diff --git a/.eslintrc.cjs b/.eslintrc.cjs index 030209c47dea..95f91b2a78be 100644 --- a/.eslintrc.cjs +++ b/.eslintrc.cjs @@ -1,14 +1,7 @@ module.exports = { root: true, extends: ['plugin:prettier/recommended'], - plugins: [ - '@nx', - 'prefer-arrow', - 'import', - 'simple-import-sort', - 'unused-imports', - 'unicorn', - ], + plugins: ['@nx', 'prefer-arrow', 'import', 'unused-imports', 'unicorn'], rules: { 'func-style': ['error', 'declaration', { allowArrowFunctions: true }], 'no-console': ['warn', { allow: ['group', 'groupCollapsed', 'groupEnd'] }], @@ -53,26 +46,6 @@ module.exports = { }, ], - 'simple-import-sort/imports': [ - 'error', - { - groups: [ - // Packages - ['^react', '^@?\\w'], - // Internal modules - ['^(@|~|src|@ui)(/.*|$)'], - // Side effect imports - ['^\\u0000'], - // Relative imports - ['^\\.\\.(?!/?$)', '^\\.\\./?$'], - ['^\\./(?=.*/)(?!/?$)', '^\\.(?!/?$)', '^\\./?$'], - // CSS imports - ['^.+\\.?(css)$'], - ], - }, - ], - 'simple-import-sort/exports': 'error', - 'unused-imports/no-unused-imports': 'warn', 'unused-imports/no-unused-vars': [ 'warn', diff --git a/.github/workflows/ci-front.yaml b/.github/workflows/ci-front.yaml index 265bf6d03a0d..aa6955723796 100644 --- a/.github/workflows/ci-front.yaml +++ b/.github/workflows/ci-front.yaml @@ -39,7 +39,7 @@ jobs: - name: Front / Write .env run: npx nx reset:env twenty-front - name: Front / Build storybook - run: npx nx storybook:build twenty-front --configuration=test + run: npx nx storybook:build twenty-front front-sb-test: runs-on: ci-8-cores needs: front-sb-build @@ -64,7 +64,7 @@ jobs: - name: Front / Write .env run: npx nx reset:env twenty-front - name: Run storybook tests - run: npx nx storybook:static:test twenty-front --configuration=${{ matrix.storybook_scope }} + run: npx nx storybook:serve-and-test:static twenty-front --configuration=${{ matrix.storybook_scope }} front-sb-test-performance: runs-on: ci-8-cores env: @@ -80,7 +80,7 @@ jobs: - name: Front / Write .env run: npx nx reset:env twenty-front - name: Run storybook tests - run: npx nx storybook:performance:test twenty-front + run: npx nx storybook:serve-and-test:static:performance twenty-front front-chromatic-deployment: if: contains(github.event.pull_request.labels.*.name, 'run-chromatic') || github.event_name == 'push' needs: front-sb-build diff --git a/.vscode/settings.json b/.vscode/settings.json index c6ff47f129ec..d63c92973cfc 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -5,21 +5,24 @@ "editor.formatOnSave": false, "editor.codeActionsOnSave": { "source.fixAll.eslint": "explicit", - "source.addMissingImports": "always" + "source.addMissingImports": "always", + "source.organizeImports": "always" } }, "[javascript]": { "editor.formatOnSave": false, "editor.codeActionsOnSave": { "source.fixAll.eslint": "explicit", - "source.addMissingImports": "always" + "source.addMissingImports": "always", + "source.organizeImports": "always" } }, "[typescriptreact]": { "editor.formatOnSave": false, "editor.codeActionsOnSave": { "source.fixAll.eslint": "explicit", - "source.addMissingImports": "always" + "source.addMissingImports": "always", + "source.organizeImports": "always" } }, "[json]": { diff --git a/.vscode/twenty.code-workspace b/.vscode/twenty.code-workspace index 49c9ebed098c..c791ac11a0c8 100644 --- a/.vscode/twenty.code-workspace +++ b/.vscode/twenty.code-workspace @@ -43,21 +43,21 @@ "[typescript]": { "editor.formatOnSave": false, "editor.codeActionsOnSave": { - "source.fixAll.eslint": true, + "source.fixAll.eslint": "explicit", "source.addMissingImports": "always" } }, "[javascript]": { "editor.formatOnSave": false, "editor.codeActionsOnSave": { - "source.fixAll.eslint": true, + "source.fixAll.eslint": "explicit", "source.addMissingImports": "always" } }, "[typescriptreact]": { "editor.formatOnSave": false, "editor.codeActionsOnSave": { - "source.fixAll.eslint": true, + "source.fixAll.eslint": "explicit", "source.addMissingImports": "always" } }, diff --git a/nx.json b/nx.json index 40a7d1a3d77e..35b6e7501700 100644 --- a/nx.json +++ b/nx.json @@ -116,14 +116,9 @@ "command": "storybook build", "output-dir": "storybook-static", "config-dir": ".storybook" - }, - "configurations": { - "test": { - "command": "storybook build --test" - } } }, - "storybook:dev": { + "storybook:serve:dev": { "executor": "nx:run-commands", "cache": true, "dependsOn": ["^build"], @@ -133,7 +128,7 @@ "config-dir": ".storybook" } }, - "storybook:static": { + "storybook:serve:static": { "executor": "nx:run-commands", "dependsOn": ["storybook:build"], "options": { @@ -143,30 +138,6 @@ "host": "localhost", "port": 6006, "silent": true - }, - "configurations": { - "test": {} - } - }, - "storybook:coverage": { - "executor": "nx:run-commands", - "cache": true, - "inputs": [ - "^default", - "excludeTests", - "{projectRoot}/coverage/storybook/coverage-storybook.json" - ], - "outputs": [ - "{projectRoot}/coverage/storybook", - "!{projectRoot}/coverage/storybook/coverage-storybook.json" - ], - "options": { - "command": "npx nyc report --reporter={args.reporter} --reporter=text-summary -t {args.coverageDir} --report-dir {args.coverageDir} --check-coverage --cwd={projectRoot}", - "coverageDir": "coverage/storybook", - "reporter": "lcov" - }, - "configurations": { - "text": { "reporter": "text" } } }, "storybook:test": { @@ -185,31 +156,52 @@ "port": 6006 } }, - "storybook:test:nocoverage": { + "storybook:test:no-coverage": { "executor": "nx:run-commands", "inputs": ["^default", "excludeTests"], "options": { "cwd": "{projectRoot}", "commands": [ - "test-storybook --url http://localhost:{args.port} --maxWorkers=3" + "test-storybook --url http://localhost:{args.port} --maxWorkers=2" ], "port": 6006 } }, - "storybook:static:test": { + "storybook:coverage": { + "executor": "nx:run-commands", + "cache": true, + "inputs": [ + "^default", + "excludeTests", + "{projectRoot}/coverage/storybook/coverage-storybook.json" + ], + "outputs": [ + "{projectRoot}/coverage/storybook", + "!{projectRoot}/coverage/storybook/coverage-storybook.json" + ], + "options": { + "command": "npx nyc report --reporter={args.reporter} --reporter=text-summary -t {args.coverageDir} --report-dir {args.coverageDir} --check-coverage --cwd={projectRoot}", + "coverageDir": "coverage/storybook", + "reporter": "lcov" + }, + "configurations": { + "text": { "reporter": "text" } + } + }, + "storybook:serve-and-test:static": { "executor": "nx:run-commands", "options": { "commands": [ - "npx concurrently --kill-others --success=first -n SB,TEST 'nx storybook:static {projectName} --port={args.port} --configuration=test' 'npx wait-on tcp:{args.port} && nx storybook:test {projectName} --port={args.port}'" + "npx concurrently --kill-others --success=first -n SB,TEST 'nx storybook:serve:static {projectName} --port={args.port}' 'npx wait-on tcp:{args.port} && nx storybook:test {projectName} --port={args.port}'" ], "port": 6006 } }, - "storybook:performance:test": { + "storybook:serve-and-test:static:performance": { "executor": "nx:run-commands", "options": { "commands": [ - "npx concurrently --kill-others --success=first -n SB,TEST 'nx storybook:dev {projectName} --configuration=performance --port={args.port}' 'npx wait-on tcp:{args.port} && nx storybook:test:nocoverage {projectName} --port={args.port} --configuration=performance'" + "npx concurrently --kill-others --success=first -n SB,TEST 'nx storybook:serve:dev {projectName} --configuration=performance --port={args.port}' 'npx wait-on tcp:{args.port} && nx storybook:test:no-coverage {projectName} --port={args.port} --configuration=performance'" ], "port": 6006 } @@ -220,7 +212,7 @@ "cwd": "{projectRoot}", "commands": [ { - "command": "nx storybook:build {projectName} --configuration=test", + "command": "nx storybook:build {projectName}", "forwardAllArgs": false }, "cross-var chromatic --project-token=$CHROMATIC_PROJECT_TOKEN --storybook-build-dir=storybook-static {args.ci}" diff --git a/package.json b/package.json index eec9a9e57987..04271989c853 100644 --- a/package.json +++ b/package.json @@ -59,6 +59,7 @@ "@types/dompurify": "^3.0.5", "@types/facepaint": "^1.2.5", "@types/lodash.camelcase": "^4.3.7", + "@types/lodash.chunk": "^4.2.9", "@types/lodash.merge": "^4.6.7", "@types/lodash.pick": "^4.3.7", "@types/nodemailer": "^6.4.14", @@ -114,6 +115,7 @@ "jsonwebtoken": "^9.0.0", "libphonenumber-js": "^1.10.26", "lodash.camelcase": "^4.3.0", + "lodash.chunk": "^4.2.0", "lodash.compact": "^3.0.1", "lodash.debounce": "^4.0.8", "lodash.groupby": "^4.6.0", diff --git a/packages/twenty-chrome-extension/src/generated/graphql.tsx b/packages/twenty-chrome-extension/src/generated/graphql.tsx index 69e043e63d65..14a8def8fc05 100644 --- a/packages/twenty-chrome-extension/src/generated/graphql.tsx +++ b/packages/twenty-chrome-extension/src/generated/graphql.tsx @@ -1,5 +1,5 @@ -import { gql } from '@apollo/client'; import * as Apollo from '@apollo/client'; +import { gql } from '@apollo/client'; export type Maybe = T | null; export type InputMaybe = Maybe; export type Exact = { [K in keyof T]: T[K] }; @@ -74,7 +74,7 @@ export type ActivityActivityTargetsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -86,7 +86,7 @@ export type ActivityAttachmentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -98,7 +98,7 @@ export type ActivityCommentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** An activity */ @@ -397,6 +397,10 @@ export type Analytics = { success: Scalars['Boolean']; }; +export type ApiConfig = { + mutationMaximumAffectedRecords: Scalars['Float']; +}; + /** An api key */ export type ApiKey = { /** Creation date */ @@ -853,8 +857,8 @@ export type Billing = { export type BillingSubscription = { id: Scalars['UUID']; - interval?: Maybe; - status: Scalars['String']; + interval?: Maybe; + status: SubscriptionStatus; }; export type BillingSubscriptionFilter = { @@ -980,6 +984,8 @@ export type CalendarChannel = { connectedAccount?: Maybe; /** Connected Account id foreign key */ connectedAccountId?: Maybe; + /** Automatically create records for people you participated with in an event. */ + contactAutoCreationPolicy?: Maybe; /** Creation date */ createdAt?: Maybe; /** Handle */ @@ -992,6 +998,12 @@ export type CalendarChannel = { isSyncEnabled?: Maybe; /** Sync Cursor. Used for syncing events from the calendar provider */ syncCursor?: Maybe; + /** Sync stage */ + syncStage?: Maybe; + /** Sync stage started at */ + syncStageStartedAt?: Maybe; + /** Sync status */ + syncStatus?: Maybe; /** Throttle Failure Count */ throttleFailureCount?: Maybe; /** Update date */ @@ -1009,7 +1021,7 @@ export type CalendarChannelCalendarChannelEventAssociationsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** Calendar Channels */ @@ -1020,10 +1032,31 @@ export type CalendarChannelConnection = { totalCount?: Maybe; }; +/** Automatically create records for people you participated with in an event. */ +export enum CalendarChannelContactAutoCreationPolicyEnum { + /** As Organizer */ + AsOrganizer = 'AS_ORGANIZER', + /** As Participant */ + AsParticipant = 'AS_PARTICIPANT', + /** As Participant and Organizer */ + AsParticipantAndOrganizer = 'AS_PARTICIPANT_AND_ORGANIZER', + /** None */ + None = 'NONE' +} + +export type CalendarChannelContactAutoCreationPolicyEnumFilter = { + eq?: InputMaybe; + in?: InputMaybe>>; + is?: InputMaybe; + neq?: InputMaybe; +}; + /** Calendar Channels */ export type CalendarChannelCreateInput = { /** Connected Account id foreign key */ connectedAccountId: Scalars['UUID']; + /** Automatically create records for people you participated with in an event. */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; /** Handle */ @@ -1036,6 +1069,12 @@ export type CalendarChannelCreateInput = { isSyncEnabled?: InputMaybe; /** Sync Cursor. Used for syncing events from the calendar provider */ syncCursor?: InputMaybe; + /** Sync stage */ + syncStage?: InputMaybe; + /** Sync stage started at */ + syncStageStartedAt?: InputMaybe; + /** Sync status */ + syncStatus?: InputMaybe; /** Throttle Failure Count */ throttleFailureCount?: InputMaybe; /** Update date */ @@ -1156,6 +1195,8 @@ export type CalendarChannelFilterInput = { and?: InputMaybe>>; /** Connected Account id foreign key */ connectedAccountId?: InputMaybe; + /** Automatically create records for people you participated with in an event. */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; /** Handle */ @@ -1170,6 +1211,12 @@ export type CalendarChannelFilterInput = { or?: InputMaybe>>; /** Sync Cursor. Used for syncing events from the calendar provider */ syncCursor?: InputMaybe; + /** Sync stage */ + syncStage?: InputMaybe; + /** Sync stage started at */ + syncStageStartedAt?: InputMaybe; + /** Sync status */ + syncStatus?: InputMaybe; /** Throttle Failure Count */ throttleFailureCount?: InputMaybe; /** Update date */ @@ -1182,6 +1229,8 @@ export type CalendarChannelFilterInput = { export type CalendarChannelOrderByInput = { /** Connected Account id foreign key */ connectedAccountId?: InputMaybe; + /** Automatically create records for people you participated with in an event. */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; /** Handle */ @@ -1194,6 +1243,12 @@ export type CalendarChannelOrderByInput = { isSyncEnabled?: InputMaybe; /** Sync Cursor. Used for syncing events from the calendar provider */ syncCursor?: InputMaybe; + /** Sync stage */ + syncStage?: InputMaybe; + /** Sync stage started at */ + syncStageStartedAt?: InputMaybe; + /** Sync status */ + syncStatus?: InputMaybe; /** Throttle Failure Count */ throttleFailureCount?: InputMaybe; /** Update date */ @@ -1202,10 +1257,56 @@ export type CalendarChannelOrderByInput = { visibility?: InputMaybe; }; +/** Sync stage */ +export enum CalendarChannelSyncStageEnum { + /** Calendar events import ongoing */ + CalendarEventsImportOngoing = 'CALENDAR_EVENTS_IMPORT_ONGOING', + /** Calendar events import pending */ + CalendarEventsImportPending = 'CALENDAR_EVENTS_IMPORT_PENDING', + /** Calendar event list fetch ongoing */ + CalendarEventListFetchOngoing = 'CALENDAR_EVENT_LIST_FETCH_ONGOING', + /** Failed */ + Failed = 'FAILED', + /** Full calendar event list fetch pending */ + FullCalendarEventListFetchPending = 'FULL_CALENDAR_EVENT_LIST_FETCH_PENDING', + /** Partial calendar event list fetch pending */ + PartialCalendarEventListFetchPending = 'PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING' +} + +export type CalendarChannelSyncStageEnumFilter = { + eq?: InputMaybe; + in?: InputMaybe>>; + is?: InputMaybe; + neq?: InputMaybe; +}; + +/** Sync status */ +export enum CalendarChannelSyncStatusEnum { + /** Active */ + Active = 'ACTIVE', + /** Failed Insufficient Permissions */ + FailedInsufficientPermissions = 'FAILED_INSUFFICIENT_PERMISSIONS', + /** Failed Unknown */ + FailedUnknown = 'FAILED_UNKNOWN', + /** Not Synced */ + NotSynced = 'NOT_SYNCED', + /** Ongoing */ + Ongoing = 'ONGOING' +} + +export type CalendarChannelSyncStatusEnumFilter = { + eq?: InputMaybe; + in?: InputMaybe>>; + is?: InputMaybe; + neq?: InputMaybe; +}; + /** Calendar Channels */ export type CalendarChannelUpdateInput = { /** Connected Account id foreign key */ connectedAccountId?: InputMaybe; + /** Automatically create records for people you participated with in an event. */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; /** Handle */ @@ -1218,6 +1319,12 @@ export type CalendarChannelUpdateInput = { isSyncEnabled?: InputMaybe; /** Sync Cursor. Used for syncing events from the calendar provider */ syncCursor?: InputMaybe; + /** Sync stage */ + syncStage?: InputMaybe; + /** Sync stage started at */ + syncStageStartedAt?: InputMaybe; + /** Sync status */ + syncStatus?: InputMaybe; /** Throttle Failure Count */ throttleFailureCount?: InputMaybe; /** Update date */ @@ -1296,7 +1403,7 @@ export type CalendarEventCalendarChannelEventAssociationsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -1308,7 +1415,7 @@ export type CalendarEventCalendarEventParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** Calendar events */ @@ -1640,11 +1747,12 @@ export type Captcha = { }; export enum CaptchaDriverType { - GoogleRecatpcha = 'GoogleRecatpcha', + GoogleRecaptcha = 'GoogleRecaptcha', Turnstile = 'Turnstile' } export type ClientConfig = { + api: ApiConfig; authProviders: AuthProviders; billing: Billing; captcha: Captcha; @@ -1811,7 +1919,7 @@ export type CompanyActivityTargetsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -1823,7 +1931,7 @@ export type CompanyAttachmentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -1835,7 +1943,7 @@ export type CompanyFavoritesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -1847,7 +1955,7 @@ export type CompanyOpportunitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -1859,7 +1967,7 @@ export type CompanyPeopleArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -1871,7 +1979,7 @@ export type CompanyTimelineActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** A company */ @@ -2050,7 +2158,7 @@ export type ConnectedAccountCalendarChannelsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -2062,7 +2170,7 @@ export type ConnectedAccountMessageChannelsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** A connected account */ @@ -2420,7 +2528,6 @@ export enum FieldMetadataType { Numeric = 'NUMERIC', Phone = 'PHONE', Position = 'POSITION', - Probability = 'PROBABILITY', Rating = 'RATING', RawJson = 'RAW_JSON', Relation = 'RELATION', @@ -2597,7 +2704,7 @@ export type MessageMessageChannelMessageAssociationsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -2609,7 +2716,7 @@ export type MessageMessageParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** Message Channels */ @@ -2618,8 +2725,14 @@ export type MessageChannel = { connectedAccount?: Maybe; /** Connected Account id foreign key */ connectedAccountId?: Maybe; + /** Automatically create People records when receiving or sending emails */ + contactAutoCreationPolicy?: Maybe; /** Creation date */ createdAt?: Maybe; + /** Exclude group emails */ + excludeGroupEmails?: Maybe; + /** Exclude non professional emails */ + excludeNonProfessionalEmails?: Maybe; /** Handle */ handle?: Maybe; /** Id */ @@ -2659,7 +2772,7 @@ export type MessageChannelMessageChannelMessageAssociationsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** Message Channels */ @@ -2670,12 +2783,35 @@ export type MessageChannelConnection = { totalCount?: Maybe; }; +/** Automatically create People records when receiving or sending emails */ +export enum MessageChannelContactAutoCreationPolicyEnum { + /** None */ + None = 'NONE', + /** Sent */ + Sent = 'SENT', + /** Sent and Received */ + SentAndReceived = 'SENT_AND_RECEIVED' +} + +export type MessageChannelContactAutoCreationPolicyEnumFilter = { + eq?: InputMaybe; + in?: InputMaybe>>; + is?: InputMaybe; + neq?: InputMaybe; +}; + /** Message Channels */ export type MessageChannelCreateInput = { /** Connected Account id foreign key */ connectedAccountId: Scalars['UUID']; + /** Automatically create People records when receiving or sending emails */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; + /** Exclude group emails */ + excludeGroupEmails?: InputMaybe; + /** Exclude non professional emails */ + excludeNonProfessionalEmails?: InputMaybe; /** Handle */ handle?: InputMaybe; /** Id */ @@ -2715,8 +2851,14 @@ export type MessageChannelFilterInput = { and?: InputMaybe>>; /** Connected Account id foreign key */ connectedAccountId?: InputMaybe; + /** Automatically create People records when receiving or sending emails */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; + /** Exclude group emails */ + excludeGroupEmails?: InputMaybe; + /** Exclude non professional emails */ + excludeNonProfessionalEmails?: InputMaybe; /** Handle */ handle?: InputMaybe; /** Id */ @@ -2874,8 +3016,14 @@ export type MessageChannelMessageAssociationUpdateInput = { export type MessageChannelOrderByInput = { /** Connected Account id foreign key */ connectedAccountId?: InputMaybe; + /** Automatically create People records when receiving or sending emails */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; + /** Exclude group emails */ + excludeGroupEmails?: InputMaybe; + /** Exclude non professional emails */ + excludeNonProfessionalEmails?: InputMaybe; /** Handle */ handle?: InputMaybe; /** Id */ @@ -2973,8 +3121,14 @@ export type MessageChannelTypeEnumFilter = { export type MessageChannelUpdateInput = { /** Connected Account id foreign key */ connectedAccountId?: InputMaybe; + /** Automatically create People records when receiving or sending emails */ + contactAutoCreationPolicy?: InputMaybe; /** Creation date */ createdAt?: InputMaybe; + /** Exclude group emails */ + excludeGroupEmails?: InputMaybe; + /** Exclude non professional emails */ + excludeNonProfessionalEmails?: InputMaybe; /** Handle */ handle?: InputMaybe; /** Id */ @@ -3300,7 +3454,7 @@ export type MessageThreadMessageChannelMessageAssociationsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -3312,7 +3466,7 @@ export type MessageThreadMessagesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** Message Thread */ @@ -3517,7 +3671,9 @@ export type Mutation = { deleteWebhooks?: Maybe>; deleteWorkspaceMember?: Maybe; deleteWorkspaceMembers?: Maybe>; + disablePostgresProxy: PostgresCredentials; emailPasswordResetLink: EmailPasswordResetLink; + enablePostgresProxy: PostgresCredentials; exchangeAuthorizationCode: ExchangeAuthCode; executeQuickActionOnActivity?: Maybe; executeQuickActionOnActivityTarget?: Maybe; @@ -3649,288 +3805,344 @@ export type MutationChallengeArgs = { export type MutationCheckoutSessionArgs = { - recurringInterval: Scalars['String']; + recurringInterval: SubscriptionInterval; successUrlPath?: InputMaybe; }; export type MutationCreateActivitiesArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateActivityArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateActivityTargetArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateActivityTargetsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateApiKeyArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateApiKeysArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateAttachmentArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateAttachmentsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateAuditLogArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateAuditLogsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateBlocklistArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateBlocklistsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateCalendarChannelArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateCalendarChannelEventAssociationArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateCalendarChannelEventAssociationsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateCalendarChannelsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateCalendarEventArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateCalendarEventParticipantArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateCalendarEventParticipantsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateCalendarEventsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateCommentArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateCommentsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateCompaniesArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateCompanyArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateConnectedAccountArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateConnectedAccountsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateFavoriteArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateFavoritesArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateMessageArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateMessageChannelArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateMessageChannelMessageAssociationArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateMessageChannelMessageAssociationsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateMessageChannelsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateMessageParticipantArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateMessageParticipantsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateMessageThreadArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateMessageThreadsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateMessagesArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateOpportunitiesArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateOpportunityArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreatePeopleArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreatePersonArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateTimelineActivitiesArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateTimelineActivityArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateViewArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateViewFieldArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateViewFieldsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateViewFilterArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateViewFiltersArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateViewSortArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateViewSortsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateViewsArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateWebhookArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateWebhooksArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; export type MutationCreateWorkspaceMemberArgs = { data?: InputMaybe; + upsert?: InputMaybe; }; export type MutationCreateWorkspaceMembersArgs = { data?: InputMaybe>; + upsert?: InputMaybe; }; @@ -4803,10 +5015,14 @@ export type ObjectFieldsConnection = { pageInfo: PageInfo; }; -/** Onboarding step */ -export enum OnboardingStep { +/** Onboarding status */ +export enum OnboardingStatus { + Completed = 'COMPLETED', InviteTeam = 'INVITE_TEAM', - SyncEmail = 'SYNC_EMAIL' + PlanRequired = 'PLAN_REQUIRED', + ProfileCreation = 'PROFILE_CREATION', + SyncEmail = 'SYNC_EMAIL', + WorkspaceActivation = 'WORKSPACE_ACTIVATION' } export type OnboardingStepSuccess = { @@ -4861,7 +5077,7 @@ export type OpportunityActivityTargetsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -4873,7 +5089,7 @@ export type OpportunityAttachmentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -4885,7 +5101,7 @@ export type OpportunityFavoritesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -4897,7 +5113,7 @@ export type OpportunityTimelineActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** An opportunity */ @@ -5120,7 +5336,7 @@ export type PersonActivityTargetsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5132,7 +5348,7 @@ export type PersonAttachmentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5144,7 +5360,7 @@ export type PersonCalendarEventParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5156,7 +5372,7 @@ export type PersonFavoritesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5168,7 +5384,7 @@ export type PersonMessageParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5180,7 +5396,7 @@ export type PersonPointOfContactForOpportunitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5192,7 +5408,7 @@ export type PersonTimelineActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** A person */ @@ -5332,9 +5548,16 @@ export type PersonUpdateInput = { xLink?: InputMaybe; }; +export type PostgresCredentials = { + id: Scalars['UUID']; + password: Scalars['String']; + user: Scalars['String']; + workspaceId: Scalars['String']; +}; + export type ProductPriceEntity = { created: Scalars['Float']; - recurringInterval: Scalars['String']; + recurringInterval: SubscriptionInterval; stripePriceId: Scalars['String']; unitAmount: Scalars['Float']; }; @@ -5347,53 +5570,54 @@ export type ProductPricesEntity = { export type Query = { activities?: Maybe; activity?: Maybe; - activityDuplicates?: Maybe; + activityDuplicates?: Maybe>; activityTarget?: Maybe; - activityTargetDuplicates?: Maybe; + activityTargetDuplicates?: Maybe>; activityTargets?: Maybe; apiKey?: Maybe; - apiKeyDuplicates?: Maybe; + apiKeyDuplicates?: Maybe>; apiKeys?: Maybe; attachment?: Maybe; - attachmentDuplicates?: Maybe; + attachmentDuplicates?: Maybe>; attachments?: Maybe; auditLog?: Maybe; - auditLogDuplicates?: Maybe; + auditLogDuplicates?: Maybe>; auditLogs?: Maybe; billingPortalSession: SessionEntity; blocklist?: Maybe; - blocklistDuplicates?: Maybe; + blocklistDuplicates?: Maybe>; blocklists?: Maybe; calendarChannel?: Maybe; - calendarChannelDuplicates?: Maybe; + calendarChannelDuplicates?: Maybe>; calendarChannelEventAssociation?: Maybe; - calendarChannelEventAssociationDuplicates?: Maybe; + calendarChannelEventAssociationDuplicates?: Maybe>; calendarChannelEventAssociations?: Maybe; calendarChannels?: Maybe; calendarEvent?: Maybe; - calendarEventDuplicates?: Maybe; + calendarEventDuplicates?: Maybe>; calendarEventParticipant?: Maybe; - calendarEventParticipantDuplicates?: Maybe; + calendarEventParticipantDuplicates?: Maybe>; calendarEventParticipants?: Maybe; calendarEvents?: Maybe; checkUserExists: UserExists; checkWorkspaceInviteHashIsValid: WorkspaceInviteHashValid; clientConfig: ClientConfig; comment?: Maybe; - commentDuplicates?: Maybe; + commentDuplicates?: Maybe>; comments?: Maybe; companies?: Maybe; company?: Maybe; - companyDuplicates?: Maybe; + companyDuplicates?: Maybe>; connectedAccount?: Maybe; - connectedAccountDuplicates?: Maybe; + connectedAccountDuplicates?: Maybe>; connectedAccounts?: Maybe; currentUser: User; currentWorkspace: Workspace; favorite?: Maybe; - favoriteDuplicates?: Maybe; + favoriteDuplicates?: Maybe>; favorites?: Maybe; findWorkspaceFromInviteHash: Workspace; + getPostgresCredentials?: Maybe; getProductPrices: ProductPricesEntity; getTimelineCalendarEventsFromCompanyId: TimelineCalendarEventsWithTotal; getTimelineCalendarEventsFromPersonId: TimelineCalendarEventsWithTotal; @@ -5401,48 +5625,48 @@ export type Query = { getTimelineThreadsFromPersonId: TimelineThreadsWithTotal; message?: Maybe; messageChannel?: Maybe; - messageChannelDuplicates?: Maybe; + messageChannelDuplicates?: Maybe>; messageChannelMessageAssociation?: Maybe; - messageChannelMessageAssociationDuplicates?: Maybe; + messageChannelMessageAssociationDuplicates?: Maybe>; messageChannelMessageAssociations?: Maybe; messageChannels?: Maybe; - messageDuplicates?: Maybe; + messageDuplicates?: Maybe>; messageParticipant?: Maybe; - messageParticipantDuplicates?: Maybe; + messageParticipantDuplicates?: Maybe>; messageParticipants?: Maybe; messageThread?: Maybe; - messageThreadDuplicates?: Maybe; + messageThreadDuplicates?: Maybe>; messageThreads?: Maybe; messages?: Maybe; object: Object; objects: ObjectConnection; opportunities?: Maybe; opportunity?: Maybe; - opportunityDuplicates?: Maybe; + opportunityDuplicates?: Maybe>; people?: Maybe; person?: Maybe; - personDuplicates?: Maybe; + personDuplicates?: Maybe>; timelineActivities?: Maybe; timelineActivity?: Maybe; - timelineActivityDuplicates?: Maybe; + timelineActivityDuplicates?: Maybe>; validatePasswordResetToken: ValidatePasswordResetToken; view?: Maybe; - viewDuplicates?: Maybe; + viewDuplicates?: Maybe>; viewField?: Maybe; - viewFieldDuplicates?: Maybe; + viewFieldDuplicates?: Maybe>; viewFields?: Maybe; viewFilter?: Maybe; - viewFilterDuplicates?: Maybe; + viewFilterDuplicates?: Maybe>; viewFilters?: Maybe; viewSort?: Maybe; - viewSortDuplicates?: Maybe; + viewSortDuplicates?: Maybe>; viewSorts?: Maybe; views?: Maybe; webhook?: Maybe; - webhookDuplicates?: Maybe; + webhookDuplicates?: Maybe>; webhooks?: Maybe; workspaceMember?: Maybe; - workspaceMemberDuplicates?: Maybe; + workspaceMemberDuplicates?: Maybe>; workspaceMembers?: Maybe; }; @@ -5454,7 +5678,7 @@ export type QueryActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5464,8 +5688,8 @@ export type QueryActivityArgs = { export type QueryActivityDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5475,8 +5699,8 @@ export type QueryActivityTargetArgs = { export type QueryActivityTargetDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5487,7 +5711,7 @@ export type QueryActivityTargetsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5497,8 +5721,8 @@ export type QueryApiKeyArgs = { export type QueryApiKeyDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5509,7 +5733,7 @@ export type QueryApiKeysArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5519,8 +5743,8 @@ export type QueryAttachmentArgs = { export type QueryAttachmentDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5531,7 +5755,7 @@ export type QueryAttachmentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5541,8 +5765,8 @@ export type QueryAuditLogArgs = { export type QueryAuditLogDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5553,7 +5777,7 @@ export type QueryAuditLogsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5568,8 +5792,8 @@ export type QueryBlocklistArgs = { export type QueryBlocklistDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5580,7 +5804,7 @@ export type QueryBlocklistsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5590,8 +5814,8 @@ export type QueryCalendarChannelArgs = { export type QueryCalendarChannelDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5601,8 +5825,8 @@ export type QueryCalendarChannelEventAssociationArgs = { export type QueryCalendarChannelEventAssociationDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5613,7 +5837,7 @@ export type QueryCalendarChannelEventAssociationsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5624,7 +5848,7 @@ export type QueryCalendarChannelsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5634,8 +5858,8 @@ export type QueryCalendarEventArgs = { export type QueryCalendarEventDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5645,8 +5869,8 @@ export type QueryCalendarEventParticipantArgs = { export type QueryCalendarEventParticipantDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5657,7 +5881,7 @@ export type QueryCalendarEventParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5668,7 +5892,7 @@ export type QueryCalendarEventsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5689,8 +5913,8 @@ export type QueryCommentArgs = { export type QueryCommentDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5701,7 +5925,7 @@ export type QueryCommentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5712,7 +5936,7 @@ export type QueryCompaniesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5722,8 +5946,8 @@ export type QueryCompanyArgs = { export type QueryCompanyDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5733,8 +5957,8 @@ export type QueryConnectedAccountArgs = { export type QueryConnectedAccountDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5745,7 +5969,7 @@ export type QueryConnectedAccountsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5755,8 +5979,8 @@ export type QueryFavoriteArgs = { export type QueryFavoriteDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5767,7 +5991,7 @@ export type QueryFavoritesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5820,8 +6044,8 @@ export type QueryMessageChannelArgs = { export type QueryMessageChannelDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5831,8 +6055,8 @@ export type QueryMessageChannelMessageAssociationArgs = { export type QueryMessageChannelMessageAssociationDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5843,7 +6067,7 @@ export type QueryMessageChannelMessageAssociationsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5854,13 +6078,13 @@ export type QueryMessageChannelsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; export type QueryMessageDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5870,8 +6094,8 @@ export type QueryMessageParticipantArgs = { export type QueryMessageParticipantDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5882,7 +6106,7 @@ export type QueryMessageParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5892,8 +6116,8 @@ export type QueryMessageThreadArgs = { export type QueryMessageThreadDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5904,7 +6128,7 @@ export type QueryMessageThreadsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5915,7 +6139,7 @@ export type QueryMessagesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5926,7 +6150,7 @@ export type QueryOpportunitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5936,8 +6160,8 @@ export type QueryOpportunityArgs = { export type QueryOpportunityDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5948,7 +6172,7 @@ export type QueryPeopleArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5958,8 +6182,8 @@ export type QueryPersonArgs = { export type QueryPersonDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5970,7 +6194,7 @@ export type QueryTimelineActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -5980,8 +6204,8 @@ export type QueryTimelineActivityArgs = { export type QueryTimelineActivityDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -5996,8 +6220,8 @@ export type QueryViewArgs = { export type QueryViewDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -6007,8 +6231,8 @@ export type QueryViewFieldArgs = { export type QueryViewFieldDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -6019,7 +6243,7 @@ export type QueryViewFieldsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -6029,8 +6253,8 @@ export type QueryViewFilterArgs = { export type QueryViewFilterDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -6041,7 +6265,7 @@ export type QueryViewFiltersArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -6051,8 +6275,8 @@ export type QueryViewSortArgs = { export type QueryViewSortDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -6063,7 +6287,7 @@ export type QueryViewSortsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -6074,7 +6298,7 @@ export type QueryViewsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -6084,8 +6308,8 @@ export type QueryWebhookArgs = { export type QueryWebhookDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -6096,7 +6320,7 @@ export type QueryWebhooksArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -6106,8 +6330,8 @@ export type QueryWorkspaceMemberArgs = { export type QueryWorkspaceMemberDuplicatesArgs = { - data?: InputMaybe; - id?: InputMaybe; + data?: InputMaybe>>; + ids?: InputMaybe>>; }; @@ -6118,7 +6342,7 @@ export type QueryWorkspaceMembersArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; export type RawJsonFilter = { @@ -6226,13 +6450,30 @@ export type StringFilter = { startsWith?: InputMaybe; }; +export enum SubscriptionInterval { + Day = 'Day', + Month = 'Month', + Week = 'Week', + Year = 'Year' +} + +export enum SubscriptionStatus { + Active = 'Active', + Canceled = 'Canceled', + Incomplete = 'Incomplete', + IncompleteExpired = 'IncompleteExpired', + PastDue = 'PastDue', + Paused = 'Paused', + Trialing = 'Trialing', + Unpaid = 'Unpaid' +} + export type Support = { supportDriver: Scalars['String']; supportFrontChatId?: Maybe; }; export type Telemetry = { - anonymizationEnabled: Scalars['Boolean']; enabled: Scalars['Boolean']; }; @@ -6534,7 +6775,7 @@ export type User = { firstName: Scalars['String']; id: Scalars['UUID']; lastName: Scalars['String']; - onboardingStep?: Maybe; + onboardingStatus?: Maybe; passwordHash?: Maybe; /** @deprecated field migrated into the AppTokens Table ref: https://github.com/twentyhq/twenty/issues/5021 */ passwordResetToken?: Maybe; @@ -6623,7 +6864,7 @@ export type ViewViewFieldsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -6635,7 +6876,7 @@ export type ViewViewFiltersArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -6647,7 +6888,7 @@ export type ViewViewSortsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** (System) Views */ @@ -7222,8 +7463,8 @@ export type Workspace = { id: Scalars['UUID']; inviteHash?: Maybe; logo?: Maybe; - subscriptionStatus: Scalars['String']; updatedAt: Scalars['DateTime']; + workspaceMembersCount?: Maybe; }; @@ -7304,7 +7545,7 @@ export type WorkspaceMemberAccountOwnerForCompaniesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7316,7 +7557,7 @@ export type WorkspaceMemberAssignedActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7328,7 +7569,7 @@ export type WorkspaceMemberAuditLogsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7340,7 +7581,7 @@ export type WorkspaceMemberAuthoredActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7352,7 +7593,7 @@ export type WorkspaceMemberAuthoredAttachmentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7364,7 +7605,7 @@ export type WorkspaceMemberAuthoredCommentsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7376,7 +7617,7 @@ export type WorkspaceMemberBlocklistArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7388,7 +7629,7 @@ export type WorkspaceMemberCalendarEventParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7400,7 +7641,7 @@ export type WorkspaceMemberConnectedAccountsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7412,7 +7653,7 @@ export type WorkspaceMemberFavoritesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7424,7 +7665,7 @@ export type WorkspaceMemberMessageParticipantsArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; @@ -7436,7 +7677,7 @@ export type WorkspaceMemberTimelineActivitiesArgs = { first?: InputMaybe; last?: InputMaybe; limit?: InputMaybe; - orderBy?: InputMaybe; + orderBy?: InputMaybe>>; }; /** A workspace member */ diff --git a/packages/twenty-docker/k8s/README.md b/packages/twenty-docker/k8s/README.md new file mode 100644 index 000000000000..df1cde709784 --- /dev/null +++ b/packages/twenty-docker/k8s/README.md @@ -0,0 +1,113 @@ +# README + +## Overview + +This repository contains Kubernetes manifests and Terraform files to help you deploy and manage the TwentyCRM application. The files are located in the `packages/twenty-docker/k8s` directory. + +## Prerequisites + +Before using these files, ensure you have the following installed and configured on your system: + +- Kubernetes cluster (e.g., Minikube, EKS, GKE) +- kubectl +- Terraform +- Docker + +## Setup Instructions + +### Step 1: Clone the Repository + +Clone the repository to your local machine: + +``` bash +git clone https://github.com/twentyhq/twenty.git +cd twentycrm/packages/twenty-docker/k8s +``` + +### Step 2: Customize the Manifests and Terraform Files + +**Important:** These files require customization for your specific implementation. Update the placeholders and configurations according to your environment and requirements. + +### Step 3: Deploy with Terraform + +1. Navigate to the Terraform directory: + + ```bash + cd terraform + ``` + +2. Initialize Terraform: + + ```bash + terraform init + ``` + +3. Plan the deployment: + + ```bash + terraform plan + ``` + +4. Apply the deployment: + + ```bash + terraform apply + ``` + +## OR + +### Step 3: Deploy with Kubernetes Manifests + +1. Navigate to the Kubernetes manifests directory: + + ```bash + cd ../k8s + ``` + +2. Create Server Secret + + ``` bash + kubectl create secret generic -n twentycrm tokens --from-literal accessToken=changeme --from-literal loginToken="changeme" --from-literal refreshToken="changeme" --from-literal fileToken="changeme" + ``` + +3. Apply the manifests: + + ```bash + kubectl apply -f . + ``` + +## Customization + +### Kubernetes Manifests + +- **Namespace:** Update the `namespace` in the manifests as needed. +- **Resource Limits:** Adjust the resource limits and requests according to your application's requirements. +- **Environment Variables:** Configure server tokens in the `Secret` command above. + +### Terraform Files + +- **Variables:** Update the variables in the `variables.tf` file to match your environment. +- **Locals:** Update the locals in the `main.tf` file to match your environment. +- **Providers:** Ensure the provider configurations (e.g., AWS, GCP) are correct for your setup. +- **Resources:** Modify the resource definitions as needed to fit your infrastructure. + +## Troubleshooting + +### Common Issues + +- **Connectivity:** Ensure your Kubernetes cluster is accessible and configured correctly. +- **Permissions:** Verify that you have the necessary permissions to deploy resources in your cloud provider. +- **Resource Limits:** Adjust resource limits if you encounter issues related to insufficient resources. + +### Logs and Debugging + +- Use `kubectl logs` to check the logs of your Kubernetes pods. +- Use `terraform show` and `terraform state` to inspect your Terraform state and configurations. + +## Conclusion + +This setup provides a basic structure for deploying the TwentyCRM application using Kubernetes and Terraform. Ensure you thoroughly customize the manifests and Terraform files to suit your specific needs. For any issues or questions, please refer to the official documentation of Kubernetes and Terraform or seek support from your cloud provider. + +--- + +Feel free to contribute and improve this repository by submitting pull requests or opening issues. Happy deploying! diff --git a/packages/twenty-docker/k8s/manifests/deployment-db.yaml b/packages/twenty-docker/k8s/manifests/deployment-db.yaml new file mode 100644 index 000000000000..8d9dec9df404 --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/deployment-db.yaml @@ -0,0 +1,54 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app: twentycrm-db + name: twentycrm-db + namespace: twentycrm +spec: + progressDeadlineSeconds: 600 + replicas: 1 + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 1 + type: RollingUpdate + selector: + matchLabels: + app: twentycrm-db + template: + metadata: + labels: + app: twentycrm-db + spec: + volumes: + - name: twentycrm-db-data + persistentVolumeClaim: + claimName: twentycrm-db-pvc + containers: + - env: + - name: POSTGRES_PASSWORD + value: "twenty" + - name: BITNAMI_DEBUG + value: "true" + - image: twentycrm/twenty-postgres:latest + imagePullPolicy: Always + name: twentycrm + ports: + - containerPort: 5432 + name: tcp + protocol: TCP + resources: + requests: + memory: "256Mi" + cpu: "250m" + limits: + memory: "1024Mi" + cpu: "1000m" + stdin: true + tty: true + volumeMounts: + - mountPath: /bitnami/postgresql + name: twentycrm-db-data + dnsPolicy: ClusterFirst + restartPolicy: Always diff --git a/packages/twenty-docker/k8s/manifests/deployment-server.yaml b/packages/twenty-docker/k8s/manifests/deployment-server.yaml new file mode 100644 index 000000000000..cf740722f55e --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/deployment-server.yaml @@ -0,0 +1,82 @@ +apiVersion: apps/v1 +kind: Deployment +metadata: + labels: + app: twentycrm-server + name: twentycrm-server + namespace: twentycrm +spec: + progressDeadlineSeconds: 600 + replicas: 1 + strategy: + rollingUpdate: + maxSurge: 1 + maxUnavailable: 1 + type: RollingUpdate + selector: + matchLabels: + app: twentycrm-server + template: + metadata: + labels: + app: twentycrm-server + spec: + volumes: + - name: twentycrm-server-data + persistentVolumeClaim: + claimName: twentycrm-server-pvc + containers: + - env: + - name: PORT + value: 3000 + - name: SERVER_URL + value: "https://crm.example.com:443" + - name: PG_DATABASE_URL + value: "postgres://twenty:twenty@twenty-db.twentycrm.svc.cluster.local/default" + - name: ENABLE_DB_MIGRATIONS + value: "true" + - name: SIGN_IN_PREFILLED + value: "true" + - name: STORAGE_TYPE + value: "local" + - name: ACCESS_TOKEN_SECRET + valueFrom: + secretKeyRef: + name: tokens + key: accessToken + - name: LOGIN_TOKEN_SECRET + valueFrom: + secretKeyRef: + name: tokens + key: loginToken + - name: REFRESH_TOKEN_SECRET + valueFrom: + secretKeyRef: + name: tokens + key: refreshToken + - name: FILE_TOKEN_SECRET + valueFrom: + secretKeyRef: + name: tokens + key: fileToken + - image: twentycrm/twenty:latest + imagePullPolicy: Always + name: twentycrm + ports: + - containerPort: 3000 + name: http-tcp + protocol: TCP + resources: + requests: + memory: "256Mi" + cpu: "250m" + limits: + memory: "1024Mi" + cpu: "1000m" + stdin: true + tty: true + volumeMounts: + - mountPath: /app/.local-storage + name: twentycrm-server-data + dnsPolicy: ClusterFirst + restartPolicy: Always diff --git a/packages/twenty-docker/k8s/manifests/ingress.yaml b/packages/twenty-docker/k8s/manifests/ingress.yaml new file mode 100644 index 000000000000..19663f7a4f51 --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/ingress.yaml @@ -0,0 +1,24 @@ +apiVersion: networking.k8s.io/v1 +kind: Ingress +metadata: + name: twentycrm + namespace: twentycrm + annotations: + nginx.ingress.kubernetes.io/configuration-snippet: | + more_set_headers "X-Forwarded-For $http_x_forwarded_for"; + nginx.ingress.kubernetes.io/force-ssl-redirect: "false" + kubernetes.io/ingress.class: "nginx" + nginx.ingress.kubernetes.io/backend-protocol: "HTTP" +spec: + ingressClassName: nginx + rules: + - host: crm.example.com + http: + paths: + - path: / + pathType: Prefix + backend: + service: + name: twentycrm-server + port: + name: http-tcp diff --git a/packages/twenty-docker/k8s/manifests/pv-db.yaml b/packages/twenty-docker/k8s/manifests/pv-db.yaml new file mode 100644 index 000000000000..9caa4ca4d919 --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/pv-db.yaml @@ -0,0 +1,11 @@ +apiVersion: v1 +kind: PersistentVolume +metadata: + name: twentycrm-db-pv +spec: + storageClassName: default + capacity: + storage: 10Gi + accessModes: + - ReadWriteOnce + persistentVolumeReclaimPolicy: Retain diff --git a/packages/twenty-docker/k8s/manifests/pv-server.yaml b/packages/twenty-docker/k8s/manifests/pv-server.yaml new file mode 100644 index 000000000000..721de7d5668a --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/pv-server.yaml @@ -0,0 +1,12 @@ +apiVersion: v1 +kind: PersistentVolume +metadata: + name: twentycrm-server-pv + namespace: twentycrm +spec: + storageClassName: default + capacity: + storage: 10Gi + accessModes: + - ReadWriteOnce + persistentVolumeReclaimPolicy: Retain diff --git a/packages/twenty-docker/k8s/manifests/pvc-db.yaml b/packages/twenty-docker/k8s/manifests/pvc-db.yaml new file mode 100644 index 000000000000..146596ea1050 --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/pvc-db.yaml @@ -0,0 +1,13 @@ +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: twentycrm-db-pvc + namespace: twentycrm +spec: + storageClassName: default + volumeName: twentycrm-db-pv + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 10Gi diff --git a/packages/twenty-docker/k8s/manifests/pvc-server.yaml b/packages/twenty-docker/k8s/manifests/pvc-server.yaml new file mode 100644 index 000000000000..f265057cf569 --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/pvc-server.yaml @@ -0,0 +1,13 @@ +apiVersion: v1 +kind: PersistentVolumeClaim +metadata: + name: twentycrm-server-pvc + namespace: twentycrm +spec: + storageClassName: default + volumeName: twentycrm-server-pv + accessModes: + - ReadWriteOnce + resources: + requests: + storage: 10Gi diff --git a/packages/twenty-docker/k8s/manifests/service-db.yaml b/packages/twenty-docker/k8s/manifests/service-db.yaml new file mode 100644 index 000000000000..bb0e38df6d6d --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/service-db.yaml @@ -0,0 +1,18 @@ +apiVersion: v1 +kind: Service +metadata: + name: twentycrm-db + namespace: twentycrm +spec: + internalTrafficPolicy: Cluster + ports: + - port: 5432 + protocol: TCP + targetPort: 5432 + selector: + app: twentycrm-db + sessionAffinity: ClientIP + sessionAffinityConfig: + clientIP: + timeoutSeconds: 10800 + type: ClusterIP diff --git a/packages/twenty-docker/k8s/manifests/service-server.yaml b/packages/twenty-docker/k8s/manifests/service-server.yaml new file mode 100644 index 000000000000..7fcc869a6edc --- /dev/null +++ b/packages/twenty-docker/k8s/manifests/service-server.yaml @@ -0,0 +1,19 @@ +apiVersion: v1 +kind: Service +metadata: + name: twentycrm-server + namespace: twentycrm +spec: + internalTrafficPolicy: Cluster + ports: + - name: http-tcp + port: 3000 + protocol: TCP + targetPort: 3000 + selector: + app: twentycrm-server + sessionAffinity: ClientIP + sessionAffinityConfig: + clientIP: + timeoutSeconds: 10800 + type: ClusterIP diff --git a/packages/twenty-docker/k8s/terraform/deployment-db.tf b/packages/twenty-docker/k8s/terraform/deployment-db.tf new file mode 100644 index 000000000000..c2a5a64b11a2 --- /dev/null +++ b/packages/twenty-docker/k8s/terraform/deployment-db.tf @@ -0,0 +1,90 @@ +resource "kubernetes_deployment" "twentycrm_db" { + metadata { + name = "${local.twentycrm_app_name}-db" + namespace = kubernetes_namespace.twentycrm.metadata.0.name + labels = { + app = "${local.twentycrm_app_name}-db" + } + } + + spec { + replicas = 1 + selector { + match_labels = { + app = "${local.twentycrm_app_name}-db" + } + } + + strategy { + type = "RollingUpdate" + rolling_update { + max_surge = "1" + max_unavailable = "1" + } + } + + template { + metadata { + labels = { + app = "${local.twentycrm_app_name}-db" + } + } + + spec { + # security_context { + # fs_group = 0 + # } + container { + image = local.twentycrm_db_image + name = local.twentycrm_app_name + stdin = true + tty = true + security_context { + allow_privilege_escalation = true + } + + env { + name = "POSTGRES_PASSWORD" + value = "twenty" + } + env { + name = "BITNAMI_DEBUG" + value = true + } + + port { + container_port = 5432 + protocol = "TCP" + } + + resources { + requests = { + cpu = "250m" + memory = "256Mi" + } + limits = { + cpu = "1000m" + memory = "1024Mi" + } + } + + volume_mount { + name = "nfs-twentycrm-db-data" + mount_path = "/bitnami/postgresql" + } + } + + volume { + name = "nfs-twentycrm-db-data" + + persistent_volume_claim { + claim_name = "nfs-twentycrm-db-data-pvc" + } + } + + dns_policy = "ClusterFirst" + restart_policy = "Always" + } + } + } +} diff --git a/packages/twenty-docker/k8s/terraform/deployment-server.tf b/packages/twenty-docker/k8s/terraform/deployment-server.tf new file mode 100644 index 000000000000..13a4d30bf708 --- /dev/null +++ b/packages/twenty-docker/k8s/terraform/deployment-server.tf @@ -0,0 +1,169 @@ +resource "kubernetes_deployment" "twentycrm_server" { + metadata { + name = "${local.twentycrm_app_name}-server" + namespace = kubernetes_namespace.twentycrm.metadata.0.name + labels = { + app = "${local.twentycrm_app_name}-server" + } + } + + spec { + replicas = 1 + selector { + match_labels = { + app = "${local.twentycrm_app_name}-server" + } + } + + strategy { + type = "RollingUpdate" + rolling_update { + max_surge = "1" + max_unavailable = "1" + } + } + + template { + metadata { + labels = { + app = "${local.twentycrm_app_name}-server" + } + } + + spec { + container { + image = local.twentycrm_server_image + name = local.twentycrm_app_name + stdin = true + tty = true + + security_context { + allow_privilege_escalation = true + privileged = true + run_as_user = 1000 + } + + env { + name = "PORT" + value = "3000" + } + env { + name = "DEBUG_MODE" + value = false + } + + env { + name = "SERVER_URL" + value = "https://crm.example.com:443" + } + + env { + name = "FRONT_BASE_URL" + value = "https://crm.example.com:443" + } + + env { + name = "BACKEND_SERVER_URL" + value = "https://crm.example.com:443" + } + + env { + name = "PG_DATABASE_URL" + value = "postgres://twenty:twenty@twentycrm-db.twentycrm.svc.cluster.local/default" + } + + env { + name = "ENABLE_DB_MIGRATIONS" + value = "true" + } + + env { + name = "SIGN_IN_PREFILLED" + value = "true" + } + + env { + name = "STORAGE_TYPE" + value = "local" + } + + env { + name = "ACCESS_TOKEN_SECRET" + value_from { + secret_key_ref { + name = "tokens" + key = "accessToken" + } + } + } + + env { + name = "LOGIN_TOKEN_SECRET" + value_from { + secret_key_ref { + name = "tokens" + key = "loginToken" + } + } + } + + env { + name = "REFRESH_TOKEN_SECRET" + value_from { + secret_key_ref { + name = "tokens" + key = "refreshToken" + } + } + } + + env { + name = "FILE_TOKEN_SECRET" + value_from { + secret_key_ref { + name = "tokens" + key = "fileToken" + } + } + } + + port { + container_port = 3000 + protocol = "TCP" + } + + resources { + requests = { + cpu = "250m" + memory = "256Mi" + } + limits = { + cpu = "1000m" + memory = "1024Mi" + } + } + + volume_mount { + name = "nfs-twentycrm-server-data" + mount_path = "/app/.local-storage" + } + } + + volume { + name = "nfs-twentycrm-server-data" + + persistent_volume_claim { + claim_name = "nfs-twentycrm-server-data-pvc" + } + } + + dns_policy = "ClusterFirst" + restart_policy = "Always" + } + } + } + depends_on = [ + kubernetes_deployment.twentycrm_db, + kubernetes_secret.twentycrm_tokens + ] +} diff --git a/packages/twenty-docker/k8s/terraform/ingress.tf b/packages/twenty-docker/k8s/terraform/ingress.tf new file mode 100644 index 000000000000..4276333b7f54 --- /dev/null +++ b/packages/twenty-docker/k8s/terraform/ingress.tf @@ -0,0 +1,30 @@ +resource "kubernetes_ingress" "twentycrm" { + wait_for_load_balancer = true + metadata { + name = "${local.twentycrm_app_name}-ingress" + namespace = kubernetes_namespace.twentycrm.metadata.0.name + annotations = { + "kubernetes.io/ingress.class" = "nginx" + "nginx.ingress.kubernetes.io/configuration-snippet" = < { + link, + workspace, + sender, + serverUrl, + }: SendInviteLinkEmailProps) => { + const workspaceLogo = getImageAbsoluteURIOrBase64(workspace.logo, serverUrl); return ( @@ -34,7 +38,7 @@ export const SendInviteLinkEmail = ({ <br /> </MainText> <HighlightedContainer> - {workspace.logo && <Img src={workspace.logo} width={40} height={40} />} + {workspaceLogo && <Img src={workspaceLogo} width={40} height={40} />} {workspace.name && <HighlightedText value={workspace.name} />} <CallToAction href={link} value="Accept invite" /> </HighlightedContainer> diff --git a/packages/twenty-emails/src/utils/getImageAbsoluteURIOrBase64.ts b/packages/twenty-emails/src/utils/getImageAbsoluteURIOrBase64.ts new file mode 100644 index 000000000000..cab930f76518 --- /dev/null +++ b/packages/twenty-emails/src/utils/getImageAbsoluteURIOrBase64.ts @@ -0,0 +1,16 @@ +export const getImageAbsoluteURIOrBase64 = ( + imageUrl?: string | null, + serverUrl?: string, +) => { + if (!imageUrl) { + return null; + } + + if (imageUrl?.startsWith('data:') || imageUrl?.startsWith('https:')) { + return imageUrl; + } + + return serverUrl?.endsWith('/') + ? `${serverUrl.substring(0, serverUrl.length - 1)}/files/${imageUrl}` + : `${serverUrl || ''}/files/${imageUrl}`; +}; diff --git a/packages/twenty-front/.storybook/main.ts b/packages/twenty-front/.storybook/main.ts index 764ae8aecf5e..633df49b3dae 100644 --- a/packages/twenty-front/.storybook/main.ts +++ b/packages/twenty-front/.storybook/main.ts @@ -45,17 +45,5 @@ const config: StorybookConfig = { name: '@storybook/react-vite', options: {}, }, - build: { - test: { - disableMDXEntries: true, - disabledAddons: [ - '@storybook/addon-docs', - '@storybook/addon-essentials/docs', - ], - }, - }, - docs: { - autodocs: false, - }, }; export default config; diff --git a/packages/twenty-front/codegen-metadata.cjs b/packages/twenty-front/codegen-metadata.cjs index 2d4e3fa895cf..d7ee2eb00c09 100644 --- a/packages/twenty-front/codegen-metadata.cjs +++ b/packages/twenty-front/codegen-metadata.cjs @@ -1,5 +1,5 @@ module.exports = { - schema: process.env.REACT_APP_SERVER_BASE_URL + '/metadata', + schema: (process.env.REACT_APP_SERVER_BASE_URL ?? 'http://localhost:3000') + '/metadata', documents: [ './src/modules/databases/graphql/**/*.ts', './src/modules/object-metadata/graphql/*.ts', diff --git a/packages/twenty-front/codegen.cjs b/packages/twenty-front/codegen.cjs index 461f7366cb99..fcc0ef27a5ce 100644 --- a/packages/twenty-front/codegen.cjs +++ b/packages/twenty-front/codegen.cjs @@ -1,5 +1,5 @@ module.exports = { - schema: process.env.REACT_APP_SERVER_BASE_URL + '/graphql', + schema: (process.env.REACT_APP_SERVER_BASE_URL ?? 'http://localhost:3000') + '/graphql', documents: [ '!./src/modules/databases/**', '!./src/modules/object-metadata/**', diff --git a/packages/twenty-front/jest.config.ts b/packages/twenty-front/jest.config.ts index 63c026a8c4ae..e21e9cbaf0a7 100644 --- a/packages/twenty-front/jest.config.ts +++ b/packages/twenty-front/jest.config.ts @@ -25,7 +25,7 @@ const jestConfig: JestConfigWithTsJest = { coverageThreshold: { global: { statements: 65, - lines: 65, + lines: 64, functions: 55, }, }, diff --git a/packages/twenty-front/package.json b/packages/twenty-front/package.json index 0e009c59c675..8ff1595fab09 100644 --- a/packages/twenty-front/package.json +++ b/packages/twenty-front/package.json @@ -1,6 +1,6 @@ { "name": "twenty-front", - "version": "0.20.0", + "version": "0.22.0", "private": true, "type": "module", "scripts": { diff --git a/packages/twenty-front/project.json b/packages/twenty-front/project.json index 332bfc4fafb1..4c427109b051 100644 --- a/packages/twenty-front/project.json +++ b/packages/twenty-front/project.json @@ -62,12 +62,9 @@ "storybook:build": { "options": { "env": { "NODE_OPTIONS": "--max_old_space_size=5000" } - }, - "configurations": { - "test": {} } }, - "storybook:dev": { + "storybook:serve:dev": { "options": { "port": 6006 }, "configurations": { "docs": { "env": { "STORYBOOK_SCOPE": "ui-docs" } }, @@ -76,11 +73,8 @@ "performance": { "env": { "STORYBOOK_SCOPE": "performance" } } } }, - "storybook:static": { - "options": { "port": 6006 }, - "configurations": { - "test": {} - } + "storybook:serve:static": { + "options": { "port": 6006 } }, "storybook:coverage": { "configurations": { @@ -100,18 +94,10 @@ "performance": { "env": { "STORYBOOK_SCOPE": "performance" } } } }, - "storybook:test:nocoverage": { - "configurations": { - "docs": { "env": { "STORYBOOK_SCOPE": "ui-docs" } }, - "modules": { "env": { "STORYBOOK_SCOPE": "modules" } }, - "pages": { "env": { "STORYBOOK_SCOPE": "pages" } }, - "performance": { "env": { "STORYBOOK_SCOPE": "performance" } } - } - }, - "storybook:static:test": { + "storybook:serve-and-test:static": { "options": { "commands": [ - "npx concurrently --kill-others --success=first -n SB,TEST 'nx storybook:static {projectName} --configuration=test --port={args.port}' 'npx wait-on tcp:{args.port} && nx storybook:test {projectName} --port={args.port} --configuration={args.scope}'" + "npx concurrently --kill-others --success=first -n SB,TEST 'nx storybook:serve:static {projectName} --port={args.port}' 'npx wait-on tcp:{args.port} && nx storybook:test {projectName} --port={args.port} --configuration={args.scope}'" ], "port": 6006 }, @@ -122,7 +108,15 @@ "performance": { "scope": "performance" } } }, - "storybook:performance:test": {}, + "storybook:serve-and-test:static:performance": {}, + "storybook:test:no-coverage": { + "configurations": { + "docs": { "env": { "STORYBOOK_SCOPE": "ui-docs" } }, + "modules": { "env": { "STORYBOOK_SCOPE": "modules" } }, + "pages": { "env": { "STORYBOOK_SCOPE": "pages" } }, + "performance": { "env": { "STORYBOOK_SCOPE": "performance" } } + } + }, "graphql:generate": { "executor": "nx:run-commands", "defaultConfiguration": "data", diff --git a/packages/twenty-front/src/App.tsx b/packages/twenty-front/src/App.tsx index 956a6c59e33a..4ea9a6ad3df5 100644 --- a/packages/twenty-front/src/App.tsx +++ b/packages/twenty-front/src/App.tsx @@ -53,9 +53,7 @@ import { PaymentSuccess } from '~/pages/onboarding/PaymentSuccess'; import { SyncEmails } from '~/pages/onboarding/SyncEmails'; import { SettingsAccounts } from '~/pages/settings/accounts/SettingsAccounts'; import { SettingsAccountsCalendars } from '~/pages/settings/accounts/SettingsAccountsCalendars'; -import { SettingsAccountsCalendarsSettings } from '~/pages/settings/accounts/SettingsAccountsCalendarsSettings'; import { SettingsAccountsEmails } from '~/pages/settings/accounts/SettingsAccountsEmails'; -import { SettingsAccountsEmailsInboxSettings } from '~/pages/settings/accounts/SettingsAccountsEmailsInboxSettings'; import { SettingsNewAccount } from '~/pages/settings/accounts/SettingsNewAccount'; import { SettingsNewObject } from '~/pages/settings/data-model/SettingsNewObject'; import { SettingsObjectDetail } from '~/pages/settings/data-model/SettingsObjectDetail'; @@ -180,18 +178,10 @@ const createRouter = (isBillingEnabled?: boolean) => path={SettingsPath.AccountsCalendars} element={<SettingsAccountsCalendars />} /> - <Route - path={SettingsPath.AccountsCalendarsSettings} - element={<SettingsAccountsCalendarsSettings />} - /> <Route path={SettingsPath.AccountsEmails} element={<SettingsAccountsEmails />} /> - <Route - path={SettingsPath.AccountsEmailsInboxSettings} - element={<SettingsAccountsEmailsInboxSettings />} - /> <Route path={SettingsPath.Billing} element={<SettingsBilling />} diff --git a/packages/twenty-front/src/__stories__/App.stories.tsx b/packages/twenty-front/src/__stories__/App.stories.tsx index 761c64f3bb82..c5314b5652a1 100644 --- a/packages/twenty-front/src/__stories__/App.stories.tsx +++ b/packages/twenty-front/src/__stories__/App.stories.tsx @@ -13,7 +13,7 @@ import { SnackBarProviderScope } from '@/ui/feedback/snack-bar-manager/scopes/Sn import { GET_CURRENT_USER } from '@/users/graphql/queries/getCurrentUser'; import { App } from '~/App'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedUsersData } from '~/testing/mock-data/users'; +import { mockedUserData } from '~/testing/mock-data/users'; const meta: Meta<typeof App> = { title: 'App/App', @@ -67,9 +67,9 @@ export const DarkMode: Story = { return HttpResponse.json({ data: { currentUser: { - ...mockedUsersData[0], + ...mockedUserData, workspaceMember: { - ...mockedUsersData[0].workspaceMember, + ...mockedUserData.workspaceMember, colorScheme: 'Dark', }, }, diff --git a/packages/twenty-front/src/generated-metadata/graphql.ts b/packages/twenty-front/src/generated-metadata/graphql.ts index 05155eaa108e..958328038e5c 100644 --- a/packages/twenty-front/src/generated-metadata/graphql.ts +++ b/packages/twenty-front/src/generated-metadata/graphql.ts @@ -36,6 +36,11 @@ export type Analytics = { success: Scalars['Boolean']['output']; }; +export type ApiConfig = { + __typename?: 'ApiConfig'; + mutationMaximumAffectedRecords: Scalars['Float']['output']; +}; + export type ApiKeyToken = { __typename?: 'ApiKeyToken'; token: Scalars['String']['output']; @@ -98,8 +103,8 @@ export type Billing = { export type BillingSubscription = { __typename?: 'BillingSubscription'; id: Scalars['UUID']['output']; - interval?: Maybe<Scalars['String']['output']>; - status: Scalars['String']['output']; + interval?: Maybe<SubscriptionInterval>; + status: SubscriptionStatus; }; export type BillingSubscriptionFilter = { @@ -136,12 +141,13 @@ export type Captcha = { }; export enum CaptchaDriverType { - GoogleRecatpcha = 'GoogleRecatpcha', + GoogleRecaptcha = 'GoogleRecaptcha', Turnstile = 'Turnstile' } export type ClientConfig = { __typename?: 'ClientConfig'; + api: ApiConfig; authProviders: AuthProviders; billing: Billing; captcha: Captcha; @@ -327,7 +333,6 @@ export enum FieldMetadataType { Numeric = 'NUMERIC', Phone = 'PHONE', Position = 'POSITION', - Probability = 'PROBABILITY', Rating = 'RATING', RawJson = 'RAW_JSON', Relation = 'RELATION', @@ -398,7 +403,9 @@ export type Mutation = { deleteOneRelation: Relation; deleteOneRemoteServer: RemoteServer; deleteUser: User; + disablePostgresProxy: PostgresCredentials; emailPasswordResetLink: EmailPasswordResetLink; + enablePostgresProxy: PostgresCredentials; exchangeAuthorizationCode: ExchangeAuthCode; generateApiKeyToken: ApiKeyToken; generateJWT: AuthTokens; @@ -451,7 +458,7 @@ export type MutationChallengeArgs = { export type MutationCheckoutSessionArgs = { - recurringInterval: Scalars['String']['input']; + recurringInterval: SubscriptionInterval; successUrlPath?: InputMaybe<Scalars['String']['input']>; }; @@ -636,10 +643,14 @@ export type ObjectFieldsConnection = { pageInfo: PageInfo; }; -/** Onboarding step */ -export enum OnboardingStep { +/** Onboarding status */ +export enum OnboardingStatus { + Completed = 'COMPLETED', InviteTeam = 'INVITE_TEAM', - SyncEmail = 'SYNC_EMAIL' + PlanRequired = 'PLAN_REQUIRED', + ProfileCreation = 'PROFILE_CREATION', + SyncEmail = 'SYNC_EMAIL', + WorkspaceActivation = 'WORKSPACE_ACTIVATION' } export type OnboardingStepSuccess = { @@ -660,10 +671,18 @@ export type PageInfo = { startCursor?: Maybe<Scalars['ConnectionCursor']['output']>; }; +export type PostgresCredentials = { + __typename?: 'PostgresCredentials'; + id: Scalars['UUID']['output']; + password: Scalars['String']['output']; + user: Scalars['String']['output']; + workspaceId: Scalars['String']['output']; +}; + export type ProductPriceEntity = { __typename?: 'ProductPriceEntity'; created: Scalars['Float']['output']; - recurringInterval: Scalars['String']['output']; + recurringInterval: SubscriptionInterval; stripePriceId: Scalars['String']['output']; unitAmount: Scalars['Float']['output']; }; @@ -688,6 +707,7 @@ export type Query = { findManyRemoteServersByType: Array<RemoteServer>; findOneRemoteServerById: RemoteServer; findWorkspaceFromInviteHash: Workspace; + getPostgresCredentials?: Maybe<PostgresCredentials>; getProductPrices: ProductPricesEntity; getTimelineCalendarEventsFromCompanyId: TimelineCalendarEventsWithTotal; getTimelineCalendarEventsFromPersonId: TimelineCalendarEventsWithTotal; @@ -912,6 +932,24 @@ export enum SortNulls { NullsLast = 'NULLS_LAST' } +export enum SubscriptionInterval { + Day = 'Day', + Month = 'Month', + Week = 'Week', + Year = 'Year' +} + +export enum SubscriptionStatus { + Active = 'Active', + Canceled = 'Canceled', + Incomplete = 'Incomplete', + IncompleteExpired = 'IncompleteExpired', + PastDue = 'PastDue', + Paused = 'Paused', + Trialing = 'Trialing', + Unpaid = 'Unpaid' +} + export type Support = { __typename?: 'Support'; supportDriver: Scalars['String']['output']; @@ -1084,7 +1122,7 @@ export type User = { firstName: Scalars['String']['output']; id: Scalars['UUID']['output']; lastName: Scalars['String']['output']; - onboardingStep?: Maybe<OnboardingStep>; + onboardingStatus?: Maybe<OnboardingStatus>; passwordHash?: Maybe<Scalars['String']['output']>; /** @deprecated field migrated into the AppTokens Table ref: https://github.com/twentyhq/twenty/issues/5021 */ passwordResetToken?: Maybe<Scalars['String']['output']>; @@ -1163,8 +1201,8 @@ export type Workspace = { id: Scalars['UUID']['output']; inviteHash?: Maybe<Scalars['String']['output']>; logo?: Maybe<Scalars['String']['output']>; - subscriptionStatus: Scalars['String']['output']; updatedAt: Scalars['DateTime']['output']; + workspaceMembersCount?: Maybe<Scalars['Float']['output']>; }; diff --git a/packages/twenty-front/src/generated/graphql.tsx b/packages/twenty-front/src/generated/graphql.tsx index d153c73c0ada..4b1a828b4469 100644 --- a/packages/twenty-front/src/generated/graphql.tsx +++ b/packages/twenty-front/src/generated/graphql.tsx @@ -20,6 +20,13 @@ export type Scalars = { Upload: any; }; +export type AisqlQueryResult = { + __typename?: 'AISQLQueryResult'; + queryFailedErrorMessage?: Maybe<Scalars['String']>; + sqlQuery: Scalars['String']; + sqlQueryResult?: Maybe<Scalars['String']>; +}; + export type ActivateWorkspaceInput = { displayName?: InputMaybe<Scalars['String']>; }; @@ -30,6 +37,11 @@ export type Analytics = { success: Scalars['Boolean']; }; +export type ApiConfig = { + __typename?: 'ApiConfig'; + mutationMaximumAffectedRecords: Scalars['Float']; +}; + export type ApiKeyToken = { __typename?: 'ApiKeyToken'; token: Scalars['String']; @@ -92,8 +104,8 @@ export type Billing = { export type BillingSubscription = { __typename?: 'BillingSubscription'; id: Scalars['UUID']; - interval?: Maybe<Scalars['String']>; - status: Scalars['String']; + interval?: Maybe<SubscriptionInterval>; + status: SubscriptionStatus; }; export type BillingSubscriptionFilter = { @@ -130,12 +142,13 @@ export type Captcha = { }; export enum CaptchaDriverType { - GoogleRecatpcha = 'GoogleRecatpcha', + GoogleRecaptcha = 'GoogleRecaptcha', Turnstile = 'Turnstile' } export type ClientConfig = { __typename?: 'ClientConfig'; + api: ApiConfig; authProviders: AuthProviders; billing: Billing; captcha: Captcha; @@ -233,7 +246,6 @@ export enum FieldMetadataType { Numeric = 'NUMERIC', Phone = 'PHONE', Position = 'POSITION', - Probability = 'PROBABILITY', Rating = 'RATING', RawJson = 'RAW_JSON', Relation = 'RELATION', @@ -291,7 +303,9 @@ export type Mutation = { deleteCurrentWorkspace: Workspace; deleteOneObject: Object; deleteUser: User; + disablePostgresProxy: PostgresCredentials; emailPasswordResetLink: EmailPasswordResetLink; + enablePostgresProxy: PostgresCredentials; exchangeAuthorizationCode: ExchangeAuthCode; generateApiKeyToken: ApiKeyToken; generateJWT: AuthTokens; @@ -339,7 +353,7 @@ export type MutationChallengeArgs = { export type MutationCheckoutSessionArgs = { - recurringInterval: Scalars['String']; + recurringInterval: SubscriptionInterval; successUrlPath?: InputMaybe<Scalars['String']>; }; @@ -459,10 +473,14 @@ export type ObjectFieldsConnection = { pageInfo: PageInfo; }; -/** Onboarding step */ -export enum OnboardingStep { +/** Onboarding status */ +export enum OnboardingStatus { + Completed = 'COMPLETED', InviteTeam = 'INVITE_TEAM', - SyncEmail = 'SYNC_EMAIL' + PlanRequired = 'PLAN_REQUIRED', + ProfileCreation = 'PROFILE_CREATION', + SyncEmail = 'SYNC_EMAIL', + WorkspaceActivation = 'WORKSPACE_ACTIVATION' } export type OnboardingStepSuccess = { @@ -483,10 +501,18 @@ export type PageInfo = { startCursor?: Maybe<Scalars['ConnectionCursor']>; }; +export type PostgresCredentials = { + __typename?: 'PostgresCredentials'; + id: Scalars['UUID']; + password: Scalars['String']; + user: Scalars['String']; + workspaceId: Scalars['String']; +}; + export type ProductPriceEntity = { __typename?: 'ProductPriceEntity'; created: Scalars['Float']; - recurringInterval: Scalars['String']; + recurringInterval: SubscriptionInterval; stripePriceId: Scalars['String']; unitAmount: Scalars['Float']; }; @@ -506,6 +532,8 @@ export type Query = { currentUser: User; currentWorkspace: Workspace; findWorkspaceFromInviteHash: Workspace; + getAISQLQuery: AisqlQueryResult; + getPostgresCredentials?: Maybe<PostgresCredentials>; getProductPrices: ProductPricesEntity; getTimelineCalendarEventsFromCompanyId: TimelineCalendarEventsWithTotal; getTimelineCalendarEventsFromPersonId: TimelineCalendarEventsWithTotal; @@ -538,6 +566,11 @@ export type QueryFindWorkspaceFromInviteHashArgs = { }; +export type QueryGetAisqlQueryArgs = { + text: Scalars['String']; +}; + + export type QueryGetProductPricesArgs = { product: Scalars['String']; }; @@ -667,6 +700,24 @@ export enum SortNulls { NullsLast = 'NULLS_LAST' } +export enum SubscriptionInterval { + Day = 'Day', + Month = 'Month', + Week = 'Week', + Year = 'Year' +} + +export enum SubscriptionStatus { + Active = 'Active', + Canceled = 'Canceled', + Incomplete = 'Incomplete', + IncompleteExpired = 'IncompleteExpired', + PastDue = 'PastDue', + Paused = 'Paused', + Trialing = 'Trialing', + Unpaid = 'Unpaid' +} + export type Support = { __typename?: 'Support'; supportDriver: Scalars['String']; @@ -810,7 +861,7 @@ export type User = { firstName: Scalars['String']; id: Scalars['UUID']; lastName: Scalars['String']; - onboardingStep?: Maybe<OnboardingStep>; + onboardingStatus?: Maybe<OnboardingStatus>; passwordHash?: Maybe<Scalars['String']>; /** @deprecated field migrated into the AppTokens Table ref: https://github.com/twentyhq/twenty/issues/5021 */ passwordResetToken?: Maybe<Scalars['String']>; @@ -879,8 +930,8 @@ export type Workspace = { id: Scalars['UUID']; inviteHash?: Maybe<Scalars['String']>; logo?: Maybe<Scalars['String']>; - subscriptionStatus: Scalars['String']; updatedAt: Scalars['DateTime']; + workspaceMembersCount?: Maybe<Scalars['Float']>; }; @@ -1061,8 +1112,6 @@ export type GetTimelineThreadsFromPersonIdQueryVariables = Exact<{ export type GetTimelineThreadsFromPersonIdQuery = { __typename?: 'Query', getTimelineThreadsFromPersonId: { __typename?: 'TimelineThreadsWithTotal', totalNumberOfThreads: number, timelineThreads: Array<{ __typename?: 'TimelineThread', id: any, read: boolean, visibility: MessageChannelVisibility, lastMessageReceivedAt: string, lastMessageBody: string, subject: string, numberOfMessagesInThread: number, participantCount: number, firstParticipant: { __typename?: 'TimelineThreadParticipant', personId?: any | null, workspaceMemberId?: any | null, firstName: string, lastName: string, displayName: string, avatarUrl: string, handle: string }, lastTwoParticipants: Array<{ __typename?: 'TimelineThreadParticipant', personId?: any | null, workspaceMemberId?: any | null, firstName: string, lastName: string, displayName: string, avatarUrl: string, handle: string }> }> } }; -export type TimelineThreadFragment = { __typename?: 'TimelineThread', id: any, subject: string, lastMessageReceivedAt: string }; - export type TrackMutationVariables = Exact<{ type: Scalars['String']; data: Scalars['JSON']; @@ -1141,7 +1190,7 @@ export type ImpersonateMutationVariables = Exact<{ }>; -export type ImpersonateMutation = { __typename?: 'Mutation', impersonate: { __typename?: 'Verify', user: { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStep?: OnboardingStep | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, subscriptionStatus: string, activationStatus: string, currentCacheVersion?: string | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: string, interval?: string | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> }, tokens: { __typename?: 'AuthTokenPair', accessToken: { __typename?: 'AuthToken', token: string, expiresAt: string }, refreshToken: { __typename?: 'AuthToken', token: string, expiresAt: string } } } }; +export type ImpersonateMutation = { __typename?: 'Mutation', impersonate: { __typename?: 'Verify', user: { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStatus?: OnboardingStatus | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, activationStatus: string, currentCacheVersion?: string | null, workspaceMembersCount?: number | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: SubscriptionStatus, interval?: SubscriptionInterval | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> }, tokens: { __typename?: 'AuthTokenPair', accessToken: { __typename?: 'AuthToken', token: string, expiresAt: string }, refreshToken: { __typename?: 'AuthToken', token: string, expiresAt: string } } } }; export type RenewTokenMutationVariables = Exact<{ appToken: Scalars['String']; @@ -1173,7 +1222,7 @@ export type VerifyMutationVariables = Exact<{ }>; -export type VerifyMutation = { __typename?: 'Mutation', verify: { __typename?: 'Verify', user: { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStep?: OnboardingStep | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, subscriptionStatus: string, activationStatus: string, currentCacheVersion?: string | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: string, interval?: string | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> }, tokens: { __typename?: 'AuthTokenPair', accessToken: { __typename?: 'AuthToken', token: string, expiresAt: string }, refreshToken: { __typename?: 'AuthToken', token: string, expiresAt: string } } } }; +export type VerifyMutation = { __typename?: 'Mutation', verify: { __typename?: 'Verify', user: { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStatus?: OnboardingStatus | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, activationStatus: string, currentCacheVersion?: string | null, workspaceMembersCount?: number | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: SubscriptionStatus, interval?: SubscriptionInterval | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> }, tokens: { __typename?: 'AuthTokenPair', accessToken: { __typename?: 'AuthToken', token: string, expiresAt: string }, refreshToken: { __typename?: 'AuthToken', token: string, expiresAt: string } } } }; export type CheckUserExistsQueryVariables = Exact<{ email: Scalars['String']; @@ -1198,7 +1247,7 @@ export type BillingPortalSessionQueryVariables = Exact<{ export type BillingPortalSessionQuery = { __typename?: 'Query', billingPortalSession: { __typename?: 'SessionEntity', url?: string | null } }; export type CheckoutSessionMutationVariables = Exact<{ - recurringInterval: Scalars['String']; + recurringInterval: SubscriptionInterval; successUrlPath?: InputMaybe<Scalars['String']>; }>; @@ -1210,7 +1259,7 @@ export type GetProductPricesQueryVariables = Exact<{ }>; -export type GetProductPricesQuery = { __typename?: 'Query', getProductPrices: { __typename?: 'ProductPricesEntity', productPrices: Array<{ __typename?: 'ProductPriceEntity', created: number, recurringInterval: string, stripePriceId: string, unitAmount: number }> } }; +export type GetProductPricesQuery = { __typename?: 'Query', getProductPrices: { __typename?: 'ProductPricesEntity', productPrices: Array<{ __typename?: 'ProductPriceEntity', created: number, recurringInterval: SubscriptionInterval, stripePriceId: string, unitAmount: number }> } }; export type UpdateBillingSubscriptionMutationVariables = Exact<{ [key: string]: never; }>; @@ -1220,14 +1269,21 @@ export type UpdateBillingSubscriptionMutation = { __typename?: 'Mutation', updat export type GetClientConfigQueryVariables = Exact<{ [key: string]: never; }>; -export type GetClientConfigQuery = { __typename?: 'Query', clientConfig: { __typename?: 'ClientConfig', signInPrefilled: boolean, signUpDisabled: boolean, debugMode: boolean, chromeExtensionId?: string | null, authProviders: { __typename?: 'AuthProviders', google: boolean, password: boolean, microsoft: boolean }, billing: { __typename?: 'Billing', isBillingEnabled: boolean, billingUrl?: string | null, billingFreeTrialDurationInDays?: number | null }, telemetry: { __typename?: 'Telemetry', enabled: boolean }, support: { __typename?: 'Support', supportDriver: string, supportFrontChatId?: string | null }, sentry: { __typename?: 'Sentry', dsn?: string | null, environment?: string | null, release?: string | null }, captcha: { __typename?: 'Captcha', provider?: CaptchaDriverType | null, siteKey?: string | null } } }; +export type GetClientConfigQuery = { __typename?: 'Query', clientConfig: { __typename?: 'ClientConfig', signInPrefilled: boolean, signUpDisabled: boolean, debugMode: boolean, chromeExtensionId?: string | null, authProviders: { __typename?: 'AuthProviders', google: boolean, password: boolean, microsoft: boolean }, billing: { __typename?: 'Billing', isBillingEnabled: boolean, billingUrl?: string | null, billingFreeTrialDurationInDays?: number | null }, telemetry: { __typename?: 'Telemetry', enabled: boolean }, support: { __typename?: 'Support', supportDriver: string, supportFrontChatId?: string | null }, sentry: { __typename?: 'Sentry', dsn?: string | null, environment?: string | null, release?: string | null }, captcha: { __typename?: 'Captcha', provider?: CaptchaDriverType | null, siteKey?: string | null }, api: { __typename?: 'ApiConfig', mutationMaximumAffectedRecords: number } } }; export type SkipSyncEmailOnboardingStepMutationVariables = Exact<{ [key: string]: never; }>; export type SkipSyncEmailOnboardingStepMutation = { __typename?: 'Mutation', skipSyncEmailOnboardingStep: { __typename?: 'OnboardingStepSuccess', success: boolean } }; -export type UserQueryFragmentFragment = { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStep?: OnboardingStep | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, subscriptionStatus: string, activationStatus: string, currentCacheVersion?: string | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: string, interval?: string | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> }; +export type GetAisqlQueryQueryVariables = Exact<{ + text: Scalars['String']; +}>; + + +export type GetAisqlQueryQuery = { __typename?: 'Query', getAISQLQuery: { __typename?: 'AISQLQueryResult', sqlQuery: string, sqlQueryResult?: string | null, queryFailedErrorMessage?: string | null } }; + +export type UserQueryFragmentFragment = { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStatus?: OnboardingStatus | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, activationStatus: string, currentCacheVersion?: string | null, workspaceMembersCount?: number | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: SubscriptionStatus, interval?: SubscriptionInterval | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> }; export type DeleteUserAccountMutationVariables = Exact<{ [key: string]: never; }>; @@ -1244,7 +1300,7 @@ export type UploadProfilePictureMutation = { __typename?: 'Mutation', uploadProf export type GetCurrentUserQueryVariables = Exact<{ [key: string]: never; }>; -export type GetCurrentUserQuery = { __typename?: 'Query', currentUser: { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStep?: OnboardingStep | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, subscriptionStatus: string, activationStatus: string, currentCacheVersion?: string | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: string, interval?: string | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> } }; +export type GetCurrentUserQuery = { __typename?: 'Query', currentUser: { __typename?: 'User', id: any, firstName: string, lastName: string, email: string, canImpersonate: boolean, supportUserHash?: string | null, onboardingStatus?: OnboardingStatus | null, workspaceMember?: { __typename?: 'WorkspaceMember', id: any, colorScheme: string, avatarUrl?: string | null, locale: string, name: { __typename?: 'FullName', firstName: string, lastName: string } } | null, defaultWorkspace: { __typename?: 'Workspace', id: any, displayName?: string | null, logo?: string | null, domainName?: string | null, inviteHash?: string | null, allowImpersonation: boolean, activationStatus: string, currentCacheVersion?: string | null, workspaceMembersCount?: number | null, featureFlags?: Array<{ __typename?: 'FeatureFlag', id: any, key: string, value: boolean, workspaceId: string }> | null, currentBillingSubscription?: { __typename?: 'BillingSubscription', id: any, status: SubscriptionStatus, interval?: SubscriptionInterval | null } | null }, workspaces: Array<{ __typename?: 'UserWorkspace', workspace?: { __typename?: 'Workspace', id: any, logo?: string | null, displayName?: string | null, domainName?: string | null } | null }> } }; export type AddUserToWorkspaceMutationVariables = Exact<{ inviteHash: Scalars['String']; @@ -1277,7 +1333,7 @@ export type UpdateWorkspaceMutationVariables = Exact<{ }>; -export type UpdateWorkspaceMutation = { __typename?: 'Mutation', updateWorkspace: { __typename?: 'Workspace', id: any, domainName?: string | null, displayName?: string | null, logo?: string | null, allowImpersonation: boolean, subscriptionStatus: string } }; +export type UpdateWorkspaceMutation = { __typename?: 'Mutation', updateWorkspace: { __typename?: 'Workspace', id: any, domainName?: string | null, displayName?: string | null, logo?: string | null, allowImpersonation: boolean } }; export type UploadWorkspaceLogoMutationVariables = Exact<{ file: Scalars['Upload']; @@ -1364,13 +1420,6 @@ export const TimelineThreadsWithTotalFragmentFragmentDoc = gql` } } ${TimelineThreadFragmentFragmentDoc}`; -export const TimelineThreadFragmentDoc = gql` - fragment timelineThread on TimelineThread { - id - subject - lastMessageReceivedAt -} - `; export const AuthTokenFragmentFragmentDoc = gql` fragment AuthTokenFragment on AuthToken { token @@ -1395,7 +1444,7 @@ export const UserQueryFragmentFragmentDoc = gql` email canImpersonate supportUserHash - onboardingStep + onboardingStatus workspaceMember { id name { @@ -1413,7 +1462,6 @@ export const UserQueryFragmentFragmentDoc = gql` domainName inviteHash allowImpersonation - subscriptionStatus activationStatus featureFlags { id @@ -1427,6 +1475,7 @@ export const UserQueryFragmentFragmentDoc = gql` status interval } + workspaceMembersCount } workspaces { workspace { @@ -2213,7 +2262,7 @@ export type BillingPortalSessionQueryHookResult = ReturnType<typeof useBillingPo export type BillingPortalSessionLazyQueryHookResult = ReturnType<typeof useBillingPortalSessionLazyQuery>; export type BillingPortalSessionQueryResult = Apollo.QueryResult<BillingPortalSessionQuery, BillingPortalSessionQueryVariables>; export const CheckoutSessionDocument = gql` - mutation CheckoutSession($recurringInterval: String!, $successUrlPath: String) { + mutation CheckoutSession($recurringInterval: SubscriptionInterval!, $successUrlPath: String) { checkoutSession( recurringInterval: $recurringInterval successUrlPath: $successUrlPath @@ -2353,6 +2402,9 @@ export const GetClientConfigDocument = gql` provider siteKey } + api { + mutationMaximumAffectedRecords + } chromeExtensionId } } @@ -2416,6 +2468,43 @@ export function useSkipSyncEmailOnboardingStepMutation(baseOptions?: Apollo.Muta export type SkipSyncEmailOnboardingStepMutationHookResult = ReturnType<typeof useSkipSyncEmailOnboardingStepMutation>; export type SkipSyncEmailOnboardingStepMutationResult = Apollo.MutationResult<SkipSyncEmailOnboardingStepMutation>; export type SkipSyncEmailOnboardingStepMutationOptions = Apollo.BaseMutationOptions<SkipSyncEmailOnboardingStepMutation, SkipSyncEmailOnboardingStepMutationVariables>; +export const GetAisqlQueryDocument = gql` + query GetAISQLQuery($text: String!) { + getAISQLQuery(text: $text) { + sqlQuery + sqlQueryResult + queryFailedErrorMessage + } +} + `; + +/** + * __useGetAisqlQueryQuery__ + * + * To run a query within a React component, call `useGetAisqlQueryQuery` and pass it any options that fit your needs. + * When your component renders, `useGetAisqlQueryQuery` returns an object from Apollo Client that contains loading, error, and data properties + * you can use to render your UI. + * + * @param baseOptions options that will be passed into the query, supported options are listed on: https://www.apollographql.com/docs/react/api/react-hooks/#options; + * + * @example + * const { data, loading, error } = useGetAisqlQueryQuery({ + * variables: { + * text: // value for 'text' + * }, + * }); + */ +export function useGetAisqlQueryQuery(baseOptions: Apollo.QueryHookOptions<GetAisqlQueryQuery, GetAisqlQueryQueryVariables>) { + const options = {...defaultOptions, ...baseOptions} + return Apollo.useQuery<GetAisqlQueryQuery, GetAisqlQueryQueryVariables>(GetAisqlQueryDocument, options); + } +export function useGetAisqlQueryLazyQuery(baseOptions?: Apollo.LazyQueryHookOptions<GetAisqlQueryQuery, GetAisqlQueryQueryVariables>) { + const options = {...defaultOptions, ...baseOptions} + return Apollo.useLazyQuery<GetAisqlQueryQuery, GetAisqlQueryQueryVariables>(GetAisqlQueryDocument, options); + } +export type GetAisqlQueryQueryHookResult = ReturnType<typeof useGetAisqlQueryQuery>; +export type GetAisqlQueryLazyQueryHookResult = ReturnType<typeof useGetAisqlQueryLazyQuery>; +export type GetAisqlQueryQueryResult = Apollo.QueryResult<GetAisqlQueryQuery, GetAisqlQueryQueryVariables>; export const DeleteUserAccountDocument = gql` mutation DeleteUserAccount { deleteUser { @@ -2652,7 +2741,6 @@ export const UpdateWorkspaceDocument = gql` displayName logo allowImpersonation - subscriptionStatus } } `; diff --git a/packages/twenty-front/src/hooks/__tests__/useDefaultHomePagePath.test.ts b/packages/twenty-front/src/hooks/__tests__/useDefaultHomePagePath.test.ts index 1d2834937e35..709c071cce76 100644 --- a/packages/twenty-front/src/hooks/__tests__/useDefaultHomePagePath.test.ts +++ b/packages/twenty-front/src/hooks/__tests__/useDefaultHomePagePath.test.ts @@ -7,7 +7,7 @@ import { getObjectMetadataItemsMock } from '@/object-metadata/utils/getObjectMet import { usePrefetchedData } from '@/prefetch/hooks/usePrefetchedData'; import { AppPath } from '@/types/AppPath'; import { useDefaultHomePagePath } from '~/hooks/useDefaultHomePagePath'; -import { mockedUsersData } from '~/testing/mock-data/users'; +import { mockedUserData } from '~/testing/mock-data/users'; const objectMetadataItem = getObjectMetadataItemsMock()[0]; jest.mock('@/object-metadata/hooks/useObjectMetadataItem'); @@ -36,7 +36,7 @@ const renderHooks = (withCurrentUser: boolean) => { () => { const setCurrentUser = useSetRecoilState(currentUserState); if (withCurrentUser) { - setCurrentUser(mockedUsersData[0]); + setCurrentUser(mockedUserData); } return useDefaultHomePagePath(); }, diff --git a/packages/twenty-front/src/hooks/__tests__/usePageChangeEffectNavigateLocation.test.ts b/packages/twenty-front/src/hooks/__tests__/usePageChangeEffectNavigateLocation.test.ts index cf9a7730cd94..327e940ac3a5 100644 --- a/packages/twenty-front/src/hooks/__tests__/usePageChangeEffectNavigateLocation.test.ts +++ b/packages/twenty-front/src/hooks/__tests__/usePageChangeEffectNavigateLocation.test.ts @@ -1,15 +1,26 @@ -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; -import { OnboardingStatus } from '@/auth/utils/getOnboardingStatus'; +import { useIsLogged } from '@/auth/hooks/useIsLogged'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; import { AppPath } from '@/types/AppPath'; +import { useSubscriptionStatus } from '@/workspace/hooks/useSubscriptionStatus'; +import { OnboardingStatus, SubscriptionStatus } from '~/generated/graphql'; import { useDefaultHomePagePath } from '~/hooks/useDefaultHomePagePath'; import { useIsMatchingLocation } from '~/hooks/useIsMatchingLocation'; import { usePageChangeEffectNavigateLocation } from '~/hooks/usePageChangeEffectNavigateLocation'; -jest.mock('@/auth/hooks/useOnboardingStatus'); -const setupMockOnboardingStatus = (onboardingStatus: OnboardingStatus) => { +jest.mock('@/onboarding/hooks/useOnboardingStatus'); +const setupMockOnboardingStatus = ( + onboardingStatus: OnboardingStatus | undefined, +) => { jest.mocked(useOnboardingStatus).mockReturnValueOnce(onboardingStatus); }; +jest.mock('@/workspace/hooks/useSubscriptionStatus'); +const setupMockSubscriptionStatus = ( + subscriptionStatus: SubscriptionStatus | undefined, +) => { + jest.mocked(useSubscriptionStatus).mockReturnValueOnce(subscriptionStatus); +}; + jest.mock('~/hooks/useIsMatchingLocation'); const mockUseIsMatchingLocation = jest.mocked(useIsMatchingLocation); @@ -19,281 +30,276 @@ const setupMockIsMatchingLocation = (pathname: string) => { ); }; +jest.mock('@/auth/hooks/useIsLogged'); +const setupMockIsLogged = (isLogged: boolean) => { + jest.mocked(useIsLogged).mockReturnValueOnce(isLogged); +}; + const defaultHomePagePath = '/objects/companies'; jest.mock('~/hooks/useDefaultHomePagePath'); jest.mocked(useDefaultHomePagePath).mockReturnValue({ - defaultHomePagePath: '/objects/companies', + defaultHomePagePath, }); // prettier-ignore const testCases = [ - { loc: AppPath.Verify, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.Verify, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.Verify, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.Verify, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingUserCreation, res: undefined }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.Verify, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.Verify, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.Verify, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: undefined }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.Verify, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.SignInUp, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.SignInUp, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.SignInUp, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.SignInUp, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingUserCreation, res: undefined }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.SignInUp, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.SignInUp, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.SignInUp, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: undefined }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.SignInUp, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.Invite, status: OnboardingStatus.Incomplete, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.Canceled, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.Unpaid, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingUserCreation, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingWorkspaceActivation, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingProfileCreation, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingSyncEmail, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingInviteTeam, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.Invite, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: undefined }, + { loc: AppPath.Invite, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.Incomplete, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.Canceled, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.Unpaid, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingUserCreation, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingWorkspaceActivation, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingProfileCreation, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingSyncEmail, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingInviteTeam, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: undefined }, + { loc: AppPath.ResetPassword, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingWorkspaceActivation, res: undefined }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.CreateWorkspace, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: undefined }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.CreateWorkspace, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingProfileCreation, res: undefined }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.CreateProfile, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: undefined }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.CreateProfile, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingSyncEmail, res: undefined }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.SyncEmails, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: undefined }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.SyncEmails, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingInviteTeam, res: undefined }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.InviteTeam, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: undefined }, + { loc: AppPath.InviteTeam, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.Incomplete, res: undefined }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.Canceled, res: undefined }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.Unpaid, res: undefined }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: undefined }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.PlanRequired, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.PlanRequired, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingWorkspaceActivation, res: undefined }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: undefined }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.PlanRequiredSuccess, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.Index, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.Index, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.Index, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.Index, status: OnboardingStatus.PastDue, res: defaultHomePagePath }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.Index, status: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.Index, status: OnboardingStatus.CompletedWithoutSubscription, res: defaultHomePagePath }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, + { loc: AppPath.Index, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.Index, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: defaultHomePagePath }, - { loc: AppPath.TasksPage, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.TasksPage, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.TasksPage, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.TasksPage, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.TasksPage, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.TasksPage, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.TasksPage, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.TasksPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.OpportunitiesPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.RecordIndexPage, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.RecordIndexPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.RecordShowPage, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.RecordShowPage, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Canceled, res: undefined }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Unpaid, res: undefined }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.SettingsCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.DevelopersCatchAll, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.Impersonate, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.Impersonate, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.Impersonate, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.Impersonate, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.Impersonate, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.Impersonate, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.Impersonate, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.Impersonate, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.Authorize, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.Authorize, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.Authorize, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.Authorize, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.Authorize, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.Authorize, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.Authorize, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.Authorize, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.NotFoundWildcard, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.NotFound, status: OnboardingStatus.Incomplete, res: AppPath.PlanRequired }, - { loc: AppPath.NotFound, status: OnboardingStatus.Canceled, res: '/settings/billing' }, - { loc: AppPath.NotFound, status: OnboardingStatus.Unpaid, res: '/settings/billing' }, - { loc: AppPath.NotFound, status: OnboardingStatus.PastDue, res: undefined }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingUserCreation, res: AppPath.SignInUp }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingWorkspaceActivation, res: AppPath.CreateWorkspace }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingProfileCreation, res: AppPath.CreateProfile }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingSyncEmail, res: AppPath.SyncEmails }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingInviteTeam, res: AppPath.InviteTeam }, - { loc: AppPath.NotFound, status: OnboardingStatus.Completed, res: undefined }, - { loc: AppPath.NotFound, status: OnboardingStatus.CompletedWithoutSubscription, res: undefined }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: AppPath.PlanRequired }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: '/settings/billing' }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: undefined }, + { loc: AppPath.NotFound, isLoggedIn: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: AppPath.SignInUp }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: AppPath.CreateWorkspace }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: AppPath.CreateProfile }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: AppPath.SyncEmails }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: AppPath.InviteTeam }, + { loc: AppPath.NotFound, isLoggedIn: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: undefined }, ]; describe('usePageChangeEffectNavigateLocation', () => { testCases.forEach((testCase) => { - it(`with location ${testCase.loc} and onboardingStatus ${testCase.status} should return ${testCase.res}`, () => { + it(`with location ${testCase.loc} and onboardingStatus ${testCase.onboardingStatus} and subscriptionStatus ${testCase.subscriptionStatus} should return ${testCase.res}`, () => { setupMockIsMatchingLocation(testCase.loc); - setupMockOnboardingStatus(testCase.status); + setupMockOnboardingStatus(testCase.onboardingStatus); + setupMockSubscriptionStatus(testCase.subscriptionStatus); + setupMockIsLogged(testCase.isLoggedIn); expect(usePageChangeEffectNavigateLocation()).toEqual(testCase.res); }); }); describe('tests should be exhaustive', () => { it('all location and onboarding status should be tested', () => { + const untestedSubscriptionStatus = [ + SubscriptionStatus.Incomplete, + SubscriptionStatus.IncompleteExpired, + SubscriptionStatus.Paused, + SubscriptionStatus.Trialing, + ]; expect(testCases.length).toEqual( - Object.keys(AppPath).length * Object.keys(OnboardingStatus).length, + Object.keys(AppPath).length * + (Object.keys(OnboardingStatus).length + + (Object.keys(SubscriptionStatus).length - + untestedSubscriptionStatus.length)), ); }); }); diff --git a/packages/twenty-front/src/hooks/usePageChangeEffectNavigateLocation.ts b/packages/twenty-front/src/hooks/usePageChangeEffectNavigateLocation.ts index 5b34f0771ed4..5492b7b988a7 100644 --- a/packages/twenty-front/src/hooks/usePageChangeEffectNavigateLocation.ts +++ b/packages/twenty-front/src/hooks/usePageChangeEffectNavigateLocation.ts @@ -1,14 +1,18 @@ -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; -import { OnboardingStatus } from '@/auth/utils/getOnboardingStatus'; +import { useIsLogged } from '@/auth/hooks/useIsLogged'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; import { AppPath } from '@/types/AppPath'; import { SettingsPath } from '@/types/SettingsPath'; +import { useSubscriptionStatus } from '@/workspace/hooks/useSubscriptionStatus'; +import { OnboardingStatus, SubscriptionStatus } from '~/generated/graphql'; import { useDefaultHomePagePath } from '~/hooks/useDefaultHomePagePath'; import { useIsMatchingLocation } from '~/hooks/useIsMatchingLocation'; import { isDefined } from '~/utils/isDefined'; export const usePageChangeEffectNavigateLocation = () => { const isMatchingLocation = useIsMatchingLocation(); + const isLoggedIn = useIsLogged(); const onboardingStatus = useOnboardingStatus(); + const subscriptionStatus = useSubscriptionStatus(); const { defaultHomePagePath } = useDefaultHomePagePath(); const isMatchingOpenRoute = @@ -33,25 +37,28 @@ export const usePageChangeEffectNavigateLocation = () => { return; } - if ( - onboardingStatus === OnboardingStatus.OngoingUserCreation && - !isMatchingOngoingUserCreationRoute - ) { + if (!isLoggedIn && !isMatchingOngoingUserCreationRoute) { return AppPath.SignInUp; } if ( - onboardingStatus === OnboardingStatus.Incomplete && + onboardingStatus === OnboardingStatus.PlanRequired && !isMatchingLocation(AppPath.PlanRequired) ) { return AppPath.PlanRequired; } if ( - isDefined(onboardingStatus) && - [OnboardingStatus.Unpaid, OnboardingStatus.Canceled].includes( - onboardingStatus, - ) && + subscriptionStatus === SubscriptionStatus.Unpaid && + !isMatchingLocation(AppPath.SettingsCatchAll) + ) { + return `${AppPath.SettingsCatchAll.replace('/*', '')}/${ + SettingsPath.Billing + }`; + } + + if ( + subscriptionStatus === SubscriptionStatus.Canceled && !( isMatchingLocation(AppPath.SettingsCatchAll) || isMatchingLocation(AppPath.PlanRequired) @@ -63,7 +70,7 @@ export const usePageChangeEffectNavigateLocation = () => { } if ( - onboardingStatus === OnboardingStatus.OngoingWorkspaceActivation && + onboardingStatus === OnboardingStatus.WorkspaceActivation && !isMatchingLocation(AppPath.CreateWorkspace) && !isMatchingLocation(AppPath.PlanRequiredSuccess) ) { @@ -71,21 +78,21 @@ export const usePageChangeEffectNavigateLocation = () => { } if ( - onboardingStatus === OnboardingStatus.OngoingProfileCreation && + onboardingStatus === OnboardingStatus.ProfileCreation && !isMatchingLocation(AppPath.CreateProfile) ) { return AppPath.CreateProfile; } if ( - onboardingStatus === OnboardingStatus.OngoingSyncEmail && + onboardingStatus === OnboardingStatus.SyncEmail && !isMatchingLocation(AppPath.SyncEmails) ) { return AppPath.SyncEmails; } if ( - onboardingStatus === OnboardingStatus.OngoingInviteTeam && + onboardingStatus === OnboardingStatus.InviteTeam && !isMatchingLocation(AppPath.InviteTeam) ) { return AppPath.InviteTeam; @@ -93,15 +100,9 @@ export const usePageChangeEffectNavigateLocation = () => { if ( onboardingStatus === OnboardingStatus.Completed && - isMatchingOnboardingRoute - ) { - return defaultHomePagePath; - } - - if ( - onboardingStatus === OnboardingStatus.CompletedWithoutSubscription && isMatchingOnboardingRoute && - !isMatchingLocation(AppPath.PlanRequired) + subscriptionStatus !== SubscriptionStatus.Canceled && + (isDefined(subscriptionStatus) || !isMatchingLocation(AppPath.PlanRequired)) ) { return defaultHomePagePath; } diff --git a/packages/twenty-front/src/hooks/useScrollToPosition.ts b/packages/twenty-front/src/hooks/useScrollToPosition.ts new file mode 100644 index 000000000000..34994c1f3b5a --- /dev/null +++ b/packages/twenty-front/src/hooks/useScrollToPosition.ts @@ -0,0 +1,20 @@ +import { overlayScrollbarsState } from '@/ui/utilities/scroll/states/overlayScrollbarsState'; +import { useRecoilCallback } from 'recoil'; + +export const useScrollToPosition = () => { + const scrollToPosition = useRecoilCallback( + ({ snapshot }) => + (scrollPositionInPx: number) => { + const overlayScrollbars = snapshot + .getLoadable(overlayScrollbarsState) + .getValue(); + + const scrollWrapper = overlayScrollbars?.elements().viewport; + + scrollWrapper?.scrollTo({ top: scrollPositionInPx }); + }, + [], + ); + + return { scrollToPosition }; +}; diff --git a/packages/twenty-front/src/loading/components/LeftPanelSkeletonLoader.tsx b/packages/twenty-front/src/loading/components/LeftPanelSkeletonLoader.tsx index e1f73257f0bb..a6caf8861aee 100644 --- a/packages/twenty-front/src/loading/components/LeftPanelSkeletonLoader.tsx +++ b/packages/twenty-front/src/loading/components/LeftPanelSkeletonLoader.tsx @@ -1,6 +1,6 @@ -import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; import styled from '@emotion/styled'; import { motion } from 'framer-motion'; +import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; import { ANIMATION, BACKGROUND_LIGHT, GRAY_SCALE } from 'twenty-ui'; import { DESKTOP_NAV_DRAWER_WIDTHS } from '@/ui/navigation/navigation-drawer/constants/DesktopNavDrawerWidths'; @@ -18,7 +18,7 @@ const StyledItemsContainer = styled.div` gap: 32px; margin-bottom: auto; overflow-y: auto; - height: calc(100vh - 32px); + height: calc(100dvh - 32px); min-width: 216px; max-width: 216px; `; diff --git a/packages/twenty-front/src/loading/components/UserOrMetadataLoader.tsx b/packages/twenty-front/src/loading/components/UserOrMetadataLoader.tsx index d4d078619bb9..da129175ef2c 100644 --- a/packages/twenty-front/src/loading/components/UserOrMetadataLoader.tsx +++ b/packages/twenty-front/src/loading/components/UserOrMetadataLoader.tsx @@ -11,7 +11,7 @@ const StyledContainer = styled.div` display: flex; flex-direction: row; gap: 12px; - height: 100vh; + height: 100dvh; min-width: ${DESKTOP_NAV_DRAWER_WIDTHS.menu}px; width: 100%; padding: 12px 8px 12px; diff --git a/packages/twenty-front/src/loading/components/__stories__/UserOrMetadataLoader.stories.tsx b/packages/twenty-front/src/loading/components/__stories__/UserOrMetadataLoader.stories.tsx index 15c18390f697..ac5a371869be 100644 --- a/packages/twenty-front/src/loading/components/__stories__/UserOrMetadataLoader.stories.tsx +++ b/packages/twenty-front/src/loading/components/__stories__/UserOrMetadataLoader.stories.tsx @@ -14,7 +14,7 @@ import { } from '~/testing/decorators/PageDecorator'; import { graphqlMocks, metadataGraphql } from '~/testing/graphqlMocks'; import { mockedClientConfig } from '~/testing/mock-data/config'; -import { mockedUsersData } from '~/testing/mock-data/users'; +import { mockedUserData } from '~/testing/mock-data/users'; const userMetadataLoaderMocks = { msw: { @@ -22,7 +22,7 @@ const userMetadataLoaderMocks = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: mockedUsersData[0], + currentUser: mockedUserData, }, }); }), diff --git a/packages/twenty-front/src/modules/accounts/types/MessageChannel.ts b/packages/twenty-front/src/modules/accounts/types/MessageChannel.ts index b875854f191c..33df519b4503 100644 --- a/packages/twenty-front/src/modules/accounts/types/MessageChannel.ts +++ b/packages/twenty-front/src/modules/accounts/types/MessageChannel.ts @@ -1,9 +1,17 @@ import { MessageChannelVisibility } from '~/generated/graphql'; +export enum MessageChannelContactAutoCreationPolicy { + SENT_AND_RECEIVED = 'SENT_AND_RECEIVED', + SENT = 'SENT', + NONE = 'NONE', +} + export type MessageChannel = { id: string; handle: string; - isContactAutoCreationEnabled?: boolean; + contactAutoCreationPolicy?: MessageChannelContactAutoCreationPolicy; + excludeNonProfessionalEmails: boolean; + excludeGroupEmails: boolean; isSyncEnabled: boolean; visibility: MessageChannelVisibility; syncStatus: string; diff --git a/packages/twenty-front/src/modules/activities/calendar/components/Calendar.tsx b/packages/twenty-front/src/modules/activities/calendar/components/Calendar.tsx index 7032895aadd1..5bc8afa2171a 100644 --- a/packages/twenty-front/src/modules/activities/calendar/components/Calendar.tsx +++ b/packages/twenty-front/src/modules/activities/calendar/components/Calendar.tsx @@ -9,6 +9,7 @@ import { useCalendarEvents } from '@/activities/calendar/hooks/useCalendarEvents import { getTimelineCalendarEventsFromCompanyId } from '@/activities/calendar/queries/getTimelineCalendarEventsFromCompanyId'; import { getTimelineCalendarEventsFromPersonId } from '@/activities/calendar/queries/getTimelineCalendarEventsFromPersonId'; import { CustomResolverFetchMoreLoader } from '@/activities/components/CustomResolverFetchMoreLoader'; +import { SkeletonLoader } from '@/activities/components/SkeletonLoader'; import { useCustomResolver } from '@/activities/hooks/useCustomResolver'; import { ActivityTargetableObject } from '@/activities/types/ActivityTargetableEntity'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; @@ -18,6 +19,7 @@ import { AnimatedPlaceholderEmptySubTitle, AnimatedPlaceholderEmptyTextContainer, AnimatedPlaceholderEmptyTitle, + EMPTY_PLACEHOLDER_TRANSITION_PROPS, } from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; import { Section } from '@/ui/layout/section/components/Section'; import { TimelineCalendarEventsWithTotal } from '~/generated/graphql'; @@ -74,14 +76,16 @@ export const Calendar = ({ } = useCalendarEvents(timelineCalendarEvents || []); if (firstQueryLoading) { - // TODO: implement loader - return; + return <SkeletonLoader />; } if (!firstQueryLoading && !timelineCalendarEvents?.length) { // TODO: change animated placeholder return ( - <AnimatedPlaceholderEmptyContainer> + <AnimatedPlaceholderEmptyContainer + // eslint-disable-next-line react/jsx-props-no-spreading + {...EMPTY_PLACEHOLDER_TRANSITION_PROPS} + > <AnimatedPlaceholder type="noMatchRecord" /> <AnimatedPlaceholderEmptyTextContainer> <AnimatedPlaceholderEmptyTitle> diff --git a/packages/twenty-front/src/modules/activities/calendar/components/CalendarEventDetailsEffect.tsx b/packages/twenty-front/src/modules/activities/calendar/components/CalendarEventDetailsEffect.tsx new file mode 100644 index 000000000000..970cfc412b55 --- /dev/null +++ b/packages/twenty-front/src/modules/activities/calendar/components/CalendarEventDetailsEffect.tsx @@ -0,0 +1,23 @@ +import { CalendarEvent } from '@/activities/calendar/types/CalendarEvent'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; +import { useEffect } from 'react'; + +type CalendarEventDetailsEffectProps = { + record: CalendarEvent; +}; + +export const CalendarEventDetailsEffect = ({ + record, +}: CalendarEventDetailsEffectProps) => { + const { upsertRecords } = useUpsertRecordsInStore(); + + useEffect(() => { + if (!record) { + return; + } + + upsertRecords([record]); + }, [record, upsertRecords]); + + return <></>; +}; diff --git a/packages/twenty-front/src/modules/activities/calendar/components/CalendarEventRow.tsx b/packages/twenty-front/src/modules/activities/calendar/components/CalendarEventRow.tsx index 2a658225bb0f..e5ad988d3153 100644 --- a/packages/twenty-front/src/modules/activities/calendar/components/CalendarEventRow.tsx +++ b/packages/twenty-front/src/modules/activities/calendar/components/CalendarEventRow.tsx @@ -1,7 +1,7 @@ -import { useContext } from 'react'; import { css, useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { format } from 'date-fns'; +import { useContext } from 'react'; import { useRecoilValue } from 'recoil'; import { Avatar, AvatarGroup, IconArrowRight, IconLock } from 'twenty-ui'; @@ -14,8 +14,8 @@ import { hasCalendarEventEnded } from '@/activities/calendar/utils/hasCalendarEv import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; import { Card } from '@/ui/layout/card/components/Card'; import { CardContent } from '@/ui/layout/card/components/CardContent'; -import { CalendarChannelVisibility } from '~/generated/graphql'; import { TimelineCalendarEvent } from '~/generated-metadata/graphql'; +import { CalendarChannelVisibility } from '~/generated/graphql'; import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; import { isDefined } from '~/utils/isDefined'; @@ -169,7 +169,9 @@ export const CalendarEventRow = ({ ? `${participant.firstName} ${participant.lastName}` : participant.displayName } - entityId={participant.workspaceMemberId ?? participant.personId} + placeholderColorSeed={ + participant.workspaceMemberId ?? participant.personId + } type="rounded" /> ))} diff --git a/packages/twenty-front/src/modules/activities/calendar/right-drawer/components/RightDrawerCalendarEvent.tsx b/packages/twenty-front/src/modules/activities/calendar/right-drawer/components/RightDrawerCalendarEvent.tsx index 06aa66012f55..24f78d71539e 100644 --- a/packages/twenty-front/src/modules/activities/calendar/right-drawer/components/RightDrawerCalendarEvent.tsx +++ b/packages/twenty-front/src/modules/activities/calendar/right-drawer/components/RightDrawerCalendarEvent.tsx @@ -1,26 +1,35 @@ import { useRecoilValue } from 'recoil'; import { CalendarEventDetails } from '@/activities/calendar/components/CalendarEventDetails'; +import { CalendarEventDetailsEffect } from '@/activities/calendar/components/CalendarEventDetailsEffect'; import { FIND_ONE_CALENDAR_EVENT_OPERATION_SIGNATURE } from '@/activities/calendar/graphql/operation-signatures/FindOneCalendarEventOperationSignature'; import { CalendarEvent } from '@/activities/calendar/types/CalendarEvent'; import { useFindOneRecord } from '@/object-record/hooks/useFindOneRecord'; import { viewableRecordIdState } from '@/object-record/record-right-drawer/states/viewableRecordIdState'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { RecordValueSetterEffect } from '@/object-record/record-store/components/RecordValueSetterEffect'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; export const RightDrawerCalendarEvent = () => { - const { setRecords } = useSetRecordInStore(); + const { upsertRecords } = useUpsertRecordsInStore(); const viewableRecordId = useRecoilValue(viewableRecordIdState); + const { record: calendarEvent } = useFindOneRecord<CalendarEvent>({ objectNameSingular: FIND_ONE_CALENDAR_EVENT_OPERATION_SIGNATURE.objectNameSingular, objectRecordId: viewableRecordId ?? '', recordGqlFields: FIND_ONE_CALENDAR_EVENT_OPERATION_SIGNATURE.fields, - onCompleted: (record) => setRecords([record]), + onCompleted: (record) => upsertRecords([record]), }); if (!calendarEvent) { return null; } - return <CalendarEventDetails calendarEvent={calendarEvent} />; + return ( + <> + <CalendarEventDetailsEffect record={calendarEvent} /> + <RecordValueSetterEffect recordId={calendarEvent.id} /> + <CalendarEventDetails calendarEvent={calendarEvent} /> + </> + ); }; diff --git a/packages/twenty-front/src/modules/activities/comment/CommentHeader.tsx b/packages/twenty-front/src/modules/activities/comment/CommentHeader.tsx index 13ed1aa16352..d48f114e9d7f 100644 --- a/packages/twenty-front/src/modules/activities/comment/CommentHeader.tsx +++ b/packages/twenty-front/src/modules/activities/comment/CommentHeader.tsx @@ -1,6 +1,5 @@ -import { Tooltip } from 'react-tooltip'; import styled from '@emotion/styled'; -import { Avatar } from 'twenty-ui'; +import { AppTooltip, Avatar } from 'twenty-ui'; import { Comment } from '@/activities/types/Comment'; import { @@ -42,21 +41,6 @@ const StyledDate = styled.div` margin-left: ${({ theme }) => theme.spacing(1)}; `; -const StyledTooltip = styled(Tooltip)` - background-color: ${({ theme }) => theme.background.primary}; - - box-shadow: 0px 2px 4px 3px - ${({ theme }) => theme.background.transparent.light}; - - box-shadow: 2px 4px 16px 6px - ${({ theme }) => theme.background.transparent.light}; - - color: ${({ theme }) => theme.font.color.primary}; - - opacity: 1; - padding: 8px; -`; - type CommentHeaderProps = { comment: Pick<Comment, 'id' | 'author' | 'createdAt'>; actionBar?: React.ReactNode; @@ -78,7 +62,7 @@ export const CommentHeader = ({ comment, actionBar }: CommentHeaderProps) => { <Avatar avatarUrl={getImageAbsoluteURIOrBase64(avatarUrl)} size="md" - entityId={author?.id} + placeholderColorSeed={author?.id} placeholder={authorName} /> <StyledName>{authorName}</StyledName> @@ -87,7 +71,7 @@ export const CommentHeader = ({ comment, actionBar }: CommentHeaderProps) => { <StyledDate id={`id-${commentId}`}> {beautifiedCreatedAt} </StyledDate> - <StyledTooltip + <AppTooltip anchorSelect={`#id-${commentId}`} content={exactCreatedAt} clickable diff --git a/packages/twenty-front/src/modules/activities/components/ActivityEditorFields.tsx b/packages/twenty-front/src/modules/activities/components/ActivityEditorFields.tsx index 9f221d840559..e8b189258544 100644 --- a/packages/twenty-front/src/modules/activities/components/ActivityEditorFields.tsx +++ b/packages/twenty-front/src/modules/activities/components/ActivityEditorFields.tsx @@ -86,14 +86,6 @@ export const ActivityEditorFields = ({ customUseUpdateOneObjectHook: useUpsertOneActivityMutation, }); - const { FieldContextProvider: ActivityTargetsContextProvider } = - useFieldContext({ - objectNameSingular: CoreObjectNameSingular.Activity, - objectRecordId: activityId, - fieldMetadataName: 'activityTargets', - fieldPosition: 3, - }); - return ( <StyledPropertyBox> {activity.type === 'Task' && @@ -112,16 +104,12 @@ export const ActivityEditorFields = ({ </AssigneeFieldContextProvider> </> )} - {ActivityTargetsContextProvider && - isDefined(activityFromCache) && - isRightDrawerAnimationCompleted && ( - <ActivityTargetsContextProvider> - <ActivityTargetsInlineCell - activity={activityFromCache} - maxWidth={340} - /> - </ActivityTargetsContextProvider> - )} + {isDefined(activityFromCache) && isRightDrawerAnimationCompleted && ( + <ActivityTargetsInlineCell + activity={activityFromCache} + maxWidth={340} + /> + )} </StyledPropertyBox> ); }; diff --git a/packages/twenty-front/src/modules/activities/timeline/components/TimelineSkeletonLoader.tsx b/packages/twenty-front/src/modules/activities/components/SkeletonLoader.tsx similarity index 62% rename from packages/twenty-front/src/modules/activities/timeline/components/TimelineSkeletonLoader.tsx rename to packages/twenty-front/src/modules/activities/components/SkeletonLoader.tsx index f328fda44085..21478ed73c32 100644 --- a/packages/twenty-front/src/modules/activities/timeline/components/TimelineSkeletonLoader.tsx +++ b/packages/twenty-front/src/modules/activities/components/SkeletonLoader.tsx @@ -18,18 +18,14 @@ const StyledSkeletonSubSection = styled.div` gap: ${({ theme }) => theme.spacing(4)}; `; -const StyledSkeletonColumn = styled.div` +const StyledSkeletonSubSectionContent = styled.div` display: flex; flex-direction: column; gap: ${({ theme }) => theme.spacing(3)}; justify-content: center; `; -const StyledSkeletonLoader = ({ - isSecondColumn, -}: { - isSecondColumn: boolean; -}) => { +const SkeletonColumnLoader = ({ height }: { height: number }) => { const theme = useTheme(); return ( <SkeletonTheme @@ -37,12 +33,16 @@ const StyledSkeletonLoader = ({ highlightColor={theme.background.transparent.lighter} borderRadius={80} > - <Skeleton width={24} height={isSecondColumn ? 120 : 84} /> + <Skeleton width={24} height={height} /> </SkeletonTheme> ); }; -export const TimelineSkeletonLoader = () => { +export const SkeletonLoader = ({ + withSubSections = false, +}: { + withSubSections?: boolean; +}) => { const theme = useTheme(); const skeletonItems = Array.from({ length: 3 }).map((_, index) => ({ id: `skeleton-item-${index}`, @@ -56,16 +56,17 @@ export const TimelineSkeletonLoader = () => { > <StyledSkeletonContainer> <Skeleton width={440} height={16} /> - {skeletonItems.map(({ id }, index) => ( - <StyledSkeletonSubSection key={id}> - <StyledSkeletonLoader isSecondColumn={index === 1} /> - <StyledSkeletonColumn> - <Skeleton width={400} height={24} /> - <Skeleton width={400} height={24} /> - {index === 1 && <Skeleton width={400} height={24} />} - </StyledSkeletonColumn> - </StyledSkeletonSubSection> - ))} + {withSubSections && + skeletonItems.map(({ id }, index) => ( + <StyledSkeletonSubSection key={id}> + <SkeletonColumnLoader height={index === 1 ? 120 : 84} /> + <StyledSkeletonSubSectionContent> + <Skeleton width={400} height={24} /> + <Skeleton width={400} height={24} /> + {index === 1 && <Skeleton width={400} height={24} />} + </StyledSkeletonSubSectionContent> + </StyledSkeletonSubSection> + ))} </StyledSkeletonContainer> </SkeletonTheme> ); diff --git a/packages/twenty-front/src/modules/activities/copilot/right-drawer/components/RightDrawerAIChat.tsx b/packages/twenty-front/src/modules/activities/copilot/right-drawer/components/RightDrawerAIChat.tsx new file mode 100644 index 000000000000..4ad2c423c77b --- /dev/null +++ b/packages/twenty-front/src/modules/activities/copilot/right-drawer/components/RightDrawerAIChat.tsx @@ -0,0 +1,54 @@ +import styled from '@emotion/styled'; +import { useSetRecoilState } from 'recoil'; + +import { copilotQueryState } from '@/activities/copilot/right-drawer/states/copilotQueryState'; +import { + AutosizeTextInput, + AutosizeTextInputVariant, +} from '@/ui/input/components/AutosizeTextInput'; + +const StyledContainer = styled.div` + box-sizing: border-box; + display: flex; + flex-direction: column; + height: 100%; + justify-content: flex-start; + overflow-y: auto; + position: relative; +`; + +const StyledChatArea = styled.div` + flex: 1; + display: flex; + flex-direction: column; + overflow-y: scroll; + padding: ${({ theme }) => theme.spacing(6)}; + padding-bottom: 0px; +`; + +const StyledNewMessageArea = styled.div` + display: flex; + flex-direction: column; + padding: ${({ theme }) => theme.spacing(6)}; + padding-top: 0px; +`; + +export const RightDrawerAIChat = () => { + const setCopilotQuery = useSetRecoilState(copilotQueryState); + + return ( + <StyledContainer> + <StyledChatArea>{/* TODO */}</StyledChatArea> + <StyledNewMessageArea> + <AutosizeTextInput + autoFocus + placeholder="Ask anything" + variant={AutosizeTextInputVariant.Icon} + onValidate={(text) => { + setCopilotQuery(text); + }} + /> + </StyledNewMessageArea> + </StyledContainer> + ); +}; diff --git a/packages/twenty-front/src/modules/activities/copilot/right-drawer/hooks/useOpenCopilotRightDrawer.ts b/packages/twenty-front/src/modules/activities/copilot/right-drawer/hooks/useOpenCopilotRightDrawer.ts new file mode 100644 index 000000000000..5369451624e1 --- /dev/null +++ b/packages/twenty-front/src/modules/activities/copilot/right-drawer/hooks/useOpenCopilotRightDrawer.ts @@ -0,0 +1,14 @@ +import { useRightDrawer } from '@/ui/layout/right-drawer/hooks/useRightDrawer'; +import { RightDrawerHotkeyScope } from '@/ui/layout/right-drawer/types/RightDrawerHotkeyScope'; +import { RightDrawerPages } from '@/ui/layout/right-drawer/types/RightDrawerPages'; +import { useSetHotkeyScope } from '@/ui/utilities/hotkey/hooks/useSetHotkeyScope'; + +export const useOpenCopilotRightDrawer = () => { + const { openRightDrawer } = useRightDrawer(); + const setHotkeyScope = useSetHotkeyScope(); + + return () => { + setHotkeyScope(RightDrawerHotkeyScope.RightDrawer, { goto: false }); + openRightDrawer(RightDrawerPages.Copilot); + }; +}; diff --git a/packages/twenty-front/src/modules/activities/copilot/right-drawer/states/copilotQueryState.ts b/packages/twenty-front/src/modules/activities/copilot/right-drawer/states/copilotQueryState.ts new file mode 100644 index 000000000000..ee56e8b0fe80 --- /dev/null +++ b/packages/twenty-front/src/modules/activities/copilot/right-drawer/states/copilotQueryState.ts @@ -0,0 +1,6 @@ +import { createState } from 'twenty-ui'; + +export const copilotQueryState = createState({ + key: 'activities/copilot-query', + defaultValue: '', +}); diff --git a/packages/twenty-front/src/modules/activities/emails/components/EmailThreads.tsx b/packages/twenty-front/src/modules/activities/emails/components/EmailThreads.tsx index d0766c517b17..53038755dc88 100644 --- a/packages/twenty-front/src/modules/activities/emails/components/EmailThreads.tsx +++ b/packages/twenty-front/src/modules/activities/emails/components/EmailThreads.tsx @@ -2,7 +2,7 @@ import styled from '@emotion/styled'; import { H1Title, H1TitleFontColor } from 'twenty-ui'; import { CustomResolverFetchMoreLoader } from '@/activities/components/CustomResolverFetchMoreLoader'; -import { EmailLoader } from '@/activities/emails/components/EmailLoader'; +import { SkeletonLoader } from '@/activities/components/SkeletonLoader'; import { EmailThreadPreview } from '@/activities/emails/components/EmailThreadPreview'; import { TIMELINE_THREADS_DEFAULT_PAGE_SIZE } from '@/activities/emails/constants/Messaging'; import { getTimelineThreadsFromCompanyId } from '@/activities/emails/queries/getTimelineThreadsFromCompanyId'; @@ -16,6 +16,7 @@ import { AnimatedPlaceholderEmptySubTitle, AnimatedPlaceholderEmptyTextContainer, AnimatedPlaceholderEmptyTitle, + EMPTY_PLACEHOLDER_TRANSITION_PROPS, } from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; import { Card } from '@/ui/layout/card/components/Card'; import { Section } from '@/ui/layout/section/components/Section'; @@ -61,12 +62,15 @@ export const EmailThreads = ({ const { totalNumberOfThreads, timelineThreads } = data?.[queryName] ?? {}; if (firstQueryLoading) { - return <EmailLoader />; + return <SkeletonLoader />; } if (!firstQueryLoading && !timelineThreads?.length) { return ( - <AnimatedPlaceholderEmptyContainer> + <AnimatedPlaceholderEmptyContainer + // eslint-disable-next-line react/jsx-props-no-spreading + {...EMPTY_PLACEHOLDER_TRANSITION_PROPS} + > <AnimatedPlaceholder type="emptyInbox" /> <AnimatedPlaceholderEmptyTextContainer> <AnimatedPlaceholderEmptyTitle> diff --git a/packages/twenty-front/src/modules/activities/emails/graphql/operation-signatures/factories/fetchAllThreadMessagesOperationSignatureFactory.ts b/packages/twenty-front/src/modules/activities/emails/graphql/operation-signatures/factories/fetchAllThreadMessagesOperationSignatureFactory.ts index 619a99b0dd23..ee5e93da74fa 100644 --- a/packages/twenty-front/src/modules/activities/emails/graphql/operation-signatures/factories/fetchAllThreadMessagesOperationSignatureFactory.ts +++ b/packages/twenty-front/src/modules/activities/emails/graphql/operation-signatures/factories/fetchAllThreadMessagesOperationSignatureFactory.ts @@ -10,9 +10,11 @@ export const fetchAllThreadMessagesOperationSignatureFactory: RecordGqlOperation eq: messageThreadId || '', }, }, - orderBy: { - receivedAt: 'AscNullsLast', - }, + orderBy: [ + { + receivedAt: 'AscNullsLast', + }, + ], limit: 10, }, fields: { diff --git a/packages/twenty-front/src/modules/activities/emails/right-drawer/components/RightDrawerEmailThread.tsx b/packages/twenty-front/src/modules/activities/emails/right-drawer/components/RightDrawerEmailThread.tsx index d7d040726145..09c98810b2e3 100644 --- a/packages/twenty-front/src/modules/activities/emails/right-drawer/components/RightDrawerEmailThread.tsx +++ b/packages/twenty-front/src/modules/activities/emails/right-drawer/components/RightDrawerEmailThread.tsx @@ -71,6 +71,7 @@ export const RightDrawerEmailThread = () => { ? visibleMessages.slice(2, visibleMessagesCount - 1) : []; const lastMessage = visibleMessages[visibleMessagesCount - 1]; + const subject = visibleMessages[0]?.subject; return ( <StyledContainer> @@ -79,7 +80,7 @@ export const RightDrawerEmailThread = () => { ) : ( <> <EmailThreadHeader - subject={thread.subject} + subject={subject} lastMessageSentAt={lastMessage.receivedAt} /> {firstMessages.map((message) => ( diff --git a/packages/twenty-front/src/modules/activities/emails/right-drawer/hooks/useRightDrawerEmailThread.ts b/packages/twenty-front/src/modules/activities/emails/right-drawer/hooks/useRightDrawerEmailThread.ts index b63e429ef26c..47c30f4d0eb4 100644 --- a/packages/twenty-front/src/modules/activities/emails/right-drawer/hooks/useRightDrawerEmailThread.ts +++ b/packages/twenty-front/src/modules/activities/emails/right-drawer/hooks/useRightDrawerEmailThread.ts @@ -8,11 +8,11 @@ import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSi import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; import { useFindOneRecord } from '@/object-record/hooks/useFindOneRecord'; import { viewableRecordIdState } from '@/object-record/record-right-drawer/states/viewableRecordIdState'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; export const useRightDrawerEmailThread = () => { const viewableRecordId = useRecoilValue(viewableRecordIdState); - const { setRecords } = useSetRecordInStore(); + const { upsertRecords } = useUpsertRecordsInStore(); const { record: thread } = useFindOneRecord<EmailThread>({ objectNameSingular: CoreObjectNameSingular.MessageThread, @@ -20,7 +20,7 @@ export const useRightDrawerEmailThread = () => { recordGqlFields: { id: true, }, - onCompleted: (record) => setRecords([record]), + onCompleted: (record) => upsertRecords([record]), }); const FETCH_ALL_MESSAGES_OPERATION_SIGNATURE = diff --git a/packages/twenty-front/src/modules/activities/files/components/Attachments.tsx b/packages/twenty-front/src/modules/activities/files/components/Attachments.tsx index aacc0ec6ad3f..482b8e18f88f 100644 --- a/packages/twenty-front/src/modules/activities/files/components/Attachments.tsx +++ b/packages/twenty-front/src/modules/activities/files/components/Attachments.tsx @@ -1,8 +1,8 @@ import { ChangeEvent, useRef, useState } from 'react'; import styled from '@emotion/styled'; -import { isNonEmptyArray } from '@sniptt/guards'; import { IconPlus } from 'twenty-ui'; +import { SkeletonLoader } from '@/activities/components/SkeletonLoader'; import { AttachmentList } from '@/activities/files/components/AttachmentList'; import { DropZone } from '@/activities/files/components/DropZone'; import { useAttachments } from '@/activities/files/hooks/useAttachments'; @@ -15,6 +15,7 @@ import { AnimatedPlaceholderEmptySubTitle, AnimatedPlaceholderEmptyTextContainer, AnimatedPlaceholderEmptyTitle, + EMPTY_PLACEHOLDER_TRANSITION_PROPS, } from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; import { isDefined } from '~/utils/isDefined'; @@ -41,7 +42,7 @@ export const Attachments = ({ targetableObject: ActivityTargetableObject; }) => { const inputFileRef = useRef<HTMLInputElement>(null); - const { attachments } = useAttachments(targetableObject); + const { attachments, loading } = useAttachments(targetableObject); const { uploadAttachmentFile } = useUploadAttachmentFile(); const [isDraggingFile, setIsDraggingFile] = useState(false); @@ -58,7 +59,13 @@ export const Attachments = ({ await uploadAttachmentFile(file, targetableObject); }; - if (!isNonEmptyArray(attachments)) { + const isAttachmentsEmpty = !attachments || attachments.length === 0; + + if (loading && isAttachmentsEmpty) { + return <SkeletonLoader />; + } + + if (isAttachmentsEmpty) { return ( <StyledDropZoneContainer onDragEnter={() => setIsDraggingFile(true)}> {isDraggingFile ? ( @@ -67,7 +74,10 @@ export const Attachments = ({ onUploadFile={onUploadFile} /> ) : ( - <AnimatedPlaceholderEmptyContainer> + <AnimatedPlaceholderEmptyContainer + // eslint-disable-next-line react/jsx-props-no-spreading + {...EMPTY_PLACEHOLDER_TRANSITION_PROPS} + > <AnimatedPlaceholder type="noFile" /> <AnimatedPlaceholderEmptyTextContainer> <AnimatedPlaceholderEmptyTitle> diff --git a/packages/twenty-front/src/modules/activities/files/hooks/useAttachments.tsx b/packages/twenty-front/src/modules/activities/files/hooks/useAttachments.tsx index 2d8539889d27..a91eb40313f3 100644 --- a/packages/twenty-front/src/modules/activities/files/hooks/useAttachments.tsx +++ b/packages/twenty-front/src/modules/activities/files/hooks/useAttachments.tsx @@ -4,25 +4,27 @@ import { getActivityTargetObjectFieldIdName } from '@/activities/utils/getActivi import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; -// do we need to test this? export const useAttachments = (targetableObject: ActivityTargetableObject) => { const targetableObjectFieldIdName = getActivityTargetObjectFieldIdName({ nameSingular: targetableObject.targetObjectNameSingular, }); - const { records: attachments } = useFindManyRecords<Attachment>({ + const { records: attachments, loading } = useFindManyRecords<Attachment>({ objectNameSingular: CoreObjectNameSingular.Attachment, filter: { [targetableObjectFieldIdName]: { eq: targetableObject.id, }, }, - orderBy: { - createdAt: 'DescNullsFirst', - }, + orderBy: [ + { + createdAt: 'DescNullsFirst', + }, + ], }); return { attachments, + loading, }; }; diff --git a/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivities.test.tsx b/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivities.test.tsx index 4e149d280fbf..b3ca9e5349d2 100644 --- a/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivities.test.tsx +++ b/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivities.test.tsx @@ -49,7 +49,7 @@ const mocks: MockedResponse[] = [ query: gql` query FindManyActivityTargets( $filter: ActivityTargetFilterInput - $orderBy: ActivityTargetOrderByInput + $orderBy: [ActivityTargetOrderByInput] $lastCursor: String $limit: Int ) { @@ -103,7 +103,7 @@ const mocks: MockedResponse[] = [ query: gql` query FindManyActivities( $filter: ActivityFilterInput - $orderBy: ActivityOrderByInput + $orderBy: [ActivityOrderByInput] $lastCursor: String $limit: Int ) { @@ -142,7 +142,7 @@ const mocks: MockedResponse[] = [ variables: { filter: { id: { in: ['234'] } }, limit: undefined, - orderBy: {}, + orderBy: [{}], }, }, result: jest.fn(() => ({ @@ -178,7 +178,7 @@ describe('useActivities', () => { useActivities({ targetableObjects: [], activitiesFilters: {}, - activitiesOrderByVariables: {}, + activitiesOrderByVariables: [{}], skip: false, }), { wrapper: Wrapper }, @@ -202,7 +202,7 @@ describe('useActivities', () => { { targetObjectNameSingular: 'company', id: '123' }, ], activitiesFilters: {}, - activitiesOrderByVariables: {}, + activitiesOrderByVariables: [{}], skip: false, }); return { activities, setCurrentWorkspaceMember }; diff --git a/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivityTargetsForTargetableObject.test.tsx b/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivityTargetsForTargetableObject.test.tsx index 9426e1cb7fbb..21f429c6b8c6 100644 --- a/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivityTargetsForTargetableObject.test.tsx +++ b/packages/twenty-front/src/modules/activities/hooks/__tests__/useActivityTargetsForTargetableObject.test.tsx @@ -34,7 +34,7 @@ const mocks: MockedResponse[] = [ query: gql` query FindManyActivityTargets( $filter: ActivityTargetFilterInput - $orderBy: ActivityTargetOrderByInput + $orderBy: [ActivityTargetOrderByInput] $lastCursor: String $limit: Int ) { diff --git a/packages/twenty-front/src/modules/activities/hooks/useCreateActivityInDB.ts b/packages/twenty-front/src/modules/activities/hooks/useCreateActivityInDB.ts index 961514ede05e..470e9fda030e 100644 --- a/packages/twenty-front/src/modules/activities/hooks/useCreateActivityInDB.ts +++ b/packages/twenty-front/src/modules/activities/hooks/useCreateActivityInDB.ts @@ -3,35 +3,98 @@ import { isNonEmptyArray } from '@sniptt/guards'; import { CREATE_ONE_ACTIVITY_OPERATION_SIGNATURE } from '@/activities/graphql/operation-signatures/CreateOneActivityOperationSignature'; import { ActivityForEditor } from '@/activities/types/ActivityForEditor'; import { ActivityTarget } from '@/activities/types/ActivityTarget'; +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { useObjectMetadataItems } from '@/object-metadata/hooks/useObjectMetadataItems'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; +import { getRecordConnectionFromRecords } from '@/object-record/cache/utils/getRecordConnectionFromRecords'; +import { modifyRecordFromCache } from '@/object-record/cache/utils/modifyRecordFromCache'; import { useCreateManyRecords } from '@/object-record/hooks/useCreateManyRecords'; import { useCreateOneRecord } from '@/object-record/hooks/useCreateOneRecord'; +import { recordStoreFamilyState } from '@/object-record/record-store/states/recordStoreFamilyState'; +import { useApolloClient } from '@apollo/client'; + +import { useRecoilCallback } from 'recoil'; +import { capitalize } from '~/utils/string/capitalize'; export const useCreateActivityInDB = () => { const { createOneRecord: createOneActivity } = useCreateOneRecord({ objectNameSingular: CREATE_ONE_ACTIVITY_OPERATION_SIGNATURE.objectNameSingular, recordGqlFields: CREATE_ONE_ACTIVITY_OPERATION_SIGNATURE.fields, + shouldMatchRootQueryFilter: true, }); const { createManyRecords: createManyActivityTargets } = useCreateManyRecords<ActivityTarget>({ objectNameSingular: CoreObjectNameSingular.ActivityTarget, - skipPostOptmisticEffect: true, + shouldMatchRootQueryFilter: true, + }); + + const { objectMetadataItems } = useObjectMetadataItems(); + + const { objectMetadataItem: objectMetadataItemActivityTarget } = + useObjectMetadataItem({ + objectNameSingular: CoreObjectNameSingular.ActivityTarget, }); - const createActivityInDB = async (activityToCreate: ActivityForEditor) => { - await createOneActivity?.({ - ...activityToCreate, - updatedAt: new Date().toISOString(), + const { objectMetadataItem: objectMetadataItemActivity } = + useObjectMetadataItem({ + objectNameSingular: CoreObjectNameSingular.Activity, }); - const activityTargetsToCreate = activityToCreate.activityTargets ?? []; + const cache = useApolloClient().cache; - if (isNonEmptyArray(activityTargetsToCreate)) { - await createManyActivityTargets(activityTargetsToCreate); - } - }; + const createActivityInDB = useRecoilCallback( + ({ set }) => + async (activityToCreate: ActivityForEditor) => { + const createdActivity = await createOneActivity?.({ + ...activityToCreate, + updatedAt: new Date().toISOString(), + }); + + const activityTargetsToCreate = activityToCreate.activityTargets ?? []; + + if (isNonEmptyArray(activityTargetsToCreate)) { + await createManyActivityTargets(activityTargetsToCreate); + } + + const activityTargetsConnection = getRecordConnectionFromRecords({ + objectMetadataItems, + objectMetadataItem: objectMetadataItemActivityTarget, + records: activityTargetsToCreate.map((activityTarget) => ({ + ...activityTarget, + __typename: capitalize( + objectMetadataItemActivityTarget.nameSingular, + ), + })), + withPageInfo: false, + computeReferences: true, + isRootLevel: false, + }); + + modifyRecordFromCache({ + recordId: createdActivity.id, + cache, + fieldModifiers: { + activityTargets: () => activityTargetsConnection, + }, + objectMetadataItem: objectMetadataItemActivity, + }); + + set(recordStoreFamilyState(createdActivity.id), { + ...createdActivity, + activityTargets: activityTargetsToCreate, + }); + }, + [ + cache, + createManyActivityTargets, + createOneActivity, + objectMetadataItemActivity, + objectMetadataItemActivityTarget, + objectMetadataItems, + ], + ); return { createActivityInDB, diff --git a/packages/twenty-front/src/modules/activities/hooks/useObjectRecordMultiSelectScopedStates.ts b/packages/twenty-front/src/modules/activities/hooks/useObjectRecordMultiSelectScopedStates.ts new file mode 100644 index 000000000000..b49c9aefa47b --- /dev/null +++ b/packages/twenty-front/src/modules/activities/hooks/useObjectRecordMultiSelectScopedStates.ts @@ -0,0 +1,35 @@ +import { objectRecordsIdsMultiSelecComponentState } from '@/activities/states/objectRecordsIdsMultiSelectComponentState'; +import { objectRecordMultiSelectCheckedRecordsIdsComponentState } from '@/object-record/record-field/states/objectRecordMultiSelectCheckedRecordsIdsComponentState'; +import { objectRecordMultiSelectComponentFamilyState } from '@/object-record/record-field/states/objectRecordMultiSelectComponentFamilyState'; +import { recordMultiSelectIsLoadingComponentState } from '@/object-record/record-field/states/recordMultiSelectIsLoadingComponentState'; +import { extractComponentFamilyState } from '@/ui/utilities/state/component-state/utils/extractComponentFamilyState'; +import { extractComponentState } from '@/ui/utilities/state/component-state/utils/extractComponentState'; + +export const useObjectRecordMultiSelectScopedStates = (scopeId: string) => { + const objectRecordsIdsMultiSelectState = extractComponentState( + objectRecordsIdsMultiSelecComponentState, + scopeId, + ); + + const objectRecordMultiSelectCheckedRecordsIdsState = extractComponentState( + objectRecordMultiSelectCheckedRecordsIdsComponentState, + scopeId, + ); + + const objectRecordMultiSelectFamilyState = extractComponentFamilyState( + objectRecordMultiSelectComponentFamilyState, + scopeId, + ); + + const recordMultiSelectIsLoadingState = extractComponentState( + recordMultiSelectIsLoadingComponentState, + scopeId, + ); + + return { + objectRecordsIdsMultiSelectState, + objectRecordMultiSelectCheckedRecordsIdsState, + objectRecordMultiSelectFamilyState, + recordMultiSelectIsLoadingState, + }; +}; diff --git a/packages/twenty-front/src/modules/activities/hooks/usePrepareFindManyActivitiesQuery.ts b/packages/twenty-front/src/modules/activities/hooks/usePrepareFindManyActivitiesQuery.ts index 7f5d9c492c8a..2951b7522ce0 100644 --- a/packages/twenty-front/src/modules/activities/hooks/usePrepareFindManyActivitiesQuery.ts +++ b/packages/twenty-front/src/modules/activities/hooks/usePrepareFindManyActivitiesQuery.ts @@ -109,7 +109,7 @@ export const usePrepareFindManyActivitiesQuery = () => { objectRecordsToOverwrite: filteredActivities, queryVariables: { ...nextFindManyActivitiesQueryFilter, - orderBy: { createdAt: 'DescNullsFirst' }, + orderBy: [{ createdAt: 'DescNullsFirst' }], }, recordGqlFields: FIND_ACTIVITIES_OPERATION_SIGNATURE.fields, computeReferences: true, diff --git a/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetInlineCellEditMode.tsx b/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetInlineCellEditMode.tsx index 0c884b2035a0..0b85e20074b7 100644 --- a/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetInlineCellEditMode.tsx +++ b/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetInlineCellEditMode.tsx @@ -1,9 +1,10 @@ import styled from '@emotion/styled'; -import { isNonEmptyArray, isNull } from '@sniptt/guards'; -import { useRecoilState, useSetRecoilState } from 'recoil'; +import { isNull } from '@sniptt/guards'; +import { useRecoilCallback, useRecoilState, useSetRecoilState } from 'recoil'; import { v4 } from 'uuid'; import { useUpsertActivity } from '@/activities/hooks/useUpsertActivity'; +import { ActivityTargetObjectRecordEffect } from '@/activities/inline-cell/components/ActivityTargetObjectRecordEffect'; import { isActivityInCreateModeState } from '@/activities/states/isActivityInCreateModeState'; import { Activity } from '@/activities/types/Activity'; import { ActivityTarget } from '@/activities/types/ActivityTarget'; @@ -15,16 +16,23 @@ import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSi import { useCreateManyRecordsInCache } from '@/object-record/cache/hooks/useCreateManyRecordsInCache'; import { useCreateManyRecords } from '@/object-record/hooks/useCreateManyRecords'; import { useDeleteManyRecords } from '@/object-record/hooks/useDeleteManyRecords'; +import { activityTargetObjectRecordFamilyState } from '@/object-record/record-field/states/activityTargetObjectRecordFamilyState'; +import { objectRecordMultiSelectCheckedRecordsIdsComponentState } from '@/object-record/record-field/states/objectRecordMultiSelectCheckedRecordsIdsComponentState'; +import { + ObjectRecordAndSelected, + objectRecordMultiSelectComponentFamilyState, +} from '@/object-record/record-field/states/objectRecordMultiSelectComponentFamilyState'; import { useInlineCell } from '@/object-record/record-inline-cell/hooks/useInlineCell'; import { recordStoreFamilyState } from '@/object-record/record-store/states/recordStoreFamilyState'; -import { MultipleObjectRecordSelect } from '@/object-record/relation-picker/components/MultipleObjectRecordSelect'; -import { ObjectRecordForSelect } from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; +import { ActivityTargetInlineCellEditModeMultiRecordsEffect } from '@/object-record/relation-picker/components/ActivityTargetInlineCellEditModeMultiRecordsEffect'; +import { MultiRecordSelect } from '@/object-record/relation-picker/components/MultiRecordSelect'; +import { RelationPickerScope } from '@/object-record/relation-picker/scopes/RelationPickerScope'; import { prefillRecord } from '@/object-record/utils/prefillRecord'; const StyledSelectContainer = styled.div` - left: 0px; position: absolute; - top: -8px; + left: 0; + top: 0; `; type ActivityTargetInlineCellEditModeProps = { @@ -37,6 +45,7 @@ export const ActivityTargetInlineCellEditMode = ({ activityTargetWithTargetRecords, }: ActivityTargetInlineCellEditModeProps) => { const [isActivityInCreateMode] = useRecoilState(isActivityInCreateModeState); + const relationPickerScopeId = `relation-picker-${activity.id}`; const selectedTargetObjectIds = activityTargetWithTargetRecords.map( (activityTarget) => ({ @@ -74,109 +83,181 @@ export const ActivityTargetInlineCellEditMode = ({ objectNameSingular: CoreObjectNameSingular.ActivityTarget, }); - const handleSubmit = async (selectedRecords: ObjectRecordForSelect[]) => { - closeEditableField(); - - const activityTargetsToDelete = activityTargetWithTargetRecords.filter( - (activityTargetObjectRecord) => - !selectedRecords.some( - (selectedRecord) => - selectedRecord.recordIdentifier.id === - activityTargetObjectRecord.targetObject.id, - ), - ); - - const selectedTargetObjectsToCreate = selectedRecords.filter( - (selectedRecord) => - !activityTargetWithTargetRecords.some( - (activityTargetWithTargetRecord) => - activityTargetWithTargetRecord.targetObject.id === - selectedRecord.recordIdentifier.id, - ), - ); - - const existingActivityTargets = activityTargetWithTargetRecords.map( - (activityTargetObjectRecord) => activityTargetObjectRecord.activityTarget, - ); - - let activityTargetsAfterUpdate = Array.from(existingActivityTargets); - - const activityTargetsToCreate = selectedTargetObjectsToCreate.map( - (selectedRecord) => { - const emptyActivityTarget = prefillRecord<ActivityTarget>({ - objectMetadataItem: objectMetadataItemActivityTarget, - input: { - id: v4(), - activityId: activity.id, - activity, - createdAt: new Date().toISOString(), - updatedAt: new Date().toISOString(), - [getActivityTargetObjectFieldName({ - nameSingular: selectedRecord.objectMetadataItem.nameSingular, - })]: selectedRecord.record, - [getActivityTargetObjectFieldIdName({ - nameSingular: selectedRecord.objectMetadataItem.nameSingular, - })]: selectedRecord.recordIdentifier.id, - }, - }); + const handleSubmit = useRecoilCallback( + ({ snapshot }) => + async () => { + const activityTargetsAfterUpdate = + activityTargetWithTargetRecords.filter((activityTarget) => { + const record = snapshot + .getLoadable( + objectRecordMultiSelectComponentFamilyState({ + scopeId: relationPickerScopeId, + familyKey: activityTarget.targetObject.id, + }), + ) + .getValue() as ObjectRecordAndSelected; + + return record.selected; + }); + setActivityFromStore((currentActivity) => { + if (isNull(currentActivity)) { + return null; + } - return emptyActivityTarget; + return { + ...currentActivity, + activityTargets: activityTargetsAfterUpdate, + }; + }); + closeEditableField(); }, - ); - - activityTargetsAfterUpdate.push(...activityTargetsToCreate); - - if (isNonEmptyArray(activityTargetsToDelete)) { - activityTargetsAfterUpdate = activityTargetsAfterUpdate.filter( - (activityTarget) => - !activityTargetsToDelete.some( - (activityTargetToDelete) => - activityTargetToDelete.activityTarget.id === activityTarget.id, - ), - ); - } - - if (isActivityInCreateMode) { - createManyActivityTargetsInCache(activityTargetsToCreate); - upsertActivity({ - activity, - input: { - activityTargets: activityTargetsAfterUpdate, - }, - }); - } else { - if (activityTargetsToCreate.length > 0) { - await createManyActivityTargets(activityTargetsToCreate); - } - - if (activityTargetsToDelete.length > 0) { - await deleteManyActivityTargets( - activityTargetsToDelete.map( - (activityTargetObjectRecord) => - activityTargetObjectRecord.activityTarget.id, - ), + [ + activityTargetWithTargetRecords, + closeEditableField, + relationPickerScopeId, + setActivityFromStore, + ], + ); + + const handleChange = useRecoilCallback( + ({ snapshot, set }) => + async (recordId: string) => { + const existingActivityTargets = activityTargetWithTargetRecords.map( + (activityTargetObjectRecord) => + activityTargetObjectRecord.activityTarget, ); - } - } - - setActivityFromStore((currentActivity) => { - if (isNull(currentActivity)) { - return null; - } - - return { - ...currentActivity, - activityTargets: activityTargetsAfterUpdate, - }; - }); - }; + + let activityTargetsAfterUpdate = Array.from(existingActivityTargets); + + const previouslyCheckedRecordsIds = snapshot + .getLoadable( + objectRecordMultiSelectCheckedRecordsIdsComponentState({ + scopeId: relationPickerScopeId, + }), + ) + .getValue(); + + const isNewlySelected = !previouslyCheckedRecordsIds.includes(recordId); + + if (isNewlySelected) { + const record = snapshot + .getLoadable( + objectRecordMultiSelectComponentFamilyState({ + scopeId: relationPickerScopeId, + familyKey: recordId, + }), + ) + .getValue(); + + if (!record) { + throw new Error( + `Could not find selected record with id ${recordId}`, + ); + } + + set( + objectRecordMultiSelectCheckedRecordsIdsComponentState({ + scopeId: relationPickerScopeId, + }), + (prev) => [...prev, recordId], + ); + + const newActivityTargetId = v4(); + const fieldName = getActivityTargetObjectFieldName({ + nameSingular: record.objectMetadataItem.nameSingular, + }); + const fieldNameWithIdSuffix = getActivityTargetObjectFieldIdName({ + nameSingular: record.objectMetadataItem.nameSingular, + }); + const newActivityTarget = prefillRecord<ActivityTarget>({ + objectMetadataItem: objectMetadataItemActivityTarget, + input: { + id: newActivityTargetId, + activityId: activity.id, + activity, + createdAt: new Date().toISOString(), + updatedAt: new Date().toISOString(), + [fieldName]: record.record, + [fieldNameWithIdSuffix]: recordId, + }, + }); + + activityTargetsAfterUpdate.push(newActivityTarget); + + if (isActivityInCreateMode) { + createManyActivityTargetsInCache([newActivityTarget]); + upsertActivity({ + activity, + input: { + activityTargets: activityTargetsAfterUpdate, + }, + }); + } else { + await createManyActivityTargets([newActivityTarget]); + } + + set(activityTargetObjectRecordFamilyState(recordId), { + activityTargetId: newActivityTargetId, + }); + } else { + const activityTargetToDeleteId = snapshot + .getLoadable(activityTargetObjectRecordFamilyState(recordId)) + .getValue().activityTargetId; + + if (!activityTargetToDeleteId) { + throw new Error('Could not delete this activity target.'); + } + + set( + objectRecordMultiSelectCheckedRecordsIdsComponentState({ + scopeId: relationPickerScopeId, + }), + previouslyCheckedRecordsIds.filter((id) => id !== recordId), + ); + activityTargetsAfterUpdate = activityTargetsAfterUpdate.filter( + (activityTarget) => activityTarget.id !== activityTargetToDeleteId, + ); + + if (isActivityInCreateMode) { + upsertActivity({ + activity, + input: { + activityTargets: activityTargetsAfterUpdate, + }, + }); + } else { + await deleteManyActivityTargets([activityTargetToDeleteId]); + } + + set(activityTargetObjectRecordFamilyState(recordId), { + activityTargetId: null, + }); + } + }, + [ + activity, + activityTargetWithTargetRecords, + createManyActivityTargets, + createManyActivityTargetsInCache, + deleteManyActivityTargets, + isActivityInCreateMode, + objectMetadataItemActivityTarget, + relationPickerScopeId, + upsertActivity, + ], + ); return ( <StyledSelectContainer> - <MultipleObjectRecordSelect - selectedObjectRecordIds={selectedTargetObjectIds} - onSubmit={handleSubmit} - /> + <RelationPickerScope relationPickerScopeId={relationPickerScopeId}> + <ActivityTargetObjectRecordEffect + activityTargetWithTargetRecords={activityTargetWithTargetRecords} + /> + <ActivityTargetInlineCellEditModeMultiRecordsEffect + selectedObjectRecordIds={selectedTargetObjectIds} + /> + <MultiRecordSelect onSubmit={handleSubmit} onChange={handleChange} /> + </RelationPickerScope> </StyledSelectContainer> ); }; diff --git a/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetObjectRecordEffect.tsx b/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetObjectRecordEffect.tsx new file mode 100644 index 000000000000..aee9ea49d2d6 --- /dev/null +++ b/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetObjectRecordEffect.tsx @@ -0,0 +1,42 @@ +import { useEffect } from 'react'; +import { useRecoilCallback } from 'recoil'; + +import { ActivityTargetWithTargetRecord } from '@/activities/types/ActivityTargetObject'; +import { activityTargetObjectRecordFamilyState } from '@/object-record/record-field/states/activityTargetObjectRecordFamilyState'; +import { isDeeplyEqual } from '~/utils/isDeeplyEqual'; + +export const ActivityTargetObjectRecordEffect = ({ + activityTargetWithTargetRecords, +}: { + activityTargetWithTargetRecords: ActivityTargetWithTargetRecord[]; +}) => { + const updateActivityTargets = useRecoilCallback( + ({ snapshot, set }) => + (newActivityTargets: ActivityTargetWithTargetRecord[]) => { + for (const newActivityTarget of newActivityTargets) { + const objectRecordId = newActivityTarget.targetObject.id; + const record = snapshot + .getLoadable(activityTargetObjectRecordFamilyState(objectRecordId)) + .getValue(); + + if ( + !isDeeplyEqual( + record.activityTargetId, + newActivityTarget.activityTarget.id, + ) + ) { + set(activityTargetObjectRecordFamilyState(objectRecordId), { + activityTargetId: newActivityTarget.activityTarget.id, + }); + } + } + }, + [], + ); + + useEffect(() => { + updateActivityTargets(activityTargetWithTargetRecords); + }, [activityTargetWithTargetRecords, updateActivityTargets]); + + return <></>; +}; diff --git a/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetsInlineCell.tsx b/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetsInlineCell.tsx index 2755acefd1c4..f1490e1d49b0 100644 --- a/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetsInlineCell.tsx +++ b/packages/twenty-front/src/modules/activities/inline-cell/components/ActivityTargetsInlineCell.tsx @@ -7,6 +7,8 @@ import { useActivityTargetObjectRecords } from '@/activities/hooks/useActivityTa import { ActivityTargetInlineCellEditMode } from '@/activities/inline-cell/components/ActivityTargetInlineCellEditMode'; import { Activity } from '@/activities/types/Activity'; import { ActivityEditorHotkeyScope } from '@/activities/types/ActivityEditorHotkeyScope'; +import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; +import { useFieldContext } from '@/object-record/hooks/useFieldContext'; import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; import { FieldFocusContextProvider } from '@/object-record/record-field/contexts/FieldFocusContextProvider'; import { RecordFieldInputScope } from '@/object-record/record-field/scopes/RecordFieldInputScope'; @@ -41,33 +43,45 @@ export const ActivityTargetsInlineCell = ({ ActivityEditorHotkeyScope.ActivityTargets, ); + const { FieldContextProvider: ActivityTargetsContextProvider } = + useFieldContext({ + objectNameSingular: CoreObjectNameSingular.Activity, + objectRecordId: activity.id, + fieldMetadataName: 'activityTargets', + fieldPosition: 3, + overridenIsFieldEmpty: activityTargetObjectRecords.length === 0, + }); + return ( <RecordFieldInputScope recordFieldInputScopeId={activity?.id ?? ''}> <FieldFocusContextProvider> - <RecordInlineCellContainer - buttonIcon={IconPencil} - customEditHotkeyScope={{ - scope: ActivityEditorHotkeyScope.ActivityTargets, - }} - IconLabel={showLabel ? IconArrowUpRight : undefined} - showLabel={showLabel} - readonly={readonly} - labelWidth={fieldDefinition?.labelWidth} - editModeContent={ - <ActivityTargetInlineCellEditMode - activity={activity} - activityTargetWithTargetRecords={activityTargetObjectRecords} - /> - } - label="Relations" - displayModeContent={ - <ActivityTargetChips - activityTargetObjectRecords={activityTargetObjectRecords} - maxWidth={maxWidth} + {ActivityTargetsContextProvider && ( + <ActivityTargetsContextProvider> + <RecordInlineCellContainer + buttonIcon={IconPencil} + customEditHotkeyScope={{ + scope: ActivityEditorHotkeyScope.ActivityTargets, + }} + IconLabel={showLabel ? IconArrowUpRight : undefined} + showLabel={showLabel} + readonly={readonly} + labelWidth={fieldDefinition?.labelWidth} + editModeContent={ + <ActivityTargetInlineCellEditMode + activity={activity} + activityTargetWithTargetRecords={activityTargetObjectRecords} + /> + } + label="Relations" + displayModeContent={ + <ActivityTargetChips + activityTargetObjectRecords={activityTargetObjectRecords} + maxWidth={maxWidth} + /> + } /> - } - isDisplayModeContentEmpty={activityTargetObjectRecords.length === 0} - /> + </ActivityTargetsContextProvider> + )} </FieldFocusContextProvider> </RecordFieldInputScope> ); diff --git a/packages/twenty-front/src/modules/activities/notes/components/Notes.tsx b/packages/twenty-front/src/modules/activities/notes/components/Notes.tsx index 173bda4ede21..4f3e83656eb4 100644 --- a/packages/twenty-front/src/modules/activities/notes/components/Notes.tsx +++ b/packages/twenty-front/src/modules/activities/notes/components/Notes.tsx @@ -1,6 +1,7 @@ import styled from '@emotion/styled'; import { IconPlus } from 'twenty-ui'; +import { SkeletonLoader } from '@/activities/components/SkeletonLoader'; import { useOpenCreateActivityDrawer } from '@/activities/hooks/useOpenCreateActivityDrawer'; import { NoteList } from '@/activities/notes/components/NoteList'; import { useNotes } from '@/activities/notes/hooks/useNotes'; @@ -12,6 +13,7 @@ import { AnimatedPlaceholderEmptySubTitle, AnimatedPlaceholderEmptyTextContainer, AnimatedPlaceholderEmptyTitle, + EMPTY_PLACEHOLDER_TRANSITION_PROPS, } from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; const StyledNotesContainer = styled.div` @@ -27,13 +29,22 @@ export const Notes = ({ }: { targetableObject: ActivityTargetableObject; }) => { - const { notes } = useNotes(targetableObject); + const { notes, loading } = useNotes(targetableObject); const openCreateActivity = useOpenCreateActivityDrawer(); - if (notes?.length === 0) { + const isNotesEmpty = !notes || notes.length === 0; + + if (loading && isNotesEmpty) { + return <SkeletonLoader />; + } + + if (isNotesEmpty) { return ( - <AnimatedPlaceholderEmptyContainer> + <AnimatedPlaceholderEmptyContainer + // eslint-disable-next-line react/jsx-props-no-spreading + {...EMPTY_PLACEHOLDER_TRANSITION_PROPS} + > <AnimatedPlaceholder type="noNote" /> <AnimatedPlaceholderEmptyTextContainer> <AnimatedPlaceholderEmptyTitle> @@ -62,7 +73,7 @@ export const Notes = ({ <StyledNotesContainer> <NoteList title="All" - notes={notes ?? []} + notes={notes} button={ <Button Icon={IconPlus} diff --git a/packages/twenty-front/src/modules/activities/notes/hooks/useNotes.ts b/packages/twenty-front/src/modules/activities/notes/hooks/useNotes.ts index b50231f9e413..383061c5e04f 100644 --- a/packages/twenty-front/src/modules/activities/notes/hooks/useNotes.ts +++ b/packages/twenty-front/src/modules/activities/notes/hooks/useNotes.ts @@ -24,7 +24,7 @@ export const useNotes = (targetableObject: ActivityTargetableObject) => { const { activities, loading } = useActivities({ activitiesFilters: notesQueryVariables.filter ?? {}, - activitiesOrderByVariables: notesQueryVariables.orderBy ?? {}, + activitiesOrderByVariables: notesQueryVariables.orderBy ?? [{}], targetableObjects: [targetableObject], }); diff --git a/packages/twenty-front/src/modules/activities/states/objectRecordsIdsMultiSelectComponentState.ts b/packages/twenty-front/src/modules/activities/states/objectRecordsIdsMultiSelectComponentState.ts new file mode 100644 index 000000000000..67959df63be7 --- /dev/null +++ b/packages/twenty-front/src/modules/activities/states/objectRecordsIdsMultiSelectComponentState.ts @@ -0,0 +1,8 @@ +import { createComponentState } from '@/ui/utilities/state/component-state/utils/createComponentState'; + +export const objectRecordsIdsMultiSelecComponentState = createComponentState< + string[] +>({ + key: 'objectRecordsIdsMultiSelectComponentState', + defaultValue: [], +}); diff --git a/packages/twenty-front/src/modules/activities/tasks/components/TaskGroups.tsx b/packages/twenty-front/src/modules/activities/tasks/components/TaskGroups.tsx index 9103bb2d630b..03c2bdb97863 100644 --- a/packages/twenty-front/src/modules/activities/tasks/components/TaskGroups.tsx +++ b/packages/twenty-front/src/modules/activities/tasks/components/TaskGroups.tsx @@ -2,6 +2,7 @@ import styled from '@emotion/styled'; import { useRecoilValue } from 'recoil'; import { IconPlus } from 'twenty-ui'; +import { SkeletonLoader } from '@/activities/components/SkeletonLoader'; import { useOpenCreateActivityDrawer } from '@/activities/hooks/useOpenCreateActivityDrawer'; import { TASKS_TAB_LIST_COMPONENT_ID } from '@/activities/tasks/constants/TasksTabListComponentId'; import { useTasks } from '@/activities/tasks/hooks/useTasks'; @@ -13,6 +14,7 @@ import { AnimatedPlaceholderEmptySubTitle, AnimatedPlaceholderEmptyTextContainer, AnimatedPlaceholderEmptyTitle, + EMPTY_PLACEHOLDER_TRANSITION_PROPS, } from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; import { useTabList } from '@/ui/layout/tab/hooks/useTabList'; @@ -40,6 +42,8 @@ export const TaskGroups = ({ upcomingTasks, unscheduledTasks, completedTasks, + incompleteTasksLoading, + completeTasksLoading, } = useTasks({ filterDropdownId: filterDropdownId, targetableObjects: targetableObjects ?? [], @@ -50,15 +54,27 @@ export const TaskGroups = ({ const { activeTabIdState } = useTabList(TASKS_TAB_LIST_COMPONENT_ID); const activeTabId = useRecoilValue(activeTabIdState); - if ( + const isLoading = + (activeTabId !== 'done' && incompleteTasksLoading) || + (activeTabId === 'done' && completeTasksLoading); + + const isTasksEmpty = (activeTabId !== 'done' && todayOrPreviousTasks?.length === 0 && upcomingTasks?.length === 0 && unscheduledTasks?.length === 0) || - (activeTabId === 'done' && completedTasks?.length === 0) - ) { + (activeTabId === 'done' && completedTasks?.length === 0); + + if (isLoading && isTasksEmpty) { + return <SkeletonLoader />; + } + + if (isTasksEmpty) { return ( - <AnimatedPlaceholderEmptyContainer> + <AnimatedPlaceholderEmptyContainer + // eslint-disable-next-line react/jsx-props-no-spreading + {...EMPTY_PLACEHOLDER_TRANSITION_PROPS} + > <AnimatedPlaceholder type="noTask" /> <AnimatedPlaceholderEmptyTextContainer> <AnimatedPlaceholderEmptyTitle> diff --git a/packages/twenty-front/src/modules/activities/tasks/hooks/useTasks.ts b/packages/twenty-front/src/modules/activities/tasks/hooks/useTasks.ts index bb8fb16566ac..e560e2607940 100644 --- a/packages/twenty-front/src/modules/activities/tasks/hooks/useTasks.ts +++ b/packages/twenty-front/src/modules/activities/tasks/hooks/useTasks.ts @@ -107,17 +107,19 @@ export const useTasks = ({ setCurrentIncompleteTaskQueryVariables, ]); - const { activities: completeTasksData } = useActivities({ - targetableObjects, - activitiesFilters: completedQueryVariables.filter ?? {}, - activitiesOrderByVariables: completedQueryVariables.orderBy ?? {}, - }); - - const { activities: incompleteTaskData } = useActivities({ - targetableObjects, - activitiesFilters: incompleteQueryVariables.filter ?? {}, - activitiesOrderByVariables: incompleteQueryVariables.orderBy ?? {}, - }); + const { activities: completeTasksData, loading: completeTasksLoading } = + useActivities({ + targetableObjects, + activitiesFilters: completedQueryVariables.filter ?? {}, + activitiesOrderByVariables: completedQueryVariables.orderBy ?? [{}], + }); + + const { activities: incompleteTaskData, loading: incompleteTasksLoading } = + useActivities({ + targetableObjects, + activitiesFilters: incompleteQueryVariables.filter ?? {}, + activitiesOrderByVariables: incompleteQueryVariables.orderBy ?? [{}], + }); const todayOrPreviousTasks = incompleteTaskData?.filter((task) => { if (!task.dueAt) { @@ -148,5 +150,7 @@ export const useTasks = ({ upcomingTasks: (upcomingTasks ?? []) as Activity[], unscheduledTasks: (unscheduledTasks ?? []) as Activity[], completedTasks: (completedTasks ?? []) as Activity[], + completeTasksLoading, + incompleteTasksLoading, }; }; diff --git a/packages/twenty-front/src/modules/activities/timeline/components/Timeline.tsx b/packages/twenty-front/src/modules/activities/timeline/components/Timeline.tsx deleted file mode 100644 index 8ee67657592d..000000000000 --- a/packages/twenty-front/src/modules/activities/timeline/components/Timeline.tsx +++ /dev/null @@ -1,68 +0,0 @@ -import styled from '@emotion/styled'; -import { useRecoilValue } from 'recoil'; - -import { TimelineCreateButtonGroup } from '@/activities/timeline/components/TimelineCreateButtonGroup'; -import { TimelineSkeletonLoader } from '@/activities/timeline/components/TimelineSkeletonLoader'; -import { timelineActivitiesForGroupState } from '@/activities/timeline/states/timelineActivitiesForGroupState'; -import { ActivityTargetableObject } from '@/activities/types/ActivityTargetableEntity'; -import AnimatedPlaceholder from '@/ui/layout/animated-placeholder/components/AnimatedPlaceholder'; -import { - AnimatedPlaceholderEmptyContainer, - AnimatedPlaceholderEmptySubTitle, - AnimatedPlaceholderEmptyTextContainer, - AnimatedPlaceholderEmptyTitle, -} from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; -import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; - -import { TimelineItemsContainer } from './TimelineItemsContainer'; - -const StyledMainContainer = styled.div` - align-items: flex-start; - align-self: stretch; - border-top: ${({ theme }) => - useIsMobile() ? `1px solid ${theme.border.color.medium}` : 'none'}; - display: flex; - flex-direction: column; - height: 100%; - - justify-content: center; -`; - -export const Timeline = ({ - targetableObject, - loading, -}: { - targetableObject: ActivityTargetableObject; - loading: boolean; -}) => { - const timelineActivitiesForGroup = useRecoilValue( - timelineActivitiesForGroupState, - ); - - if (loading) { - return <TimelineSkeletonLoader />; - } - - if (timelineActivitiesForGroup.length === 0) { - return ( - <AnimatedPlaceholderEmptyContainer> - <AnimatedPlaceholder type="emptyTimeline" /> - <AnimatedPlaceholderEmptyTextContainer> - <AnimatedPlaceholderEmptyTitle> - Add your first Activity - </AnimatedPlaceholderEmptyTitle> - <AnimatedPlaceholderEmptySubTitle> - There are no activities associated with this record.{' '} - </AnimatedPlaceholderEmptySubTitle> - </AnimatedPlaceholderEmptyTextContainer> - <TimelineCreateButtonGroup targetableObject={targetableObject} /> - </AnimatedPlaceholderEmptyContainer> - ); - } - - return ( - <StyledMainContainer> - <TimelineItemsContainer /> - </StyledMainContainer> - ); -}; diff --git a/packages/twenty-front/src/modules/activities/timeline/components/TimelineActivity.tsx b/packages/twenty-front/src/modules/activities/timeline/components/TimelineActivity.tsx index 2b4ff934ff95..fc915a1ce1bc 100644 --- a/packages/twenty-front/src/modules/activities/timeline/components/TimelineActivity.tsx +++ b/packages/twenty-front/src/modules/activities/timeline/components/TimelineActivity.tsx @@ -1,8 +1,7 @@ -import { Tooltip } from 'react-tooltip'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { useRecoilValue } from 'recoil'; -import { Avatar, IconCheckbox, IconNotes } from 'twenty-ui'; +import { AppTooltip, Avatar, IconCheckbox, IconNotes } from 'twenty-ui'; import { useOpenActivityRightDrawer } from '@/activities/hooks/useOpenActivityRightDrawer'; import { timelineActivityWithoutTargetsFamilyState } from '@/activities/timeline/states/timelineActivityWithoutTargetsFamilyState'; @@ -109,21 +108,6 @@ const StyledVerticalLine = styled.div` width: 2px; `; -const StyledTooltip = styled(Tooltip)` - background-color: ${({ theme }) => theme.background.primary}; - - box-shadow: 0px 2px 4px 3px - ${({ theme }) => theme.background.transparent.light}; - - box-shadow: 2px 4px 16px 6px - ${({ theme }) => theme.background.transparent.light}; - - color: ${({ theme }) => theme.font.color.primary}; - - opacity: 1; - padding: ${({ theme }) => theme.spacing(2)}; -`; - const StyledTimelineItemContainer = styled.div<{ isGap?: boolean }>` align-items: center; align-self: stretch; @@ -217,7 +201,7 @@ export const TimelineActivity = ({ <StyledItemTitleDate id={`id-${activityForTimeline.id}`}> {beautifiedCreatedAt} </StyledItemTitleDate> - <StyledTooltip + <AppTooltip anchorSelect={`#id-${activityForTimeline.id}`} content={exactCreatedAt} clickable diff --git a/packages/twenty-front/src/modules/activities/timeline/components/TimelineCreateButtonGroup.tsx b/packages/twenty-front/src/modules/activities/timeline/components/TimelineCreateButtonGroup.tsx index 21c0c5713014..a4ca491496d0 100644 --- a/packages/twenty-front/src/modules/activities/timeline/components/TimelineCreateButtonGroup.tsx +++ b/packages/twenty-front/src/modules/activities/timeline/components/TimelineCreateButtonGroup.tsx @@ -1,44 +1,30 @@ import { useSetRecoilState } from 'recoil'; import { IconCheckbox, IconNotes, IconPaperclip } from 'twenty-ui'; -import { useOpenCreateActivityDrawer } from '@/activities/hooks/useOpenCreateActivityDrawer'; -import { ActivityTargetableObject } from '@/activities/types/ActivityTargetableEntity'; import { Button } from '@/ui/input/button/components/Button'; import { ButtonGroup } from '@/ui/input/button/components/ButtonGroup'; import { TAB_LIST_COMPONENT_ID } from '@/ui/layout/show-page/components/ShowPageRightContainer'; import { useTabList } from '@/ui/layout/tab/hooks/useTabList'; -export const TimelineCreateButtonGroup = ({ - targetableObject, -}: { - targetableObject: ActivityTargetableObject; -}) => { +export const TimelineCreateButtonGroup = () => { const { activeTabIdState } = useTabList(TAB_LIST_COMPONENT_ID); const setActiveTabId = useSetRecoilState(activeTabIdState); - const openCreateActivity = useOpenCreateActivityDrawer(); - return ( <ButtonGroup variant={'secondary'}> <Button Icon={IconNotes} title="Note" - onClick={() => - openCreateActivity({ - type: 'Note', - targetableObjects: [targetableObject], - }) - } + onClick={() => { + setActiveTabId('notes'); + }} /> <Button Icon={IconCheckbox} title="Task" - onClick={() => - openCreateActivity({ - type: 'Task', - targetableObjects: [targetableObject], - }) - } + onClick={() => { + setActiveTabId('tasks'); + }} /> <Button Icon={IconPaperclip} diff --git a/packages/twenty-front/src/modules/activities/timeline/components/TimelineQueryEffect.tsx b/packages/twenty-front/src/modules/activities/timeline/components/TimelineQueryEffect.tsx index 88c9669119b4..60367044b336 100644 --- a/packages/twenty-front/src/modules/activities/timeline/components/TimelineQueryEffect.tsx +++ b/packages/twenty-front/src/modules/activities/timeline/components/TimelineQueryEffect.tsx @@ -4,7 +4,7 @@ import { useRecoilCallback, useRecoilState, useSetRecoilState } from 'recoil'; import { useActivities } from '@/activities/hooks/useActivities'; import { FIND_MANY_TIMELINE_ACTIVITIES_ORDER_BY } from '@/activities/timeline/constants/FindManyTimelineActivitiesOrderBy'; import { objectShowPageTargetableObjectState } from '@/activities/timeline/states/objectShowPageTargetableObjectIdState'; -import { timelineActivitiesFammilyState } from '@/activities/timeline/states/timelineActivitiesFamilyState'; +import { timelineActivitiesFamilyState } from '@/activities/timeline/states/timelineActivitiesFamilyState'; import { timelineActivitiesForGroupState } from '@/activities/timeline/states/timelineActivitiesForGroupState'; import { timelineActivityWithoutTargetsFamilyState } from '@/activities/timeline/states/timelineActivityWithoutTargetsFamilyState'; import { Activity } from '@/activities/types/Activity'; @@ -68,11 +68,11 @@ export const TimelineQueryEffect = ({ (newActivities: Activity[]) => { for (const newActivity of newActivities) { const currentActivity = snapshot - .getLoadable(timelineActivitiesFammilyState(newActivity.id)) + .getLoadable(timelineActivitiesFamilyState(newActivity.id)) .getValue(); if (!isDeeplyEqual(newActivity, currentActivity)) { - set(timelineActivitiesFammilyState(newActivity.id), newActivity); + set(timelineActivitiesFamilyState(newActivity.id), newActivity); } const currentActivityWithoutTarget = snapshot diff --git a/packages/twenty-front/src/modules/activities/timeline/constants/FindManyTimelineActivitiesOrderBy.ts b/packages/twenty-front/src/modules/activities/timeline/constants/FindManyTimelineActivitiesOrderBy.ts index 4e48c6dc5f65..ea6c4763571a 100644 --- a/packages/twenty-front/src/modules/activities/timeline/constants/FindManyTimelineActivitiesOrderBy.ts +++ b/packages/twenty-front/src/modules/activities/timeline/constants/FindManyTimelineActivitiesOrderBy.ts @@ -1,6 +1,8 @@ import { RecordGqlOperationOrderBy } from '@/object-record/graphql/types/RecordGqlOperationOrderBy'; export const FIND_MANY_TIMELINE_ACTIVITIES_ORDER_BY: RecordGqlOperationOrderBy = - { - createdAt: 'DescNullsFirst', - }; + [ + { + createdAt: 'DescNullsFirst', + }, + ]; diff --git a/packages/twenty-front/src/modules/activities/timeline/states/timelineActivitiesFamilyState.ts b/packages/twenty-front/src/modules/activities/timeline/states/timelineActivitiesFamilyState.ts index f2acc7a3312f..88903b6c8e38 100644 --- a/packages/twenty-front/src/modules/activities/timeline/states/timelineActivitiesFamilyState.ts +++ b/packages/twenty-front/src/modules/activities/timeline/states/timelineActivitiesFamilyState.ts @@ -1,10 +1,10 @@ import { Activity } from '@/activities/types/Activity'; import { createFamilyState } from '@/ui/utilities/state/utils/createFamilyState'; -export const timelineActivitiesFammilyState = createFamilyState< +export const timelineActivitiesFamilyState = createFamilyState< Activity | null, string >({ - key: 'timelineActivitiesFammilyState', + key: 'timelineActivitiesFamilyState', defaultValue: null, }); diff --git a/packages/twenty-front/src/modules/activities/timeline/utils/makeTimelineActivitiesQueryVariables.ts b/packages/twenty-front/src/modules/activities/timeline/utils/makeTimelineActivitiesQueryVariables.ts index c8a7030d2d99..4a9c67f5b54c 100644 --- a/packages/twenty-front/src/modules/activities/timeline/utils/makeTimelineActivitiesQueryVariables.ts +++ b/packages/twenty-front/src/modules/activities/timeline/utils/makeTimelineActivitiesQueryVariables.ts @@ -13,8 +13,10 @@ export const makeTimelineActivitiesQueryVariables = ({ in: [...activityIds].sort(sortByAscString), }, }, - orderBy: { - createdAt: 'DescNullsFirst', - }, + orderBy: [ + { + createdAt: 'DescNullsFirst', + }, + ], }; }; diff --git a/packages/twenty-front/src/modules/activities/timelineActivities/components/EventRow.tsx b/packages/twenty-front/src/modules/activities/timelineActivities/components/EventRow.tsx index b1072ee5790e..050b5c1b3206 100644 --- a/packages/twenty-front/src/modules/activities/timelineActivities/components/EventRow.tsx +++ b/packages/twenty-front/src/modules/activities/timelineActivities/components/EventRow.tsx @@ -10,10 +10,23 @@ import { TimelineActivity } from '@/activities/timelineActivities/types/Timeline import { getTimelineActivityAuthorFullName } from '@/activities/timelineActivities/utils/getTimelineActivityAuthorFullName'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; -import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; import { beautifyPastDateRelativeToNow } from '~/utils/date-utils'; import { isUndefinedOrNull } from '~/utils/isUndefinedOrNull'; +const StyledTimelineItemContainer = styled.div` + display: flex; + gap: ${({ theme }) => theme.spacing(4)}; + height: 'auto'; + justify-content: space-between; + overflow: hidden; + white-space: nowrap; +`; + +const StyledLeftContainer = styled.div` + display: flex; + flex-direction: column; +`; + const StyledIconContainer = styled.div` display: flex; align-items: center; @@ -25,62 +38,50 @@ const StyledIconContainer = styled.div` user-select: none; text-decoration-line: underline; z-index: 2; - align-self: normal; -`; - -const StyledItemContainer = styled.div` - display: flex; - align-items: center; - gap: ${({ theme }) => theme.spacing(1)}; - flex: 1; - overflow: hidden; -`; - -const StyledItemTitleDate = styled.div` - align-items: center; - color: ${({ theme }) => theme.font.color.tertiary}; - display: flex; - gap: ${({ theme }) => theme.spacing(2)}; - justify-content: flex-end; - margin-left: auto; - align-self: baseline; `; const StyledVerticalLineContainer = styled.div` - align-items: center; - align-self: stretch; display: flex; - gap: ${({ theme }) => theme.spacing(2)}; + flex-shrink: 0; justify-content: center; - width: 26px; z-index: 2; + height: 100%; `; const StyledVerticalLine = styled.div` - align-self: stretch; background: ${({ theme }) => theme.border.color.light}; - flex-shrink: 0; width: 2px; + height: 100%; `; -const StyledTimelineItemContainer = styled.div<{ isGap?: boolean }>` - display: flex; +const StyledSummary = styled.summary` align-items: center; - justify-content: space-between; - gap: ${({ theme }) => theme.spacing(4)}; - height: ${({ isGap, theme }) => - isGap ? (useIsMobile() ? theme.spacing(6) : theme.spacing(3)) : 'auto'}; - overflow: hidden; - white-space: nowrap; + display: flex; + flex: 1; + flex-direction: row; + gap: ${({ theme }) => theme.spacing(1)}; `; -const StyledSummary = styled.summary` +const StyledItemContainer = styled.div<{ isMarginBottom?: boolean }>` + align-items: flex-start; display: flex; flex: 1; - flex-direction: row; + flex-direction: column; gap: ${({ theme }) => theme.spacing(1)}; - align-items: center; overflow: hidden; + margin-bottom: ${({ isMarginBottom, theme }) => + isMarginBottom ? theme.spacing(3) : 0}; + min-height: 26px; +`; + +const StyledItemTitleDate = styled.div` + align-items: flex-start; + padding-top: ${({ theme }) => theme.spacing(1)}; + color: ${({ theme }) => theme.font.color.tertiary}; + display: flex; + gap: ${({ theme }) => theme.spacing(1)}; + justify-content: flex-end; + margin-left: auto; `; type EventRowProps = { @@ -118,13 +119,20 @@ export const EventRow = ({ return ( <> <StyledTimelineItemContainer> - <StyledIconContainer> - <EventIconDynamicComponent - event={event} - linkedObjectMetadataItem={linkedObjectMetadataItem} - /> - </StyledIconContainer> - <StyledItemContainer> + <StyledLeftContainer> + <StyledIconContainer> + <EventIconDynamicComponent + event={event} + linkedObjectMetadataItem={linkedObjectMetadataItem} + /> + </StyledIconContainer> + {!isLastEvent && ( + <StyledVerticalLineContainer> + <StyledVerticalLine /> + </StyledVerticalLineContainer> + )} + </StyledLeftContainer> + <StyledItemContainer isMarginBottom={!isLastEvent}> <StyledSummary> <EventRowDynamicComponent authorFullName={authorFullName} @@ -134,18 +142,11 @@ export const EventRow = ({ linkedObjectMetadataItem={linkedObjectMetadataItem} /> </StyledSummary> - <StyledItemTitleDate id={`id-${event.id}`}> - {beautifiedCreatedAt} - </StyledItemTitleDate> </StyledItemContainer> + <StyledItemTitleDate id={`id-${event.id}`}> + {beautifiedCreatedAt} + </StyledItemTitleDate> </StyledTimelineItemContainer> - {!isLastEvent && ( - <StyledTimelineItemContainer isGap> - <StyledVerticalLineContainer> - <StyledVerticalLine /> - </StyledVerticalLineContainer> - </StyledTimelineItemContainer> - )} </> ); }; diff --git a/packages/twenty-front/src/modules/activities/timelineActivities/components/TimelineActivities.tsx b/packages/twenty-front/src/modules/activities/timelineActivities/components/TimelineActivities.tsx index c0b47efdac09..f24972e9509c 100644 --- a/packages/twenty-front/src/modules/activities/timelineActivities/components/TimelineActivities.tsx +++ b/packages/twenty-front/src/modules/activities/timelineActivities/components/TimelineActivities.tsx @@ -1,7 +1,7 @@ import styled from '@emotion/styled'; -import { isNonEmptyArray } from '@sniptt/guards'; import { CustomResolverFetchMoreLoader } from '@/activities/components/CustomResolverFetchMoreLoader'; +import { SkeletonLoader } from '@/activities/components/SkeletonLoader'; import { TimelineCreateButtonGroup } from '@/activities/timeline/components/TimelineCreateButtonGroup'; import { EventList } from '@/activities/timelineActivities/components/EventList'; import { useTimelineActivities } from '@/activities/timelineActivities/hooks/useTimelineActivities'; @@ -12,6 +12,7 @@ import { AnimatedPlaceholderEmptySubTitle, AnimatedPlaceholderEmptyTextContainer, AnimatedPlaceholderEmptyTitle, + EMPTY_PLACEHOLDER_TRANSITION_PROPS, } from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; @@ -23,11 +24,11 @@ const StyledMainContainer = styled.div` display: flex; flex-direction: column; height: 100%; + overflow: auto; justify-content: center; padding-top: ${({ theme }) => theme.spacing(6)}; padding-right: ${({ theme }) => theme.spacing(6)}; - padding-bottom: ${({ theme }) => theme.spacing(16)}; padding-left: ${({ theme }) => theme.spacing(6)}; gap: ${({ theme }) => theme.spacing(4)}; `; @@ -40,9 +41,19 @@ export const TimelineActivities = ({ const { timelineActivities, loading, fetchMoreRecords } = useTimelineActivities(targetableObject); - if (!isNonEmptyArray(timelineActivities)) { + const isTimelineActivitiesEmpty = + !timelineActivities || timelineActivities.length === 0; + + if (loading && isTimelineActivitiesEmpty) { + return <SkeletonLoader withSubSections />; + } + + if (isTimelineActivitiesEmpty) { return ( - <AnimatedPlaceholderEmptyContainer> + <AnimatedPlaceholderEmptyContainer + // eslint-disable-next-line react/jsx-props-no-spreading + {...EMPTY_PLACEHOLDER_TRANSITION_PROPS} + > <AnimatedPlaceholder type="emptyTimeline" /> <AnimatedPlaceholderEmptyTextContainer> <AnimatedPlaceholderEmptyTitle> @@ -52,7 +63,7 @@ export const TimelineActivities = ({ There are no activities associated with this record.{' '} </AnimatedPlaceholderEmptySubTitle> </AnimatedPlaceholderEmptyTextContainer> - <TimelineCreateButtonGroup targetableObject={targetableObject} /> + <TimelineCreateButtonGroup /> </AnimatedPlaceholderEmptyContainer> ); } diff --git a/packages/twenty-front/src/modules/activities/timelineActivities/hooks/useTimelineActivities.tsx b/packages/twenty-front/src/modules/activities/timelineActivities/hooks/useTimelineActivities.tsx index a6173b5d9953..e18dff4befe6 100644 --- a/packages/twenty-front/src/modules/activities/timelineActivities/hooks/useTimelineActivities.tsx +++ b/packages/twenty-front/src/modules/activities/timelineActivities/hooks/useTimelineActivities.tsx @@ -23,9 +23,11 @@ export const useTimelineActivities = ( eq: targetableObject.id, }, }, - orderBy: { - createdAt: 'DescNullsFirst', - }, + orderBy: [ + { + createdAt: 'DescNullsFirst', + }, + ], recordGqlFields: { id: true, createdAt: true, diff --git a/packages/twenty-front/src/modules/activities/timelineActivities/rows/calendar/components/EventCardCalendarEvent.tsx b/packages/twenty-front/src/modules/activities/timelineActivities/rows/calendar/components/EventCardCalendarEvent.tsx index 413a70e213ca..8d572594f4fc 100644 --- a/packages/twenty-front/src/modules/activities/timelineActivities/rows/calendar/components/EventCardCalendarEvent.tsx +++ b/packages/twenty-front/src/modules/activities/timelineActivities/rows/calendar/components/EventCardCalendarEvent.tsx @@ -5,7 +5,7 @@ import { useOpenCalendarEventRightDrawer } from '@/activities/calendar/right-dra import { CalendarEvent } from '@/activities/calendar/types/CalendarEvent'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { useFindOneRecord } from '@/object-record/hooks/useFindOneRecord'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; import { formatToHumanReadableDay, formatToHumanReadableMonth, @@ -85,7 +85,7 @@ export const EventCardCalendarEvent = ({ }: { calendarEventId: string; }) => { - const { setRecords } = useSetRecordInStore(); + const { upsertRecords } = useUpsertRecordsInStore(); const { record: calendarEvent, @@ -101,7 +101,7 @@ export const EventCardCalendarEvent = ({ endsAt: true, }, onCompleted: (data) => { - setRecords([data]); + upsertRecords([data]); }, }); diff --git a/packages/twenty-front/src/modules/activities/timelineActivities/rows/components/EventCard.tsx b/packages/twenty-front/src/modules/activities/timelineActivities/rows/components/EventCard.tsx index baed69eb738a..67e4f78cd726 100644 --- a/packages/twenty-front/src/modules/activities/timelineActivities/rows/components/EventCard.tsx +++ b/packages/twenty-front/src/modules/activities/timelineActivities/rows/components/EventCard.tsx @@ -15,7 +15,7 @@ const StyledCardContainer = styled.div` gap: ${({ theme }) => theme.spacing(2)}; width: 400px; padding: ${({ theme }) => theme.spacing(2)} 0px - ${({ theme }) => theme.spacing(4)} 0px; + ${({ theme }) => theme.spacing(1)} 0px; `; const StyledCard = styled(Card)` diff --git a/packages/twenty-front/src/modules/activities/timelineActivities/rows/message/components/EventCardMessage.tsx b/packages/twenty-front/src/modules/activities/timelineActivities/rows/message/components/EventCardMessage.tsx index 9c2b41dd4d7f..134dd95ba3a7 100644 --- a/packages/twenty-front/src/modules/activities/timelineActivities/rows/message/components/EventCardMessage.tsx +++ b/packages/twenty-front/src/modules/activities/timelineActivities/rows/message/components/EventCardMessage.tsx @@ -7,7 +7,7 @@ import { EmailThreadMessage } from '@/activities/emails/types/EmailThreadMessage import { EventCardMessageNotShared } from '@/activities/timelineActivities/rows/message/components/EventCardMessageNotShared'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { useFindOneRecord } from '@/object-record/hooks/useFindOneRecord'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; import { isDefined } from '~/utils/isDefined'; const StyledEventCardMessageContainer = styled.div` @@ -56,7 +56,7 @@ export const EventCardMessage = ({ messageId: string; authorFullName: string; }) => { - const { setRecords } = useSetRecordInStore(); + const { upsertRecords } = useUpsertRecordsInStore(); const { record: message, @@ -75,7 +75,7 @@ export const EventCardMessage = ({ }, }, onCompleted: (data) => { - setRecords([data]); + upsertRecords([data]); }, }); diff --git a/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/sortCachedObjectEdges.ts b/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/sortCachedObjectEdges.ts index aa8434f0ba80..6f76dd66a5a8 100644 --- a/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/sortCachedObjectEdges.ts +++ b/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/sortCachedObjectEdges.ts @@ -16,7 +16,7 @@ export const sortCachedObjectEdges = ({ orderBy: RecordGqlOperationOrderBy; readCacheField: ReadFieldFunction; }) => { - const [orderByFieldName, orderByFieldValue] = Object.entries(orderBy)[0]; + const [orderByFieldName, orderByFieldValue] = Object.entries(orderBy[0])[0]; const [orderBySubFieldName, orderBySubFieldValue] = typeof orderByFieldValue === 'string' ? [] diff --git a/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerCreateRecordsOptimisticEffect.ts b/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerCreateRecordsOptimisticEffect.ts index 5df2b2acbcc7..057f93006fa0 100644 --- a/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerCreateRecordsOptimisticEffect.ts +++ b/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerCreateRecordsOptimisticEffect.ts @@ -7,7 +7,11 @@ import { RecordGqlRefEdge } from '@/object-record/cache/types/RecordGqlRefEdge'; import { getEdgeTypename } from '@/object-record/cache/utils/getEdgeTypename'; import { isObjectRecordConnectionWithRefs } from '@/object-record/cache/utils/isObjectRecordConnectionWithRefs'; import { RecordGqlNode } from '@/object-record/graphql/types/RecordGqlNode'; +import { isRecordMatchingFilter } from '@/object-record/record-filter/utils/isRecordMatchingFilter'; + +import { CachedObjectRecordQueryVariables } from '@/apollo/types/CachedObjectRecordQueryVariables'; import { isDefined } from '~/utils/isDefined'; +import { parseApolloStoreFieldName } from '~/utils/parseApolloStoreFieldName'; /* TODO: for now new records are added to all cached record lists, no matter what the variables (filters, orderBy, etc.) are. @@ -19,11 +23,13 @@ export const triggerCreateRecordsOptimisticEffect = ({ objectMetadataItem, recordsToCreate, objectMetadataItems, + shouldMatchRootQueryFilter, }: { cache: ApolloCache<unknown>; objectMetadataItem: ObjectMetadataItem; recordsToCreate: RecordGqlNode[]; objectMetadataItems: ObjectMetadataItem[]; + shouldMatchRootQueryFilter?: boolean; }) => { recordsToCreate.forEach((record) => triggerUpdateRelationsOptimisticEffect({ @@ -39,12 +45,7 @@ export const triggerCreateRecordsOptimisticEffect = ({ fields: { [objectMetadataItem.namePlural]: ( rootQueryCachedResponse, - { - DELETE: _DELETE, - readField, - storeFieldName: _storeFieldName, - toReference, - }, + { DELETE: _DELETE, readField, storeFieldName, toReference }, ) => { const shouldSkip = !isObjectRecordConnectionWithRefs( objectMetadataItem.nameSingular, @@ -55,6 +56,13 @@ export const triggerCreateRecordsOptimisticEffect = ({ return rootQueryCachedResponse; } + const { fieldVariables: rootQueryVariables } = + parseApolloStoreFieldName<CachedObjectRecordQueryVariables>( + storeFieldName, + ); + + const rootQueryFilter = rootQueryVariables?.filter; + const rootQueryCachedObjectRecordConnection = rootQueryCachedResponse; const rootQueryCachedRecordEdges = readField<RecordGqlRefEdge[]>( @@ -74,6 +82,22 @@ export const triggerCreateRecordsOptimisticEffect = ({ const hasAddedRecords = recordsToCreate .map((recordToCreate) => { if (isNonEmptyString(recordToCreate.id)) { + if ( + isDefined(rootQueryFilter) && + shouldMatchRootQueryFilter === true + ) { + const recordToCreateMatchesThisRootQueryFilter = + isRecordMatchingFilter({ + record: recordToCreate, + filter: rootQueryFilter, + objectMetadataItem, + }); + + if (!recordToCreateMatchesThisRootQueryFilter) { + return false; + } + } + const recordToCreateReference = toReference(recordToCreate); if (!recordToCreateReference) { diff --git a/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerUpdateRelationsOptimisticEffect.ts b/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerUpdateRelationsOptimisticEffect.ts index 0b12d7524bd0..16d103532b77 100644 --- a/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerUpdateRelationsOptimisticEffect.ts +++ b/packages/twenty-front/src/modules/apollo/optimistic-effect/utils/triggerUpdateRelationsOptimisticEffect.ts @@ -70,6 +70,7 @@ export const triggerUpdateRelationsOptimisticEffect = ({ isDeeplyEqual( currentFieldValueOnSourceRecord, updatedFieldValueOnSourceRecord, + { strict: true }, ) ) { return; diff --git a/packages/twenty-front/src/modules/auth/hooks/__test__/useOnboardingStatus.test.ts b/packages/twenty-front/src/modules/auth/hooks/__test__/useOnboardingStatus.test.ts deleted file mode 100644 index 13d8e23fdaff..000000000000 --- a/packages/twenty-front/src/modules/auth/hooks/__test__/useOnboardingStatus.test.ts +++ /dev/null @@ -1,373 +0,0 @@ -import { act } from 'react-dom/test-utils'; -import { renderHook } from '@testing-library/react'; -import { RecoilRoot, useSetRecoilState } from 'recoil'; - -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; -import { CurrentUser, currentUserState } from '@/auth/states/currentUserState'; -import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; -import { - CurrentWorkspace, - currentWorkspaceState, -} from '@/auth/states/currentWorkspaceState'; -import { isVerifyPendingState } from '@/auth/states/isVerifyPendingState'; -import { tokenPairState } from '@/auth/states/tokenPairState'; -import { billingState } from '@/client-config/states/billingState'; -import { OnboardingStep } from '~/generated/graphql'; - -const tokenPair = { - accessToken: { token: 'accessToken', expiresAt: 'expiresAt' }, - refreshToken: { token: 'refreshToken', expiresAt: 'expiresAt' }, -}; -const billing = { - billingUrl: 'testing.com', - isBillingEnabled: true, -}; -const currentUser = { - id: '1', - email: 'test@test', - supportUserHash: '1', - canImpersonate: false, - onboardingStep: null, -} as CurrentUser; -const currentWorkspace = { - activationStatus: 'active', - id: '1', - allowImpersonation: true, - currentBillingSubscription: { - status: 'trialing', - }, -} as CurrentWorkspace; -const currentWorkspaceMember = { - id: '1', - locale: '', - name: { - firstName: '', - lastName: '', - }, -}; - -const renderHooks = () => { - const { result } = renderHook( - () => { - const onboardingStatus = useOnboardingStatus(); - const setBilling = useSetRecoilState(billingState); - const setCurrentWorkspace = useSetRecoilState(currentWorkspaceState); - const setCurrentWorkspaceMember = useSetRecoilState( - currentWorkspaceMemberState, - ); - const setCurrentUser = useSetRecoilState(currentUserState); - const setTokenPair = useSetRecoilState(tokenPairState); - const setVerifyPending = useSetRecoilState(isVerifyPendingState); - - return { - onboardingStatus, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - setTokenPair, - setVerifyPending, - }; - }, - { - wrapper: RecoilRoot, - }, - ); - return { result }; -}; - -describe('useOnboardingStatus', () => { - it('should return "ongoing_user_creation" when user is not logged in', async () => { - const { result } = renderHooks(); - - expect(result.current.onboardingStatus).toBe('ongoing_user_creation'); - }); - - it('should return "incomplete"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'incomplete', - }); - setCurrentWorkspaceMember(currentWorkspaceMember); - }); - - expect(result.current.onboardingStatus).toBe('incomplete'); - }); - - it('should return "canceled"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'canceled', - }); - setCurrentWorkspaceMember({ - ...currentWorkspaceMember, - name: { - firstName: 'John', - lastName: 'Doe', - }, - }); - }); - - expect(result.current.onboardingStatus).toBe('canceled'); - }); - - it('should return "ongoing_workspace_activation"', async () => { - const { result } = renderHooks(); - const { setTokenPair, setBilling, setCurrentUser, setCurrentWorkspace } = - result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - activationStatus: 'inactive', - subscriptionStatus: 'active', - }); - }); - - expect(result.current.onboardingStatus).toBe( - 'ongoing_workspace_activation', - ); - }); - - it('should return "ongoing_profile_creation"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'active', - }); - setCurrentWorkspaceMember(currentWorkspaceMember); - }); - - expect(result.current.onboardingStatus).toBe('ongoing_profile_creation'); - }); - - it('should return "ongoing_sync_email"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser({ - ...currentUser, - onboardingStep: OnboardingStep.SyncEmail, - }); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'active', - }); - setCurrentWorkspaceMember({ - ...currentWorkspaceMember, - name: { - firstName: 'John', - lastName: 'Doe', - }, - }); - }); - - expect(result.current.onboardingStatus).toBe('ongoing_sync_email'); - }); - - it('should return "ongoing_invite_team"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser({ - ...currentUser, - onboardingStep: OnboardingStep.InviteTeam, - }); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'active', - }); - setCurrentWorkspaceMember({ - ...currentWorkspaceMember, - name: { - firstName: 'John', - lastName: 'Doe', - }, - }); - }); - - expect(result.current.onboardingStatus).toBe('ongoing_invite_team'); - }); - - it('should return "completed"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'active', - }); - setCurrentWorkspaceMember({ - ...currentWorkspaceMember, - name: { - firstName: 'John', - lastName: 'Doe', - }, - }); - }); - - expect(result.current.onboardingStatus).toBe('completed'); - }); - - it('should return "past_due"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'past_due', - }); - setCurrentWorkspaceMember({ - ...currentWorkspaceMember, - name: { - firstName: 'John', - lastName: 'Doe', - }, - }); - }); - - expect(result.current.onboardingStatus).toBe('past_due'); - }); - - it('should return "unpaid"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'unpaid', - }); - setCurrentWorkspaceMember({ - ...currentWorkspaceMember, - name: { - firstName: 'John', - lastName: 'Doe', - }, - }); - }); - - expect(result.current.onboardingStatus).toBe('unpaid'); - }); - - it('should return "completed_without_subscription"', async () => { - const { result } = renderHooks(); - const { - setTokenPair, - setBilling, - setCurrentUser, - setCurrentWorkspace, - setCurrentWorkspaceMember, - } = result.current; - - act(() => { - setTokenPair(tokenPair); - setBilling(billing); - setCurrentUser(currentUser); - setCurrentWorkspace({ - ...currentWorkspace, - subscriptionStatus: 'trialing', - currentBillingSubscription: null, - }); - setCurrentWorkspaceMember({ - ...currentWorkspaceMember, - name: { - firstName: 'John', - lastName: 'Doe', - }, - }); - }); - - expect(result.current.onboardingStatus).toBe( - 'completed_without_subscription', - ); - }); -}); diff --git a/packages/twenty-front/src/modules/auth/hooks/useOnboardingStatus.ts b/packages/twenty-front/src/modules/auth/hooks/useOnboardingStatus.ts deleted file mode 100644 index b638b19da85f..000000000000 --- a/packages/twenty-front/src/modules/auth/hooks/useOnboardingStatus.ts +++ /dev/null @@ -1,28 +0,0 @@ -import { useRecoilValue } from 'recoil'; - -import { currentUserState } from '@/auth/states/currentUserState'; -import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; -import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; -import { billingState } from '@/client-config/states/billingState'; - -import { useIsLogged } from '../hooks/useIsLogged'; -import { - getOnboardingStatus, - OnboardingStatus, -} from '../utils/getOnboardingStatus'; - -export const useOnboardingStatus = (): OnboardingStatus | undefined => { - const billing = useRecoilValue(billingState); - const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); - const currentWorkspace = useRecoilValue(currentWorkspaceState); - const currentUser = useRecoilValue(currentUserState); - const isLoggedIn = useIsLogged(); - - return getOnboardingStatus({ - isLoggedIn, - currentWorkspaceMember, - currentWorkspace, - currentUser, - isBillingEnabled: billing?.isBillingEnabled || false, - }); -}; diff --git a/packages/twenty-front/src/modules/auth/sign-in-up/components/FooterNote.tsx b/packages/twenty-front/src/modules/auth/sign-in-up/components/FooterNote.tsx index c9d5066d9585..3bad98dcb083 100644 --- a/packages/twenty-front/src/modules/auth/sign-in-up/components/FooterNote.tsx +++ b/packages/twenty-front/src/modules/auth/sign-in-up/components/FooterNote.tsx @@ -1,15 +1,23 @@ -import React from 'react'; import styled from '@emotion/styled'; +import React from 'react'; type FooterNoteProps = { children: React.ReactNode }; const StyledContainer = styled.div` align-items: center; color: ${({ theme }) => theme.font.color.tertiary}; - display: flex; font-size: ${({ theme }) => theme.font.size.sm}; max-width: 280px; text-align: center; + + & > a { + color: ${({ theme }) => theme.font.color.tertiary}; + text-decoration: none; + + &:hover { + text-decoration: underline; + } + } `; export const FooterNote = ({ children }: FooterNoteProps) => ( diff --git a/packages/twenty-front/src/modules/auth/sign-in-up/components/SignInUpForm.tsx b/packages/twenty-front/src/modules/auth/sign-in-up/components/SignInUpForm.tsx index 86565460419a..1ebbfd9bf112 100644 --- a/packages/twenty-front/src/modules/auth/sign-in-up/components/SignInUpForm.tsx +++ b/packages/twenty-front/src/modules/auth/sign-in-up/components/SignInUpForm.tsx @@ -1,8 +1,8 @@ -import { useMemo, useState } from 'react'; -import { Controller } from 'react-hook-form'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { motion } from 'framer-motion'; +import { useMemo, useState } from 'react'; +import { Controller } from 'react-hook-form'; import { useRecoilState, useRecoilValue } from 'recoil'; import { Key } from 'ts-key-enum'; import { IconGoogle, IconMicrosoft } from 'twenty-ui'; @@ -28,23 +28,23 @@ import { ActionLink } from '@/ui/navigation/link/components/ActionLink'; import { isDefined } from '~/utils/isDefined'; const StyledContentContainer = styled.div` - margin-bottom: ${({ theme }) => theme.spacing(8)}; - margin-top: ${({ theme }) => theme.spacing(4)}; + margin-bottom: ${({ theme }) => theme.spacing(8)}; + margin-top: ${({ theme }) => theme.spacing(4)}; `; const StyledForm = styled.form` - align-items: center; - display: flex; - flex-direction: column; - width: 100%; + align-items: center; + display: flex; + flex-direction: column; + width: 100%; `; const StyledFullWidthMotionDiv = styled(motion.div)` - width: 100%; + width: 100%; `; const StyledInputContainer = styled.div` - margin-bottom: ${({ theme }) => theme.spacing(3)}; + margin-bottom: ${({ theme }) => theme.spacing(3)}; `; export const SignInUpForm = () => { @@ -168,9 +168,9 @@ export const SignInUpForm = () => { name="email" control={form.control} render={({ - field: { onChange, onBlur, value }, - fieldState: { error }, - }) => ( + field: { onChange, onBlur, value }, + fieldState: { error }, + }) => ( <StyledInputContainer> <TextInput autoFocus @@ -207,9 +207,9 @@ export const SignInUpForm = () => { name="password" control={form.control} render={({ - field: { onChange, onBlur, value }, - fieldState: { error }, - }) => ( + field: { onChange, onBlur, value }, + fieldState: { error }, + }) => ( <StyledInputContainer> <TextInput autoFocus @@ -258,7 +258,23 @@ export const SignInUpForm = () => { )} {signInUpStep === SignInUpStep.Init && ( <FooterNote> - By using Funnelmink, you agree to the Terms of Service and Privacy Policy. + By using Funnelmink, you agree to the{' '} + <a + href="https://funnelmink.com/legal/terms" + target="_blank" + rel="noopener noreferrer" + > + Terms of Service + </a>{' '} + and{' '} + <a + href="https://funnelmink.com/legal/privacy" + target="_blank" + rel="noopener noreferrer" + > + Privacy Policy + </a> + . </FooterNote> )} </> diff --git a/packages/twenty-front/src/modules/auth/sign-in-up/hooks/useWorkspaceFromInviteHash.ts b/packages/twenty-front/src/modules/auth/sign-in-up/hooks/useWorkspaceFromInviteHash.ts index ee6234feb343..feee086ef7c0 100644 --- a/packages/twenty-front/src/modules/auth/sign-in-up/hooks/useWorkspaceFromInviteHash.ts +++ b/packages/twenty-front/src/modules/auth/sign-in-up/hooks/useWorkspaceFromInviteHash.ts @@ -22,8 +22,8 @@ export const useWorkspaceFromInviteHash = () => { const { data: workspaceFromInviteHash, loading } = useGetWorkspaceFromInviteHashQuery({ variables: { inviteHash: workspaceInviteHash || '' }, - onError: () => { - enqueueSnackBar('workspace does not exist', { + onError: (error) => { + enqueueSnackBar(error.message, { variant: SnackBarVariant.Error, }); navigate(AppPath.Index); diff --git a/packages/twenty-front/src/modules/auth/states/currentUserState.ts b/packages/twenty-front/src/modules/auth/states/currentUserState.ts index 07efc7dfb1ec..2aab02507ff6 100644 --- a/packages/twenty-front/src/modules/auth/states/currentUserState.ts +++ b/packages/twenty-front/src/modules/auth/states/currentUserState.ts @@ -4,7 +4,7 @@ import { User } from '~/generated/graphql'; export type CurrentUser = Pick< User, - 'id' | 'email' | 'supportUserHash' | 'canImpersonate' | 'onboardingStep' + 'id' | 'email' | 'supportUserHash' | 'canImpersonate' | 'onboardingStatus' >; export const currentUserState = createState<CurrentUser | null>({ diff --git a/packages/twenty-front/src/modules/auth/states/currentWorkspaceState.ts b/packages/twenty-front/src/modules/auth/states/currentWorkspaceState.ts index c9187c3ea694..a66d7bd1cb06 100644 --- a/packages/twenty-front/src/modules/auth/states/currentWorkspaceState.ts +++ b/packages/twenty-front/src/modules/auth/states/currentWorkspaceState.ts @@ -10,9 +10,9 @@ export type CurrentWorkspace = Pick< | 'displayName' | 'allowImpersonation' | 'featureFlags' - | 'subscriptionStatus' | 'activationStatus' | 'currentBillingSubscription' + | 'workspaceMembersCount' | 'currentCacheVersion' >; diff --git a/packages/twenty-front/src/modules/auth/utils/__test__/getOnboardingStatus.test.ts b/packages/twenty-front/src/modules/auth/utils/__test__/getOnboardingStatus.test.ts deleted file mode 100644 index 813e9b38299b..000000000000 --- a/packages/twenty-front/src/modules/auth/utils/__test__/getOnboardingStatus.test.ts +++ /dev/null @@ -1,174 +0,0 @@ -import { CurrentUser } from '@/auth/states/currentUserState'; -import { CurrentWorkspace } from '@/auth/states/currentWorkspaceState'; -import { WorkspaceMember } from '@/workspace-member/types/WorkspaceMember'; -import { OnboardingStep } from '~/generated/graphql'; - -import { getOnboardingStatus } from '../getOnboardingStatus'; - -describe('getOnboardingStatus', () => { - it('should return the correct status', () => { - const ongoingUserCreation = getOnboardingStatus({ - isLoggedIn: false, - currentWorkspaceMember: null, - currentWorkspace: null, - currentUser: null, - isBillingEnabled: false, - }); - - const ongoingWorkspaceActivation = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: null, - currentWorkspace: { - id: '1', - activationStatus: 'inactive', - } as CurrentWorkspace, - currentUser: { - onboardingStep: null, - } as CurrentUser, - isBillingEnabled: false, - }); - - const ongoingProfileCreation = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: { - id: '1', - name: {}, - } as WorkspaceMember, - currentWorkspace: { - id: '1', - activationStatus: 'active', - } as CurrentWorkspace, - currentUser: { - onboardingStep: null, - } as CurrentUser, - isBillingEnabled: false, - }); - - const ongoingSyncEmail = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: { - id: '1', - name: { - firstName: 'John', - lastName: 'Doe', - }, - } as WorkspaceMember, - currentWorkspace: { - id: '1', - activationStatus: 'active', - } as CurrentWorkspace, - currentUser: { - onboardingStep: OnboardingStep.SyncEmail, - } as CurrentUser, - isBillingEnabled: false, - }); - - const ongoingInviteTeam = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: { - id: '1', - name: { - firstName: 'John', - lastName: 'Doe', - }, - } as WorkspaceMember, - currentWorkspace: { - id: '1', - activationStatus: 'active', - } as CurrentWorkspace, - currentUser: { - onboardingStep: OnboardingStep.InviteTeam, - } as CurrentUser, - isBillingEnabled: false, - }); - - const completed = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: { - id: '1', - name: { - firstName: 'John', - lastName: 'Doe', - }, - } as WorkspaceMember, - currentWorkspace: { - id: '1', - activationStatus: 'active', - } as CurrentWorkspace, - currentUser: { - onboardingStep: null, - } as CurrentUser, - isBillingEnabled: false, - }); - - const incomplete = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: { - id: '1', - name: { - firstName: 'John', - lastName: 'Doe', - }, - } as WorkspaceMember, - currentWorkspace: { - id: '1', - activationStatus: 'active', - subscriptionStatus: 'incomplete', - } as CurrentWorkspace, - currentUser: { - onboardingStep: null, - } as CurrentUser, - isBillingEnabled: true, - }); - - const incompleteButBillingDisabled = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: { - id: '1', - name: { - firstName: 'John', - lastName: 'Doe', - }, - } as WorkspaceMember, - currentWorkspace: { - id: '1', - activationStatus: 'active', - subscriptionStatus: 'incomplete', - } as CurrentWorkspace, - currentUser: { - onboardingStep: null, - } as CurrentUser, - isBillingEnabled: false, - }); - - const canceled = getOnboardingStatus({ - isLoggedIn: true, - currentWorkspaceMember: { - id: '1', - name: { - firstName: 'John', - lastName: 'Doe', - }, - } as WorkspaceMember, - currentWorkspace: { - id: '1', - activationStatus: 'active', - subscriptionStatus: 'canceled', - } as CurrentWorkspace, - currentUser: { - onboardingStep: null, - } as CurrentUser, - isBillingEnabled: true, - }); - - expect(ongoingUserCreation).toBe('ongoing_user_creation'); - expect(ongoingWorkspaceActivation).toBe('ongoing_workspace_activation'); - expect(ongoingProfileCreation).toBe('ongoing_profile_creation'); - expect(ongoingSyncEmail).toBe('ongoing_sync_email'); - expect(ongoingInviteTeam).toBe('ongoing_invite_team'); - expect(completed).toBe('completed'); - expect(incomplete).toBe('incomplete'); - expect(canceled).toBe('canceled'); - expect(incompleteButBillingDisabled).toBe('completed'); - }); -}); diff --git a/packages/twenty-front/src/modules/auth/utils/getOnboardingStatus.ts b/packages/twenty-front/src/modules/auth/utils/getOnboardingStatus.ts deleted file mode 100644 index 0dc254705228..000000000000 --- a/packages/twenty-front/src/modules/auth/utils/getOnboardingStatus.ts +++ /dev/null @@ -1,89 +0,0 @@ -import { CurrentUser } from '@/auth/states/currentUserState'; -import { CurrentWorkspace } from '@/auth/states/currentWorkspaceState'; -import { WorkspaceMember } from '@/workspace-member/types/WorkspaceMember'; -import { OnboardingStep } from '~/generated/graphql'; - -export enum OnboardingStatus { - Incomplete = 'incomplete', - Canceled = 'canceled', - Unpaid = 'unpaid', - PastDue = 'past_due', - OngoingUserCreation = 'ongoing_user_creation', - OngoingWorkspaceActivation = 'ongoing_workspace_activation', - OngoingProfileCreation = 'ongoing_profile_creation', - OngoingSyncEmail = 'ongoing_sync_email', - OngoingInviteTeam = 'ongoing_invite_team', - Completed = 'completed', - CompletedWithoutSubscription = 'completed_without_subscription', -} - -export const getOnboardingStatus = ({ - isLoggedIn, - currentWorkspaceMember, - currentWorkspace, - currentUser, - isBillingEnabled, -}: { - isLoggedIn: boolean; - currentWorkspaceMember: Omit< - WorkspaceMember, - 'createdAt' | 'updatedAt' | 'userId' | 'userEmail' | '__typename' - > | null; - currentWorkspace: CurrentWorkspace | null; - currentUser: CurrentUser | null; - isBillingEnabled: boolean; -}) => { - if (!isLoggedIn) { - return OnboardingStatus.OngoingUserCreation; - } - - // After SignInUp, the user should have a current workspace assigned. - // If not, it indicates that the data is still being requested. - if (!currentWorkspace || !currentUser) { - return undefined; - } - - if ( - isBillingEnabled && - currentWorkspace.subscriptionStatus === 'incomplete' - ) { - return OnboardingStatus.Incomplete; - } - - if (currentWorkspace.activationStatus !== 'active') { - return OnboardingStatus.OngoingWorkspaceActivation; - } - - if ( - !currentWorkspaceMember?.name.firstName || - !currentWorkspaceMember?.name.lastName - ) { - return OnboardingStatus.OngoingProfileCreation; - } - - if (currentUser.onboardingStep === OnboardingStep.SyncEmail) { - return OnboardingStatus.OngoingSyncEmail; - } - - if (currentUser.onboardingStep === OnboardingStep.InviteTeam) { - return OnboardingStatus.OngoingInviteTeam; - } - - if (isBillingEnabled && currentWorkspace.subscriptionStatus === 'canceled') { - return OnboardingStatus.Canceled; - } - - if (isBillingEnabled && currentWorkspace.subscriptionStatus === 'past_due') { - return OnboardingStatus.PastDue; - } - - if (isBillingEnabled && currentWorkspace.subscriptionStatus === 'unpaid') { - return OnboardingStatus.Unpaid; - } - - if (isBillingEnabled && !currentWorkspace.currentBillingSubscription) { - return OnboardingStatus.CompletedWithoutSubscription; - } - - return OnboardingStatus.Completed; -}; diff --git a/packages/twenty-front/src/modules/billing/graphql/checkoutSession.ts b/packages/twenty-front/src/modules/billing/graphql/checkoutSession.ts index bff619a82f85..53821543cef2 100644 --- a/packages/twenty-front/src/modules/billing/graphql/checkoutSession.ts +++ b/packages/twenty-front/src/modules/billing/graphql/checkoutSession.ts @@ -2,7 +2,7 @@ import { gql } from '@apollo/client'; export const CHECKOUT_SESSION = gql` mutation CheckoutSession( - $recurringInterval: String! + $recurringInterval: SubscriptionInterval! $successUrlPath: String ) { checkoutSession( diff --git a/packages/twenty-front/src/modules/captcha/components/CaptchaProviderScriptLoaderEffect.tsx b/packages/twenty-front/src/modules/captcha/components/CaptchaProviderScriptLoaderEffect.tsx index 59dedab8fee3..aae90964f1dd 100644 --- a/packages/twenty-front/src/modules/captcha/components/CaptchaProviderScriptLoaderEffect.tsx +++ b/packages/twenty-front/src/modules/captcha/components/CaptchaProviderScriptLoaderEffect.tsx @@ -32,7 +32,7 @@ export const CaptchaProviderScriptLoaderEffect = () => { scriptElement = document.createElement('script'); scriptElement.src = scriptUrl; scriptElement.onload = () => { - if (captchaProvider.provider === CaptchaDriverType.GoogleRecatpcha) { + if (captchaProvider.provider === CaptchaDriverType.GoogleRecaptcha) { window.grecaptcha?.ready(() => { setIsCaptchaScriptLoaded(true); }); diff --git a/packages/twenty-front/src/modules/captcha/hooks/useRequestFreshCaptchaToken.ts b/packages/twenty-front/src/modules/captcha/hooks/useRequestFreshCaptchaToken.ts index f2e2821fc27f..63e6ee725480 100644 --- a/packages/twenty-front/src/modules/captcha/hooks/useRequestFreshCaptchaToken.ts +++ b/packages/twenty-front/src/modules/captcha/hooks/useRequestFreshCaptchaToken.ts @@ -35,7 +35,7 @@ export const useRequestFreshCaptchaToken = () => { let captchaWidget: any; switch (captchaProvider.provider) { - case CaptchaDriverType.GoogleRecatpcha: + case CaptchaDriverType.GoogleRecaptcha: window.grecaptcha .execute(captchaProvider.siteKey, { action: 'submit', diff --git a/packages/twenty-front/src/modules/captcha/utils/__tests__/getCaptchaUrlByProvider.test.ts b/packages/twenty-front/src/modules/captcha/utils/__tests__/getCaptchaUrlByProvider.test.ts new file mode 100644 index 000000000000..40cba1f43081 --- /dev/null +++ b/packages/twenty-front/src/modules/captcha/utils/__tests__/getCaptchaUrlByProvider.test.ts @@ -0,0 +1,38 @@ +import { expect } from '@storybook/test'; + +import { CaptchaDriverType } from '~/generated/graphql'; + +import { getCaptchaUrlByProvider } from '../getCaptchaUrlByProvider'; + +describe('getCaptchaUrlByProvider', () => { + it('handles GoogleRecaptcha', async () => { + const captchaUrl = getCaptchaUrlByProvider( + CaptchaDriverType.GoogleRecaptcha, + 'siteKey', + ); + + expect(captchaUrl).toEqual( + 'https://www.google.com/recaptcha/api.js?render=siteKey', + ); + + expect(() => + getCaptchaUrlByProvider(CaptchaDriverType.GoogleRecaptcha, ''), + ).toThrow( + 'SiteKey must be provided while generating url for GoogleRecaptcha provider', + ); + }); + + it('handles Turnstile', async () => { + const captchaUrl = getCaptchaUrlByProvider(CaptchaDriverType.Turnstile, ''); + + expect(captchaUrl).toEqual( + 'https://challenges.cloudflare.com/turnstile/v0/api.js', + ); + }); + + it('handles unknown provider', async () => { + expect(() => + getCaptchaUrlByProvider('Unknown' as CaptchaDriverType, ''), + ).toThrow('Unknown captcha provider'); + }); +}); diff --git a/packages/twenty-front/src/modules/captcha/utils/getCaptchaUrlByProvider.ts b/packages/twenty-front/src/modules/captcha/utils/getCaptchaUrlByProvider.ts index 5c0abe89e8a1..b3c1874ea897 100644 --- a/packages/twenty-front/src/modules/captcha/utils/getCaptchaUrlByProvider.ts +++ b/packages/twenty-front/src/modules/captcha/utils/getCaptchaUrlByProvider.ts @@ -1,16 +1,22 @@ -import { CaptchaDriverType } from '~/generated-metadata/graphql'; +import { isNonEmptyString } from '@sniptt/guards'; -export const getCaptchaUrlByProvider = (name: string, siteKey: string) => { - if (!name) { - return ''; - } +import { CaptchaDriverType } from '~/generated-metadata/graphql'; +export const getCaptchaUrlByProvider = ( + name: CaptchaDriverType, + siteKey: string, +) => { switch (name) { - case CaptchaDriverType.GoogleRecatpcha: + case CaptchaDriverType.GoogleRecaptcha: + if (!isNonEmptyString(siteKey)) { + throw new Error( + 'SiteKey must be provided while generating url for GoogleRecaptcha provider', + ); + } return `https://www.google.com/recaptcha/api.js?render=${siteKey}`; case CaptchaDriverType.Turnstile: return 'https://challenges.cloudflare.com/turnstile/v0/api.js'; default: - return ''; + throw new Error('Unknown captcha provider'); } }; diff --git a/packages/twenty-front/src/modules/client-config/components/ClientConfigProviderEffect.tsx b/packages/twenty-front/src/modules/client-config/components/ClientConfigProviderEffect.tsx index af45dc0fd34b..8bec4cc7dbd6 100644 --- a/packages/twenty-front/src/modules/client-config/components/ClientConfigProviderEffect.tsx +++ b/packages/twenty-front/src/modules/client-config/components/ClientConfigProviderEffect.tsx @@ -1,6 +1,7 @@ import { useEffect } from 'react'; import { useRecoilState, useSetRecoilState } from 'recoil'; +import { apiConfigState } from '@/client-config/states/apiConfigState'; import { authProvidersState } from '@/client-config/states/authProvidersState'; import { billingState } from '@/client-config/states/billingState'; import { captchaProviderState } from '@/client-config/states/captchaProviderState'; @@ -35,6 +36,8 @@ export const ClientConfigProviderEffect = () => { const setChromeExtensionId = useSetRecoilState(chromeExtensionIdState); + const setApiConfig = useSetRecoilState(apiConfigState); + const { data, loading } = useGetClientConfigQuery({ skip: isClientConfigLoaded, }); @@ -68,6 +71,7 @@ export const ClientConfigProviderEffect = () => { }); setChromeExtensionId(data?.clientConfig?.chromeExtensionId); + setApiConfig(data?.clientConfig?.api); } }, [ data, @@ -83,6 +87,7 @@ export const ClientConfigProviderEffect = () => { setIsClientConfigLoaded, setCaptchaProvider, setChromeExtensionId, + setApiConfig, ]); return <></>; diff --git a/packages/twenty-front/src/modules/client-config/graphql/queries/getClientConfig.ts b/packages/twenty-front/src/modules/client-config/graphql/queries/getClientConfig.ts index 6b98067167c4..3143bbc5f65a 100644 --- a/packages/twenty-front/src/modules/client-config/graphql/queries/getClientConfig.ts +++ b/packages/twenty-front/src/modules/client-config/graphql/queries/getClientConfig.ts @@ -32,6 +32,9 @@ export const GET_CLIENT_CONFIG = gql` provider siteKey } + api { + mutationMaximumAffectedRecords + } chromeExtensionId } } diff --git a/packages/twenty-front/src/modules/client-config/states/apiConfigState.ts b/packages/twenty-front/src/modules/client-config/states/apiConfigState.ts new file mode 100644 index 000000000000..9a01493e4f84 --- /dev/null +++ b/packages/twenty-front/src/modules/client-config/states/apiConfigState.ts @@ -0,0 +1,8 @@ +import { createState } from 'twenty-ui'; + +import { ApiConfig } from '~/generated/graphql'; + +export const apiConfigState = createState<ApiConfig | null>({ + key: 'apiConfigState', + defaultValue: null, +}); diff --git a/packages/twenty-front/src/modules/command-menu/components/CommandMenu.tsx b/packages/twenty-front/src/modules/command-menu/components/CommandMenu.tsx index efd72c3b9373..cb941c4e1861 100644 --- a/packages/twenty-front/src/modules/command-menu/components/CommandMenu.tsx +++ b/packages/twenty-front/src/modules/command-menu/components/CommandMenu.tsx @@ -1,10 +1,12 @@ -import { useMemo, useRef } from 'react'; import styled from '@emotion/styled'; import { isNonEmptyString } from '@sniptt/guards'; -import { useRecoilState, useRecoilValue } from 'recoil'; +import { useMemo, useRef } from 'react'; +import { useRecoilState, useRecoilValue, useSetRecoilState } from 'recoil'; import { Key } from 'ts-key-enum'; -import { Avatar, IconNotes } from 'twenty-ui'; +import { Avatar, IconNotes, IconSparkles } from 'twenty-ui'; +import { useOpenCopilotRightDrawer } from '@/activities/copilot/right-drawer/hooks/useOpenCopilotRightDrawer'; +import { copilotQueryState } from '@/activities/copilot/right-drawer/states/copilotQueryState'; import { useOpenActivityRightDrawer } from '@/activities/hooks/useOpenActivityRightDrawer'; import { Activity } from '@/activities/types/Activity'; import { commandMenuSearchState } from '@/command-menu/states/commandMenuSearchState'; @@ -21,6 +23,7 @@ import { AppHotkeyScope } from '@/ui/utilities/hotkey/types/AppHotkeyScope'; import { useListenClickOutside } from '@/ui/utilities/pointer-event/hooks/useListenClickOutside'; import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; import { ScrollWrapper } from '@/ui/utilities/scroll/components/ScrollWrapper'; +import { useIsFeatureEnabled } from '@/workspace/hooks/useIsFeatureEnabled'; import { getLogoUrlFromDomainName } from '~/utils'; import { generateILikeFiltersForCompositeFields } from '~/utils/array/generateILikeFiltersForCompositeFields'; import { isDefined } from '~/utils/isDefined'; @@ -248,8 +251,27 @@ export const CommandMenu = () => { callback: closeCommandMenu, }); - const selectableItemIds = matchingCreateCommand + const isCopilotEnabled = useIsFeatureEnabled('IS_COPILOT_ENABLED'); + const setCopilotQuery = useSetRecoilState(copilotQueryState); + const openCopilotRightDrawer = useOpenCopilotRightDrawer(); + + const copilotCommand: Command = { + id: 'copilot', + to: '', // TODO + Icon: IconSparkles, + label: 'Open Copilot', + type: CommandType.Navigate, + onCommandClick: () => { + setCopilotQuery(commandMenuSearch); + openCopilotRightDrawer(); + }, + }; + + const copilotCommands: Command[] = isCopilotEnabled ? [copilotCommand] : []; + + const selectableItemIds = copilotCommands .map((cmd) => cmd.id) + .concat(matchingCreateCommand.map((cmd) => cmd.id)) .concat(matchingNavigateCommand.map((cmd) => cmd.id)) .concat(people.map((person) => person.id)) .concat(companies.map((company) => company.id)) @@ -275,6 +297,7 @@ export const CommandMenu = () => { hotkeyScope={AppHotkeyScope.CommandMenu} onEnter={(itemId) => { const command = [ + ...copilotCommands, ...commandMenuCommands, ...otherCommands, ].find((cmd) => cmd.id === itemId); @@ -292,6 +315,22 @@ export const CommandMenu = () => { !activities.length && ( <StyledEmpty>No results found</StyledEmpty> )} + {isCopilotEnabled && ( + <CommandGroup heading="Copilot"> + <SelectableItem itemId={copilotCommand.id}> + <CommandMenuItem + id={copilotCommand.id} + Icon={copilotCommand.Icon} + label={`${copilotCommand.label} ${ + commandMenuSearch.length > 2 + ? `"${commandMenuSearch}"` + : '' + }`} + onClick={copilotCommand.onCommandClick} + /> + </SelectableItem> + </CommandGroup> + )} <CommandGroup heading="Create"> {matchingCreateCommand.map((cmd) => ( <SelectableItem itemId={cmd.id} key={cmd.id}> @@ -338,7 +377,7 @@ export const CommandMenu = () => { <Avatar type="rounded" avatarUrl={null} - entityId={person.id} + placeholderColorSeed={person.id} placeholder={ person.name.firstName + ' ' + @@ -360,7 +399,7 @@ export const CommandMenu = () => { to={`object/company/${company.id}`} Icon={() => ( <Avatar - entityId={company.id} + placeholderColorSeed={company.id} placeholder={company.name} avatarUrl={getLogoUrlFromDomainName( company.domainName, diff --git a/packages/twenty-front/src/modules/databases/utils/__tests__/getForeignDataWrapperType.test.ts b/packages/twenty-front/src/modules/databases/utils/__tests__/getForeignDataWrapperType.test.ts new file mode 100644 index 000000000000..3db265dd9444 --- /dev/null +++ b/packages/twenty-front/src/modules/databases/utils/__tests__/getForeignDataWrapperType.test.ts @@ -0,0 +1,15 @@ +import { getForeignDataWrapperType } from '../getForeignDataWrapperType'; + +describe('getForeignDataWrapperType', () => { + it('should handle postgres', () => { + expect(getForeignDataWrapperType('postgresql')).toBe('postgres_fdw'); + }); + + it('should handle stripe', () => { + expect(getForeignDataWrapperType('stripe')).toBe('stripe_fdw'); + }); + + it('should return null for unknown', () => { + expect(getForeignDataWrapperType('unknown')).toBeNull(); + }); +}); diff --git a/packages/twenty-front/src/modules/error-handler/components/GenericErrorFallback.tsx b/packages/twenty-front/src/modules/error-handler/components/GenericErrorFallback.tsx index 87f84036911b..2109891aca14 100644 --- a/packages/twenty-front/src/modules/error-handler/components/GenericErrorFallback.tsx +++ b/packages/twenty-front/src/modules/error-handler/components/GenericErrorFallback.tsx @@ -1,4 +1,6 @@ +import { useEffect, useState } from 'react'; import { FallbackProps } from 'react-error-boundary'; +import { useLocation } from 'react-router-dom'; import { ThemeProvider, useTheme } from '@emotion/react'; import isEmpty from 'lodash.isempty'; import { IconRefresh, THEME_LIGHT } from 'twenty-ui'; @@ -11,6 +13,7 @@ import { AnimatedPlaceholderEmptyTextContainer, AnimatedPlaceholderEmptyTitle, } from '@/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled'; +import { isDeeplyEqual } from '~/utils/isDeeplyEqual'; type GenericErrorFallbackProps = FallbackProps; @@ -18,7 +21,18 @@ export const GenericErrorFallback = ({ error, resetErrorBoundary, }: GenericErrorFallbackProps) => { + const location = useLocation(); + + const [previousLocation] = useState(location); + + useEffect(() => { + if (!isDeeplyEqual(previousLocation, location)) { + resetErrorBoundary(); + } + }, [previousLocation, location, resetErrorBoundary]); + const theme = useTheme(); + return ( <ThemeProvider theme={isEmpty(theme) ? THEME_LIGHT : theme}> <AnimatedPlaceholderEmptyContainer> diff --git a/packages/twenty-front/src/modules/favorites/components/Favorites.tsx b/packages/twenty-front/src/modules/favorites/components/Favorites.tsx index c3bfeeb6aefe..9835a860b41f 100644 --- a/packages/twenty-front/src/modules/favorites/components/Favorites.tsx +++ b/packages/twenty-front/src/modules/favorites/components/Favorites.tsx @@ -1,4 +1,5 @@ import styled from '@emotion/styled'; +import { useRecoilValue } from 'recoil'; import { Avatar } from 'twenty-ui'; import { FavoritesSkeletonLoader } from '@/favorites/components/FavoritesSkeletonLoader'; @@ -8,6 +9,7 @@ import { DraggableList } from '@/ui/layout/draggable-list/components/DraggableLi import { NavigationDrawerItem } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerItem'; import { NavigationDrawerSection } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerSection'; import { NavigationDrawerSectionTitle } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitle'; +import { useNavigationSection } from '@/ui/navigation/navigation-drawer/hooks/useNavigationSection'; import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; import { useFavorites } from '../hooks/useFavorites'; @@ -36,6 +38,10 @@ export const Favorites = () => { const { favorites, handleReorderFavorite } = useFavorites(); const loading = useIsPrefetchLoading(); + const { toggleNavigationSection, isNavigationSectionOpenState } = + useNavigationSection('Favorites'); + const isNavigationSectionOpen = useRecoilValue(isNavigationSectionOpenState); + if (loading) { return <FavoritesSkeletonLoader />; } @@ -44,48 +50,53 @@ export const Favorites = () => { return ( <StyledContainer> - <NavigationDrawerSectionTitle label="Favorites" /> - <DraggableList - onDragEnd={handleReorderFavorite} - draggableItems={ - <> - {favorites.map((favorite, index) => { - const { - id, - labelIdentifier, - avatarUrl, - avatarType, - link, - recordId, - } = favorite; - - return ( - <DraggableItem - key={id} - draggableId={id} - index={index} - itemComponent={ - <StyledNavigationDrawerItem - key={id} - label={labelIdentifier} - Icon={() => ( - <StyledAvatar - entityId={recordId} - avatarUrl={getImageAbsoluteURIOrBase64(avatarUrl)} - type={avatarType} - placeholder={labelIdentifier} - className="fav-avatar" - /> - )} - to={link} - /> - } - /> - ); - })} - </> - } + <NavigationDrawerSectionTitle + label="Favorites" + onClick={() => toggleNavigationSection()} /> + {isNavigationSectionOpen && ( + <DraggableList + onDragEnd={handleReorderFavorite} + draggableItems={ + <> + {favorites.map((favorite, index) => { + const { + id, + labelIdentifier, + avatarUrl, + avatarType, + link, + recordId, + } = favorite; + + return ( + <DraggableItem + key={id} + draggableId={id} + index={index} + itemComponent={ + <StyledNavigationDrawerItem + key={id} + label={labelIdentifier} + Icon={() => ( + <StyledAvatar + placeholderColorSeed={recordId} + avatarUrl={getImageAbsoluteURIOrBase64(avatarUrl)} + type={avatarType} + placeholder={labelIdentifier} + className="fav-avatar" + /> + )} + to={link} + /> + } + /> + ); + })} + </> + } + /> + )} </StyledContainer> ); }; diff --git a/packages/twenty-front/src/modules/favorites/hooks/__mocks__/useFavorites.ts b/packages/twenty-front/src/modules/favorites/hooks/__mocks__/useFavorites.ts index 73b4c855a610..c00b9f5da9ba 100644 --- a/packages/twenty-front/src/modules/favorites/hooks/__mocks__/useFavorites.ts +++ b/packages/twenty-front/src/modules/favorites/hooks/__mocks__/useFavorites.ts @@ -145,7 +145,16 @@ export const mocks = [ currencyCode } createdAt - address + address { + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng + } updatedAt name accountOwnerId @@ -262,7 +271,16 @@ export const mocks = [ currencyCode } createdAt - address + address { + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng + } updatedAt name accountOwnerId diff --git a/packages/twenty-front/src/modules/navigation/components/MainNavigationDrawerItems.tsx b/packages/twenty-front/src/modules/navigation/components/MainNavigationDrawerItems.tsx index a002010e8b9f..43d7b3da3eb9 100644 --- a/packages/twenty-front/src/modules/navigation/components/MainNavigationDrawerItems.tsx +++ b/packages/twenty-front/src/modules/navigation/components/MainNavigationDrawerItems.tsx @@ -9,7 +9,6 @@ import { Favorites } from '@/favorites/components/Favorites'; import { ObjectMetadataNavItems } from '@/object-metadata/components/ObjectMetadataNavItems'; import { NavigationDrawerItem } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerItem'; import { NavigationDrawerSection } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerSection'; -import { NavigationDrawerSectionTitle } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitle'; import { navigationMemorizedUrlState } from '@/ui/navigation/states/navigationMemorizedUrlState'; import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; @@ -56,10 +55,8 @@ export const MainNavigationDrawerItems = () => { <Favorites /> - <NavigationDrawerSection> - <NavigationDrawerSectionTitle label="Workspace" /> - <ObjectMetadataNavItems /> - </NavigationDrawerSection> + <ObjectMetadataNavItems isRemote={false} /> + <ObjectMetadataNavItems isRemote={true} /> </> ); }; diff --git a/packages/twenty-front/src/modules/navigation/utils/__tests__/indexAppPath.test.ts b/packages/twenty-front/src/modules/navigation/utils/__tests__/indexAppPath.test.ts new file mode 100644 index 000000000000..8920f6fa7445 --- /dev/null +++ b/packages/twenty-front/src/modules/navigation/utils/__tests__/indexAppPath.test.ts @@ -0,0 +1,10 @@ +import { AppPath } from '@/types/AppPath'; + +import indexAppPath from '../indexAppPath'; + +describe('getIndexAppPath', () => { + it('returns the index app path', () => { + const { getIndexAppPath } = indexAppPath; + expect(getIndexAppPath()).toEqual(AppPath.Index); + }); +}); diff --git a/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataItemsProvider.tsx b/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataItemsProvider.tsx index 8071b00b65a8..7f18bb2c254f 100644 --- a/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataItemsProvider.tsx +++ b/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataItemsProvider.tsx @@ -1,11 +1,10 @@ -import React, { useMemo } from 'react'; +import React from 'react'; import { useRecoilValue } from 'recoil'; import { ObjectMetadataItemsLoadEffect } from '@/object-metadata/components/ObjectMetadataItemsLoadEffect'; -import { PreComputedChipGeneratorsContext } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; +import { PreComputedChipGeneratorsProvider } from '@/object-metadata/components/PreComputedChipGeneratorsProvider'; import { objectMetadataItemsState } from '@/object-metadata/states/objectMetadataItemsState'; import { RelationPickerScope } from '@/object-record/relation-picker/scopes/RelationPickerScope'; -import { getRecordChipGeneratorPerObjectPerField } from '@/object-record/utils/getRecordChipGeneratorPerObjectPerField'; import { UserOrMetadataLoader } from '~/loading/components/UserOrMetadataLoader'; export const ObjectMetadataItemsProvider = ({ @@ -15,23 +14,15 @@ export const ObjectMetadataItemsProvider = ({ const shouldDisplayChildren = objectMetadataItems.length > 0; - const chipGeneratorPerObjectPerField = useMemo(() => { - return getRecordChipGeneratorPerObjectPerField(objectMetadataItems); - }, [objectMetadataItems]); - return ( <> <ObjectMetadataItemsLoadEffect /> {shouldDisplayChildren ? ( - <PreComputedChipGeneratorsContext.Provider - value={{ - chipGeneratorPerObjectPerField, - }} - > + <PreComputedChipGeneratorsProvider> <RelationPickerScope relationPickerScopeId="relation-picker"> {children} </RelationPickerScope> - </PreComputedChipGeneratorsContext.Provider> + </PreComputedChipGeneratorsProvider> ) : ( <UserOrMetadataLoader /> )} diff --git a/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataNavItems.tsx b/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataNavItems.tsx index cb28ad6d31d2..10a89da89cfb 100644 --- a/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataNavItems.tsx +++ b/packages/twenty-front/src/modules/object-metadata/components/ObjectMetadataNavItems.tsx @@ -1,17 +1,32 @@ import { useLocation } from 'react-router-dom'; +import { useRecoilValue } from 'recoil'; import { useIcons } from 'twenty-ui'; -import { ObjectMetadataNavItemsSkeletonLoader } from '@/object-metadata/components/ObjectMetadataNavItemsSkeletonLoader'; +import { + ObjectMetadataNavItemsSkeletonLoader, +} from '@/object-metadata/components/ObjectMetadataNavItemsSkeletonLoader'; import { useFilteredObjectMetadataItems } from '@/object-metadata/hooks/useFilteredObjectMetadataItems'; import { useIsPrefetchLoading } from '@/prefetch/hooks/useIsPrefetchLoading'; import { usePrefetchedData } from '@/prefetch/hooks/usePrefetchedData'; import { PrefetchKey } from '@/prefetch/types/PrefetchKey'; import { NavigationDrawerItem } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerItem'; +import { NavigationDrawerSection } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerSection'; +import { + NavigationDrawerSectionTitle, +} from '@/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitle'; +import { useNavigationSection } from '@/ui/navigation/navigation-drawer/hooks/useNavigationSection'; import { View } from '@/views/types/View'; import { getObjectMetadataItemViews } from '@/views/utils/getObjectMetadataItemViews'; -export const ObjectMetadataNavItems = () => { +export const ObjectMetadataNavItems = ({ isRemote }: { isRemote: boolean }) => { + const { toggleNavigationSection, isNavigationSectionOpenState } = + useNavigationSection('Objects' + (isRemote ? 'Remote' : 'Workspace')); + const isNavigationSectionOpen = useRecoilValue(isNavigationSectionOpenState); + const { activeObjectMetadataItems } = useFilteredObjectMetadataItems(); + const filteredActiveObjectMetadataItems = activeObjectMetadataItems.filter( + (item) => (isRemote ? item.isRemote : !item.isRemote), + ); const { getIcon } = useIcons(); const currentPath = useLocation().pathname; @@ -23,55 +38,41 @@ export const ObjectMetadataNavItems = () => { } return ( - <> - {[ - ...activeObjectMetadataItems - .filter((item) => - ['person', 'company', 'opportunity'].includes(item.nameSingular), - ) - .sort((objectMetadataItemA, objectMetadataItemB) => { - const order = ['person', 'company', 'opportunity']; - const indexA = order.indexOf(objectMetadataItemA.nameSingular); - const indexB = order.indexOf(objectMetadataItemB.nameSingular); - if (indexA === -1 || indexB === -1) { - return objectMetadataItemA.nameSingular.localeCompare( - objectMetadataItemB.nameSingular, + filteredActiveObjectMetadataItems.length > 0 && ( + <NavigationDrawerSection> + <NavigationDrawerSectionTitle + label={isRemote ? 'Remote' : 'Workspace'} + onClick={() => toggleNavigationSection()} + /> + + {isNavigationSectionOpen && + filteredActiveObjectMetadataItems + .sort((objectMetadataItemA, objectMetadataItemB) => + objectMetadataItemA.labelPlural.localeCompare(objectMetadataItemB.labelPlural)) + .map((objectMetadataItem) => { + const objectMetadataViews = getObjectMetadataItemViews( + objectMetadataItem.id, + views, ); - } - return indexA - indexB; - }), - ...activeObjectMetadataItems - .filter( - (item) => - !['person', 'company', 'opportunity'].includes(item.nameSingular), - ) - .sort((objectMetadataItemA, objectMetadataItemB) => { - return new Date(objectMetadataItemA.createdAt) < - new Date(objectMetadataItemB.createdAt) - ? 1 - : -1; - }), - ].map((objectMetadataItem) => { - const objectMetadataViews = getObjectMetadataItemViews( - objectMetadataItem.id, - views, - ); - const viewId = objectMetadataViews[0]?.id; + const viewId = objectMetadataViews[0]?.id; - const navigationPath = `/objects/${objectMetadataItem.namePlural}${ - viewId ? `?view=${viewId}` : '' - }`; + const navigationPath = `/objects/${objectMetadataItem.namePlural}${ + viewId ? `?view=${viewId}` : '' + }`; - return ( - <NavigationDrawerItem - key={objectMetadataItem.id} - label={objectMetadataItem.labelPlural} - to={navigationPath} - active={currentPath === `/objects/${objectMetadataItem.namePlural}`} - Icon={getIcon(objectMetadataItem.icon)} - /> - ); - })} - </> + return ( + <NavigationDrawerItem + key={objectMetadataItem.id} + label={objectMetadataItem.labelPlural} + to={navigationPath} + active={ + currentPath === `/objects/${objectMetadataItem.namePlural}` + } + Icon={getIcon(objectMetadataItem.icon)} + /> + ); + })} + </NavigationDrawerSection> + ) ); }; diff --git a/packages/twenty-front/src/modules/object-metadata/components/PreComputedChipGeneratorsProvider.tsx b/packages/twenty-front/src/modules/object-metadata/components/PreComputedChipGeneratorsProvider.tsx new file mode 100644 index 000000000000..dcf9bc5cc73d --- /dev/null +++ b/packages/twenty-front/src/modules/object-metadata/components/PreComputedChipGeneratorsProvider.tsx @@ -0,0 +1,30 @@ +import React, { useMemo } from 'react'; +import { useRecoilValue } from 'recoil'; + +import { PreComputedChipGeneratorsContext } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; +import { objectMetadataItemsState } from '@/object-metadata/states/objectMetadataItemsState'; +import { getRecordChipGenerators } from '@/object-record/utils/getRecordChipGenerators'; + +export const PreComputedChipGeneratorsProvider = ({ + children, +}: React.PropsWithChildren) => { + const objectMetadataItems = useRecoilValue(objectMetadataItemsState); + + const { chipGeneratorPerObjectPerField, identifierChipGeneratorPerObject } = + useMemo(() => { + return getRecordChipGenerators(objectMetadataItems); + }, [objectMetadataItems]); + + return ( + <> + <PreComputedChipGeneratorsContext.Provider + value={{ + chipGeneratorPerObjectPerField, + identifierChipGeneratorPerObject, + }} + > + {children} + </PreComputedChipGeneratorsContext.Provider> + </> + ); +}; diff --git a/packages/twenty-front/src/modules/object-metadata/context/PreComputedChipGeneratorsContext.ts b/packages/twenty-front/src/modules/object-metadata/context/PreComputedChipGeneratorsContext.ts index fcb5b7d46b68..ed7b734bcc98 100644 --- a/packages/twenty-front/src/modules/object-metadata/context/PreComputedChipGeneratorsContext.ts +++ b/packages/twenty-front/src/modules/object-metadata/context/PreComputedChipGeneratorsContext.ts @@ -3,13 +3,19 @@ import { createContext } from 'react'; import { RecordChipData } from '@/object-record/record-field/types/RecordChipData'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; -export type ChipGeneratorPerObjectPerField = Record< +export type ChipGeneratorPerObjectNameSingularPerFieldName = Record< string, Record<string, (record: ObjectRecord) => RecordChipData> >; +export type IdentifierChipGeneratorPerObject = Record< + string, + (record: ObjectRecord) => RecordChipData +>; + export type PreComputedChipGeneratorsContextProps = { - chipGeneratorPerObjectPerField: ChipGeneratorPerObjectPerField; + chipGeneratorPerObjectPerField: ChipGeneratorPerObjectNameSingularPerFieldName; + identifierChipGeneratorPerObject: IdentifierChipGeneratorPerObject; }; export const PreComputedChipGeneratorsContext = diff --git a/packages/twenty-front/src/modules/object-metadata/hooks/__mocks__/useCreateOneObjectMetadataItem.ts b/packages/twenty-front/src/modules/object-metadata/hooks/__mocks__/useCreateOneObjectMetadataItem.ts index f142bffbde25..ce4d5cfbe41b 100644 --- a/packages/twenty-front/src/modules/object-metadata/hooks/__mocks__/useCreateOneObjectMetadataItem.ts +++ b/packages/twenty-front/src/modules/object-metadata/hooks/__mocks__/useCreateOneObjectMetadataItem.ts @@ -24,7 +24,7 @@ export const query = gql` export const findManyViewsQuery = gql` query FindManyViews( $filter: ViewFilterInput - $orderBy: ViewOrderByInput + $orderBy: [ViewOrderByInput] $lastCursor: String $limit: Int ) { diff --git a/packages/twenty-front/src/modules/object-metadata/hooks/__tests__/useGetObjectOrderByField.test.tsx b/packages/twenty-front/src/modules/object-metadata/hooks/__tests__/useGetObjectOrderByField.test.tsx index 646806c9762a..21ec56ad60b4 100644 --- a/packages/twenty-front/src/modules/object-metadata/hooks/__tests__/useGetObjectOrderByField.test.tsx +++ b/packages/twenty-front/src/modules/object-metadata/hooks/__tests__/useGetObjectOrderByField.test.tsx @@ -26,8 +26,8 @@ describe('useGetObjectOrderByField', () => { }, ); - expect(result.current).toEqual({ - name: { firstName: 'AscNullsLast', lastName: 'AscNullsLast' }, - }); + expect(result.current).toEqual([ + { name: { firstName: 'AscNullsLast', lastName: 'AscNullsLast' } }, + ]); }); }); diff --git a/packages/twenty-front/src/modules/object-metadata/utils/__tests__/getObjectMetadataItemBySingularName.test.ts b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/getObjectMetadataItemBySingularName.test.ts new file mode 100644 index 000000000000..07e88a26f066 --- /dev/null +++ b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/getObjectMetadataItemBySingularName.test.ts @@ -0,0 +1,17 @@ +import { getObjectMetadataItemByNameSingular } from '@/object-metadata/utils/getObjectMetadataItemBySingularName'; +import { getObjectMetadataItemsMock } from '@/object-metadata/utils/getObjectMetadataItemsMock'; + +const mockObjectMetadataItems = getObjectMetadataItemsMock(); + +describe('getObjectMetadataItemBySingularName', () => { + it('should work as expected', () => { + const firstObjectMetadataItem = mockObjectMetadataItems[0]; + + const foundObjectMetadataItem = getObjectMetadataItemByNameSingular({ + objectMetadataItems: mockObjectMetadataItems, + objectNameSingular: firstObjectMetadataItem.nameSingular, + }); + + expect(foundObjectMetadataItem.id).toEqual(firstObjectMetadataItem.id); + }); +}); diff --git a/packages/twenty-front/src/modules/object-metadata/utils/__tests__/getObjectOrderByField.test.ts b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/getObjectOrderByField.test.ts index 7cecf5668b25..baa61c853751 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/__tests__/getObjectOrderByField.test.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/getObjectOrderByField.test.ts @@ -9,8 +9,8 @@ describe('getObjectOrderByField', () => { (item) => item.nameSingular === 'person', )!; const res = getOrderByFieldForObjectMetadataItem(objectMetadataItem); - expect(res).toEqual({ - name: { firstName: 'AscNullsLast', lastName: 'AscNullsLast' }, - }); + expect(res).toEqual([ + { name: { firstName: 'AscNullsLast', lastName: 'AscNullsLast' } }, + ]); }); }); diff --git a/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapFieldMetadataToGraphQLQuery.test.tsx b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapFieldMetadataToGraphQLQuery.test.tsx index 116874b23290..fc841bd27068 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapFieldMetadataToGraphQLQuery.test.tsx +++ b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapFieldMetadataToGraphQLQuery.test.tsx @@ -66,6 +66,16 @@ annualRecurringRevenue } createdAt address +{ + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng +} updatedAt name accountOwnerId @@ -86,7 +96,7 @@ idealCustomerProfile domainName: true, annualRecurringRevenue: true, createdAt: true, - address: true, + address: { addressStreet1: true }, updatedAt: true, name: true, accountOwnerId: true, @@ -129,6 +139,16 @@ annualRecurringRevenue } createdAt address +{ + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng +} updatedAt people { diff --git a/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapObjectMetadataToGraphQLQuery.test.tsx b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapObjectMetadataToGraphQLQuery.test.tsx index a5261cbfdc01..53442862491e 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapObjectMetadataToGraphQLQuery.test.tsx +++ b/packages/twenty-front/src/modules/object-metadata/utils/__tests__/mapObjectMetadataToGraphQLQuery.test.tsx @@ -65,6 +65,16 @@ annualRecurringRevenue } createdAt address +{ + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng +} updatedAt name accountOwnerId diff --git a/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemAsFieldDefinition.ts b/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemAsFieldDefinition.ts index 0c1cc9a3dae1..a579d5ce4de3 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemAsFieldDefinition.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemAsFieldDefinition.ts @@ -40,6 +40,7 @@ export const formatFieldMetadataItemAsFieldDefinition = ({ targetFieldMetadataName: field.relationDefinition?.targetFieldMetadata?.name ?? '', options: field.options, + isNullable: field.isNullable, }; return { diff --git a/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsFilterDefinitions.ts b/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsFilterDefinitions.ts index c5df10c85c8f..318073af17bd 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsFilterDefinitions.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsFilterDefinitions.ts @@ -1,6 +1,8 @@ import { FilterDefinition } from '@/object-record/object-filter-dropdown/types/FilterDefinition'; -import { FieldMetadataType } from '~/generated-metadata/graphql'; -import { isDefined } from '~/utils/isDefined'; +import { + FieldMetadataType, + RelationDefinitionType, +} from '~/generated-metadata/graphql'; import { ObjectMetadataItem } from '../types/ObjectMetadataItem'; @@ -10,6 +12,15 @@ export const formatFieldMetadataItemsAsFilterDefinitions = ({ fields: Array<ObjectMetadataItem['fields'][0]>; }): FilterDefinition[] => fields.reduce((acc, field) => { + if ( + field.type === FieldMetadataType.Relation && + field.relationDefinition?.direction !== + RelationDefinitionType.ManyToOne && + field.relationDefinition?.direction !== RelationDefinitionType.OneToOne + ) { + return acc; + } + if ( ![ FieldMetadataType.DateTime, @@ -22,24 +33,12 @@ export const formatFieldMetadataItemsAsFilterDefinitions = ({ FieldMetadataType.Address, FieldMetadataType.Relation, FieldMetadataType.Select, - FieldMetadataType.MultiSelect, FieldMetadataType.Currency, ].includes(field.type) ) { return acc; } - // Todo: remove once Rating fieldtype is implemented - if (field.name === 'probability') { - return acc; - } - - if (field.type === FieldMetadataType.Relation) { - if (isDefined(field.fromRelationMetadata)) { - return acc; - } - } - return [...acc, formatFieldMetadataItemAsFilterDefinition({ field })]; }, [] as FilterDefinition[]); @@ -52,9 +51,9 @@ export const formatFieldMetadataItemAsFilterDefinition = ({ label: field.label, iconName: field.icon ?? 'Icon123', relationObjectMetadataNamePlural: - field.toRelationMetadata?.fromObjectMetadata.namePlural, + field.relationDefinition?.targetObjectMetadata.namePlural, relationObjectMetadataNameSingular: - field.toRelationMetadata?.fromObjectMetadata.nameSingular, + field.relationDefinition?.targetObjectMetadata.nameSingular, type: getFilterTypeFromFieldType(field.type), }); diff --git a/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsSortDefinitions.ts b/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsSortDefinitions.ts index 56b84aec08ac..b62f7fb35ba6 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsSortDefinitions.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/formatFieldMetadataItemsAsSortDefinitions.ts @@ -18,6 +18,8 @@ export const formatFieldMetadataItemsAsSortDefinitions = ({ FieldMetadataType.Boolean, FieldMetadataType.Select, FieldMetadataType.Phone, + FieldMetadataType.Email, + FieldMetadataType.FullName, ].includes(field.type) ) { return acc; diff --git a/packages/twenty-front/src/modules/object-metadata/utils/getLabelIdentifierFieldValue.ts b/packages/twenty-front/src/modules/object-metadata/utils/getLabelIdentifierFieldValue.ts index 7960047804ba..001cf4ecb06b 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/getLabelIdentifierFieldValue.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/getLabelIdentifierFieldValue.ts @@ -8,7 +8,7 @@ export const getLabelIdentifierFieldValue = ( record: ObjectRecord, labelIdentifierFieldMetadataItem: FieldMetadataItem | undefined, objectNameSingular: string, -) => { +): string => { if ( objectNameSingular === CoreObjectNameSingular.WorkspaceMember || labelIdentifierFieldMetadataItem?.type === FieldMetadataType.FullName @@ -17,7 +17,7 @@ export const getLabelIdentifierFieldValue = ( } if (isDefined(labelIdentifierFieldMetadataItem?.name)) { - return record[labelIdentifierFieldMetadataItem.name] as string | number; + return String(record[labelIdentifierFieldMetadataItem.name]); } return ''; diff --git a/packages/twenty-front/src/modules/object-metadata/utils/getLinkToShowPage.ts b/packages/twenty-front/src/modules/object-metadata/utils/getLinkToShowPage.ts index c5b747a1ebbb..4a6510dea2c1 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/getLinkToShowPage.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/getLinkToShowPage.ts @@ -4,7 +4,7 @@ import { ObjectRecord } from '@/object-record/types/ObjectRecord'; export const getLinkToShowPage = ( objectNameSingular: string, - record: ObjectRecord, + record: Pick<ObjectRecord, 'id'>, ) => { const basePathToShowPage = getBasePathToShowPage({ objectNameSingular, diff --git a/packages/twenty-front/src/modules/object-metadata/utils/getObjectMetadataItemsMock.ts b/packages/twenty-front/src/modules/object-metadata/utils/getObjectMetadataItemsMock.ts index c50755a51f92..7ae1d86e7993 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/getObjectMetadataItemsMock.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/getObjectMetadataItemsMock.ts @@ -466,23 +466,6 @@ export const getObjectMetadataItemsMock = () => { fromRelationMetadata: null, toRelationMetadata: null, }, - { - __typename: 'field', - id: '20202020-3b9c-4e58-a3d2-c617d3b596b1', - type: 'TEXT', - name: 'probability', - label: 'Probability', - description: 'Opportunity probability', - icon: 'IconProgressCheck', - isCustom: false, - isActive: true, - isSystem: false, - isNullable: true, - createdAt: '2023-11-30T11:13:15.308Z', - updatedAt: '2023-11-30T11:13:15.308Z', - fromRelationMetadata: null, - toRelationMetadata: null, - }, { __typename: 'fieldEdge', node: { @@ -3039,7 +3022,7 @@ export const getObjectMetadataItemsMock = () => { { __typename: 'field', id: '20202020-ad10-4117-a039-3f04b7a5f939', - type: 'TEXT', + type: 'ADDRESS', name: 'address', label: 'Address', description: 'The company address', @@ -3155,6 +3138,14 @@ export const getObjectMetadataItemsMock = () => { }, toFieldMetadataId: '20202020-64e1-4080-b6ad-db03c3809885', }, + relationDefinition: { + targetObjectMetadata: { + nameSingular: 'person', + }, + targetFieldMetadata: { + name: 'company', + }, + }, toRelationMetadata: null, }, { diff --git a/packages/twenty-front/src/modules/object-metadata/utils/getObjectOrderByField.ts b/packages/twenty-front/src/modules/object-metadata/utils/getObjectOrderByField.ts index 97d31c9a6b55..4b3b080f848a 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/getObjectOrderByField.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/getObjectOrderByField.ts @@ -15,20 +15,26 @@ export const getOrderByFieldForObjectMetadataItem = ( if (isDefined(labelIdentifierFieldMetadata)) { switch (labelIdentifierFieldMetadata.type) { case FieldMetadataType.FullName: - return { - [labelIdentifierFieldMetadata.name]: { - firstName: orderBy ?? 'AscNullsLast', - lastName: orderBy ?? 'AscNullsLast', + return [ + { + [labelIdentifierFieldMetadata.name]: { + firstName: orderBy ?? 'AscNullsLast', + lastName: orderBy ?? 'AscNullsLast', + }, }, - }; + ]; default: - return { - [labelIdentifierFieldMetadata.name]: orderBy ?? 'AscNullsLast', - }; + return [ + { + [labelIdentifierFieldMetadata.name]: orderBy ?? 'AscNullsLast', + }, + ]; } } else { - return { - createdAt: orderBy ?? 'DescNullsLast', - }; + return [ + { + createdAt: orderBy ?? 'DescNullsLast', + }, + ]; } }; diff --git a/packages/twenty-front/src/modules/object-metadata/utils/getObjectSlug.ts b/packages/twenty-front/src/modules/object-metadata/utils/getObjectSlug.ts index 465af568df1c..fa2132f9c369 100644 --- a/packages/twenty-front/src/modules/object-metadata/utils/getObjectSlug.ts +++ b/packages/twenty-front/src/modules/object-metadata/utils/getObjectSlug.ts @@ -3,5 +3,5 @@ import toKebabCase from 'lodash.kebabcase'; import { ObjectMetadataItem } from '../types/ObjectMetadataItem'; export const getObjectSlug = ( - objectMetadataItem: Pick<ObjectMetadataItem, 'labelPlural'>, -) => toKebabCase(objectMetadataItem.labelPlural); + objectMetadataItem: Pick<ObjectMetadataItem, 'namePlural'>, +) => toKebabCase(objectMetadataItem.namePlural); diff --git a/packages/twenty-front/src/modules/object-record/cache/utils/__tests__/isObjectRecordConnection.test.ts b/packages/twenty-front/src/modules/object-record/cache/utils/__tests__/isObjectRecordConnection.test.ts new file mode 100644 index 000000000000..bbe8b38db7bb --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/cache/utils/__tests__/isObjectRecordConnection.test.ts @@ -0,0 +1,27 @@ +import { peopleQueryResult } from '~/testing/mock-data/people'; + +import { isObjectRecordConnection } from '@/object-record/cache/utils/isObjectRecordConnection'; + +describe('isObjectRecordConnection', () => { + it('should work with query result', () => { + const validQueryResult = peopleQueryResult.people; + + const isValidQueryResult = isObjectRecordConnection( + 'person', + validQueryResult, + ); + + expect(isValidQueryResult).toEqual(true); + }); + + it('should fail with invalid result', () => { + const invalidResult = { test: 123 }; + + const isValidQueryResult = isObjectRecordConnection( + 'person', + invalidResult, + ); + + expect(isValidQueryResult).toEqual(false); + }); +}); diff --git a/packages/twenty-front/src/modules/object-record/cache/utils/getRecordNodeFromRecord.ts b/packages/twenty-front/src/modules/object-record/cache/utils/getRecordNodeFromRecord.ts index f162131f526c..d7faf5a77f29 100644 --- a/packages/twenty-front/src/modules/object-record/cache/utils/getRecordNodeFromRecord.ts +++ b/packages/twenty-front/src/modules/object-record/cache/utils/getRecordNodeFromRecord.ts @@ -4,6 +4,7 @@ import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; import { getNodeTypename } from '@/object-record/cache/utils/getNodeTypename'; import { getObjectTypename } from '@/object-record/cache/utils/getObjectTypename'; import { getRecordConnectionFromRecords } from '@/object-record/cache/utils/getRecordConnectionFromRecords'; +import { getRefName } from '@/object-record/cache/utils/getRefName'; import { RecordGqlNode } from '@/object-record/graphql/types/RecordGqlNode'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; import { @@ -39,7 +40,7 @@ export const getRecordNodeFromRecord = <T extends ObjectRecord>({ if (!isRootLevel && computeReferences) { return { - __ref: `${nodeTypeName}:${record.id}`, + __ref: getRefName(objectMetadataItem.nameSingular, record.id), } as unknown as RecordGqlNode; // Fix typing: we want a Reference in computeReferences mode } diff --git a/packages/twenty-front/src/modules/object-record/cache/utils/getRefName.ts b/packages/twenty-front/src/modules/object-record/cache/utils/getRefName.ts new file mode 100644 index 000000000000..3a2e12a88976 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/cache/utils/getRefName.ts @@ -0,0 +1,7 @@ +import { getNodeTypename } from '@/object-record/cache/utils/getNodeTypename'; + +export const getRefName = (objectNameSingular: string, id: string) => { + const nodeTypeName = getNodeTypename(objectNameSingular); + + return `${nodeTypeName}:${id}`; +}; diff --git a/packages/twenty-front/src/modules/object-record/cache/utils/updateRecordFromCache.ts b/packages/twenty-front/src/modules/object-record/cache/utils/updateRecordFromCache.ts index 7ce423c09222..26df70cecd18 100644 --- a/packages/twenty-front/src/modules/object-record/cache/utils/updateRecordFromCache.ts +++ b/packages/twenty-front/src/modules/object-record/cache/utils/updateRecordFromCache.ts @@ -13,11 +13,13 @@ export const updateRecordFromCache = <T extends ObjectRecord>({ objectMetadataItems, objectMetadataItem, cache, + recordGqlFields = undefined, record, }: { objectMetadataItems: ObjectMetadataItem[]; objectMetadataItem: ObjectMetadataItem; cache: ApolloCache<object>; + recordGqlFields?: Record<string, any>; record: T; }) => { if (isUndefinedOrNull(objectMetadataItem)) { @@ -32,6 +34,7 @@ export const updateRecordFromCache = <T extends ObjectRecord>({ objectMetadataItems, objectMetadataItem, computeReferences: true, + recordGqlFields, }, )} `; diff --git a/packages/twenty-front/src/modules/object-record/components/RecordChip.tsx b/packages/twenty-front/src/modules/object-record/components/RecordChip.tsx index ad5cc2217b79..94674aa57b21 100644 --- a/packages/twenty-front/src/modules/object-record/components/RecordChip.tsx +++ b/packages/twenty-front/src/modules/object-record/components/RecordChip.tsx @@ -1,14 +1,17 @@ -import { EntityChip, EntityChipVariant } from 'twenty-ui'; +import { AvatarChip, AvatarChipVariant } from 'twenty-ui'; -import { useMapToObjectRecordIdentifier } from '@/object-metadata/hooks/useMapToObjectRecordIdentifier'; +import { getLinkToShowPage } from '@/object-metadata/utils/getLinkToShowPage'; +import { useRecordChipData } from '@/object-record/hooks/useRecordChipData'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; -import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; +import { isNonEmptyString } from '@sniptt/guards'; +import { MouseEvent } from 'react'; +import { useNavigate } from 'react-router-dom'; export type RecordChipProps = { objectNameSingular: string; record: ObjectRecord; className?: string; - variant?: EntityChipVariant; + variant?: AvatarChipVariant; }; export const RecordChip = ({ @@ -17,21 +20,29 @@ export const RecordChip = ({ className, variant, }: RecordChipProps) => { - const { mapToObjectRecordIdentifier } = useMapToObjectRecordIdentifier({ + const navigate = useNavigate(); + + const { recordChipData } = useRecordChipData({ objectNameSingular, + record, }); - const objectRecordIdentifier = mapToObjectRecordIdentifier(record); + const handleAvatarChipClick = (event: MouseEvent) => { + const linkToShowPage = getLinkToShowPage(objectNameSingular, record); + + if (isNonEmptyString(linkToShowPage)) { + event.stopPropagation(); + navigate(linkToShowPage); + } + }; return ( - <EntityChip - entityId={record.id} - name={objectRecordIdentifier.name} - avatarType={objectRecordIdentifier.avatarType} - avatarUrl={ - getImageAbsoluteURIOrBase64(objectRecordIdentifier.avatarUrl) || '' - } - linkToEntity={objectRecordIdentifier.linkToShowPage} + <AvatarChip + placeholderColorSeed={record.id} + name={recordChipData.name} + avatarType={recordChipData.avatarType} + avatarUrl={recordChipData.avatarUrl ?? ''} + onClick={handleAvatarChipClick} className={className} variant={variant} /> diff --git a/packages/twenty-front/src/modules/object-record/constants/DefaultMutationBatchSize.ts b/packages/twenty-front/src/modules/object-record/constants/DefaultMutationBatchSize.ts new file mode 100644 index 000000000000..6864c5e05e4b --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/constants/DefaultMutationBatchSize.ts @@ -0,0 +1 @@ +export const DEFAULT_MUTATION_BATCH_SIZE = 30; diff --git a/packages/twenty-front/src/modules/object-record/constants/DefaultQueryPageSize.ts b/packages/twenty-front/src/modules/object-record/constants/DefaultQueryPageSize.ts new file mode 100644 index 000000000000..cb8e9f295d4d --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/constants/DefaultQueryPageSize.ts @@ -0,0 +1 @@ +export const DEFAULT_QUERY_PAGE_SIZE = 30; diff --git a/packages/twenty-front/src/modules/object-record/constants/DeleteMaxCount.ts b/packages/twenty-front/src/modules/object-record/constants/DeleteMaxCount.ts new file mode 100644 index 000000000000..0f1b3a7c3bdf --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/constants/DeleteMaxCount.ts @@ -0,0 +1 @@ +export const DELETE_MAX_COUNT = 10000; diff --git a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlConnection.ts b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlConnection.ts index 543a55b2fc48..df64c8e35fe2 100644 --- a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlConnection.ts +++ b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlConnection.ts @@ -6,6 +6,7 @@ export type RecordGqlConnection = { __typename?: string; edges: RecordGqlEdge[]; pageInfo: { + __typename?: string; hasNextPage?: boolean; hasPreviousPage?: boolean; startCursor?: Nullable<string>; diff --git a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationFilter.ts b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationFilter.ts index 66acadc80bee..fd6de3f7090d 100644 --- a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationFilter.ts +++ b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationFilter.ts @@ -9,6 +9,11 @@ export type UUIDFilter = { is?: IsFilter; }; +export type RelationFilter = { + is?: IsFilter; + in?: UUIDFilterValue[]; +}; + export type BooleanFilter = { eq?: boolean; is?: IsFilter; diff --git a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationFindDuplicatesResults.ts b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationFindDuplicatesResults.ts new file mode 100644 index 000000000000..09134da76d41 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationFindDuplicatesResults.ts @@ -0,0 +1,5 @@ +import { RecordGqlConnection } from '@/object-record/graphql/types/RecordGqlConnection'; + +export type RecordGqlOperationFindDuplicatesResult = { + [objectNamePlural: string]: RecordGqlConnection[]; +}; diff --git a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationOrderBy.ts b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationOrderBy.ts index 67592770d176..b1180d6c6860 100644 --- a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationOrderBy.ts +++ b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationOrderBy.ts @@ -1,5 +1,5 @@ import { OrderBy } from '@/object-metadata/types/OrderBy'; -export type RecordGqlOperationOrderBy = { +export type RecordGqlOperationOrderBy = Array<{ [fieldName: string]: OrderBy | { [subFieldName: string]: OrderBy }; -}; +}>; diff --git a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationVariables.ts b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationVariables.ts index 9dc0fed4f010..3abc3358a654 100644 --- a/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationVariables.ts +++ b/packages/twenty-front/src/modules/object-record/graphql/types/RecordGqlOperationVariables.ts @@ -1,8 +1,14 @@ import { RecordGqlOperationFilter } from '@/object-record/graphql/types/RecordGqlOperationFilter'; import { RecordGqlOperationOrderBy } from '@/object-record/graphql/types/RecordGqlOperationOrderBy'; +import { QueryCursorDirection } from '@/object-record/utils/generateFindManyRecordsQuery'; export type RecordGqlOperationVariables = { filter?: RecordGqlOperationFilter; orderBy?: RecordGqlOperationOrderBy; limit?: number; + cursorFilter?: { + cursor: string; + cursorDirection: QueryCursorDirection; + limit: number; + }; }; diff --git a/packages/twenty-front/src/modules/object-record/graphql/utils/generateDepthOneRecordGqlFields.ts b/packages/twenty-front/src/modules/object-record/graphql/utils/generateDepthOneRecordGqlFields.ts index ce52cd786d3f..7870c00cc9fb 100644 --- a/packages/twenty-front/src/modules/object-record/graphql/utils/generateDepthOneRecordGqlFields.ts +++ b/packages/twenty-front/src/modules/object-record/graphql/utils/generateDepthOneRecordGqlFields.ts @@ -1,14 +1,31 @@ import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; +import { isDefined } from '~/utils/isDefined'; export const generateDepthOneRecordGqlFields = ({ objectMetadataItem, + record, }: { objectMetadataItem: ObjectMetadataItem; + record?: Record<string, any>; }) => { - return objectMetadataItem.fields.reduce((acc, field) => { - return { - ...acc, - [field.name]: true, - }; - }, {}); + const gqlFieldsFromObjectMetadataItem = objectMetadataItem.fields.reduce( + (acc, field) => { + return { + ...acc, + [field.name]: true, + }; + }, + {}, + ); + + if (isDefined(record)) { + return Object.keys(gqlFieldsFromObjectMetadataItem).reduce((acc, key) => { + return { + ...acc, + [key]: Object.keys(record).includes(key), + }; + }, gqlFieldsFromObjectMetadataItem); + } + + return gqlFieldsFromObjectMetadataItem; }; diff --git a/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useCreateManyRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useCreateManyRecords.ts index d0a867d70924..7a8a4bf3a78a 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useCreateManyRecords.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useCreateManyRecords.ts @@ -3,8 +3,8 @@ import { gql } from '@apollo/client'; import { Person } from '@/people/types/Person'; export const query = gql` - mutation CreatePeople($data: [PersonCreateInput!]!) { - createPeople(data: $data) { + mutation CreatePeople($data: [PersonCreateInput!]!, $upsert: Boolean) { + createPeople(data: $data, upsert: $upsert) { __typename xLink { label diff --git a/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFetchAllRecordIds.ts b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFetchAllRecordIds.ts new file mode 100644 index 000000000000..6309686fb9e2 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFetchAllRecordIds.ts @@ -0,0 +1,81 @@ +import { RecordGqlConnection } from '@/object-record/graphql/types/RecordGqlConnection'; +import { gql } from '@apollo/client'; + +import { peopleQueryResult } from '~/testing/mock-data/people'; + + +export const query = gql` + query FindManyPeople($filter: PersonFilterInput, $orderBy: [PersonOrderByInput], $lastCursor: String, $limit: Int) { + people(filter: $filter, orderBy: $orderBy, first: $limit, after: $lastCursor){ + edges { + node { + __typename + id + } + cursor + } + pageInfo { + hasNextPage + startCursor + endCursor + } + totalCount + } + } +`; + +export const mockPageSize = 2; + +export const peopleMockWithIdsOnly: RecordGqlConnection = { ...peopleQueryResult.people,edges: peopleQueryResult.people.edges.map((edge) => ({ ...edge, node: { __typename: 'Person', id: edge.node.id } })) }; + +export const firstRequestLastCursor = peopleMockWithIdsOnly.edges[mockPageSize].cursor; +export const secondRequestLastCursor = peopleMockWithIdsOnly.edges[mockPageSize * 2].cursor; +export const thirdRequestLastCursor = peopleMockWithIdsOnly.edges[mockPageSize * 3].cursor; + +export const variablesFirstRequest = { + filter: undefined, + limit: mockPageSize, + orderBy: undefined +}; + +export const variablesSecondRequest = { + filter: undefined, + limit: mockPageSize, + orderBy: undefined, + lastCursor: firstRequestLastCursor +}; + +export const variablesThirdRequest = { + filter: undefined, + limit: mockPageSize, + orderBy: undefined, + lastCursor: secondRequestLastCursor +} + +const paginateRequestResponse = (response: RecordGqlConnection, start: number, end: number, hasNextPage: boolean, totalCount: number) => { + return { + ...response, + edges: [ + ...response.edges.slice(start, end) + ], + pageInfo: { + ...response.pageInfo, + startCursor: response.edges[start].cursor, + endCursor: response.edges[end].cursor, + hasNextPage, + } satisfies RecordGqlConnection['pageInfo'], + totalCount, + } +} + +export const responseFirstRequest = { + people: paginateRequestResponse(peopleMockWithIdsOnly, 0, mockPageSize, true, 6), +}; + +export const responseSecondRequest = { + people: paginateRequestResponse(peopleMockWithIdsOnly, mockPageSize, mockPageSize * 2, true, 6), +}; + +export const responseThirdRequest = { + people: paginateRequestResponse(peopleMockWithIdsOnly, mockPageSize * 2, mockPageSize * 3, false, 6), +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindDuplicateRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindDuplicateRecords.ts index 6cc35d507098..fec1fb988a94 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindDuplicateRecords.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindDuplicateRecords.ts @@ -4,8 +4,8 @@ import { getPeopleMock } from '~/testing/mock-data/people'; const peopleMock = getPeopleMock(); export const query = gql` - query FindDuplicatePerson($id: ID!) { - personDuplicates(id: $id) { + query FindDuplicatePerson($ids: [ID!]!) { + personDuplicates(ids: $ids) { edges { node { __typename @@ -38,32 +38,32 @@ export const query = gql` startCursor endCursor } - totalCount } } `; export const variables = { - id: '6205681e-7c11-40b4-9e32-f523dbe54590', + ids: ['6205681e-7c11-40b4-9e32-f523dbe54590'], }; export const responseData = { - personDuplicates: { - edges: [ - { - node: { ...peopleMock[0], updatedAt: '' }, - cursor: 'cursor1', + personDuplicates: [ + { + edges: [ + { + node: { ...peopleMock[0], updatedAt: '' }, + cursor: 'cursor1', + }, + { + node: { ...peopleMock[1], updatedAt: '' }, + cursor: 'cursor2', + }, + ], + pageInfo: { + hasNextPage: false, + startCursor: 'cursor1', + endCursor: 'cursor2', }, - { - node: { ...peopleMock[1], updatedAt: '' }, - cursor: 'cursor2', - }, - ], - pageInfo: { - hasNextPage: false, - startCursor: 'cursor1', - endCursor: 'cursor2', }, - totalCount: 2, - }, + ], }; diff --git a/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindManyRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindManyRecords.ts index 6de4625bfa7c..22caeadc1cf5 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindManyRecords.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/__mocks__/useFindManyRecords.ts @@ -27,7 +27,6 @@ export const query = gql` updatedAt companyId stage - probability closeDate amount { amountMicros @@ -53,7 +52,6 @@ export const query = gql` updatedAt companyId stage - probability closeDate amount { amountMicros @@ -82,7 +80,16 @@ export const query = gql` currencyCode } createdAt - address + address { + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng + } updatedAt name accountOwnerId diff --git a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useCreateManyRecordsMutation.test.tsx b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useCreateManyRecordsMutation.test.tsx index 131707d0fbd0..9574e3b28b7d 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useCreateManyRecordsMutation.test.tsx +++ b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useCreateManyRecordsMutation.test.tsx @@ -5,8 +5,8 @@ import { RecoilRoot } from 'recoil'; import { useCreateManyRecordsMutation } from '@/object-record/hooks/useCreateManyRecordsMutation'; const expectedQueryTemplate = ` - mutation CreatePeople($data: [PersonCreateInput!]!) { - createPeople(data: $data) { + mutation CreatePeople($data: [PersonCreateInput!]!, $upsert: Boolean) { + createPeople(data: $data, upsert: $upsert) { __typename xLink { label diff --git a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useDeleteManyRecords.test.tsx b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useDeleteManyRecords.test.tsx index 548c1f0d6aa5..1c7dd33d33e6 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useDeleteManyRecords.test.tsx +++ b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useDeleteManyRecords.test.tsx @@ -1,6 +1,6 @@ -import { ReactNode } from 'react'; import { MockedProvider } from '@apollo/client/testing'; import { act, renderHook } from '@testing-library/react'; +import { ReactNode } from 'react'; import { RecoilRoot } from 'recoil'; import { @@ -23,7 +23,7 @@ const mocks = [ }, result: jest.fn(() => ({ data: { - deletePeople: responseData, + deletePeople: [responseData], }, })), }, @@ -49,7 +49,7 @@ describe('useDeleteManyRecords', () => { await act(async () => { const res = await result.current.deleteManyRecords(people); expect(res).toBeDefined(); - expect(res).toHaveProperty('id'); + expect(res[0]).toHaveProperty('id'); }); expect(mocks[0].result).toHaveBeenCalled(); diff --git a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFetchAllRecordIds.test.tsx b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFetchAllRecordIds.test.tsx new file mode 100644 index 000000000000..fc32df77ebb0 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFetchAllRecordIds.test.tsx @@ -0,0 +1,107 @@ +import { MockedProvider } from '@apollo/client/testing'; +import { act, renderHook } from '@testing-library/react'; +import { ReactNode, useEffect } from 'react'; +import { RecoilRoot, useRecoilState } from 'recoil'; + +import { objectMetadataItemsState } from '@/object-metadata/states/objectMetadataItemsState'; +import { getObjectMetadataItemsMock } from '@/object-metadata/utils/getObjectMetadataItemsMock'; +import { + mockPageSize, + peopleMockWithIdsOnly, + query, + responseFirstRequest, + responseSecondRequest, + responseThirdRequest, + variablesFirstRequest, + variablesSecondRequest, + variablesThirdRequest, +} from '@/object-record/hooks/__mocks__/useFetchAllRecordIds'; +import { useFetchAllRecordIds } from '@/object-record/hooks/useFetchAllRecordIds'; +import { SnackBarManagerScopeInternalContext } from '@/ui/feedback/snack-bar-manager/scopes/scope-internal-context/SnackBarManagerScopeInternalContext'; + +const mocks = [ + { + delay: 100, + request: { + query, + variables: variablesFirstRequest, + }, + result: jest.fn(() => ({ + data: responseFirstRequest, + })), + }, + { + delay: 100, + request: { + query, + variables: variablesSecondRequest, + }, + result: jest.fn(() => ({ + data: responseSecondRequest, + })), + }, + { + delay: 100, + request: { + query, + variables: variablesThirdRequest, + }, + result: jest.fn(() => ({ + data: responseThirdRequest, + })), + }, +]; + +describe('useFetchAllRecordIds', () => { + it('fetches all record ids with fetch more synchronous loop', async () => { + const Wrapper = ({ children }: { children: ReactNode }) => ( + <RecoilRoot> + <SnackBarManagerScopeInternalContext.Provider + value={{ + scopeId: 'snack-bar-manager', + }} + > + <MockedProvider mocks={mocks} addTypename={false}> + {children} + </MockedProvider> + </SnackBarManagerScopeInternalContext.Provider> + </RecoilRoot> + ); + + const { result } = renderHook( + () => { + const [, setObjectMetadataItems] = useRecoilState( + objectMetadataItemsState, + ); + + useEffect(() => { + setObjectMetadataItems(getObjectMetadataItemsMock()); + }, [setObjectMetadataItems]); + + return useFetchAllRecordIds({ + objectNameSingular: 'person', + pageSize: mockPageSize, + }); + }, + { + wrapper: Wrapper, + }, + ); + + const { fetchAllRecordIds } = result.current; + + let recordIds: string[] = []; + + await act(async () => { + recordIds = await fetchAllRecordIds(); + }); + + expect(mocks[0].result).toHaveBeenCalled(); + expect(mocks[1].result).toHaveBeenCalled(); + expect(mocks[2].result).toHaveBeenCalled(); + + expect(recordIds).toEqual( + peopleMockWithIdsOnly.edges.map((edge) => edge.node.id).slice(0, 6), + ); + }); +}); diff --git a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecords.test.tsx b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecords.test.tsx index 98d3cad285c0..e8616d1da1a7 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecords.test.tsx +++ b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecords.test.tsx @@ -42,7 +42,7 @@ describe('useFindDuplicateRecords', () => { const { result } = renderHook( () => useFindDuplicateRecords({ - objectRecordId, + objectRecordIds: [objectRecordId], objectNameSingular, }), { @@ -54,7 +54,7 @@ describe('useFindDuplicateRecords', () => { await waitFor(() => { expect(result.current.loading).toBe(false); - expect(result.current.records).toBeDefined(); + expect(result.current.results).toBeDefined(); }); expect(mocks[0].result).toHaveBeenCalled(); diff --git a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecordsQuery.test.tsx b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecordsQuery.test.tsx index 5650e59e13aa..a32f1a6f24d0 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecordsQuery.test.tsx +++ b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindDuplicateRecordsQuery.test.tsx @@ -5,8 +5,8 @@ import { RecoilRoot } from 'recoil'; import { useFindDuplicateRecordsQuery } from '@/object-record/hooks/useFindDuplicatesRecordsQuery'; const expectedQueryTemplate = ` - query FindDuplicatePerson($id: ID!) { - personDuplicates(id: $id) { + query FindDuplicatePerson($ids: [ID!]!) { + personDuplicates(ids: $ids) { edges { node { __typename @@ -39,7 +39,6 @@ const expectedQueryTemplate = ` startCursor endCursor } - totalCount } } `.replace(/\s/g, ''); diff --git a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecords.test.tsx b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecords.test.tsx index 57e18be23cba..6b7d71c18b5d 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecords.test.tsx +++ b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecords.test.tsx @@ -1,6 +1,6 @@ -import { ReactNode } from 'react'; import { MockedProvider } from '@apollo/client/testing'; import { renderHook } from '@testing-library/react'; +import { ReactNode } from 'react'; import { RecoilRoot, useSetRecoilState } from 'recoil'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; diff --git a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecordsQuery.test.tsx b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecordsQuery.test.tsx index 21cfb7f0f2e9..47082dec0f2f 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecordsQuery.test.tsx +++ b/packages/twenty-front/src/modules/object-record/hooks/__tests__/useFindManyRecordsQuery.test.tsx @@ -5,7 +5,7 @@ import { RecoilRoot } from 'recoil'; import { useFindManyRecordsQuery } from '@/object-record/hooks/useFindManyRecordsQuery'; const expectedQueryTemplate = ` - query FindManyPeople($filter: PersonFilterInput, $orderBy: PersonOrderByInput, $lastCursor: String, $limit: Int) { + query FindManyPeople($filter: PersonFilterInput, $orderBy: [PersonOrderByInput], $lastCursor: String, $limit: Int) { people(filter: $filter, orderBy: $orderBy, first: $limit, after: $lastCursor) { edges { node { diff --git a/packages/twenty-front/src/modules/object-record/hooks/useAttachRelatedRecordFromRecord.ts b/packages/twenty-front/src/modules/object-record/hooks/useAttachRelatedRecordFromRecord.ts new file mode 100644 index 000000000000..17c80b4c1887 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useAttachRelatedRecordFromRecord.ts @@ -0,0 +1,113 @@ +import { useApolloClient } from '@apollo/client'; + +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { useObjectMetadataItems } from '@/object-metadata/hooks/useObjectMetadataItems'; +import { useGetRecordFromCache } from '@/object-record/cache/hooks/useGetRecordFromCache'; +import { updateRecordFromCache } from '@/object-record/cache/utils/updateRecordFromCache'; +import { generateDepthOneRecordGqlFields } from '@/object-record/graphql/utils/generateDepthOneRecordGqlFields'; +import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { isDefined } from '~/utils/isDefined'; + +type useAttachRelatedRecordFromRecordProps = { + recordObjectNameSingular: string; + fieldNameOnRecordObject: string; +}; + +export const useAttachRelatedRecordFromRecord = ({ + recordObjectNameSingular, + fieldNameOnRecordObject, +}: useAttachRelatedRecordFromRecordProps) => { + const apolloClient = useApolloClient(); + + const { objectMetadataItem } = useObjectMetadataItem({ + objectNameSingular: recordObjectNameSingular, + }); + + const fieldOnObject = objectMetadataItem.fields.find((field) => { + return field.name === fieldNameOnRecordObject; + }); + + const relatedRecordObjectNameSingular = + fieldOnObject?.relationDefinition?.targetObjectMetadata.nameSingular; + + if (!relatedRecordObjectNameSingular) { + throw new Error( + `Could not find record related to ${recordObjectNameSingular}`, + ); + } + const { objectMetadataItem: relatedObjectMetadataItem } = + useObjectMetadataItem({ + objectNameSingular: relatedRecordObjectNameSingular, + }); + + const fieldOnRelatedObject = + fieldOnObject?.relationDefinition?.targetFieldMetadata.name; + + if (!fieldOnRelatedObject) { + throw new Error(`Missing target field for ${fieldNameOnRecordObject}`); + } + + const { updateOneRecord } = useUpdateOneRecord({ + objectNameSingular: relatedRecordObjectNameSingular, + }); + + const getRecordFromCache = useGetRecordFromCache({ + objectNameSingular: recordObjectNameSingular, + }); + + const getRelatedRecordFromCache = useGetRecordFromCache({ + objectNameSingular: relatedRecordObjectNameSingular, + }); + + const { objectMetadataItems } = useObjectMetadataItems(); + + const updateOneRecordAndAttachRelations = async ({ + recordId, + relatedRecordId, + }: { + recordId: string; + relatedRecordId: string; + }) => { + const cachedRelatedRecord = + getRelatedRecordFromCache<ObjectRecord>(relatedRecordId); + + if (!cachedRelatedRecord) { + throw new Error('could not find cached related record'); + } + + const previousRecordId = cachedRelatedRecord?.[`${fieldOnRelatedObject}Id`]; + + if (isDefined(previousRecordId)) { + const previousRecord = getRecordFromCache<ObjectRecord>(previousRecordId); + + const previousRecordWithRelation = { + ...cachedRelatedRecord, + [fieldOnRelatedObject]: previousRecord, + }; + const gqlFields = generateDepthOneRecordGqlFields({ + objectMetadataItem: relatedObjectMetadataItem, + record: previousRecordWithRelation, + }); + updateRecordFromCache({ + objectMetadataItems, + objectMetadataItem: relatedObjectMetadataItem, + cache: apolloClient.cache, + record: { + ...cachedRelatedRecord, + [fieldOnRelatedObject]: previousRecord, + }, + recordGqlFields: gqlFields, + }); + } + + await updateOneRecord({ + idToUpdate: relatedRecordId, + updateOneRecordInput: { + [`${fieldOnRelatedObject}Id`]: recordId, + }, + }); + }; + + return { updateOneRecordAndAttachRelations }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecords.ts index e95cbad37edf..b9313fae3ecc 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecords.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecords.ts @@ -18,6 +18,7 @@ type useCreateManyRecordsProps = { objectNameSingular: string; recordGqlFields?: RecordGqlOperationGqlRecordFields; skipPostOptmisticEffect?: boolean; + shouldMatchRootQueryFilter?: boolean; }; export const useCreateManyRecords = < @@ -26,6 +27,7 @@ export const useCreateManyRecords = < objectNameSingular, recordGqlFields, skipPostOptmisticEffect = false, + shouldMatchRootQueryFilter, }: useCreateManyRecordsProps) => { const apolloClient = useApolloClient(); @@ -49,10 +51,11 @@ export const useCreateManyRecords = < const createManyRecords = async ( recordsToCreate: Partial<CreatedObjectRecord>[], + upsert?: boolean, ) => { const sanitizedCreateManyRecordsInput = recordsToCreate.map( (recordToCreate) => { - const idForCreation = recordToCreate?.id ?? v4(); + const idForCreation = recordToCreate?.id ?? (upsert ? undefined : v4()); return { ...sanitizeRecordInput({ @@ -67,8 +70,12 @@ export const useCreateManyRecords = < const recordsCreatedInCache = []; for (const recordToCreate of sanitizedCreateManyRecordsInput) { + if (recordToCreate.id === null) { + continue; + } + const recordCreatedInCache = createOneRecordInCache({ - ...recordToCreate, + ...(recordToCreate as { id: string }), __typename: getObjectTypename(objectMetadataItem.nameSingular), }); @@ -83,6 +90,7 @@ export const useCreateManyRecords = < objectMetadataItem, recordsToCreate: recordsCreatedInCache, objectMetadataItems, + shouldMatchRootQueryFilter, }); } @@ -94,6 +102,7 @@ export const useCreateManyRecords = < mutation: createManyRecordsMutation, variables: { data: sanitizedCreateManyRecordsInput, + upsert: upsert, }, update: (cache, { data }) => { const records = data?.[mutationResponseField]; @@ -105,6 +114,7 @@ export const useCreateManyRecords = < objectMetadataItem, recordsToCreate: records, objectMetadataItems, + shouldMatchRootQueryFilter, }); }, }); diff --git a/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecordsMutation.ts b/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecordsMutation.ts index f038531dc586..28e499acbcca 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecordsMutation.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useCreateManyRecordsMutation.ts @@ -34,12 +34,16 @@ export const useCreateManyRecordsMutation = ({ const createManyRecordsMutation = gql` mutation Create${capitalize( objectMetadataItem.namePlural, - )}($data: [${capitalize(objectMetadataItem.nameSingular)}CreateInput!]!) { - ${mutationResponseField}(data: $data) ${mapObjectMetadataToGraphQLQuery({ - objectMetadataItems, - objectMetadataItem, - recordGqlFields, - })} + )}($data: [${capitalize( + objectMetadataItem.nameSingular, + )}CreateInput!]!, $upsert: Boolean) { + ${mutationResponseField}(data: $data, upsert: $upsert) ${mapObjectMetadataToGraphQLQuery( + { + objectMetadataItems, + objectMetadataItem, + recordGqlFields, + }, + )} }`; return { diff --git a/packages/twenty-front/src/modules/object-record/hooks/useCreateOneRecord.ts b/packages/twenty-front/src/modules/object-record/hooks/useCreateOneRecord.ts index 5cb2d9a436cf..f34ce0692f7a 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useCreateOneRecord.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useCreateOneRecord.ts @@ -1,5 +1,5 @@ -import { useState } from 'react'; import { useApolloClient } from '@apollo/client'; +import { useState } from 'react'; import { v4 } from 'uuid'; import { triggerCreateRecordsOptimisticEffect } from '@/apollo/optimistic-effect/utils/triggerCreateRecordsOptimisticEffect'; @@ -19,6 +19,7 @@ type useCreateOneRecordProps = { objectNameSingular: string; recordGqlFields?: RecordGqlOperationGqlRecordFields; skipPostOptmisticEffect?: boolean; + shouldMatchRootQueryFilter?: boolean; }; export const useCreateOneRecord = < @@ -27,6 +28,7 @@ export const useCreateOneRecord = < objectNameSingular, recordGqlFields, skipPostOptmisticEffect = false, + shouldMatchRootQueryFilter, }: useCreateOneRecordProps) => { const apolloClient = useApolloClient(); const [loading, setLoading] = useState(false); @@ -76,6 +78,7 @@ export const useCreateOneRecord = < objectMetadataItem, recordsToCreate: [recordCreatedInCache], objectMetadataItems, + shouldMatchRootQueryFilter, }); } @@ -97,7 +100,9 @@ export const useCreateOneRecord = < objectMetadataItem, recordsToCreate: [record], objectMetadataItems, + shouldMatchRootQueryFilter, }); + setLoading(false); }, }); diff --git a/packages/twenty-front/src/modules/object-record/hooks/useDeleteManyRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/useDeleteManyRecords.ts index f189d516da1b..1d3d82f3fe60 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useDeleteManyRecords.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useDeleteManyRecords.ts @@ -1,12 +1,16 @@ import { useApolloClient } from '@apollo/client'; import { triggerDeleteRecordsOptimisticEffect } from '@/apollo/optimistic-effect/utils/triggerDeleteRecordsOptimisticEffect'; +import { apiConfigState } from '@/client-config/states/apiConfigState'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; import { useObjectMetadataItems } from '@/object-metadata/hooks/useObjectMetadataItems'; import { useGetRecordFromCache } from '@/object-record/cache/hooks/useGetRecordFromCache'; +import { DEFAULT_MUTATION_BATCH_SIZE } from '@/object-record/constants/DefaultMutationBatchSize'; import { useDeleteManyRecordsMutation } from '@/object-record/hooks/useDeleteManyRecordsMutation'; import { getDeleteManyRecordsMutationResponseField } from '@/object-record/utils/getDeleteManyRecordsMutationResponseField'; +import { useRecoilValue } from 'recoil'; import { isDefined } from '~/utils/isDefined'; +import { sleep } from '~/utils/sleep'; import { capitalize } from '~/utils/string/capitalize'; type useDeleteOneRecordProps = { @@ -16,11 +20,17 @@ type useDeleteOneRecordProps = { type DeleteManyRecordsOptions = { skipOptimisticEffect?: boolean; + delayInMsBetweenRequests?: number; }; export const useDeleteManyRecords = ({ objectNameSingular, }: useDeleteOneRecordProps) => { + const apiConfig = useRecoilValue(apiConfigState); + + const mutationPageSize = + apiConfig?.mutationMaximumAffectedRecords ?? DEFAULT_MUTATION_BATCH_SIZE; + const apolloClient = useApolloClient(); const { objectMetadataItem } = useObjectMetadataItem({ @@ -45,40 +55,60 @@ export const useDeleteManyRecords = ({ idsToDelete: string[], options?: DeleteManyRecordsOptions, ) => { - const deletedRecords = await apolloClient.mutate({ - mutation: deleteManyRecordsMutation, - variables: { - filter: { id: { in: idsToDelete } }, - }, - optimisticResponse: options?.skipOptimisticEffect - ? undefined - : { - [mutationResponseField]: idsToDelete.map((idToDelete) => ({ - __typename: capitalize(objectNameSingular), - id: idToDelete, - })), - }, - update: options?.skipOptimisticEffect - ? undefined - : (cache, { data }) => { - const records = data?.[mutationResponseField]; - - if (!records?.length) return; - - const cachedRecords = records - .map((record) => getRecordFromCache(record.id, cache)) - .filter(isDefined); - - triggerDeleteRecordsOptimisticEffect({ - cache, - objectMetadataItem, - recordsToDelete: cachedRecords, - objectMetadataItems, - }); - }, - }); - - return deletedRecords.data?.[mutationResponseField] ?? null; + const numberOfBatches = Math.ceil(idsToDelete.length / mutationPageSize); + + const deletedRecords = []; + + for (let batchIndex = 0; batchIndex < numberOfBatches; batchIndex++) { + const batchIds = idsToDelete.slice( + batchIndex * mutationPageSize, + (batchIndex + 1) * mutationPageSize, + ); + + const deletedRecordsResponse = await apolloClient.mutate({ + mutation: deleteManyRecordsMutation, + variables: { + filter: { id: { in: batchIds } }, + }, + optimisticResponse: options?.skipOptimisticEffect + ? undefined + : { + [mutationResponseField]: batchIds.map((idToDelete) => ({ + __typename: capitalize(objectNameSingular), + id: idToDelete, + })), + }, + update: options?.skipOptimisticEffect + ? undefined + : (cache, { data }) => { + const records = data?.[mutationResponseField]; + + if (!records?.length) return; + + const cachedRecords = records + .map((record) => getRecordFromCache(record.id, cache)) + .filter(isDefined); + + triggerDeleteRecordsOptimisticEffect({ + cache, + objectMetadataItem, + recordsToDelete: cachedRecords, + objectMetadataItems, + }); + }, + }); + + const deletedRecordsForThisBatch = + deletedRecordsResponse.data?.[mutationResponseField] ?? []; + + deletedRecords.push(...deletedRecordsForThisBatch); + + if (isDefined(options?.delayInMsBetweenRequests)) { + await sleep(options.delayInMsBetweenRequests); + } + } + + return deletedRecords; }; return { deleteManyRecords }; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useDetachRelatedRecordFromRecord.ts b/packages/twenty-front/src/modules/object-record/hooks/useDetachRelatedRecordFromRecord.ts new file mode 100644 index 000000000000..b441ab37a808 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useDetachRelatedRecordFromRecord.ts @@ -0,0 +1,88 @@ +import { Reference, useApolloClient } from '@apollo/client'; + +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { getRefName } from '@/object-record/cache/utils/getRefName'; +import { modifyRecordFromCache } from '@/object-record/cache/utils/modifyRecordFromCache'; +import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; + +type useDetachRelatedRecordFromRecordProps = { + recordObjectNameSingular: string; + fieldNameOnRecordObject: string; +}; + +export const useDetachRelatedRecordFromRecord = ({ + recordObjectNameSingular, + fieldNameOnRecordObject, +}: useDetachRelatedRecordFromRecordProps) => { + const apolloClient = useApolloClient(); + + const { objectMetadataItem } = useObjectMetadataItem({ + objectNameSingular: recordObjectNameSingular, + }); + + const fieldOnObject = objectMetadataItem.fields.find((field) => { + return field.name === fieldNameOnRecordObject; + }); + + const relatedRecordObjectNameSingular = + fieldOnObject?.relationDefinition?.targetObjectMetadata.nameSingular; + + const fieldOnRelatedObject = + fieldOnObject?.relationDefinition?.targetFieldMetadata.name; + + if (!relatedRecordObjectNameSingular) { + throw new Error( + `Could not find record related to ${recordObjectNameSingular}`, + ); + } + + const { updateOneRecord } = useUpdateOneRecord({ + objectNameSingular: relatedRecordObjectNameSingular, + }); + + const updateOneRecordAndDetachRelations = async ({ + recordId, + relatedRecordId, + }: { + recordId: string; + relatedRecordId: string; + }) => { + modifyRecordFromCache({ + objectMetadataItem, + cache: apolloClient.cache, + fieldModifiers: { + [fieldNameOnRecordObject]: ( + fieldNameOnRecordObjectConnection, + { readField }, + ) => { + const edges = readField<{ node: Reference }[]>( + 'edges', + fieldNameOnRecordObjectConnection, + ); + + if (!edges) return fieldNameOnRecordObjectConnection; + + return { + ...fieldNameOnRecordObjectConnection, + edges: edges.filter( + (edge) => + !( + edge.node.__ref === + getRefName(relatedRecordObjectNameSingular, relatedRecordId) + ), + ), + }; + }, + }, + recordId, + }); + await updateOneRecord({ + idToUpdate: relatedRecordId, + updateOneRecordInput: { + [`${fieldOnRelatedObject}Id`]: null, + }, + }); + }; + + return { updateOneRecordAndDetachRelations }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFetchAllRecordIds.ts b/packages/twenty-front/src/modules/object-record/hooks/useFetchAllRecordIds.ts new file mode 100644 index 000000000000..715cdc5af15d --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useFetchAllRecordIds.ts @@ -0,0 +1,88 @@ +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { DEFAULT_QUERY_PAGE_SIZE } from '@/object-record/constants/DefaultQueryPageSize'; +import { UseFindManyRecordsParams } from '@/object-record/hooks/useFetchMoreRecordsWithPagination'; +import { useLazyFindManyRecords } from '@/object-record/hooks/useLazyFindManyRecords'; +import { useCallback } from 'react'; +import { isDefined } from '~/utils/isDefined'; + +type UseLazyFetchAllRecordIdsParams<T> = Omit< + UseFindManyRecordsParams<T>, + 'skip' +> & { pageSize?: number }; + +export const useFetchAllRecordIds = <T>({ + objectNameSingular, + filter, + orderBy, + pageSize = DEFAULT_QUERY_PAGE_SIZE, +}: UseLazyFetchAllRecordIdsParams<T>) => { + const { fetchMore, findManyRecords } = useLazyFindManyRecords({ + objectNameSingular, + filter, + orderBy, + limit: pageSize, + recordGqlFields: { id: true }, + }); + + const { objectMetadataItem } = useObjectMetadataItem({ + objectNameSingular, + }); + + const fetchAllRecordIds = useCallback(async () => { + if (!isDefined(findManyRecords)) { + return []; + } + + const findManyRecordsDataResult = await findManyRecords(); + + const firstQueryResult = + findManyRecordsDataResult?.data?.[objectMetadataItem.namePlural]; + + const totalCount = firstQueryResult?.totalCount ?? 0; + + const recordsCount = firstQueryResult?.edges.length ?? 0; + + const recordIdSet = new Set( + firstQueryResult?.edges?.map((edge) => edge.node.id) ?? [], + ); + + const remainingCount = totalCount - recordsCount; + + const remainingPages = Math.ceil(remainingCount / pageSize); + + let lastCursor = firstQueryResult?.pageInfo.endCursor ?? null; + + for (let pageIndex = 0; pageIndex < remainingPages; pageIndex++) { + if (lastCursor === null) { + break; + } + + const rawResult = await fetchMore?.({ + variables: { + lastCursor: lastCursor, + limit: pageSize, + }, + }); + + const fetchMoreResult = rawResult?.data?.[objectMetadataItem.namePlural]; + + for (const edge of fetchMoreResult.edges) { + recordIdSet.add(edge.node.id); + } + + if (fetchMoreResult.pageInfo.hasNextPage === false) { + break; + } + + lastCursor = fetchMoreResult.pageInfo.endCursor ?? null; + } + + const recordIds = Array.from(recordIdSet); + + return recordIds; + }, [fetchMore, findManyRecords, objectMetadataItem.namePlural, pageSize]); + + return { + fetchAllRecordIds, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFetchMoreRecordsWithPagination.ts b/packages/twenty-front/src/modules/object-record/hooks/useFetchMoreRecordsWithPagination.ts new file mode 100644 index 000000000000..2cc6be1c0b87 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useFetchMoreRecordsWithPagination.ts @@ -0,0 +1,229 @@ +import { + ApolloError, + ApolloQueryResult, + FetchMoreQueryOptions, + OperationVariables, + WatchQueryFetchPolicy, +} from '@apollo/client'; +import { isNonEmptyArray } from '@apollo/client/utilities'; +import { isNonEmptyString } from '@sniptt/guards'; +import { useMemo } from 'react'; +import { useRecoilCallback, useRecoilState, useSetRecoilState } from 'recoil'; + +import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; +import { ObjectMetadataItemIdentifier } from '@/object-metadata/types/ObjectMetadataItemIdentifier'; +import { isAggregationEnabled } from '@/object-metadata/utils/isAggregationEnabled'; +import { getRecordsFromRecordConnection } from '@/object-record/cache/utils/getRecordsFromRecordConnection'; +import { RecordGqlEdge } from '@/object-record/graphql/types/RecordGqlEdge'; +import { RecordGqlOperationFindManyResult } from '@/object-record/graphql/types/RecordGqlOperationFindManyResult'; +import { RecordGqlOperationGqlRecordFields } from '@/object-record/graphql/types/RecordGqlOperationGqlRecordFields'; +import { RecordGqlOperationVariables } from '@/object-record/graphql/types/RecordGqlOperationVariables'; +import { useHandleFindManyRecordsError } from '@/object-record/hooks/useHandleFindManyRecordsError'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { OnFindManyRecordsCompleted } from '@/object-record/types/OnFindManyRecordsCompleted'; +import { filterUniqueRecordEdgesByCursor } from '@/object-record/utils/filterUniqueRecordEdgesByCursor'; +import { getQueryIdentifier } from '@/object-record/utils/getQueryIdentifier'; +import { isDefined } from '~/utils/isDefined'; +import { capitalize } from '~/utils/string/capitalize'; + +import { cursorFamilyState } from '../states/cursorFamilyState'; +import { hasNextPageFamilyState } from '../states/hasNextPageFamilyState'; +import { isFetchingMoreRecordsFamilyState } from '../states/isFetchingMoreRecordsFamilyState'; + +export type UseFindManyRecordsParams<T> = ObjectMetadataItemIdentifier & + RecordGqlOperationVariables & { + onCompleted?: OnFindManyRecordsCompleted<T>; + skip?: boolean; + recordGqlFields?: RecordGqlOperationGqlRecordFields; + fetchPolicy?: WatchQueryFetchPolicy; + }; + +type UseFindManyRecordsStateParams< + T, + TData = RecordGqlOperationFindManyResult, +> = Omit< + UseFindManyRecordsParams<T>, + 'skip' | 'recordGqlFields' | 'fetchPolicy' +> & { + data: RecordGqlOperationFindManyResult | undefined; + error: ApolloError | undefined; + fetchMore< + TFetchData = TData, + TFetchVars extends OperationVariables = OperationVariables, + >( + fetchMoreOptions: FetchMoreQueryOptions<TFetchVars, TFetchData> & { + updateQuery?: ( + previousQueryResult: TData, + options: { + fetchMoreResult: TFetchData; + variables: TFetchVars; + }, + ) => TData; + }, + ): Promise<ApolloQueryResult<TFetchData>>; + objectMetadataItem: ObjectMetadataItem; +}; + +export const useFetchMoreRecordsWithPagination = < + T extends ObjectRecord = ObjectRecord, +>({ + objectNameSingular, + filter, + orderBy, + limit, + data, + error, + fetchMore, + objectMetadataItem, + onCompleted, +}: UseFindManyRecordsStateParams<T>) => { + const queryIdentifier = getQueryIdentifier({ + objectNameSingular, + filter, + limit, + orderBy, + }); + + const [hasNextPage] = useRecoilState(hasNextPageFamilyState(queryIdentifier)); + + const setIsFetchingMoreObjects = useSetRecoilState( + isFetchingMoreRecordsFamilyState(queryIdentifier), + ); + + const { handleFindManyRecordsError } = useHandleFindManyRecordsError({ + objectMetadataItem, + }); + + // TODO: put this into a util inspired from https://github.com/apollographql/apollo-client/blob/master/src/utilities/policies/pagination.ts + // This function is equivalent to merge function + read function in field policy + const fetchMoreRecords = useRecoilCallback( + ({ snapshot, set }) => + async () => { + const hasNextPageLocal = snapshot + .getLoadable(hasNextPageFamilyState(queryIdentifier)) + .getValue(); + + const lastCursorLocal = snapshot + .getLoadable(cursorFamilyState(queryIdentifier)) + .getValue(); + + // Remote objects does not support hasNextPage. We cannot rely on it to fetch more records. + if ( + hasNextPageLocal || + (!isAggregationEnabled(objectMetadataItem) && !error) + ) { + setIsFetchingMoreObjects(true); + + try { + const { data: fetchMoreDataResult } = await fetchMore({ + variables: { + filter, + orderBy, + lastCursor: isNonEmptyString(lastCursorLocal) + ? lastCursorLocal + : undefined, + }, + updateQuery: (prev, { fetchMoreResult }) => { + const previousEdges = + prev?.[objectMetadataItem.namePlural]?.edges; + const nextEdges = + fetchMoreResult?.[objectMetadataItem.namePlural]?.edges; + + let newEdges: RecordGqlEdge[] = previousEdges ?? []; + + if (isNonEmptyArray(nextEdges)) { + newEdges = filterUniqueRecordEdgesByCursor([ + ...newEdges, + ...(fetchMoreResult?.[objectMetadataItem.namePlural] + ?.edges ?? []), + ]); + } + + const pageInfo = + fetchMoreResult?.[objectMetadataItem.namePlural]?.pageInfo; + + if (isDefined(data?.[objectMetadataItem.namePlural])) { + set( + cursorFamilyState(queryIdentifier), + pageInfo.endCursor ?? '', + ); + set( + hasNextPageFamilyState(queryIdentifier), + pageInfo.hasNextPage ?? false, + ); + } + + const records = getRecordsFromRecordConnection({ + recordConnection: { + edges: newEdges, + pageInfo, + }, + }) as T[]; + + onCompleted?.(records, { + pageInfo, + totalCount: + fetchMoreResult?.[objectMetadataItem.namePlural] + ?.totalCount, + }); + + return Object.assign({}, prev, { + [objectMetadataItem.namePlural]: { + __typename: `${capitalize( + objectMetadataItem.nameSingular, + )}Connection`, + edges: newEdges, + pageInfo: + fetchMoreResult?.[objectMetadataItem.namePlural].pageInfo, + totalCount: + fetchMoreResult?.[objectMetadataItem.namePlural] + .totalCount, + }, + } as RecordGqlOperationFindManyResult); + }, + }); + + return { + data: fetchMoreDataResult?.[objectMetadataItem.namePlural], + }; + } catch (error) { + handleFindManyRecordsError(error as ApolloError); + } finally { + setIsFetchingMoreObjects(false); + } + } + }, + [ + objectMetadataItem, + error, + setIsFetchingMoreObjects, + fetchMore, + filter, + orderBy, + data, + onCompleted, + handleFindManyRecordsError, + queryIdentifier, + ], + ); + + const totalCount = data?.[objectMetadataItem.namePlural]?.totalCount; + + const records = useMemo( + () => + data?.[objectMetadataItem.namePlural] + ? getRecordsFromRecordConnection<T>({ + recordConnection: data?.[objectMetadataItem.namePlural], + }) + : ([] as T[]), + + [data, objectMetadataItem.namePlural], + ); + + return { + fetchMoreRecords, + totalCount, + records, + hasNextPage, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFieldContext.tsx b/packages/twenty-front/src/modules/object-record/hooks/useFieldContext.tsx index cbd002af74a7..9a4be7849d8f 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useFieldContext.tsx +++ b/packages/twenty-front/src/modules/object-record/hooks/useFieldContext.tsx @@ -19,6 +19,7 @@ export const useFieldContext = ({ objectNameSingular, objectRecordId, customUseUpdateOneObjectHook, + overridenIsFieldEmpty, }: { clearable?: boolean; fieldMetadataName: string; @@ -27,6 +28,7 @@ export const useFieldContext = ({ objectNameSingular: string; objectRecordId: string; customUseUpdateOneObjectHook?: RecordUpdateHook; + overridenIsFieldEmpty?: boolean; }) => { const { objectMetadataItem } = useObjectMetadataItem({ objectNameSingular, @@ -78,6 +80,7 @@ export const useFieldContext = ({ customUseUpdateOneObjectHook ?? useUpdateOneObjectMutation, hotkeyScope: InlineCellHotkeyScope.InlineCell, clearable, + overridenIsFieldEmpty, }} > {children} diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicateRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicateRecords.ts index 4bf2d0c1ebfd..bd75db4a75a4 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicateRecords.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicateRecords.ts @@ -5,7 +5,7 @@ import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadata import { ObjectMetadataItemIdentifier } from '@/object-metadata/types/ObjectMetadataItemIdentifier'; import { getRecordsFromRecordConnection } from '@/object-record/cache/utils/getRecordsFromRecordConnection'; import { RecordGqlConnection } from '@/object-record/graphql/types/RecordGqlConnection'; -import { RecordGqlOperationFindManyResult } from '@/object-record/graphql/types/RecordGqlOperationFindManyResult'; +import { RecordGqlOperationFindDuplicatesResult } from '@/object-record/graphql/types/RecordGqlOperationFindDuplicatesResults'; import { useFindDuplicateRecordsQuery } from '@/object-record/hooks/useFindDuplicatesRecordsQuery'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; import { getFindDuplicateRecordsQueryResponseField } from '@/object-record/utils/getFindDuplicateRecordsQueryResponseField'; @@ -14,12 +14,12 @@ import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; import { logError } from '~/utils/logError'; export const useFindDuplicateRecords = <T extends ObjectRecord = ObjectRecord>({ - objectRecordId = '', + objectRecordIds = [], objectNameSingular, onCompleted, }: ObjectMetadataItemIdentifier & { - objectRecordId: string | undefined; - onCompleted?: (data: RecordGqlConnection) => void; + objectRecordIds: string[] | undefined; + onCompleted?: (data: RecordGqlConnection[]) => void; skip?: boolean; }) => { const findDuplicateQueryStateIdentifier = objectNameSingular; @@ -38,46 +38,48 @@ export const useFindDuplicateRecords = <T extends ObjectRecord = ObjectRecord>({ objectMetadataItem.nameSingular, ); - const { data, loading, error } = useQuery<RecordGqlOperationFindManyResult>( - findDuplicateRecordsQuery, - { - variables: { - id: objectRecordId, + const { data, loading, error } = + useQuery<RecordGqlOperationFindDuplicatesResult>( + findDuplicateRecordsQuery, + { + variables: { + ids: objectRecordIds, + }, + onCompleted: (data) => { + onCompleted?.(data[queryResponseField]); + }, + onError: (error) => { + logError( + `useFindDuplicateRecords for "${objectMetadataItem.nameSingular}" error : ` + + error, + ); + enqueueSnackBar( + `Error during useFindDuplicateRecords for "${objectMetadataItem.nameSingular}", ${error.message}`, + { + variant: SnackBarVariant.Error, + }, + ); + }, }, - onCompleted: (data) => { - onCompleted?.(data[queryResponseField]); - }, - onError: (error) => { - logError( - `useFindDuplicateRecords for "${objectMetadataItem.nameSingular}" error : ` + - error, - ); - enqueueSnackBar( - `Error during useFindDuplicateRecords for "${objectMetadataItem.nameSingular}", ${error.message}`, - { - variant: SnackBarVariant.Error, - }, - ); - }, - }, - ); + ); - const objectRecordConnection = data?.[queryResponseField]; + const objectResults = data?.[queryResponseField]; - const records = useMemo( + const results = useMemo( () => - objectRecordConnection - ? (getRecordsFromRecordConnection({ - recordConnection: objectRecordConnection, - }) as T[]) - : [], - [objectRecordConnection], + objectResults?.map((result: RecordGqlConnection) => { + return result + ? (getRecordsFromRecordConnection({ + recordConnection: result, + }) as T[]) + : []; + }), + [objectResults], ); return { objectMetadataItem, - records, - totalCount: objectRecordConnection?.totalCount, + results, loading, error, queryStateIdentifier: findDuplicateQueryStateIdentifier, diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicatesRecordsQuery.ts b/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicatesRecordsQuery.ts index 9968d2d46a72..b3b95e270b95 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicatesRecordsQuery.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useFindDuplicatesRecordsQuery.ts @@ -22,10 +22,10 @@ export const useFindDuplicateRecordsQuery = ({ const findDuplicateRecordsQuery = gql` query FindDuplicate${capitalize( objectMetadataItem.nameSingular, - )}($id: ID!) { + )}($ids: [ID!]!) { ${getFindDuplicateRecordsQueryResponseField( objectMetadataItem.nameSingular, - )}(id: $id) { + )}(ids: $ids) { edges { node ${mapObjectMetadataToGraphQLQuery({ objectMetadataItems, @@ -38,7 +38,6 @@ export const useFindDuplicateRecordsQuery = ({ startCursor endCursor } - ${isAggregationEnabled(objectMetadataItem) ? 'totalCount' : ''} } } `; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecords.ts index befc85849d7e..da5dc238af89 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecords.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecords.ts @@ -1,236 +1,97 @@ -import { useCallback, useMemo } from 'react'; import { useQuery, WatchQueryFetchPolicy } from '@apollo/client'; -import { isNonEmptyArray } from '@apollo/client/utilities'; -import { isNonEmptyString } from '@sniptt/guards'; -import { useRecoilState, useRecoilValue, useSetRecoilState } from 'recoil'; +import { useRecoilValue } from 'recoil'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; import { ObjectMetadataItemIdentifier } from '@/object-metadata/types/ObjectMetadataItemIdentifier'; -import { isAggregationEnabled } from '@/object-metadata/utils/isAggregationEnabled'; -import { getRecordsFromRecordConnection } from '@/object-record/cache/utils/getRecordsFromRecordConnection'; -import { RecordGqlConnection } from '@/object-record/graphql/types/RecordGqlConnection'; -import { RecordGqlEdge } from '@/object-record/graphql/types/RecordGqlEdge'; import { RecordGqlOperationFindManyResult } from '@/object-record/graphql/types/RecordGqlOperationFindManyResult'; import { RecordGqlOperationGqlRecordFields } from '@/object-record/graphql/types/RecordGqlOperationGqlRecordFields'; import { RecordGqlOperationVariables } from '@/object-record/graphql/types/RecordGqlOperationVariables'; +import { useFetchMoreRecordsWithPagination } from '@/object-record/hooks/useFetchMoreRecordsWithPagination'; import { useFindManyRecordsQuery } from '@/object-record/hooks/useFindManyRecordsQuery'; +import { useHandleFindManyRecordsCompleted } from '@/object-record/hooks/useHandleFindManyRecordsCompleted'; +import { useHandleFindManyRecordsError } from '@/object-record/hooks/useHandleFindManyRecordsError'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; -import { filterUniqueRecordEdgesByCursor } from '@/object-record/utils/filterUniqueRecordEdgesByCursor'; -import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; -import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; -import { isDefined } from '~/utils/isDefined'; -import { logError } from '~/utils/logError'; -import { capitalize } from '~/utils/string/capitalize'; +import { OnFindManyRecordsCompleted } from '@/object-record/types/OnFindManyRecordsCompleted'; +import { getQueryIdentifier } from '@/object-record/utils/getQueryIdentifier'; -import { cursorFamilyState } from '../states/cursorFamilyState'; -import { hasNextPageFamilyState } from '../states/hasNextPageFamilyState'; -import { isFetchingMoreRecordsFamilyState } from '../states/isFetchingMoreRecordsFamilyState'; +export type UseFindManyRecordsParams<T> = ObjectMetadataItemIdentifier & + RecordGqlOperationVariables & { + onError?: (error?: Error) => void; + onCompleted?: OnFindManyRecordsCompleted<T>; + skip?: boolean; + recordGqlFields?: RecordGqlOperationGqlRecordFields; + fetchPolicy?: WatchQueryFetchPolicy; + }; export const useFindManyRecords = <T extends ObjectRecord = ObjectRecord>({ objectNameSingular, filter, orderBy, limit, - onCompleted, - onError, skip, recordGqlFields, fetchPolicy, -}: ObjectMetadataItemIdentifier & - RecordGqlOperationVariables & { - onCompleted?: ( - records: T[], - options?: { - pageInfo?: RecordGqlConnection['pageInfo']; - totalCount?: number; - }, - ) => void; - onError?: (error?: Error) => void; - skip?: boolean; - recordGqlFields?: RecordGqlOperationGqlRecordFields; - fetchPolicy?: WatchQueryFetchPolicy; - }) => { - const findManyQueryStateIdentifier = - objectNameSingular + - JSON.stringify(filter) + - JSON.stringify(orderBy) + - limit; - - const [lastCursor, setLastCursor] = useRecoilState( - cursorFamilyState(findManyQueryStateIdentifier), - ); - - const [hasNextPage, setHasNextPage] = useRecoilState( - hasNextPageFamilyState(findManyQueryStateIdentifier), - ); - - const setIsFetchingMoreObjects = useSetRecoilState( - isFetchingMoreRecordsFamilyState(findManyQueryStateIdentifier), - ); - + onError, + onCompleted, + cursorFilter, +}: UseFindManyRecordsParams<T>) => { + const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); const { objectMetadataItem } = useObjectMetadataItem({ objectNameSingular, }); - const { findManyRecordsQuery } = useFindManyRecordsQuery({ objectNameSingular, recordGqlFields, + cursorDirection: cursorFilter?.cursorDirection, }); - const { enqueueSnackBar } = useSnackBar(); - const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); + const { handleFindManyRecordsError } = useHandleFindManyRecordsError({ + objectMetadataItem, + handleError: onError, + }); + + const queryIdentifier = getQueryIdentifier({ + objectNameSingular, + filter, + orderBy, + limit, + }); + + const { handleFindManyRecordsCompleted } = useHandleFindManyRecordsCompleted({ + objectMetadataItem, + queryIdentifier, + onCompleted, + }); const { data, loading, error, fetchMore } = useQuery<RecordGqlOperationFindManyResult>(findManyRecordsQuery, { skip: skip || !objectMetadataItem || !currentWorkspaceMember, variables: { filter, - limit, orderBy, + lastCursor: cursorFilter?.cursor ?? undefined, + limit: cursorFilter?.limit ?? limit, }, fetchPolicy: fetchPolicy, - onCompleted: (data) => { - if (!isDefined(data)) { - onCompleted?.([]); - } - - const pageInfo = data?.[objectMetadataItem.namePlural]?.pageInfo; - - const records = getRecordsFromRecordConnection({ - recordConnection: data?.[objectMetadataItem.namePlural], - }) as T[]; - - onCompleted?.(records, { - pageInfo, - totalCount: data?.[objectMetadataItem.namePlural]?.totalCount, - }); - - if (isDefined(data?.[objectMetadataItem.namePlural])) { - setLastCursor(pageInfo.endCursor ?? ''); - setHasNextPage(pageInfo.hasNextPage ?? false); - } - }, - onError: (error) => { - logError( - `useFindManyRecords for "${objectMetadataItem.namePlural}" error : ` + - error, - ); - enqueueSnackBar( - `Error during useFindManyRecords for "${objectMetadataItem.namePlural}", ${error.message}`, - { - variant: SnackBarVariant.Error, - }, - ); - onError?.(error); - }, + onCompleted: handleFindManyRecordsCompleted, + onError: handleFindManyRecordsError, }); - const fetchMoreRecords = useCallback(async () => { - // Remote objects does not support hasNextPage. We cannot rely on it to fetch more records. - if (hasNextPage || (!isAggregationEnabled(objectMetadataItem) && !error)) { - setIsFetchingMoreObjects(true); - - try { - await fetchMore({ - variables: { - filter, - orderBy, - lastCursor: isNonEmptyString(lastCursor) ? lastCursor : undefined, - }, - updateQuery: (prev, { fetchMoreResult }) => { - const previousEdges = prev?.[objectMetadataItem.namePlural]?.edges; - const nextEdges = - fetchMoreResult?.[objectMetadataItem.namePlural]?.edges; - - let newEdges: RecordGqlEdge[] = previousEdges ?? []; - - if (isNonEmptyArray(nextEdges)) { - newEdges = filterUniqueRecordEdgesByCursor([ - ...newEdges, - ...(fetchMoreResult?.[objectMetadataItem.namePlural]?.edges ?? - []), - ]); - } - - const pageInfo = - fetchMoreResult?.[objectMetadataItem.namePlural]?.pageInfo; - - if (isDefined(data?.[objectMetadataItem.namePlural])) { - setLastCursor(pageInfo.endCursor ?? ''); - setHasNextPage(pageInfo.hasNextPage ?? false); - } - - const records = getRecordsFromRecordConnection({ - recordConnection: { - edges: newEdges, - pageInfo, - }, - }) as T[]; - - onCompleted?.(records, { - pageInfo, - totalCount: - fetchMoreResult?.[objectMetadataItem.namePlural]?.totalCount, - }); - - return Object.assign({}, prev, { - [objectMetadataItem.namePlural]: { - __typename: `${capitalize( - objectMetadataItem.nameSingular, - )}Connection`, - edges: newEdges, - pageInfo: - fetchMoreResult?.[objectMetadataItem.namePlural].pageInfo, - totalCount: - fetchMoreResult?.[objectMetadataItem.namePlural].totalCount, - }, - } as RecordGqlOperationFindManyResult); - }, - }); - } catch (error) { - logError( - `fetchMoreObjects for "${objectMetadataItem.namePlural}" error : ` + - error, - ); - enqueueSnackBar( - `Error during fetchMoreObjects for "${objectMetadataItem.namePlural}", ${error}`, - { - variant: SnackBarVariant.Error, - }, - ); - } finally { - setIsFetchingMoreObjects(false); - } - } - }, [ - hasNextPage, - objectMetadataItem, - error, - setIsFetchingMoreObjects, - fetchMore, - filter, - orderBy, - lastCursor, - data, - onCompleted, - setLastCursor, - setHasNextPage, - enqueueSnackBar, - ]); - - const totalCount = data?.[objectMetadataItem.namePlural]?.totalCount; - - const records = useMemo( - () => - data?.[objectMetadataItem.namePlural] - ? getRecordsFromRecordConnection<T>({ - recordConnection: data?.[objectMetadataItem.namePlural], - }) - : ([] as T[]), + const { fetchMoreRecords, records, hasNextPage } = + useFetchMoreRecordsWithPagination<T>({ + objectNameSingular, + filter, + orderBy, + limit, + fetchMore, + data, + error, + objectMetadataItem, + }); - [data, objectMetadataItem.namePlural], - ); + const pageInfo = data?.[objectMetadataItem.namePlural].pageInfo; + const totalCount = data?.[objectMetadataItem.namePlural].totalCount; return { objectMetadataItem, @@ -239,7 +100,8 @@ export const useFindManyRecords = <T extends ObjectRecord = ObjectRecord>({ loading, error, fetchMoreRecords, - queryStateIdentifier: findManyQueryStateIdentifier, + queryStateIdentifier: queryIdentifier, hasNextPage, + pageInfo, }; }; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecordsQuery.ts b/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecordsQuery.ts index 55a6f270dcef..0ede4f9f009c 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecordsQuery.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useFindManyRecordsQuery.ts @@ -3,16 +3,21 @@ import { useRecoilValue } from 'recoil'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; import { objectMetadataItemsState } from '@/object-metadata/states/objectMetadataItemsState'; import { RecordGqlOperationGqlRecordFields } from '@/object-record/graphql/types/RecordGqlOperationGqlRecordFields'; -import { generateFindManyRecordsQuery } from '@/object-record/utils/generateFindManyRecordsQuery'; +import { + generateFindManyRecordsQuery, + QueryCursorDirection, +} from '@/object-record/utils/generateFindManyRecordsQuery'; export const useFindManyRecordsQuery = ({ objectNameSingular, recordGqlFields, computeReferences, + cursorDirection = 'after', }: { objectNameSingular: string; recordGqlFields?: RecordGqlOperationGqlRecordFields; computeReferences?: boolean; + cursorDirection?: QueryCursorDirection; }) => { const { objectMetadataItem } = useObjectMetadataItem({ objectNameSingular, @@ -25,6 +30,7 @@ export const useFindManyRecordsQuery = ({ objectMetadataItems, recordGqlFields, computeReferences, + cursorDirection, }); return { diff --git a/packages/twenty-front/src/modules/object-record/hooks/useFindOneRecord.ts b/packages/twenty-front/src/modules/object-record/hooks/useFindOneRecord.ts index af72ad296adc..76c9645bd850 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useFindOneRecord.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useFindOneRecord.ts @@ -1,5 +1,5 @@ -import { useMemo } from 'react'; import { useQuery } from '@apollo/client'; +import { useMemo } from 'react'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; import { ObjectMetadataItemIdentifier } from '@/object-metadata/types/ObjectMetadataItemIdentifier'; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useHandleFindManyRecordsCompleted.ts b/packages/twenty-front/src/modules/object-record/hooks/useHandleFindManyRecordsCompleted.ts new file mode 100644 index 000000000000..f0f89bfb2e6f --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useHandleFindManyRecordsCompleted.ts @@ -0,0 +1,53 @@ +import { useRecoilState } from 'recoil'; + +import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; +import { getRecordsFromRecordConnection } from '@/object-record/cache/utils/getRecordsFromRecordConnection'; +import { RecordGqlOperationFindManyResult } from '@/object-record/graphql/types/RecordGqlOperationFindManyResult'; +import { cursorFamilyState } from '@/object-record/states/cursorFamilyState'; +import { hasNextPageFamilyState } from '@/object-record/states/hasNextPageFamilyState'; +import { OnFindManyRecordsCompleted } from '@/object-record/types/OnFindManyRecordsCompleted'; +import { isDefined } from '~/utils/isDefined'; + +export const useHandleFindManyRecordsCompleted = <T>({ + queryIdentifier, + onCompleted, + objectMetadataItem, +}: { + queryIdentifier: string; + objectMetadataItem: ObjectMetadataItem; + onCompleted?: OnFindManyRecordsCompleted<T>; +}) => { + const [, setLastCursor] = useRecoilState(cursorFamilyState(queryIdentifier)); + + const [, setHasNextPage] = useRecoilState( + hasNextPageFamilyState(queryIdentifier), + ); + + const handleFindManyRecordsCompleted = ( + data: RecordGqlOperationFindManyResult, + ) => { + if (!isDefined(data)) { + onCompleted?.([]); + } + + const pageInfo = data?.[objectMetadataItem.namePlural]?.pageInfo; + + const records = getRecordsFromRecordConnection({ + recordConnection: data?.[objectMetadataItem.namePlural], + }) as T[]; + + onCompleted?.(records, { + pageInfo, + totalCount: data?.[objectMetadataItem.namePlural]?.totalCount, + }); + + if (isDefined(data?.[objectMetadataItem.namePlural])) { + setLastCursor(pageInfo.endCursor ?? ''); + setHasNextPage(pageInfo.hasNextPage ?? false); + } + }; + + return { + handleFindManyRecordsCompleted, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useHandleFindManyRecordsError.ts b/packages/twenty-front/src/modules/object-record/hooks/useHandleFindManyRecordsError.ts new file mode 100644 index 000000000000..1b5a8cae64ef --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useHandleFindManyRecordsError.ts @@ -0,0 +1,34 @@ +import { ApolloError } from '@apollo/client'; + +import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; +import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; +import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; +import { logError } from '~/utils/logError'; + +export const useHandleFindManyRecordsError = ({ + handleError, + objectMetadataItem, +}: { + objectMetadataItem: ObjectMetadataItem; + handleError?: (error?: Error) => void; +}) => { + const { enqueueSnackBar } = useSnackBar(); + + const handleFindManyRecordsError = (error: ApolloError) => { + logError( + `useFindManyRecords for "${objectMetadataItem.namePlural}" error : ` + + error, + ); + enqueueSnackBar( + `Error during useFindManyRecords for "${objectMetadataItem.namePlural}", ${error.message}`, + { + variant: SnackBarVariant.Error, + }, + ); + handleError?.(error); + }; + + return { + handleFindManyRecordsError, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useLazyFindManyRecords.ts b/packages/twenty-front/src/modules/object-record/hooks/useLazyFindManyRecords.ts new file mode 100644 index 000000000000..caf315296d4c --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useLazyFindManyRecords.ts @@ -0,0 +1,115 @@ +import { useLazyQuery } from '@apollo/client'; +import { useRecoilCallback } from 'recoil'; + +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { RecordGqlOperationFindManyResult } from '@/object-record/graphql/types/RecordGqlOperationFindManyResult'; +import { useFetchMoreRecordsWithPagination } from '@/object-record/hooks/useFetchMoreRecordsWithPagination'; +import { UseFindManyRecordsParams } from '@/object-record/hooks/useFindManyRecords'; +import { useFindManyRecordsQuery } from '@/object-record/hooks/useFindManyRecordsQuery'; +import { useHandleFindManyRecordsCompleted } from '@/object-record/hooks/useHandleFindManyRecordsCompleted'; +import { useHandleFindManyRecordsError } from '@/object-record/hooks/useHandleFindManyRecordsError'; +import { cursorFamilyState } from '@/object-record/states/cursorFamilyState'; +import { hasNextPageFamilyState } from '@/object-record/states/hasNextPageFamilyState'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { getQueryIdentifier } from '@/object-record/utils/getQueryIdentifier'; + +type UseLazyFindManyRecordsParams<T> = Omit< + UseFindManyRecordsParams<T>, + 'skip' +>; + +export const useLazyFindManyRecords = <T extends ObjectRecord = ObjectRecord>({ + objectNameSingular, + filter, + orderBy, + limit, + recordGqlFields, + fetchPolicy, + onCompleted, + onError, +}: UseLazyFindManyRecordsParams<T>) => { + const { objectMetadataItem } = useObjectMetadataItem({ + objectNameSingular, + }); + + const { findManyRecordsQuery } = useFindManyRecordsQuery({ + objectNameSingular, + recordGqlFields, + }); + + const { handleFindManyRecordsError } = useHandleFindManyRecordsError({ + objectMetadataItem, + handleError: onError, + }); + + const queryIdentifier = getQueryIdentifier({ + objectNameSingular, + filter, + orderBy, + limit, + }); + + const { handleFindManyRecordsCompleted } = useHandleFindManyRecordsCompleted({ + objectMetadataItem, + queryIdentifier, + onCompleted, + }); + + const [findManyRecords, { data, loading, error, fetchMore }] = + useLazyQuery<RecordGqlOperationFindManyResult>(findManyRecordsQuery, { + variables: { + filter, + limit, + orderBy, + }, + fetchPolicy: fetchPolicy, + onCompleted: handleFindManyRecordsCompleted, + onError: handleFindManyRecordsError, + }); + + const { fetchMoreRecords, totalCount, records } = + useFetchMoreRecordsWithPagination<T>({ + objectNameSingular, + filter, + orderBy, + limit, + onCompleted, + fetchMore, + data, + error, + objectMetadataItem, + }); + + const findManyRecordsLazy = useRecoilCallback( + ({ set }) => + async () => { + const result = await findManyRecords(); + + const hasNextPage = + result?.data?.[objectMetadataItem.namePlural]?.pageInfo.hasNextPage ?? + false; + + const lastCursor = + result?.data?.[objectMetadataItem.namePlural]?.pageInfo.endCursor ?? + ''; + + set(hasNextPageFamilyState(queryIdentifier), hasNextPage); + set(cursorFamilyState(queryIdentifier), lastCursor); + + return result; + }, + [queryIdentifier, findManyRecords, objectMetadataItem], + ); + + return { + objectMetadataItem, + records, + totalCount, + loading, + error, + fetchMore, + fetchMoreRecordsWithPagination: fetchMoreRecords, + queryStateIdentifier: queryIdentifier, + findManyRecords: findManyRecordsLazy, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useRecordChipData.ts b/packages/twenty-front/src/modules/object-record/hooks/useRecordChipData.ts new file mode 100644 index 000000000000..1958a09eb535 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/hooks/useRecordChipData.ts @@ -0,0 +1,24 @@ +import { PreComputedChipGeneratorsContext } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; +import { generateDefaultRecordChipData } from '@/object-metadata/utils/generateDefaultRecordChipData'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { useContext } from 'react'; + +export const useRecordChipData = ({ + objectNameSingular, + record, +}: { + objectNameSingular: string; + record: ObjectRecord; +}) => { + const { identifierChipGeneratorPerObject } = useContext( + PreComputedChipGeneratorsContext, + ); + + const generateRecordChipData = + identifierChipGeneratorPerObject[objectNameSingular] ?? + generateDefaultRecordChipData; + + const recordChipData = generateRecordChipData(record); + + return { recordChipData }; +}; diff --git a/packages/twenty-front/src/modules/object-record/hooks/useUpdateOneRecord.ts b/packages/twenty-front/src/modules/object-record/hooks/useUpdateOneRecord.ts index fc0b5a1eb903..427c48547172 100644 --- a/packages/twenty-front/src/modules/object-record/hooks/useUpdateOneRecord.ts +++ b/packages/twenty-front/src/modules/object-record/hooks/useUpdateOneRecord.ts @@ -47,9 +47,11 @@ export const useUpdateOneRecord = < const updateOneRecord = async ({ idToUpdate, updateOneRecordInput, + optimisticRecord, }: { idToUpdate: string; updateOneRecordInput: Partial<Omit<UpdatedObjectRecord, 'id'>>; + optimisticRecord?: Partial<ObjectRecord>; }) => { const sanitizedInput = { ...sanitizeRecordInput({ @@ -68,16 +70,16 @@ export const useUpdateOneRecord = < computeReferences: true, }); - const optimisticRecord = { + const computedOptimisticRecord = { ...cachedRecord, - ...sanitizedInput, + ...(optimisticRecord ?? sanitizedInput), ...{ id: idToUpdate }, ...{ __typename: capitalize(objectMetadataItem.nameSingular) }, }; const optimisticRecordWithConnection = getRecordNodeFromRecord<ObjectRecord>({ - record: optimisticRecord, + record: computedOptimisticRecord, objectMetadataItem, objectMetadataItems, recordGqlFields: computedRecordGqlFields, @@ -92,7 +94,7 @@ export const useUpdateOneRecord = < objectMetadataItems, objectMetadataItem, cache: apolloClient.cache, - record: optimisticRecord, + record: computedOptimisticRecord, }); triggerUpdateRecordOptimisticEffect({ diff --git a/packages/twenty-front/src/modules/object-record/multiple-objects/hooks/useGenerateCombinedFindManyRecordsQuery.ts b/packages/twenty-front/src/modules/object-record/multiple-objects/hooks/useGenerateCombinedFindManyRecordsQuery.ts index a60d7d5b67aa..9e7935792c6c 100644 --- a/packages/twenty-front/src/modules/object-record/multiple-objects/hooks/useGenerateCombinedFindManyRecordsQuery.ts +++ b/packages/twenty-front/src/modules/object-record/multiple-objects/hooks/useGenerateCombinedFindManyRecordsQuery.ts @@ -32,9 +32,9 @@ export const useGenerateCombinedFindManyRecordsQuery = ({ const orderByPerMetadataItemArray = operationSignatures .map( ({ objectNameSingular }) => - `$orderBy${capitalize(objectNameSingular)}: ${capitalize( + `$orderBy${capitalize(objectNameSingular)}: [${capitalize( objectNameSingular, - )}OrderByInput`, + )}OrderByInput]`, ) .join(', '); diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/GenericEntityFilterChip.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/GenericEntityFilterChip.tsx index e11ae14c966d..fae9454c187a 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/GenericEntityFilterChip.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/GenericEntityFilterChip.tsx @@ -1,4 +1,4 @@ -import { EntityChip, IconComponent } from 'twenty-ui'; +import { AvatarChip, IconComponent } from 'twenty-ui'; import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; @@ -13,8 +13,8 @@ export const GenericEntityFilterChip = ({ filter, Icon, }: GenericEntityFilterChipProps) => ( - <EntityChip - entityId={filter.value} + <AvatarChip + placeholderColorSeed={filter.value} name={filter.displayValue} avatarType="rounded" avatarUrl={getImageAbsoluteURIOrBase64(filter.displayAvatarUrl) || ''} diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/MultipleFiltersDropdownContent.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/MultipleFiltersDropdownContent.tsx index 1bd909b72f98..cd49fd096b89 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/MultipleFiltersDropdownContent.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/MultipleFiltersDropdownContent.tsx @@ -3,6 +3,7 @@ import { useRecoilValue } from 'recoil'; import { ObjectFilterDropdownSearchInput } from '@/object-record/object-filter-dropdown/components/ObjectFilterDropdownSearchInput'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; import { DropdownMenuSeparator } from '@/ui/layout/dropdown/components/DropdownMenuSeparator'; +import { ViewFilterOperand } from '@/views/types/ViewFilterOperand'; import { MultipleFiltersDropdownFilterOnFilterChangedEffect } from './MultipleFiltersDropdownFilterOnFilterChangedEffect'; import { ObjectFilterDropdownDateInput } from './ObjectFilterDropdownDateInput'; @@ -36,6 +37,11 @@ export const MultipleFiltersDropdownContent = ({ const selectedOperandInDropdown = useRecoilValue( selectedOperandInDropdownState, ); + const isEmptyOperand = + selectedOperandInDropdown && + [ViewFilterOperand.IsEmpty, ViewFilterOperand.IsNotEmpty].includes( + selectedOperandInDropdown, + ); return ( <> @@ -43,6 +49,8 @@ export const MultipleFiltersDropdownContent = ({ <ObjectFilterDropdownFilterSelect /> ) : isObjectFilterDropdownOperandSelectUnfolded ? ( <ObjectFilterDropdownOperandSelect /> + ) : isEmptyOperand ? ( + <ObjectFilterDropdownOperandButton /> ) : ( selectedOperandInDropdown && ( <> diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownDateInput.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownDateInput.tsx index 11c9f1ab2820..741b448ab07c 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownDateInput.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownDateInput.tsx @@ -1,13 +1,18 @@ import { useRecoilValue } from 'recoil'; +import { v4 } from 'uuid'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; import { InternalDatePicker } from '@/ui/input/components/internal/date/components/InternalDatePicker'; +import { useState } from 'react'; import { isDefined } from '~/utils/isDefined'; export const ObjectFilterDropdownDateInput = () => { + const [internalDate, setInternalDate] = useState<Date | null>(new Date()); + const { filterDefinitionUsedInDropdownState, selectedOperandInDropdownState, + selectedFilterState, setIsObjectFilterDropdownUnfolded, selectFilter, } = useFilterDropdown(); @@ -19,10 +24,14 @@ export const ObjectFilterDropdownDateInput = () => { selectedOperandInDropdownState, ); + const selectedFilter = useRecoilValue(selectedFilterState); + const handleChange = (date: Date | null) => { - if (!filterDefinitionUsedInDropdown || !selectedOperandInDropdown) return; + setInternalDate(date); + if (!filterDefinitionUsedInDropdown || !selectedOperandInDropdown) return; selectFilter?.({ + id: selectedFilter?.id ? selectedFilter.id : v4(), fieldMetadataId: filterDefinitionUsedInDropdown.fieldMetadataId, value: isDefined(date) ? date.toISOString() : '', operand: selectedOperandInDropdown, @@ -35,7 +44,7 @@ export const ObjectFilterDropdownDateInput = () => { return ( <InternalDatePicker - date={new Date()} + date={internalDate} onChange={handleChange} onMouseSelect={handleChange} /> diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownEntitySearchSelect.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownEntitySearchSelect.tsx deleted file mode 100644 index 27b155975058..000000000000 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownEntitySearchSelect.tsx +++ /dev/null @@ -1,127 +0,0 @@ -import { useEffect, useState } from 'react'; -import { useRecoilValue } from 'recoil'; - -import { OBJECT_FILTER_DROPDOWN_ID } from '@/object-record/object-filter-dropdown/constants/ObjectFilterDropdownId'; -import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; -import { SingleEntitySelectMenuItems } from '@/object-record/relation-picker/components/SingleEntitySelectMenuItems'; -import { EntitiesForMultipleEntitySelect } from '@/object-record/relation-picker/types/EntitiesForMultipleEntitySelect'; -import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; -import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown'; -import { ViewFilterOperand } from '@/views/types/ViewFilterOperand'; - -export const ObjectFilterDropdownEntitySearchSelect = ({ - entitiesForSelect, -}: { - entitiesForSelect: EntitiesForMultipleEntitySelect<EntityForSelect>; -}) => { - const { - setObjectFilterDropdownSelectedEntityId, - filterDefinitionUsedInDropdownState, - selectedOperandInDropdownState, - objectFilterDropdownSearchInputState, - selectedFilterState, - selectFilter, - } = useFilterDropdown(); - - const filterDefinitionUsedInDropdown = useRecoilValue( - filterDefinitionUsedInDropdownState, - ); - const selectedOperandInDropdown = useRecoilValue( - selectedOperandInDropdownState, - ); - const objectFilterDropdownSearchInput = useRecoilValue( - objectFilterDropdownSearchInputState, - ); - const selectedFilter = useRecoilValue(selectedFilterState); - - const { closeDropdown } = useDropdown(OBJECT_FILTER_DROPDOWN_ID); - - const [isAllEntitySelected, setIsAllEntitySelected] = useState(false); - - const handleRecordSelected = ( - selectedEntity: EntityForSelect | null | undefined, - ) => { - if ( - !filterDefinitionUsedInDropdown || - !selectedOperandInDropdown || - !selectedEntity - ) { - return; - } - - if (isAllEntitySelected) { - setIsAllEntitySelected(false); - } - - setObjectFilterDropdownSelectedEntityId(selectedEntity.id); - - selectFilter?.({ - displayValue: selectedEntity.name, - fieldMetadataId: filterDefinitionUsedInDropdown.fieldMetadataId, - operand: selectedOperandInDropdown, - value: selectedEntity.id, - displayAvatarUrl: selectedEntity.avatarUrl, - definition: filterDefinitionUsedInDropdown, - }); - closeDropdown(); - }; - - const isAllEntitySelectShown = - !!filterDefinitionUsedInDropdown?.selectAllLabel && - !!filterDefinitionUsedInDropdown?.SelectAllIcon && - (isAllEntitySelected || - filterDefinitionUsedInDropdown?.selectAllLabel - .toLocaleLowerCase() - .includes(objectFilterDropdownSearchInput.toLocaleLowerCase())); - - const handleAllEntitySelectClick = () => { - if ( - !filterDefinitionUsedInDropdown || - !selectedOperandInDropdown || - !filterDefinitionUsedInDropdown.selectAllLabel - ) { - return; - } - - setIsAllEntitySelected(true); - setObjectFilterDropdownSelectedEntityId(null); - closeDropdown(); - - selectFilter?.({ - displayValue: filterDefinitionUsedInDropdown.selectAllLabel, - fieldMetadataId: filterDefinitionUsedInDropdown.fieldMetadataId, - operand: ViewFilterOperand.IsNotNull, - value: '', - definition: filterDefinitionUsedInDropdown, - }); - }; - - useEffect(() => { - if (!selectedFilter) { - setObjectFilterDropdownSelectedEntityId(null); - } else { - setObjectFilterDropdownSelectedEntityId(selectedFilter.value); - setIsAllEntitySelected( - selectedFilter.operand === ViewFilterOperand.IsNotNull, - ); - } - }, [ - selectedFilter, - setObjectFilterDropdownSelectedEntityId, - entitiesForSelect.selectedEntities, - ]); - - return ( - <SingleEntitySelectMenuItems - entitiesToSelect={entitiesForSelect.entitiesToSelect} - selectedEntity={entitiesForSelect.selectedEntities[0]} - loading={entitiesForSelect.loading} - onEntitySelected={handleRecordSelected} - SelectAllIcon={filterDefinitionUsedInDropdown?.SelectAllIcon} - selectAllLabel={filterDefinitionUsedInDropdown?.selectAllLabel} - isAllEntitySelected={isAllEntitySelected} - isAllEntitySelectShown={isAllEntitySelectShown} - onAllEntitySelected={handleAllEntitySelectClick} - /> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownNumberInput.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownNumberInput.tsx index d6cc92eaaf1f..b0ad3cc85386 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownNumberInput.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownNumberInput.tsx @@ -1,5 +1,6 @@ import { ChangeEvent } from 'react'; import { useRecoilValue } from 'recoil'; +import { v4 } from 'uuid'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; import { DropdownMenuInput } from '@/ui/layout/dropdown/components/DropdownMenuInput'; @@ -8,6 +9,7 @@ export const ObjectFilterDropdownNumberInput = () => { const { selectedOperandInDropdownState, filterDefinitionUsedInDropdownState, + selectedFilterState, selectFilter, } = useFilterDropdown(); @@ -18,6 +20,8 @@ export const ObjectFilterDropdownNumberInput = () => { selectedOperandInDropdownState, ); + const selectedFilter = useRecoilValue(selectedFilterState); + return ( filterDefinitionUsedInDropdown && selectedOperandInDropdown && ( @@ -27,6 +31,7 @@ export const ObjectFilterDropdownNumberInput = () => { placeholder={filterDefinitionUsedInDropdown.label} onChange={(event: ChangeEvent<HTMLInputElement>) => { selectFilter?.({ + id: selectedFilter?.id ? selectedFilter.id : v4(), fieldMetadataId: filterDefinitionUsedInDropdown.fieldMetadataId, value: event.target.value, operand: selectedOperandInDropdown, diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOperandSelect.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOperandSelect.tsx index d3f0ccccfd6a..5f500b916461 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOperandSelect.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOperandSelect.tsx @@ -1,6 +1,8 @@ import { useRecoilValue } from 'recoil'; +import { v4 } from 'uuid'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; +import { FilterDefinition } from '@/object-record/object-filter-dropdown/types/FilterDefinition'; import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer'; import { MenuItem } from '@/ui/navigation/menu-item/components/MenuItem'; import { ViewFilterOperand } from '@/views/types/ViewFilterOperand'; @@ -33,15 +35,33 @@ export const ObjectFilterDropdownOperandSelect = () => { filterDefinitionUsedInDropdown?.type, ); - const handleOperangeChange = (newOperand: ViewFilterOperand) => { + const handleOperandChange = (newOperand: ViewFilterOperand) => { + const isEmptyOperand = [ + ViewFilterOperand.IsEmpty, + ViewFilterOperand.IsNotEmpty, + ].includes(newOperand); + setSelectedOperandInDropdown(newOperand); setIsObjectFilterDropdownOperandSelectUnfolded(false); + if (isEmptyOperand) { + selectFilter?.({ + id: v4(), + fieldMetadataId: filterDefinitionUsedInDropdown?.fieldMetadataId ?? '', + displayValue: '', + operand: newOperand, + value: '', + definition: filterDefinitionUsedInDropdown as FilterDefinition, + }); + return; + } + if ( isDefined(filterDefinitionUsedInDropdown) && isDefined(selectedFilter) ) { selectFilter?.({ + id: selectedFilter.id ? selectedFilter.id : v4(), fieldMetadataId: selectedFilter.fieldMetadataId, displayValue: selectedFilter.displayValue, operand: newOperand, @@ -61,7 +81,7 @@ export const ObjectFilterDropdownOperandSelect = () => { <MenuItem key={`select-filter-operand-${index}`} onClick={() => { - handleOperangeChange(filterOperand); + handleOperandChange(filterOperand); }} text={getOperandLabel(filterOperand)} /> diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOptionSelect.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOptionSelect.tsx index 46b294b1ebd8..6180ff3d3ce7 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOptionSelect.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownOptionSelect.tsx @@ -1,5 +1,6 @@ import { useEffect, useState } from 'react'; import { useRecoilValue } from 'recoil'; +import { v4 } from 'uuid'; import { FieldMetadataItemOption } from '@/object-metadata/types/FieldMetadataItem'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; @@ -22,6 +23,7 @@ export const ObjectFilterDropdownOptionSelect = () => { objectFilterDropdownSearchInputState, selectedOperandInDropdownState, objectFilterDropdownSelectedOptionValuesState, + selectedFilterState, selectFilter, } = useFilterDropdown(); @@ -38,6 +40,8 @@ export const ObjectFilterDropdownOptionSelect = () => { objectFilterDropdownSelectedOptionValuesState, ); + const selectedFilter = useRecoilValue(selectedFilterState); + const fieldMetaDataId = filterDefinitionUsedInDropdown?.fieldMetadataId ?? ''; const { selectOptions } = useOptionsForSelect(fieldMetaDataId); @@ -96,6 +100,7 @@ export const ObjectFilterDropdownOptionSelect = () => { : EMPTY_FILTER_VALUE; selectFilter({ + id: selectedFilter?.id ? selectedFilter.id : v4(), definition: filterDefinitionUsedInDropdown, operand: selectedOperandInDropdown, displayValue: filterDisplayValue, diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownRecordSelect.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownRecordSelect.tsx index 65a2c3263dc3..a5cf8185455d 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownRecordSelect.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownRecordSelect.tsx @@ -1,25 +1,39 @@ +import { useState } from 'react'; import { useRecoilValue } from 'recoil'; +import { v4 } from 'uuid'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; import { MultipleRecordSelectDropdown } from '@/object-record/select/components/MultipleRecordSelectDropdown'; import { useRecordsForSelect } from '@/object-record/select/hooks/useRecordsForSelect'; import { SelectableRecord } from '@/object-record/select/types/SelectableRecord'; +import { useCombinedViewFilters } from '@/views/hooks/useCombinedViewFilters'; +import { useGetCurrentView } from '@/views/hooks/useGetCurrentView'; import { isDefined } from '~/utils/isDefined'; export const EMPTY_FILTER_VALUE = '[]'; export const MAX_RECORDS_TO_DISPLAY = 3; -export const ObjectFilterDropdownRecordSelect = () => { +type ObjectFilterDropdownRecordSelectProps = { + viewComponentId?: string; +}; +export const ObjectFilterDropdownRecordSelect = ({ + viewComponentId, +}: ObjectFilterDropdownRecordSelectProps) => { const { filterDefinitionUsedInDropdownState, objectFilterDropdownSearchInputState, selectedOperandInDropdownState, + selectedFilterState, setObjectFilterDropdownSelectedRecordIds, objectFilterDropdownSelectedRecordIdsState, selectFilter, emptyFilterButKeepDefinition, } = useFilterDropdown(); + const { removeCombinedViewFilter } = useCombinedViewFilters(viewComponentId); + const { currentViewWithCombinedFiltersAndSorts } = + useGetCurrentView(viewComponentId); + const filterDefinitionUsedInDropdown = useRecoilValue( filterDefinitionUsedInDropdownState, ); @@ -32,6 +46,9 @@ export const ObjectFilterDropdownRecordSelect = () => { const objectFilterDropdownSelectedRecordIds = useRecoilValue( objectFilterDropdownSelectedRecordIdsState, ); + const [fieldId] = useState(v4()); + + const selectedFilter = useRecoilValue(selectedFilterState); const objectNameSingular = filterDefinitionUsedInDropdown?.relationObjectMetadataNameSingular ?? ''; @@ -60,6 +77,7 @@ export const ObjectFilterDropdownRecordSelect = () => { if (newSelectedRecordIds.length === 0) { emptyFilterButKeepDefinition(); + removeCombinedViewFilter(fieldId); return; } @@ -91,7 +109,17 @@ export const ObjectFilterDropdownRecordSelect = () => { ? JSON.stringify(newSelectedRecordIds) : EMPTY_FILTER_VALUE; + const viewFilter = + currentViewWithCombinedFiltersAndSorts?.viewFilters.find( + (viewFilter) => + viewFilter.fieldMetadataId === + filterDefinitionUsedInDropdown.fieldMetadataId, + ); + + const filterId = viewFilter?.id ?? fieldId; + selectFilter({ + id: selectedFilter?.id ? selectedFilter.id : filterId, definition: filterDefinitionUsedInDropdown, operand: selectedOperandInDropdown, displayValue: filterDisplayValue, diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownTextSearchInput.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownTextSearchInput.tsx index 445d080ad699..f2e1bbeb9ce7 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownTextSearchInput.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/ObjectFilterDropdownTextSearchInput.tsx @@ -1,5 +1,6 @@ -import { ChangeEvent } from 'react'; +import { ChangeEvent, useState } from 'react'; import { useRecoilValue } from 'recoil'; +import { v4 } from 'uuid'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; import { DropdownMenuSearchInput } from '@/ui/layout/dropdown/components/DropdownMenuSearchInput'; @@ -14,6 +15,8 @@ export const ObjectFilterDropdownTextSearchInput = () => { selectFilter, } = useFilterDropdown(); + const [filterId] = useState(v4()); + const filterDefinitionUsedInDropdown = useRecoilValue( filterDefinitionUsedInDropdownState, ); @@ -37,6 +40,7 @@ export const ObjectFilterDropdownTextSearchInput = () => { setObjectFilterDropdownSearchInput(event.target.value); selectFilter?.({ + id: selectedFilter?.id ? selectedFilter.id : filterId, fieldMetadataId: filterDefinitionUsedInDropdown.fieldMetadataId, value: event.target.value, operand: selectedOperandInDropdown, diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/SingleEntityObjectFilterDropdownButton.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/SingleEntityObjectFilterDropdownButton.tsx index 7ddbdb5f5d59..5f57a990d365 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/SingleEntityObjectFilterDropdownButton.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/components/SingleEntityObjectFilterDropdownButton.tsx @@ -17,6 +17,8 @@ import { GenericEntityFilterChip } from './GenericEntityFilterChip'; import { ObjectFilterDropdownRecordSelect } from './ObjectFilterDropdownRecordSelect'; import { ObjectFilterDropdownSearchInput } from './ObjectFilterDropdownSearchInput'; +const SINGLE_ENTITY_FILTER_DROPDOWN_ID = 'single-entity-filter-dropdown'; + export const SingleEntityObjectFilterDropdownButton = ({ hotkeyScope, }: { @@ -50,7 +52,7 @@ export const SingleEntityObjectFilterDropdownButton = ({ return ( <Dropdown - dropdownId="single-entity-filter-dropdown" + dropdownId={SINGLE_ENTITY_FILTER_DROPDOWN_ID} dropdownHotkeyScope={hotkeyScope} dropdownOffset={{ x: 0, y: -28 }} clickableComponent={ @@ -75,7 +77,9 @@ export const SingleEntityObjectFilterDropdownButton = ({ <ObjectFilterDropdownSearchInput /> <DropdownMenuSeparator /> <ObjectFilterDropdownRecordRemoveFilterMenuItem /> - <ObjectFilterDropdownRecordSelect /> + <ObjectFilterDropdownRecordSelect + viewComponentId={SINGLE_ENTITY_FILTER_DROPDOWN_ID} + /> </> } /> diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/hooks/__tests__/useFilterDropdown.test.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/hooks/__tests__/useFilterDropdown.test.tsx index 311764d5892a..d6757cd310c4 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/hooks/__tests__/useFilterDropdown.test.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/hooks/__tests__/useFilterDropdown.test.tsx @@ -23,6 +23,7 @@ const filterDefinitions: FilterDefinition[] = [ ]; const mockFilter: Filter = { + id: 'id', definition: filterDefinitions[0], displayValue: '', fieldMetadataId: '', diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/types/Filter.ts b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/types/Filter.ts index bb72dfd46ae1..862e99bed5f4 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/types/Filter.ts +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/types/Filter.ts @@ -3,6 +3,7 @@ import { ViewFilterOperand } from '@/views/types/ViewFilterOperand'; import { FilterDefinition } from './FilterDefinition'; export type Filter = { + id: string; fieldMetadataId: string; value: string; displayValue: string; diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/__tests__/getOperandsForFilterType.test.tsx b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/__tests__/getOperandsForFilterType.test.tsx index 1644d9472545..d5eded8bf666 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/__tests__/getOperandsForFilterType.test.tsx +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/__tests__/getOperandsForFilterType.test.tsx @@ -4,20 +4,34 @@ import { ViewFilterOperand } from '@/views/types/ViewFilterOperand'; import { getOperandsForFilterType } from '../getOperandsForFilterType'; describe('getOperandsForFilterType', () => { + const emptyOperands = [ + ViewFilterOperand.IsEmpty, + ViewFilterOperand.IsNotEmpty, + ]; + + const containsOperands = [ + ViewFilterOperand.Contains, + ViewFilterOperand.DoesNotContain, + ]; + + const numberOperands = [ + ViewFilterOperand.GreaterThan, + ViewFilterOperand.LessThan, + ]; + + const relationOperand = [ViewFilterOperand.Is, ViewFilterOperand.IsNot]; + const testCases = [ - ['TEXT', [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain]], - ['EMAIL', [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain]], - [ - 'FULL_NAME', - [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain], - ], - ['ADDRESS', [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain]], - ['LINK', [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain]], - ['LINKS', [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain]], - ['CURRENCY', [ViewFilterOperand.GreaterThan, ViewFilterOperand.LessThan]], - ['NUMBER', [ViewFilterOperand.GreaterThan, ViewFilterOperand.LessThan]], - ['DATE_TIME', [ViewFilterOperand.GreaterThan, ViewFilterOperand.LessThan]], - ['RELATION', [ViewFilterOperand.Is, ViewFilterOperand.IsNot]], + ['TEXT', [...containsOperands, ...emptyOperands]], + ['EMAIL', [...containsOperands, ...emptyOperands]], + ['FULL_NAME', [...containsOperands, ...emptyOperands]], + ['ADDRESS', [...containsOperands, ...emptyOperands]], + ['LINK', [...containsOperands, ...emptyOperands]], + ['LINKS', [...containsOperands, ...emptyOperands]], + ['CURRENCY', [...numberOperands, ...emptyOperands]], + ['NUMBER', [...numberOperands, ...emptyOperands]], + ['DATE_TIME', [...numberOperands, ...emptyOperands]], + ['RELATION', [...relationOperand, ...emptyOperands]], [undefined, []], [null, []], ['UNKNOWN_TYPE', []], diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandLabel.ts b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandLabel.ts index a58bc9db53c8..9c9e297ef960 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandLabel.ts +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandLabel.ts @@ -18,6 +18,10 @@ export const getOperandLabel = ( return 'Is not'; case ViewFilterOperand.IsNotNull: return 'Is not null'; + case ViewFilterOperand.IsEmpty: + return 'Is empty'; + case ViewFilterOperand.IsNotEmpty: + return 'Is not empty'; default: return ''; } @@ -35,6 +39,10 @@ export const getOperandLabelShort = ( return ': Not'; case ViewFilterOperand.IsNotNull: return ': NotNull'; + case ViewFilterOperand.IsNotEmpty: + return ': NotEmpty'; + case ViewFilterOperand.IsEmpty: + return ': Empty'; case ViewFilterOperand.GreaterThan: return '\u00A0> '; case ViewFilterOperand.LessThan: diff --git a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandsForFilterType.ts b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandsForFilterType.ts index 147e6ee9ecd5..7f189009b41a 100644 --- a/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandsForFilterType.ts +++ b/packages/twenty-front/src/modules/object-record/object-filter-dropdown/utils/getOperandsForFilterType.ts @@ -5,6 +5,13 @@ import { FilterType } from '../types/FilterType'; export const getOperandsForFilterType = ( filterType: FilterType | null | undefined, ): ViewFilterOperand[] => { + const emptyOperands = [ + ViewFilterOperand.IsEmpty, + ViewFilterOperand.IsNotEmpty, + ]; + + const relationOperands = [ViewFilterOperand.Is, ViewFilterOperand.IsNot]; + switch (filterType) { case 'TEXT': case 'EMAIL': @@ -12,17 +19,25 @@ export const getOperandsForFilterType = ( case 'ADDRESS': case 'PHONE': case 'LINK': - return [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain]; case 'LINKS': - return [ViewFilterOperand.Contains, ViewFilterOperand.DoesNotContain]; + return [ + ViewFilterOperand.Contains, + ViewFilterOperand.DoesNotContain, + ...emptyOperands, + ]; case 'CURRENCY': case 'NUMBER': case 'DATE_TIME': case 'DATE': - return [ViewFilterOperand.GreaterThan, ViewFilterOperand.LessThan]; + return [ + ViewFilterOperand.GreaterThan, + ViewFilterOperand.LessThan, + ...emptyOperands, + ]; case 'RELATION': + return [...relationOperands, ...emptyOperands]; case 'SELECT': - return [ViewFilterOperand.Is, ViewFilterOperand.IsNot]; + return [...relationOperands]; default: return []; } diff --git a/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/__tests__/turnSortsIntoOrderBy.test.tsx b/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/__tests__/turnSortsIntoOrderBy.test.tsx index 0abde78e58e3..8534f11cabc3 100644 --- a/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/__tests__/turnSortsIntoOrderBy.test.tsx +++ b/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/__tests__/turnSortsIntoOrderBy.test.tsx @@ -30,9 +30,11 @@ describe('turnSortsIntoOrderBy', () => { it('should sort by recordPosition if no sorts', () => { const fields = [{ id: 'field1', name: 'createdAt' }] as FieldMetadataItem[]; expect(turnSortsIntoOrderBy({ ...objectMetadataItem, fields }, [])).toEqual( - { - position: 'AscNullsFirst', - }, + [ + { + position: 'AscNullsFirst', + }, + ], ); }); @@ -47,10 +49,7 @@ describe('turnSortsIntoOrderBy', () => { const fields = [{ id: 'field1', name: 'field1' }] as FieldMetadataItem[]; expect( turnSortsIntoOrderBy({ ...objectMetadataItem, fields }, sorts), - ).toEqual({ - field1: 'AscNullsFirst', - position: 'AscNullsFirst', - }); + ).toEqual([{ field1: 'AscNullsFirst' }, { position: 'AscNullsFirst' }]); }); it('should create OrderByField with multiple sorts', () => { @@ -72,11 +71,11 @@ describe('turnSortsIntoOrderBy', () => { ] as FieldMetadataItem[]; expect( turnSortsIntoOrderBy({ ...objectMetadataItem, fields }, sorts), - ).toEqual({ - field1: 'AscNullsFirst', - field2: 'DescNullsLast', - position: 'AscNullsFirst', - }); + ).toEqual([ + { field1: 'AscNullsFirst' }, + { field2: 'DescNullsLast' }, + { position: 'AscNullsFirst' }, + ]); }); it('should ignore if field not found', () => { @@ -87,9 +86,9 @@ describe('turnSortsIntoOrderBy', () => { definition: sortDefinition, }, ]; - expect(turnSortsIntoOrderBy(objectMetadataItem, sorts)).toEqual({ - position: 'AscNullsFirst', - }); + expect(turnSortsIntoOrderBy(objectMetadataItem, sorts)).toEqual([ + { position: 'AscNullsFirst' }, + ]); }); it('should not return position for remotes', () => { @@ -102,6 +101,6 @@ describe('turnSortsIntoOrderBy', () => { ]; expect( turnSortsIntoOrderBy({ ...objectMetadataItem, isRemote: true }, sorts), - ).toEqual({}); + ).toEqual([]); }); }); diff --git a/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/turnSortsIntoOrderBy.ts b/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/turnSortsIntoOrderBy.ts index b30f245707f6..04e877109434 100644 --- a/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/turnSortsIntoOrderBy.ts +++ b/packages/twenty-front/src/modules/object-record/object-sort-dropdown/utils/turnSortsIntoOrderBy.ts @@ -2,7 +2,7 @@ import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; import { OrderBy } from '@/object-metadata/types/OrderBy'; import { hasPositionField } from '@/object-metadata/utils/hasPositionField'; import { RecordGqlOperationOrderBy } from '@/object-record/graphql/types/RecordGqlOperationOrderBy'; -import { Field } from '~/generated/graphql'; +import { Field, FieldMetadataType } from '~/generated/graphql'; import { mapArrayToObject } from '~/utils/array/mapArrayToObject'; import { isDefined } from '~/utils/isDefined'; import { isUndefinedOrNull } from '~/utils/isUndefinedOrNull'; @@ -13,31 +13,45 @@ export const turnSortsIntoOrderBy = ( objectMetadataItem: ObjectMetadataItem, sorts: Sort[], ): RecordGqlOperationOrderBy => { - const fields: Pick<Field, 'id' | 'name'>[] = objectMetadataItem?.fields ?? []; + const fields: Pick<Field, 'id' | 'name' | 'type'>[] = + objectMetadataItem?.fields ?? []; const fieldsById = mapArrayToObject(fields, ({ id }) => id); - const sortsOrderBy = Object.fromEntries( - sorts - .map((sort) => { - const correspondingField = fieldsById[sort.fieldMetadataId]; + const sortsOrderBy = sorts + .map((sort) => { + const correspondingField = fieldsById[sort.fieldMetadataId]; - if (isUndefinedOrNull(correspondingField)) { - return undefined; - } + if (isUndefinedOrNull(correspondingField)) { + return undefined; + } - const direction: OrderBy = - sort.direction === 'asc' ? 'AscNullsFirst' : 'DescNullsLast'; + const direction: OrderBy = + sort.direction === 'asc' ? 'AscNullsFirst' : 'DescNullsLast'; - return [correspondingField.name, direction]; - }) - .filter(isDefined), - ); + return getOrderByForFieldMetadataType(correspondingField, direction); + }) + .filter(isDefined); if (hasPositionField(objectMetadataItem)) { - return { - ...sortsOrderBy, - position: 'AscNullsFirst', - }; + return [...sortsOrderBy, { position: 'AscNullsFirst' }]; } return sortsOrderBy; }; + +const getOrderByForFieldMetadataType = ( + field: Pick<Field, 'id' | 'name' | 'type'>, + direction: OrderBy, +) => { + switch (field.type) { + case FieldMetadataType.FullName: + return { + [field.name]: { + firstName: direction, + lastName: direction, + }, + }; + + default: + return { [field.name]: direction }; + } +}; diff --git a/packages/twenty-front/src/modules/object-record/record-action-bar/hooks/useRecordActionBar.tsx b/packages/twenty-front/src/modules/object-record/record-action-bar/hooks/useRecordActionBar.tsx index 3333ead34719..490467dae757 100644 --- a/packages/twenty-front/src/modules/object-record/record-action-bar/hooks/useRecordActionBar.tsx +++ b/packages/twenty-front/src/modules/object-record/record-action-bar/hooks/useRecordActionBar.tsx @@ -1,5 +1,5 @@ -import { useCallback, useMemo, useState } from 'react'; import { isNonEmptyString } from '@sniptt/guards'; +import { useCallback, useMemo, useState } from 'react'; import { useRecoilCallback, useSetRecoilState } from 'recoil'; import { IconClick, @@ -13,8 +13,9 @@ import { import { useFavorites } from '@/favorites/hooks/useFavorites'; import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; -import { useDeleteManyRecords } from '@/object-record/hooks/useDeleteManyRecords'; +import { DELETE_MAX_COUNT } from '@/object-record/constants/DeleteMaxCount'; import { useExecuteQuickActionOnOneRecord } from '@/object-record/hooks/useExecuteQuickActionOnOneRecord'; +import { useDeleteTableData } from '@/object-record/record-index/options/hooks/useDeleteTableData'; import { displayedExportProgress, useExportTableData, @@ -31,12 +32,14 @@ type useRecordActionBarProps = { objectMetadataItem: ObjectMetadataItem; selectedRecordIds: string[]; callback?: () => void; + totalNumberOfRecordsSelected?: number; }; export const useRecordActionBar = ({ objectMetadataItem, selectedRecordIds, callback, + totalNumberOfRecordsSelected, }: useRecordActionBarProps) => { const setContextMenuEntries = useSetRecoilState(contextMenuEntriesState); const setActionBarEntriesState = useSetRecoilState(actionBarEntriesState); @@ -45,10 +48,6 @@ export const useRecordActionBar = ({ const { createFavorite, favorites, deleteFavorite } = useFavorites(); - const { deleteManyRecords } = useDeleteManyRecords({ - objectNameSingular: objectMetadataItem.nameSingular, - }); - const { executeQuickActionOnOneRecord } = useExecuteQuickActionOnOneRecord({ objectNameSingular: objectMetadataItem.nameSingular, }); @@ -88,24 +87,17 @@ export const useRecordActionBar = ({ ], ); - const handleDeleteClick = useCallback(async () => { - callback?.(); - selectedRecordIds.forEach((recordId) => { - const foundFavorite = favorites?.find( - (favorite) => favorite.recordId === recordId, - ); - if (foundFavorite !== undefined) { - deleteFavorite(foundFavorite.id); - } - }); - await deleteManyRecords(selectedRecordIds); - }, [ - callback, - deleteManyRecords, - selectedRecordIds, - favorites, - deleteFavorite, - ]); + const baseTableDataParams = { + delayMs: 100, + objectNameSingular: objectMetadataItem.nameSingular, + recordIndexId: objectMetadataItem.namePlural, + }; + + const { deleteTableData } = useDeleteTableData(baseTableDataParams); + + const handleDeleteClick = useCallback(() => { + deleteTableData(); + }, [deleteTableData]); const handleExecuteQuickActionOnClick = useCallback(async () => { callback?.(); @@ -117,56 +109,63 @@ export const useRecordActionBar = ({ }, [callback, executeQuickActionOnOneRecord, selectedRecordIds]); const { progress, download } = useExportTableData({ - delayMs: 100, + ...baseTableDataParams, filename: `${objectMetadataItem.nameSingular}.csv`, - objectNameSingular: objectMetadataItem.nameSingular, - recordIndexId: objectMetadataItem.namePlural, }); const isRemoteObject = objectMetadataItem.isRemote; - const baseActions: ContextMenuEntry[] = useMemo( - () => [ - { - label: displayedExportProgress(progress), - Icon: IconFileExport, - accent: 'default', - onClick: () => download(), - }, - ], - [download, progress], - ); - - const deletionActions: ContextMenuEntry[] = useMemo( - () => [ - { - label: 'Delete', - Icon: IconTrash, - accent: 'danger', - onClick: () => setIsDeleteRecordsModalOpen(true), - ConfirmationModal: ( - <ConfirmationModal - isOpen={isDeleteRecordsModalOpen} - setIsOpen={setIsDeleteRecordsModalOpen} - title={`Delete ${selectedRecordIds.length} ${ - selectedRecordIds.length === 1 ? `record` : 'records' - }`} - subtitle={`This action cannot be undone. This will permanently delete ${ - selectedRecordIds.length === 1 ? 'this record' : 'these records' - }`} - onConfirmClick={() => handleDeleteClick()} - deleteButtonText={`Delete ${ - selectedRecordIds.length > 1 ? 'Records' : 'Record' - }`} - /> - ), - }, - ], + const numberOfSelectedRecords = + totalNumberOfRecordsSelected ?? selectedRecordIds.length; + const canDelete = + !isRemoteObject && numberOfSelectedRecords < DELETE_MAX_COUNT; + + const menuActions: ContextMenuEntry[] = useMemo( + () => + [ + { + label: displayedExportProgress(progress), + Icon: IconFileExport, + accent: 'default', + onClick: () => download(), + } satisfies ContextMenuEntry, + canDelete + ? ({ + label: 'Delete', + Icon: IconTrash, + accent: 'danger', + onClick: () => { + setIsDeleteRecordsModalOpen(true); + handleDeleteClick(); + }, + ConfirmationModal: ( + <ConfirmationModal + isOpen={isDeleteRecordsModalOpen} + setIsOpen={setIsDeleteRecordsModalOpen} + title={`Delete ${numberOfSelectedRecords} ${ + numberOfSelectedRecords === 1 ? `record` : 'records' + }`} + subtitle={`This action cannot be undone. This will permanently delete ${ + numberOfSelectedRecords === 1 + ? 'this record' + : 'these records' + }`} + onConfirmClick={() => handleDeleteClick()} + deleteButtonText={`Delete ${ + numberOfSelectedRecords > 1 ? 'Records' : 'Record' + }`} + /> + ), + } satisfies ContextMenuEntry) + : undefined, + ].filter(isDefined), [ + download, + progress, + canDelete, handleDeleteClick, - selectedRecordIds, isDeleteRecordsModalOpen, - setIsDeleteRecordsModalOpen, + numberOfSelectedRecords, ], ); @@ -183,8 +182,7 @@ export const useRecordActionBar = ({ return { setContextMenuEntries: useCallback(() => { setContextMenuEntries([ - ...(isRemoteObject ? [] : deletionActions), - ...baseActions, + ...menuActions, ...(!isRemoteObject && isFavorite && hasOnlyOneRecordSelected ? [ { @@ -205,8 +203,7 @@ export const useRecordActionBar = ({ : []), ]); }, [ - baseActions, - deletionActions, + menuActions, handleFavoriteButtonClick, hasOnlyOneRecordSelected, isFavorite, @@ -235,15 +232,12 @@ export const useRecordActionBar = ({ }, ] : []), - ...(isRemoteObject ? [] : deletionActions), - ...baseActions, + ...menuActions, ]); }, [ - baseActions, + menuActions, dataExecuteQuickActionOnmentEnabled, - deletionActions, handleExecuteQuickActionOnClick, - isRemoteObject, setActionBarEntriesState, ]), }; diff --git a/packages/twenty-front/src/modules/object-record/record-board/components/RecordBoard.tsx b/packages/twenty-front/src/modules/object-record/record-board/components/RecordBoard.tsx index 23aa8612882b..d6ffe57f2e22 100644 --- a/packages/twenty-front/src/modules/object-record/record-board/components/RecordBoard.tsx +++ b/packages/twenty-front/src/modules/object-record/record-board/components/RecordBoard.tsx @@ -1,6 +1,6 @@ -import { useContext, useRef } from 'react'; import styled from '@emotion/styled'; import { DragDropContext, OnDragEndResponder } from '@hello-pangea/dnd'; // Atlassian dnd does not support StrictMode from RN 18, so we use a fork @hello-pangea/dnd https://github.com/atlassian/react-beautiful-dnd/issues/2350 +import { useContext, useRef } from 'react'; import { useRecoilCallback, useRecoilValue } from 'recoil'; import { Key } from 'ts-key-enum'; diff --git a/packages/twenty-front/src/modules/object-record/record-board/record-board-card/components/RecordBoardCard.tsx b/packages/twenty-front/src/modules/object-record/record-board/record-board-card/components/RecordBoardCard.tsx index 9b695a3f3c83..a551db4cb58f 100644 --- a/packages/twenty-front/src/modules/object-record/record-board/record-board-card/components/RecordBoardCard.tsx +++ b/packages/twenty-front/src/modules/object-record/record-board/record-board-card/components/RecordBoardCard.tsx @@ -1,10 +1,9 @@ +import styled from '@emotion/styled'; import { ReactNode, useContext, useState } from 'react'; import { useInView } from 'react-intersection-observer'; -import styled from '@emotion/styled'; import { useRecoilState, useRecoilValue, useSetRecoilState } from 'recoil'; -import { EntityChipVariant, IconEye } from 'twenty-ui'; +import { AvatarChipVariant, IconEye } from 'twenty-ui'; -import { RecordChip } from '@/object-record/components/RecordChip'; import { RecordBoardContext } from '@/object-record/record-board/contexts/RecordBoardContext'; import { useRecordBoardStates } from '@/object-record/record-board/hooks/internal/useRecordBoardStates'; import { RecordBoardCardContext } from '@/object-record/record-board/record-board-card/contexts/RecordBoardCardContext'; @@ -14,6 +13,7 @@ import { RecordUpdateHookParams, } from '@/object-record/record-field/contexts/FieldContext'; import { getFieldButtonIcon } from '@/object-record/record-field/utils/getFieldButtonIcon'; +import { RecordIndexRecordChip } from '@/object-record/record-index/components/RecordIndexRecordChip'; import { RecordInlineCell } from '@/object-record/record-inline-cell/components/RecordInlineCell'; import { InlineCellHotkeyScope } from '@/object-record/record-inline-cell/types/InlineCellHotkeyScope'; import { RecordValueSetterEffect } from '@/object-record/record-store/components/RecordValueSetterEffect'; @@ -66,7 +66,7 @@ const StyledBoardCardWrapper = styled.div` width: 100%; `; -const StyledBoardCardHeader = styled.div<{ +export const StyledBoardCardHeader = styled.div<{ showCompactView: boolean; }>` align-items: center; @@ -89,7 +89,7 @@ const StyledBoardCardHeader = styled.div<{ } `; -const StyledBoardCardBody = styled.div` +export const StyledBoardCardBody = styled.div` display: flex; flex-direction: column; gap: ${({ theme }) => theme.spacing(0.5)}; @@ -117,6 +117,7 @@ const StyledFieldContainer = styled.div` display: flex; flex-direction: row; width: fit-content; + max-width: 100%; `; const StyledCompactIconContainer = styled.div` @@ -221,10 +222,10 @@ export const RecordBoardCard = () => { }} > <StyledBoardCardHeader showCompactView={isCompactModeActive}> - <RecordChip + <RecordIndexRecordChip objectNameSingular={objectMetadataItem.nameSingular} record={record} - variant={EntityChipVariant.Transparent} + variant={AvatarChipVariant.Transparent} /> {isCompactModeActive && ( <StyledCompactIconContainer className="compact-icon-container"> @@ -262,7 +263,7 @@ export const RecordBoardCard = () => { recoilScopeId: recordId + fieldDefinition.fieldMetadataId, isLabelIdentifier: false, fieldDefinition: { - disableTooltip: true, + disableTooltip: false, fieldMetadataId: fieldDefinition.fieldMetadataId, label: fieldDefinition.label, iconName: fieldDefinition.iconName, diff --git a/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnCardContainerSkeletonLoader.tsx b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnCardContainerSkeletonLoader.tsx new file mode 100644 index 000000000000..4aeafd96f1fd --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnCardContainerSkeletonLoader.tsx @@ -0,0 +1,61 @@ +import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; +import { useTheme } from '@emotion/react'; +import styled from '@emotion/styled'; + +import { + StyledBoardCardBody, + StyledBoardCardHeader, +} from '@/object-record/record-board/record-board-card/components/RecordBoardCard'; + +const StyledSkeletonIconAndText = styled.div` + display: flex; + gap: ${({ theme }) => theme.spacing(1)}; +`; + +const StyledSkeletonTitle = styled.div` + padding-left: ${({ theme }) => theme.spacing(2)}; +`; + +const StyledSeparator = styled.div` + height: ${({ theme }) => theme.spacing(2)}; +`; + +export const RecordBoardColumnCardContainerSkeletonLoader = ({ + numberOfFields, + titleSkeletonWidth, + isCompactModeActive, +}: { + numberOfFields: number; + titleSkeletonWidth: number; + isCompactModeActive: boolean; +}) => { + const theme = useTheme(); + const skeletonItems = Array.from({ length: numberOfFields }).map( + (_, index) => ({ + id: `skeleton-item-${index}`, + }), + ); + return ( + <SkeletonTheme + baseColor={theme.background.tertiary} + highlightColor={theme.background.transparent.lighter} + borderRadius={4} + > + <StyledBoardCardHeader showCompactView={isCompactModeActive}> + <StyledSkeletonTitle> + <Skeleton width={titleSkeletonWidth} height={16} /> + </StyledSkeletonTitle> + </StyledBoardCardHeader> + <StyledSeparator /> + {!isCompactModeActive && + skeletonItems.map(({ id }) => ( + <StyledBoardCardBody key={id}> + <StyledSkeletonIconAndText> + <Skeleton width={16} height={16} /> + <Skeleton width={151} height={16} /> + </StyledSkeletonIconAndText> + </StyledBoardCardBody> + ))} + </SkeletonTheme> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnCardsContainer.tsx b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnCardsContainer.tsx index 96d14a011bd4..79c786df1225 100644 --- a/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnCardsContainer.tsx +++ b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnCardsContainer.tsx @@ -1,14 +1,19 @@ import React, { useContext } from 'react'; import styled from '@emotion/styled'; import { Draggable, DroppableProvided } from '@hello-pangea/dnd'; +import { useRecoilValue } from 'recoil'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { RecordBoardContext } from '@/object-record/record-board/contexts/RecordBoardContext'; +import { useRecordBoardStates } from '@/object-record/record-board/hooks/internal/useRecordBoardStates'; +import { RecordBoardColumnCardContainerSkeletonLoader } from '@/object-record/record-board/record-board-column/components/RecordBoardColumnCardContainerSkeletonLoader'; import { RecordBoardColumnCardsMemo } from '@/object-record/record-board/record-board-column/components/RecordBoardColumnCardsMemo'; import { RecordBoardColumnFetchMoreLoader } from '@/object-record/record-board/record-board-column/components/RecordBoardColumnFetchMoreLoader'; import { RecordBoardColumnNewButton } from '@/object-record/record-board/record-board-column/components/RecordBoardColumnNewButton'; import { RecordBoardColumnNewOpportunityButton } from '@/object-record/record-board/record-board-column/components/RecordBoardColumnNewOpportunityButton'; import { RecordBoardColumnContext } from '@/object-record/record-board/record-board-column/contexts/RecordBoardColumnContext'; +import { getNumberOfCardsPerColumnForSkeletonLoading } from '@/object-record/record-board/record-board-column/utils/getNumberOfCardsPerColumnForSkeletonLoading'; +import { isRecordIndexBoardColumnLoadingFamilyState } from '@/object-record/states/isRecordBoardColumnLoadingFamilyState'; const StyledColumnCardsContainer = styled.div` display: flex; @@ -20,6 +25,17 @@ const StyledNewButtonContainer = styled.div` padding-bottom: ${({ theme }) => theme.spacing(4)}; `; +const StyledSkeletonCardContainer = styled.div` + background-color: ${({ theme }) => theme.background.secondary}; + border: 1px solid ${({ theme }) => theme.background.quaternary}; + border-radius: ${({ theme }) => theme.border.radius.md}; + box-shadow: + 0px 4px 8px 0px rgba(0, 0, 0, 0.08), + 0px 0px 4px 0px rgba(0, 0, 0, 0.08); + color: ${({ theme }) => theme.font.color.primary}; + margin-bottom: ${({ theme }) => theme.spacing(2)}; +`; + type RecordBoardColumnCardsContainerProps = { recordIds: string[]; droppableProvided: DroppableProvided; @@ -32,13 +48,51 @@ export const RecordBoardColumnCardsContainer = ({ const { columnDefinition } = useContext(RecordBoardColumnContext); const { objectMetadataItem } = useContext(RecordBoardContext); + const columnId = columnDefinition.id; + + const isRecordIndexBoardColumnLoading = useRecoilValue( + isRecordIndexBoardColumnLoadingFamilyState(columnId), + ); + + const { isCompactModeActiveState, visibleFieldDefinitionsState } = + useRecordBoardStates(); + + const visibleFieldDefinitions = useRecoilValue( + visibleFieldDefinitionsState(), + ); + + const numberOfFields = visibleFieldDefinitions.length; + + const isCompactModeActive = useRecoilValue(isCompactModeActiveState); + return ( <StyledColumnCardsContainer ref={droppableProvided?.innerRef} // eslint-disable-next-line react/jsx-props-no-spreading {...droppableProvided?.droppableProps} > - <RecordBoardColumnCardsMemo recordIds={recordIds} /> + {isRecordIndexBoardColumnLoading ? ( + Array.from( + { + length: getNumberOfCardsPerColumnForSkeletonLoading( + columnDefinition.position, + ), + }, + (_, index) => ( + <StyledSkeletonCardContainer + key={`${columnDefinition.id}-${index}`} + > + <RecordBoardColumnCardContainerSkeletonLoader + numberOfFields={numberOfFields} + titleSkeletonWidth={isCompactModeActive ? 72 : 54} + isCompactModeActive={isCompactModeActive} + /> + </StyledSkeletonCardContainer> + ), + ) + ) : ( + <RecordBoardColumnCardsMemo recordIds={recordIds} /> + )} <RecordBoardColumnFetchMoreLoader /> <Draggable draggableId={`new-${columnDefinition.id}`} diff --git a/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnHeader.tsx b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnHeader.tsx index e4f59a85a238..a4c76e5bcfbd 100644 --- a/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnHeader.tsx +++ b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/components/RecordBoardColumnHeader.tsx @@ -1,10 +1,11 @@ -import React, { useContext, useState } from 'react'; import styled from '@emotion/styled'; +import { useContext, useState } from 'react'; import { IconDotsVertical, Tag } from 'twenty-ui'; import { RecordBoardColumnDropdownMenu } from '@/object-record/record-board/record-board-column/components/RecordBoardColumnDropdownMenu'; import { RecordBoardColumnContext } from '@/object-record/record-board/record-board-column/contexts/RecordBoardColumnContext'; import { RecordBoardColumnHotkeyScope } from '@/object-record/record-board/types/BoardColumnHotkeyScope'; +import { RecordBoardColumnDefinitionType } from '@/object-record/record-board/types/RecordBoardColumnDefinition'; import { LightIconButton } from '@/ui/input/button/components/LightIconButton'; import { usePreviousHotkeyScope } from '@/ui/utilities/hotkey/hooks/usePreviousHotkeyScope'; @@ -79,14 +80,23 @@ export const RecordBoardColumnHeader = () => { > <Tag onClick={handleBoardColumnMenuOpen} - color={columnDefinition.color} + variant={ + columnDefinition.type === RecordBoardColumnDefinitionType.Value + ? 'solid' + : 'outline' + } + color={ + columnDefinition.type === RecordBoardColumnDefinitionType.Value + ? columnDefinition.color + : 'transparent' + } text={columnDefinition.title} /> {!!boardColumnTotal && <StyledAmount>${boardColumnTotal}</StyledAmount>} {!isHeaderHovered && ( <StyledNumChildren>{recordCount}</StyledNumChildren> )} - {isHeaderHovered && ( + {isHeaderHovered && columnDefinition.actions.length > 0 && ( <StyledHeaderActions> <LightIconButton accent="tertiary" @@ -96,7 +106,7 @@ export const RecordBoardColumnHeader = () => { </StyledHeaderActions> )} </StyledHeader> - {isBoardColumnMenuOpen && ( + {isBoardColumnMenuOpen && columnDefinition.actions.length > 0 && ( <RecordBoardColumnDropdownMenu onClose={handleBoardColumnMenuClose} stageId={columnDefinition.id} diff --git a/packages/twenty-front/src/modules/object-record/record-board/record-board-column/utils/getNumberOfCardsPerColumnForSkeletonLoading.ts b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/utils/getNumberOfCardsPerColumnForSkeletonLoading.ts new file mode 100644 index 000000000000..1f6fc58aaa95 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-board/record-board-column/utils/getNumberOfCardsPerColumnForSkeletonLoading.ts @@ -0,0 +1,13 @@ +export const getNumberOfCardsPerColumnForSkeletonLoading = ( + columnIndex: number, +): number => { + const skeletonCounts: Record<number, number> = { + 0: 2, + 1: 1, + 2: 3, + 3: 0, + 4: 1, + }; + + return skeletonCounts[columnIndex] || 0; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-board/types/RecordBoardColumnDefinition.ts b/packages/twenty-front/src/modules/object-record/record-board/types/RecordBoardColumnDefinition.ts index 5a9197dff42a..b5e443b0fd9a 100644 --- a/packages/twenty-front/src/modules/object-record/record-board/types/RecordBoardColumnDefinition.ts +++ b/packages/twenty-front/src/modules/object-record/record-board/types/RecordBoardColumnDefinition.ts @@ -2,11 +2,30 @@ import { ThemeColor } from 'twenty-ui'; import { RecordBoardColumnAction } from '@/object-record/record-board/types/RecordBoardColumnAction'; -export type RecordBoardColumnDefinition = { +export const enum RecordBoardColumnDefinitionType { + Value = 'value', + NoValue = 'no-value', +} + +export type RecordBoardColumnDefinitionNoValue = { + id: 'no-value'; + type: RecordBoardColumnDefinitionType.NoValue; + title: 'No Value'; + position: number; + value: null; + actions: RecordBoardColumnAction[]; +}; + +export type RecordBoardColumnDefinitionValue = { id: string; + type: RecordBoardColumnDefinitionType.Value; title: string; value: string; - position: number; color: ThemeColor; + position: number; actions: RecordBoardColumnAction[]; }; + +export type RecordBoardColumnDefinition = + | RecordBoardColumnDefinitionValue + | RecordBoardColumnDefinitionNoValue; diff --git a/packages/twenty-front/src/modules/object-record/record-field/__mocks__/fieldDefinitions.ts b/packages/twenty-front/src/modules/object-record/record-field/__mocks__/fieldDefinitions.ts index 9ea73c15dfc3..4be023caf378 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/__mocks__/fieldDefinitions.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/__mocks__/fieldDefinitions.ts @@ -7,7 +7,6 @@ import { FieldSelectMetadata, FieldTextMetadata, } from '@/object-record/record-field/types/FieldMetadata'; -import { type } from 'os'; import { FieldMetadataType } from '~/generated-metadata/graphql'; import { mockedCompanyObjectMetadataItem, @@ -44,6 +43,7 @@ export const selectFieldDefinition: FieldDefinition<FieldSelectMetadata> = { metadata: { fieldName: 'accountOwner', options: [{ label: 'Elon Musk', color: 'blue', value: 'userId' }], + isNullable: true, }, }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/components/FieldDisplay.tsx b/packages/twenty-front/src/modules/object-record/record-field/components/FieldDisplay.tsx index 31a7692f21e5..f3448846d214 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/components/FieldDisplay.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/components/FieldDisplay.tsx @@ -2,11 +2,15 @@ import { useContext } from 'react'; import { BooleanFieldDisplay } from '@/object-record/record-field/meta-types/display/components/BooleanFieldDisplay'; import { LinksFieldDisplay } from '@/object-record/record-field/meta-types/display/components/LinksFieldDisplay'; +import { RatingFieldDisplay } from '@/object-record/record-field/meta-types/display/components/RatingFieldDisplay'; +import { RelationFromManyFieldDisplay } from '@/object-record/record-field/meta-types/display/components/RelationFromManyFieldDisplay'; +import { isFieldChipDisplay } from '@/object-record/record-field/meta-types/display/utils/isFieldChipDisplay'; import { isFieldBoolean } from '@/object-record/record-field/types/guards/isFieldBoolean'; import { isFieldDisplayedAsPhone } from '@/object-record/record-field/types/guards/isFieldDisplayedAsPhone'; import { isFieldLinks } from '@/object-record/record-field/types/guards/isFieldLinks'; -import { isFieldChipDisplay } from '@/object-record/utils/getRecordChipGeneratorPerObjectPerField'; - +import { isFieldRating } from '@/object-record/record-field/types/guards/isFieldRating'; +import { isFieldRelationFromManyObjects } from '@/object-record/record-field/types/guards/isFieldRelationFromManyObjects'; +import { isFieldRelationToOneObject } from '@/object-record/record-field/types/guards/isFieldRelationToOneObject'; import { FieldContext } from '../contexts/FieldContext'; import { AddressFieldDisplay } from '../meta-types/display/components/AddressFieldDisplay'; import { ChipFieldDisplay } from '../meta-types/display/components/ChipFieldDisplay'; @@ -20,7 +24,7 @@ import { LinkFieldDisplay } from '../meta-types/display/components/LinkFieldDisp import { MultiSelectFieldDisplay } from '../meta-types/display/components/MultiSelectFieldDisplay'; import { NumberFieldDisplay } from '../meta-types/display/components/NumberFieldDisplay'; import { PhoneFieldDisplay } from '../meta-types/display/components/PhoneFieldDisplay'; -import { RelationFieldDisplay } from '../meta-types/display/components/RelationFieldDisplay'; +import { RelationToOneFieldDisplay } from '../meta-types/display/components/RelationToOneFieldDisplay'; import { SelectFieldDisplay } from '../meta-types/display/components/SelectFieldDisplay'; import { TextFieldDisplay } from '../meta-types/display/components/TextFieldDisplay'; import { UuidFieldDisplay } from '../meta-types/display/components/UuidFieldDisplay'; @@ -35,7 +39,6 @@ import { isFieldMultiSelect } from '../types/guards/isFieldMultiSelect'; import { isFieldNumber } from '../types/guards/isFieldNumber'; import { isFieldPhone } from '../types/guards/isFieldPhone'; import { isFieldRawJson } from '../types/guards/isFieldRawJson'; -import { isFieldRelation } from '../types/guards/isFieldRelation'; import { isFieldSelect } from '../types/guards/isFieldSelect'; import { isFieldText } from '../types/guards/isFieldText'; import { isFieldUuid } from '../types/guards/isFieldUuid'; @@ -47,8 +50,10 @@ export const FieldDisplay = () => { return isChipDisplay ? ( <ChipFieldDisplay /> - ) : isFieldRelation(fieldDefinition) ? ( - <RelationFieldDisplay /> + ) : isFieldRelationToOneObject(fieldDefinition) ? ( + <RelationToOneFieldDisplay /> + ) : isFieldRelationFromManyObjects(fieldDefinition) ? ( + <RelationFromManyFieldDisplay /> ) : isFieldPhone(fieldDefinition) || isFieldDisplayedAsPhone(fieldDefinition) ? ( <PhoneFieldDisplay /> @@ -82,5 +87,7 @@ export const FieldDisplay = () => { <JsonFieldDisplay /> ) : isFieldBoolean(fieldDefinition) ? ( <BooleanFieldDisplay /> + ) : isFieldRating(fieldDefinition) ? ( + <RatingFieldDisplay /> ) : null; }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/components/FieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/components/FieldInput.tsx index ce1240e6449c..5ac286d7e745 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/components/FieldInput.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/components/FieldInput.tsx @@ -6,7 +6,7 @@ import { FullNameFieldInput } from '@/object-record/record-field/meta-types/inpu import { LinksFieldInput } from '@/object-record/record-field/meta-types/input/components/LinksFieldInput'; import { MultiSelectFieldInput } from '@/object-record/record-field/meta-types/input/components/MultiSelectFieldInput'; import { RawJsonFieldInput } from '@/object-record/record-field/meta-types/input/components/RawJsonFieldInput'; -import { RelationManyFieldInput } from '@/object-record/record-field/meta-types/input/components/RelationManyFieldInput'; +import { RelationFromManyFieldInput } from '@/object-record/record-field/meta-types/input/components/RelationFromManyFieldInput'; import { SelectFieldInput } from '@/object-record/record-field/meta-types/input/components/SelectFieldInput'; import { RecordFieldInputScope } from '@/object-record/record-field/scopes/RecordFieldInputScope'; import { isFieldDate } from '@/object-record/record-field/types/guards/isFieldDate'; @@ -16,6 +16,7 @@ import { isFieldLinks } from '@/object-record/record-field/types/guards/isFieldL import { isFieldMultiSelect } from '@/object-record/record-field/types/guards/isFieldMultiSelect'; import { isFieldRawJson } from '@/object-record/record-field/types/guards/isFieldRawJson'; import { isFieldRelationFromManyObjects } from '@/object-record/record-field/types/guards/isFieldRelationFromManyObjects'; +import { isFieldRelationToOneObject } from '@/object-record/record-field/types/guards/isFieldRelationToOneObject'; import { isFieldSelect } from '@/object-record/record-field/types/guards/isFieldSelect'; import { getScopeIdFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdFromComponentId'; @@ -28,7 +29,7 @@ import { LinkFieldInput } from '../meta-types/input/components/LinkFieldInput'; import { NumberFieldInput } from '../meta-types/input/components/NumberFieldInput'; import { PhoneFieldInput } from '../meta-types/input/components/PhoneFieldInput'; import { RatingFieldInput } from '../meta-types/input/components/RatingFieldInput'; -import { RelationFieldInput } from '../meta-types/input/components/RelationFieldInput'; +import { RelationToOneFieldInput } from '../meta-types/input/components/RelationToOneFieldInput'; import { TextFieldInput } from '../meta-types/input/components/TextFieldInput'; import { FieldInputEvent } from '../types/FieldInputEvent'; import { isFieldAddress } from '../types/guards/isFieldAddress'; @@ -40,7 +41,6 @@ import { isFieldLink } from '../types/guards/isFieldLink'; import { isFieldNumber } from '../types/guards/isFieldNumber'; import { isFieldPhone } from '../types/guards/isFieldPhone'; import { isFieldRating } from '../types/guards/isFieldRating'; -import { isFieldRelation } from '../types/guards/isFieldRelation'; import { isFieldText } from '../types/guards/isFieldText'; type FieldInputProps = { @@ -72,16 +72,10 @@ export const FieldInput = ({ <RecordFieldInputScope recordFieldInputScopeId={getScopeIdFromComponentId(recordFieldInputdId)} > - {isFieldRelation(fieldDefinition) ? ( - isFieldRelationFromManyObjects(fieldDefinition) ? ( - <RelationManyFieldInput - relationPickerScopeId={getScopeIdFromComponentId( - `relation-picker-${fieldDefinition.fieldMetadataId}`, - )} - /> - ) : ( - <RelationFieldInput onSubmit={onSubmit} onCancel={onCancel} /> - ) + {isFieldRelationToOneObject(fieldDefinition) ? ( + <RelationToOneFieldInput onSubmit={onSubmit} onCancel={onCancel} /> + ) : isFieldRelationFromManyObjects(fieldDefinition) ? ( + <RelationFromManyFieldInput onSubmit={onSubmit} /> ) : isFieldPhone(fieldDefinition) || isFieldDisplayedAsPhone(fieldDefinition) ? ( <PhoneFieldInput @@ -121,6 +115,7 @@ export const FieldInput = ({ onEscape={onEscape} onClickOutside={onClickOutside} onClear={onSubmit} + onSubmit={onSubmit} /> ) : isFieldDate(fieldDefinition) ? ( <DateFieldInput @@ -128,6 +123,7 @@ export const FieldInput = ({ onEscape={onEscape} onClickOutside={onClickOutside} onClear={onSubmit} + onSubmit={onSubmit} /> ) : isFieldNumber(fieldDefinition) ? ( <NumberFieldInput diff --git a/packages/twenty-front/src/modules/object-record/record-field/contexts/FieldContext.ts b/packages/twenty-front/src/modules/object-record/record-field/contexts/FieldContext.ts index 824e7a3b5f74..571e39cbe9e4 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/contexts/FieldContext.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/contexts/FieldContext.ts @@ -30,6 +30,7 @@ export type GenericFieldContextType = { clearable?: boolean; maxWidth?: number; isCentered?: boolean; + overridenIsFieldEmpty?: boolean; }; export const FieldContext = createContext<GenericFieldContextType>( diff --git a/packages/twenty-front/src/modules/object-record/record-field/hooks/__tests__/useToggleEditOnlyInput.test.tsx b/packages/twenty-front/src/modules/object-record/record-field/hooks/__tests__/useToggleEditOnlyInput.test.tsx index 3a073219f9b6..3000ca0eb5b5 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/hooks/__tests__/useToggleEditOnlyInput.test.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/hooks/__tests__/useToggleEditOnlyInput.test.tsx @@ -40,7 +40,16 @@ const mocks: MockedResponse[] = [ currencyCode } createdAt - address + address { + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng + } updatedAt name accountOwnerId diff --git a/packages/twenty-front/src/modules/object-record/record-field/hooks/useClearField.ts b/packages/twenty-front/src/modules/object-record/record-field/hooks/useClearField.ts index 548e212f98c7..30577442254f 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/hooks/useClearField.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/hooks/useClearField.ts @@ -2,6 +2,7 @@ import { useContext } from 'react'; import { useRecoilCallback } from 'recoil'; import { objectMetadataItemsState } from '@/object-metadata/states/objectMetadataItemsState'; +import { useSetRecordFieldValue } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; import { recordStoreFamilySelector } from '@/object-record/record-store/states/selectors/recordStoreFamilySelector'; import { generateEmptyFieldValue } from '@/object-record/utils/generateEmptyFieldValue'; @@ -16,6 +17,8 @@ export const useClearField = () => { const [updateRecord] = useUpdateRecord(); + const setRecordFieldValue = useSetRecordFieldValue(); + const clearField = useRecoilCallback( ({ snapshot, set }) => () => { @@ -46,6 +49,8 @@ export const useClearField = () => { emptyFieldValue, ); + setRecordFieldValue(entityId, fieldName, emptyFieldValue); + updateRecord?.({ variables: { where: { id: entityId }, @@ -55,7 +60,7 @@ export const useClearField = () => { }, }); }, - [entityId, fieldDefinition, updateRecord], + [entityId, fieldDefinition, updateRecord, setRecordFieldValue], ); return clearField; diff --git a/packages/twenty-front/src/modules/object-record/record-field/hooks/useIsFieldEmpty.ts b/packages/twenty-front/src/modules/object-record/record-field/hooks/useIsFieldEmpty.ts index 01c4573c0000..f1e100566e72 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/hooks/useIsFieldEmpty.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/hooks/useIsFieldEmpty.ts @@ -2,17 +2,23 @@ import { useContext } from 'react'; import { isFieldValueEmpty } from '@/object-record/record-field/utils/isFieldValueEmpty'; import { useRecordFieldValue } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; +import { isDefined } from '~/utils/isDefined'; import { FieldContext } from '../contexts/FieldContext'; export const useIsFieldEmpty = () => { - const { entityId, fieldDefinition } = useContext(FieldContext); + const { entityId, fieldDefinition, overridenIsFieldEmpty } = + useContext(FieldContext); const fieldValue = useRecordFieldValue( entityId, - fieldDefinition.metadata.fieldName, + fieldDefinition?.metadata?.fieldName ?? '', ); + if (isDefined(overridenIsFieldEmpty)) { + return overridenIsFieldEmpty; + } + return isFieldValueEmpty({ fieldDefinition, fieldValue, diff --git a/packages/twenty-front/src/modules/object-record/record-field/hooks/usePersistField.ts b/packages/twenty-front/src/modules/object-record/record-field/hooks/usePersistField.ts index d2f1a1db8bca..946b6046dc10 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/hooks/usePersistField.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/hooks/usePersistField.ts @@ -15,7 +15,8 @@ import { isFieldMultiSelect } from '@/object-record/record-field/types/guards/is import { isFieldMultiSelectValue } from '@/object-record/record-field/types/guards/isFieldMultiSelectValue'; import { isFieldRawJson } from '@/object-record/record-field/types/guards/isFieldRawJson'; import { isFieldRawJsonValue } from '@/object-record/record-field/types/guards/isFieldRawJsonValue'; -import { isFieldRelationFromManyObjects } from '@/object-record/record-field/types/guards/isFieldRelationFromManyObjects'; +import { isFieldRelationToOneObject } from '@/object-record/record-field/types/guards/isFieldRelationToOneObject'; +import { isFieldRelationToOneValue } from '@/object-record/record-field/types/guards/isFieldRelationToOneValue'; import { isFieldSelect } from '@/object-record/record-field/types/guards/isFieldSelect'; import { isFieldSelectValue } from '@/object-record/record-field/types/guards/isFieldSelectValue'; import { recordStoreFamilySelector } from '@/object-record/record-store/states/selectors/recordStoreFamilySelector'; @@ -38,8 +39,6 @@ import { isFieldPhone } from '../types/guards/isFieldPhone'; import { isFieldPhoneValue } from '../types/guards/isFieldPhoneValue'; import { isFieldRating } from '../types/guards/isFieldRating'; import { isFieldRatingValue } from '../types/guards/isFieldRatingValue'; -import { isFieldRelation } from '../types/guards/isFieldRelation'; -import { isFieldRelationValue } from '../types/guards/isFieldRelationValue'; import { isFieldText } from '../types/guards/isFieldText'; import { isFieldTextValue } from '../types/guards/isFieldTextValue'; @@ -55,14 +54,10 @@ export const usePersistField = () => { const persistField = useRecoilCallback( ({ set }) => (valueToPersist: unknown) => { - const fieldIsRelation = - isFieldRelation(fieldDefinition) && - isFieldRelationValue(valueToPersist); - - const fieldIsRelationFromManyObjects = - isFieldRelationFromManyObjects( + const fieldIsRelationToOneObject = + isFieldRelationToOneObject( fieldDefinition as FieldDefinition<FieldRelationMetadata>, - ) && isFieldRelationValue(valueToPersist); + ) && isFieldRelationToOneValue(valueToPersist); const fieldIsText = isFieldText(fieldDefinition) && isFieldTextValue(valueToPersist); @@ -87,7 +82,7 @@ export const usePersistField = () => { isFieldBoolean(fieldDefinition) && isFieldBooleanValue(valueToPersist); - const fieldIsProbability = + const fieldIsRating = isFieldRating(fieldDefinition) && isFieldRatingValue(valueToPersist); const fieldIsNumber = @@ -120,11 +115,11 @@ export const usePersistField = () => { isFieldRawJsonValue(valueToPersist); const isValuePersistable = - (fieldIsRelation && !fieldIsRelationFromManyObjects) || + fieldIsRelationToOneObject || fieldIsText || fieldIsBoolean || fieldIsEmail || - fieldIsProbability || + fieldIsRating || fieldIsNumber || fieldIsDateTime || fieldIsDate || @@ -145,7 +140,7 @@ export const usePersistField = () => { valueToPersist, ); - if (fieldIsRelation && !fieldIsRelationFromManyObjects) { + if (fieldIsRelationToOneObject) { const value = valueToPersist as EntityForSelect; updateRecord?.({ variables: { diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/ChipFieldDisplay.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/ChipFieldDisplay.tsx index 30fdcb7a7774..4a8682f3859e 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/ChipFieldDisplay.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/ChipFieldDisplay.tsx @@ -1,24 +1,21 @@ -import { EntityChip } from 'twenty-ui'; - +import { RecordChip } from '@/object-record/components/RecordChip'; import { useChipFieldDisplay } from '@/object-record/record-field/meta-types/hooks/useChipFieldDisplay'; -import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; +import { RecordIndexRecordChip } from '@/object-record/record-index/components/RecordIndexRecordChip'; export const ChipFieldDisplay = () => { - const { recordValue, generateRecordChipData } = useChipFieldDisplay(); + const { recordValue, objectNameSingular, isLabelIdentifier } = + useChipFieldDisplay(); if (!recordValue) { return null; } - const recordChipData = generateRecordChipData(recordValue); - - return ( - <EntityChip - entityId={recordValue.id} - name={recordChipData.name as any} - avatarType={recordChipData.avatarType} - avatarUrl={getImageAbsoluteURIOrBase64(recordChipData.avatarUrl) ?? ''} - linkToEntity={recordChipData.linkToShowPage} + return isLabelIdentifier ? ( + <RecordIndexRecordChip + objectNameSingular={objectNameSingular} + record={recordValue} /> + ) : ( + <RecordChip objectNameSingular={objectNameSingular} record={recordValue} /> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RatingFieldDisplay.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RatingFieldDisplay.tsx new file mode 100644 index 000000000000..7514480e4b00 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RatingFieldDisplay.tsx @@ -0,0 +1,8 @@ +import { useRatingFieldDisplay } from '@/object-record/record-field/meta-types/hooks/useRatingFieldDisplay'; +import { RatingInput } from '@/ui/field/input/components/RatingInput'; + +export const RatingFieldDisplay = () => { + const { rating } = useRatingFieldDisplay(); + + return <RatingInput value={rating} readonly />; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationFieldDisplay.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationFieldDisplay.tsx deleted file mode 100644 index dace0123781c..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationFieldDisplay.tsx +++ /dev/null @@ -1,38 +0,0 @@ -import { isArray } from '@sniptt/guards'; -import { EntityChip } from 'twenty-ui'; - -import { RelationFromManyFieldDisplay } from '@/object-record/record-field/meta-types/display/components/RelationFromManyFieldDisplay'; -import { useRelationFieldDisplay } from '@/object-record/record-field/meta-types/hooks/useRelationFieldDisplay'; -import { isFieldRelationFromManyObjects } from '@/object-record/record-field/types/guards/isFieldRelationFromManyObjects'; -import { ObjectRecord } from '@/object-record/types/ObjectRecord'; -import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; - -export const RelationFieldDisplay = () => { - const { fieldValue, fieldDefinition, generateRecordChipData } = - useRelationFieldDisplay(); - - if ( - !fieldValue || - !fieldDefinition?.metadata.relationObjectMetadataNameSingular - ) { - return null; - } - - if (isArray(fieldValue) && isFieldRelationFromManyObjects(fieldDefinition)) { - return ( - <RelationFromManyFieldDisplay fieldValue={fieldValue as ObjectRecord[]} /> - ); - } - - const recordChipData = generateRecordChipData(fieldValue); - - return ( - <EntityChip - entityId={fieldValue.id} - name={recordChipData.name as any} - avatarType={recordChipData.avatarType} - avatarUrl={getImageAbsoluteURIOrBase64(recordChipData.avatarUrl) || ''} - linkToEntity={recordChipData.linkToShowPage} - /> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationFromManyFieldDisplay.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationFromManyFieldDisplay.tsx index 808565a5a932..adfaea988958 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationFromManyFieldDisplay.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationFromManyFieldDisplay.tsx @@ -1,34 +1,27 @@ -import { EntityChip } from 'twenty-ui'; - +import { RecordChip } from '@/object-record/components/RecordChip'; import { useFieldFocus } from '@/object-record/record-field/hooks/useFieldFocus'; -import { useRelationFieldDisplay } from '@/object-record/record-field/meta-types/hooks/useRelationFieldDisplay'; -import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { useRelationFromManyFieldDisplay } from '@/object-record/record-field/meta-types/hooks/useRelationFromManyFieldDisplay'; import { ExpandableList } from '@/ui/layout/expandable-list/components/ExpandableList'; -import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; -export const RelationFromManyFieldDisplay = ({ - fieldValue, -}: { - fieldValue: ObjectRecord[]; -}) => { +export const RelationFromManyFieldDisplay = () => { + const { fieldValue, fieldDefinition } = useRelationFromManyFieldDisplay(); const { isFocused } = useFieldFocus(); - const { generateRecordChipData } = useRelationFieldDisplay(); - const recordChipsData = fieldValue.map((fieldValueItem) => - generateRecordChipData(fieldValueItem), - ); + const relationObjectNameSingular = + fieldDefinition?.metadata.relationObjectMetadataNameSingular; + + if (!fieldValue || !relationObjectNameSingular) { + return null; + } return ( <ExpandableList isChipCountDisplayed={isFocused}> - {recordChipsData.map((record) => { + {fieldValue.map((record) => { return ( - <EntityChip - key={record.recordId} - entityId={record.recordId} - name={record.name as any} - avatarType={record.avatarType} - avatarUrl={getImageAbsoluteURIOrBase64(record.avatarUrl) || ''} - linkToEntity={record.linkToShowPage} + <RecordChip + key={record.id} + objectNameSingular={relationObjectNameSingular} + record={record} /> ); })} diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationToOneFieldDisplay.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationToOneFieldDisplay.tsx new file mode 100644 index 000000000000..dede0f879534 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/RelationToOneFieldDisplay.tsx @@ -0,0 +1,24 @@ +import { RecordChip } from '@/object-record/components/RecordChip'; +import { useRelationToOneFieldDisplay } from '@/object-record/record-field/meta-types/hooks/useRelationToOneFieldDisplay'; + +export const RelationToOneFieldDisplay = () => { + const { fieldValue, fieldDefinition, generateRecordChipData } = + useRelationToOneFieldDisplay(); + + if ( + !fieldValue || + !fieldDefinition?.metadata.relationObjectMetadataNameSingular + ) { + return null; + } + + const recordChipData = generateRecordChipData(fieldValue); + + return ( + <RecordChip + key={recordChipData.recordId} + objectNameSingular={recordChipData.objectNameSingular} + record={fieldValue} + /> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/AddressFieldDisplay.perf.stories.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/AddressFieldDisplay.perf.stories.tsx index 6c138c79548e..f97418a5af23 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/AddressFieldDisplay.perf.stories.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/AddressFieldDisplay.perf.stories.tsx @@ -49,6 +49,6 @@ export const Elipsis: Story = { export const Performance = getProfilingStory({ componentName: 'AddressFieldDisplay', averageThresholdInMs: 0.15, - numberOfRuns: 50, + numberOfRuns: 20, numberOfTestsPerRun: 100, }); diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RatingFieldDisplay.perf.stories.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RatingFieldDisplay.perf.stories.tsx new file mode 100644 index 000000000000..cffa101426cf --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RatingFieldDisplay.perf.stories.tsx @@ -0,0 +1,34 @@ +import { Meta, StoryObj } from '@storybook/react'; +import { ComponentDecorator } from 'twenty-ui'; + +import { RatingFieldDisplay } from '@/object-record/record-field/meta-types/display/components/RatingFieldDisplay'; +import { getFieldDecorator } from '~/testing/decorators/getFieldDecorator'; +import { MemoryRouterDecorator } from '~/testing/decorators/MemoryRouterDecorator'; +import { getProfilingStory } from '~/testing/profiling/utils/getProfilingStory'; + +const meta: Meta = { + title: 'UI/Data/Field/Display/RatingFieldDisplay', + decorators: [ + MemoryRouterDecorator, + getFieldDecorator('person', 'testRating'), + ComponentDecorator, + ], + component: RatingFieldDisplay, + args: {}, + parameters: { + chromatic: { disableSnapshot: true }, + }, +}; + +export default meta; + +type Story = StoryObj<typeof RatingFieldDisplay>; + +export const Default: Story = {}; + +export const Performance = getProfilingStory({ + componentName: 'RatingFieldDisplay', + averageThresholdInMs: 0.5, + numberOfRuns: 50, + numberOfTestsPerRun: 100, +}); diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationFromManyFieldDisplay.perf.stories.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationFromManyFieldDisplay.perf.stories.tsx index 7c9ce09da874..f4f2a3393b3a 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationFromManyFieldDisplay.perf.stories.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationFromManyFieldDisplay.perf.stories.tsx @@ -9,7 +9,7 @@ import { FieldDefinition } from '@/object-record/record-field/types/FieldDefinit import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; import { RecordFieldValueSelectorContextProvider, - useSetRecordValue, + useSetRecordFieldValue, } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; import { recordStoreFamilyState } from '@/object-record/record-store/states/recordStoreFamilyState'; import { ChipGeneratorsDecorator } from '~/testing/decorators/ChipGeneratorsDecorator'; @@ -18,6 +18,7 @@ import { getProfilingStory } from '~/testing/profiling/utils/getProfilingStory'; import { fieldValue, + otherPersonMock, relationFromManyFieldDisplayMock, } from './relationFromManyFieldDisplayMock'; @@ -30,21 +31,26 @@ const RelationFieldValueSetterEffect = () => { recordStoreFamilyState(relationFromManyFieldDisplayMock.relationEntityId), ); - const setRecordValue = useSetRecordValue(); + const setRecordFieldValue = useSetRecordFieldValue(); useEffect(() => { setEntity(relationFromManyFieldDisplayMock.entityValue); setRelationEntity(relationFromManyFieldDisplayMock.relationFieldValue); - setRecordValue( + setRecordFieldValue( relationFromManyFieldDisplayMock.entityValue.id, - relationFromManyFieldDisplayMock.entityValue, + 'company', + [relationFromManyFieldDisplayMock.entityValue], ); - setRecordValue( + setRecordFieldValue(otherPersonMock.entityValue.id, 'company', [ + relationFromManyFieldDisplayMock.entityValue, + ]); + setRecordFieldValue( relationFromManyFieldDisplayMock.relationFieldValue.id, + 'company', relationFromManyFieldDisplayMock.relationFieldValue, ); - }, [setEntity, setRelationEntity, setRecordValue]); + }, [setEntity, setRelationEntity, setRecordFieldValue]); return null; }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationFieldDisplay.perf.stories.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationToOneFieldDisplay.perf.stories.tsx similarity index 80% rename from packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationFieldDisplay.perf.stories.tsx rename to packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationToOneFieldDisplay.perf.stories.tsx index 6c985926d94e..49a076d80a47 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationFieldDisplay.perf.stories.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/RelationToOneFieldDisplay.perf.stories.tsx @@ -1,7 +1,7 @@ import { Meta, StoryObj } from '@storybook/react'; import { ComponentDecorator } from 'twenty-ui'; -import { RelationFieldDisplay } from '@/object-record/record-field/meta-types/display/components/RelationFieldDisplay'; +import { RelationToOneFieldDisplay } from '@/object-record/record-field/meta-types/display/components/RelationToOneFieldDisplay'; import { ChipGeneratorsDecorator } from '~/testing/decorators/ChipGeneratorsDecorator'; import { getFieldDecorator } from '~/testing/decorators/getFieldDecorator'; import { MemoryRouterDecorator } from '~/testing/decorators/MemoryRouterDecorator'; @@ -15,7 +15,7 @@ const meta: Meta = { getFieldDecorator('person', 'company'), ComponentDecorator, ], - component: RelationFieldDisplay, + component: RelationToOneFieldDisplay, args: {}, parameters: { chromatic: { disableSnapshot: true }, @@ -24,7 +24,7 @@ const meta: Meta = { export default meta; -type Story = StoryObj<typeof RelationFieldDisplay>; +type Story = StoryObj<typeof RelationToOneFieldDisplay>; export const Default: Story = {}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/relationFromManyFieldDisplayMock.ts b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/relationFromManyFieldDisplayMock.ts index cbc63cb40be5..674d00fe90e2 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/relationFromManyFieldDisplayMock.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/components/__stories__/perf/relationFromManyFieldDisplayMock.ts @@ -59,6 +59,60 @@ export const fieldValue = [ }, ]; +export const otherPersonMock = { + entityValue: { + __typename: 'Person', + asd: '', + city: 'Paris', + jobTitle: '', + name: 'John Doe', + createdAt: '2024-05-01T13:16:29.046Z', + company: { + __typename: 'Company', + domainName: 'google.com', + xLink: { + __typename: 'Link', + label: '', + url: '', + }, + name: 'Google', + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, + employees: null, + accountOwnerId: null, + address: '', + idealCustomerProfile: false, + createdAt: '2024-05-01T13:16:29.046Z', + id: '20202020-c21e-4ec2-873b-de4264d89025', + position: 6, + updatedAt: '2024-05-01T13:16:29.046Z', + linkedinLink: { + __typename: 'Link', + label: '', + url: '', + }, + }, + id: 'd3e70589-c449-4e64-8268-065640fdaff0', + email: 'john.doe@google.com', + phone: '+33744332211', + linkedinLink: { + __typename: 'Link', + label: '', + url: '', + }, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, + tEst: '', + position: 14, + }, +}; + export const relationFromManyFieldDisplayMock = { entityId: '20202020-2d40-4e49-8df4-9c6a049191df', relationEntityId: '20202020-c21e-4ec2-873b-de4264d89025', @@ -67,11 +121,7 @@ export const relationFromManyFieldDisplayMock = { asd: '', city: 'Seattle', jobTitle: '', - name: { - __typename: 'FullName', - firstName: 'Lorie', - lastName: 'Vladim', - }, + name: 'Lorie Vladim', createdAt: '2024-05-01T13:16:29.046Z', company: { __typename: 'Company', diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/utils/isFieldChipDisplay.ts b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/utils/isFieldChipDisplay.ts new file mode 100644 index 000000000000..cb45decfe7c3 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/display/utils/isFieldChipDisplay.ts @@ -0,0 +1,11 @@ +import { FieldMetadataItem } from '@/object-metadata/types/FieldMetadataItem'; +import { isFieldFullName } from '@/object-record/record-field/types/guards/isFieldFullName'; +import { isFieldNumber } from '@/object-record/record-field/types/guards/isFieldNumber'; +import { isFieldText } from '@/object-record/record-field/types/guards/isFieldText'; + +export const isFieldChipDisplay = ( + field: Pick<FieldMetadataItem, 'type'>, + isLabelIdentifier: boolean, +) => + isLabelIdentifier && + (isFieldText(field) || isFieldFullName(field) || isFieldNumber(field)); diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useChipFieldDisplay.ts b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useChipFieldDisplay.ts index 29cf1d3dc651..c0840bf7c06a 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useChipFieldDisplay.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useChipFieldDisplay.ts @@ -1,8 +1,7 @@ -import { useContext } from 'react'; import { isNonEmptyString } from '@sniptt/guards'; +import { useContext } from 'react'; import { PreComputedChipGeneratorsContext } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; -import { generateDefaultRecordChipData } from '@/object-metadata/utils/generateDefaultRecordChipData'; import { isFieldFullName } from '@/object-record/record-field/types/guards/isFieldFullName'; import { isFieldNumber } from '@/object-record/record-field/types/guards/isFieldNumber'; import { isFieldText } from '@/object-record/record-field/types/guards/isFieldText'; @@ -12,7 +11,8 @@ import { isDefined } from '~/utils/isDefined'; import { FieldContext } from '../../contexts/FieldContext'; export const useChipFieldDisplay = () => { - const { entityId, fieldDefinition } = useContext(FieldContext); + const { entityId, fieldDefinition, isLabelIdentifier } = + useContext(FieldContext); const { chipGeneratorPerObjectPerField } = useContext( PreComputedChipGeneratorsContext, @@ -31,18 +31,13 @@ export const useChipFieldDisplay = () => { const recordValue = useRecordValue(entityId); - if (!isNonEmptyString(fieldDefinition.metadata.objectMetadataNameSingular)) { + if (!isNonEmptyString(objectNameSingular)) { throw new Error('Object metadata name singular is not a non-empty string'); } - const generateRecordChipData = - chipGeneratorPerObjectPerField[ - fieldDefinition.metadata.objectMetadataNameSingular - ]?.[fieldDefinition.metadata.fieldName] ?? generateDefaultRecordChipData; - return { objectNameSingular, recordValue, - generateRecordChipData, + isLabelIdentifier, }; }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRatingFieldDisplay.ts b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRatingFieldDisplay.ts new file mode 100644 index 000000000000..f6d0ff6603f2 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRatingFieldDisplay.ts @@ -0,0 +1,23 @@ +import { useContext } from 'react'; + +import { FieldRatingValue } from '@/object-record/record-field/types/FieldMetadata'; +import { useRecordFieldValue } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; + +import { FieldContext } from '../../contexts/FieldContext'; + +export const useRatingFieldDisplay = () => { + const { entityId, fieldDefinition } = useContext(FieldContext); + + const fieldName = fieldDefinition.metadata.fieldName; + + const fieldValue = useRecordFieldValue(entityId, fieldName) as + | FieldRatingValue + | undefined; + + const rating = fieldValue ?? 'RATING_1'; + + return { + fieldDefinition, + rating, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationFromManyFieldDisplay.ts b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationFromManyFieldDisplay.ts new file mode 100644 index 000000000000..251852afd987 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationFromManyFieldDisplay.ts @@ -0,0 +1,63 @@ +import { useContext } from 'react'; +import { isNonEmptyString } from '@sniptt/guards'; + +import { PreComputedChipGeneratorsContext } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; +import { generateDefaultRecordChipData } from '@/object-metadata/utils/generateDefaultRecordChipData'; +import { useRecordFieldValue } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { FIELD_EDIT_BUTTON_WIDTH } from '@/ui/field/display/constants/FieldEditButtonWidth'; +import { FieldMetadataType } from '~/generated-metadata/graphql'; +import { isDefined } from '~/utils/isDefined'; + +import { FieldContext } from '../../contexts/FieldContext'; +import { assertFieldMetadata } from '../../types/guards/assertFieldMetadata'; +import { isFieldRelation } from '../../types/guards/isFieldRelation'; + +export const useRelationFromManyFieldDisplay = () => { + const { entityId, fieldDefinition, maxWidth } = useContext(FieldContext); + + const { chipGeneratorPerObjectPerField } = useContext( + PreComputedChipGeneratorsContext, + ); + + if (!isDefined(chipGeneratorPerObjectPerField)) { + throw new Error('Chip generator per object per field is not defined'); + } + + assertFieldMetadata( + FieldMetadataType.Relation, + isFieldRelation, + fieldDefinition, + ); + + const button = fieldDefinition.editButtonIcon; + + const fieldName = fieldDefinition.metadata.fieldName; + + const fieldValue = useRecordFieldValue<ObjectRecord[] | undefined>( + entityId, + fieldName, + ); + + const maxWidthForField = + isDefined(button) && isDefined(maxWidth) + ? maxWidth - FIELD_EDIT_BUTTON_WIDTH + : maxWidth; + + if (!isNonEmptyString(fieldDefinition.metadata.objectMetadataNameSingular)) { + throw new Error('Object metadata name singular is not a non-empty string'); + } + + const generateRecordChipData = + chipGeneratorPerObjectPerField[ + fieldDefinition.metadata.objectMetadataNameSingular + ]?.[fieldDefinition.metadata.fieldName] ?? generateDefaultRecordChipData; + + return { + fieldDefinition, + fieldValue, + maxWidth: maxWidthForField, + entityId, + generateRecordChipData, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationFieldDisplay.ts b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationToOneFieldDisplay.ts similarity index 97% rename from packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationFieldDisplay.ts rename to packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationToOneFieldDisplay.ts index 510cd7fbc2f4..c33464dc37b3 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationFieldDisplay.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/hooks/useRelationToOneFieldDisplay.ts @@ -13,7 +13,7 @@ import { FieldContext } from '../../contexts/FieldContext'; import { assertFieldMetadata } from '../../types/guards/assertFieldMetadata'; import { isFieldRelation } from '../../types/guards/isFieldRelation'; -export const useRelationFieldDisplay = () => { +export const useRelationToOneFieldDisplay = () => { const { entityId, fieldDefinition, maxWidth } = useContext(FieldContext); const { chipGeneratorPerObjectPerField } = useContext( diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateFieldInput.tsx index acc4dfff175b..4776642e91cd 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateFieldInput.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateFieldInput.tsx @@ -13,6 +13,7 @@ export type DateFieldInputProps = { onEnter?: FieldInputEvent; onEscape?: FieldInputEvent; onClear?: FieldInputEvent; + onSubmit?: FieldInputEvent; }; export const DateFieldInput = ({ @@ -20,6 +21,7 @@ export const DateFieldInput = ({ onEscape, onClickOutside, onClear, + onSubmit, }: DateFieldInputProps) => { const { fieldValue, setDraftValue } = useDateField(); @@ -39,6 +41,10 @@ export const DateFieldInput = ({ onEnter?.(() => persistDate(newDate)); }; + const handleSubmit = (newDate: Nullable<Date>) => { + onSubmit?.(() => persistDate(newDate)); + }; + const handleEscape = (newDate: Nullable<Date>) => { onEscape?.(() => persistDate(newDate)); }; @@ -69,6 +75,7 @@ export const DateFieldInput = ({ clearable onChange={handleChange} onClear={handleClear} + onSubmit={handleSubmit} /> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateTimeFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateTimeFieldInput.tsx index 15c6280ee9ac..6a57490084ac 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateTimeFieldInput.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/DateTimeFieldInput.tsx @@ -12,6 +12,7 @@ export type DateTimeFieldInputProps = { onEnter?: FieldInputEvent; onEscape?: FieldInputEvent; onClear?: FieldInputEvent; + onSubmit?: FieldInputEvent; }; export const DateTimeFieldInput = ({ @@ -19,6 +20,7 @@ export const DateTimeFieldInput = ({ onEscape, onClickOutside, onClear, + onSubmit, }: DateTimeFieldInputProps) => { const { fieldValue, setDraftValue } = useDateTimeField(); @@ -57,6 +59,10 @@ export const DateTimeFieldInput = ({ onClear?.(() => persistDate(null)); }; + const handleSubmit = (newDate: Nullable<Date>) => { + onSubmit?.(() => persistDate(newDate)); + }; + const dateValue = fieldValue ? new Date(fieldValue) : null; return ( @@ -69,6 +75,7 @@ export const DateTimeFieldInput = ({ onChange={handleChange} isDateTimeInput onClear={handleClear} + onSubmit={handleSubmit} /> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldInput.tsx index 9724bc3e6836..3fdbb0abf5f1 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldInput.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldInput.tsx @@ -124,6 +124,33 @@ export const LinksFieldInput = ({ onCancel }: LinksFieldInputProps) => { }; const handleDeleteLink = (index: number) => { + const hasOnlyOneLastLink = links.length === 1; + + if (hasOnlyOneLastLink) { + persistLinksField({ + primaryLinkUrl: '', + primaryLinkLabel: '', + secondaryLinks: null, + }); + + handleDropdownClose(); + + return; + } + + const isRemovingPrimary = index === 0; + if (isRemovingPrimary) { + const [, nextPrimaryLink, ...nextSecondaryLinks] = links; + + persistLinksField({ + primaryLinkUrl: nextPrimaryLink.url ?? '', + primaryLinkLabel: nextPrimaryLink.label ?? '', + secondaryLinks: nextSecondaryLinks, + }); + + return; + } + persistLinksField({ ...fieldValue, secondaryLinks: toSpliced(fieldValue.secondaryLinks ?? [], index - 1, 1), @@ -151,7 +178,7 @@ export const LinksFieldInput = ({ onCancel }: LinksFieldInputProps) => { <DropdownMenuSeparator /> </> )} - {isInputDisplayed ? ( + {isInputDisplayed || !links.length ? ( <DropdownMenuInput autoFocus placeholder="URL" diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldMenuItem.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldMenuItem.tsx index 39be332373aa..5b1fedb88486 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldMenuItem.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/LinksFieldMenuItem.tsx @@ -46,6 +46,11 @@ export const LinksFieldMenuItem = ({ const handleMouseEnter = () => setIsHovered(true); const handleMouseLeave = () => setIsHovered(false); + const handleDeleteClick = () => { + setIsHovered(false); + onDelete?.(); + }; + // Make sure dropdown closes on unmount. useEffect(() => { if (isDropdownOpen) { @@ -86,14 +91,12 @@ export const LinksFieldMenuItem = ({ text="Edit" onClick={onEdit} /> - {!isPrimary && ( - <MenuItem - accent="danger" - LeftIcon={IconTrash} - text="Delete" - onClick={onDelete} - /> - )} + <MenuItem + accent="danger" + LeftIcon={IconTrash} + text="Delete" + onClick={handleDeleteClick} + /> </DropdownMenuItemsContainer> } /> diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFromManyFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFromManyFieldInput.tsx new file mode 100644 index 000000000000..92e1dc80c5f3 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFromManyFieldInput.tsx @@ -0,0 +1,37 @@ +import { useContext } from 'react'; + +import { ObjectMetadataItemsRelationPickerEffect } from '@/object-metadata/components/ObjectMetadataItemsRelationPickerEffect'; +import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; +import { RelationFromManyFieldInputMultiRecordsEffect } from '@/object-record/record-field/meta-types/input/components/RelationFromManyFieldInputMultiRecordsEffect'; +import { useUpdateRelationFromManyFieldInput } from '@/object-record/record-field/meta-types/input/hooks/useUpdateRelationFromManyFieldInput'; +import { FieldInputEvent } from '@/object-record/record-field/types/FieldInputEvent'; +import { MultiRecordSelect } from '@/object-record/relation-picker/components/MultiRecordSelect'; +import { RelationPickerScope } from '@/object-record/relation-picker/scopes/RelationPickerScope'; + +export type RelationFromManyFieldInputProps = { + onSubmit?: FieldInputEvent; +}; + +export const RelationFromManyFieldInput = ({ + onSubmit, +}: RelationFromManyFieldInputProps) => { + const { fieldDefinition } = useContext(FieldContext); + const relationPickerScopeId = `relation-picker-${fieldDefinition.fieldMetadataId}`; + const { updateRelation } = useUpdateRelationFromManyFieldInput({ + scopeId: relationPickerScopeId, + }); + + const handleSubmit = () => { + onSubmit?.(() => {}); + }; + + return ( + <> + <RelationPickerScope relationPickerScopeId={relationPickerScopeId}> + <ObjectMetadataItemsRelationPickerEffect /> + <RelationFromManyFieldInputMultiRecordsEffect /> + <MultiRecordSelect onSubmit={handleSubmit} onChange={updateRelation} /> + </RelationPickerScope> + </> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFromManyFieldInputMultiRecordsEffect.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFromManyFieldInputMultiRecordsEffect.tsx new file mode 100644 index 000000000000..6f42a7fbad2e --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFromManyFieldInputMultiRecordsEffect.tsx @@ -0,0 +1,126 @@ +import { useEffect, useMemo } from 'react'; +import { useRecoilCallback, useRecoilState, useSetRecoilState } from 'recoil'; + +import { useObjectRecordMultiSelectScopedStates } from '@/activities/hooks/useObjectRecordMultiSelectScopedStates'; +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { useRelationField } from '@/object-record/record-field/meta-types/hooks/useRelationField'; +import { objectRecordMultiSelectComponentFamilyState } from '@/object-record/record-field/states/objectRecordMultiSelectComponentFamilyState'; +import { ObjectRecordForSelect } from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; +import { useRelationPickerEntitiesOptions } from '@/object-record/relation-picker/hooks/useRelationPickerEntitiesOptions'; +import { RelationPickerScopeInternalContext } from '@/object-record/relation-picker/scopes/scope-internal-context/RelationPickerScopeInternalContext'; +import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; +import { isDeeplyEqual } from '~/utils/isDeeplyEqual'; + +export const RelationFromManyFieldInputMultiRecordsEffect = () => { + const { fieldValue, fieldDefinition } = useRelationField<EntityForSelect[]>(); + const scopeId = useAvailableScopeIdOrThrow( + RelationPickerScopeInternalContext, + ); + const { + objectRecordsIdsMultiSelectState, + objectRecordMultiSelectCheckedRecordsIdsState, + recordMultiSelectIsLoadingState, + } = useObjectRecordMultiSelectScopedStates(scopeId); + const [objectRecordsIdsMultiSelect, setObjectRecordsIdsMultiSelect] = + useRecoilState(objectRecordsIdsMultiSelectState); + + const { entities } = useRelationPickerEntitiesOptions({ + relationObjectNameSingular: + fieldDefinition.metadata.relationObjectMetadataNameSingular, + }); + + const setRecordMultiSelectIsLoading = useSetRecoilState( + recordMultiSelectIsLoadingState, + ); + + const { objectMetadataItem } = useObjectMetadataItem({ + objectNameSingular: + fieldDefinition.metadata.relationObjectMetadataNameSingular, + }); + + const allRecords = useMemo( + () => [ + ...entities.entitiesToSelect.map((entity) => { + const { record, ...recordIdentifier } = entity; + return { + objectMetadataItem: objectMetadataItem, + record: record, + recordIdentifier: recordIdentifier, + }; + }), + ], + [entities.entitiesToSelect, objectMetadataItem], + ); + + const [ + objectRecordMultiSelectCheckedRecordsIds, + setObjectRecordMultiSelectCheckedRecordsIds, + ] = useRecoilState(objectRecordMultiSelectCheckedRecordsIdsState); + + const updateRecords = useRecoilCallback( + ({ snapshot, set }) => + (newRecords: ObjectRecordForSelect[]) => { + for (const newRecord of newRecords) { + const currentRecord = snapshot + .getLoadable( + objectRecordMultiSelectComponentFamilyState({ + scopeId: scopeId, + familyKey: newRecord.record.id, + }), + ) + .getValue(); + + const newRecordWithSelected = { + ...newRecord, + selected: objectRecordMultiSelectCheckedRecordsIds.includes( + newRecord.record.id, + ), + }; + + if ( + !isDeeplyEqual( + newRecordWithSelected.selected, + currentRecord?.selected, + ) + ) { + set( + objectRecordMultiSelectComponentFamilyState({ + scopeId: scopeId, + familyKey: newRecordWithSelected.record.id, + }), + newRecordWithSelected, + ); + } + } + }, + [objectRecordMultiSelectCheckedRecordsIds, scopeId], + ); + + useEffect(() => { + updateRecords(allRecords); + const allRecordsIds = allRecords.map((record) => record.record.id); + if (!isDeeplyEqual(allRecordsIds, objectRecordsIdsMultiSelect)) { + setObjectRecordsIdsMultiSelect(allRecordsIds); + } + }, [ + allRecords, + objectRecordsIdsMultiSelect, + setObjectRecordsIdsMultiSelect, + updateRecords, + ]); + + useEffect(() => { + setObjectRecordMultiSelectCheckedRecordsIds( + fieldValue + ? fieldValue.map((fieldValueItem: EntityForSelect) => fieldValueItem.id) + : [], + ); + }, [fieldValue, setObjectRecordMultiSelectCheckedRecordsIds]); + + useEffect(() => { + setRecordMultiSelectIsLoading(entities.loading); + }, [entities.loading, setRecordMultiSelectIsLoading]); + + return <></>; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationManyFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationManyFieldInput.tsx deleted file mode 100644 index 9e3629528463..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationManyFieldInput.tsx +++ /dev/null @@ -1,82 +0,0 @@ -import { useMemo } from 'react'; - -import { ObjectMetadataItemsRelationPickerEffect } from '@/object-metadata/components/ObjectMetadataItemsRelationPickerEffect'; -import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; -import { useUpdateRelationManyFieldInput } from '@/object-record/record-field/meta-types/input/hooks/useUpdateRelationManyFieldInput'; -import { useInlineCell } from '@/object-record/record-inline-cell/hooks/useInlineCell'; -import { MultiRecordSelect } from '@/object-record/relation-picker/components/MultiRecordSelect'; -import { useRelationPicker } from '@/object-record/relation-picker/hooks/useRelationPicker'; -import { useRelationPickerEntitiesOptions } from '@/object-record/relation-picker/hooks/useRelationPickerEntitiesOptions'; -import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; - -import { useRelationField } from '../../hooks/useRelationField'; - -export const RelationManyFieldInput = ({ - relationPickerScopeId = 'relation-picker', -}: { - relationPickerScopeId?: string; -}) => { - const { closeInlineCell: closeEditableField } = useInlineCell(); - - const { fieldDefinition, fieldValue } = useRelationField<EntityForSelect[]>(); - const { entities, relationPickerSearchFilter } = - useRelationPickerEntitiesOptions({ - relationObjectNameSingular: - fieldDefinition.metadata.relationObjectMetadataNameSingular, - relationPickerScopeId, - }); - - const { setRelationPickerSearchFilter } = useRelationPicker({ - relationPickerScopeId, - }); - - const { handleChange } = useUpdateRelationManyFieldInput({ entities }); - - const { objectMetadataItem } = useObjectMetadataItem({ - objectNameSingular: - fieldDefinition.metadata.relationObjectMetadataNameSingular, - }); - const allRecords = useMemo( - () => [ - ...entities.entitiesToSelect.map((entity) => { - const { record, ...recordIdentifier } = entity; - return { - objectMetadataItem: objectMetadataItem, - record: record, - recordIdentifier: recordIdentifier, - }; - }), - ], - [entities.entitiesToSelect, objectMetadataItem], - ); - - const selectedRecords = useMemo( - () => - allRecords.filter( - (entity) => - fieldValue?.some((f) => { - return f.id === entity.recordIdentifier.id; - }), - ), - [allRecords, fieldValue], - ); - - return ( - <> - <ObjectMetadataItemsRelationPickerEffect - relationPickerScopeId={relationPickerScopeId} - /> - <MultiRecordSelect - allRecords={allRecords} - selectedObjectRecords={selectedRecords} - loading={entities.loading} - searchFilter={relationPickerSearchFilter} - setSearchFilter={setRelationPickerSearchFilter} - onSubmit={() => { - closeEditableField(); - }} - onChange={handleChange} - /> - </> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationToOneFieldInput.tsx similarity index 90% rename from packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFieldInput.tsx rename to packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationToOneFieldInput.tsx index ed752bef9130..5366907c9332 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationFieldInput.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/RelationToOneFieldInput.tsx @@ -14,15 +14,15 @@ const StyledRelationPickerContainer = styled.div` top: -1px; `; -export type RelationFieldInputProps = { +export type RelationToOneFieldInputProps = { onSubmit?: FieldInputEvent; onCancel?: () => void; }; -export const RelationFieldInput = ({ +export const RelationToOneFieldInput = ({ onSubmit, onCancel, -}: RelationFieldInputProps) => { +}: RelationToOneFieldInputProps) => { const { fieldDefinition, initialSearchValue, fieldValue } = useRelationField<EntityForSelect>(); diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/SelectFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/SelectFieldInput.tsx index 0f745de75420..a1d9a8631a72 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/SelectFieldInput.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/SelectFieldInput.tsx @@ -1,7 +1,8 @@ -import React, { useRef, useState } from 'react'; import styled from '@emotion/styled'; +import { useRef, useState } from 'react'; import { Key } from 'ts-key-enum'; +import { useClearField } from '@/object-record/record-field/hooks/useClearField'; import { useSelectField } from '@/object-record/record-field/meta-types/hooks/useSelectField'; import { FieldInputEvent } from '@/object-record/record-field/types/FieldInputEvent'; import { DropdownMenu } from '@/ui/layout/dropdown/components/DropdownMenu'; @@ -30,12 +31,15 @@ export const SelectFieldInput = ({ }: SelectFieldInputProps) => { const { persistField, fieldDefinition, fieldValue, hotkeyScope } = useSelectField(); + const clearField = useClearField(); + const [searchFilter, setSearchFilter] = useState(''); const containerRef = useRef<HTMLDivElement>(null); const selectedOption = fieldDefinition.metadata.options.find( (option) => option.value === fieldValue, ); + const optionsToSelect = fieldDefinition.metadata.options.filter((option) => { return ( @@ -43,10 +47,17 @@ export const SelectFieldInput = ({ option.label.toLowerCase().includes(searchFilter.toLowerCase()) ); }) || []; + const optionsInDropDown = selectedOption ? [selectedOption, ...optionsToSelect] : optionsToSelect; + // handlers + const handleClearField = () => { + clearField(); + onCancel?.(); + }; + useListenClickOutside({ refs: [containerRef], callback: (event) => { @@ -85,7 +96,19 @@ export const SelectFieldInput = ({ autoFocus /> <DropdownMenuSeparator /> + <DropdownMenuItemsContainer hasMaxHeight> + {fieldDefinition.metadata.isNullable && ( + <MenuItemSelectTag + key={`No ${fieldDefinition.label}`} + selected={false} + text={`No ${fieldDefinition.label}`} + color="transparent" + variant="outline" + onClick={handleClearField} + /> + )} + {optionsInDropDown.map((option) => { return ( <MenuItemSelectTag diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/DateTimeFieldInput.stories.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/DateTimeFieldInput.stories.tsx index 00813a0bb1a5..5a8d23898b32 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/DateTimeFieldInput.stories.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/DateTimeFieldInput.stories.tsx @@ -12,7 +12,7 @@ import { DateTimeFieldInputProps, } from '../DateTimeFieldInput'; -const formattedDate = new Date(2022, 1, 1); +const formattedDate = new Date(2022, 0, 1, 2, 0, 0); const DateFieldValueSetterEffect = ({ value }: { value: Date }) => { const { setFieldValue } = useDateTimeField(); @@ -126,9 +126,9 @@ type Story = StoryObj<typeof DateFieldInputWithContext>; export const Default: Story = { play: async ({ canvasElement }) => { const canvas = within(canvasElement); - const div = await canvas.findByText('February'); + const div = await canvas.findByText('January'); - await expect(div.innerText).toContain('February'); + await expect(div.innerText).toContain('January'); }, }; @@ -138,7 +138,7 @@ export const ClickOutside: Story = { await expect(clickOutsideJestFn).toHaveBeenCalledTimes(0); - await canvas.findByText('February'); + await canvas.findByText('January'); const emptyDiv = canvas.getByTestId('data-field-input-click-outside-div'); await userEvent.click(emptyDiv); @@ -151,7 +151,7 @@ export const Escape: Story = { await expect(escapeJestFn).toHaveBeenCalledTimes(0); const canvas = within(canvasElement); - await canvas.findByText('February'); + await canvas.findByText('January'); await userEvent.keyboard('{escape}'); await expect(escapeJestFn).toHaveBeenCalledTimes(1); @@ -163,7 +163,7 @@ export const Enter: Story = { await expect(enterJestFn).toHaveBeenCalledTimes(0); const canvas = within(canvasElement); - await canvas.findByText('February'); + await canvas.findByText('January'); await userEvent.keyboard('{enter}'); await expect(enterJestFn).toHaveBeenCalledTimes(1); diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationManyFieldInput.stories.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationManyFieldInput.stories.tsx index b768056cb87f..a02b1ffb38d1 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationManyFieldInput.stories.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationManyFieldInput.stories.tsx @@ -5,7 +5,7 @@ import { useSetRecoilState } from 'recoil'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; -import { RelationManyFieldInput } from '@/object-record/record-field/meta-types/input/components/RelationManyFieldInput'; +import { RelationFromManyFieldInput } from '@/object-record/record-field/meta-types/input/components/RelationFromManyFieldInput'; import { useSetHotkeyScope } from '@/ui/utilities/hotkey/hooks/useSetHotkeyScope'; import { FieldMetadataType } from '~/generated/graphql'; import { ComponentWithRecoilScopeDecorator } from '~/testing/decorators/ComponentWithRecoilScopeDecorator'; @@ -45,22 +45,21 @@ const RelationManyFieldInputWithContext = () => { <FieldContextProvider fieldDefinition={{ fieldMetadataId: 'relation', - label: 'Relation', + label: 'People', type: FieldMetadataType.Relation, iconName: 'IconLink', metadata: { - fieldName: 'Relation', - relationObjectMetadataNamePlural: 'workspaceMembers', - relationObjectMetadataNameSingular: - CoreObjectNameSingular.WorkspaceMember, - objectMetadataNameSingular: 'person', + fieldName: 'people', + relationObjectMetadataNamePlural: 'companies', + relationObjectMetadataNameSingular: CoreObjectNameSingular.Company, + objectMetadataNameSingular: 'company', relationFieldMetadataId: '20202020-8c37-4163-ba06-1dada334ce3e', }, }} entityId={'entityId'} > <RelationWorkspaceSetterEffect /> - <RelationManyFieldInput /> + <RelationFromManyFieldInput /> </FieldContextProvider> <div data-testid="data-field-input-click-outside-div" /> </div> diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationFieldInput.stories.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationToOneFieldInput.stories.tsx similarity index 83% rename from packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationFieldInput.stories.tsx rename to packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationToOneFieldInput.stories.tsx index 9874373e81fd..30b723880b7b 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationFieldInput.stories.tsx +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/components/__stories__/RelationToOneFieldInput.stories.tsx @@ -13,6 +13,7 @@ import { useSetRecoilState } from 'recoil'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; +import { RelationPickerScope } from '@/object-record/relation-picker/scopes/RelationPickerScope'; import { useSetHotkeyScope } from '@/ui/utilities/hotkey/hooks/useSetHotkeyScope'; import { FieldMetadataType } from '~/generated/graphql'; import { ComponentWithRecoilScopeDecorator } from '~/testing/decorators/ComponentWithRecoilScopeDecorator'; @@ -26,9 +27,9 @@ import { import { FieldContextProvider } from '../../../__stories__/FieldContextProvider'; import { - RelationFieldInput, - RelationFieldInputProps, -} from '../RelationFieldInput'; + RelationToOneFieldInput, + RelationToOneFieldInputProps, +} from '../RelationToOneFieldInput'; const RelationWorkspaceSetterEffect = () => { const setCurrentWorkspace = useSetRecoilState(currentWorkspaceState); @@ -44,16 +45,16 @@ const RelationWorkspaceSetterEffect = () => { return <></>; }; -type RelationFieldInputWithContextProps = RelationFieldInputProps & { +type RelationToOneFieldInputWithContextProps = RelationToOneFieldInputProps & { value: number; entityId?: string; }; -const RelationFieldInputWithContext = ({ +const RelationToOneFieldInputWithContext = ({ entityId, onSubmit, onCancel, -}: RelationFieldInputWithContextProps) => { +}: RelationToOneFieldInputWithContextProps) => { const setHotKeyScope = useSetHotkeyScope(); useEffect(() => { @@ -79,8 +80,12 @@ const RelationFieldInputWithContext = ({ }} entityId={entityId} > - <RelationWorkspaceSetterEffect /> - <RelationFieldInput onSubmit={onSubmit} onCancel={onCancel} /> + <RelationPickerScope + relationPickerScopeId={'relation-to-one-field-input'} + > + <RelationWorkspaceSetterEffect /> + <RelationToOneFieldInput onSubmit={onSubmit} onCancel={onCancel} /> + </RelationPickerScope> </FieldContextProvider> <div data-testid="data-field-input-click-outside-div" /> </div> @@ -99,8 +104,8 @@ const clearMocksDecorator: Decorator = (Story, context) => { }; const meta: Meta = { - title: 'UI/Data/Field/Input/RelationFieldInput', - component: RelationFieldInputWithContext, + title: 'UI/Data/Field/Input/RelationToOneFieldInput', + component: RelationToOneFieldInputWithContext, args: { useEditButton: true, onSubmit: submitJestFn, @@ -123,7 +128,7 @@ const meta: Meta = { export default meta; -type Story = StoryObj<typeof RelationFieldInputWithContext>; +type Story = StoryObj<typeof RelationToOneFieldInputWithContext>; export const Default: Story = { decorators: [ComponentWithRecoilScopeDecorator], diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/hooks/useUpdateRelationFromManyFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/hooks/useUpdateRelationFromManyFieldInput.tsx new file mode 100644 index 000000000000..ec07c96c205c --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/hooks/useUpdateRelationFromManyFieldInput.tsx @@ -0,0 +1,93 @@ +import { useContext } from 'react'; +import { useRecoilCallback } from 'recoil'; + +import { useAttachRelatedRecordFromRecord } from '@/object-record/hooks/useAttachRelatedRecordFromRecord'; +import { useDetachRelatedRecordFromRecord } from '@/object-record/hooks/useDetachRelatedRecordFromRecord'; +import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; +import { objectRecordMultiSelectCheckedRecordsIdsComponentState } from '@/object-record/record-field/states/objectRecordMultiSelectCheckedRecordsIdsComponentState'; +import { assertFieldMetadata } from '@/object-record/record-field/types/guards/assertFieldMetadata'; +import { isFieldRelation } from '@/object-record/record-field/types/guards/isFieldRelation'; +import { FieldMetadataType } from '~/generated-metadata/graphql'; + +export const useUpdateRelationFromManyFieldInput = ({ + scopeId, +}: { + scopeId: string; +}) => { + const { entityId, fieldDefinition } = useContext(FieldContext); + + assertFieldMetadata( + FieldMetadataType.Relation, + isFieldRelation, + fieldDefinition, + ); + + if (!fieldDefinition.metadata.objectMetadataNameSingular) { + throw new Error('ObjectMetadataNameSingular is required'); + } + + const { updateOneRecordAndDetachRelations } = + useDetachRelatedRecordFromRecord({ + recordObjectNameSingular: + fieldDefinition.metadata.objectMetadataNameSingular, + fieldNameOnRecordObject: fieldDefinition.metadata.fieldName, + }); + + const { updateOneRecordAndAttachRelations } = + useAttachRelatedRecordFromRecord({ + recordObjectNameSingular: + fieldDefinition.metadata.objectMetadataNameSingular, + fieldNameOnRecordObject: fieldDefinition.metadata.fieldName, + }); + + const updateRelation = useRecoilCallback( + ({ snapshot, set }) => + async (objectRecordId: string) => { + const previouslyCheckedRecordsIds = snapshot + .getLoadable( + objectRecordMultiSelectCheckedRecordsIdsComponentState({ + scopeId, + }), + ) + .getValue(); + + const isNewlySelected = + !previouslyCheckedRecordsIds.includes(objectRecordId); + if (isNewlySelected) { + set( + objectRecordMultiSelectCheckedRecordsIdsComponentState({ + scopeId, + }), + (prev) => [...prev, objectRecordId], + ); + } else { + set( + objectRecordMultiSelectCheckedRecordsIdsComponentState({ + scopeId, + }), + (prev) => prev.filter((id) => id !== objectRecordId), + ); + } + + if (isNewlySelected) { + await updateOneRecordAndAttachRelations({ + recordId: entityId, + relatedRecordId: objectRecordId, + }); + } else { + await updateOneRecordAndDetachRelations({ + recordId: entityId, + relatedRecordId: objectRecordId, + }); + } + }, + [ + entityId, + scopeId, + updateOneRecordAndAttachRelations, + updateOneRecordAndDetachRelations, + ], + ); + + return { updateRelation }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/hooks/useUpdateRelationManyFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/hooks/useUpdateRelationManyFieldInput.tsx deleted file mode 100644 index a0d274418af5..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-field/meta-types/input/hooks/useUpdateRelationManyFieldInput.tsx +++ /dev/null @@ -1,52 +0,0 @@ -import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; -import { useRelationField } from '@/object-record/record-field/meta-types/hooks/useRelationField'; -import { ObjectRecordForSelect } from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; -import { EntitiesForMultipleEntitySelect } from '@/object-record/relation-picker/types/EntitiesForMultipleEntitySelect'; -import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; -import { isDefined } from '~/utils/isDefined'; - -export const useUpdateRelationManyFieldInput = ({ - entities, -}: { - entities: EntitiesForMultipleEntitySelect<EntityForSelect>; -}) => { - const { fieldDefinition, fieldValue, setFieldValue, entityId } = - useRelationField<EntityForSelect[]>(); - - const { updateOneRecord } = useUpdateOneRecord({ - objectNameSingular: - fieldDefinition.metadata.relationObjectMetadataNameSingular, - }); - - const fieldName = fieldDefinition.metadata.targetFieldMetadataName; - - const handleChange = ( - objectRecord: ObjectRecordForSelect | null, - isSelected: boolean, - ) => { - const entityToAddOrRemove = entities.entitiesToSelect.find( - (entity) => entity.id === objectRecord?.recordIdentifier.id, - ); - - const updatedFieldValue = isSelected - ? [...(fieldValue ?? []), entityToAddOrRemove] - : (fieldValue ?? []).filter( - (value) => value.id !== objectRecord?.recordIdentifier.id, - ); - setFieldValue( - updatedFieldValue.filter((value) => - isDefined(value), - ) as EntityForSelect[], - ); - if (isDefined(objectRecord)) { - updateOneRecord({ - idToUpdate: objectRecord.record?.id, - updateOneRecordInput: { - [`${fieldName}Id`]: isSelected ? entityId : null, - }, - }); - } - }; - - return { handleChange }; -}; diff --git a/packages/twenty-front/src/modules/object-record/record-field/states/activityTargetObjectRecordFamilyState.ts b/packages/twenty-front/src/modules/object-record/record-field/states/activityTargetObjectRecordFamilyState.ts new file mode 100644 index 000000000000..45d54e009ce1 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/states/activityTargetObjectRecordFamilyState.ts @@ -0,0 +1,13 @@ +import { createFamilyState } from '@/ui/utilities/state/utils/createFamilyState'; + +export type ActivityTargetObjectRecord = { + activityTargetId: string | null; +}; + +export const activityTargetObjectRecordFamilyState = createFamilyState< + ActivityTargetObjectRecord, + string +>({ + key: 'activityTargetObjectRecordFamilyState', + defaultValue: { activityTargetId: null }, +}); diff --git a/packages/twenty-front/src/modules/object-record/record-field/states/lastShowPageRecordId.ts b/packages/twenty-front/src/modules/object-record/record-field/states/lastShowPageRecordId.ts new file mode 100644 index 000000000000..dce923b37bda --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/states/lastShowPageRecordId.ts @@ -0,0 +1,6 @@ +import { createState } from 'twenty-ui'; + +export const lastShowPageRecordIdState = createState<string | null>({ + key: 'lastShowPageRecordIdState', + defaultValue: null, +}); diff --git a/packages/twenty-front/src/modules/object-record/record-field/states/objectRecordMultiSelectCheckedRecordsIdsComponentState.ts b/packages/twenty-front/src/modules/object-record/record-field/states/objectRecordMultiSelectCheckedRecordsIdsComponentState.ts new file mode 100644 index 000000000000..4f51db21d8dd --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/states/objectRecordMultiSelectCheckedRecordsIdsComponentState.ts @@ -0,0 +1,7 @@ +import { createComponentState } from '@/ui/utilities/state/component-state/utils/createComponentState'; + +export const objectRecordMultiSelectCheckedRecordsIdsComponentState = + createComponentState<string[]>({ + key: 'objectRecordMultiSelectCheckedRecordsIdsComponentState', + defaultValue: [], + }); diff --git a/packages/twenty-front/src/modules/object-record/record-field/states/objectRecordMultiSelectComponentFamilyState.ts b/packages/twenty-front/src/modules/object-record/record-field/states/objectRecordMultiSelectComponentFamilyState.ts new file mode 100644 index 000000000000..0e15c962e11b --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/states/objectRecordMultiSelectComponentFamilyState.ts @@ -0,0 +1,12 @@ +import { ObjectRecordForSelect } from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; +import { createComponentFamilyState } from '@/ui/utilities/state/component-state/utils/createComponentFamilyState'; + +export type ObjectRecordAndSelected = ObjectRecordForSelect & { + selected: boolean; +}; + +export const objectRecordMultiSelectComponentFamilyState = + createComponentFamilyState<ObjectRecordAndSelected | undefined, string>({ + key: 'objectRecordMultiSelectComponentFamilyState', + defaultValue: undefined, + }); diff --git a/packages/twenty-front/src/modules/object-record/record-field/states/recordMultiSelectIsLoadingComponentState.ts b/packages/twenty-front/src/modules/object-record/record-field/states/recordMultiSelectIsLoadingComponentState.ts new file mode 100644 index 000000000000..880207ecbeae --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/states/recordMultiSelectIsLoadingComponentState.ts @@ -0,0 +1,7 @@ +import { createComponentState } from '@/ui/utilities/state/component-state/utils/createComponentState'; + +export const recordMultiSelectIsLoadingComponentState = + createComponentState<boolean>({ + key: 'recordMultiSelectIsLoadingComponentState', + defaultValue: false, + }); diff --git a/packages/twenty-front/src/modules/object-record/record-field/states/recordPositionInternalState.ts b/packages/twenty-front/src/modules/object-record/record-field/states/recordPositionInternalState.ts new file mode 100644 index 000000000000..9b83e0dcd8d7 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/states/recordPositionInternalState.ts @@ -0,0 +1,6 @@ +import { createState } from 'twenty-ui'; + +export const recordPositionInternalState = createState<number | null>({ + key: 'recordPositionInternalState', + defaultValue: null, +}); diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/CurrencyCode.ts b/packages/twenty-front/src/modules/object-record/record-field/types/CurrencyCode.ts index 71e58d242553..9e86b3c87939 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/types/CurrencyCode.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/types/CurrencyCode.ts @@ -10,7 +10,11 @@ export enum CurrencyCode { USD = 'USD', NOK = 'NOK', SEK = 'SEK', + BHT = 'BHT', MAD = 'MAD', QAR = 'QAR', AED = 'AED', + KRW = 'KRW', + BRL = 'BRL', + AUD = 'AUD', } diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/FieldInputDraftValue.ts b/packages/twenty-front/src/modules/object-record/record-field/types/FieldInputDraftValue.ts index 0282ccd02f86..ae1d63e549eb 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/types/FieldInputDraftValue.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/types/FieldInputDraftValue.ts @@ -13,12 +13,12 @@ import { FieldNumberValue, FieldPhoneValue, FieldRatingValue, - FieldRelationValue, + FieldRelationFromManyValue, + FieldRelationToOneValue, FieldSelectValue, FieldTextValue, FieldUUidValue, } from '@/object-record/record-field/types/FieldMetadata'; -import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; export type FieldTextDraftValue = string; export type FieldNumberDraftValue = string; @@ -28,6 +28,7 @@ export type FieldEmailDraftValue = string; export type FieldSelectDraftValue = string; export type FieldMultiSelectDraftValue = string[]; export type FieldRelationDraftValue = string; +export type FieldRelationManyDraftValue = string[]; export type FieldLinkDraftValue = { url: string; label: string }; export type FieldLinksDraftValue = { primaryLinkLabel: string; @@ -79,12 +80,12 @@ export type FieldInputDraftValue<FieldValue> = FieldValue extends FieldTextValue ? FieldSelectDraftValue : FieldValue extends FieldMultiSelectValue ? FieldMultiSelectDraftValue - : FieldValue extends - | FieldRelationValue<EntityForSelect> - | FieldRelationValue<EntityForSelect[]> + : FieldValue extends FieldRelationToOneValue ? FieldRelationDraftValue - : FieldValue extends FieldAddressValue - ? FieldAddressDraftValue - : FieldValue extends FieldJsonValue - ? FieldJsonDraftValue - : never; + : FieldValue extends FieldRelationFromManyValue + ? FieldRelationManyDraftValue + : FieldValue extends FieldAddressValue + ? FieldAddressDraftValue + : FieldValue extends FieldJsonValue + ? FieldJsonDraftValue + : never; diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/FieldMetadata.ts b/packages/twenty-front/src/modules/object-record/record-field/types/FieldMetadata.ts index f54794613652..3208e5debbae 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/types/FieldMetadata.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/types/FieldMetadata.ts @@ -1,7 +1,9 @@ import { ThemeColor } from 'twenty-ui'; import { RATING_VALUES } from '@/object-record/record-field/meta-types/constants/RatingValues'; +import { ZodHelperLiteral } from '@/object-record/record-field/types/ZodHelperLiteral'; import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; +import { WithNarrowedStringLiteralProperty } from '~/types/WithNarrowedStringLiteralProperty'; import { CurrencyCode } from './CurrencyCode'; @@ -110,10 +112,23 @@ export type FieldRelationMetadata = { useEditButton?: boolean; }; +export type FieldRelationOneMetadata = WithNarrowedStringLiteralProperty< + FieldRelationMetadata, + 'relationType', + 'TO_ONE_OBJECT' +>; + +export type FieldRelationManyMetadata = WithNarrowedStringLiteralProperty< + FieldRelationMetadata, + 'relationType', + 'FROM_MANY_OBJECTS' +>; + export type FieldSelectMetadata = { objectMetadataNameSingular?: string; fieldName: string; options: { label: string; color: ThemeColor; value: string }[]; + isNullable: boolean; }; export type FieldMultiSelectMetadata = { @@ -174,10 +189,13 @@ export type FieldRatingValue = (typeof RATING_VALUES)[number]; export type FieldSelectValue = string | null; export type FieldMultiSelectValue = string[] | null; -export type FieldRelationValue<T extends EntityForSelect | EntityForSelect[]> = - T | null; +export type FieldRelationToOneValue = EntityForSelect | null; + +export type FieldRelationFromManyValue = EntityForSelect[] | []; + +export type FieldRelationValue< + T extends FieldRelationToOneValue | FieldRelationFromManyValue, +> = T; -// See https://zod.dev/?id=json-type -type Literal = string | number | boolean | null; -export type Json = Literal | { [key: string]: Json } | Json[]; +export type Json = ZodHelperLiteral | { [key: string]: Json } | Json[]; export type FieldJsonValue = Record<string, Json> | Json[] | null; diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/RecordChipData.ts b/packages/twenty-front/src/modules/object-record/record-field/types/RecordChipData.ts index 13eb99f84cd0..d6ef0c2f6948 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/types/RecordChipData.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/types/RecordChipData.ts @@ -2,8 +2,9 @@ import { AvatarType } from 'twenty-ui'; export type RecordChipData = { recordId: string; - name: string | number; + name: string; avatarType: AvatarType; avatarUrl: string; - linkToShowPage: string; + isLabelIdentifier: boolean; + objectNameSingular: string; }; diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/ZodHelperLiteral.ts b/packages/twenty-front/src/modules/object-record/record-field/types/ZodHelperLiteral.ts new file mode 100644 index 000000000000..81c9b3dbde06 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/types/ZodHelperLiteral.ts @@ -0,0 +1,2 @@ +/** See https://zod.dev/?id=json-type */ +export type ZodHelperLiteral = string | number | boolean | null; diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/guards/assertFieldMetadata.ts b/packages/twenty-front/src/modules/object-record/record-field/types/guards/assertFieldMetadata.ts index 9d1f8782848b..c82974a1a7b8 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/types/guards/assertFieldMetadata.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/types/guards/assertFieldMetadata.ts @@ -51,19 +51,17 @@ type AssertFieldMetadataFunction = < ? FieldNumberMetadata : E extends 'PHONE' ? FieldPhoneMetadata - : E extends 'PROBABILITY' - ? FieldRatingMetadata - : E extends 'RELATION' - ? FieldRelationMetadata - : E extends 'TEXT' - ? FieldTextMetadata - : E extends 'UUID' - ? FieldUuidMetadata - : E extends 'ADDRESS' - ? FieldAddressMetadata - : E extends 'RAW_JSON' - ? FieldRawJsonMetadata - : never, + : E extends 'RELATION' + ? FieldRelationMetadata + : E extends 'TEXT' + ? FieldTextMetadata + : E extends 'UUID' + ? FieldUuidMetadata + : E extends 'ADDRESS' + ? FieldAddressMetadata + : E extends 'RAW_JSON' + ? FieldRawJsonMetadata + : never, >( fieldType: E, fieldTypeGuard: ( diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationFromManyObjects.ts b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationFromManyObjects.ts index 03b56892afd6..03559df5d356 100644 --- a/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationFromManyObjects.ts +++ b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationFromManyObjects.ts @@ -1,10 +1,9 @@ -import { FieldMetadataType } from '~/generated-metadata/graphql'; +import { isFieldRelation } from '@/object-record/record-field/types/guards/isFieldRelation'; import { FieldDefinition } from '../FieldDefinition'; -import { FieldRelationMetadata } from '../FieldMetadata'; +import { FieldMetadata, FieldRelationManyMetadata } from '../FieldMetadata'; export const isFieldRelationFromManyObjects = ( - field: Pick<FieldDefinition<FieldRelationMetadata>, 'type' | 'metadata'>, -): field is FieldDefinition<FieldRelationMetadata> => - field.type === FieldMetadataType.Relation && - field.metadata.relationType === 'FROM_MANY_OBJECTS'; + field: Pick<FieldDefinition<FieldMetadata>, 'type' | 'metadata'>, +): field is FieldDefinition<FieldRelationManyMetadata> => + isFieldRelation(field) && field.metadata.relationType === 'FROM_MANY_OBJECTS'; diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationFromManyValue.ts b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationFromManyValue.ts new file mode 100644 index 000000000000..a2d0df7d995c --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationFromManyValue.ts @@ -0,0 +1,9 @@ +import { isNull, isObject, isUndefined } from '@sniptt/guards'; + +import { FieldRelationFromManyValue } from '@/object-record/record-field/types/FieldMetadata'; + +// TODO: add zod +export const isFieldRelationFromManyValue = ( + fieldValue: unknown, +): fieldValue is FieldRelationFromManyValue => + !isUndefined(fieldValue) && (isObject(fieldValue) || isNull(fieldValue)); diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationToOneObject.ts b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationToOneObject.ts new file mode 100644 index 000000000000..d1a1bb365aab --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationToOneObject.ts @@ -0,0 +1,9 @@ +import { isFieldRelation } from '@/object-record/record-field/types/guards/isFieldRelation'; + +import { FieldDefinition } from '../FieldDefinition'; +import { FieldMetadata, FieldRelationOneMetadata } from '../FieldMetadata'; + +export const isFieldRelationToOneObject = ( + field: Pick<FieldDefinition<FieldMetadata>, 'type' | 'metadata'>, +): field is FieldDefinition<FieldRelationOneMetadata> => + isFieldRelation(field) && field.metadata.relationType === 'TO_ONE_OBJECT'; diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationToOneValue.ts b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationToOneValue.ts new file mode 100644 index 000000000000..dc1f5b3615a1 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationToOneValue.ts @@ -0,0 +1,9 @@ +import { isNull, isObject, isUndefined } from '@sniptt/guards'; + +import { FieldRelationToOneValue } from '@/object-record/record-field/types/FieldMetadata'; + +// TODO: add zod +export const isFieldRelationToOneValue = ( + fieldValue: unknown, +): fieldValue is FieldRelationToOneValue => + !isUndefined(fieldValue) && (isObject(fieldValue) || isNull(fieldValue)); diff --git a/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationValue.ts b/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationValue.ts deleted file mode 100644 index 1919c493e327..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-field/types/guards/isFieldRelationValue.ts +++ /dev/null @@ -1,13 +0,0 @@ -import { isNull, isObject, isUndefined } from '@sniptt/guards'; - -import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; - -import { FieldRelationValue } from '../FieldMetadata'; - -// TODO: add zod -export const isFieldRelationValue = < - T extends EntityForSelect | EntityForSelect[], ->( - fieldValue: unknown, -): fieldValue is FieldRelationValue<T> => - !isUndefined(fieldValue) && (isObject(fieldValue) || isNull(fieldValue)); diff --git a/packages/twenty-front/src/modules/object-record/record-filter/utils/turnObjectDropdownFilterIntoQueryFilter.ts b/packages/twenty-front/src/modules/object-record/record-filter/utils/turnObjectDropdownFilterIntoQueryFilter.ts index 3235ac572dbe..1b08eace5a6b 100644 --- a/packages/twenty-front/src/modules/object-record/record-filter/utils/turnObjectDropdownFilterIntoQueryFilter.ts +++ b/packages/twenty-front/src/modules/object-record/record-filter/utils/turnObjectDropdownFilterIntoQueryFilter.ts @@ -6,10 +6,12 @@ import { DateFilter, FloatFilter, RecordGqlOperationFilter, + RelationFilter, StringFilter, URLFilter, UUIDFilter, } from '@/object-record/graphql/types/RecordGqlOperationFilter'; +import { FilterType } from '@/object-record/object-filter-dropdown/types/FilterType'; import { makeAndFilterVariables } from '@/object-record/utils/makeAndFilterVariables'; import { ViewFilterOperand } from '@/views/types/ViewFilterOperand'; import { Field } from '~/generated/graphql'; @@ -24,6 +26,200 @@ export type ObjectDropdownFilter = Omit<Filter, 'definition'> & { }; }; +const applyEmptyFilters = ( + operand: ViewFilterOperand, + correspondingField: Pick<Field, 'id' | 'name'>, + objectRecordFilters: RecordGqlOperationFilter[], + filterType: FilterType, +) => { + let emptyRecordFilter: RecordGqlOperationFilter = {}; + + switch (filterType) { + case 'TEXT': + case 'EMAIL': + case 'PHONE': + emptyRecordFilter = { + or: [ + { [correspondingField.name]: { ilike: '' } as StringFilter }, + { [correspondingField.name]: { is: 'NULL' } as StringFilter }, + ], + }; + break; + case 'CURRENCY': + emptyRecordFilter = { + or: [ + { + [correspondingField.name]: { + amountMicros: { is: 'NULL' }, + } as CurrencyFilter, + }, + ], + }; + break; + case 'FULL_NAME': { + const fullNameFilters = generateILikeFiltersForCompositeFields( + '', + correspondingField.name, + ['firstName', 'lastName'], + true, + ); + + emptyRecordFilter = { + and: fullNameFilters, + }; + break; + } + case 'LINK': + emptyRecordFilter = { + or: [ + { [correspondingField.name]: { url: { ilike: '' } } as URLFilter }, + { + [correspondingField.name]: { url: { is: 'NULL' } } as URLFilter, + }, + ], + }; + break; + case 'LINKS': { + const linksFilters = generateILikeFiltersForCompositeFields( + '', + correspondingField.name, + ['primaryLinkLabel', 'primaryLinkUrl'], + true, + ); + + emptyRecordFilter = { + and: linksFilters, + }; + break; + } + case 'ADDRESS': + emptyRecordFilter = { + and: [ + { + or: [ + { + [correspondingField.name]: { + addressStreet1: { ilike: '' }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressStreet1: { is: 'NULL' }, + } as AddressFilter, + }, + ], + }, + { + or: [ + { + [correspondingField.name]: { + addressStreet2: { ilike: '' }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressStreet2: { is: 'NULL' }, + } as AddressFilter, + }, + ], + }, + { + or: [ + { + [correspondingField.name]: { + addressCity: { ilike: '' }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressCity: { is: 'NULL' }, + } as AddressFilter, + }, + ], + }, + { + or: [ + { + [correspondingField.name]: { + addressState: { ilike: '' }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressState: { is: 'NULL' }, + } as AddressFilter, + }, + ], + }, + { + or: [ + { + [correspondingField.name]: { + addressCountry: { ilike: '' }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressCountry: { is: 'NULL' }, + } as AddressFilter, + }, + ], + }, + { + or: [ + { + [correspondingField.name]: { + addressPostcode: { ilike: '' }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressPostcode: { is: 'NULL' }, + } as AddressFilter, + }, + ], + }, + ], + }; + break; + case 'NUMBER': + emptyRecordFilter = { + [correspondingField.name]: { is: 'NULL' } as FloatFilter, + }; + break; + case 'DATE_TIME': + emptyRecordFilter = { + [correspondingField.name]: { is: 'NULL' } as DateFilter, + }; + break; + case 'SELECT': + emptyRecordFilter = { + [correspondingField.name]: { is: 'NULL' } as UUIDFilter, + }; + break; + case 'RELATION': + emptyRecordFilter = { + [correspondingField.name + 'Id']: { is: 'NULL' } as RelationFilter, + }; + break; + default: + throw new Error(`Unsupported empty filter type ${filterType}`); + } + + switch (operand) { + case ViewFilterOperand.IsEmpty: + objectRecordFilters.push(emptyRecordFilter); + break; + case ViewFilterOperand.IsNotEmpty: + objectRecordFilters.push({ + not: emptyRecordFilter, + }); + break; + default: + throw new Error(`Unknown operand ${operand} for ${filterType} filter`); + } +}; + export const turnObjectDropdownFilterIntoQueryFilter = ( rawUIFilters: ObjectDropdownFilter[], fields: Pick<Field, 'id' | 'name'>[], @@ -35,12 +231,19 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( (field) => field.id === rawUIFilter.fieldMetadataId, ); + const isEmptyOperand = [ + ViewFilterOperand.IsEmpty, + ViewFilterOperand.IsNotEmpty, + ].includes(rawUIFilter.operand); + if (!correspondingField) { continue; } - if (!isDefined(rawUIFilter.value) || rawUIFilter.value === '') { - continue; + if (!isEmptyOperand) { + if (!isDefined(rawUIFilter.value) || rawUIFilter.value === '') { + continue; + } } switch (rawUIFilter.definition.type) { @@ -64,6 +267,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( }, }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -86,6 +298,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( } as DateFilter, }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -108,6 +329,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( } as FloatFilter, }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -115,39 +345,57 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( } break; case 'RELATION': { - try { - JSON.parse(rawUIFilter.value); - } catch (e) { - throw new Error( - `Cannot parse filter value for RELATION filter : "${rawUIFilter.value}"`, - ); - } + if (!isEmptyOperand) { + try { + JSON.parse(rawUIFilter.value); + } catch (e) { + throw new Error( + `Cannot parse filter value for RELATION filter : "${rawUIFilter.value}"`, + ); + } - const parsedRecordIds = JSON.parse(rawUIFilter.value) as string[]; + const parsedRecordIds = JSON.parse(rawUIFilter.value) as string[]; - if (parsedRecordIds.length > 0) { - switch (rawUIFilter.operand) { - case ViewFilterOperand.Is: - objectRecordFilters.push({ - [correspondingField.name + 'Id']: { - in: parsedRecordIds, - } as UUIDFilter, - }); - break; - case ViewFilterOperand.IsNot: - if (parsedRecordIds.length > 0) { + if (parsedRecordIds.length > 0) { + switch (rawUIFilter.operand) { + case ViewFilterOperand.Is: objectRecordFilters.push({ - not: { - [correspondingField.name + 'Id']: { - in: parsedRecordIds, - } as UUIDFilter, - }, + [correspondingField.name + 'Id']: { + in: parsedRecordIds, + } as RelationFilter, }); - } + break; + case ViewFilterOperand.IsNot: + if (parsedRecordIds.length > 0) { + objectRecordFilters.push({ + not: { + [correspondingField.name + 'Id']: { + in: parsedRecordIds, + } as RelationFilter, + }, + }); + } + break; + default: + throw new Error( + `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, + ); + } + } + } else { + switch (rawUIFilter.operand) { + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); break; default: throw new Error( - `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, + `Unknown empty operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, ); } } @@ -169,6 +417,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( } as CurrencyFilter, }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -197,6 +454,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( }, }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -224,6 +490,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( }), }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -231,7 +506,6 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( } break; } - case 'FULL_NAME': { const fullNameFilters = generateILikeFiltersForCompositeFields( rawUIFilter.value, @@ -253,6 +527,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( }), }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -286,6 +569,27 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( }, } as AddressFilter, }, + { + [correspondingField.name]: { + addressState: { + ilike: `%${rawUIFilter.value}%`, + }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressCountry: { + ilike: `%${rawUIFilter.value}%`, + }, + } as AddressFilter, + }, + { + [correspondingField.name]: { + addressPostcode: { + ilike: `%${rawUIFilter.value}%`, + }, + } as AddressFilter, + }, ], }); break; @@ -322,6 +626,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( ], }); break; + case ViewFilterOperand.IsEmpty: + case ViewFilterOperand.IsNotEmpty: + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; default: throw new Error( `Unknown operand ${rawUIFilter.operand} for ${rawUIFilter.definition.type} filter`, @@ -329,6 +642,15 @@ export const turnObjectDropdownFilterIntoQueryFilter = ( } break; case 'SELECT': { + if (isEmptyOperand) { + applyEmptyFilters( + rawUIFilter.operand, + correspondingField, + objectRecordFilters, + rawUIFilter.definition.type, + ); + break; + } const stringifiedSelectValues = rawUIFilter.value; let parsedOptionValues: string[] = []; diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardColumnLoaderEffect.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardColumnLoaderEffect.tsx index 804df9806dd5..e9c909b72418 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardColumnLoaderEffect.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardColumnLoaderEffect.tsx @@ -1,9 +1,10 @@ import { useEffect } from 'react'; -import { useRecoilState } from 'recoil'; +import { useRecoilState, useSetRecoilState } from 'recoil'; import { isRecordBoardFetchingRecordsByColumnFamilyState } from '@/object-record/record-board/states/isRecordBoardFetchingRecordsByColumnFamilyState'; import { recordBoardShouldFetchMoreInColumnComponentFamilyState } from '@/object-record/record-board/states/recordBoardShouldFetchMoreInColumnComponentFamilyState'; import { useLoadRecordIndexBoardColumn } from '@/object-record/record-index/hooks/useLoadRecordIndexBoardColumn'; +import { isRecordIndexBoardColumnLoadingFamilyState } from '@/object-record/states/isRecordBoardColumnLoadingFamilyState'; import { getScopeIdFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdFromComponentId'; export const RecordIndexBoardColumnLoaderEffect = ({ @@ -15,7 +16,7 @@ export const RecordIndexBoardColumnLoaderEffect = ({ }: { recordBoardId: string; objectNameSingular: string; - boardFieldSelectValue: string; + boardFieldSelectValue: string | null; boardFieldMetadataId: string | null; columnId: string; }) => { @@ -34,7 +35,7 @@ export const RecordIndexBoardColumnLoaderEffect = ({ }), ); - const { fetchMoreRecords, loading, hasNextPage } = + const { fetchMoreRecords, loading, records, hasNextPage } = useLoadRecordIndexBoardColumn({ objectNameSingular, recordBoardId, @@ -43,6 +44,14 @@ export const RecordIndexBoardColumnLoaderEffect = ({ columnId, }); + const setIsRecordIndexLoading = useSetRecoilState( + isRecordIndexBoardColumnLoadingFamilyState(columnId), + ); + + useEffect(() => { + setIsRecordIndexLoading(loading && records.length === 0); + }, [records, loading, setIsRecordIndexLoading]); + useEffect(() => { const run = async () => { if (!loading && shouldFetchMore && hasNextPage) { diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoader.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoader.tsx index d6ca8ddfd8ab..1906285f521f 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoader.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoader.tsx @@ -45,6 +45,15 @@ export const RecordIndexBoardDataLoader = ({ columnId={columnIds[index]} /> ))} + {recordIndexKanbanFieldMetadataItem?.isNullable && ( + <RecordIndexBoardColumnLoaderEffect + objectNameSingular={objectNameSingular} + boardFieldMetadataId={recordIndexKanbanFieldMetadataId} + boardFieldSelectValue={null} + recordBoardId={recordBoardId} + columnId={'no-value'} + /> + )} </> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoaderEffect.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoaderEffect.tsx index b7b31f73c985..79c0e606817c 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoaderEffect.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexBoardDataLoaderEffect.tsx @@ -3,6 +3,7 @@ import { useNavigate } from 'react-router-dom'; import { useRecoilValue, useSetRecoilState } from 'recoil'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { getObjectSlug } from '@/object-metadata/utils/getObjectSlug'; import { useRecordActionBar } from '@/object-record/record-action-bar/hooks/useRecordActionBar'; import { useRecordBoard } from '@/object-record/record-board/hooks/useRecordBoard'; import { useRecordBoardSelection } from '@/object-record/record-board/hooks/useRecordBoardSelection'; @@ -59,10 +60,9 @@ export const RecordIndexBoardDataLoaderEffect = ({ }, [recordIndexFieldDefinitions, setFieldDefinitions]); const navigate = useNavigate(); - const navigateToSelectSettings = useCallback(() => { - navigate(`/settings/objects/${objectMetadataItem.namePlural}`); - }, [navigate, objectMetadataItem.namePlural]); + navigate(`/settings/objects/${getObjectSlug(objectMetadataItem)}`); + }, [navigate, objectMetadataItem]); const { resetRecordSelection } = useRecordBoardSelection(recordBoardId); diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx index 28b6aa9a6a9d..c7098481c06d 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx @@ -1,15 +1,24 @@ import styled from '@emotion/styled'; -import { useRecoilCallback, useRecoilState, useSetRecoilState } from 'recoil'; +import { + useRecoilCallback, + useRecoilState, + useRecoilValue, + useSetRecoilState, +} from 'recoil'; import { useColumnDefinitionsFromFieldMetadata } from '@/object-metadata/hooks/useColumnDefinitionsFromFieldMetadata'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; import { useObjectNameSingularFromPlural } from '@/object-metadata/hooks/useObjectNameSingularFromPlural'; +import { turnSortsIntoOrderBy } from '@/object-record/object-sort-dropdown/utils/turnSortsIntoOrderBy'; +import { lastShowPageRecordIdState } from '@/object-record/record-field/states/lastShowPageRecordId'; +import { turnObjectDropdownFilterIntoQueryFilter } from '@/object-record/record-filter/utils/turnObjectDropdownFilterIntoQueryFilter'; import { RecordIndexBoardContainer } from '@/object-record/record-index/components/RecordIndexBoardContainer'; import { RecordIndexBoardDataLoader } from '@/object-record/record-index/components/RecordIndexBoardDataLoader'; import { RecordIndexBoardDataLoaderEffect } from '@/object-record/record-index/components/RecordIndexBoardDataLoaderEffect'; import { RecordIndexTableContainer } from '@/object-record/record-index/components/RecordIndexTableContainer'; import { RecordIndexTableContainerEffect } from '@/object-record/record-index/components/RecordIndexTableContainerEffect'; import { RecordIndexViewBarEffect } from '@/object-record/record-index/components/RecordIndexViewBarEffect'; +import { RecordIndexEventContext } from '@/object-record/record-index/contexts/RecordIndexEventContext'; import { RecordIndexOptionsDropdown } from '@/object-record/record-index/options/components/RecordIndexOptionsDropdown'; import { recordIndexFieldDefinitionsState } from '@/object-record/record-index/states/recordIndexFieldDefinitionsState'; import { recordIndexFiltersState } from '@/object-record/record-index/states/recordIndexFiltersState'; @@ -17,15 +26,22 @@ import { recordIndexIsCompactModeActiveState } from '@/object-record/record-inde import { recordIndexKanbanFieldMetadataIdState } from '@/object-record/record-index/states/recordIndexKanbanFieldMetadataIdState'; import { recordIndexSortsState } from '@/object-record/record-index/states/recordIndexSortsState'; import { recordIndexViewTypeState } from '@/object-record/record-index/states/recordIndexViewTypeState'; +import { useFindRecordCursorFromFindManyCacheRootQuery } from '@/object-record/record-show/hooks/useFindRecordCursorFromFindManyCacheRootQuery'; +import { findView } from '@/object-record/record-show/hooks/useRecordShowPagePagination'; import { RecordFieldValueSelectorContextProvider } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; import { useRecordTable } from '@/object-record/record-table/hooks/useRecordTable'; +import { usePrefetchedData } from '@/prefetch/hooks/usePrefetchedData'; +import { PrefetchKey } from '@/prefetch/types/PrefetchKey'; import { SpreadsheetImportProvider } from '@/spreadsheet-import/provider/components/SpreadsheetImportProvider'; import { ViewBar } from '@/views/components/ViewBar'; +import { currentViewIdComponentState } from '@/views/states/currentViewIdComponentState'; +import { View } from '@/views/types/View'; import { ViewField } from '@/views/types/ViewField'; import { ViewType } from '@/views/types/ViewType'; import { mapViewFieldsToColumnDefinitions } from '@/views/utils/mapViewFieldsToColumnDefinitions'; import { mapViewFiltersToFilters } from '@/views/utils/mapViewFiltersToFilters'; import { mapViewSortsToSorts } from '@/views/utils/mapViewSortsToSorts'; +import { useNavigate } from 'react-router-dom'; import { isDeeplyEqual } from '~/utils/isDeeplyEqual'; const StyledContainer = styled.div` @@ -108,6 +124,63 @@ export const RecordIndexContainer = ({ [columnDefinitions, setTableColumns], ); + const navigate = useNavigate(); + + const { records: views } = usePrefetchedData<View>(PrefetchKey.AllViews); + + const currentViewId = useRecoilValue( + currentViewIdComponentState({ + scopeId: recordIndexId, + }), + ); + + const view = findView({ + objectMetadataItemId: objectMetadataItem?.id ?? '', + viewId: currentViewId ?? null, + views, + }); + + const filter = turnObjectDropdownFilterIntoQueryFilter( + mapViewFiltersToFilters(view?.viewFilters ?? [], filterDefinitions), + objectMetadataItem?.fields ?? [], + ); + + const orderBy = turnSortsIntoOrderBy( + objectMetadataItem, + mapViewSortsToSorts(view?.viewSorts ?? [], sortDefinitions), + ); + + const { findCursorInCache } = useFindRecordCursorFromFindManyCacheRootQuery({ + fieldVariables: { + filter, + orderBy, + }, + objectNamePlural: objectNamePlural, + }); + + const handleIndexIdentifierClick = (recordId: string) => { + const cursor = findCursorInCache(recordId); + + // TODO: use URL builder + navigate( + `/object/${objectNameSingular}/${recordId}?view=${currentViewId}`, + { + state: { + cursor, + }, + }, + ); + }; + + const handleIndexRecordsLoaded = useRecoilCallback( + ({ set }) => + () => { + // TODO: find a better way to reset this state ? + set(lastShowPageRecordIdState, null); + }, + [], + ); + return ( <StyledContainer> <RecordFieldValueSelectorContextProvider> @@ -153,41 +226,46 @@ export const RecordIndexContainer = ({ /> </StyledContainerWithPadding> </SpreadsheetImportProvider> - - {recordIndexViewType === ViewType.Table && ( - <> - <RecordIndexTableContainer - recordTableId={recordIndexId} - viewBarId={recordIndexId} - objectNameSingular={objectNameSingular} - createRecord={createRecord} - /> - <RecordIndexTableContainerEffect - objectNameSingular={objectNameSingular} - recordTableId={recordIndexId} - viewBarId={recordIndexId} - /> - </> - )} - - {recordIndexViewType === ViewType.Kanban && ( - <StyledContainerWithPadding> - <RecordIndexBoardContainer - recordBoardId={recordIndexId} - viewBarId={recordIndexId} - objectNameSingular={objectNameSingular} - createRecord={createRecord} - /> - <RecordIndexBoardDataLoader - objectNameSingular={objectNameSingular} - recordBoardId={recordIndexId} - /> - <RecordIndexBoardDataLoaderEffect - objectNameSingular={objectNameSingular} - recordBoardId={recordIndexId} - /> - </StyledContainerWithPadding> - )} + <RecordIndexEventContext.Provider + value={{ + onIndexIdentifierClick: handleIndexIdentifierClick, + onIndexRecordsLoaded: handleIndexRecordsLoaded, + }} + > + {recordIndexViewType === ViewType.Table && ( + <> + <RecordIndexTableContainer + recordTableId={recordIndexId} + viewBarId={recordIndexId} + objectNameSingular={objectNameSingular} + createRecord={createRecord} + /> + <RecordIndexTableContainerEffect + objectNameSingular={objectNameSingular} + recordTableId={recordIndexId} + viewBarId={recordIndexId} + /> + </> + )} + {recordIndexViewType === ViewType.Kanban && ( + <StyledContainerWithPadding> + <RecordIndexBoardContainer + recordBoardId={recordIndexId} + viewBarId={recordIndexId} + objectNameSingular={objectNameSingular} + createRecord={createRecord} + /> + <RecordIndexBoardDataLoader + objectNameSingular={objectNameSingular} + recordBoardId={recordIndexId} + /> + <RecordIndexBoardDataLoaderEffect + objectNameSingular={objectNameSingular} + recordBoardId={recordIndexId} + /> + </StyledContainerWithPadding> + )} + </RecordIndexEventContext.Provider> </RecordFieldValueSelectorContextProvider> </StyledContainer> ); diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexRecordChip.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexRecordChip.tsx new file mode 100644 index 000000000000..1a064570ca56 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexRecordChip.tsx @@ -0,0 +1,40 @@ +import { AvatarChip, AvatarChipVariant } from 'twenty-ui'; + +import { useRecordChipData } from '@/object-record/hooks/useRecordChipData'; +import { RecordIndexEventContext } from '@/object-record/record-index/contexts/RecordIndexEventContext'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { useContext } from 'react'; + +export type RecordIndexRecordChipProps = { + objectNameSingular: string; + record: ObjectRecord; + variant?: AvatarChipVariant; +}; + +export const RecordIndexRecordChip = ({ + objectNameSingular, + record, + variant, +}: RecordIndexRecordChipProps) => { + const { onIndexIdentifierClick } = useContext(RecordIndexEventContext); + + const { recordChipData } = useRecordChipData({ + objectNameSingular, + record, + }); + + const handleAvatarChipClick = () => { + onIndexIdentifierClick(record.id); + }; + + return ( + <AvatarChip + placeholderColorSeed={record.id} + name={recordChipData.name} + avatarType={recordChipData.avatarType} + avatarUrl={recordChipData.avatarUrl ?? ''} + onClick={handleAvatarChipClick} + variant={variant} + /> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RemoveSortingModal.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexRemoveSortingModal.tsx similarity index 96% rename from packages/twenty-front/src/modules/object-record/record-table/components/RemoveSortingModal.tsx rename to packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexRemoveSortingModal.tsx index 9bfb93b60430..efe7e4cb9236 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RemoveSortingModal.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexRemoveSortingModal.tsx @@ -5,7 +5,7 @@ import { ConfirmationModal } from '@/ui/layout/modal/components/ConfirmationModa import { useCombinedViewSorts } from '@/views/hooks/useCombinedViewSorts'; import { useGetCurrentView } from '@/views/hooks/useGetCurrentView'; -export const RemoveSortingModal = ({ +export const RecordIndexRemoveSortingModal = ({ recordTableId, }: { recordTableId: string; diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainer.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainer.tsx index 31be052ff407..f5af2e96fa20 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainer.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainer.tsx @@ -1,8 +1,8 @@ import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; import { RecordUpdateHookParams } from '@/object-record/record-field/contexts/FieldContext'; +import { RecordIndexRemoveSortingModal } from '@/object-record/record-index/components/RecordIndexRemoveSortingModal'; import { RecordTableActionBar } from '@/object-record/record-table/action-bar/components/RecordTableActionBar'; import { RecordTableWithWrappers } from '@/object-record/record-table/components/RecordTableWithWrappers'; -import { RemoveSortingModal } from '@/object-record/record-table/components/RemoveSortingModal'; import { RecordTableContextMenu } from '@/object-record/record-table/context-menu/components/RecordTableContextMenu'; type RecordIndexTableContainerProps = { @@ -39,7 +39,7 @@ export const RecordIndexTableContainer = ({ createRecord={createRecord} /> <RecordTableActionBar recordTableId={recordTableId} /> - <RemoveSortingModal recordTableId={recordTableId} /> + <RecordIndexRemoveSortingModal recordTableId={recordTableId} /> <RecordTableContextMenu recordTableId={recordTableId} /> </> ); diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainerEffect.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainerEffect.tsx index 6677d042665d..32ae2ffc5091 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainerEffect.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexTableContainerEffect.tsx @@ -6,7 +6,9 @@ import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadata import { useRecordActionBar } from '@/object-record/record-action-bar/hooks/useRecordActionBar'; import { useHandleToggleColumnFilter } from '@/object-record/record-index/hooks/useHandleToggleColumnFilter'; import { useHandleToggleColumnSort } from '@/object-record/record-index/hooks/useHandleToggleColumnSort'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; import { useRecordTable } from '@/object-record/record-table/hooks/useRecordTable'; +import { useViewStates } from '@/views/hooks/internal/useViewStates'; import { useSetRecordCountInCurrentView } from '@/views/hooks/useSetRecordCountInCurrentView'; type RecordIndexTableContainerEffectProps = { @@ -45,12 +47,31 @@ export const RecordIndexTableContainerEffect = ({ setAvailableTableColumns(columnDefinitions); }, [columnDefinitions, setAvailableTableColumns]); + const { tableRowIdsState, hasUserSelectedAllRowsState } = + useRecordTableStates(recordTableId); + + const { entityCountInCurrentViewState } = useViewStates(recordTableId); + const entityCountInCurrentView = useRecoilValue( + entityCountInCurrentViewState, + ); + const hasUserSelectedAllRows = useRecoilValue(hasUserSelectedAllRowsState); + const tableRowIds = useRecoilValue(tableRowIdsState); + const selectedRowIds = useRecoilValue(selectedRowIdsSelector()); + const numSelected = + hasUserSelectedAllRows && entityCountInCurrentView + ? selectedRowIds.length === tableRowIds.length + ? entityCountInCurrentView + : entityCountInCurrentView - + (tableRowIds.length - selectedRowIds.length) // unselected row Ids + : selectedRowIds.length; + const { setActionBarEntries, setContextMenuEntries } = useRecordActionBar({ objectMetadataItem, selectedRecordIds: selectedRowIds, callback: resetTableRowSelection, + totalNumberOfRecordsSelected: numSelected, }); const handleToggleColumnFilter = useHandleToggleColumnFilter({ diff --git a/packages/twenty-front/src/modules/object-record/record-index/contexts/RecordIndexEventContext.tsx b/packages/twenty-front/src/modules/object-record/record-index/contexts/RecordIndexEventContext.tsx new file mode 100644 index 000000000000..7de19221dc79 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-index/contexts/RecordIndexEventContext.tsx @@ -0,0 +1,9 @@ +import { createEventContext } from '~/utils/createEventContext'; + +export type RecordIndexEventContextProps = { + onIndexIdentifierClick: (recordId: string) => void; + onIndexRecordsLoaded: () => void; +}; + +export const RecordIndexEventContext = + createEventContext<RecordIndexEventContextProps>(); diff --git a/packages/twenty-front/src/modules/object-record/record-index/hooks/useHandleToggleColumnFilter.ts b/packages/twenty-front/src/modules/object-record/record-index/hooks/useHandleToggleColumnFilter.ts index 774e604178af..18997fbadbae 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/hooks/useHandleToggleColumnFilter.ts +++ b/packages/twenty-front/src/modules/object-record/record-index/hooks/useHandleToggleColumnFilter.ts @@ -1,4 +1,5 @@ import { useCallback } from 'react'; +import { v4 } from 'uuid'; import { useColumnDefinitionsFromFieldMetadata } from '@/object-metadata/hooks/useColumnDefinitionsFromFieldMetadata'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; @@ -46,6 +47,7 @@ export const useHandleToggleColumnFilter = ({ const defaultOperand = availableOperandsForFilter[0]; const newFilter: Filter = { + id: v4(), fieldMetadataId, operand: defaultOperand, displayValue: '', @@ -60,8 +62,8 @@ export const useHandleToggleColumnFilter = ({ upsertCombinedViewFilter(newFilter); - openDropdown(fieldMetadataId, { - scope: fieldMetadataId, + openDropdown(newFilter.id, { + scope: newFilter.id, }); }, [columnDefinitions, upsertCombinedViewFilter, openDropdown], diff --git a/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoard.ts b/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoard.ts index 9cd621effee5..297f1dcf8088 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoard.ts +++ b/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoard.ts @@ -11,7 +11,7 @@ import { recordIndexFieldDefinitionsState } from '@/object-record/record-index/s import { recordIndexFiltersState } from '@/object-record/record-index/states/recordIndexFiltersState'; import { recordIndexIsCompactModeActiveState } from '@/object-record/record-index/states/recordIndexIsCompactModeActiveState'; import { recordIndexSortsState } from '@/object-record/record-index/states/recordIndexSortsState'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; import { useSetRecordCountInCurrentView } from '@/views/hooks/useSetRecordCountInCurrentView'; type UseLoadRecordIndexBoardProps = { @@ -33,7 +33,7 @@ export const useLoadRecordIndexBoard = ({ setFieldDefinitions, isCompactModeActiveState, } = useRecordBoard(recordBoardId); - const { setRecords: setRecordsInStore } = useSetRecordInStore(); + const { upsertRecords: upsertRecordsInStore } = useUpsertRecordsInStore(); const recordIndexFieldDefinitions = useRecoilValue( recordIndexFieldDefinitionsState, @@ -82,8 +82,8 @@ export const useLoadRecordIndexBoard = ({ }, [records, setRecordIdsInBoard]); useEffect(() => { - setRecordsInStore(records); - }, [records, setRecordsInStore]); + upsertRecordsInStore(records); + }, [records, upsertRecordsInStore]); useEffect(() => { setRecordCountInCurrentView(totalCount); diff --git a/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoardColumn.ts b/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoardColumn.ts index a65d67abf679..02485fe0d78b 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoardColumn.ts +++ b/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexBoardColumn.ts @@ -9,13 +9,14 @@ import { turnObjectDropdownFilterIntoQueryFilter } from '@/object-record/record- import { useRecordBoardRecordGqlFields } from '@/object-record/record-index/hooks/useRecordBoardRecordGqlFields'; import { recordIndexFiltersState } from '@/object-record/record-index/states/recordIndexFiltersState'; import { recordIndexSortsState } from '@/object-record/record-index/states/recordIndexSortsState'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; +import { isDefined } from '~/utils/isDefined'; type UseLoadRecordIndexBoardProps = { objectNameSingular: string; boardFieldMetadataId: string | null; recordBoardId: string; - columnFieldSelectValue: string; + columnFieldSelectValue: string | null; columnId: string; }; @@ -30,7 +31,7 @@ export const useLoadRecordIndexBoardColumn = ({ objectNameSingular, }); const { setRecordIdsForColumn } = useRecordBoard(recordBoardId); - const { setRecords: setRecordsInStore } = useSetRecordInStore(); + const { upsertRecords: upsertRecordsInStore } = useUpsertRecordsInStore(); const recordIndexFilters = useRecoilValue(recordIndexFiltersState); const recordIndexSorts = useRecoilValue(recordIndexSortsState); @@ -51,9 +52,11 @@ export const useLoadRecordIndexBoardColumn = ({ const filter = { ...requestFilters, - [recordIndexKanbanFieldMetadataItem?.name ?? '']: { - in: [columnFieldSelectValue], - }, + [recordIndexKanbanFieldMetadataItem?.name ?? '']: isDefined( + columnFieldSelectValue, + ) + ? { in: [columnFieldSelectValue] } + : { is: 'NULL' }, }; const { @@ -75,8 +78,8 @@ export const useLoadRecordIndexBoardColumn = ({ }, [records, setRecordIdsForColumn, columnId]); useEffect(() => { - setRecordsInStore(records); - }, [records, setRecordsInStore]); + upsertRecordsInStore(records); + }, [records, upsertRecordsInStore]); return { records, diff --git a/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexTable.ts b/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexTable.ts index c6a26daada1d..99f613521b2d 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexTable.ts +++ b/packages/twenty-front/src/modules/object-record/record-index/hooks/useLoadRecordIndexTable.ts @@ -1,4 +1,4 @@ -import { useRecoilValue, useSetRecoilState } from 'recoil'; +import { useRecoilValue } from 'recoil'; import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; @@ -41,8 +41,6 @@ export const useLoadRecordIndexTable = (objectNameSingular: string) => { const { setRecordTableData, setIsRecordTableInitialLoading } = useRecordTable(); - const { tableLastRowVisibleState } = useRecordTableStates(); - const setLastRowVisible = useSetRecoilState(tableLastRowVisibleState); const currentWorkspace = useRecoilValue(currentWorkspaceState); const params = useFindManyParams(objectNameSingular); @@ -58,7 +56,6 @@ export const useLoadRecordIndexTable = (objectNameSingular: string) => { ...params, recordGqlFields, onCompleted: () => { - setLastRowVisible(false); setIsRecordTableInitialLoading(false); }, onError: () => { diff --git a/packages/twenty-front/src/modules/object-record/record-index/options/components/RecordIndexOptionsDropdownContent.tsx b/packages/twenty-front/src/modules/object-record/record-index/options/components/RecordIndexOptionsDropdownContent.tsx index 73149fb6eba5..c93c8bfdadf1 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/options/components/RecordIndexOptionsDropdownContent.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/options/components/RecordIndexOptionsDropdownContent.tsx @@ -132,6 +132,7 @@ export const RecordIndexOptionsDropdownContent = ({ onClick={() => handleSelectMenu('fields')} LeftIcon={IconTag} text="Fields" + hasSubMenu /> <MenuItem onClick={() => openRecordSpreadsheetImport()} diff --git a/packages/twenty-front/src/modules/object-record/record-index/options/hooks/__tests__/useExportTableData.test.ts b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/__tests__/useExportTableData.test.ts index f5d35d92bcbe..b7dfc586016f 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/options/hooks/__tests__/useExportTableData.test.ts +++ b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/__tests__/useExportTableData.test.ts @@ -40,6 +40,7 @@ describe('generateCsv', () => { ] as ColumnDefinition<FieldMetadata>[]; const rows = [ { + id: '1', bar: 'another field', empty: null, foo: 'some field', @@ -48,8 +49,8 @@ describe('generateCsv', () => { }, ]; const csv = generateCsv({ columns, rows }); - expect(csv).toEqual(`Foo,Empty,Nested Foo,Nested Nested,Relation -some field,,foo,nested,a relation`); + expect(csv).toEqual(`Id,Foo,Empty,Nested Foo,Nested Nested,Relation +1,some field,,foo,nested,a relation`); }); }); @@ -62,6 +63,7 @@ describe('csvDownloader', () => { { id: 2, name: 'Alice' }, ], columns: [], + objectNameSingular: '', }; const link = document.createElement('a'); diff --git a/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useDeleteTableData.ts b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useDeleteTableData.ts new file mode 100644 index 000000000000..8c2fca1b9ede --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useDeleteTableData.ts @@ -0,0 +1,76 @@ +import { useFavorites } from '@/favorites/hooks/useFavorites'; +import { useDeleteManyRecords } from '@/object-record/hooks/useDeleteManyRecords'; +import { useFetchAllRecordIds } from '@/object-record/hooks/useFetchAllRecordIds'; +import { UseTableDataOptions } from '@/object-record/record-index/options/hooks/useTableData'; +import { useRecordTable } from '@/object-record/record-table/hooks/useRecordTable'; +import { tableRowIdsComponentState } from '@/object-record/record-table/states/tableRowIdsComponentState'; +import { getScopeIdFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdFromComponentId'; +import { useRecoilValue } from 'recoil'; + +type UseDeleteTableDataOptions = Omit<UseTableDataOptions, 'callback'>; + +export const useDeleteTableData = ({ + objectNameSingular, + recordIndexId, +}: UseDeleteTableDataOptions) => { + const { fetchAllRecordIds } = useFetchAllRecordIds({ + objectNameSingular, + }); + + const { + resetTableRowSelection, + selectedRowIdsSelector, + hasUserSelectedAllRowsState, + } = useRecordTable({ + recordTableId: recordIndexId, + }); + + const tableRowIds = useRecoilValue( + tableRowIdsComponentState({ + scopeId: getScopeIdFromComponentId(recordIndexId), + }), + ); + + const { deleteManyRecords } = useDeleteManyRecords({ + objectNameSingular, + }); + const { favorites, deleteFavorite } = useFavorites(); + + const selectedRowIds = useRecoilValue(selectedRowIdsSelector()); + + const hasUserSelectedAllRows = useRecoilValue(hasUserSelectedAllRowsState); + + const deleteRecords = async () => { + let recordIdsToDelete = selectedRowIds; + + if (hasUserSelectedAllRows) { + const allRecordIds = await fetchAllRecordIds(); + + const unselectedRecordIds = tableRowIds.filter( + (recordId) => !selectedRowIds.includes(recordId), + ); + + recordIdsToDelete = allRecordIds.filter( + (recordId) => !unselectedRecordIds.includes(recordId), + ); + } + + resetTableRowSelection(); + + for (const recordIdToDelete of recordIdsToDelete) { + const foundFavorite = favorites?.find( + (favorite) => favorite.recordId === recordIdToDelete, + ); + + if (foundFavorite !== undefined) { + deleteFavorite(foundFavorite.id); + } + } + + await deleteManyRecords(recordIdsToDelete, { + delayInMsBetweenRequests: 50, + }); + }; + + return { deleteTableData: deleteRecords }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useExportTableData.ts b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useExportTableData.ts index 785bb8f51d94..c186bf1e2903 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useExportTableData.ts +++ b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useExportTableData.ts @@ -1,16 +1,16 @@ -import { useEffect, useState } from 'react'; +import { useMemo } from 'react'; import { json2csv } from 'json-2-csv'; -import { useRecoilValue } from 'recoil'; -import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { + useTableData, + UseTableDataOptions, +} from '@/object-record/record-index/options/hooks/useTableData'; import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { FieldMetadataType } from '~/generated/graphql'; import { isDefined } from '~/utils/isDefined'; import { isUndefinedOrNull } from '~/utils/isUndefinedOrNull'; -import { sleep } from '~/utils/sleep'; - -import { useFindManyParams } from '../../hooks/useLoadRecordIndexTable'; export const download = (blob: Blob, filename: string) => { const url = URL.createObjectURL(blob); @@ -45,14 +45,27 @@ export const generateCsv: GenerateExport = ({ col.metadata.relationType === 'TO_ONE_OBJECT', ); - const keys = columnsToExport.flatMap((col) => { + const objectIdColumn: ColumnDefinition<FieldMetadata> = { + fieldMetadataId: '', + type: FieldMetadataType.Uuid, + iconName: '', + label: `Id`, + metadata: { + fieldName: 'id', + }, + position: 0, + size: 0, + }; + + const columnsToExportWithIdColumn = [objectIdColumn, ...columnsToExport]; + + const keys = columnsToExportWithIdColumn.flatMap((col) => { const column = { field: `${col.metadata.fieldName}${col.type === 'RELATION' ? 'Id' : ''}`, title: [col.label, col.type === 'RELATION' ? 'Id' : null] .filter(isDefined) .join(' '), }; - const fieldsWithSubFields = rows.find((row) => { const fieldValue = (row as any)[column.field]; const hasSubFields = @@ -113,13 +126,8 @@ const downloader = (mimeType: string, generator: GenerateExport) => { export const csvDownloader = downloader('text/csv', generateCsv); -type UseExportTableDataOptions = { - delayMs: number; +type UseExportTableDataOptions = Omit<UseTableDataOptions, 'callback'> & { filename: string; - maximumRequests?: number; - objectNameSingular: string; - pageSize?: number; - recordIndexId: string; }; export const useExportTableData = ({ @@ -130,100 +138,22 @@ export const useExportTableData = ({ pageSize = 30, recordIndexId, }: UseExportTableDataOptions) => { - const [isDownloading, setIsDownloading] = useState(false); - const [inflight, setInflight] = useState(false); - const [pageCount, setPageCount] = useState(0); - const [progress, setProgress] = useState<ExportProgress>({ - displayType: 'number', - }); - const [previousRecordCount, setPreviousRecordCount] = useState(0); - - const { visibleTableColumnsSelector, selectedRowIdsSelector } = - useRecordTableStates(recordIndexId); - - const columns = useRecoilValue(visibleTableColumnsSelector()); - const selectedRowIds = useRecoilValue(selectedRowIdsSelector()); - - const hasSelectedRows = selectedRowIds.length > 0; - - const findManyRecordsParams = useFindManyParams( - objectNameSingular, - recordIndexId, - ); - - const selectedFindManyParams = { - ...findManyRecordsParams, - filter: { - ...findManyRecordsParams.filter, - id: { - in: selectedRowIds, + const downloadCsv = useMemo( + () => + (rows: ObjectRecord[], columns: ColumnDefinition<FieldMetadata>[]) => { + csvDownloader(filename, { rows, columns }); }, - }, - }; - - const usedFindManyParams = hasSelectedRows - ? selectedFindManyParams - : findManyRecordsParams; - - // Todo: this needs to be done on click on the Export not button, not to be reactive. Use Lazy query for example - const { totalCount, records, fetchMoreRecords } = useFindManyRecords({ - ...usedFindManyParams, - limit: pageSize, - }); - - useEffect(() => { - const MAXIMUM_REQUESTS = isDefined(totalCount) - ? Math.min(maximumRequests, totalCount / pageSize) - : maximumRequests; - - const downloadCsv = (rows: object[]) => { - csvDownloader(filename, { rows, columns }); - setIsDownloading(false); - setProgress({ - displayType: 'number', - }); - }; - - const fetchNextPage = async () => { - setInflight(true); - setPreviousRecordCount(records.length); - await fetchMoreRecords(); - setPageCount((state) => state + 1); - setProgress({ - exportedRecordCount: records.length, - totalRecordCount: totalCount, - displayType: totalCount ? 'percentage' : 'number', - }); - await sleep(delayMs); - setInflight(false); - }; - - if (!isDownloading || inflight) { - return; - } + [filename], + ); - if ( - pageCount >= MAXIMUM_REQUESTS || - records.length === previousRecordCount - ) { - downloadCsv(records); - } else { - fetchNextPage(); - } - }, [ + const { getTableData: download, progress } = useTableData({ delayMs, - fetchMoreRecords, - filename, - inflight, - isDownloading, - pageCount, - records, - totalCount, - columns, maximumRequests, + objectNameSingular, pageSize, - previousRecordCount, - ]); + recordIndexId, + callback: downloadCsv, + }); - return { progress, download: () => setIsDownloading(true) }; + return { progress, download }; }; diff --git a/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useTableData.ts b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useTableData.ts new file mode 100644 index 000000000000..b389d906dbb8 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-index/options/hooks/useTableData.ts @@ -0,0 +1,204 @@ +import { useEffect, useMemo, useState } from 'react'; +import { useRecoilValue } from 'recoil'; + +import { useLazyFindManyRecords } from '@/object-record/hooks/useLazyFindManyRecords'; +import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { isDefined } from '~/utils/isDefined'; + +import { useFindManyParams } from '../../hooks/useLoadRecordIndexTable'; + +export const sleep = (ms: number) => + new Promise((resolve) => setTimeout(resolve, ms)); + +export const percentage = (part: number, whole: number): number => { + return Math.round((part / whole) * 100); +}; + +export type UseTableDataOptions = { + delayMs: number; + maximumRequests?: number; + objectNameSingular: string; + pageSize?: number; + recordIndexId: string; + callback: ( + rows: ObjectRecord[], + columns: ColumnDefinition<FieldMetadata>[], + ) => void | Promise<void>; +}; + +type ExportProgress = { + exportedRecordCount?: number; + totalRecordCount?: number; + displayType: 'percentage' | 'number'; +}; + +export const useTableData = ({ + delayMs, + maximumRequests = 100, + objectNameSingular, + pageSize = 30, + recordIndexId, + callback, +}: UseTableDataOptions) => { + const [isDownloading, setIsDownloading] = useState(false); + const [inflight, setInflight] = useState(false); + const [pageCount, setPageCount] = useState(0); + const [progress, setProgress] = useState<ExportProgress>({ + displayType: 'number', + }); + const [previousRecordCount, setPreviousRecordCount] = useState(0); + + const { + visibleTableColumnsSelector, + selectedRowIdsSelector, + tableRowIdsState, + hasUserSelectedAllRowsState, + } = useRecordTableStates(recordIndexId); + + const columns = useRecoilValue(visibleTableColumnsSelector()); + const selectedRowIds = useRecoilValue(selectedRowIdsSelector()); + + const hasUserSelectedAllRows = useRecoilValue(hasUserSelectedAllRowsState); + const tableRowIds = useRecoilValue(tableRowIdsState); + + // user has checked select all and then unselected some rows + const userHasUnselectedSomeRows = + hasUserSelectedAllRows && selectedRowIds.length < tableRowIds.length; + + const hasSelectedRows = + selectedRowIds.length > 0 && + !(hasUserSelectedAllRows && selectedRowIds.length === tableRowIds.length); + + const unselectedRowIds = useMemo( + () => + userHasUnselectedSomeRows + ? tableRowIds.filter((id) => !selectedRowIds.includes(id)) + : [], + [userHasUnselectedSomeRows, tableRowIds, selectedRowIds], + ); + + const findManyRecordsParams = useFindManyParams( + objectNameSingular, + recordIndexId, + ); + + const selectedFindManyParams = { + ...findManyRecordsParams, + filter: { + ...findManyRecordsParams.filter, + id: { + in: selectedRowIds, + }, + }, + }; + + const unselectedFindManyParams = { + ...findManyRecordsParams, + filter: { + ...findManyRecordsParams.filter, + not: { + id: { + in: unselectedRowIds, + }, + }, + }, + }; + + const usedFindManyParams = + hasSelectedRows && !userHasUnselectedSomeRows + ? selectedFindManyParams + : userHasUnselectedSomeRows + ? unselectedFindManyParams + : findManyRecordsParams; + + const { + findManyRecords, + totalCount, + records, + fetchMoreRecordsWithPagination, + loading, + } = useLazyFindManyRecords({ + ...usedFindManyParams, + limit: pageSize, + }); + + useEffect(() => { + const MAXIMUM_REQUESTS = isDefined(totalCount) + ? Math.min(maximumRequests, totalCount / pageSize) + : maximumRequests; + + const fetchNextPage = async () => { + setInflight(true); + setPreviousRecordCount(records.length); + + await fetchMoreRecordsWithPagination(); + + setPageCount((state) => state + 1); + setProgress({ + exportedRecordCount: records.length, + totalRecordCount: totalCount, + displayType: totalCount ? 'percentage' : 'number', + }); + await sleep(delayMs); + setInflight(false); + }; + + if (!isDownloading || inflight || loading) { + return; + } + + if ( + pageCount >= MAXIMUM_REQUESTS || + (isDefined(totalCount) && records.length === totalCount) + ) { + setPageCount(0); + + const complete = () => { + setPageCount(0); + setPreviousRecordCount(0); + setIsDownloading(false); + setProgress({ + displayType: 'number', + }); + }; + + const res = callback(records, columns); + + if (res instanceof Promise) { + res.then(complete); + } else { + complete(); + } + } else { + fetchNextPage(); + } + }, [ + delayMs, + fetchMoreRecordsWithPagination, + inflight, + isDownloading, + pageCount, + records, + totalCount, + columns, + maximumRequests, + pageSize, + loading, + callback, + previousRecordCount, + ]); + + return { + progress, + isDownloading, + getTableData: () => { + setPageCount(0); + setPreviousRecordCount(0); + setIsDownloading(true); + findManyRecords?.(); + }, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-index/options/states/recordDeleteProgressState.ts b/packages/twenty-front/src/modules/object-record/record-index/options/states/recordDeleteProgressState.ts new file mode 100644 index 000000000000..e69de29bb2d1 diff --git a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCell.tsx b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCell.tsx index 8eef4e126c5c..f7927be24ece 100644 --- a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCell.tsx +++ b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCell.tsx @@ -6,7 +6,6 @@ import { FieldInput } from '@/object-record/record-field/components/FieldInput'; import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; import { FieldFocusContextProvider } from '@/object-record/record-field/contexts/FieldFocusContextProvider'; import { useGetButtonIcon } from '@/object-record/record-field/hooks/useGetButtonIcon'; -import { useIsFieldEmpty } from '@/object-record/record-field/hooks/useIsFieldEmpty'; import { useIsFieldInputOnly } from '@/object-record/record-field/hooks/useIsFieldInputOnly'; import { FieldInputEvent } from '@/object-record/record-field/types/FieldInputEvent'; import { isFieldRelation } from '@/object-record/record-field/types/guards/isFieldRelation'; @@ -26,11 +25,8 @@ export const RecordInlineCell = ({ loading, }: RecordInlineCellProps) => { const { fieldDefinition, entityId } = useContext(FieldContext); - const buttonIcon = useGetButtonIcon(); - const isFieldEmpty = useIsFieldEmpty(); - const isFieldInputOnly = useIsFieldInputOnly(); const { closeInlineCell } = useInlineCell(); @@ -104,7 +100,6 @@ export const RecordInlineCell = ({ /> } displayModeContent={<FieldDisplay />} - isDisplayModeContentEmpty={isFieldEmpty} isDisplayModeFixHeight editModeContentOnly={isFieldInputOnly} loading={loading} diff --git a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellContainer.tsx b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellContainer.tsx index 926d934c8f8e..4e6073c9c383 100644 --- a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellContainer.tsx +++ b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellContainer.tsx @@ -1,8 +1,7 @@ import React, { ReactElement, useContext } from 'react'; -import { Tooltip } from 'react-tooltip'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; -import { IconComponent } from 'twenty-ui'; +import { AppTooltip, IconComponent, TooltipDelay } from 'twenty-ui'; import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; import { useFieldFocus } from '@/object-record/record-field/hooks/useFieldFocus'; @@ -34,7 +33,8 @@ const StyledLabelAndIconContainer = styled.div` const StyledValueContainer = styled.div` display: flex; - min-width: 100%; + flex-grow: 1; + min-width: 0; `; const StyledLabelContainer = styled.div<{ width?: number }>` @@ -54,17 +54,6 @@ const StyledInlineCellBaseContainer = styled.div` user-select: none; `; -const StyledTooltip = styled(Tooltip)` - background-color: ${({ theme }) => theme.background.primary}; - box-shadow: ${({ theme }) => theme.boxShadow.light}; - - color: ${({ theme }) => theme.font.color.primary}; - - font-size: ${({ theme }) => theme.font.size.sm}; - font-weight: ${({ theme }) => theme.font.weight.regular}; - padding: ${({ theme }) => theme.spacing(2)}; -`; - export const StyledSkeletonDiv = styled.div` height: 24px; `; @@ -80,7 +69,6 @@ export type RecordInlineCellContainerProps = { editModeContentOnly?: boolean; displayModeContent: ReactElement; customEditHotkeyScope?: HotkeyScope; - isDisplayModeContentEmpty?: boolean; isDisplayModeFixHeight?: boolean; disableHoverEffect?: boolean; loading?: boolean; @@ -96,7 +84,6 @@ export const RecordInlineCellContainer = ({ editModeContent, displayModeContent, customEditHotkeyScope, - isDisplayModeContentEmpty, editModeContentOnly, isDisplayModeFixHeight, disableHoverEffect, @@ -140,13 +127,14 @@ export const RecordInlineCellContainer = ({ )} {/* TODO: Displaying Tooltips on the board is causing performance issues https://react-tooltip.com/docs/examples/render */} {!showLabel && !fieldDefinition?.disableTooltip && ( - <StyledTooltip + <AppTooltip anchorSelect={`#${labelId}`} content={label} clickable noArrow place="bottom" positionStrategy="fixed" + delay={TooltipDelay.shortDelay} /> )} </StyledLabelAndIconContainer> @@ -159,7 +147,6 @@ export const RecordInlineCellContainer = ({ disableHoverEffect, editModeContent, editModeContentOnly, - isDisplayModeContentEmpty, isDisplayModeFixHeight, buttonIcon, label, diff --git a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellDisplayMode.tsx b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellDisplayMode.tsx index 0b8770973598..eacdf2dc4b7f 100644 --- a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellDisplayMode.tsx +++ b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellDisplayMode.tsx @@ -1,13 +1,16 @@ import { css } from '@emotion/react'; import styled from '@emotion/styled'; +import { useFieldFocus } from '@/object-record/record-field/hooks/useFieldFocus'; +import { useIsFieldEmpty } from '@/object-record/record-field/hooks/useIsFieldEmpty'; +import { useIsFieldInputOnly } from '@/object-record/record-field/hooks/useIsFieldInputOnly'; +import { RecordInlineCellContainerProps } from '@/object-record/record-inline-cell/components/RecordInlineCellContainer'; +import { RecordInlineCellButton } from '@/object-record/record-inline-cell/components/RecordInlineCellEditButton'; + const StyledRecordInlineCellNormalModeOuterContainer = styled.div< Pick< RecordInlineCellDisplayModeProps, - | 'isDisplayModeContentEmpty' - | 'disableHoverEffect' - | 'isDisplayModeFixHeight' - | 'isHovered' + 'disableHoverEffect' | 'isDisplayModeFixHeight' | 'isHovered' > >` align-items: center; @@ -51,33 +54,50 @@ const StyledEmptyField = styled.div` `; type RecordInlineCellDisplayModeProps = { - isDisplayModeContentEmpty?: boolean; disableHoverEffect?: boolean; isDisplayModeFixHeight?: boolean; isHovered?: boolean; emptyPlaceholder?: string; -}; +} & Pick<RecordInlineCellContainerProps, 'buttonIcon' | 'editModeContentOnly'>; export const RecordInlineCellDisplayMode = ({ children, - isDisplayModeContentEmpty, disableHoverEffect, isDisplayModeFixHeight, emptyPlaceholder = 'Empty', isHovered, -}: React.PropsWithChildren<RecordInlineCellDisplayModeProps>) => ( - <StyledRecordInlineCellNormalModeOuterContainer - isDisplayModeContentEmpty={isDisplayModeContentEmpty} - disableHoverEffect={disableHoverEffect} - isDisplayModeFixHeight={isDisplayModeFixHeight} - isHovered={isHovered} - > - <StyledRecordInlineCellNormalModeInnerContainer> - {isDisplayModeContentEmpty || !children ? ( - <StyledEmptyField>{emptyPlaceholder}</StyledEmptyField> - ) : ( - children - )} - </StyledRecordInlineCellNormalModeInnerContainer> - </StyledRecordInlineCellNormalModeOuterContainer> -); + buttonIcon, + editModeContentOnly, +}: React.PropsWithChildren<RecordInlineCellDisplayModeProps>) => { + const { isFocused } = useFieldFocus(); + const isDisplayModeContentEmpty = useIsFieldEmpty(); + const showEditButton = + buttonIcon && + isFocused && + !isDisplayModeContentEmpty && + !editModeContentOnly; + + const isFieldInputOnly = useIsFieldInputOnly(); + + const shouldDisplayEditModeOnFocus = isFocused && isFieldInputOnly; + + return ( + <> + <StyledRecordInlineCellNormalModeOuterContainer + disableHoverEffect={disableHoverEffect} + isDisplayModeFixHeight={isDisplayModeFixHeight} + isHovered={isHovered} + > + <StyledRecordInlineCellNormalModeInnerContainer> + {(isDisplayModeContentEmpty && !shouldDisplayEditModeOnFocus) || + !children ? ( + <StyledEmptyField>{emptyPlaceholder}</StyledEmptyField> + ) : ( + children + )} + </StyledRecordInlineCellNormalModeInnerContainer> + </StyledRecordInlineCellNormalModeOuterContainer> + {showEditButton && <RecordInlineCellButton Icon={buttonIcon} />} + </> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellEditButton.tsx b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellEditButton.tsx index 0475a925606a..2af1e91d7c21 100644 --- a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellEditButton.tsx +++ b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellEditButton.tsx @@ -1,8 +1,8 @@ import styled from '@emotion/styled'; import { IconComponent } from 'twenty-ui'; -import { AnimatedContainer } from '@/object-record/record-table/components/AnimatedContainer'; import { FloatingIconButton } from '@/ui/input/button/components/FloatingIconButton'; +import { AnimatedContainer } from '@/ui/utilities/animation/components/AnimatedContainer'; const StyledInlineCellButtonContainer = styled.div` align-items: center; diff --git a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellValue.tsx b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellValue.tsx index a1042ddd0593..152ef8927a1f 100644 --- a/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellValue.tsx +++ b/packages/twenty-front/src/modules/object-record/record-inline-cell/components/RecordInlineCellValue.tsx @@ -4,7 +4,6 @@ import styled from '@emotion/styled'; import { useFieldFocus } from '@/object-record/record-field/hooks/useFieldFocus'; import { RecordInlineCellContainerProps } from '@/object-record/record-inline-cell/components/RecordInlineCellContainer'; import { RecordInlineCellDisplayMode } from '@/object-record/record-inline-cell/components/RecordInlineCellDisplayMode'; -import { RecordInlineCellButton } from '@/object-record/record-inline-cell/components/RecordInlineCellEditButton'; import { RecordInlineCellEditMode } from '@/object-record/record-inline-cell/components/RecordInlineCellEditMode'; import { RecordInlineCellSkeletonLoader } from '@/object-record/record-inline-cell/components/RecordInlineCellSkeletonLoader'; import { useInlineCell } from '@/object-record/record-inline-cell/hooks/useInlineCell'; @@ -27,7 +26,6 @@ type RecordInlineCellValueProps = Pick< | 'customEditHotkeyScope' | 'editModeContent' | 'editModeContentOnly' - | 'isDisplayModeContentEmpty' | 'isDisplayModeFixHeight' | 'disableHoverEffect' | 'readonly' @@ -43,7 +41,6 @@ export const RecordInlineCellValue = ({ disableHoverEffect, editModeContent, editModeContentOnly, - isDisplayModeContentEmpty, isDisplayModeFixHeight, readonly, buttonIcon, @@ -61,46 +58,43 @@ export const RecordInlineCellValue = ({ } }; - const showEditButton = - buttonIcon && - !isInlineCellInEditMode && - isFocused && - !editModeContentOnly && - !isDisplayModeContentEmpty; - if (loading === true) { return <RecordInlineCellSkeletonLoader />; } - return !readonly && isInlineCellInEditMode ? ( - <RecordInlineCellEditMode>{editModeContent}</RecordInlineCellEditMode> - ) : editModeContentOnly ? ( - <StyledClickableContainer readonly={readonly}> - <RecordInlineCellDisplayMode - disableHoverEffect={disableHoverEffect} - isDisplayModeContentEmpty={isDisplayModeContentEmpty} - isDisplayModeFixHeight={isDisplayModeFixHeight} - isHovered={isFocused} - emptyPlaceholder={showLabel ? 'Empty' : label} - > - {editModeContent} - </RecordInlineCellDisplayMode> - </StyledClickableContainer> - ) : ( - <StyledClickableContainer - readonly={readonly} - onClick={handleDisplayModeClick} - > - <RecordInlineCellDisplayMode - disableHoverEffect={disableHoverEffect} - isDisplayModeContentEmpty={isDisplayModeContentEmpty} - isDisplayModeFixHeight={isDisplayModeFixHeight} - isHovered={isFocused} - emptyPlaceholder={showLabel ? 'Empty' : label} - > - {displayModeContent} - </RecordInlineCellDisplayMode> - {showEditButton && <RecordInlineCellButton Icon={buttonIcon} />} - </StyledClickableContainer> + return ( + <> + {!readonly && isInlineCellInEditMode && ( + <RecordInlineCellEditMode>{editModeContent}</RecordInlineCellEditMode> + )} + {editModeContentOnly ? ( + <StyledClickableContainer readonly={readonly}> + <RecordInlineCellDisplayMode + disableHoverEffect={disableHoverEffect} + isDisplayModeFixHeight={isDisplayModeFixHeight} + isHovered={isFocused} + emptyPlaceholder={showLabel ? 'Empty' : label} + > + {editModeContent} + </RecordInlineCellDisplayMode> + </StyledClickableContainer> + ) : ( + <StyledClickableContainer + readonly={readonly} + onClick={handleDisplayModeClick} + > + <RecordInlineCellDisplayMode + disableHoverEffect={disableHoverEffect} + isDisplayModeFixHeight={isDisplayModeFixHeight} + isHovered={isFocused} + emptyPlaceholder={showLabel ? 'Empty' : label} + buttonIcon={buttonIcon} + editModeContentOnly={editModeContentOnly} + > + {displayModeContent} + </RecordInlineCellDisplayMode> + </StyledClickableContainer> + )} + </> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-show/hooks/useFindRecordCursorFromFindManyCacheRootQuery.ts b/packages/twenty-front/src/modules/object-record/record-show/hooks/useFindRecordCursorFromFindManyCacheRootQuery.ts new file mode 100644 index 000000000000..878309217a11 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-show/hooks/useFindRecordCursorFromFindManyCacheRootQuery.ts @@ -0,0 +1,34 @@ +import { RecordGqlRefEdge } from '@/object-record/cache/types/RecordGqlRefEdge'; +import { useApolloClient } from '@apollo/client'; +import { createApolloStoreFieldName } from '~/utils/createApolloStoreFieldName'; + +export const useFindRecordCursorFromFindManyCacheRootQuery = ({ + objectNamePlural, + fieldVariables, +}: { + objectNamePlural: string; + fieldVariables: { + filter: any; + orderBy: any; + }; +}) => { + const apollo = useApolloClient(); + + const testsFieldNameOnRootQuery = createApolloStoreFieldName({ + fieldName: objectNamePlural, + fieldVariables: fieldVariables, + }); + + const findCursorInCache = (recordId: string) => { + const extractedCache = apollo.cache.extract() as any; + + const edgesInCache = + extractedCache?.['ROOT_QUERY']?.[testsFieldNameOnRootQuery]?.edges ?? []; + + return edgesInCache.find( + (edge: RecordGqlRefEdge) => edge.node?.__ref.split(':')[1] === recordId, + )?.cursor; + }; + + return { findCursorInCache }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-show/hooks/useRecordIdsFromFindManyCacheRootQuery.ts b/packages/twenty-front/src/modules/object-record/record-show/hooks/useRecordIdsFromFindManyCacheRootQuery.ts new file mode 100644 index 000000000000..522c8112f219 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-show/hooks/useRecordIdsFromFindManyCacheRootQuery.ts @@ -0,0 +1,30 @@ +import { RecordGqlRefEdge } from '@/object-record/cache/types/RecordGqlRefEdge'; +import { useApolloClient } from '@apollo/client'; +import { createApolloStoreFieldName } from '~/utils/createApolloStoreFieldName'; + +export const useRecordIdsFromFindManyCacheRootQuery = ({ + objectNamePlural, + fieldVariables, +}: { + objectNamePlural: string; + fieldVariables: { + filter: any; + orderBy: any; + }; +}) => { + const apollo = useApolloClient(); + + const testsFieldNameOnRootQuery = createApolloStoreFieldName({ + fieldName: objectNamePlural, + fieldVariables: fieldVariables, + }); + + const extractedCache = apollo.cache.extract() as any; + + const recordIdsInCache: string[] = + extractedCache?.['ROOT_QUERY']?.[testsFieldNameOnRootQuery]?.edges?.map( + (edge: RecordGqlRefEdge) => edge.node?.__ref.split(':')[1], + ) ?? []; + + return { recordIdsInCache }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-show/hooks/useRecordShowPagePagination.ts b/packages/twenty-front/src/modules/object-record/record-show/hooks/useRecordShowPagePagination.ts new file mode 100644 index 000000000000..0909fcf78cd7 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-show/hooks/useRecordShowPagePagination.ts @@ -0,0 +1,253 @@ +/* eslint-disable @nx/workspace-no-navigate-prefer-link */ +import { useMemo, useState } from 'react'; +import { + useLocation, + useNavigate, + useParams, + useSearchParams, +} from 'react-router-dom'; +import { useSetRecoilState } from 'recoil'; + +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { formatFieldMetadataItemsAsFilterDefinitions } from '@/object-metadata/utils/formatFieldMetadataItemsAsFilterDefinitions'; +import { formatFieldMetadataItemsAsSortDefinitions } from '@/object-metadata/utils/formatFieldMetadataItemsAsSortDefinitions'; +import { generateDepthOneRecordGqlFields } from '@/object-record/graphql/utils/generateDepthOneRecordGqlFields'; +import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; +import { turnSortsIntoOrderBy } from '@/object-record/object-sort-dropdown/utils/turnSortsIntoOrderBy'; +import { lastShowPageRecordIdState } from '@/object-record/record-field/states/lastShowPageRecordId'; +import { turnObjectDropdownFilterIntoQueryFilter } from '@/object-record/record-filter/utils/turnObjectDropdownFilterIntoQueryFilter'; +import { useRecordIdsFromFindManyCacheRootQuery } from '@/object-record/record-show/hooks/useRecordIdsFromFindManyCacheRootQuery'; +import { usePrefetchedData } from '@/prefetch/hooks/usePrefetchedData'; +import { PrefetchKey } from '@/prefetch/types/PrefetchKey'; +import { View } from '@/views/types/View'; +import { mapViewFiltersToFilters } from '@/views/utils/mapViewFiltersToFilters'; +import { mapViewSortsToSorts } from '@/views/utils/mapViewSortsToSorts'; +import { isNonEmptyString } from '@sniptt/guards'; +import { capitalize } from '~/utils/string/capitalize'; + +export const findView = ({ + viewId, + objectMetadataItemId, + views, +}: { + viewId: string | null; + objectMetadataItemId: string; + views: View[]; +}) => { + if (!viewId) { + return views.find( + (view: any) => + view.key === 'INDEX' && view?.objectMetadataId === objectMetadataItemId, + ) as View; + } else { + return views.find( + (view: any) => + view?.id === viewId && view?.objectMetadataId === objectMetadataItemId, + ) as View; + } +}; + +export const useRecordShowPagePagination = ( + propsObjectNameSingular: string, + propsObjectRecordId: string, +) => { + const { + objectNameSingular: paramObjectNameSingular, + objectRecordId: paramObjectRecordId, + } = useParams(); + const navigate = useNavigate(); + const [searchParams] = useSearchParams(); + const viewIdQueryParam = searchParams.get('view'); + + const setLastShowPageRecordId = useSetRecoilState(lastShowPageRecordIdState); + + const [isLoadedRecords, setIsLoadedRecords] = useState(false); + + const objectNameSingular = propsObjectNameSingular || paramObjectNameSingular; + const objectRecordId = propsObjectRecordId || paramObjectRecordId; + + if (!objectNameSingular || !objectRecordId) { + throw new Error('Object name or Record id is not defined'); + } + + const { objectMetadataItem } = useObjectMetadataItem({ objectNameSingular }); + const { records: views } = usePrefetchedData<View>(PrefetchKey.AllViews); + + const view = useMemo(() => { + return findView({ + objectMetadataItemId: objectMetadataItem?.id ?? '', + viewId: viewIdQueryParam, + views, + }); + }, [viewIdQueryParam, objectMetadataItem, views]); + + const activeFieldMetadataItems = useMemo( + () => + objectMetadataItem + ? objectMetadataItem.fields.filter( + ({ isActive, isSystem }) => isActive && !isSystem, + ) + : [], + [objectMetadataItem], + ); + + const filterDefinitions = formatFieldMetadataItemsAsFilterDefinitions({ + fields: activeFieldMetadataItems, + }); + + const sortDefinitions = formatFieldMetadataItemsAsSortDefinitions({ + fields: activeFieldMetadataItems, + }); + + const filter = turnObjectDropdownFilterIntoQueryFilter( + mapViewFiltersToFilters(view?.viewFilters ?? [], filterDefinitions), + objectMetadataItem?.fields ?? [], + ); + + const orderBy = turnSortsIntoOrderBy( + objectMetadataItem, + mapViewSortsToSorts(view?.viewSorts ?? [], sortDefinitions), + ); + + const recordGqlFields = generateDepthOneRecordGqlFields({ + objectMetadataItem, + }); + + const { state } = useLocation(); + + const cursorFromIndexPage = state?.cursor; + + const { loading: loadingCurrentRecord, pageInfo: currentRecordsPageInfo } = + useFindManyRecords({ + filter: { + id: { eq: objectRecordId }, + }, + orderBy, + skip: isLoadedRecords, + limit: 1, + objectNameSingular, + recordGqlFields, + }); + + const currentRecordCursor = currentRecordsPageInfo?.endCursor; + + const cursor = cursorFromIndexPage ?? currentRecordCursor; + + const { + loading: loadingRecordBefore, + records: recordsBefore, + pageInfo: pageInfoBefore, + totalCount: totalCountBefore, + } = useFindManyRecords({ + filter, + orderBy, + skip: isLoadedRecords, + cursorFilter: isNonEmptyString(cursor) + ? { + cursorDirection: 'before', + cursor: cursor, + limit: 1, + } + : undefined, + objectNameSingular, + recordGqlFields, + }); + + const { + loading: loadingRecordAfter, + records: recordsAfter, + pageInfo: pageInfoAfter, + totalCount: totalCountAfter, + } = useFindManyRecords({ + filter, + orderBy, + skip: isLoadedRecords, + cursorFilter: cursor + ? { + cursorDirection: 'after', + cursor: cursor, + limit: 1, + } + : undefined, + objectNameSingular, + recordGqlFields, + }); + + const totalCount = Math.max(totalCountBefore ?? 0, totalCountAfter ?? 0); + + const loading = + loadingRecordAfter || loadingRecordBefore || loadingCurrentRecord; + + const isThereARecordBefore = recordsBefore.length > 0; + const isThereARecordAfter = recordsAfter.length > 0; + + const recordBefore = recordsBefore[0]; + const recordAfter = recordsAfter[0]; + + const recordBeforeCursor = pageInfoBefore?.endCursor; + const recordAfterCursor = pageInfoAfter?.endCursor; + + const navigateToPreviousRecord = () => { + navigate( + `/object/${objectNameSingular}/${recordBefore.id}${ + viewIdQueryParam ? `?view=${viewIdQueryParam}` : '' + }`, + { + state: { + cursor: recordBeforeCursor, + }, + }, + ); + }; + + const navigateToNextRecord = () => { + navigate( + `/object/${objectNameSingular}/${recordAfter.id}${ + viewIdQueryParam ? `?view=${viewIdQueryParam}` : '' + }`, + { + state: { + cursor: recordAfterCursor, + }, + }, + ); + }; + + const navigateToIndexView = () => { + const indexPath = `/objects/${objectMetadataItem.namePlural}${ + viewIdQueryParam ? `?view=${viewIdQueryParam}` : '' + }`; + + setLastShowPageRecordId(objectRecordId); + + navigate(indexPath); + }; + + const { recordIdsInCache } = useRecordIdsFromFindManyCacheRootQuery({ + objectNamePlural: objectMetadataItem.namePlural, + fieldVariables: { + filter, + orderBy, + }, + }); + + const rankInView = recordIdsInCache.findIndex((id) => id === objectRecordId); + + const rankFoundInFiew = rankInView > -1; + + const objectLabel = capitalize(objectMetadataItem.namePlural); + + const viewNameWithCount = rankFoundInFiew + ? `${rankInView + 1} of ${totalCount} in ${objectLabel}` + : `${objectLabel} (${totalCount})`; + + return { + viewName: viewNameWithCount, + hasPreviousRecord: isThereARecordBefore, + isLoadingPagination: loading, + hasNextRecord: isThereARecordAfter, + navigateToPreviousRecord, + navigateToNextRecord, + navigateToIndexView, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailDuplicatesSection.tsx b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailDuplicatesSection.tsx index 3a82313b87e7..74690db36295 100644 --- a/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailDuplicatesSection.tsx +++ b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailDuplicatesSection.tsx @@ -12,18 +12,19 @@ export const RecordDetailDuplicatesSection = ({ objectRecordId: string; objectNameSingular: string; }) => { - const { records: duplicateRecords } = useFindDuplicateRecords({ - objectRecordId, + const { results: queryResults } = useFindDuplicateRecords({ + objectRecordIds: [objectRecordId], objectNameSingular, }); - if (!duplicateRecords.length) return null; + if (!queryResults || !queryResults[0] || queryResults[0].length === 0) + return null; return ( <RecordDetailSection> <RecordDetailSectionHeader title="Duplicates" /> <RecordDetailRecordsList> - {duplicateRecords.slice(0, 5).map((duplicateRecord) => ( + {queryResults[0].slice(0, 5).map((duplicateRecord) => ( <RecordDetailRecordsListItem key={duplicateRecord.id}> <RecordChip record={duplicateRecord} diff --git a/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsList.tsx b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsList.tsx index c8ce45c0b9ad..e4dae45402b3 100644 --- a/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsList.tsx +++ b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsList.tsx @@ -1,7 +1,8 @@ -import { useState } from 'react'; +import { Fragment, useState } from 'react'; import { RecordDetailRecordsList } from '@/object-record/record-show/record-detail-section/components/RecordDetailRecordsList'; import { RecordDetailRelationRecordsListItem } from '@/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItem'; +import { RecordDetailRelationRecordsListItemEffect } from '@/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItemEffect'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; type RecordDetailRelationRecordsListProps = { @@ -19,12 +20,18 @@ export const RecordDetailRelationRecordsList = ({ return ( <RecordDetailRecordsList> {relationRecords.slice(0, 5).map((relationRecord) => ( - <RecordDetailRelationRecordsListItem - key={relationRecord.id} - isExpanded={expandedItem === relationRecord.id} - onClick={handleItemClick} - relationRecord={relationRecord} - /> + <Fragment key={relationRecord.id}> + <RecordDetailRelationRecordsListItemEffect + key={`${relationRecord.id}-effect`} + relationRecordId={relationRecord.id} + /> + <RecordDetailRelationRecordsListItem + key={relationRecord.id} + isExpanded={expandedItem === relationRecord.id} + onClick={handleItemClick} + relationRecord={relationRecord} + /> + </Fragment> ))} </RecordDetailRecordsList> ); diff --git a/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItem.tsx b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItem.tsx index 8e35a01b0c66..a0653f933e5b 100644 --- a/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItem.tsx +++ b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItem.tsx @@ -15,7 +15,6 @@ import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSi import { formatFieldMetadataItemAsColumnDefinition } from '@/object-metadata/utils/formatFieldMetadataItemAsColumnDefinition'; import { RecordChip } from '@/object-record/components/RecordChip'; import { useDeleteOneRecord } from '@/object-record/hooks/useDeleteOneRecord'; -import { useLazyFindOneRecord } from '@/object-record/hooks/useLazyFindOneRecord'; import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; import { FieldContext, @@ -29,7 +28,6 @@ import { PropertyBox } from '@/object-record/record-inline-cell/property-box/com import { InlineCellHotkeyScope } from '@/object-record/record-inline-cell/types/InlineCellHotkeyScope'; import { RecordDetailRecordsListItem } from '@/object-record/record-show/record-detail-section/components/RecordDetailRecordsListItem'; import { RecordValueSetterEffect } from '@/object-record/record-store/components/RecordValueSetterEffect'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; import { isFieldCellSupported } from '@/object-record/utils/isFieldCellSupported'; import { LightIconButton } from '@/ui/input/button/components/LightIconButton'; @@ -99,12 +97,6 @@ export const RecordDetailRelationRecordsListItem = ({ const persistField = usePersistField(); - const { - called: hasFetchedRelationRecord, - findOneRecord: findOneRelationRecord, - } = useLazyFindOneRecord({ - objectNameSingular: relationObjectMetadataNameSingular, - }); const { updateOneRecord: updateOneRelationRecord } = useUpdateOneRecord({ objectNameSingular: relationObjectMetadataNameSingular, }); @@ -168,8 +160,6 @@ export const RecordDetailRelationRecordsListItem = ({ return [updateEntity, { loading: false }]; }; - const { setRecords } = useSetRecordInStore(); - const handleClick = () => onClick(relationRecord.id); const AnimatedIconChevronDown = useCallback<IconComponent>( @@ -194,16 +184,7 @@ export const RecordDetailRelationRecordsListItem = ({ record={relationRecord} objectNameSingular={relationObjectMetadataItem.nameSingular} /> - <StyledClickableZone - onClick={handleClick} - onMouseOver={() => - !hasFetchedRelationRecord && - findOneRelationRecord({ - objectRecordId: relationRecord.id, - onCompleted: (record) => setRecords([record]), - }) - } - > + <StyledClickableZone onClick={handleClick}> <LightIconButton className="displayOnHover" Icon={AnimatedIconChevronDown} diff --git a/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItemEffect.tsx b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItemEffect.tsx new file mode 100644 index 000000000000..0d7451bd5ce5 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-show/record-detail-section/components/RecordDetailRelationRecordsListItemEffect.tsx @@ -0,0 +1,35 @@ +import { useContext, useEffect } from 'react'; + +import { useFindOneRecord } from '@/object-record/hooks/useFindOneRecord'; +import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; +import { FieldRelationMetadata } from '@/object-record/record-field/types/FieldMetadata'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; +import { isDefined } from '~/utils/isDefined'; + +type RecordDetailRelationRecordsListItemEffectProps = { + relationRecordId: string; +}; + +export const RecordDetailRelationRecordsListItemEffect = ({ + relationRecordId, +}: RecordDetailRelationRecordsListItemEffectProps) => { + const { fieldDefinition } = useContext(FieldContext); + + const { relationObjectMetadataNameSingular } = + fieldDefinition.metadata as FieldRelationMetadata; + + const { record } = useFindOneRecord({ + objectNameSingular: relationObjectMetadataNameSingular, + objectRecordId: relationRecordId, + }); + + const { upsertRecords } = useUpsertRecordsInStore(); + + useEffect(() => { + if (isDefined(record)) { + upsertRecords([record]); + } + }, [record, upsertRecords]); + + return null; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-store/hooks/useSetRecordInStore.ts b/packages/twenty-front/src/modules/object-record/record-store/hooks/useUpsertRecordsInStore.ts similarity index 85% rename from packages/twenty-front/src/modules/object-record/record-store/hooks/useSetRecordInStore.ts rename to packages/twenty-front/src/modules/object-record/record-store/hooks/useUpsertRecordsInStore.ts index 63c4b9836c5b..0c7692749768 100644 --- a/packages/twenty-front/src/modules/object-record/record-store/hooks/useSetRecordInStore.ts +++ b/packages/twenty-front/src/modules/object-record/record-store/hooks/useUpsertRecordsInStore.ts @@ -3,8 +3,8 @@ import { useRecoilCallback } from 'recoil'; import { recordStoreFamilyState } from '@/object-record/record-store/states/recordStoreFamilyState'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; -export const useSetRecordInStore = () => { - const setRecords = useRecoilCallback( +export const useUpsertRecordsInStore = () => { + const upsertRecords = useRecoilCallback( ({ set, snapshot }) => (records: ObjectRecord[]) => { for (const record of records) { @@ -21,6 +21,6 @@ export const useSetRecordInStore = () => { ); return { - setRecords, + upsertRecords, }; }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/action-bar/components/RecordTableActionBar.tsx b/packages/twenty-front/src/modules/object-record/record-table/action-bar/components/RecordTableActionBar.tsx index f7241a5e2a48..f41025398c22 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/action-bar/components/RecordTableActionBar.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/action-bar/components/RecordTableActionBar.tsx @@ -2,19 +2,43 @@ import { useRecoilValue } from 'recoil'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; import { ActionBar } from '@/ui/navigation/action-bar/components/ActionBar'; +import { useViewStates } from '@/views/hooks/internal/useViewStates'; export const RecordTableActionBar = ({ recordTableId, }: { recordTableId: string; }) => { - const { selectedRowIdsSelector } = useRecordTableStates(recordTableId); + const { + selectedRowIdsSelector, + tableRowIdsState, + hasUserSelectedAllRowsState, + } = useRecordTableStates(recordTableId); + const { entityCountInCurrentViewState } = useViewStates(recordTableId); + const entityCountInCurrentView = useRecoilValue( + entityCountInCurrentViewState, + ); + const hasUserSelectedAllRows = useRecoilValue(hasUserSelectedAllRowsState); + const tableRowIds = useRecoilValue(tableRowIdsState); const selectedRowIds = useRecoilValue(selectedRowIdsSelector()); + const totalNumberOfSelectedRecords = + hasUserSelectedAllRows && entityCountInCurrentView + ? selectedRowIds.length === tableRowIds.length + ? entityCountInCurrentView + : entityCountInCurrentView - + (tableRowIds.length - selectedRowIds.length) // unselected row Ids + : selectedRowIds.length; + if (!selectedRowIds.length) { return null; } - return <ActionBar selectedIds={selectedRowIds} />; + return ( + <ActionBar + selectedIds={selectedRowIds} + totalNumberOfSelectedRecords={totalNumberOfSelectedRecords} + /> + ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/GripCell.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/GripCell.tsx deleted file mode 100644 index b9900026dbf6..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/components/GripCell.tsx +++ /dev/null @@ -1,29 +0,0 @@ -import styled from '@emotion/styled'; - -import { IconListViewGrip } from '@/ui/input/components/IconListViewGrip'; - -const StyledContainer = styled.div` - cursor: grab; - width: 16px; - height: 32px; - z-index: 200; - display: flex; - &:hover .icon { - opacity: 1; - } -`; - -const StyledIconWrapper = styled.div<{ isDragging: boolean }>` - opacity: ${({ isDragging }) => (isDragging ? 1 : 0)}; - transition: opacity 0.1s; -`; - -export const GripCell = ({ isDragging }: { isDragging: boolean }) => { - return ( - <StyledContainer> - <StyledIconWrapper className="icon" isDragging={isDragging}> - <IconListViewGrip /> - </StyledIconWrapper> - </StyledContainer> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTable.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTable.tsx index 2afb86e9875c..5e3b8c23af3e 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTable.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTable.tsx @@ -1,155 +1,27 @@ -import { css } from '@emotion/react'; import styled from '@emotion/styled'; -import { useRecoilValue } from 'recoil'; -import { MOBILE_VIEWPORT, RGBA } from 'twenty-ui'; +import { isNonEmptyString, isNull } from '@sniptt/guards'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; -import { RecordTableBody } from '@/object-record/record-table/components/RecordTableBody'; -import { RecordTableBodyEffect } from '@/object-record/record-table/components/RecordTableBodyEffect'; -import { RecordTableHeader } from '@/object-record/record-table/components/RecordTableHeader'; -import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; -import { useHandleContainerMouseEnter } from '@/object-record/record-table/hooks/internal/useHandleContainerMouseEnter'; +import { RecordTableContextProvider } from '@/object-record/record-table/components/RecordTableContextProvider'; +import { RecordTableEmptyState } from '@/object-record/record-table/components/RecordTableEmptyState'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; -import { useRecordTableMoveFocus } from '@/object-record/record-table/hooks/useRecordTableMoveFocus'; -import { useCloseRecordTableCellV2 } from '@/object-record/record-table/record-table-cell/hooks/useCloseRecordTableCellV2'; -import { useMoveSoftFocusToCellOnHoverV2 } from '@/object-record/record-table/record-table-cell/hooks/useMoveSoftFocusToCellOnHoverV2'; -import { - OpenTableCellArgs, - useOpenRecordTableCellV2, -} from '@/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellV2'; -import { useTriggerContextMenu } from '@/object-record/record-table/record-table-cell/hooks/useTriggerContextMenu'; -import { useUpsertRecordV2 } from '@/object-record/record-table/record-table-cell/hooks/useUpsertRecordV2'; +import { RecordTableBody } from '@/object-record/record-table/record-table-body/components/RecordTableBody'; +import { RecordTableBodyEffect } from '@/object-record/record-table/record-table-body/components/RecordTableBodyEffect'; +import { RecordTableHeader } from '@/object-record/record-table/record-table-header/components/RecordTableHeader'; import { RecordTableScope } from '@/object-record/record-table/scopes/RecordTableScope'; -import { MoveFocusDirection } from '@/object-record/record-table/types/MoveFocusDirection'; -import { TableCellPosition } from '@/object-record/record-table/types/TableCellPosition'; +import { useRecoilValue } from 'recoil'; -const StyledTable = styled.table<{ - freezeFirstColumns?: boolean; -}>` +const StyledTable = styled.table` border-radius: ${({ theme }) => theme.border.radius.sm}; border-spacing: 0; margin-right: ${({ theme }) => theme.table.horizontalCellMargin}; table-layout: fixed; width: calc(100% - ${({ theme }) => theme.table.horizontalCellMargin} * 2); - - th { - border-block: 1px solid ${({ theme }) => theme.border.color.light}; - color: ${({ theme }) => theme.font.color.tertiary}; - padding: 0; - text-align: left; - - :last-child { - border-right-color: transparent; - } - :first-of-type { - border-top-color: transparent; - border-bottom-color: transparent; - } - } - - td { - border-bottom: 1px solid ${({ theme }) => theme.border.color.light}; - color: ${({ theme }) => theme.font.color.primary}; - border-right: 1px solid ${({ theme }) => theme.border.color.light}; - - text-align: left; - - :last-child { - border-right-color: transparent; - } - :first-of-type { - border-top-color: transparent; - border-bottom-color: transparent; - } - } - - th { - background-color: ${({ theme }) => theme.background.primary}; - border-right: 1px solid ${({ theme }) => theme.border.color.light}; - } - - thead th { - position: sticky; - top: 0; - z-index: 9; - } - - thead th:nth-of-type(1), - thead th:nth-of-type(2), - thead th:nth-of-type(3) { - z-index: 12; - background-color: ${({ theme }) => theme.background.primary}; - } - - thead th:nth-of-type(1) { - width: 9px; - left: 0; - border-right-color: ${({ theme }) => theme.background.primary}; - } - - thead th:nth-of-type(2) { - left: 9px; - border-right-color: ${({ theme }) => theme.background.primary}; - } - - thead th:nth-of-type(3) { - left: 39px; - } - - tbody td:nth-of-type(1), - tbody td:nth-of-type(2), - tbody td:nth-of-type(3) { - position: sticky; - z-index: 1; - } - - tbody td:nth-of-type(1) { - left: 0; - z-index: 7; - } - - tbody td:nth-of-type(2) { - left: 9px; - z-index: 5; - } - - tbody td:nth-of-type(3) { - left: 39px; - z-index: 6; - } - - thead th:nth-of-type(3), - tbody td:nth-of-type(3) { - ${({ freezeFirstColumns }) => - freezeFirstColumns && - css` - @media (max-width: ${MOBILE_VIEWPORT}px) { - width: 35px; - max-width: 35px; - } - `} - - &::after { - content: ''; - height: calc(100% + 1px); - position: absolute; - width: 4px; - right: -4px; - top: 0; - - ${({ freezeFirstColumns, theme }) => - freezeFirstColumns && - css` - box-shadow: 4px 0px 4px -4px ${theme.name === 'dark' - ? RGBA(theme.grayScale.gray50, 0.8) - : RGBA(theme.grayScale.gray100, 0.25)} inset; - `} - } - } `; type RecordTableProps = { + viewBarId: string; recordTableId: string; objectNameSingular: string; onColumnsChange: (columns: any) => void; @@ -157,102 +29,66 @@ type RecordTableProps = { }; export const RecordTable = ({ + viewBarId, recordTableId, objectNameSingular, onColumnsChange, createRecord, }: RecordTableProps) => { - const { scopeId, visibleTableColumnsSelector } = - useRecordTableStates(recordTableId); + const { scopeId } = useRecordTableStates(recordTableId); - const { objectMetadataItem } = useObjectMetadataItem({ - objectNameSingular, - }); + const { + isRecordTableInitialLoadingState, + tableRowIdsState, + pendingRecordIdState, + } = useRecordTableStates(recordTableId); - const { upsertRecord } = useUpsertRecordV2({ - objectNameSingular, - }); - - const handleUpsertRecord = ({ - persistField, - entityId, - fieldName, - }: { - persistField: () => void; - entityId: string; - fieldName: string; - }) => { - upsertRecord(persistField, entityId, fieldName, recordTableId); - }; - - const { openTableCell } = useOpenRecordTableCellV2(recordTableId); - - const handleOpenTableCell = (args: OpenTableCellArgs) => { - openTableCell(args); - }; - - const { moveFocus } = useRecordTableMoveFocus(recordTableId); - - const handleMoveFocus = (direction: MoveFocusDirection) => { - moveFocus(direction); - }; - - const { closeTableCell } = useCloseRecordTableCellV2(recordTableId); - - const handleCloseTableCell = () => { - closeTableCell(); - }; - - const { moveSoftFocusToCell } = - useMoveSoftFocusToCellOnHoverV2(recordTableId); + const isRecordTableInitialLoading = useRecoilValue( + isRecordTableInitialLoadingState, + ); - const handleMoveSoftFocusToCell = (cellPosition: TableCellPosition) => { - moveSoftFocusToCell(cellPosition); - }; + const tableRowIds = useRecoilValue(tableRowIdsState); - const { triggerContextMenu } = useTriggerContextMenu({ - recordTableId, - }); + const pendingRecordId = useRecoilValue(pendingRecordIdState); - const handleContextMenu = (event: React.MouseEvent, recordId: string) => { - triggerContextMenu(event, recordId); - }; + const { objectMetadataItem: foundObjectMetadataItem } = useObjectMetadataItem( + { objectNameSingular }, + ); - const { handleContainerMouseEnter } = useHandleContainerMouseEnter({ - recordTableId, - }); + const objectLabel = foundObjectMetadataItem?.labelSingular; + const isRemote = foundObjectMetadataItem?.isRemote ?? false; - const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); + if (!isNonEmptyString(objectNameSingular)) { + return <></>; + } return ( <RecordTableScope recordTableScopeId={scopeId} onColumnsChange={onColumnsChange} > - {!!objectNameSingular && ( - <RecordTableContext.Provider - value={{ - objectMetadataItem, - onUpsertRecord: handleUpsertRecord, - onOpenTableCell: handleOpenTableCell, - onMoveFocus: handleMoveFocus, - onCloseTableCell: handleCloseTableCell, - onMoveSoftFocusToCell: handleMoveSoftFocusToCell, - onContextMenu: handleContextMenu, - onCellMouseEnter: handleContainerMouseEnter, - visibleTableColumns, - }} - > + <RecordTableContextProvider + objectNameSingular={objectNameSingular} + recordTableId={recordTableId} + viewBarId={viewBarId} + > + <RecordTableBodyEffect /> + {!isRecordTableInitialLoading && + tableRowIds.length === 0 && + isNull(pendingRecordId) ? ( + <RecordTableEmptyState + objectNameSingular={objectNameSingular} + objectLabel={objectLabel} + createRecord={createRecord} + isRemote={isRemote} + /> + ) : ( <StyledTable className="entity-table-cell"> <RecordTableHeader createRecord={createRecord} /> - <RecordTableBodyEffect objectNameSingular={objectNameSingular} /> - <RecordTableBody - objectNameSingular={objectNameSingular} - recordTableId={recordTableId} - /> + <RecordTableBody /> </StyledTable> - </RecordTableContext.Provider> - )} + )} + </RecordTableContextProvider> </RecordTableScope> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBody.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBody.tsx deleted file mode 100644 index 640bfffee55b..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBody.tsx +++ /dev/null @@ -1,43 +0,0 @@ -import { useRecoilValue } from 'recoil'; - -import { RecordTableBodyFetchMoreLoader } from '@/object-record/record-table/components/RecordTableBodyFetchMoreLoader'; -import { RecordTableRow } from '@/object-record/record-table/components/RecordTableRow'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; -import { DraggableTableBody } from '@/ui/layout/draggable-list/components/DraggableTableBody'; - -type RecordTableBodyProps = { - objectNameSingular: string; - recordTableId: string; -}; - -export const RecordTableBody = ({ - objectNameSingular, - recordTableId, -}: RecordTableBodyProps) => { - const { tableRowIdsState } = useRecordTableStates(); - - const tableRowIds = useRecoilValue(tableRowIdsState); - - return ( - <> - <DraggableTableBody - objectNameSingular={objectNameSingular} - recordTableId={recordTableId} - draggableItems={ - <> - {tableRowIds.map((recordId, rowIndex) => { - return ( - <RecordTableRow - key={recordId} - recordId={recordId} - rowIndex={rowIndex} - /> - ); - })} - </> - } - /> - <RecordTableBodyFetchMoreLoader objectNameSingular={objectNameSingular} /> - </> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBodyEffect.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBodyEffect.tsx deleted file mode 100644 index 4bea69f3d446..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBodyEffect.tsx +++ /dev/null @@ -1,58 +0,0 @@ -import { useEffect } from 'react'; -import { useRecoilState, useRecoilValue } from 'recoil'; - -import { useLoadRecordIndexTable } from '@/object-record/record-index/hooks/useLoadRecordIndexTable'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; -import { isFetchingMoreRecordsFamilyState } from '@/object-record/states/isFetchingMoreRecordsFamilyState'; -import { useScrollRestoration } from '~/hooks/useScrollRestoration'; - -type RecordTableBodyEffectProps = { - objectNameSingular: string; -}; - -export const RecordTableBodyEffect = ({ - objectNameSingular, -}: RecordTableBodyEffectProps) => { - const { - fetchMoreRecords: fetchMoreObjects, - records, - totalCount, - setRecordTableData, - queryStateIdentifier, - loading, - } = useLoadRecordIndexTable(objectNameSingular); - - const { tableLastRowVisibleState } = useRecordTableStates(); - - const [tableLastRowVisible, setTableLastRowVisible] = useRecoilState( - tableLastRowVisibleState, - ); - - const isFetchingMoreObjects = useRecoilValue( - isFetchingMoreRecordsFamilyState(queryStateIdentifier), - ); - - const rowHeight = 32; - const viewportHeight = records.length * rowHeight; - - useScrollRestoration(viewportHeight); - - useEffect(() => { - if (!loading) { - setRecordTableData(records, totalCount); - } - }, [records, totalCount, setRecordTableData, loading]); - - useEffect(() => { - if (tableLastRowVisible && !isFetchingMoreObjects) { - fetchMoreObjects(); - } - }, [ - fetchMoreObjects, - isFetchingMoreObjects, - setTableLastRowVisible, - tableLastRowVisible, - ]); - - return <></>; -}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableContextProvider.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableContextProvider.tsx new file mode 100644 index 000000000000..7fe7343c9298 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableContextProvider.tsx @@ -0,0 +1,112 @@ +import { ReactNode } from 'react'; +import { useRecoilValue } from 'recoil'; + +import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; +import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; +import { useHandleContainerMouseEnter } from '@/object-record/record-table/hooks/internal/useHandleContainerMouseEnter'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { useRecordTableMoveFocus } from '@/object-record/record-table/hooks/useRecordTableMoveFocus'; +import { useCloseRecordTableCellV2 } from '@/object-record/record-table/record-table-cell/hooks/useCloseRecordTableCellV2'; +import { useMoveSoftFocusToCellOnHoverV2 } from '@/object-record/record-table/record-table-cell/hooks/useMoveSoftFocusToCellOnHoverV2'; +import { + OpenTableCellArgs, + useOpenRecordTableCellV2, +} from '@/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellV2'; +import { useTriggerContextMenu } from '@/object-record/record-table/record-table-cell/hooks/useTriggerContextMenu'; +import { useUpsertRecord } from '@/object-record/record-table/record-table-cell/hooks/useUpsertRecord'; +import { MoveFocusDirection } from '@/object-record/record-table/types/MoveFocusDirection'; +import { TableCellPosition } from '@/object-record/record-table/types/TableCellPosition'; + +export const RecordTableContextProvider = ({ + viewBarId, + recordTableId, + objectNameSingular, + children, +}: { + viewBarId: string; + recordTableId: string; + objectNameSingular: string; + children: ReactNode; +}) => { + const { visibleTableColumnsSelector } = useRecordTableStates(recordTableId); + + const { objectMetadataItem } = useObjectMetadataItem({ + objectNameSingular, + }); + + const { upsertRecord } = useUpsertRecord({ + objectNameSingular, + }); + + const handleUpsertRecord = ({ + persistField, + entityId, + fieldName, + }: { + persistField: () => void; + entityId: string; + fieldName: string; + }) => { + upsertRecord(persistField, entityId, fieldName, recordTableId); + }; + + const { openTableCell } = useOpenRecordTableCellV2(recordTableId); + + const handleOpenTableCell = (args: OpenTableCellArgs) => { + openTableCell(args); + }; + + const { moveFocus } = useRecordTableMoveFocus(recordTableId); + + const handleMoveFocus = (direction: MoveFocusDirection) => { + moveFocus(direction); + }; + + const { closeTableCell } = useCloseRecordTableCellV2(recordTableId); + + const handleCloseTableCell = () => { + closeTableCell(); + }; + + const { moveSoftFocusToCell } = + useMoveSoftFocusToCellOnHoverV2(recordTableId); + + const handleMoveSoftFocusToCell = (cellPosition: TableCellPosition) => { + moveSoftFocusToCell(cellPosition); + }; + + const { triggerContextMenu } = useTriggerContextMenu({ + recordTableId, + }); + + const handleContextMenu = (event: React.MouseEvent, recordId: string) => { + triggerContextMenu(event, recordId); + }; + + const { handleContainerMouseEnter } = useHandleContainerMouseEnter({ + recordTableId, + }); + + const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); + + return ( + <RecordTableContext.Provider + value={{ + viewBarId, + objectMetadataItem, + onUpsertRecord: handleUpsertRecord, + onOpenTableCell: handleOpenTableCell, + onMoveFocus: handleMoveFocus, + onCloseTableCell: handleCloseTableCell, + onMoveSoftFocusToCell: handleMoveSoftFocusToCell, + onContextMenu: handleContextMenu, + onCellMouseEnter: handleContainerMouseEnter, + visibleTableColumns, + recordTableId, + objectNameSingular, + }} + > + {children} + </RecordTableContext.Provider> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeader.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeader.tsx deleted file mode 100644 index 9fe828dd9750..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeader.tsx +++ /dev/null @@ -1,112 +0,0 @@ -import { useTheme } from '@emotion/react'; -import styled from '@emotion/styled'; -import { useRecoilValue } from 'recoil'; -import { IconPlus } from 'twenty-ui'; - -import { RecordTableHeaderCell } from '@/object-record/record-table/components/RecordTableHeaderCell'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; -import { Dropdown } from '@/ui/layout/dropdown/components/Dropdown'; -import { useScrollWrapperScopedRef } from '@/ui/utilities/scroll/hooks/useScrollWrapperScopedRef'; - -import { RecordTableHeaderPlusButtonContent } from './RecordTableHeaderPlusButtonContent'; -import { SelectAllCheckbox } from './SelectAllCheckbox'; - -const StyledTableHead = styled.thead` - cursor: pointer; -`; - -const StyledPlusIconHeaderCell = styled.th<{ isTableWiderThanScreen: boolean }>` - ${({ theme }) => { - return ` - &:hover { - background: ${theme.background.transparent.light}; - }; - padding-left: ${theme.spacing(3)}; - `; - }}; - border-left: none !important; - min-width: 32px; - ${({ isTableWiderThanScreen, theme }) => - isTableWiderThanScreen && - ` - width: 32px; - border-right: none !important; - background-color: ${theme.background.primary}; - `}; - z-index: 1; -`; - -const StyledPlusIconContainer = styled.div` - align-items: center; - display: flex; - height: 32px; - justify-content: center; - width: 32px; -`; - -export const HIDDEN_TABLE_COLUMN_DROPDOWN_ID = - 'hidden-table-columns-dropdown-scope-id'; - -const HIDDEN_TABLE_COLUMN_DROPDOWN_HOTKEY_SCOPE_ID = - 'hidden-table-columns-dropdown-hotkey-scope-id'; - -export const RecordTableHeader = ({ - createRecord, -}: { - createRecord: () => void; -}) => { - const { visibleTableColumnsSelector } = useRecordTableStates(); - - const scrollWrapper = useScrollWrapperScopedRef(); - const isTableWiderThanScreen = - (scrollWrapper.current?.clientWidth ?? 0) < - (scrollWrapper.current?.scrollWidth ?? 0); - - const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); - const hiddenTableColumns = useRecoilValue(visibleTableColumnsSelector()); - - const theme = useTheme(); - - return ( - <StyledTableHead data-select-disable> - <tr> - <th></th> - <th - style={{ - width: 30, - minWidth: 30, - maxWidth: 30, - }} - > - <SelectAllCheckbox /> - </th> - {visibleTableColumns.map((column) => ( - <RecordTableHeaderCell - key={column.fieldMetadataId} - column={column} - createRecord={createRecord} - /> - ))} - <StyledPlusIconHeaderCell - isTableWiderThanScreen={isTableWiderThanScreen} - > - {hiddenTableColumns.length > 0 && ( - <Dropdown - dropdownId={HIDDEN_TABLE_COLUMN_DROPDOWN_ID} - clickableComponent={ - <StyledPlusIconContainer> - <IconPlus size={theme.icon.size.md} /> - </StyledPlusIconContainer> - } - dropdownComponents={<RecordTableHeaderPlusButtonContent />} - dropdownPlacement="bottom-start" - dropdownHotkeyScope={{ - scope: HIDDEN_TABLE_COLUMN_DROPDOWN_HOTKEY_SCOPE_ID, - }} - /> - )} - </StyledPlusIconHeaderCell> - </tr> - </StyledTableHead> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableInternalEffect.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableInternalEffect.tsx index 9a3f062e0f2c..8b94a325f36a 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableInternalEffect.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableInternalEffect.tsx @@ -5,7 +5,10 @@ import { useRecordTable } from '@/object-record/record-table/hooks/useRecordTabl import { TableHotkeyScope } from '@/object-record/record-table/types/TableHotkeyScope'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; import { useClickOutsideListener } from '@/ui/utilities/pointer-event/hooks/useClickOutsideListener'; -import { useListenClickOutsideByClassName } from '@/ui/utilities/pointer-event/hooks/useListenClickOutside'; +import { + ClickOutsideMode, + useListenClickOutsideByClassName, +} from '@/ui/utilities/pointer-event/hooks/useListenClickOutside'; type RecordTableInternalEffectProps = { recordTableId: string; @@ -30,6 +33,7 @@ export const RecordTableInternalEffect = ({ callback: () => { leaveTableFocus(); }, + mode: ClickOutsideMode.compareHTMLRef, }); useScopedHotkeys( diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableRow.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableRow.tsx deleted file mode 100644 index 5ec7e8ff4269..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableRow.tsx +++ /dev/null @@ -1,156 +0,0 @@ -import { useContext } from 'react'; -import { useInView } from 'react-intersection-observer'; -import { useTheme } from '@emotion/react'; -import styled from '@emotion/styled'; -import { Draggable } from '@hello-pangea/dnd'; -import { useRecoilValue } from 'recoil'; - -import { getBasePathToShowPage } from '@/object-metadata/utils/getBasePathToShowPage'; -import { RecordValueSetterEffect } from '@/object-record/record-store/components/RecordValueSetterEffect'; -import { RecordTableCellFieldContextWrapper } from '@/object-record/record-table/components/RecordTableCellFieldContextWrapper'; -import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; -import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; -import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; -import { ScrollWrapperContext } from '@/ui/utilities/scroll/components/ScrollWrapper'; - -import { CheckboxCell } from './CheckboxCell'; -import { GripCell } from './GripCell'; - -type RecordTableRowProps = { - recordId: string; - rowIndex: number; - isPendingRow?: boolean; -}; - -const StyledTd = styled.td` - position: relative; - user-select: none; -`; - -const StyledTr = styled.tr<{ isDragging: boolean }>` - border: 1px solid transparent; - transition: border-left-color 0.2s ease-in-out; - - td:nth-of-type(-n + 2) { - background-color: ${({ theme }) => theme.background.primary}; - border-right-color: ${({ theme }) => theme.background.primary}; - } - - ${({ isDragging }) => - isDragging && - ` - td:nth-of-type(1) { - background-color: transparent; - border-color: transparent; - } - - td:nth-of-type(2) { - background-color: transparent; - border-color: transparent; - } - - td:nth-of-type(3) { - background-color: transparent; - border-color: transparent; - } - - `} -`; - -export const RecordTableRow = ({ - recordId, - rowIndex, - isPendingRow, -}: RecordTableRowProps) => { - const { visibleTableColumnsSelector, isRowSelectedFamilyState } = - useRecordTableStates(); - const currentRowSelected = useRecoilValue(isRowSelectedFamilyState(recordId)); - const { objectMetadataItem } = useContext(RecordTableContext); - - const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); - - const scrollWrapperRef = useContext(ScrollWrapperContext); - - const { ref: elementRef, inView } = useInView({ - root: scrollWrapperRef.current?.querySelector( - '[data-overlayscrollbars-viewport="scrollbarHidden"]', - ), - rootMargin: '1000px', - }); - - const theme = useTheme(); - - return ( - <RecordTableRowContext.Provider - value={{ - recordId, - rowIndex, - pathToShowPage: - getBasePathToShowPage({ - objectNameSingular: objectMetadataItem.nameSingular, - }) + recordId, - objectNameSingular: objectMetadataItem.nameSingular, - isSelected: currentRowSelected, - isReadOnly: objectMetadataItem.isRemote ?? false, - isPendingRow, - }} - > - <RecordValueSetterEffect recordId={recordId} /> - - <Draggable key={recordId} draggableId={recordId} index={rowIndex}> - {(draggableProvided, draggableSnapshot) => ( - <StyledTr - ref={(node) => { - elementRef(node); - draggableProvided.innerRef(node); - }} - // eslint-disable-next-line react/jsx-props-no-spreading - {...draggableProvided.draggableProps} - style={{ - ...draggableProvided.draggableProps.style, - background: draggableSnapshot.isDragging - ? theme.background.transparent.light - : 'none', - borderColor: draggableSnapshot.isDragging - ? `${theme.border.color.medium}` - : 'transparent', - }} - isDragging={draggableSnapshot.isDragging} - data-testid={`row-id-${recordId}`} - data-selectable-id={recordId} - > - <StyledTd - // eslint-disable-next-line react/jsx-props-no-spreading - {...draggableProvided.dragHandleProps} - data-select-disable - > - <GripCell isDragging={draggableSnapshot.isDragging} /> - </StyledTd> - <StyledTd> - {!draggableSnapshot.isDragging && <CheckboxCell />} - </StyledTd> - {inView || draggableSnapshot.isDragging - ? visibleTableColumns.map((column, columnIndex) => ( - <RecordTableCellContext.Provider - value={{ - columnDefinition: column, - columnIndex, - }} - key={column.fieldMetadataId} - > - {draggableSnapshot.isDragging && columnIndex > 0 ? null : ( - <RecordTableCellFieldContextWrapper /> - )} - </RecordTableCellContext.Provider> - )) - : visibleTableColumns.map((column) => ( - <StyledTd key={column.fieldMetadataId}></StyledTd> - ))} - <StyledTd /> - </StyledTr> - )} - </Draggable> - </RecordTableRowContext.Provider> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableRows.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableRows.tsx new file mode 100644 index 000000000000..40db65bbb80c --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableRows.tsx @@ -0,0 +1,16 @@ +import { useRecoilValue } from 'recoil'; + +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableRow } from '@/object-record/record-table/record-table-row/components/RecordTableRow'; + +export const RecordTableRows = () => { + const { tableRowIdsState } = useRecordTableStates(); + + const tableRowIds = useRecoilValue(tableRowIdsState); + + return tableRowIds.map((recordId, rowIndex) => { + return ( + <RecordTableRow key={recordId} recordId={recordId} rowIndex={rowIndex} /> + ); + }); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableWithWrappers.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableWithWrappers.tsx index 63cced18d373..85e860436e93 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableWithWrappers.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableWithWrappers.tsx @@ -1,14 +1,11 @@ -import { useRef } from 'react'; import styled from '@emotion/styled'; -import { useRecoilCallback, useRecoilValue } from 'recoil'; +import { useRef } from 'react'; +import { useRecoilCallback } from 'recoil'; -import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; import { useDeleteOneRecord } from '@/object-record/hooks/useDeleteOneRecord'; import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; import { RecordTable } from '@/object-record/record-table/components/RecordTable'; -import { RecordTableEmptyState } from '@/object-record/record-table/components/RecordTableEmptyState'; import { EntityDeleteContext } from '@/object-record/record-table/contexts/EntityDeleteHookContext'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; import { DragSelect } from '@/ui/utilities/drag-select/components/DragSelect'; import { ScrollWrapper } from '@/ui/utilities/scroll/components/ScrollWrapper'; @@ -31,6 +28,10 @@ const StyledTableContainer = styled.div` position: relative; `; +const StyledTableInternalContainer = styled.div` + height: 100%; +`; + type RecordTableWithWrappersProps = { objectNameSingular: string; recordTableId: string; @@ -48,32 +49,24 @@ export const RecordTableWithWrappers = ({ }: RecordTableWithWrappersProps) => { const tableBodyRef = useRef<HTMLDivElement>(null); - const { isRecordTableInitialLoadingState, tableRowIdsState } = - useRecordTableStates(recordTableId); - - const isRecordTableInitialLoading = useRecoilValue( - isRecordTableInitialLoadingState, - ); - - const tableRowIds = useRecoilValue(tableRowIdsState); - const { resetTableRowSelection, setRowSelected } = useRecordTable({ recordTableId, }); - const { objectMetadataItem: foundObjectMetadataItem } = useObjectMetadataItem( - { - objectNameSingular, - }, - ); - const { saveViewFields } = useSaveCurrentViewFields(viewBarId); const { deleteOneRecord } = useDeleteOneRecord({ objectNameSingular }); - const objectLabel = foundObjectMetadataItem?.labelSingular; - - const isRemote = foundObjectMetadataItem?.isRemote ?? false; + const handleColumnsChange = useRecoilCallback( + () => (columns) => { + saveViewFields( + mapColumnDefinitionsToViewFields( + columns as ColumnDefinition<FieldMetadata>[], + ), + ); + }, + [saveViewFields], + ); return ( <EntityDeleteContext.Provider value={deleteOneRecord}> @@ -81,20 +74,12 @@ export const RecordTableWithWrappers = ({ <RecordUpdateContext.Provider value={updateRecordMutation}> <StyledTableWithHeader> <StyledTableContainer> - <div ref={tableBodyRef}> + <StyledTableInternalContainer ref={tableBodyRef}> <RecordTable + viewBarId={viewBarId} recordTableId={recordTableId} objectNameSingular={objectNameSingular} - onColumnsChange={useRecoilCallback( - () => (columns) => { - saveViewFields( - mapColumnDefinitionsToViewFields( - columns as ColumnDefinition<FieldMetadata>[], - ), - ); - }, - [saveViewFields], - )} + onColumnsChange={handleColumnsChange} createRecord={createRecord} /> <DragSelect @@ -102,21 +87,11 @@ export const RecordTableWithWrappers = ({ onDragSelectionStart={resetTableRowSelection} onDragSelectionChange={setRowSelected} /> - </div> + </StyledTableInternalContainer> <RecordTableInternalEffect recordTableId={recordTableId} tableBodyRef={tableBodyRef} /> - {!isRecordTableInitialLoading && - // we cannot rely on count states because this is not available for remote objects - tableRowIds.length === 0 && ( - <RecordTableEmptyState - objectNameSingular={objectNameSingular} - objectLabel={objectLabel} - createRecord={createRecord} - isRemote={isRemote} - /> - )} </StyledTableContainer> </StyledTableWithHeader> </RecordUpdateContext.Provider> diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/RecordTableCell.perf.stories.tsx b/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/RecordTableCell.perf.stories.tsx index 487465b609a2..add76f790a97 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/RecordTableCell.perf.stories.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/RecordTableCell.perf.stories.tsx @@ -1,5 +1,5 @@ -import { useEffect } from 'react'; import { Meta, StoryObj } from '@storybook/react'; +import { useEffect } from 'react'; import { useRecoilState, useSetRecoilState } from 'recoil'; import { ComponentDecorator } from 'twenty-ui'; @@ -12,7 +12,6 @@ import { useSetRecordValue, } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; import { recordStoreFamilyState } from '@/object-record/record-store/states/recordStoreFamilyState'; -import { RecordTableCellFieldContextWrapper } from '@/object-record/record-table/components/RecordTableCellFieldContextWrapper'; import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; @@ -21,6 +20,7 @@ import { ChipGeneratorsDecorator } from '~/testing/decorators/ChipGeneratorsDeco import { MemoryRouterDecorator } from '~/testing/decorators/MemoryRouterDecorator'; import { getProfilingStory } from '~/testing/profiling/utils/getProfilingStory'; +import { RecordTableCellFieldContextWrapper } from '@/object-record/record-table/record-table-cell/components/RecordTableCellFieldContextWrapper'; import { mockPerformance } from './mock'; const objectMetadataItems = getObjectMetadataItemsMock(); @@ -64,6 +64,7 @@ const meta: Meta = { <RecordFieldValueSelectorContextProvider> <RecordTableContext.Provider value={{ + viewBarId: mockPerformance.entityId, objectMetadataItem: mockPerformance.objectMetadataItem as any, onUpsertRecord: () => {}, onOpenTableCell: () => {}, @@ -73,6 +74,9 @@ const meta: Meta = { onContextMenu: () => {}, onCellMouseEnter: () => {}, visibleTableColumns: mockPerformance.visibleTableColumns as any, + objectNameSingular: + mockPerformance.objectMetadataItem.nameSingular, + recordTableId: 'recordTableId', }} > <RecordTableScope @@ -92,12 +96,19 @@ const meta: Meta = { }) + mockPerformance.entityId, isSelected: false, isReadOnly: false, + isDragging: false, + dragHandleProps: null, + inView: true, + isPendingRow: false, }} > <RecordTableCellContext.Provider value={{ columnDefinition: mockPerformance.fieldDefinition, columnIndex: 0, + cellPosition: { row: 0, column: 0 }, + hasSoftFocus: false, + isInEditMode: false, }} > <FieldContext.Provider diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/mock.ts b/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/mock.ts index ec60d500fbb4..925e8c70e7a7 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/mock.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/components/__stories__/perf/mock.ts @@ -800,7 +800,9 @@ export const mockPerformance = { }, employees: null, accountOwnerId: null, - address: '', + address: { + addressCity: 'San Francisco', + }, idealCustomerProfile: false, createdAt: '2024-05-01T13:16:29.046Z', id: '20202020-c21e-4ec2-873b-de4264d89025', @@ -844,7 +846,7 @@ export const mockPerformance = { }, employees: null, accountOwnerId: null, - address: '', + address: { addressCity: 'San Francisco' }, idealCustomerProfile: false, createdAt: '2024-05-01T13:16:29.046Z', id: '20202020-ed89-413a-b31a-962986e67bb4', diff --git a/packages/twenty-front/src/modules/object-record/record-table/constants/ColumnHeadDropdownId.ts b/packages/twenty-front/src/modules/object-record/record-table/constants/ColumnHeadDropdownId.ts deleted file mode 100644 index 8598c854650d..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/constants/ColumnHeadDropdownId.ts +++ /dev/null @@ -1 +0,0 @@ -export const COLUMN_HEAD_DROPDOWN_ID = 'table-head-options-dropdown-id'; diff --git a/packages/twenty-front/src/modules/object-record/record-table/constants/HiddenTableColumnDropdownId.ts b/packages/twenty-front/src/modules/object-record/record-table/constants/HiddenTableColumnDropdownId.ts new file mode 100644 index 000000000000..b2a90ce6fedf --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/constants/HiddenTableColumnDropdownId.ts @@ -0,0 +1,2 @@ +export const HIDDEN_TABLE_COLUMN_DROPDOWN_ID = + 'hidden-table-columns-dropdown-scope-id'; diff --git a/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableCellContext.ts b/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableCellContext.ts index 3a17eedacbae..78eb5a8a55cb 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableCellContext.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableCellContext.ts @@ -2,12 +2,15 @@ import { createContext } from 'react'; import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; +import { TableCellPosition } from '@/object-record/record-table/types/TableCellPosition'; -type RecordTableRowContextProps = { +export type RecordTableCellContextProps = { columnDefinition: ColumnDefinition<FieldMetadata>; columnIndex: number; + isInEditMode: boolean; + hasSoftFocus: boolean; + cellPosition: TableCellPosition; }; -export const RecordTableCellContext = createContext<RecordTableRowContextProps>( - {} as RecordTableRowContextProps, -); +export const RecordTableCellContext = + createContext<RecordTableCellContextProps>({} as RecordTableCellContextProps); diff --git a/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableContext.ts b/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableContext.ts index 4d25606854c0..960e544620a2 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableContext.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableContext.ts @@ -9,6 +9,7 @@ import { MoveFocusDirection } from '@/object-record/record-table/types/MoveFocus import { TableCellPosition } from '@/object-record/record-table/types/TableCellPosition'; export type RecordTableContextProps = { + viewBarId: string; objectMetadataItem: ObjectMetadataItem; onUpsertRecord: ({ persistField, @@ -26,6 +27,8 @@ export type RecordTableContextProps = { onContextMenu: (event: React.MouseEvent, recordId: string) => void; onCellMouseEnter: (args: HandleContainerMouseEnterArgs) => void; visibleTableColumns: ColumnDefinition<FieldMetadata>[]; + recordTableId: string; + objectNameSingular: string; }; export const RecordTableContext = createContext<RecordTableContextProps>( diff --git a/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableRowContext.ts b/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableRowContext.ts index f04afe492820..d0ed1aea296b 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableRowContext.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/contexts/RecordTableRowContext.ts @@ -1,4 +1,5 @@ import { createContext } from 'react'; +import { DraggableProvidedDragHandleProps } from '@hello-pangea/dnd'; export type RecordTableRowContextProps = { pathToShowPage: string; @@ -8,6 +9,9 @@ export type RecordTableRowContextProps = { isSelected: boolean; isReadOnly: boolean; isPendingRow?: boolean; + isDragging: boolean; + dragHandleProps: DraggableProvidedDragHandleProps | null; + inView?: boolean; }; export const RecordTableRowContext = createContext<RecordTableRowContextProps>( diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useCloseCurrentTableCellInEditMode.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useCloseCurrentTableCellInEditMode.ts index b894046ce057..a9976aa05fac 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useCloseCurrentTableCellInEditMode.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useCloseCurrentTableCellInEditMode.ts @@ -21,13 +21,6 @@ export const useCloseCurrentTableCellInEditMode = (recordTableId?: string) => { isTableCellInEditModeFamilyState(currentTableCellInEditModePosition), false, ); - - document.dispatchEvent( - new CustomEvent( - `edit-mode-change-${currentTableCellInEditModePosition.row}:${currentTableCellInEditModePosition.column}`, - { detail: false }, - ), - ); }; }, [currentTableCellInEditModePositionState, isTableCellInEditModeFamilyState], diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useLeaveTableFocus.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useLeaveTableFocus.ts index 0ee3aa62d6a8..0a52eb083239 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useLeaveTableFocus.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useLeaveTableFocus.ts @@ -8,12 +8,17 @@ import { TableHotkeyScope } from '../../types/TableHotkeyScope'; import { useCloseCurrentTableCellInEditMode } from './useCloseCurrentTableCellInEditMode'; import { useDisableSoftFocus } from './useDisableSoftFocus'; +import { useSetHasUserSelectedAllRows } from './useSetAllRowSelectedState'; export const useLeaveTableFocus = (recordTableId?: string) => { const disableSoftFocus = useDisableSoftFocus(recordTableId); const closeCurrentCellInEditMode = useCloseCurrentTableCellInEditMode(recordTableId); + const setHasUserSelectedAllRows = useSetHasUserSelectedAllRows(recordTableId); + + const selectAllRows = useSetHasUserSelectedAllRows(recordTableId); + const { isSoftFocusActiveState } = useRecordTableStates(recordTableId); return useRecoilCallback( @@ -38,7 +43,15 @@ export const useLeaveTableFocus = (recordTableId?: string) => { closeCurrentCellInEditMode(); disableSoftFocus(); + setHasUserSelectedAllRows(false); + selectAllRows(false); }, - [closeCurrentCellInEditMode, disableSoftFocus, isSoftFocusActiveState], + [ + closeCurrentCellInEditMode, + disableSoftFocus, + isSoftFocusActiveState, + selectAllRows, + setHasUserSelectedAllRows, + ], ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useMoveEditModeToCellPosition.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useMoveEditModeToCellPosition.ts index c51be2fa6bdb..02e04a259d31 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useMoveEditModeToCellPosition.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useMoveEditModeToCellPosition.ts @@ -24,23 +24,9 @@ export const useMoveEditModeToTableCellPosition = (recordTableId?: string) => { false, ); - document.dispatchEvent( - new CustomEvent( - `edit-mode-change-${currentTableCellInEditModePosition.row}:${currentTableCellInEditModePosition.column}`, - { detail: false }, - ), - ); - set(currentTableCellInEditModePositionState, newPosition); set(isTableCellInEditModeFamilyState(newPosition), true); - - document.dispatchEvent( - new CustomEvent( - `edit-mode-change-${newPosition.row}:${newPosition.column}`, - { detail: true }, - ), - ); }; }, [currentTableCellInEditModePositionState, isTableCellInEditModeFamilyState], diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useRecordTableStates.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useRecordTableStates.ts index 1e178da5bd7a..1fbc4a8bd810 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useRecordTableStates.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useRecordTableStates.ts @@ -109,7 +109,7 @@ export const useRecordTableStates = (recordTableId?: string) => { isRowSelectedComponentFamilyState, scopeId, ), - hasUserSelectedAllRowState: extractComponentState( + hasUserSelectedAllRowsState: extractComponentState( hasUserSelectedAllRowsComponentState, scopeId, ), diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useResetTableRowSelection.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useResetTableRowSelection.ts index 7e86f581909c..1b6263739111 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useResetTableRowSelection.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useResetTableRowSelection.ts @@ -4,8 +4,11 @@ import { useRecordTableStates } from '@/object-record/record-table/hooks/interna import { getSnapshotValue } from '@/ui/utilities/recoil-scope/utils/getSnapshotValue'; export const useResetTableRowSelection = (recordTableId?: string) => { - const { tableRowIdsState, isRowSelectedFamilyState } = - useRecordTableStates(recordTableId); + const { + tableRowIdsState, + isRowSelectedFamilyState, + hasUserSelectedAllRowsState, + } = useRecordTableStates(recordTableId); return useRecoilCallback( ({ snapshot, set }) => @@ -15,7 +18,9 @@ export const useResetTableRowSelection = (recordTableId?: string) => { for (const rowId of tableRowIds) { set(isRowSelectedFamilyState(rowId), false); } + + set(hasUserSelectedAllRowsState, false); }, - [tableRowIdsState, isRowSelectedFamilyState], + [tableRowIdsState, isRowSelectedFamilyState, hasUserSelectedAllRowsState], ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetAllRowSelectedState.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetAllRowSelectedState.ts index 4544c4b71f95..5df32d9006d2 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetAllRowSelectedState.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetAllRowSelectedState.ts @@ -3,14 +3,13 @@ import { useRecoilCallback } from 'recoil'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; export const useSetHasUserSelectedAllRows = (recordTableId?: string) => { - const { hasUserSelectedAllRowState: hasUserSelectedAllRowFamilyState } = - useRecordTableStates(recordTableId); + const { hasUserSelectedAllRowsState } = useRecordTableStates(recordTableId); return useRecoilCallback( ({ set }) => (selected: boolean) => { - set(hasUserSelectedAllRowFamilyState, selected); + set(hasUserSelectedAllRowsState, selected); }, - [hasUserSelectedAllRowFamilyState], + [hasUserSelectedAllRowsState], ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetRecordTableData.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetRecordTableData.ts index 64ca715b896a..eeaf15147c4a 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetRecordTableData.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetRecordTableData.ts @@ -19,7 +19,7 @@ export const useSetRecordTableData = ({ tableRowIdsState, numberOfTableRowsState, isRowSelectedFamilyState, - hasUserSelectedAllRowState, + hasUserSelectedAllRowsState, } = useRecordTableStates(recordTableId); return useRecoilCallback( @@ -39,7 +39,7 @@ export const useSetRecordTableData = ({ const hasUserSelectedAllRows = getSnapshotValue( snapshot, - hasUserSelectedAllRowState, + hasUserSelectedAllRowsState, ); const entityIds = newEntityArray.map((entity) => entity.id); @@ -62,7 +62,7 @@ export const useSetRecordTableData = ({ tableRowIdsState, onEntityCountChange, isRowSelectedFamilyState, - hasUserSelectedAllRowState, + hasUserSelectedAllRowsState, ], ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetSoftFocusPosition.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetSoftFocusPosition.ts index d645c7bb9036..edf8f7a904e0 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetSoftFocusPosition.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/internal/useSetSoftFocusPosition.ts @@ -24,23 +24,9 @@ export const useSetSoftFocusPosition = (recordTableId?: string) => { set(isSoftFocusOnTableCellFamilyState(currentPosition), false); - document.dispatchEvent( - new CustomEvent( - `soft-focus-move-${currentPosition.row}:${currentPosition.column}`, - { detail: false }, - ), - ); - set(softFocusPositionState, newPosition); set(isSoftFocusOnTableCellFamilyState(newPosition), true); - - document.dispatchEvent( - new CustomEvent( - `soft-focus-move-${newPosition.row}:${newPosition.column}`, - { detail: true }, - ), - ); }; }, [ diff --git a/packages/twenty-front/src/modules/object-record/record-table/hooks/useRecordTable.ts b/packages/twenty-front/src/modules/object-record/record-table/hooks/useRecordTable.ts index 785baebe165c..6cad63df5448 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/hooks/useRecordTable.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/hooks/useRecordTable.ts @@ -45,6 +45,7 @@ export const useRecordTable = (props?: useRecordTableProps) => { onToggleColumnFilterState, onToggleColumnSortState, pendingRecordIdState, + hasUserSelectedAllRowsState, } = useRecordTableStates(recordTableId); const setAvailableTableColumns = useRecoilCallback( @@ -226,5 +227,6 @@ export const useRecordTable = (props?: useRecordTableProps) => { setOnToggleColumnFilter, setOnToggleColumnSort, setPendingRecordId, + hasUserSelectedAllRowsState, }; }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBody.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBody.tsx new file mode 100644 index 000000000000..2a6e46c21987 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBody.tsx @@ -0,0 +1,38 @@ +import { useRecoilValue } from 'recoil'; + +import { RecordTableRows } from '@/object-record/record-table/components/RecordTableRows'; +import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableBodyDragDropContext } from '@/object-record/record-table/record-table-body/components/RecordTableBodyDragDropContext'; +import { RecordTableBodyDroppable } from '@/object-record/record-table/record-table-body/components/RecordTableBodyDroppable'; +import { RecordTableBodyFetchMoreLoader } from '@/object-record/record-table/record-table-body/components/RecordTableBodyFetchMoreLoader'; +import { RecordTableBodyLoading } from '@/object-record/record-table/record-table-body/components/RecordTableBodyLoading'; +import { RecordTablePendingRow } from '@/object-record/record-table/record-table-row/components/RecordTablePendingRow'; +import { useContext } from 'react'; + +export const RecordTableBody = () => { + const { tableRowIdsState, isRecordTableInitialLoadingState } = + useRecordTableStates(); + + const { objectNameSingular } = useContext(RecordTableContext); + + const tableRowIds = useRecoilValue(tableRowIdsState); + + const isRecordTableInitialLoading = useRecoilValue( + isRecordTableInitialLoadingState, + ); + + if (isRecordTableInitialLoading && tableRowIds.length === 0) { + return <RecordTableBodyLoading />; + } + + return ( + <RecordTableBodyDragDropContext> + <RecordTableBodyDroppable> + <RecordTablePendingRow /> + <RecordTableRows /> + </RecordTableBodyDroppable> + <RecordTableBodyFetchMoreLoader objectNameSingular={objectNameSingular} /> + </RecordTableBodyDragDropContext> + ); +}; diff --git a/packages/twenty-front/src/modules/ui/layout/draggable-list/components/DraggableTableBody.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyDragDropContext.tsx similarity index 58% rename from packages/twenty-front/src/modules/ui/layout/draggable-list/components/DraggableTableBody.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyDragDropContext.tsx index c1513d828b74..ae352714590e 100644 --- a/packages/twenty-front/src/modules/ui/layout/draggable-list/components/DraggableTableBody.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyDragDropContext.tsx @@ -1,42 +1,30 @@ -import { useState } from 'react'; -import styled from '@emotion/styled'; -import { DragDropContext, Droppable, DropResult } from '@hello-pangea/dnd'; +import { ReactNode, useContext } from 'react'; +import { DragDropContext, DropResult } from '@hello-pangea/dnd'; import { useRecoilValue, useSetRecoilState } from 'recoil'; -import { v4 } from 'uuid'; import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; -import { RecordTablePendingRow } from '@/object-record/record-table/components/RecordTablePendingRow'; +import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; import { useComputeNewRowPosition } from '@/object-record/record-table/hooks/useComputeNewRowPosition'; import { isRemoveSortingModalOpenState } from '@/object-record/record-table/states/isRemoveSortingModalOpenState'; import { useGetCurrentView } from '@/views/hooks/useGetCurrentView'; import { isDefined } from '~/utils/isDefined'; -type DraggableTableBodyProps = { - draggableItems: React.ReactNode; - objectNameSingular: string; - recordTableId: string; -}; - -const StyledTbody = styled.tbody` - overflow: hidden; -`; +export const RecordTableBodyDragDropContext = ({ + children, +}: { + children: ReactNode; +}) => { + const { objectNameSingular, recordTableId } = useContext(RecordTableContext); -export const DraggableTableBody = ({ - objectNameSingular, - draggableItems, - recordTableId, -}: DraggableTableBodyProps) => { - const [v4Persistable] = useState(v4()); + const { updateOneRecord: updateOneRow } = useUpdateOneRecord({ + objectNameSingular, + }); const { tableRowIdsState } = useRecordTableStates(); const tableRowIds = useRecoilValue(tableRowIdsState); - const { updateOneRecord: updateOneRow } = useUpdateOneRecord({ - objectNameSingular, - }); - const { currentViewWithCombinedFiltersAndSorts } = useGetCurrentView(recordTableId); @@ -45,6 +33,7 @@ export const DraggableTableBody = ({ const setIsRemoveSortingModalOpenState = useSetRecoilState( isRemoveSortingModalOpenState, ); + const computeNewRowPosition = useComputeNewRowPosition(); const handleDragEnd = (result: DropResult) => { @@ -68,20 +57,6 @@ export const DraggableTableBody = ({ }; return ( - <DragDropContext onDragEnd={handleDragEnd}> - <Droppable droppableId={v4Persistable}> - {(provided) => ( - <StyledTbody - ref={provided.innerRef} - // eslint-disable-next-line react/jsx-props-no-spreading - {...provided.droppableProps} - > - <RecordTablePendingRow /> - {draggableItems} - {provided.placeholder} - </StyledTbody> - )} - </Droppable> - </DragDropContext> + <DragDropContext onDragEnd={handleDragEnd}>{children}</DragDropContext> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyDroppable.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyDroppable.tsx new file mode 100644 index 000000000000..bd062d3abb4f --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyDroppable.tsx @@ -0,0 +1,67 @@ +import { Theme } from '@emotion/react'; +import { Droppable } from '@hello-pangea/dnd'; +import { styled } from '@linaria/react'; +import { ReactNode, useContext, useState } from 'react'; +import { ThemeContext } from 'twenty-ui'; +import { v4 } from 'uuid'; + +const StyledTbody = styled.tbody<{ + theme: Theme; +}>` + overflow: hidden; + + &.first-columns-sticky { + td:nth-of-type(1) { + position: sticky; + left: 0; + z-index: 5; + } + td:nth-of-type(2) { + position: sticky; + left: 9px; + z-index: 5; + } + td:nth-of-type(3) { + position: sticky; + left: 39px; + z-index: 5; + &::after { + content: ''; + position: absolute; + top: -1px; + height: calc(100% + 2px); + width: 4px; + right: 0px; + box-shadow: ${({ theme }) => theme.boxShadow.light}; + clip-path: inset(0px -4px 0px 0px); + } + } + } +`; + +export const RecordTableBodyDroppable = ({ + children, +}: { + children: ReactNode; +}) => { + const [v4Persistable] = useState(v4()); + + const { theme } = useContext(ThemeContext); + + return ( + <Droppable droppableId={v4Persistable}> + {(provided) => ( + <StyledTbody + id="record-table-body" + theme={theme} + ref={provided.innerRef} + // eslint-disable-next-line react/jsx-props-no-spreading + {...provided.droppableProps} + > + {children} + {provided.placeholder} + </StyledTbody> + )} + </Droppable> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyEffect.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyEffect.tsx new file mode 100644 index 000000000000..33816ad4d3ec --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyEffect.tsx @@ -0,0 +1,148 @@ +import { useContext, useEffect, useState } from 'react'; +import { useRecoilState, useRecoilValue } from 'recoil'; +import { useDebouncedCallback } from 'use-debounce'; + +import { lastShowPageRecordIdState } from '@/object-record/record-field/states/lastShowPageRecordId'; +import { useLoadRecordIndexTable } from '@/object-record/record-index/hooks/useLoadRecordIndexTable'; +import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { isRecordTableScrolledLeftComponentState } from '@/object-record/record-table/states/isRecordTableScrolledLeftComponentState'; +import { isRecordTableScrolledTopComponentState } from '@/object-record/record-table/states/isRecordTableScrolledTopComponentState'; +import { isFetchingMoreRecordsFamilyState } from '@/object-record/states/isFetchingMoreRecordsFamilyState'; +import { scrollLeftState } from '@/ui/utilities/scroll/states/scrollLeftState'; +import { scrollTopState } from '@/ui/utilities/scroll/states/scrollTopState'; +import { useSetRecoilComponentState } from '@/ui/utilities/state/component-state/hooks/useSetRecoilComponentState'; +import { isNonEmptyString } from '@sniptt/guards'; +import { useScrollRestoration } from '~/hooks/useScrollRestoration'; +import { useScrollToPosition } from '~/hooks/useScrollToPosition'; + +export const ROW_HEIGHT = 32; + +export const RecordTableBodyEffect = () => { + const { objectNameSingular } = useContext(RecordTableContext); + + const [hasInitializedScroll, setHasInitiazedScroll] = useState(false); + + const { + fetchMoreRecords: fetchMoreObjects, + records, + totalCount, + setRecordTableData, + loading, + queryStateIdentifier, + } = useLoadRecordIndexTable(objectNameSingular); + + const isFetchingMoreObjects = useRecoilValue( + isFetchingMoreRecordsFamilyState(queryStateIdentifier), + ); + + const { tableLastRowVisibleState } = useRecordTableStates(); + + const tableLastRowVisible = useRecoilValue(tableLastRowVisibleState); + + const scrollTop = useRecoilValue(scrollTopState); + const setIsRecordTableScrolledTop = useSetRecoilComponentState( + isRecordTableScrolledTopComponentState, + ); + + useEffect(() => { + setIsRecordTableScrolledTop(scrollTop === 0); + if (scrollTop > 0) { + document + .getElementById('record-table-header') + ?.classList.add('header-sticky'); + } else { + document + .getElementById('record-table-header') + ?.classList.remove('header-sticky'); + } + }, [scrollTop, setIsRecordTableScrolledTop]); + + const scrollLeft = useRecoilValue(scrollLeftState); + + const setIsRecordTableScrolledLeft = useSetRecoilComponentState( + isRecordTableScrolledLeftComponentState, + ); + + useEffect(() => { + setIsRecordTableScrolledLeft(scrollLeft === 0); + if (scrollLeft > 0) { + document + .getElementById('record-table-body') + ?.classList.add('first-columns-sticky'); + document + .getElementById('record-table-header') + ?.classList.add('first-columns-sticky'); + } else { + document + .getElementById('record-table-body') + ?.classList.remove('first-columns-sticky'); + document + .getElementById('record-table-header') + ?.classList.remove('first-columns-sticky'); + } + }, [scrollLeft, setIsRecordTableScrolledLeft]); + + const rowHeight = ROW_HEIGHT; + const viewportHeight = records.length * rowHeight; + + const [lastShowPageRecordId, setLastShowPageRecordId] = useRecoilState( + lastShowPageRecordIdState, + ); + + const { scrollToPosition } = useScrollToPosition(); + + useEffect(() => { + if (isNonEmptyString(lastShowPageRecordId) && !hasInitializedScroll) { + const isRecordAlreadyFetched = records.some( + (record) => record.id === lastShowPageRecordId, + ); + + if (isRecordAlreadyFetched) { + const recordPosition = records.findIndex( + (record) => record.id === lastShowPageRecordId, + ); + + const positionInPx = recordPosition * ROW_HEIGHT; + + scrollToPosition(positionInPx); + + setHasInitiazedScroll(true); + } + } + }, [ + loading, + isFetchingMoreObjects, + lastShowPageRecordId, + fetchMoreObjects, + records, + scrollToPosition, + hasInitializedScroll, + setLastShowPageRecordId, + ]); + + useScrollRestoration(viewportHeight); + + useEffect(() => { + if (!loading) { + setRecordTableData(records, totalCount); + } + }, [records, totalCount, setRecordTableData, loading]); + + const fetchMoreDebouncedIfRequested = useDebouncedCallback(async () => { + // We are debouncing here to give the user some room to scroll if they want to within this throttle window + await fetchMoreObjects(); + }, 100); + + useEffect(() => { + if (!isFetchingMoreObjects && tableLastRowVisible) { + fetchMoreDebouncedIfRequested(); + } + }, [ + fetchMoreDebouncedIfRequested, + isFetchingMoreObjects, + tableLastRowVisible, + ]); + + return <></>; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBodyFetchMoreLoader.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyFetchMoreLoader.tsx similarity index 100% rename from packages/twenty-front/src/modules/object-record/record-table/components/RecordTableBodyFetchMoreLoader.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyFetchMoreLoader.tsx diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyLoading.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyLoading.tsx new file mode 100644 index 000000000000..8a80403ded4f --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-body/components/RecordTableBodyLoading.tsx @@ -0,0 +1,31 @@ +import { useRecoilValue } from 'recoil'; + +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableCellCheckbox } from '@/object-record/record-table/record-table-cell/components/RecordTableCellCheckbox'; +import { RecordTableCellGrip } from '@/object-record/record-table/record-table-cell/components/RecordTableCellGrip'; +import { RecordTableCellLoading } from '@/object-record/record-table/record-table-cell/components/RecordTableCellLoading'; +import { RecordTableTr } from '@/object-record/record-table/record-table-row/components/RecordTableTr'; + +export const RecordTableBodyLoading = () => { + const { visibleTableColumnsSelector } = useRecordTableStates(); + const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); + + return ( + <tbody> + {Array.from({ length: 8 }).map((_, rowIndex) => ( + <RecordTableTr + isDragging={false} + data-testid={`row-id-${rowIndex}`} + data-selectable-id={`row-id-${rowIndex}`} + key={rowIndex} + > + <RecordTableCellGrip /> + <RecordTableCellCheckbox /> + {visibleTableColumns.map((column) => ( + <RecordTableCellLoading key={column.fieldMetadataId} /> + ))} + </RecordTableTr> + ))} + </tbody> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCell.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCell.tsx index a38d61a39dc5..4d6fcdd74fab 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCell.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCell.tsx @@ -1,109 +1,13 @@ -import { useContext } from 'react'; - import { FieldDisplay } from '@/object-record/record-field/components/FieldDisplay'; -import { FieldInput } from '@/object-record/record-field/components/FieldInput'; -import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; import { FieldFocusContextProvider } from '@/object-record/record-field/contexts/FieldFocusContextProvider'; -import { FieldInputEvent } from '@/object-record/record-field/types/FieldInputEvent'; -import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; -import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; import { RecordTableCellContainer } from '@/object-record/record-table/record-table-cell/components/RecordTableCellContainer'; -import { HotkeyScope } from '@/ui/utilities/hotkey/types/HotkeyScope'; - -export const RecordTableCell = ({ - customHotkeyScope, -}: { - customHotkeyScope: HotkeyScope; -}) => { - const { onUpsertRecord, onMoveFocus, onCloseTableCell } = - useContext(RecordTableContext); - const { entityId, fieldDefinition } = useContext(FieldContext); - const { isReadOnly } = useContext(RecordTableRowContext); - - const handleEnter: FieldInputEvent = (persistField) => { - onUpsertRecord({ - persistField, - entityId, - fieldName: fieldDefinition.metadata.fieldName, - }); - - onCloseTableCell(); - onMoveFocus('down'); - }; - - const handleSubmit: FieldInputEvent = (persistField) => { - onUpsertRecord({ - persistField, - entityId, - fieldName: fieldDefinition.metadata.fieldName, - }); - - onCloseTableCell(); - }; - - const handleCancel = () => { - onCloseTableCell(); - }; - - const handleClickOutside: FieldInputEvent = (persistField) => { - onUpsertRecord({ - persistField, - entityId, - fieldName: fieldDefinition.metadata.fieldName, - }); - - onCloseTableCell(); - }; - - const handleEscape: FieldInputEvent = (persistField) => { - onUpsertRecord({ - persistField, - entityId, - fieldName: fieldDefinition.metadata.fieldName, - }); - - onCloseTableCell(); - }; - - const handleTab: FieldInputEvent = (persistField) => { - onUpsertRecord({ - persistField, - entityId, - fieldName: fieldDefinition.metadata.fieldName, - }); - - onCloseTableCell(); - onMoveFocus('right'); - }; - - const handleShiftTab: FieldInputEvent = (persistField) => { - onUpsertRecord({ - persistField, - entityId, - fieldName: fieldDefinition.metadata.fieldName, - }); - - onCloseTableCell(); - onMoveFocus('left'); - }; +import { RecordTableCellFieldInput } from '@/object-record/record-table/record-table-cell/components/RecordTableCellFieldInput'; +export const RecordTableCell = () => { return ( <FieldFocusContextProvider> <RecordTableCellContainer - editHotkeyScope={customHotkeyScope} - editModeContent={ - <FieldInput - recordFieldInputdId={`${entityId}-${fieldDefinition?.metadata?.fieldName}`} - onCancel={handleCancel} - onClickOutside={handleClickOutside} - onEnter={handleEnter} - onEscape={handleEscape} - onShiftTab={handleShiftTab} - onSubmit={handleSubmit} - onTab={handleTab} - isReadOnly={isReadOnly} - /> - } + editModeContent={<RecordTableCellFieldInput />} nonEditModeContent={<FieldDisplay />} /> </FieldFocusContextProvider> diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellBaseContainer.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellBaseContainer.tsx new file mode 100644 index 000000000000..3a2a8c5c9bf8 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellBaseContainer.tsx @@ -0,0 +1,101 @@ +import { ReactNode, useContext } from 'react'; +import { styled } from '@linaria/react'; +import { BORDER_COMMON, ThemeContext } from 'twenty-ui'; + +import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; +import { useFieldFocus } from '@/object-record/record-field/hooks/useFieldFocus'; +import { CellHotkeyScopeContext } from '@/object-record/record-table/contexts/CellHotkeyScopeContext'; +import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; +import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { + DEFAULT_CELL_SCOPE, + useOpenRecordTableCellFromCell, +} from '@/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellFromCell'; + +const StyledBaseContainer = styled.div<{ + hasSoftFocus: boolean; + fontColorExtraLight: string; + backgroundColorTransparentSecondary: string; +}>` + align-items: center; + box-sizing: border-box; + cursor: pointer; + display: flex; + height: 32px; + position: relative; + user-select: none; + + background: ${({ hasSoftFocus, backgroundColorTransparentSecondary }) => + hasSoftFocus ? backgroundColorTransparentSecondary : 'none'}; + + border-radius: ${({ hasSoftFocus }) => + hasSoftFocus ? BORDER_COMMON.radius.sm : 'none'}; + + outline: ${({ hasSoftFocus, fontColorExtraLight }) => + hasSoftFocus ? `1px solid ${fontColorExtraLight}` : 'none'}; +`; + +export const RecordTableCellBaseContainer = ({ + children, +}: { + children: ReactNode; +}) => { + const { setIsFocused } = useFieldFocus(); + const { openTableCell } = useOpenRecordTableCellFromCell(); + const { theme } = useContext(ThemeContext); + const { recordId } = useContext(RecordTableRowContext); + + const { hasSoftFocus, cellPosition } = useContext(RecordTableCellContext); + + const { onMoveSoftFocusToCell, onCellMouseEnter } = + useContext(RecordTableContext); + + const handleContainerMouseMove = () => { + setIsFocused(true); + if (!hasSoftFocus) { + onCellMouseEnter({ + cellPosition, + }); + } + }; + + const handleContainerMouseLeave = () => { + setIsFocused(false); + }; + + const handleContainerClick = () => { + if (!hasSoftFocus) { + onMoveSoftFocusToCell(cellPosition); + openTableCell(); + } + }; + + const { onContextMenu } = useContext(RecordTableContext); + + const handleContextMenu = (event: React.MouseEvent) => { + onContextMenu(event, recordId); + }; + + const { hotkeyScope } = useContext(FieldContext); + + const editHotkeyScope = { scope: hotkeyScope } ?? DEFAULT_CELL_SCOPE; + + return ( + <CellHotkeyScopeContext.Provider value={editHotkeyScope}> + <StyledBaseContainer + onMouseLeave={handleContainerMouseLeave} + onMouseMove={handleContainerMouseMove} + onClick={handleContainerClick} + onContextMenu={handleContextMenu} + backgroundColorTransparentSecondary={ + theme.background.transparent.secondary + } + fontColorExtraLight={theme.font.color.extraLight} + hasSoftFocus={hasSoftFocus} + > + {children} + </StyledBaseContainer> + </CellHotkeyScopeContext.Provider> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellButton.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellButton.tsx index 287df8331f07..25b0e10f6379 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellButton.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellButton.tsx @@ -1,8 +1,8 @@ import styled from '@emotion/styled'; import { IconComponent } from 'twenty-ui'; -import { AnimatedContainer } from '@/object-record/record-table/components/AnimatedContainer'; import { FloatingIconButton } from '@/ui/input/button/components/FloatingIconButton'; +import { AnimatedContainer } from '@/ui/utilities/animation/components/AnimatedContainer'; const StyledButtonContainer = styled.div` margin: ${({ theme }) => theme.spacing(1)}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/CheckboxCell.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellCheckbox.tsx similarity index 75% rename from packages/twenty-front/src/modules/object-record/record-table/components/CheckboxCell.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellCheckbox.tsx index 2aff77e1b162..a261fa2ae375 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/CheckboxCell.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellCheckbox.tsx @@ -1,9 +1,10 @@ -import { useCallback, useContext } from 'react'; import styled from '@emotion/styled'; +import { useCallback, useContext } from 'react'; import { useRecoilValue, useSetRecoilState } from 'recoil'; import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableTd } from '@/object-record/record-table/record-table-cell/components/RecordTableTd'; import { useSetCurrentRowSelected } from '@/object-record/record-table/record-table-row/hooks/useSetCurrentRowSelected'; import { Checkbox } from '@/ui/input/components/Checkbox'; import { actionBarOpenState } from '@/ui/navigation/action-bar/states/actionBarIsOpenState'; @@ -16,10 +17,11 @@ const StyledContainer = styled.div` height: 32px; justify-content: center; - background-color: ${({ theme }) => theme.background.primary}; `; -export const CheckboxCell = () => { +export const RecordTableCellCheckbox = () => { + const { isSelected } = useContext(RecordTableRowContext); + const { recordId } = useContext(RecordTableRowContext); const { isRowSelectedFamilyState } = useRecordTableStates(); const setActionBarOpenState = useSetRecoilState(actionBarOpenState); @@ -32,8 +34,10 @@ export const CheckboxCell = () => { }, [currentRowSelected, setActionBarOpenState, setCurrentRowSelected]); return ( - <StyledContainer onClick={handleClick}> - <Checkbox checked={currentRowSelected} /> - </StyledContainer> + <RecordTableTd isSelected={isSelected} hasRightBorder={false}> + <StyledContainer onClick={handleClick}> + <Checkbox checked={currentRowSelected} /> + </StyledContainer> + </RecordTableTd> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellContainer.module.css b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellContainer.module.css deleted file mode 100644 index eef506c67a98..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellContainer.module.css +++ /dev/null @@ -1,32 +0,0 @@ -.td-in-edit-mode { - z-index: 4 !important; -} - -.td-not-in-edit-mode { - z-index: 3; -} - -.td-is-selected { - background: var(--twentycrm-accent-quaternary); -} - -.td-is-not-selected { - background: var(--twentycrm-background-primary); -} - -.cell-base-container { - align-items: center; - box-sizing: border-box; - cursor: pointer; - display: flex; - height: 32px; - position: relative; - user-select: none; -} - -.cell-base-container-soft-focus { - background: var(--twentycrm-background-transparent-secondary); - border-radius: var(--twentycrm-border-radius-sm); - outline: 1px solid var(--twentycrm-font-color-extra-light); -} - diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellContainer.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellContainer.tsx index 6ff27ac7339b..0197e7e02697 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellContainer.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellContainer.tsx @@ -1,163 +1,41 @@ -import React, { ReactElement, useContext, useEffect, useState } from 'react'; -import { clsx } from 'clsx'; +import { ReactElement, useContext } from 'react'; -import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; -import { useFieldFocus } from '@/object-record/record-field/hooks/useFieldFocus'; -import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; -import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; +import { RecordTableCellBaseContainer } from '@/object-record/record-table/record-table-cell/components/RecordTableCellBaseContainer'; import { RecordTableCellSoftFocusMode } from '@/object-record/record-table/record-table-cell/components/RecordTableCellSoftFocusMode'; -import { useCurrentTableCellPosition } from '@/object-record/record-table/record-table-cell/hooks/useCurrentCellPosition'; -import { useOpenRecordTableCellFromCell } from '@/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellFromCell'; -import { HotkeyScope } from '@/ui/utilities/hotkey/types/HotkeyScope'; - -import { CellHotkeyScopeContext } from '../../contexts/CellHotkeyScopeContext'; -import { TableHotkeyScope } from '../../types/TableHotkeyScope'; import { RecordTableCellDisplayMode } from './RecordTableCellDisplayMode'; import { RecordTableCellEditMode } from './RecordTableCellEditMode'; -import styles from './RecordTableCellContainer.module.css'; - export type RecordTableCellContainerProps = { editModeContent: ReactElement; nonEditModeContent: ReactElement; - editHotkeyScope?: HotkeyScope; transparent?: boolean; maxContentWidth?: number; onSubmit?: () => void; onCancel?: () => void; }; -const DEFAULT_CELL_SCOPE: HotkeyScope = { - scope: TableHotkeyScope.CellEditMode, -}; - export const RecordTableCellContainer = ({ editModeContent, nonEditModeContent, - editHotkeyScope, }: RecordTableCellContainerProps) => { - const { setIsFocused } = useFieldFocus(); - const { openTableCell } = useOpenRecordTableCellFromCell(); - - const { isSelected, recordId, isPendingRow } = useContext( - RecordTableRowContext, - ); - const { isLabelIdentifier } = useContext(FieldContext); - const { onContextMenu, onCellMouseEnter } = useContext(RecordTableContext); - - const shouldBeInitiallyInEditMode = - isPendingRow === true && isLabelIdentifier; - - const [hasSoftFocus, setHasSoftFocus] = useState(false); - const [isInEditMode, setIsInEditMode] = useState(shouldBeInitiallyInEditMode); - - const cellPosition = useCurrentTableCellPosition(); - - const handleContextMenu = (event: React.MouseEvent) => { - onContextMenu(event, recordId); - }; - - const handleContainerMouseMove = () => { - if (!hasSoftFocus) { - onCellMouseEnter({ - cellPosition, - }); - } - }; - - const handleContainerMouseLeave = () => { - setHasSoftFocus(false); - setIsFocused(false); - }; - - const handleContainerClick = () => { - if (!hasSoftFocus) { - openTableCell(); - } - }; - - useEffect(() => { - const customEventListener = (event: any) => { - event.stopPropagation(); - - const newHasSoftFocus = event.detail; - - setHasSoftFocus(newHasSoftFocus); - setIsFocused(newHasSoftFocus); - }; - - document.addEventListener( - `soft-focus-move-${cellPosition.row}:${cellPosition.column}`, - customEventListener, - ); - - return () => { - document.removeEventListener( - `soft-focus-move-${cellPosition.row}:${cellPosition.column}`, - customEventListener, - ); - }; - }, [cellPosition, setIsFocused]); - - useEffect(() => { - const customEventListener = (event: any) => { - const newIsInEditMode = event.detail; - - setIsInEditMode(newIsInEditMode); - }; - - document.addEventListener( - `edit-mode-change-${cellPosition.row}:${cellPosition.column}`, - customEventListener, - ); - - return () => { - document.removeEventListener( - `edit-mode-change-${cellPosition.row}:${cellPosition.column}`, - customEventListener, - ); - }; - }, [cellPosition]); + const { hasSoftFocus, isInEditMode } = useContext(RecordTableCellContext); return ( - <td - className={clsx({ - [styles.tdInEditMode]: isInEditMode, - [styles.tdNotInEditMode]: !isInEditMode, - [styles.tdIsSelected]: isSelected, - [styles.tdIsNotSelected]: !isSelected, - })} - onContextMenu={handleContextMenu} - > - <CellHotkeyScopeContext.Provider - value={editHotkeyScope ?? DEFAULT_CELL_SCOPE} - > - <div - onMouseLeave={handleContainerMouseLeave} - onMouseMove={handleContainerMouseMove} - onClick={handleContainerClick} - className={clsx({ - [styles.cellBaseContainer]: true, - [styles.cellBaseContainerSoftFocus]: hasSoftFocus, - })} - > - {isInEditMode ? ( - <RecordTableCellEditMode>{editModeContent}</RecordTableCellEditMode> - ) : hasSoftFocus ? ( - <> - <RecordTableCellSoftFocusMode - editModeContent={editModeContent} - nonEditModeContent={nonEditModeContent} - /> - </> - ) : ( - <RecordTableCellDisplayMode> - {nonEditModeContent} - </RecordTableCellDisplayMode> - )} - </div> - </CellHotkeyScopeContext.Provider> - </td> + <RecordTableCellBaseContainer> + {isInEditMode ? ( + <RecordTableCellEditMode>{editModeContent}</RecordTableCellEditMode> + ) : hasSoftFocus ? ( + <RecordTableCellSoftFocusMode + editModeContent={editModeContent} + nonEditModeContent={nonEditModeContent} + /> + ) : ( + <RecordTableCellDisplayMode> + {nonEditModeContent} + </RecordTableCellDisplayMode> + )} + </RecordTableCellBaseContainer> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayContainer.module.css b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayContainer.module.css deleted file mode 100644 index d2184fcda0ba..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayContainer.module.css +++ /dev/null @@ -1,24 +0,0 @@ -.cell-display-outer-container { - align-items: center; - display: flex; - height: 100%; - overflow: hidden; - padding-left: 8px; - padding-right: 4px; - width: 100%; -} - -.cell-display-outer-container-soft-focus { - background: var(--twentycrm-background-transparent-secondary); - border-radius: var(--twentycrm-border-radius-sm); - outline: 1px solid var(--twentycrm-font-color-extra-light); -} - -.cell-display-inner-container { - align-items: center; - display: flex; - height: 100%; - overflow: hidden; - width: 100%; -} - diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayContainer.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayContainer.tsx index f2db2901da0e..20941b7c63a6 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayContainer.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayContainer.tsx @@ -1,7 +1,24 @@ import { Ref } from 'react'; -import clsx from 'clsx'; +import { styled } from '@linaria/react'; -import styles from './RecordTableCellDisplayContainer.module.css'; +const StyledOuterContainer = styled.div<{ + hasSoftFocus?: boolean; +}>` + align-items: center; + display: flex; + height: 100%; + overflow: hidden; + padding-left: 6px; + width: 100%; +`; + +const StyledInnerContainer = styled.div` + align-items: center; + display: flex; + height: 100%; + overflow: hidden; + width: 100%; +`; export type EditableCellDisplayContainerProps = { softFocus?: boolean; @@ -16,17 +33,14 @@ export const RecordTableCellDisplayContainer = ({ onClick, scrollRef, }: React.PropsWithChildren<EditableCellDisplayContainerProps>) => ( - <div + <StyledOuterContainer data-testid={ softFocus ? 'editable-cell-soft-focus-mode' : 'editable-cell-display-mode' } onClick={onClick} - className={clsx({ - [styles.cellDisplayOuterContainer]: true, - [styles.cellDisplayOuterContainerSoftFocus]: softFocus, - })} ref={scrollRef} + hasSoftFocus={softFocus} > - <div className={clsx(styles.cellDisplayInnerContainer)}>{children}</div> - </div> + <StyledInnerContainer>{children}</StyledInnerContainer> + </StyledOuterContainer> ); diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayMode.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayMode.tsx index 566be79baf32..f562348e7974 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayMode.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellDisplayMode.tsx @@ -4,7 +4,8 @@ import { RecordTableCellDisplayContainer } from './RecordTableCellDisplayContain export const RecordTableCellDisplayMode = ({ children, -}: React.PropsWithChildren<unknown>) => { + softFocus, +}: React.PropsWithChildren<{ softFocus?: boolean }>) => { const isEmpty = useIsFieldEmpty(); if (isEmpty) { @@ -12,7 +13,7 @@ export const RecordTableCellDisplayMode = ({ } return ( - <RecordTableCellDisplayContainer> + <RecordTableCellDisplayContainer softFocus={softFocus}> {children} </RecordTableCellDisplayContainer> ); diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellEditMode.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellEditMode.tsx index 99c52de425dc..16e59db8d559 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellEditMode.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellEditMode.tsx @@ -4,6 +4,7 @@ import styled from '@emotion/styled'; import { autoUpdate, flip, offset, useFloating } from '@floating-ui/react'; const StyledEditableCellEditModeContainer = styled.div<RecordTableCellEditModeProps>` + position: absolute; align-items: center; display: flex; min-width: 200px; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableCellFieldContextWrapper.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellFieldContextWrapper.tsx similarity index 89% rename from packages/twenty-front/src/modules/object-record/record-table/components/RecordTableCellFieldContextWrapper.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellFieldContextWrapper.tsx index ec30ebe03eba..16a571fcd5cc 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableCellFieldContextWrapper.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellFieldContextWrapper.tsx @@ -1,4 +1,4 @@ -import { useContext } from 'react'; +import { ReactNode, useContext } from 'react'; import { isLabelIdentifierField } from '@/object-metadata/utils/isLabelIdentifierField'; import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; @@ -8,13 +8,16 @@ import { RecordUpdateContext } from '@/object-record/record-table/contexts/Entit import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; -import { RecordTableCell } from '@/object-record/record-table/record-table-cell/components/RecordTableCell'; import { TableHotkeyScope } from '@/object-record/record-table/types/TableHotkeyScope'; import { RelationPickerHotkeyScope } from '@/object-record/relation-picker/types/RelationPickerHotkeyScope'; import { SelectFieldHotkeyScope } from '@/object-record/select/types/SelectFieldHotkeyScope'; import { isUndefinedOrNull } from '~/utils/isUndefinedOrNull'; -export const RecordTableCellFieldContextWrapper = () => { +export const RecordTableCellFieldContextWrapper = ({ + children, +}: { + children: ReactNode; +}) => { const { objectMetadataItem } = useContext(RecordTableContext); const { columnDefinition } = useContext(RecordTableCellContext); const { recordId, pathToShowPage } = useContext(RecordTableRowContext); @@ -49,7 +52,7 @@ export const RecordTableCellFieldContextWrapper = () => { }), }} > - <RecordTableCell customHotkeyScope={{ scope: customHotkeyScope }} /> + {children} </FieldContext.Provider> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellFieldInput.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellFieldInput.tsx new file mode 100644 index 000000000000..8b56f8d24b39 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellFieldInput.tsx @@ -0,0 +1,95 @@ +import { useContext } from 'react'; + +import { FieldInput } from '@/object-record/record-field/components/FieldInput'; +import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; +import { FieldInputEvent } from '@/object-record/record-field/types/FieldInputEvent'; +import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; + +export const RecordTableCellFieldInput = () => { + const { onUpsertRecord, onMoveFocus, onCloseTableCell } = + useContext(RecordTableContext); + const { entityId, fieldDefinition } = useContext(FieldContext); + const { isReadOnly } = useContext(RecordTableRowContext); + + const handleEnter: FieldInputEvent = (persistField) => { + onUpsertRecord({ + persistField, + entityId, + fieldName: fieldDefinition.metadata.fieldName, + }); + + onCloseTableCell(); + onMoveFocus('down'); + }; + + const handleSubmit: FieldInputEvent = (persistField) => { + onUpsertRecord({ + persistField, + entityId, + fieldName: fieldDefinition.metadata.fieldName, + }); + + onCloseTableCell(); + }; + + const handleCancel = () => { + onCloseTableCell(); + }; + + const handleClickOutside: FieldInputEvent = (persistField) => { + onUpsertRecord({ + persistField, + entityId, + fieldName: fieldDefinition.metadata.fieldName, + }); + + onCloseTableCell(); + }; + + const handleEscape: FieldInputEvent = (persistField) => { + onUpsertRecord({ + persistField, + entityId, + fieldName: fieldDefinition.metadata.fieldName, + }); + + onCloseTableCell(); + }; + + const handleTab: FieldInputEvent = (persistField) => { + onUpsertRecord({ + persistField, + entityId, + fieldName: fieldDefinition.metadata.fieldName, + }); + + onCloseTableCell(); + onMoveFocus('right'); + }; + + const handleShiftTab: FieldInputEvent = (persistField) => { + onUpsertRecord({ + persistField, + entityId, + fieldName: fieldDefinition.metadata.fieldName, + }); + + onCloseTableCell(); + onMoveFocus('left'); + }; + + return ( + <FieldInput + recordFieldInputdId={`${entityId}-${fieldDefinition?.metadata?.fieldName}`} + onCancel={handleCancel} + onClickOutside={handleClickOutside} + onEnter={handleEnter} + onEscape={handleEscape} + onShiftTab={handleShiftTab} + onSubmit={handleSubmit} + onTab={handleTab} + isReadOnly={isReadOnly} + /> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellGrip.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellGrip.tsx new file mode 100644 index 000000000000..563646604447 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellGrip.tsx @@ -0,0 +1,44 @@ +import styled from '@emotion/styled'; +import { useContext } from 'react'; + +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { RecordTableTd } from '@/object-record/record-table/record-table-cell/components/RecordTableTd'; +import { IconListViewGrip } from '@/ui/input/components/IconListViewGrip'; + +const StyledContainer = styled.div` + cursor: grab; + width: 16px; + height: 32px; + z-index: 200; + display: flex; + &:hover .icon { + opacity: 1; + } + + border-color: transparent; +`; + +const StyledIconWrapper = styled.div<{ isDragging: boolean }>` + opacity: ${({ isDragging }) => (isDragging ? 1 : 0)}; + transition: opacity 0.1s; +`; + +export const RecordTableCellGrip = () => { + const { dragHandleProps, isDragging } = useContext(RecordTableRowContext); + + return ( + <RecordTableTd + // eslint-disable-next-line react/jsx-props-no-spreading + {...dragHandleProps} + data-select-disable + hasRightBorder={false} + hasBottomBorder={false} + > + <StyledContainer> + <StyledIconWrapper className="icon" isDragging={isDragging}> + <IconListViewGrip /> + </StyledIconWrapper> + </StyledContainer> + </RecordTableTd> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellLoading.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellLoading.tsx new file mode 100644 index 000000000000..a8ca441b60e7 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellLoading.tsx @@ -0,0 +1,10 @@ +import { RecordTableCellSkeletonLoader } from '@/object-record/record-table/record-table-cell/components/RecordTableCellSkeletonLoader'; +import { RecordTableTd } from '@/object-record/record-table/record-table-cell/components/RecordTableTd'; + +export const RecordTableCellLoading = () => { + return ( + <RecordTableTd> + <RecordTableCellSkeletonLoader /> + </RecordTableTd> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellSkeletonLoader.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellSkeletonLoader.tsx new file mode 100644 index 000000000000..b010bdaecd6d --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellSkeletonLoader.tsx @@ -0,0 +1,29 @@ +import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; +import { useTheme } from '@emotion/react'; +import styled from '@emotion/styled'; + +const StyledSkeletonContainer = styled.div` + padding-left: ${({ theme }) => theme.spacing(2)}; + padding-right: ${({ theme }) => theme.spacing(1)}; +`; + +const StyledRecordTableCellLoader = ({ width }: { width?: number }) => { + const theme = useTheme(); + return ( + <SkeletonTheme + baseColor={theme.background.tertiary} + highlightColor={theme.background.transparent.lighter} + borderRadius={4} + > + <Skeleton width={width} height={16} /> + </SkeletonTheme> + ); +}; + +export const RecordTableCellSkeletonLoader = () => { + return ( + <StyledSkeletonContainer> + <StyledRecordTableCellLoader /> + </StyledSkeletonContainer> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellSoftFocusMode.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellSoftFocusMode.tsx index e7f3a03599e3..4a2ebb10c88e 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellSoftFocusMode.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellSoftFocusMode.tsx @@ -13,7 +13,6 @@ import { RecordTableCellContext } from '@/object-record/record-table/contexts/Re import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; import { useCloseCurrentTableCellInEditMode } from '@/object-record/record-table/hooks/internal/useCloseCurrentTableCellInEditMode'; import { RecordTableCellButton } from '@/object-record/record-table/record-table-cell/components/RecordTableCellButton'; -import { useCurrentTableCellPosition } from '@/object-record/record-table/record-table-cell/hooks/useCurrentCellPosition'; import { useOpenRecordTableCellFromCell } from '@/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellFromCell'; import { isSoftFocusUsingMouseState } from '@/object-record/record-table/states/isSoftFocusUsingMouseState'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; @@ -130,15 +129,10 @@ export const RecordTableCellSoftFocusMode = ({ */ }; - const { column, row } = useCurrentTableCellPosition(); - useListenClickOutside({ refs: [scrollRef], callback: () => { closeCurrentTableCell(); - document.dispatchEvent( - new CustomEvent(`soft-focus-move-${row}:${column}`, { detail: false }), - ); }, }); @@ -159,6 +153,7 @@ export const RecordTableCellSoftFocusMode = ({ <RecordTableCellDisplayContainer onClick={handleClick} scrollRef={scrollRef} + softFocus > {editModeContentOnly ? editModeContent : nonEditModeContent} </RecordTableCellDisplayContainer> diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellWrapper.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellWrapper.tsx new file mode 100644 index 000000000000..c552fa47beb6 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableCellWrapper.tsx @@ -0,0 +1,75 @@ +import { useContext, useMemo } from 'react'; +import { useRecoilValue } from 'recoil'; + +import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; +import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { RecordTableCellFieldContextWrapper } from '@/object-record/record-table/record-table-cell/components/RecordTableCellFieldContextWrapper'; +import { RecordTableScopeInternalContext } from '@/object-record/record-table/scopes/scope-internal-context/RecordTableScopeInternalContext'; +import { isSoftFocusOnTableCellComponentFamilyState } from '@/object-record/record-table/states/isSoftFocusOnTableCellComponentFamilyState'; +import { isTableCellInEditModeComponentFamilyState } from '@/object-record/record-table/states/isTableCellInEditModeComponentFamilyState'; +import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; +import { TableCellPosition } from '@/object-record/record-table/types/TableCellPosition'; +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; +import { getScopeIdOrUndefinedFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdOrUndefinedFromComponentId'; +import { extractComponentFamilyState } from '@/ui/utilities/state/component-state/utils/extractComponentFamilyState'; + +export const RecordTableCellWrapper = ({ + children, + column, + columnIndex, +}: { + column: ColumnDefinition<FieldMetadata>; + columnIndex: number; + children: React.ReactNode; +}) => { + const tableScopeId = useAvailableScopeIdOrThrow( + RecordTableScopeInternalContext, + getScopeIdOrUndefinedFromComponentId(), + ); + + const { rowIndex } = useContext(RecordTableRowContext); + + const currentTableCellPosition: TableCellPosition = useMemo( + () => ({ + column: columnIndex, + row: rowIndex, + }), + [columnIndex, rowIndex], + ); + + const isTableCellInEditModeFamilyState = extractComponentFamilyState( + isTableCellInEditModeComponentFamilyState, + tableScopeId, + ); + + const isSoftFocusOnTableCellFamilyState = extractComponentFamilyState( + isSoftFocusOnTableCellComponentFamilyState, + tableScopeId, + ); + + const isInEditMode = useRecoilValue( + isTableCellInEditModeFamilyState(currentTableCellPosition), + ); + + const hasSoftFocus = useRecoilValue( + isSoftFocusOnTableCellFamilyState(currentTableCellPosition), + ); + + return ( + <RecordTableCellContext.Provider + value={{ + columnDefinition: column, + columnIndex, + isInEditMode, + hasSoftFocus, + cellPosition: currentTableCellPosition, + }} + key={column.fieldMetadataId} + > + <RecordTableCellFieldContextWrapper> + {children} + </RecordTableCellFieldContextWrapper> + </RecordTableCellContext.Provider> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableLastEmptyCell.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableLastEmptyCell.tsx new file mode 100644 index 000000000000..c8f5bccdfed7 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableLastEmptyCell.tsx @@ -0,0 +1,10 @@ +import { useContext } from 'react'; + +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { RecordTableTd } from '@/object-record/record-table/record-table-cell/components/RecordTableTd'; + +export const RecordTableLastEmptyCell = () => { + const { isSelected } = useContext(RecordTableRowContext); + + return <RecordTableTd isSelected={isSelected} hasRightBorder={false} />; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableTd.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableTd.tsx new file mode 100644 index 000000000000..49f51197db11 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/components/RecordTableTd.tsx @@ -0,0 +1,102 @@ +import { DraggableProvidedDragHandleProps } from '@hello-pangea/dnd'; +import { styled } from '@linaria/react'; +import { ReactNode, useContext } from 'react'; +import { MOBILE_VIEWPORT, ThemeContext } from 'twenty-ui'; + +import { isDefined } from '~/utils/isDefined'; + +const StyledTd = styled.td<{ + zIndex?: number; + backgroundColor: string; + borderColor: string; + isDragging?: boolean; + fontColor: string; + sticky?: boolean; + freezeFirstColumns?: boolean; + left?: number; + hasRightBorder?: boolean; + hasBottomBorder?: boolean; +}>` + border-bottom: 1px solid + ${({ borderColor, hasBottomBorder }) => + hasBottomBorder ? borderColor : 'transparent'}; + color: ${({ fontColor }) => fontColor}; + border-right: 1px solid + ${({ borderColor, hasRightBorder }) => + hasRightBorder ? borderColor : 'transparent'}; + + padding: 0; + + text-align: left; + + background: ${({ backgroundColor }) => backgroundColor}; + z-index: ${({ zIndex }) => (isDefined(zIndex) ? zIndex : 'auto')}; + + ${({ isDragging }) => + isDragging + ? ` + background-color: transparent; + border-color: transparent; + ` + : ''} + + ${({ freezeFirstColumns }) => + freezeFirstColumns + ? `@media (max-width: ${MOBILE_VIEWPORT}px) { + width: 35px; + max-width: 35px; + }` + : ''} +`; + +export const RecordTableTd = ({ + children, + zIndex, + isSelected, + isDragging, + sticky, + freezeFirstColumns, + left, + hasRightBorder = true, + hasBottomBorder = true, + ...dragHandleProps +}: { + className?: string; + children?: ReactNode; + zIndex?: number; + isSelected?: boolean; + isDragging?: boolean; + sticky?: boolean; + freezeFirstColumns?: boolean; + hasRightBorder?: boolean; + hasBottomBorder?: boolean; + left?: number; +} & (Partial<DraggableProvidedDragHandleProps> | null)) => { + const { theme } = useContext(ThemeContext); + + const tdBackgroundColor = isSelected + ? theme.accent.quaternary + : theme.background.primary; + + const borderColor = theme.border.color.light; + const fontColor = theme.font.color.primary; + + return ( + <StyledTd + isDragging={isDragging} + zIndex={zIndex} + backgroundColor={tdBackgroundColor} + borderColor={borderColor} + fontColor={fontColor} + sticky={sticky} + freezeFirstColumns={freezeFirstColumns} + left={left} + hasRightBorder={hasRightBorder} + hasBottomBorder={hasBottomBorder} + // eslint-disable-next-line react/jsx-props-no-spreading + {...dragHandleProps} + > + {children} + </StyledTd> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__mocks__/cell.ts b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__mocks__/cell.ts index 792b33adc009..9aad45bc4e26 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__mocks__/cell.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__mocks__/cell.ts @@ -1,6 +1,5 @@ -import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; +import { RecordTableCellContextProps } from '@/object-record/record-table/contexts/RecordTableCellContext'; import { RecordTableRowContextProps } from '@/object-record/record-table/contexts/RecordTableRowContext'; -import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; import { FieldMetadataType } from '~/generated-metadata/graphql'; export const recordTableRow: RecordTableRowContextProps = { @@ -10,12 +9,13 @@ export const recordTableRow: RecordTableRowContextProps = { pathToShowPage: '/', objectNameSingular: 'objectNameSingular', isReadOnly: false, + dragHandleProps: {} as any, + isDragging: false, + inView: true, + isPendingRow: false, }; -export const recordTableCell: { - columnDefinition: ColumnDefinition<FieldMetadata>; - columnIndex: number; -} = { +export const recordTableCell:RecordTableCellContextProps= { columnIndex: 3, columnDefinition: { size: 1, @@ -29,4 +29,10 @@ export const recordTableCell: { fieldName: 'fieldName', }, }, + cellPosition: { + row: 2, + column: 3, + }, + hasSoftFocus: false, + isInEditMode: false, }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useUpsertRecord.test.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useUpsertRecord.test.tsx index 291c7407df60..2bf6ff27a658 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useUpsertRecord.test.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useUpsertRecord.test.tsx @@ -1,5 +1,5 @@ -import { ReactNode } from 'react'; import { act, renderHook } from '@testing-library/react'; +import { ReactNode } from 'react'; import { RecoilRoot } from 'recoil'; import { createState } from 'twenty-ui'; @@ -10,9 +10,10 @@ import { FieldContext } from '@/object-record/record-field/contexts/FieldContext import { useUpsertRecord } from '@/object-record/record-table/record-table-cell/hooks/useUpsertRecord'; import { TableHotkeyScope } from '@/object-record/record-table/types/TableHotkeyScope'; -const pendingRecordId = 'a7286b9a-c039-4a89-9567-2dfa7953cda9'; const draftValue = 'updated Name'; +// Todo refactor this test to inject the states in a cleaner way instead of mocking hooks +// (this is not easy to maintain while refactoring) jest.mock('@/object-record/hooks/useCreateOneRecord', () => ({ __esModule: true, useCreateOneRecord: jest.fn(), @@ -93,17 +94,25 @@ describe('useUpsertRecord', () => { }); it('calls update record if there is no pending record', async () => { - const { result } = renderHook(() => useUpsertRecord(), { - wrapper: ({ children }) => - Wrapper({ - pendingRecordIdMockedValue: null, - draftValueMockedValue: null, - children, - }), - }); + const { result } = renderHook( + () => useUpsertRecord({ objectNameSingular: 'person' }), + { + wrapper: ({ children }) => + Wrapper({ + pendingRecordIdMockedValue: null, + draftValueMockedValue: null, + children, + }), + }, + ); await act(async () => { - await result.current.upsertRecord(updateOneRecordMock); + await result.current.upsertRecord( + updateOneRecordMock, + 'entityId', + 'name', + 'recordTableId', + ); }); expect(createOneRecordMock).not.toHaveBeenCalled(); @@ -111,42 +120,28 @@ describe('useUpsertRecord', () => { }); it('calls update record if pending record is empty', async () => { - const { result } = renderHook(() => useUpsertRecord(), { - wrapper: ({ children }) => - Wrapper({ - pendingRecordIdMockedValue: null, - draftValueMockedValue: draftValue, - children, - }), - }); + const { result } = renderHook( + () => useUpsertRecord({ objectNameSingular: 'person' }), + { + wrapper: ({ children }) => + Wrapper({ + pendingRecordIdMockedValue: null, + draftValueMockedValue: draftValue, + children, + }), + }, + ); await act(async () => { - await result.current.upsertRecord(updateOneRecordMock); + await result.current.upsertRecord( + updateOneRecordMock, + 'entityId', + 'name', + 'recordTableId', + ); }); expect(createOneRecordMock).not.toHaveBeenCalled(); expect(updateOneRecordMock).toHaveBeenCalled(); }); - - it('calls create record if pending record is not empty', async () => { - const { result } = renderHook(() => useUpsertRecord(), { - wrapper: ({ children }) => - Wrapper({ - pendingRecordIdMockedValue: pendingRecordId, - draftValueMockedValue: draftValue, - children, - }), - }); - - await act(async () => { - await result.current.upsertRecord(updateOneRecordMock); - }); - - expect(createOneRecordMock).toHaveBeenCalledWith({ - id: pendingRecordId, - name: draftValue, - position: 'first', - }); - expect(updateOneRecordMock).not.toHaveBeenCalled(); - }); }); diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCurrentCellPosition.ts b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCurrentCellPosition.ts index d3384909cf9d..383a81de4050 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCurrentCellPosition.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCurrentCellPosition.ts @@ -1,21 +1,9 @@ -import { useContext, useMemo } from 'react'; +import { useContext } from 'react'; import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; -import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; - -import { TableCellPosition } from '../../types/TableCellPosition'; export const useCurrentTableCellPosition = () => { - const { rowIndex } = useContext(RecordTableRowContext); - const { columnIndex } = useContext(RecordTableCellContext); - - const currentTableCellPosition: TableCellPosition = useMemo( - () => ({ - column: columnIndex, - row: rowIndex, - }), - [columnIndex, rowIndex], - ); + const { cellPosition } = useContext(RecordTableCellContext); - return currentTableCellPosition; + return cellPosition; }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellV2.ts b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellV2.ts index b9e279b0b77a..2a379840276f 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellV2.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCellV2.ts @@ -1,4 +1,3 @@ -import { useNavigate } from 'react-router-dom'; import { useRecoilCallback, useSetRecoilState } from 'recoil'; import { useInitDraftValueV2 } from '@/object-record/record-field/hooks/useInitDraftValueV2'; @@ -21,6 +20,8 @@ import { useClickOutsideListener } from '@/ui/utilities/pointer-event/hooks/useC import { getSnapshotValue } from '@/ui/utilities/state/utils/getSnapshotValue'; import { isDefined } from '~/utils/isDefined'; +import { RecordIndexEventContext } from '@/object-record/record-index/contexts/RecordIndexEventContext'; +import { useContext } from 'react'; import { TableHotkeyScope } from '../../types/TableHotkeyScope'; export const DEFAULT_CELL_SCOPE: HotkeyScope = { @@ -40,13 +41,13 @@ export type OpenTableCellArgs = { }; export const useOpenRecordTableCellV2 = (tableScopeId: string) => { + const { onIndexIdentifierClick } = useContext(RecordIndexEventContext); const moveEditModeToTableCellPosition = useMoveEditModeToTableCellPosition(tableScopeId); const setHotkeyScope = useSetHotkeyScope(); const { setDragSelectionStartEnabled } = useDragSelect(); - const navigate = useNavigate(); const leaveTableFocus = useLeaveTableFocus(tableScopeId); const { toggleClickOutsideListener } = useClickOutsideListener( SOFT_FOCUS_CLICK_OUTSIDE_LISTENER_ID, @@ -66,7 +67,6 @@ export const useOpenRecordTableCellV2 = (tableScopeId: string) => { initialValue, cellPosition, isReadOnly, - pathToShowPage, objectNameSingular, customCellHotkeyScope, fieldDefinition, @@ -94,7 +94,8 @@ export const useOpenRecordTableCellV2 = (tableScopeId: string) => { if (isFirstColumnCell && !isEmpty && !isActionButtonClick) { leaveTableFocus(); - navigate(pathToShowPage); + + onIndexIdentifierClick(entityId); return; } @@ -142,7 +143,7 @@ export const useOpenRecordTableCellV2 = (tableScopeId: string) => { openRightDrawer, setViewableRecordId, setViewableRecordNameSingular, - navigate, + onIndexIdentifierClick, ], ); diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useUpsertRecord.ts b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useUpsertRecord.ts index 230e4a5d321a..553c6aa835e0 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useUpsertRecord.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useUpsertRecord.ts @@ -1,41 +1,65 @@ -import { useContext } from 'react'; -import { useRecoilValue } from 'recoil'; +import { useRecoilCallback } from 'recoil'; import { useCreateOneRecord } from '@/object-record/hooks/useCreateOneRecord'; -import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; -import { useRecordFieldInputStates } from '@/object-record/record-field/hooks/internal/useRecordFieldInputStates'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { recordFieldInputDraftValueComponentSelector } from '@/object-record/record-field/states/selectors/recordFieldInputDraftValueComponentSelector'; +import { recordTablePendingRecordIdComponentState } from '@/object-record/record-table/states/recordTablePendingRecordIdComponentState'; +import { getScopeIdFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdFromComponentId'; +import { getSnapshotValue } from '@/ui/utilities/recoil-scope/utils/getSnapshotValue'; +import { extractComponentSelector } from '@/ui/utilities/state/component-state/utils/extractComponentSelector'; +import { extractComponentState } from '@/ui/utilities/state/component-state/utils/extractComponentState'; import { isDefined } from '~/utils/isDefined'; -export const useUpsertRecord = () => { - const { entityId, fieldDefinition } = useContext(FieldContext); - - const { pendingRecordIdState } = useRecordTableStates(); - - const pendingRecordId = useRecoilValue(pendingRecordIdState); - const fieldName = fieldDefinition.metadata.fieldName; - const { getDraftValueSelector } = useRecordFieldInputStates( - `${entityId}-${fieldName}`, - ); - const draftValue = useRecoilValue(getDraftValueSelector()); - - const objectNameSingular = - fieldDefinition.metadata.objectMetadataNameSingular ?? ''; +export const useUpsertRecord = ({ + objectNameSingular, +}: { + objectNameSingular: string; +}) => { const { createOneRecord } = useCreateOneRecord({ objectNameSingular, }); - const upsertRecord = (persistField: () => void) => { - if (isDefined(pendingRecordId) && isDefined(draftValue)) { - createOneRecord({ - id: pendingRecordId, - name: draftValue, - position: 'first', - }); - } else if (!pendingRecordId) { - persistField(); - } - }; + const upsertRecord = useRecoilCallback( + ({ snapshot }) => + ( + persistField: () => void, + entityId: string, + fieldName: string, + recordTableId: string, + ) => { + const tableScopeId = getScopeIdFromComponentId(recordTableId); + + const recordTablePendingRecordIdState = extractComponentState( + recordTablePendingRecordIdComponentState, + tableScopeId, + ); + + const recordTablePendingRecordId = getSnapshotValue( + snapshot, + recordTablePendingRecordIdState, + ); + const fieldScopeId = getScopeIdFromComponentId( + `${entityId}-${fieldName}`, + ); + + const draftValueSelector = extractComponentSelector( + recordFieldInputDraftValueComponentSelector, + fieldScopeId, + ); + + const draftValue = getSnapshotValue(snapshot, draftValueSelector()); + + if (isDefined(recordTablePendingRecordId) && isDefined(draftValue)) { + createOneRecord({ + id: recordTablePendingRecordId, + name: draftValue, + position: 'first', + }); + } else if (!recordTablePendingRecordId) { + persistField(); + } + }, + [createOneRecord], + ); return { upsertRecord }; }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useUpsertRecordV2.ts b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useUpsertRecordV2.ts deleted file mode 100644 index c0b714ab2cc4..000000000000 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useUpsertRecordV2.ts +++ /dev/null @@ -1,65 +0,0 @@ -import { useRecoilCallback } from 'recoil'; - -import { useCreateOneRecord } from '@/object-record/hooks/useCreateOneRecord'; -import { recordFieldInputDraftValueComponentSelector } from '@/object-record/record-field/states/selectors/recordFieldInputDraftValueComponentSelector'; -import { recordTablePendingRecordIdComponentState } from '@/object-record/record-table/states/recordTablePendingRecordIdComponentState'; -import { getScopeIdFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdFromComponentId'; -import { getSnapshotValue } from '@/ui/utilities/recoil-scope/utils/getSnapshotValue'; -import { extractComponentSelector } from '@/ui/utilities/state/component-state/utils/extractComponentSelector'; -import { extractComponentState } from '@/ui/utilities/state/component-state/utils/extractComponentState'; -import { isDefined } from '~/utils/isDefined'; - -export const useUpsertRecordV2 = ({ - objectNameSingular, -}: { - objectNameSingular: string; -}) => { - const { createOneRecord } = useCreateOneRecord({ - objectNameSingular, - }); - - const upsertRecord = useRecoilCallback( - ({ snapshot }) => - ( - persistField: () => void, - entityId: string, - fieldName: string, - recordTableId: string, - ) => { - const tableScopeId = getScopeIdFromComponentId(recordTableId); - - const recordTablePendingRecordIdState = extractComponentState( - recordTablePendingRecordIdComponentState, - tableScopeId, - ); - - const recordTablePendingRecordId = getSnapshotValue( - snapshot, - recordTablePendingRecordIdState, - ); - const fieldScopeId = getScopeIdFromComponentId( - `${entityId}-${fieldName}`, - ); - - const draftValueSelector = extractComponentSelector( - recordFieldInputDraftValueComponentSelector, - fieldScopeId, - ); - - const draftValue = getSnapshotValue(snapshot, draftValueSelector()); - - if (isDefined(recordTablePendingRecordId) && isDefined(draftValue)) { - createOneRecord({ - id: recordTablePendingRecordId, - name: draftValue, - position: 'first', - }); - } else if (!recordTablePendingRecordId) { - persistField(); - } - }, - [createOneRecord], - ); - - return { upsertRecord }; -}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/ColumnHead.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHead.tsx similarity index 67% rename from packages/twenty-front/src/modules/object-record/record-table/components/ColumnHead.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHead.tsx index 89caf5a48fc3..7a5c9f3cd6f7 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/ColumnHead.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHead.tsx @@ -1,14 +1,14 @@ import { css, useTheme } from '@emotion/react'; import styled from '@emotion/styled'; -import { useRecoilValue } from 'recoil'; import { MOBILE_VIEWPORT, useIcons } from 'twenty-ui'; import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; -import { scrollLeftState } from '@/ui/utilities/scroll/states/scrollLeftState'; +import { isRecordTableScrolledLeftComponentState } from '@/object-record/record-table/states/isRecordTableScrolledLeftComponentState'; +import { useRecoilComponentValue } from '@/ui/utilities/state/component-state/hooks/useRecoilComponentValue'; -import { ColumnDefinition } from '../types/ColumnDefinition'; +import { ColumnDefinition } from '../../types/ColumnDefinition'; -type ColumnHeadProps = { +type RecordTableColumnHeadProps = { column: ColumnDefinition<FieldMetadata>; }; @@ -46,16 +46,22 @@ const StyledText = styled.span` white-space: nowrap; `; -export const ColumnHead = ({ column }: ColumnHeadProps) => { +export const RecordTableColumnHead = ({ + column, +}: RecordTableColumnHeadProps) => { const theme = useTheme(); const { getIcon } = useIcons(); const Icon = getIcon(column.iconName); - const scrollLeft = useRecoilValue(scrollLeftState); + const isRecordTableScrolledLeft = useRecoilComponentValue( + isRecordTableScrolledLeftComponentState, + ); return ( - <StyledTitle hideTitle={!!column.isLabelIdentifier && scrollLeft > 0}> + <StyledTitle + hideTitle={!!column.isLabelIdentifier && !isRecordTableScrolledLeft} + > <StyledIcon> <Icon size={theme.icon.size.md} /> </StyledIcon> diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableColumnDropdownMenu.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHeadDropdownMenu.tsx similarity index 82% rename from packages/twenty-front/src/modules/object-record/record-table/components/RecordTableColumnDropdownMenu.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHeadDropdownMenu.tsx index 96a257041fe1..6065e45a1198 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableColumnDropdownMenu.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHeadDropdownMenu.tsx @@ -14,16 +14,16 @@ import { DropdownMenuSeparator } from '@/ui/layout/dropdown/components/DropdownM import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown'; import { MenuItem } from '@/ui/navigation/menu-item/components/MenuItem'; -import { useTableColumns } from '../hooks/useTableColumns'; -import { ColumnDefinition } from '../types/ColumnDefinition'; +import { useTableColumns } from '../../hooks/useTableColumns'; +import { ColumnDefinition } from '../../types/ColumnDefinition'; -export type RecordTableColumnDropdownMenuProps = { +export type RecordTableColumnHeadDropdownMenuProps = { column: ColumnDefinition<FieldMetadata>; }; -export const RecordTableColumnDropdownMenu = ({ +export const RecordTableColumnHeadDropdownMenu = ({ column, -}: RecordTableColumnDropdownMenuProps) => { +}: RecordTableColumnHeadDropdownMenuProps) => { const { visibleTableColumnsSelector, onToggleColumnFilterState, @@ -33,12 +33,13 @@ export const RecordTableColumnDropdownMenu = ({ const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); const secondVisibleColumn = visibleTableColumns[1]; + const canMove = column.isLabelIdentifier !== true; const canMoveLeft = - column.fieldMetadataId !== secondVisibleColumn?.fieldMetadataId; + column.fieldMetadataId !== secondVisibleColumn?.fieldMetadataId && canMove; const lastVisibleColumn = visibleTableColumns[visibleTableColumns.length - 1]; const canMoveRight = - column.fieldMetadataId !== lastVisibleColumn?.fieldMetadataId; + column.fieldMetadataId !== lastVisibleColumn?.fieldMetadataId && canMove; const { handleColumnVisibilityChange, handleMoveTableColumn } = useTableColumns(); @@ -83,7 +84,9 @@ export const RecordTableColumnDropdownMenu = ({ const isSortable = column.isSortable === true; const isFilterable = column.isFilterable === true; - const showSeparator = isFilterable || isSortable; + const showSeparator = + (isFilterable || isSortable) && column.isLabelIdentifier !== true; + const canHide = column.isLabelIdentifier !== true; return ( <DropdownMenuItemsContainer> @@ -116,11 +119,13 @@ export const RecordTableColumnDropdownMenu = ({ text="Move right" /> )} - <MenuItem - LeftIcon={IconEyeOff} - onClick={handleColumnVisibility} - text="Hide" - /> + {canHide && ( + <MenuItem + LeftIcon={IconEyeOff} + onClick={handleColumnVisibility} + text="Hide" + /> + )} </DropdownMenuItemsContainer> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/ColumnHeadWithDropdown.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHeadWithDropdown.tsx similarity index 55% rename from packages/twenty-front/src/modules/object-record/record-table/components/ColumnHeadWithDropdown.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHeadWithDropdown.tsx index 475fb4beba60..6298c77f1dac 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/ColumnHeadWithDropdown.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableColumnHeadWithDropdown.tsx @@ -4,27 +4,31 @@ import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata' import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; import { Dropdown } from '@/ui/layout/dropdown/components/Dropdown'; -import { ColumnHead } from './ColumnHead'; -import { RecordTableColumnDropdownMenu } from './RecordTableColumnDropdownMenu'; +import { RecordTableColumnHeadDropdownMenu } from './RecordTableColumnHeadDropdownMenu'; -type ColumnHeadWithDropdownProps = { +import { RecordTableColumnHead } from './RecordTableColumnHead'; + +type RecordTableColumnHeadWithDropdownProps = { column: ColumnDefinition<FieldMetadata>; }; const StyledDropdown = styled(Dropdown)` display: flex; + flex: 1; + z-index: ${({ theme }) => theme.lastLayerZIndex}; `; -export const ColumnHeadWithDropdown = ({ +export const RecordTableColumnHeadWithDropdown = ({ column, -}: ColumnHeadWithDropdownProps) => { +}: RecordTableColumnHeadWithDropdownProps) => { return ( <StyledDropdown dropdownId={column.fieldMetadataId + '-header'} - clickableComponent={<ColumnHead column={column} />} - dropdownComponents={<RecordTableColumnDropdownMenu column={column} />} + clickableComponent={<RecordTableColumnHead column={column} />} + dropdownComponents={<RecordTableColumnHeadDropdownMenu column={column} />} dropdownOffset={{ x: -1 }} + usePortal dropdownPlacement="bottom-start" dropdownHotkeyScope={{ scope: column.fieldMetadataId + '-header' }} /> diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeader.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeader.tsx new file mode 100644 index 000000000000..34009d15421d --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeader.tsx @@ -0,0 +1,101 @@ +import styled from '@emotion/styled'; +import { useRecoilValue } from 'recoil'; +import { MOBILE_VIEWPORT } from 'twenty-ui'; + +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableHeaderCell } from '@/object-record/record-table/record-table-header/components/RecordTableHeaderCell'; +import { RecordTableHeaderCheckboxColumn } from '@/object-record/record-table/record-table-header/components/RecordTableHeaderCheckboxColumn'; +import { RecordTableHeaderDragDropColumn } from '@/object-record/record-table/record-table-header/components/RecordTableHeaderDragDropColumn'; +import { RecordTableHeaderLastColumn } from '@/object-record/record-table/record-table-header/components/RecordTableHeaderLastColumn'; + +const StyledTableHead = styled.thead<{ + isScrolledTop?: boolean; + isScrolledLeft?: boolean; +}>` + cursor: pointer; + + th:nth-of-type(1) { + width: 9px; + left: 0; + border-right-color: ${({ theme }) => theme.background.primary}; + } + + th:nth-of-type(2) { + border-right-color: ${({ theme }) => theme.background.primary}; + } + + &.first-columns-sticky { + th:nth-of-type(1) { + position: sticky; + left: 0; + z-index: 5; + } + th:nth-of-type(2) { + position: sticky; + left: 9px; + z-index: 5; + } + th:nth-of-type(3) { + position: sticky; + left: 39px; + z-index: 5; + &::after { + content: ''; + position: absolute; + top: -1px; + height: calc(100% + 2px); + width: 4px; + right: 0px; + box-shadow: ${({ theme }) => theme.boxShadow.light}; + clip-path: inset(0px -4px 0px 0px); + } + @media (max-width: ${MOBILE_VIEWPORT}px) { + width: 35px; + max-width: 35px; + } + } + } + + &.header-sticky { + th { + position: sticky; + top: 0; + z-index: 5; + } + } + + &.header-sticky.first-columns-sticky { + th:nth-of-type(1), + th:nth-of-type(2), + th:nth-of-type(3) { + z-index: 10; + } + } +`; + +export const RecordTableHeader = ({ + createRecord, +}: { + createRecord: () => void; +}) => { + const { visibleTableColumnsSelector } = useRecordTableStates(); + + const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); + + return ( + <StyledTableHead id="record-table-header" data-select-disable> + <tr> + <RecordTableHeaderDragDropColumn /> + <RecordTableHeaderCheckboxColumn /> + {visibleTableColumns.map((column) => ( + <RecordTableHeaderCell + key={column.fieldMetadataId} + column={column} + createRecord={createRecord} + /> + ))} + <RecordTableHeaderLastColumn /> + </tr> + </StyledTableHead> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeaderCell.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderCell.tsx similarity index 83% rename from packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeaderCell.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderCell.tsx index 698939c85b8c..d243b2a29b6e 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeaderCell.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderCell.tsx @@ -1,28 +1,35 @@ -import { useCallback, useMemo, useState } from 'react'; import styled from '@emotion/styled'; +import { useCallback, useMemo, useState } from 'react'; import { useRecoilCallback, useRecoilState, useRecoilValue } from 'recoil'; import { IconPlus } from 'twenty-ui'; import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; -import { ColumnHead } from '@/object-record/record-table/components/ColumnHead'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; import { useTableColumns } from '@/object-record/record-table/hooks/useTableColumns'; +import { RecordTableColumnHeadWithDropdown } from '@/object-record/record-table/record-table-header/components/RecordTableColumnHeadWithDropdown'; +import { isRecordTableScrolledLeftComponentState } from '@/object-record/record-table/states/isRecordTableScrolledLeftComponentState'; import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; import { LightIconButton } from '@/ui/input/button/components/LightIconButton'; import { useTrackPointer } from '@/ui/utilities/pointer-event/hooks/useTrackPointer'; import { getSnapshotValue } from '@/ui/utilities/recoil-scope/utils/getSnapshotValue'; import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; -import { scrollLeftState } from '@/ui/utilities/scroll/states/scrollLeftState'; +import { useRecoilComponentValue } from '@/ui/utilities/state/component-state/hooks/useRecoilComponentValue'; import { mapArrayToObject } from '~/utils/array/mapArrayToObject'; -import { ColumnHeadWithDropdown } from './ColumnHeadWithDropdown'; - const COLUMN_MIN_WIDTH = 104; const StyledColumnHeaderCell = styled.th<{ columnWidth: number; isResizing?: boolean; }>` + border-bottom: 1px solid ${({ theme }) => theme.border.color.light}; + border-top: 1px solid ${({ theme }) => theme.border.color.light}; + color: ${({ theme }) => theme.font.color.tertiary}; + padding: 0; + text-align: left; + + background-color: ${({ theme }) => theme.background.primary}; + border-right: 1px solid ${({ theme }) => theme.border.color.light}; ${({ columnWidth }) => ` min-width: ${columnWidth}px; width: ${columnWidth}px; @@ -32,7 +39,10 @@ const StyledColumnHeaderCell = styled.th<{ ${({ theme }) => { return ` &:hover { - background: ${theme.background.quaternary}; + background: ${theme.background.secondary}; + }; + &:active { + background: ${theme.background.tertiary}; }; `; }}; @@ -163,11 +173,14 @@ export const RecordTableHeaderCell = ({ onMouseUp: handleResizeHandlerEnd, }); + const isRecordTableScrolledLeft = useRecoilComponentValue( + isRecordTableScrolledLeftComponentState, + ); + const isMobile = useIsMobile(); - const scrollLeft = useRecoilValue(scrollLeftState); const disableColumnResize = - column.isLabelIdentifier && isMobile && scrollLeft > 0; + column.isLabelIdentifier && isMobile && !isRecordTableScrolledLeft; return ( <StyledColumnHeaderCell @@ -183,11 +196,7 @@ export const RecordTableHeaderCell = ({ onMouseLeave={() => setIconVisibility(false)} > <StyledColumnHeadContainer> - {column.isLabelIdentifier ? ( - <ColumnHead column={column} /> - ) : ( - <ColumnHeadWithDropdown column={column} /> - )} + <RecordTableColumnHeadWithDropdown column={column} /> {(useIsMobile() || iconVisibility) && !!column.isLabelIdentifier && ( <StyledHeaderIcon> <LightIconButton diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/SelectAllCheckbox.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderCheckboxColumn.tsx similarity index 61% rename from packages/twenty-front/src/modules/object-record/record-table/components/SelectAllCheckbox.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderCheckboxColumn.tsx index 870b49a19d3d..ef88ebee3021 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/SelectAllCheckbox.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderCheckboxColumn.tsx @@ -2,9 +2,9 @@ import styled from '@emotion/styled'; import { useRecoilValue } from 'recoil'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { useRecordTable } from '@/object-record/record-table/hooks/useRecordTable'; import { Checkbox } from '@/ui/input/components/Checkbox'; - -import { useRecordTable } from '../hooks/useRecordTable'; +import { useTheme } from '@emotion/react'; const StyledContainer = styled.div` align-items: center; @@ -16,7 +16,7 @@ const StyledContainer = styled.div` background-color: ${({ theme }) => theme.background.primary}; `; -export const SelectAllCheckbox = () => { +export const RecordTableHeaderCheckboxColumn = () => { const { allRowsSelectedStatusSelector } = useRecordTableStates(); const allRowsSelectedStatus = useRecoilValue(allRowsSelectedStatusSelector()); @@ -36,13 +36,26 @@ export const SelectAllCheckbox = () => { } }; + const theme = useTheme(); + return ( - <StyledContainer> - <Checkbox - checked={checked} - onChange={onChange} - indeterminate={indeterminate} - /> - </StyledContainer> + <th + style={{ + borderBottom: `1px solid ${theme.border.color.light}`, + borderTop: `1px solid ${theme.border.color.light}`, + width: 30, + minWidth: 30, + maxWidth: 30, + borderRight: 'transparent', + }} + > + <StyledContainer> + <Checkbox + checked={checked} + onChange={onChange} + indeterminate={indeterminate} + /> + </StyledContainer> + </th> ); }; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderDragDropColumn.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderDragDropColumn.tsx new file mode 100644 index 000000000000..3acf7afa9ad1 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderDragDropColumn.tsx @@ -0,0 +1,10 @@ +import { styled } from '@linaria/react'; + +const StyledTh = styled.th` + border-bottom: none; + border-top: none; +`; + +export const RecordTableHeaderDragDropColumn = () => { + return <StyledTh></StyledTh>; +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderLastColumn.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderLastColumn.tsx new file mode 100644 index 000000000000..170ec63f74b9 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderLastColumn.tsx @@ -0,0 +1,89 @@ +import { Theme } from '@emotion/react'; +import styled from '@emotion/styled'; +import { useContext } from 'react'; +import { useRecoilValue } from 'recoil'; +import { IconPlus, ThemeContext } from 'twenty-ui'; + +import { HIDDEN_TABLE_COLUMN_DROPDOWN_ID } from '@/object-record/record-table/constants/HiddenTableColumnDropdownId'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableHeaderPlusButtonContent } from '@/object-record/record-table/record-table-header/components/RecordTableHeaderPlusButtonContent'; +import { Dropdown } from '@/ui/layout/dropdown/components/Dropdown'; +import { useScrollWrapperScopedRef } from '@/ui/utilities/scroll/hooks/useScrollWrapperScopedRef'; + +const StyledPlusIconHeaderCell = styled.th<{ + theme: Theme; + isTableWiderThanScreen: boolean; +}>` + ${({ theme }) => { + return ` + &:hover { + background: ${theme.background.transparent.light}; + }; + padding-left: ${theme.spacing(3)}; + `; + }}; + border-bottom: 1px solid ${({ theme }) => theme.border.color.light}; + border-top: 1px solid ${({ theme }) => theme.border.color.light}; + background-color: ${({ theme }) => theme.background.primary}; + border-left: none !important; + color: ${({ theme }) => theme.font.color.tertiary}; + min-width: 32px; + border-right: none !important; + + ${({ isTableWiderThanScreen, theme }) => + isTableWiderThanScreen + ? ` + width: 32px; + background-color: ${theme.background.primary}; + ` + : ''}; + z-index: 1; +`; + +const StyledPlusIconContainer = styled.div` + align-items: center; + display: flex; + height: 32px; + justify-content: center; + width: 32px; +`; + +const HIDDEN_TABLE_COLUMN_DROPDOWN_HOTKEY_SCOPE_ID = + 'hidden-table-columns-dropdown-hotkey-scope-id'; + +export const RecordTableHeaderLastColumn = () => { + const { theme } = useContext(ThemeContext); + + const scrollWrapper = useScrollWrapperScopedRef(); + + const isTableWiderThanScreen = + (scrollWrapper.current?.clientWidth ?? 0) < + (scrollWrapper.current?.scrollWidth ?? 0); + + const { hiddenTableColumnsSelector } = useRecordTableStates(); + + const hiddenTableColumns = useRecoilValue(hiddenTableColumnsSelector()); + + return ( + <StyledPlusIconHeaderCell + theme={theme} + isTableWiderThanScreen={isTableWiderThanScreen} + > + {hiddenTableColumns.length > 0 && ( + <Dropdown + dropdownId={HIDDEN_TABLE_COLUMN_DROPDOWN_ID} + clickableComponent={ + <StyledPlusIconContainer> + <IconPlus size={theme.icon.size.md} /> + </StyledPlusIconContainer> + } + dropdownComponents={<RecordTableHeaderPlusButtonContent />} + dropdownPlacement="bottom-start" + dropdownHotkeyScope={{ + scope: HIDDEN_TABLE_COLUMN_DROPDOWN_HOTKEY_SCOPE_ID, + }} + /> + )} + </StyledPlusIconHeaderCell> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeaderPlusButtonContent.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderPlusButtonContent.tsx similarity index 94% rename from packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeaderPlusButtonContent.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderPlusButtonContent.tsx index 2a19bc15f248..3fd2cc60e8b5 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTableHeaderPlusButtonContent.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-header/components/RecordTableHeaderPlusButtonContent.tsx @@ -4,6 +4,7 @@ import styled from '@emotion/styled'; import { useRecoilValue } from 'recoil'; import { IconSettings, useIcons } from 'twenty-ui'; +import { getObjectSlug } from '@/object-metadata/utils/getObjectSlug'; import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; @@ -57,7 +58,7 @@ export const RecordTableHeaderPlusButtonContent = () => { )} <DropdownMenuItemsContainer> <StyledMenuItemLink - to={`/settings/objects/${objectMetadataItem.namePlural}`} + to={`/settings/objects/${getObjectSlug(objectMetadataItem)}`} > <MenuItem LeftIcon={IconSettings} text="Customize fields" /> </StyledMenuItemLink> diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCells.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCells.tsx new file mode 100644 index 000000000000..25c557144d9f --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCells.tsx @@ -0,0 +1,17 @@ +import { useContext } from 'react'; + +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { RecordTableCellsEmpty } from '@/object-record/record-table/record-table-row/components/RecordTableCellsEmpty'; +import { RecordTableCellsVisible } from '@/object-record/record-table/record-table-row/components/RecordTableCellsVisible'; + +export const RecordTableCells = () => { + const { inView, isDragging } = useContext(RecordTableRowContext); + + const areCellsVisible = inView || isDragging; + + return areCellsVisible ? ( + <RecordTableCellsVisible /> + ) : ( + <RecordTableCellsEmpty /> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCellsEmpty.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCellsEmpty.tsx new file mode 100644 index 000000000000..2df7a27f4a5a --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCellsEmpty.tsx @@ -0,0 +1,17 @@ +import { useContext } from 'react'; +import { useRecoilValue } from 'recoil'; + +import { RecordTableTd } from '@/object-record/record-table/record-table-cell/components/RecordTableTd'; +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; + +export const RecordTableCellsEmpty = () => { + const { isSelected } = useContext(RecordTableRowContext); + const { visibleTableColumnsSelector } = useRecordTableStates(); + + const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); + + return visibleTableColumns.map((column) => ( + <RecordTableTd isSelected={isSelected} key={column.fieldMetadataId} /> + )); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCellsVisible.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCellsVisible.tsx new file mode 100644 index 000000000000..e6b75b265074 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableCellsVisible.tsx @@ -0,0 +1,39 @@ +import { useContext } from 'react'; +import { useRecoilValue } from 'recoil'; + +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableCell } from '@/object-record/record-table/record-table-cell/components/RecordTableCell'; +import { RecordTableCellWrapper } from '@/object-record/record-table/record-table-cell/components/RecordTableCellWrapper'; +import { RecordTableTd } from '@/object-record/record-table/record-table-cell/components/RecordTableTd'; + +export const RecordTableCellsVisible = () => { + const { isDragging } = useContext(RecordTableRowContext); + const { visibleTableColumnsSelector } = useRecordTableStates(); + + const visibleTableColumns = useRecoilValue(visibleTableColumnsSelector()); + + const tableColumnsAfterFirst = visibleTableColumns.slice(1); + + return ( + <> + <RecordTableCellWrapper column={visibleTableColumns[0]} columnIndex={0}> + <RecordTableTd> + <RecordTableCell /> + </RecordTableTd> + </RecordTableCellWrapper> + {!isDragging && + tableColumnsAfterFirst.map((column, columnIndex) => ( + <RecordTableCellWrapper + key={column.fieldMetadataId} + column={column} + columnIndex={columnIndex + 1} + > + <RecordTableTd> + <RecordTableCell /> + </RecordTableTd> + </RecordTableCellWrapper> + ))} + </> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTablePendingRow.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTablePendingRow.tsx similarity index 76% rename from packages/twenty-front/src/modules/object-record/record-table/components/RecordTablePendingRow.tsx rename to packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTablePendingRow.tsx index cde96d52e1a3..addf2d747b8a 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/RecordTablePendingRow.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTablePendingRow.tsx @@ -1,13 +1,13 @@ import { useRecoilValue } from 'recoil'; -import { RecordTableRow } from '@/object-record/record-table/components/RecordTableRow'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableRow } from '@/object-record/record-table/record-table-row/components/RecordTableRow'; export const RecordTablePendingRow = () => { const { pendingRecordIdState } = useRecordTableStates(); const pendingRecordId = useRecoilValue(pendingRecordIdState); - if (!pendingRecordId) return; + if (!pendingRecordId) return <></>; return ( <RecordTableRow diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableRow.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableRow.tsx new file mode 100644 index 000000000000..382fdc230372 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableRow.tsx @@ -0,0 +1,32 @@ +import { RecordValueSetterEffect } from '@/object-record/record-store/components/RecordValueSetterEffect'; +import { RecordTableCellCheckbox } from '@/object-record/record-table/record-table-cell/components/RecordTableCellCheckbox'; +import { RecordTableCellGrip } from '@/object-record/record-table/record-table-cell/components/RecordTableCellGrip'; +import { RecordTableLastEmptyCell } from '@/object-record/record-table/record-table-cell/components/RecordTableLastEmptyCell'; +import { RecordTableCells } from '@/object-record/record-table/record-table-row/components/RecordTableCells'; +import { RecordTableRowWrapper } from '@/object-record/record-table/record-table-row/components/RecordTableRowWrapper'; + +type RecordTableRowProps = { + recordId: string; + rowIndex: number; + isPendingRow?: boolean; +}; + +export const RecordTableRow = ({ + recordId, + rowIndex, + isPendingRow, +}: RecordTableRowProps) => { + return ( + <RecordTableRowWrapper + recordId={recordId} + rowIndex={rowIndex} + isPendingRow={isPendingRow} + > + <RecordTableCellGrip /> + <RecordTableCellCheckbox /> + <RecordTableCells /> + <RecordTableLastEmptyCell /> + <RecordValueSetterEffect recordId={recordId} /> + </RecordTableRowWrapper> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableRowWrapper.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableRowWrapper.tsx new file mode 100644 index 000000000000..193c852d9e88 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableRowWrapper.tsx @@ -0,0 +1,96 @@ +import { useTheme } from '@emotion/react'; +import { Draggable } from '@hello-pangea/dnd'; +import { ReactNode, useContext, useEffect } from 'react'; +import { useInView } from 'react-intersection-observer'; +import { useRecoilValue } from 'recoil'; + +import { getBasePathToShowPage } from '@/object-metadata/utils/getBasePathToShowPage'; +import { RecordIndexEventContext } from '@/object-record/record-index/contexts/RecordIndexEventContext'; +import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext'; +import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; +import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; +import { RecordTableTr } from '@/object-record/record-table/record-table-row/components/RecordTableTr'; +import { ScrollWrapperContext } from '@/ui/utilities/scroll/components/ScrollWrapper'; + +export const RecordTableRowWrapper = ({ + recordId, + rowIndex, + isPendingRow, + children, +}: { + recordId: string; + rowIndex: number; + isPendingRow?: boolean; + children: ReactNode; +}) => { + const { objectMetadataItem } = useContext(RecordTableContext); + const { onIndexRecordsLoaded } = useContext(RecordIndexEventContext); + + const theme = useTheme(); + + const { isRowSelectedFamilyState } = useRecordTableStates(); + const currentRowSelected = useRecoilValue(isRowSelectedFamilyState(recordId)); + + const scrollWrapperRef = useContext(ScrollWrapperContext); + + const { ref: elementRef, inView } = useInView({ + root: scrollWrapperRef.current?.querySelector( + '[data-overlayscrollbars-viewport="scrollbarHidden"]', + ), + rootMargin: '1000px', + }); + + // TODO: find a better way to emit this event + useEffect(() => { + if (inView) { + onIndexRecordsLoaded?.(); + } + }, [inView, onIndexRecordsLoaded]); + + return ( + <Draggable key={recordId} draggableId={recordId} index={rowIndex}> + {(draggableProvided, draggableSnapshot) => ( + <RecordTableTr + ref={(node) => { + elementRef(node); + draggableProvided.innerRef(node); + }} + // eslint-disable-next-line react/jsx-props-no-spreading + {...draggableProvided.draggableProps} + style={{ + ...draggableProvided.draggableProps.style, + background: draggableSnapshot.isDragging + ? theme.background.transparent.light + : 'none', + borderColor: draggableSnapshot.isDragging + ? `${theme.border.color.medium}` + : 'transparent', + }} + isDragging={draggableSnapshot.isDragging} + data-testid={`row-id-${recordId}`} + data-selectable-id={recordId} + > + <RecordTableRowContext.Provider + value={{ + recordId, + rowIndex, + pathToShowPage: + getBasePathToShowPage({ + objectNameSingular: objectMetadataItem.nameSingular, + }) + recordId, + objectNameSingular: objectMetadataItem.nameSingular, + isSelected: currentRowSelected, + isReadOnly: objectMetadataItem.isRemote ?? false, + isPendingRow, + isDragging: draggableSnapshot.isDragging, + dragHandleProps: draggableProvided.dragHandleProps, + inView, + }} + > + {children} + </RecordTableRowContext.Provider> + </RecordTableTr> + )} + </Draggable> + ); +}; diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableTr.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableTr.tsx new file mode 100644 index 000000000000..0ccfc98ac751 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-row/components/RecordTableTr.tsx @@ -0,0 +1,11 @@ +import styled from '@emotion/styled'; + +const StyledTr = styled.tr<{ isDragging: boolean }>` + border: ${({ isDragging, theme }) => + isDragging + ? `1px solid ${theme.border.color.medium}` + : '1px solid transparent'}; + transition: border-left-color 0.2s ease-in-out; +`; + +export const RecordTableTr = StyledTr; diff --git a/packages/twenty-front/src/modules/object-record/record-table/states/isRecordTableScrolledLeftComponentState.ts b/packages/twenty-front/src/modules/object-record/record-table/states/isRecordTableScrolledLeftComponentState.ts new file mode 100644 index 000000000000..6f26d8a6082a --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/states/isRecordTableScrolledLeftComponentState.ts @@ -0,0 +1,9 @@ +import { RecordTableScopeInternalContext } from '@/object-record/record-table/scopes/scope-internal-context/RecordTableScopeInternalContext'; +import { createComponentStateV2 } from '@/ui/utilities/state/component-state/utils/createComponentStateV2'; + +export const isRecordTableScrolledLeftComponentState = + createComponentStateV2<boolean>({ + key: 'isRecordTableScrolledLeftComponentState', + componentContext: RecordTableScopeInternalContext, + defaultValue: true, + }); diff --git a/packages/twenty-front/src/modules/object-record/record-table/states/isRecordTableScrolledTopComponentState.ts b/packages/twenty-front/src/modules/object-record/record-table/states/isRecordTableScrolledTopComponentState.ts new file mode 100644 index 000000000000..564c567a6062 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-table/states/isRecordTableScrolledTopComponentState.ts @@ -0,0 +1,9 @@ +import { RecordTableScopeInternalContext } from '@/object-record/record-table/scopes/scope-internal-context/RecordTableScopeInternalContext'; +import { createComponentStateV2 } from '@/ui/utilities/state/component-state/utils/createComponentStateV2'; + +export const isRecordTableScrolledTopComponentState = + createComponentStateV2<boolean>({ + key: 'isRecordTableScrolledTopComponentState', + componentContext: RecordTableScopeInternalContext, + defaultValue: true, + }); diff --git a/packages/twenty-front/src/modules/object-record/record-table/states/selectors/allRowsSelectedStatusComponentSelector.ts b/packages/twenty-front/src/modules/object-record/record-table/states/selectors/allRowsSelectedStatusComponentSelector.ts index c0a42ea81015..6ce6b42ff6c7 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/states/selectors/allRowsSelectedStatusComponentSelector.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/states/selectors/allRowsSelectedStatusComponentSelector.ts @@ -1,5 +1,5 @@ -import { numberOfTableRowsComponentState } from '@/object-record/record-table/states/numberOfTableRowsComponentState'; import { selectedRowIdsComponentSelector } from '@/object-record/record-table/states/selectors/selectedRowIdsComponentSelector'; +import { tableRowIdsComponentState } from '@/object-record/record-table/states/tableRowIdsComponentState'; import { createComponentReadOnlySelector } from '@/ui/utilities/state/component-state/utils/createComponentReadOnlySelector'; import { AllRowsSelectedStatus } from '../../types/AllRowSelectedStatus'; @@ -10,7 +10,7 @@ export const allRowsSelectedStatusComponentSelector = get: ({ scopeId }) => ({ get }) => { - const numberOfRows = get(numberOfTableRowsComponentState({ scopeId })); + const tableRowIds = get(tableRowIdsComponentState({ scopeId })); const selectedRowIds = get( selectedRowIdsComponentSelector({ scopeId }), @@ -21,7 +21,7 @@ export const allRowsSelectedStatusComponentSelector = const allRowsSelectedStatus = numberOfSelectedRows === 0 ? 'none' - : numberOfRows === numberOfSelectedRows + : selectedRowIds.length === tableRowIds.length ? 'all' : 'some'; diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/components/ActivityTargetInlineCellEditModeMultiRecordsEffect.tsx b/packages/twenty-front/src/modules/object-record/relation-picker/components/ActivityTargetInlineCellEditModeMultiRecordsEffect.tsx new file mode 100644 index 000000000000..440bab3c780d --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/relation-picker/components/ActivityTargetInlineCellEditModeMultiRecordsEffect.tsx @@ -0,0 +1,132 @@ +import { useEffect } from 'react'; +import { + useRecoilCallback, + useRecoilState, + useRecoilValue, + useSetRecoilState, +} from 'recoil'; + +import { useObjectRecordMultiSelectScopedStates } from '@/activities/hooks/useObjectRecordMultiSelectScopedStates'; +import { objectRecordMultiSelectComponentFamilyState } from '@/object-record/record-field/states/objectRecordMultiSelectComponentFamilyState'; +import { useRelationPickerScopedStates } from '@/object-record/relation-picker/hooks/internal/useRelationPickerScopedStates'; +import { + ObjectRecordForSelect, + SelectedObjectRecordId, + useMultiObjectSearch, +} from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; +import { RelationPickerScopeInternalContext } from '@/object-record/relation-picker/scopes/scope-internal-context/RelationPickerScopeInternalContext'; +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; +import { isDeeplyEqual } from '~/utils/isDeeplyEqual'; + +export const ActivityTargetInlineCellEditModeMultiRecordsEffect = ({ + selectedObjectRecordIds, +}: { + selectedObjectRecordIds: SelectedObjectRecordId[]; +}) => { + const scopeId = useAvailableScopeIdOrThrow( + RelationPickerScopeInternalContext, + ); + const { + objectRecordsIdsMultiSelectState, + objectRecordMultiSelectCheckedRecordsIdsState, + recordMultiSelectIsLoadingState, + } = useObjectRecordMultiSelectScopedStates(scopeId); + const [objectRecordsIdsMultiSelect, setObjectRecordsIdsMultiSelect] = + useRecoilState(objectRecordsIdsMultiSelectState); + + const setRecordMultiSelectIsLoading = useSetRecoilState( + recordMultiSelectIsLoadingState, + ); + + const relationPickerScopedId = useAvailableScopeIdOrThrow( + RelationPickerScopeInternalContext, + ); + + const { relationPickerSearchFilterState } = useRelationPickerScopedStates({ + relationPickerScopedId, + }); + const relationPickerSearchFilter = useRecoilValue( + relationPickerSearchFilterState, + ); + const { filteredSelectedObjectRecords, loading, objectRecordsToSelect } = + useMultiObjectSearch({ + searchFilterValue: relationPickerSearchFilter, + selectedObjectRecordIds, + excludedObjectRecordIds: [], + limit: 10, + }); + + const [ + objectRecordMultiSelectCheckedRecordsIds, + setObjectRecordMultiSelectCheckedRecordsIds, + ] = useRecoilState(objectRecordMultiSelectCheckedRecordsIdsState); + + const updateRecords = useRecoilCallback( + ({ snapshot, set }) => + (newRecords: ObjectRecordForSelect[]) => { + for (const newRecord of newRecords) { + const currentRecord = snapshot + .getLoadable( + objectRecordMultiSelectComponentFamilyState({ + scopeId: scopeId, + familyKey: newRecord.record.id, + }), + ) + .getValue(); + + const newRecordWithSelected = { + ...newRecord, + selected: objectRecordMultiSelectCheckedRecordsIds.some( + (checkedRecordId) => checkedRecordId === newRecord.record.id, + ), + }; + + if ( + !isDeeplyEqual( + newRecordWithSelected.selected, + currentRecord?.selected, + ) + ) { + set( + objectRecordMultiSelectComponentFamilyState({ + scopeId: scopeId, + familyKey: newRecordWithSelected.record.id, + }), + newRecordWithSelected, + ); + } + } + }, + [objectRecordMultiSelectCheckedRecordsIds, scopeId], + ); + + useEffect(() => { + const allRecords = [ + ...(filteredSelectedObjectRecords ?? []), + ...(objectRecordsToSelect ?? []), + ]; + updateRecords(allRecords); + const allRecordsIds = allRecords.map((record) => record.record.id); + if (!isDeeplyEqual(allRecordsIds, objectRecordsIdsMultiSelect)) { + setObjectRecordsIdsMultiSelect(allRecordsIds); + } + }, [ + filteredSelectedObjectRecords, + objectRecordsIdsMultiSelect, + objectRecordsToSelect, + setObjectRecordsIdsMultiSelect, + updateRecords, + ]); + + useEffect(() => { + setObjectRecordMultiSelectCheckedRecordsIds( + selectedObjectRecordIds.map((rec) => rec.id), + ); + }, [selectedObjectRecordIds, setObjectRecordMultiSelectCheckedRecordsIds]); + + useEffect(() => { + setRecordMultiSelectIsLoading(loading); + }, [loading, setRecordMultiSelectIsLoading]); + + return <></>; +}; diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/components/MultiRecordSelect.tsx b/packages/twenty-front/src/modules/object-record/relation-picker/components/MultiRecordSelect.tsx index 049176e6cb08..fe0f3453a2ea 100644 --- a/packages/twenty-front/src/modules/object-record/relation-picker/components/MultiRecordSelect.tsx +++ b/packages/twenty-front/src/modules/object-record/relation-picker/components/MultiRecordSelect.tsx @@ -1,12 +1,14 @@ -import { useEffect, useMemo, useRef, useState } from 'react'; +import { useCallback, useRef } from 'react'; import styled from '@emotion/styled'; -import { isNonEmptyString } from '@sniptt/guards'; +import { useRecoilValue, useSetRecoilState } from 'recoil'; import { useDebouncedCallback } from 'use-debounce'; +import { useObjectRecordMultiSelectScopedStates } from '@/activities/hooks/useObjectRecordMultiSelectScopedStates'; import { MultipleObjectRecordOnClickOutsideEffect } from '@/object-record/relation-picker/components/MultipleObjectRecordOnClickOutsideEffect'; import { MultipleObjectRecordSelectItem } from '@/object-record/relation-picker/components/MultipleObjectRecordSelectItem'; import { MULTI_OBJECT_RECORD_SELECT_SELECTABLE_LIST_ID } from '@/object-record/relation-picker/constants/MultiObjectRecordSelectSelectableListId'; -import { ObjectRecordForSelect } from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; +import { useRelationPickerScopedStates } from '@/object-record/relation-picker/hooks/internal/useRelationPickerScopedStates'; +import { RelationPickerScopeInternalContext } from '@/object-record/relation-picker/scopes/scope-internal-context/RelationPickerScopeInternalContext'; import { RelationPickerHotkeyScope } from '@/object-record/relation-picker/types/RelationPickerHotkeyScope'; import { DropdownMenu } from '@/ui/layout/dropdown/components/DropdownMenu'; import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer'; @@ -15,7 +17,7 @@ import { DropdownMenuSeparator } from '@/ui/layout/dropdown/components/DropdownM import { SelectableItem } from '@/ui/layout/selectable-list/components/SelectableItem'; import { SelectableList } from '@/ui/layout/selectable-list/components/SelectableList'; import { MenuItem } from '@/ui/navigation/menu-item/components/MenuItem'; -import { isDefined } from '~/utils/isDefined'; +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; export const StyledSelectableItem = styled(SelectableItem)` height: 100%; @@ -24,134 +26,80 @@ export const StyledSelectableItem = styled(SelectableItem)` export const MultiRecordSelect = ({ onChange, onSubmit, - selectedObjectRecords, - allRecords, - loading, - searchFilter, - setSearchFilter, }: { - onChange?: ( - changedRecordForSelect: ObjectRecordForSelect, - newSelectedValue: boolean, - ) => void; - onSubmit?: (objectRecordsForSelect: ObjectRecordForSelect[]) => void; - selectedObjectRecords: ObjectRecordForSelect[]; - allRecords: ObjectRecordForSelect[]; - loading: boolean; - searchFilter: string; - setSearchFilter: (searchFilter: string) => void; + onChange?: (changedRecordForSelectId: string) => void; + onSubmit?: () => void; }) => { const containerRef = useRef<HTMLDivElement>(null); - const [internalSelectedRecords, setInternalSelectedRecords] = useState< - ObjectRecordForSelect[] - >([]); - - useEffect(() => { - if (!loading) { - setInternalSelectedRecords(selectedObjectRecords); - } - }, [selectedObjectRecords, loading]); - - const debouncedSetSearchFilter = useDebouncedCallback(setSearchFilter, 100, { - leading: true, - }); - - const handleFilterChange = (event: React.ChangeEvent<HTMLInputElement>) => { - debouncedSetSearchFilter(event.currentTarget.value); - }; + const relationPickerScopedId = useAvailableScopeIdOrThrow( + RelationPickerScopeInternalContext, + ); - const handleSelectChange = ( - changedRecordForSelect: ObjectRecordForSelect, - newSelectedValue: boolean, - ) => { - const newSelectedRecords = newSelectedValue - ? [...internalSelectedRecords, changedRecordForSelect] - : internalSelectedRecords.filter( - (selectedRecord) => - selectedRecord.record.id !== changedRecordForSelect.record.id, - ); + const { objectRecordsIdsMultiSelectState, recordMultiSelectIsLoadingState } = + useObjectRecordMultiSelectScopedStates(relationPickerScopedId); - setInternalSelectedRecords(newSelectedRecords); + const recordMultiSelectIsLoading = useRecoilValue( + recordMultiSelectIsLoadingState, + ); + const objectRecordsIdsMultiSelect = useRecoilValue( + objectRecordsIdsMultiSelectState, + ); - onChange?.(changedRecordForSelect, newSelectedValue); - }; + const { relationPickerSearchFilterState } = useRelationPickerScopedStates({ + relationPickerScopedId, + }); - const entitiesInDropdown = useMemo( - () => - [...(allRecords ?? [])].filter((entity) => - isNonEmptyString(entity.recordIdentifier.id), - ), - [allRecords], + const setSearchFilter = useSetRecoilState(relationPickerSearchFilterState); + const relationPickerSearchFilter = useRecoilValue( + relationPickerSearchFilterState, ); + const debouncedSetSearchFilter = useDebouncedCallback(setSearchFilter, 100, { + leading: true, + }); - const selectableItemIds = entitiesInDropdown.map( - (entity) => entity.record.id, + const handleFilterChange = useCallback( + (event: React.ChangeEvent<HTMLInputElement>) => { + debouncedSetSearchFilter(event.currentTarget.value); + }, + [debouncedSetSearchFilter], ); - return ( <> <MultipleObjectRecordOnClickOutsideEffect containerRef={containerRef} onClickOutside={() => { - onSubmit?.(internalSelectedRecords); + onSubmit?.(); }} /> <DropdownMenu ref={containerRef} data-select-disable> <DropdownMenuSearchInput - value={searchFilter} + value={relationPickerSearchFilter} onChange={handleFilterChange} autoFocus /> <DropdownMenuSeparator /> <DropdownMenuItemsContainer hasMaxHeight> - {loading ? ( + {recordMultiSelectIsLoading ? ( <MenuItem text="Loading..." /> ) : ( <> <SelectableList selectableListId={MULTI_OBJECT_RECORD_SELECT_SELECTABLE_LIST_ID} - selectableItemIdArray={selectableItemIds} + selectableItemIdArray={objectRecordsIdsMultiSelect} hotkeyScope={RelationPickerHotkeyScope.RelationPicker} - onEnter={(recordId) => { - const recordIsSelected = internalSelectedRecords?.some( - (selectedRecord) => selectedRecord.record.id === recordId, - ); - - const correspondingRecordForSelect = entitiesInDropdown?.find( - (entity) => entity.record.id === recordId, - ); - - if (isDefined(correspondingRecordForSelect)) { - handleSelectChange( - correspondingRecordForSelect, - !recordIsSelected, - ); - } - }} > - {entitiesInDropdown?.map((objectRecordForSelect) => ( - <MultipleObjectRecordSelectItem - key={objectRecordForSelect.record.id} - objectRecordForSelect={objectRecordForSelect} - onSelectedChange={(newSelectedValue) => - handleSelectChange( - objectRecordForSelect, - newSelectedValue, - ) - } - selected={internalSelectedRecords?.some( - (selectedRecord) => { - return ( - selectedRecord.record.id === - objectRecordForSelect.record.id - ); - }, - )} - /> - ))} + {objectRecordsIdsMultiSelect?.map((recordId) => { + return ( + <MultipleObjectRecordSelectItem + key={recordId} + objectRecordId={recordId} + onChange={onChange} + /> + ); + })} </SelectableList> - {entitiesInDropdown?.length === 0 && ( + {objectRecordsIdsMultiSelect?.length === 0 && ( <MenuItem text="No result" /> )} </> diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/components/MultipleObjectRecordSelect.tsx b/packages/twenty-front/src/modules/object-record/relation-picker/components/MultipleObjectRecordSelect.tsx deleted file mode 100644 index 659461a30d57..000000000000 --- a/packages/twenty-front/src/modules/object-record/relation-picker/components/MultipleObjectRecordSelect.tsx +++ /dev/null @@ -1,68 +0,0 @@ -import { useMemo, useState } from 'react'; -import styled from '@emotion/styled'; - -import { MultiRecordSelect } from '@/object-record/relation-picker/components/MultiRecordSelect'; -import { - ObjectRecordForSelect, - SelectedObjectRecordId, - useMultiObjectSearch, -} from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; -import { SelectableItem } from '@/ui/layout/selectable-list/components/SelectableItem'; - -export const StyledSelectableItem = styled(SelectableItem)` - height: 100%; - width: 100%; -`; -export const MultipleObjectRecordSelect = ({ - onChange, - onSubmit, - selectedObjectRecordIds, -}: { - onChange?: ( - changedRecordForSelect: ObjectRecordForSelect, - newSelectedValue: boolean, - ) => void; - onSubmit?: (objectRecordsForSelect: ObjectRecordForSelect[]) => void; - selectedObjectRecordIds: SelectedObjectRecordId[]; -}) => { - const [searchFilter, setSearchFilter] = useState<string>(''); - - const { - filteredSelectedObjectRecords, - loading, - objectRecordsToSelect, - selectedObjectRecords, - } = useMultiObjectSearch({ - searchFilterValue: searchFilter, - selectedObjectRecordIds, - excludedObjectRecordIds: [], - limit: 10, - }); - - const selectedObjectRecordsForSelect = useMemo( - () => - selectedObjectRecords.filter((selectedObjectRecord) => - selectedObjectRecordIds.some( - (selectedObjectRecordId) => - selectedObjectRecordId.id === - selectedObjectRecord.recordIdentifier.id, - ), - ), - [selectedObjectRecords, selectedObjectRecordIds], - ); - - return ( - <MultiRecordSelect - onChange={onChange} - onSubmit={onSubmit} - selectedObjectRecords={selectedObjectRecordsForSelect} - allRecords={[ - ...(filteredSelectedObjectRecords ?? []), - ...(objectRecordsToSelect ?? []), - ]} - loading={loading} - searchFilter={searchFilter} - setSearchFilter={setSearchFilter} - /> - ); -}; diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/components/MultipleObjectRecordSelectItem.tsx b/packages/twenty-front/src/modules/object-record/relation-picker/components/MultipleObjectRecordSelectItem.tsx index ed975e2bbe3f..af92fedd19f0 100644 --- a/packages/twenty-front/src/modules/object-record/relation-picker/components/MultipleObjectRecordSelectItem.tsx +++ b/packages/twenty-front/src/modules/object-record/relation-picker/components/MultipleObjectRecordSelectItem.tsx @@ -3,12 +3,15 @@ import { useRecoilValue } from 'recoil'; import { Avatar } from 'twenty-ui'; import { v4 } from 'uuid'; +import { useObjectRecordMultiSelectScopedStates } from '@/activities/hooks/useObjectRecordMultiSelectScopedStates'; import { MULTI_OBJECT_RECORD_SELECT_SELECTABLE_LIST_ID } from '@/object-record/relation-picker/constants/MultiObjectRecordSelectSelectableListId'; -import { ObjectRecordForSelect } from '@/object-record/relation-picker/hooks/useMultiObjectSearch'; +import { RelationPickerScopeInternalContext } from '@/object-record/relation-picker/scopes/scope-internal-context/RelationPickerScopeInternalContext'; import { SelectableItem } from '@/ui/layout/selectable-list/components/SelectableItem'; import { useSelectableList } from '@/ui/layout/selectable-list/hooks/useSelectableList'; import { MenuItemMultiSelectAvatar } from '@/ui/navigation/menu-item/components/MenuItemMultiSelectAvatar'; +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; +import { isDefined } from '~/utils/isDefined'; export const StyledSelectableItem = styled(SelectableItem)` height: 100%; @@ -16,45 +19,60 @@ export const StyledSelectableItem = styled(SelectableItem)` `; export const MultipleObjectRecordSelectItem = ({ - objectRecordForSelect, - onSelectedChange, - selected, + objectRecordId, + onChange, }: { - objectRecordForSelect: ObjectRecordForSelect; - onSelectedChange?: (selected: boolean) => void; - selected: boolean; + objectRecordId: string; + onChange?: (changedRecordForSelectId: string) => void; }) => { const { isSelectedItemIdSelector } = useSelectableList( MULTI_OBJECT_RECORD_SELECT_SELECTABLE_LIST_ID, ); const isSelectedByKeyboard = useRecoilValue( - isSelectedItemIdSelector(objectRecordForSelect.record.id), + isSelectedItemIdSelector(objectRecordId), ); + const scopeId = useAvailableScopeIdOrThrow( + RelationPickerScopeInternalContext, + ); + + const { objectRecordMultiSelectFamilyState } = + useObjectRecordMultiSelectScopedStates(scopeId); + + const record = useRecoilValue( + objectRecordMultiSelectFamilyState(objectRecordId), + ); + + if (!record) { + return null; + } + + const handleSelectChange = () => { + onChange?.(objectRecordId); + }; + + const { selected, recordIdentifier } = record; + + if (!isDefined(recordIdentifier)) { + return null; + } return ( - <StyledSelectableItem - itemId={objectRecordForSelect.record.id} - key={objectRecordForSelect.record.id + v4()} - > + <StyledSelectableItem itemId={objectRecordId} key={objectRecordId + v4()}> <MenuItemMultiSelectAvatar - selected={selected} - onSelectChange={onSelectedChange} + onSelectChange={(_isNewlySelectedValue) => handleSelectChange()} isKeySelected={isSelectedByKeyboard} + selected={selected} avatar={ <Avatar - avatarUrl={getImageAbsoluteURIOrBase64( - objectRecordForSelect.recordIdentifier.avatarUrl, - )} - entityId={objectRecordForSelect.record.id} - placeholder={objectRecordForSelect.recordIdentifier.name} + avatarUrl={getImageAbsoluteURIOrBase64(recordIdentifier.avatarUrl)} + placeholderColorSeed={objectRecordId} + placeholder={recordIdentifier.name} size="md" - type={ - objectRecordForSelect.recordIdentifier.avatarType ?? 'rounded' - } + type={recordIdentifier.avatarType ?? 'rounded'} /> } - text={objectRecordForSelect.recordIdentifier.name} + text={recordIdentifier.name} /> </StyledSelectableItem> ); diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/components/SelectableMenuItemSelect.tsx b/packages/twenty-front/src/modules/object-record/relation-picker/components/SelectableMenuItemSelect.tsx index bd6e4a2f99e7..16eb58fad271 100644 --- a/packages/twenty-front/src/modules/object-record/relation-picker/components/SelectableMenuItemSelect.tsx +++ b/packages/twenty-front/src/modules/object-record/relation-picker/components/SelectableMenuItemSelect.tsx @@ -41,7 +41,7 @@ export const SelectableMenuItemSelect = ({ avatar={ <Avatar avatarUrl={getImageAbsoluteURIOrBase64(entity.avatarUrl)} - entityId={entity.id} + placeholderColorSeed={entity.id} placeholder={entity.name} size="md" type={entity.avatarType ?? 'rounded'} diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/components/SingleEntitySelectMenuItemsWithSearch.tsx b/packages/twenty-front/src/modules/object-record/relation-picker/components/SingleEntitySelectMenuItemsWithSearch.tsx index 70c777c7bfd3..537c2fd088da 100644 --- a/packages/twenty-front/src/modules/object-record/relation-picker/components/SingleEntitySelectMenuItemsWithSearch.tsx +++ b/packages/twenty-front/src/modules/object-record/relation-picker/components/SingleEntitySelectMenuItemsWithSearch.tsx @@ -44,7 +44,6 @@ export const SingleEntitySelectMenuItemsWithSearch = ({ const { entities, relationPickerSearchFilter } = useRelationPickerEntitiesOptions({ relationObjectNameSingular, - relationPickerScopeId, selectedRelationRecordIds, excludedRelationRecordIds, }); diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/constants/TableColumnsDenyList.ts b/packages/twenty-front/src/modules/object-record/relation-picker/constants/TableColumnsDenyList.ts new file mode 100644 index 000000000000..8feab843fc81 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/relation-picker/constants/TableColumnsDenyList.ts @@ -0,0 +1,5 @@ +export const TABLE_COLUMNS_DENY_LIST = [ + 'attachments', + 'activities', + 'timelineActivities', +]; diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/hooks/__tests__/useMultiObjectSearch.test.tsx b/packages/twenty-front/src/modules/object-record/relation-picker/hooks/__tests__/useMultiObjectSearch.test.tsx index a84dfcce9497..63868f068611 100644 --- a/packages/twenty-front/src/modules/object-record/relation-picker/hooks/__tests__/useMultiObjectSearch.test.tsx +++ b/packages/twenty-front/src/modules/object-record/relation-picker/hooks/__tests__/useMultiObjectSearch.test.tsx @@ -11,7 +11,7 @@ import { FieldMetadataType } from '~/generated/graphql'; const query = gql` query CombinedFindManyRecords( $filterNameSingular: NameSingularFilterInput - $orderByNameSingular: NameSingularOrderByInput + $orderByNameSingular: [NameSingularOrderByInput] $lastCursorNameSingular: String $limitNameSingular: Int ) { @@ -50,7 +50,7 @@ const mocks = [ query, variables: { filterNameSingular: { id: { in: ['1'] } }, - orderByNameSingular: { createdAt: 'DescNullsLast' }, + orderByNameSingular: [{ createdAt: 'DescNullsLast' }], limitNameSingular: 60, }, }, @@ -63,7 +63,7 @@ const mocks = [ query, variables: { filterNameSingular: { and: [{}, { id: { in: ['1'] } }] }, - orderByNameSingular: { createdAt: 'DescNullsLast' }, + orderByNameSingular: [{ createdAt: 'DescNullsLast' }], limitNameSingular: 60, }, }, @@ -77,7 +77,7 @@ const mocks = [ variables: { limitNameSingular: 60, filterNameSingular: { not: { id: { in: ['1'] } } }, - orderByNameSingular: { createdAt: 'DescNullsLast' }, + orderByNameSingular: [{ createdAt: 'DescNullsLast' }], }, }, result: jest.fn(() => ({ diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useOrderByFieldPerMetadataItem.ts b/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useOrderByFieldPerMetadataItem.ts index 07f60b012fce..ec20862e0807 100644 --- a/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useOrderByFieldPerMetadataItem.ts +++ b/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useOrderByFieldPerMetadataItem.ts @@ -16,9 +16,7 @@ export const useOrderByFieldPerMetadataItem = ({ return [ `orderBy${capitalize(objectMetadataItem.nameSingular)}`, - { - ...orderByField, - }, + [...orderByField], ]; }) .filter(isDefined), diff --git a/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useRelationPickerEntitiesOptions.ts b/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useRelationPickerEntitiesOptions.ts index 9eb5f990c622..efb3f9934267 100644 --- a/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useRelationPickerEntitiesOptions.ts +++ b/packages/twenty-front/src/modules/object-record/relation-picker/hooks/useRelationPickerEntitiesOptions.ts @@ -1,22 +1,26 @@ import { useRecoilValue } from 'recoil'; import { useRelationPickerScopedStates } from '@/object-record/relation-picker/hooks/internal/useRelationPickerScopedStates'; +import { RelationPickerScopeInternalContext } from '@/object-record/relation-picker/scopes/scope-internal-context/RelationPickerScopeInternalContext'; import { useFilteredSearchEntityQuery } from '@/search/hooks/useFilteredSearchEntityQuery'; +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; export const useRelationPickerEntitiesOptions = ({ relationObjectNameSingular, - relationPickerScopeId = 'relation-picker', selectedRelationRecordIds = [], excludedRelationRecordIds = [], }: { relationObjectNameSingular: string; - relationPickerScopeId?: string; selectedRelationRecordIds?: string[]; excludedRelationRecordIds?: string[]; }) => { + const scopeId = useAvailableScopeIdOrThrow( + RelationPickerScopeInternalContext, + ); + const { searchQueryState, relationPickerSearchFilterState } = useRelationPickerScopedStates({ - relationPickerScopedId: relationPickerScopeId, + relationPickerScopedId: scopeId, }); const relationPickerSearchFilter = useRecoilValue( relationPickerSearchFilterState, diff --git a/packages/twenty-front/src/modules/object-record/select/components/MultipleRecordSelectDropdown.tsx b/packages/twenty-front/src/modules/object-record/select/components/MultipleRecordSelectDropdown.tsx index 455cedced398..af608e0077d5 100644 --- a/packages/twenty-front/src/modules/object-record/select/components/MultipleRecordSelectDropdown.tsx +++ b/packages/twenty-front/src/modules/object-record/select/components/MultipleRecordSelectDropdown.tsx @@ -70,7 +70,7 @@ export const MultipleRecordSelectDropdown = ({ avatar={ <Avatar avatarUrl={getImageAbsoluteURIOrBase64(record.avatarUrl)} - entityId={record.id} + placeholderColorSeed={record.id} placeholder={record.name} size="md" type={record.avatarType ?? 'rounded'} diff --git a/packages/twenty-front/src/modules/object-record/spreadsheet-import/__tests__/useSpreadsheetRecordImport.test.tsx b/packages/twenty-front/src/modules/object-record/spreadsheet-import/__tests__/useSpreadsheetRecordImport.test.tsx index a925b36c8c74..33f43785b2b2 100644 --- a/packages/twenty-front/src/modules/object-record/spreadsheet-import/__tests__/useSpreadsheetRecordImport.test.tsx +++ b/packages/twenty-front/src/modules/object-record/spreadsheet-import/__tests__/useSpreadsheetRecordImport.test.tsx @@ -20,8 +20,11 @@ const companyMocks = [ { request: { query: gql` - mutation CreateCompanies($data: [CompanyCreateInput!]!) { - createCompanies(data: $data) { + mutation CreateCompanies( + $data: [CompanyCreateInput!]! + $upsert: Boolean + ) { + createCompanies(data: $data, upsert: $upsert) { __typename xLink { label @@ -37,7 +40,16 @@ const companyMocks = [ currencyCode } createdAt - address + address { + addressStreet1 + addressStreet2 + addressCity + addressState + addressCountry + addressPostcode + addressLat + addressLng + } updatedAt name accountOwnerId @@ -58,6 +70,7 @@ const companyMocks = [ id: companyId, }, ], + upsert: true, }, }, result: jest.fn(() => ({ diff --git a/packages/twenty-front/src/modules/object-record/spreadsheet-import/useSpreadsheetRecordImport.ts b/packages/twenty-front/src/modules/object-record/spreadsheet-import/useSpreadsheetRecordImport.ts index 700cc44622a0..28516c7939e6 100644 --- a/packages/twenty-front/src/modules/object-record/spreadsheet-import/useSpreadsheetRecordImport.ts +++ b/packages/twenty-front/src/modules/object-record/spreadsheet-import/useSpreadsheetRecordImport.ts @@ -1,11 +1,11 @@ import { isNonEmptyString } from '@sniptt/guards'; -import { IconComponent, useIcons } from 'twenty-ui'; +import { useIcons } from 'twenty-ui'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; import { useCreateManyRecords } from '@/object-record/hooks/useCreateManyRecords'; import { getSpreadSheetValidation } from '@/object-record/spreadsheet-import/util/getSpreadSheetValidation'; import { useSpreadsheetImport } from '@/spreadsheet-import/hooks/useSpreadsheetImport'; -import { SpreadsheetOptions, Validation } from '@/spreadsheet-import/types'; +import { Field, SpreadsheetOptions } from '@/spreadsheet-import/types'; import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; import { FieldMetadataType } from '~/generated-metadata/graphql'; @@ -26,21 +26,13 @@ export const useSpreadsheetRecordImport = (objectNameSingular: string) => { .filter( (x) => x.isActive && - !x.isSystem && + (!x.isSystem || x.name === 'id') && x.name !== 'createdAt' && (x.type !== FieldMetadataType.Relation || x.toRelationMetadata), ) .sort((a, b) => a.name.localeCompare(b.name)); - const templateFields: { - icon: IconComponent; - label: string; - key: string; - fieldType: { - type: 'input' | 'checkbox'; - }; - validations?: Validation[]; - }[] = []; + const templateFields: Field<string>[] = []; for (const field of fields) { if (field.type === FieldMetadataType.FullName) { templateFields.push({ @@ -80,6 +72,34 @@ export const useSpreadsheetRecordImport = (objectNameSingular: string) => { field.label + ' (ID)', ), }); + } else if (field.type === FieldMetadataType.Select) { + templateFields.push({ + icon: getIcon(field.icon), + label: field.label, + key: field.name, + fieldType: { + type: 'select', + options: + field.options?.map((option) => ({ + label: option.label, + value: option.value, + })) || [], + }, + validations: getSpreadSheetValidation( + field.type, + field.label + ' (ID)', + ), + }); + } else if (field.type === FieldMetadataType.Boolean) { + templateFields.push({ + icon: getIcon(field.icon), + label: field.label, + key: field.name, + fieldType: { + type: 'checkbox', + }, + validations: getSpreadSheetValidation(field.type, field.label), + }); } else { templateFields.push({ icon: getIcon(field.icon), @@ -110,11 +130,15 @@ export const useSpreadsheetRecordImport = (objectNameSingular: string) => { switch (field.type) { case FieldMetadataType.Boolean: - fieldMapping[field.name] = value === 'true' || value === true; + if (value !== undefined) { + fieldMapping[field.name] = value === 'true' || value === true; + } break; case FieldMetadataType.Number: case FieldMetadataType.Numeric: - fieldMapping[field.name] = Number(value); + if (value !== undefined) { + fieldMapping[field.name] = Number(value); + } break; case FieldMetadataType.Currency: if (value !== undefined) { @@ -154,14 +178,16 @@ export const useSpreadsheetRecordImport = (objectNameSingular: string) => { } break; default: - fieldMapping[field.name] = value; + if (value !== undefined) { + fieldMapping[field.name] = value; + } break; } } return fieldMapping; }); try { - await createManyRecords(createInputs); + await createManyRecords(createInputs, true); } catch (error: any) { enqueueSnackBar(error?.message || 'Something went wrong', { variant: SnackBarVariant.Error, diff --git a/packages/twenty-front/src/modules/object-record/states/isRecordBoardColumnLoadingFamilyState.ts b/packages/twenty-front/src/modules/object-record/states/isRecordBoardColumnLoadingFamilyState.ts new file mode 100644 index 000000000000..8804ff588ba1 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/states/isRecordBoardColumnLoadingFamilyState.ts @@ -0,0 +1,9 @@ +import { createFamilyState } from '@/ui/utilities/state/utils/createFamilyState'; + +export const isRecordIndexBoardColumnLoadingFamilyState = createFamilyState< + boolean, + string | undefined +>({ + key: 'isRecordIndexBoardColumnLoadingFamilyState', + defaultValue: false, +}); diff --git a/packages/twenty-front/src/modules/object-record/types/OnFindManyRecordsCompleted.ts b/packages/twenty-front/src/modules/object-record/types/OnFindManyRecordsCompleted.ts new file mode 100644 index 000000000000..97003380167d --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/types/OnFindManyRecordsCompleted.ts @@ -0,0 +1,9 @@ +import { RecordGqlConnection } from '@/object-record/graphql/types/RecordGqlConnection'; + +export type OnFindManyRecordsCompleted<T> = ( + records: T[], + options?: { + pageInfo?: RecordGqlConnection['pageInfo']; + totalCount?: number; + }, +) => void; diff --git a/packages/twenty-front/src/modules/object-record/types/UseFindManyRecordsParams.ts b/packages/twenty-front/src/modules/object-record/types/UseFindManyRecordsParams.ts new file mode 100644 index 000000000000..1a15c6040ecc --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/types/UseFindManyRecordsParams.ts @@ -0,0 +1,14 @@ +import { WatchQueryFetchPolicy } from '@apollo/client'; + +import { ObjectMetadataItemIdentifier } from '@/object-metadata/types/ObjectMetadataItemIdentifier'; +import { RecordGqlOperationGqlRecordFields } from '@/object-record/graphql/types/RecordGqlOperationGqlRecordFields'; +import { RecordGqlOperationVariables } from '@/object-record/graphql/types/RecordGqlOperationVariables'; +import { OnFindManyRecordsCompleted } from '@/object-record/types/OnFindManyRecordsCompleted'; + +export type UseFindManyRecordsParams<T> = ObjectMetadataItemIdentifier & + RecordGqlOperationVariables & { + onCompleted?: OnFindManyRecordsCompleted<T>; + skip?: boolean; + recordGqlFields?: RecordGqlOperationGqlRecordFields; + fetchPolicy?: WatchQueryFetchPolicy; + }; diff --git a/packages/twenty-front/src/modules/object-record/utils/computeRecordBoardColumnDefinitionsFromObjectMetadata.ts b/packages/twenty-front/src/modules/object-record/utils/computeRecordBoardColumnDefinitionsFromObjectMetadata.ts index 4df6d4afd655..4abbba702a98 100644 --- a/packages/twenty-front/src/modules/object-record/utils/computeRecordBoardColumnDefinitionsFromObjectMetadata.ts +++ b/packages/twenty-front/src/modules/object-record/utils/computeRecordBoardColumnDefinitionsFromObjectMetadata.ts @@ -1,7 +1,12 @@ import { IconPencil } from 'twenty-ui'; import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; -import { RecordBoardColumnDefinition } from '@/object-record/record-board/types/RecordBoardColumnDefinition'; +import { + RecordBoardColumnDefinition, + RecordBoardColumnDefinitionNoValue, + RecordBoardColumnDefinitionType, + RecordBoardColumnDefinitionValue, +} from '@/object-record/record-board/types/RecordBoardColumnDefinition'; import { FieldMetadataType } from '~/generated-metadata/graphql'; export const computeRecordBoardColumnDefinitionsFromObjectMetadata = ( @@ -25,20 +30,42 @@ export const computeRecordBoardColumnDefinitionsFromObjectMetadata = ( ); } - return selectFieldMetadataItem.options.map((selectOption) => ({ - id: selectOption.id, - title: selectOption.label, - value: selectOption.value, - color: selectOption.color, - position: selectOption.position, - actions: [ - { - id: 'edit', - label: 'Edit from settings', - icon: IconPencil, - position: 0, - callback: navigateToSelectSettings, - }, - ], - })); + const valueColumns = selectFieldMetadataItem.options.map( + (selectOption) => + ({ + id: selectOption.id, + type: RecordBoardColumnDefinitionType.Value, + title: selectOption.label, + value: selectOption.value, + color: selectOption.color, + position: selectOption.position, + actions: [ + { + id: 'edit', + label: 'Edit from settings', + icon: IconPencil, + position: 0, + callback: navigateToSelectSettings, + }, + ], + }) satisfies RecordBoardColumnDefinitionValue, + ); + + const noValueColumn = { + id: 'no-value', + title: 'No Value', + type: RecordBoardColumnDefinitionType.NoValue, + value: null, + actions: [], + position: + selectFieldMetadataItem.options + .map((option) => option.position) + .reduce((a, b) => Math.max(a, b), 0) + 1, + } satisfies RecordBoardColumnDefinitionNoValue; + + if (selectFieldMetadataItem.isNullable === true) { + return [...valueColumns, noValueColumn]; + } + + return valueColumns; }; diff --git a/packages/twenty-front/src/modules/object-record/utils/filterAvailableTableColumns.ts b/packages/twenty-front/src/modules/object-record/utils/filterAvailableTableColumns.ts index bb7cc3578b3d..eadb16fd47c8 100644 --- a/packages/twenty-front/src/modules/object-record/utils/filterAvailableTableColumns.ts +++ b/packages/twenty-front/src/modules/object-record/utils/filterAvailableTableColumns.ts @@ -1,17 +1,23 @@ import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; import { isFieldRelation } from '@/object-record/record-field/types/guards/isFieldRelation'; import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; +import { TABLE_COLUMNS_DENY_LIST } from '@/object-record/relation-picker/constants/TableColumnsDenyList'; export const filterAvailableTableColumns = ( columnDefinition: ColumnDefinition<FieldMetadata>, ): boolean => { if ( isFieldRelation(columnDefinition) && - columnDefinition.metadata?.relationType !== 'TO_ONE_OBJECT' + columnDefinition.metadata?.relationType !== 'TO_ONE_OBJECT' && + columnDefinition.metadata?.relationType !== 'FROM_MANY_OBJECTS' ) { return false; } + if (TABLE_COLUMNS_DENY_LIST.includes(columnDefinition.metadata.fieldName)) { + return false; + } + if (columnDefinition.type === 'UUID') { return false; } diff --git a/packages/twenty-front/src/modules/object-record/utils/generateFindManyRecordsQuery.ts b/packages/twenty-front/src/modules/object-record/utils/generateFindManyRecordsQuery.ts index ea68c231cbad..a62ae2c0d77f 100644 --- a/packages/twenty-front/src/modules/object-record/utils/generateFindManyRecordsQuery.ts +++ b/packages/twenty-front/src/modules/object-record/utils/generateFindManyRecordsQuery.ts @@ -1,32 +1,37 @@ import gql from 'graphql-tag'; import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; -import { isAggregationEnabled } from '@/object-metadata/utils/isAggregationEnabled'; import { mapObjectMetadataToGraphQLQuery } from '@/object-metadata/utils/mapObjectMetadataToGraphQLQuery'; import { RecordGqlOperationGqlRecordFields } from '@/object-record/graphql/types/RecordGqlOperationGqlRecordFields'; import { capitalize } from '~/utils/string/capitalize'; +export type QueryCursorDirection = 'before' | 'after'; + export const generateFindManyRecordsQuery = ({ objectMetadataItem, objectMetadataItems, recordGqlFields, computeReferences, + cursorDirection, }: { objectMetadataItem: ObjectMetadataItem; objectMetadataItems: ObjectMetadataItem[]; recordGqlFields?: RecordGqlOperationGqlRecordFields; computeReferences?: boolean; + cursorDirection?: QueryCursorDirection; }) => gql` query FindMany${capitalize( objectMetadataItem.namePlural, )}($filter: ${capitalize( objectMetadataItem.nameSingular, -)}FilterInput, $orderBy: ${capitalize( +)}FilterInput, $orderBy: [${capitalize( objectMetadataItem.nameSingular, -)}OrderByInput, $lastCursor: String, $limit: Int) { - ${ - objectMetadataItem.namePlural - }(filter: $filter, orderBy: $orderBy, first: $limit, after: $lastCursor){ +)}OrderByInput], $lastCursor: String, $limit: Int) { + ${objectMetadataItem.namePlural}(filter: $filter, orderBy: $orderBy, ${ + cursorDirection === 'before' + ? 'last: $limit, before: $lastCursor' + : 'first: $limit, after: $lastCursor' + } ){ edges { node ${mapObjectMetadataToGraphQLQuery({ objectMetadataItems, @@ -37,11 +42,12 @@ query FindMany${capitalize( cursor } pageInfo { - ${isAggregationEnabled(objectMetadataItem) ? 'hasNextPage' : ''} + hasNextPage + hasPreviousPage startCursor endCursor } - ${isAggregationEnabled(objectMetadataItem) ? 'totalCount' : ''} + totalCount } } `; diff --git a/packages/twenty-front/src/modules/object-record/utils/getQueryIdentifier.ts b/packages/twenty-front/src/modules/object-record/utils/getQueryIdentifier.ts new file mode 100644 index 000000000000..c43f7807a451 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/utils/getQueryIdentifier.ts @@ -0,0 +1,16 @@ +import { RecordGqlOperationVariables } from '@/object-record/graphql/types/RecordGqlOperationVariables'; + +export const getQueryIdentifier = ({ + objectNameSingular, + filter, + orderBy, + limit, + cursorFilter, +}: RecordGqlOperationVariables & { + objectNameSingular: string; +}) => + objectNameSingular + + JSON.stringify(filter) + + JSON.stringify(orderBy) + + limit + + (cursorFilter ? JSON.stringify(cursorFilter) : undefined); diff --git a/packages/twenty-front/src/modules/object-record/utils/getRecordChipGeneratorPerObjectPerField.ts b/packages/twenty-front/src/modules/object-record/utils/getRecordChipGeneratorPerObjectPerField.ts deleted file mode 100644 index f127e275ec39..000000000000 --- a/packages/twenty-front/src/modules/object-record/utils/getRecordChipGeneratorPerObjectPerField.ts +++ /dev/null @@ -1,116 +0,0 @@ -import { ChipGeneratorPerObjectPerField } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; -import { FieldMetadataItem } from '@/object-metadata/types/FieldMetadataItem'; -import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; -import { getAvatarType } from '@/object-metadata/utils/getAvatarType'; -import { getAvatarUrl } from '@/object-metadata/utils/getAvatarUrl'; -import { getLabelIdentifierFieldMetadataItem } from '@/object-metadata/utils/getLabelIdentifierFieldMetadataItem'; -import { getLabelIdentifierFieldValue } from '@/object-metadata/utils/getLabelIdentifierFieldValue'; -import { getLinkToShowPage } from '@/object-metadata/utils/getLinkToShowPage'; -import { isLabelIdentifierField } from '@/object-metadata/utils/isLabelIdentifierField'; -import { isFieldFullName } from '@/object-record/record-field/types/guards/isFieldFullName'; -import { isFieldNumber } from '@/object-record/record-field/types/guards/isFieldNumber'; -import { isFieldRelation } from '@/object-record/record-field/types/guards/isFieldRelation'; -import { isFieldText } from '@/object-record/record-field/types/guards/isFieldText'; -import { RecordChipData } from '@/object-record/record-field/types/RecordChipData'; -import { ObjectRecord } from '@/object-record/types/ObjectRecord'; -import { FieldMetadataType } from '~/generated-metadata/graphql'; -import { isDefined } from '~/utils/isDefined'; - -export const isFieldChipDisplay = ( - field: Pick<FieldMetadataItem, 'type'>, - isLabelIdentifier: boolean, -) => - isLabelIdentifier && - (isFieldText(field) || isFieldFullName(field) || isFieldNumber(field)); - -export const getRecordChipGeneratorPerObjectPerField = ( - objectMetadataItems: ObjectMetadataItem[], -) => { - const recordChipGeneratorPerObjectPerField: ChipGeneratorPerObjectPerField = - {}; - - for (const objectMetadataItem of objectMetadataItems) { - const generatorPerField = Object.fromEntries< - (record: ObjectRecord) => RecordChipData - >( - objectMetadataItem.fields - .filter( - (fieldMetadataItem) => - isLabelIdentifierField({ - fieldMetadataItem: fieldMetadataItem, - objectMetadataItem, - }) || - fieldMetadataItem.type === FieldMetadataType.Relation || - isFieldChipDisplay( - fieldMetadataItem, - isLabelIdentifierField({ - fieldMetadataItem: fieldMetadataItem, - objectMetadataItem, - }), - ), - ) - .map((fieldMetadataItem) => { - const objectNameSingularToFind = isLabelIdentifierField({ - fieldMetadataItem: fieldMetadataItem, - objectMetadataItem: objectMetadataItem, - }) - ? objectMetadataItem.nameSingular - : isFieldRelation(fieldMetadataItem) - ? fieldMetadataItem.relationDefinition?.targetObjectMetadata - .nameSingular ?? undefined - : undefined; - - const objectMetadataItemToUse = objectMetadataItems.find( - (objectMetadataItem) => - objectMetadataItem.nameSingular === objectNameSingularToFind, - ); - - if ( - !isDefined(objectMetadataItemToUse) || - !isDefined(objectNameSingularToFind) - ) { - return ['', () => ({}) as any]; - } - - const labelIdentifierFieldMetadataItem = - getLabelIdentifierFieldMetadataItem(objectMetadataItemToUse); - - const imageIdentifierFieldMetadata = - objectMetadataItemToUse.fields.find( - (field) => - field.id === - objectMetadataItemToUse.imageIdentifierFieldMetadataId, - ); - - const avatarType = getAvatarType(objectNameSingularToFind); - - return [ - fieldMetadataItem.name, - (record: ObjectRecord) => ({ - recordId: record.id, - name: getLabelIdentifierFieldValue( - record, - labelIdentifierFieldMetadataItem, - objectMetadataItemToUse.nameSingular, - ), - avatarUrl: getAvatarUrl( - objectMetadataItemToUse.nameSingular, - record, - imageIdentifierFieldMetadata, - ), - avatarType, - linkToShowPage: getLinkToShowPage( - objectMetadataItemToUse.nameSingular, - record, - ), - }), - ]; - }), - ); - - recordChipGeneratorPerObjectPerField[objectMetadataItem.nameSingular] = - generatorPerField; - } - - return recordChipGeneratorPerObjectPerField; -}; diff --git a/packages/twenty-front/src/modules/object-record/utils/getRecordChipGenerators.ts b/packages/twenty-front/src/modules/object-record/utils/getRecordChipGenerators.ts new file mode 100644 index 000000000000..c80fe1b76312 --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/utils/getRecordChipGenerators.ts @@ -0,0 +1,120 @@ +import { + ChipGeneratorPerObjectNameSingularPerFieldName, + IdentifierChipGeneratorPerObject, +} from '@/object-metadata/context/PreComputedChipGeneratorsContext'; +import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; +import { getAvatarType } from '@/object-metadata/utils/getAvatarType'; +import { getAvatarUrl } from '@/object-metadata/utils/getAvatarUrl'; +import { getLabelIdentifierFieldMetadataItem } from '@/object-metadata/utils/getLabelIdentifierFieldMetadataItem'; +import { getLabelIdentifierFieldValue } from '@/object-metadata/utils/getLabelIdentifierFieldValue'; +import { isLabelIdentifierField } from '@/object-metadata/utils/isLabelIdentifierField'; +import { isFieldChipDisplay } from '@/object-record/record-field/meta-types/display/utils/isFieldChipDisplay'; +import { RecordChipData } from '@/object-record/record-field/types/RecordChipData'; +import { ObjectRecord } from '@/object-record/types/ObjectRecord'; +import { FieldMetadataType } from '~/generated-metadata/graphql'; +import { isDefined } from '~/utils/isDefined'; + +export const getRecordChipGenerators = ( + objectMetadataItems: ObjectMetadataItem[], +) => { + const chipGeneratorPerObjectPerField: ChipGeneratorPerObjectNameSingularPerFieldName = + {}; + + const identifierChipGeneratorPerObject: IdentifierChipGeneratorPerObject = {}; + + for (const objectMetadataItem of objectMetadataItems) { + const labelIdentifierFieldMetadataItem = + getLabelIdentifierFieldMetadataItem(objectMetadataItem); + + const generatorPerField = Object.fromEntries< + (record: ObjectRecord) => RecordChipData + >( + objectMetadataItem.fields + .filter( + (fieldMetadataItem) => + labelIdentifierFieldMetadataItem?.id === fieldMetadataItem.id || + fieldMetadataItem.type === FieldMetadataType.Relation || + isFieldChipDisplay( + fieldMetadataItem, + isLabelIdentifierField({ + fieldMetadataItem: fieldMetadataItem, + objectMetadataItem, + }), + ), + ) + .map((fieldMetadataItem) => { + const isLabelIdentifier = + labelIdentifierFieldMetadataItem?.id === fieldMetadataItem.id; + + const currentObjectNameSingular = objectMetadataItem.nameSingular; + const fieldRelationObjectNameSingular = + fieldMetadataItem.relationDefinition?.targetObjectMetadata + .nameSingular ?? undefined; + + const objectNameSingularToFind = isLabelIdentifier + ? currentObjectNameSingular + : fieldRelationObjectNameSingular; + + const objectMetadataItemToUse = objectMetadataItems.find( + (objectMetadataItem) => + objectMetadataItem.nameSingular === objectNameSingularToFind, + ); + + if ( + !isDefined(objectMetadataItemToUse) || + !isDefined(objectNameSingularToFind) + ) { + return ['', () => ({}) as any]; + } + + const labelIdentifierFieldMetadataItemToUse = + getLabelIdentifierFieldMetadataItem(objectMetadataItemToUse); + + const imageIdentifierFieldMetadataToUse = + objectMetadataItemToUse.fields.find( + (field) => + field.id === + objectMetadataItemToUse.imageIdentifierFieldMetadataId, + ); + + const avatarType = getAvatarType(objectNameSingularToFind); + + return [ + fieldMetadataItem.name, + (record: ObjectRecord) => + ({ + recordId: record.id, + name: getLabelIdentifierFieldValue( + record, + labelIdentifierFieldMetadataItemToUse, + objectMetadataItemToUse.nameSingular, + ), + avatarUrl: getAvatarUrl( + objectMetadataItemToUse.nameSingular, + record, + imageIdentifierFieldMetadataToUse, + ), + avatarType, + isLabelIdentifier, + objectNameSingular: objectNameSingularToFind, + }) satisfies RecordChipData, + ]; + }), + ); + + chipGeneratorPerObjectPerField[objectMetadataItem.nameSingular] = + generatorPerField; + + if (isDefined(labelIdentifierFieldMetadataItem)) { + identifierChipGeneratorPerObject[objectMetadataItem.nameSingular] = + chipGeneratorPerObjectPerField[objectMetadataItem.nameSingular]?.[ + labelIdentifierFieldMetadataItem.name + ]; + } + } + + return { + chipGeneratorPerObjectPerField, + identifierChipGeneratorPerObject, + }; +}; diff --git a/packages/twenty-front/src/modules/object-record/utils/sanitizeRecordInput.ts b/packages/twenty-front/src/modules/object-record/utils/sanitizeRecordInput.ts index fce402359687..314fa1e9153e 100644 --- a/packages/twenty-front/src/modules/object-record/utils/sanitizeRecordInput.ts +++ b/packages/twenty-front/src/modules/object-record/utils/sanitizeRecordInput.ts @@ -2,8 +2,7 @@ import { isString } from '@sniptt/guards'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; -import { isFieldRelationValue } from '@/object-record/record-field/types/guards/isFieldRelationValue'; -import { EntityForSelect } from '@/object-record/relation-picker/types/EntityForSelect'; +import { isFieldRelationToOneValue } from '@/object-record/record-field/types/guards/isFieldRelationToOneValue'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; import { FieldMetadataType } from '~/generated/graphql'; import { isDefined } from '~/utils/isDefined'; @@ -31,7 +30,7 @@ export const sanitizeRecordInput = ({ if ( fieldMetadataItem.type === FieldMetadataType.Relation && - isFieldRelationValue<EntityForSelect>(fieldValue) + isFieldRelationToOneValue(fieldValue) ) { const relationIdFieldName = `${fieldMetadataItem.name}Id`; const relationIdFieldMetadataItem = objectMetadataItem.fields.find( diff --git a/packages/twenty-front/src/modules/onboarding/components/OnboardingSyncEmailsSettingsCard.tsx b/packages/twenty-front/src/modules/onboarding/components/OnboardingSyncEmailsSettingsCard.tsx index 61605ec00389..2a01962fdfe1 100644 --- a/packages/twenty-front/src/modules/onboarding/components/OnboardingSyncEmailsSettingsCard.tsx +++ b/packages/twenty-front/src/modules/onboarding/components/OnboardingSyncEmailsSettingsCard.tsx @@ -12,6 +12,7 @@ export const OnboardingSyncEmailsSettingsCard = ({ value = MessageChannelVisibility.ShareEverything, }: OnboardingSyncEmailsSettingsCardProps) => ( <SettingsAccountsRadioSettingsCard + name="sync-emails-visiblity" options={onboardingSyncEmailsOptions} value={value} onChange={onChange} diff --git a/packages/twenty-front/src/modules/onboarding/components/onboardingSyncEmailsOptions.tsx b/packages/twenty-front/src/modules/onboarding/components/onboardingSyncEmailsOptions.tsx index 71e53d472f0f..121408787316 100644 --- a/packages/twenty-front/src/modules/onboarding/components/onboardingSyncEmailsOptions.tsx +++ b/packages/twenty-front/src/modules/onboarding/components/onboardingSyncEmailsOptions.tsx @@ -1,9 +1,9 @@ +import { SettingsAccountsVisibilityIcon } from '@/settings/accounts/components/SettingsAccountsVisibilityIcon'; import styled from '@emotion/styled'; -import { SettingsAccountsVisibilitySettingCardMedia } from '@/settings/accounts/components/SettingsAccountsVisibilitySettingCardMedia'; import { MessageChannelVisibility } from '~/generated/graphql'; -const StyledCardMedia = styled(SettingsAccountsVisibilitySettingCardMedia)` +const StyledCardMedia = styled(SettingsAccountsVisibilityIcon)` width: ${({ theme }) => theme.spacing(10)}; `; diff --git a/packages/twenty-front/src/modules/onboarding/hooks/__tests__/useOnboardingStatus.test.ts b/packages/twenty-front/src/modules/onboarding/hooks/__tests__/useOnboardingStatus.test.ts new file mode 100644 index 000000000000..3aaf6449502f --- /dev/null +++ b/packages/twenty-front/src/modules/onboarding/hooks/__tests__/useOnboardingStatus.test.ts @@ -0,0 +1,61 @@ +import { act } from 'react-dom/test-utils'; +import { renderHook } from '@testing-library/react'; +import { RecoilRoot, useSetRecoilState } from 'recoil'; + +import { CurrentUser, currentUserState } from '@/auth/states/currentUserState'; +import { tokenPairState } from '@/auth/states/tokenPairState'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; +import { OnboardingStatus } from '~/generated/graphql'; + +const tokenPair = { + accessToken: { token: 'accessToken', expiresAt: 'expiresAt' }, + refreshToken: { token: 'refreshToken', expiresAt: 'expiresAt' }, +}; +const currentUser = { + id: '1', + onboardingStatus: null, +} as CurrentUser; + +const renderHooks = () => { + const { result } = renderHook( + () => { + const onboardingStatus = useOnboardingStatus(); + const setCurrentUser = useSetRecoilState(currentUserState); + const setTokenPair = useSetRecoilState(tokenPairState); + + return { + onboardingStatus, + setCurrentUser, + setTokenPair, + }; + }, + { + wrapper: RecoilRoot, + }, + ); + return { result }; +}; + +describe('useOnboardingStatus', () => { + it(`should return "undefined" when user is not logged in`, async () => { + const { result } = renderHooks(); + expect(result.current.onboardingStatus).toBe(undefined); + }); + + Object.values(OnboardingStatus).forEach((onboardingStatus) => { + it(`should return "${onboardingStatus}"`, async () => { + const { result } = renderHooks(); + const { setTokenPair, setCurrentUser } = result.current; + + act(() => { + setTokenPair(tokenPair); + setCurrentUser({ + ...currentUser, + onboardingStatus, + }); + }); + + expect(result.current.onboardingStatus).toBe(onboardingStatus); + }); + }); +}); diff --git a/packages/twenty-front/src/modules/onboarding/hooks/__tests__/useSetNextOnboardingStatus.test.ts b/packages/twenty-front/src/modules/onboarding/hooks/__tests__/useSetNextOnboardingStatus.test.ts new file mode 100644 index 000000000000..260e3d6aa48a --- /dev/null +++ b/packages/twenty-front/src/modules/onboarding/hooks/__tests__/useSetNextOnboardingStatus.test.ts @@ -0,0 +1,87 @@ +import { act, renderHook } from '@testing-library/react'; +import { RecoilRoot, useRecoilState, useSetRecoilState } from 'recoil'; +import { v4 } from 'uuid'; + +import { currentUserState } from '@/auth/states/currentUserState'; +import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; +import { useSetNextOnboardingStatus } from '@/onboarding/hooks/useSetNextOnboardingStatus'; +import { OnboardingStatus, SubscriptionStatus } from '~/generated/graphql'; +import { + mockDefaultWorkspace, + mockedUserData, +} from '~/testing/mock-data/users'; + +const renderHooks = ( + onboardingStatus: OnboardingStatus, + withCurrentBillingSubscription: boolean, + withOneWorkspaceMember = true, +) => { + const { result } = renderHook( + () => { + const [currentUser, setCurrentUser] = useRecoilState(currentUserState); + const setCurrentWorkspace = useSetRecoilState(currentWorkspaceState); + const setNextOnboardingStatus = useSetNextOnboardingStatus(); + return { + currentUser, + setCurrentUser, + setCurrentWorkspace, + setNextOnboardingStatus, + }; + }, + { + wrapper: RecoilRoot, + }, + ); + act(() => { + result.current.setCurrentUser({ ...mockedUserData, onboardingStatus }); + result.current.setCurrentWorkspace({ + ...mockDefaultWorkspace, + currentBillingSubscription: withCurrentBillingSubscription + ? { id: v4(), status: SubscriptionStatus.Active } + : undefined, + workspaceMembersCount: withOneWorkspaceMember ? 1 : 2, + }); + }); + act(() => { + result.current.setNextOnboardingStatus(); + }); + return result.current.currentUser?.onboardingStatus; +}; + +describe('useSetNextOnboardingStatus', () => { + it('should set next onboarding status for ProfileCreation', () => { + const nextOnboardingStatus = renderHooks( + OnboardingStatus.ProfileCreation, + false, + true, + ); + expect(nextOnboardingStatus).toEqual(OnboardingStatus.SyncEmail); + }); + + it('should set next onboarding status for SyncEmail', () => { + const nextOnboardingStatus = renderHooks( + OnboardingStatus.SyncEmail, + false, + true, + ); + expect(nextOnboardingStatus).toEqual(OnboardingStatus.InviteTeam); + }); + + it('should skip invite when more than 1 workspaceMember exist', () => { + const nextOnboardingStatus = renderHooks( + OnboardingStatus.SyncEmail, + true, + false, + ); + expect(nextOnboardingStatus).toEqual(OnboardingStatus.Completed); + }); + + it('should set next onboarding status for Completed', () => { + const nextOnboardingStatus = renderHooks( + OnboardingStatus.InviteTeam, + true, + true, + ); + expect(nextOnboardingStatus).toEqual(OnboardingStatus.Completed); + }); +}); diff --git a/packages/twenty-front/src/modules/onboarding/hooks/useOnboardingStatus.ts b/packages/twenty-front/src/modules/onboarding/hooks/useOnboardingStatus.ts new file mode 100644 index 000000000000..28bef7fd4a58 --- /dev/null +++ b/packages/twenty-front/src/modules/onboarding/hooks/useOnboardingStatus.ts @@ -0,0 +1,9 @@ +import { useRecoilValue } from 'recoil'; + +import { currentUserState } from '@/auth/states/currentUserState'; +import { OnboardingStatus } from '~/generated/graphql'; + +export const useOnboardingStatus = (): OnboardingStatus | null | undefined => { + const currentUser = useRecoilValue(currentUserState); + return currentUser?.onboardingStatus; +}; diff --git a/packages/twenty-front/src/modules/onboarding/hooks/useSetNextOnboardingStatus.ts b/packages/twenty-front/src/modules/onboarding/hooks/useSetNextOnboardingStatus.ts new file mode 100644 index 000000000000..abc0a2b31810 --- /dev/null +++ b/packages/twenty-front/src/modules/onboarding/hooks/useSetNextOnboardingStatus.ts @@ -0,0 +1,50 @@ +import { useRecoilCallback, useRecoilValue } from 'recoil'; + +import { CurrentUser, currentUserState } from '@/auth/states/currentUserState'; +import { + CurrentWorkspace, + currentWorkspaceState, +} from '@/auth/states/currentWorkspaceState'; +import { OnboardingStatus } from '~/generated/graphql'; +import { isDefined } from '~/utils/isDefined'; + +const getNextOnboardingStatus = ( + currentUser: CurrentUser | null, + currentWorkspace: CurrentWorkspace | null, +) => { + if (currentUser?.onboardingStatus === OnboardingStatus.ProfileCreation) { + return OnboardingStatus.SyncEmail; + } + if ( + currentUser?.onboardingStatus === OnboardingStatus.SyncEmail && + currentWorkspace?.workspaceMembersCount === 1 + ) { + return OnboardingStatus.InviteTeam; + } + return OnboardingStatus.Completed; +}; + +export const useSetNextOnboardingStatus = () => { + const currentUser = useRecoilValue(currentUserState); + const currentWorkspace = useRecoilValue(currentWorkspaceState); + + return useRecoilCallback( + ({ set }) => + () => { + const nextOnboardingStatus = getNextOnboardingStatus( + currentUser, + currentWorkspace, + ); + set(currentUserState, (current) => { + if (isDefined(current)) { + return { + ...current, + onboardingStatus: nextOnboardingStatus, + }; + } + return current; + }); + }, + [currentWorkspace, currentUser], + ); +}; diff --git a/packages/twenty-front/src/modules/onboarding/hooks/useSetNextOnboardingStep.ts b/packages/twenty-front/src/modules/onboarding/hooks/useSetNextOnboardingStep.ts deleted file mode 100644 index 42acf8feb0ac..000000000000 --- a/packages/twenty-front/src/modules/onboarding/hooks/useSetNextOnboardingStep.ts +++ /dev/null @@ -1,41 +0,0 @@ -import { useRecoilCallback, useSetRecoilState } from 'recoil'; - -import { currentUserState } from '@/auth/states/currentUserState'; -import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; -import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; -import { WorkspaceMember } from '@/workspace-member/types/WorkspaceMember'; -import { OnboardingStep } from '~/generated/graphql'; - -const getNextOnboardingStep = ( - currentOnboardingStep: OnboardingStep, - workspaceMembers: WorkspaceMember[], -) => { - if (currentOnboardingStep === OnboardingStep.SyncEmail) { - return workspaceMembers && workspaceMembers.length > 1 - ? null - : OnboardingStep.InviteTeam; - } - return null; -}; - -export const useSetNextOnboardingStep = () => { - const setCurrentUser = useSetRecoilState(currentUserState); - const { records: workspaceMembers } = useFindManyRecords<WorkspaceMember>({ - objectNameSingular: CoreObjectNameSingular.WorkspaceMember, - }); - return useRecoilCallback( - () => (currentOnboardingStep: OnboardingStep) => { - setCurrentUser( - (current) => - ({ - ...current, - onboardingStep: getNextOnboardingStep( - currentOnboardingStep, - workspaceMembers, - ), - }) as any, - ); - }, - [setCurrentUser, workspaceMembers], - ); -}; diff --git a/packages/twenty-front/src/modules/search/hooks/__mocks__/useFilteredSearchEntityQuery.ts b/packages/twenty-front/src/modules/search/hooks/__mocks__/useFilteredSearchEntityQuery.ts index d8a0481242c5..34aaee597da5 100644 --- a/packages/twenty-front/src/modules/search/hooks/__mocks__/useFilteredSearchEntityQuery.ts +++ b/packages/twenty-front/src/modules/search/hooks/__mocks__/useFilteredSearchEntityQuery.ts @@ -3,7 +3,7 @@ import { gql } from '@apollo/client'; export const query = gql` query FindManyPeople( $filter: PersonFilterInput - $orderBy: PersonOrderByInput + $orderBy: [PersonOrderByInput] $lastCursor: String $limit: Int = 60 ) { @@ -25,7 +25,6 @@ export const query = gql` updatedAt companyId stage - probability closeDate amount { amountMicros @@ -50,7 +49,6 @@ export const query = gql` updatedAt companyId stage - probability closeDate amount { amountMicros @@ -78,7 +76,9 @@ export const query = gql` currencyCode } createdAt - address + address { + adressCity + } updatedAt name accountOwnerId @@ -166,7 +166,7 @@ export const variables = { { not: { id: { in: ['1', '2'] } } }, ], }, - orderBy: { name: 'AscNullsLast' }, + orderBy: [{ name: 'AscNullsLast' }], }, filteredSelectedEntities: { limit: 60, @@ -176,12 +176,12 @@ export const variables = { { id: { in: ['1'] } }, ], }, - orderBy: { name: 'AscNullsLast' }, + orderBy: [{ name: 'AscNullsLast' }], }, selectedEntities: { limit: 60, filter: { id: { in: ['1'] } }, - orderBy: { name: 'AscNullsLast' }, + orderBy: [{ name: 'AscNullsLast' }], }, }; diff --git a/packages/twenty-front/src/modules/search/hooks/useFilteredSearchEntityQuery.ts b/packages/twenty-front/src/modules/search/hooks/useFilteredSearchEntityQuery.ts index b4d337c007a8..d43468a3ff37 100644 --- a/packages/twenty-front/src/modules/search/hooks/useFilteredSearchEntityQuery.ts +++ b/packages/twenty-front/src/modules/search/hooks/useFilteredSearchEntityQuery.ts @@ -49,7 +49,7 @@ export const useFilteredSearchEntityQuery = ({ useFindManyRecords({ objectNameSingular, filter: selectedIdsFilter, - orderBy: { [orderByField]: sortOrder }, + orderBy: [{ [orderByField]: sortOrder }], skip: !selectedIds.length, }); @@ -93,7 +93,7 @@ export const useFilteredSearchEntityQuery = ({ } = useFindManyRecords({ objectNameSingular, filter: makeAndFilterVariables([...searchFilters, selectedIdsFilter]), - orderBy: { [orderByField]: sortOrder }, + orderBy: [{ [orderByField]: sortOrder }], skip: !selectedIds.length, }); @@ -106,7 +106,7 @@ export const useFilteredSearchEntityQuery = ({ objectNameSingular, filter: makeAndFilterVariables([...searchFilters, notFilter]), limit: limit ?? DEFAULT_SEARCH_REQUEST_LIMIT, - orderBy: { [orderByField]: sortOrder }, + orderBy: [{ [orderByField]: sortOrder }], }); return { diff --git a/packages/twenty-front/src/modules/search/queries/getTextToSQL.ts b/packages/twenty-front/src/modules/search/queries/getTextToSQL.ts new file mode 100644 index 000000000000..6758209824b6 --- /dev/null +++ b/packages/twenty-front/src/modules/search/queries/getTextToSQL.ts @@ -0,0 +1,11 @@ +import { gql } from '@apollo/client'; + +export const getCopilot = gql` + query GetAISQLQuery($text: String!) { + getAISQLQuery(text: $text) { + sqlQuery + sqlQueryResult + queryFailedErrorMessage + } + } +`; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistInput.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistInput.tsx similarity index 94% rename from packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistInput.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistInput.tsx index 1794eb1d42bf..9bb4fdd98ce5 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistInput.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistInput.tsx @@ -19,7 +19,7 @@ const StyledLinkContainer = styled.div` margin-right: ${({ theme }) => theme.spacing(2)}; `; -type SettingsAccountsEmailsBlocklistInputProps = { +type SettingsAccountsBlocklistInputProps = { updateBlockedEmailList: (email: string) => void; blockedEmailOrDomainList: string[]; }; @@ -50,10 +50,10 @@ type FormInput = { emailOrDomain: string; }; -export const SettingsAccountsEmailsBlocklistInput = ({ +export const SettingsAccountsBlocklistInput = ({ updateBlockedEmailList, blockedEmailOrDomainList, -}: SettingsAccountsEmailsBlocklistInputProps) => { +}: SettingsAccountsBlocklistInputProps) => { const { reset, handleSubmit, control, formState } = useForm<FormInput>({ mode: 'onSubmit', resolver: zodResolver(validationSchema(blockedEmailOrDomainList)), diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistSection.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistSection.tsx similarity index 81% rename from packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistSection.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistSection.tsx index 39b789fae57a..51ff642b1b35 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistSection.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistSection.tsx @@ -7,11 +7,11 @@ import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSi import { useCreateOneRecord } from '@/object-record/hooks/useCreateOneRecord'; import { useDeleteOneRecord } from '@/object-record/hooks/useDeleteOneRecord'; import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; -import { SettingsAccountsEmailsBlocklistInput } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistInput'; -import { SettingsAccountsEmailsBlocklistTable } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistTable'; +import { SettingsAccountsBlocklistInput } from '@/settings/accounts/components/SettingsAccountsBlocklistInput'; +import { SettingsAccountsBlocklistTable } from '@/settings/accounts/components/SettingsAccountsBlocklistTable'; import { Section } from '@/ui/layout/section/components/Section'; -export const SettingsAccountsEmailsBlocklistSection = () => { +export const SettingsAccountsBlocklistSection = () => { const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); const { records: blocklist } = useFindManyRecords<BlocklistItem>({ @@ -44,11 +44,11 @@ export const SettingsAccountsEmailsBlocklistSection = () => { title="Blocklist" description="Exclude the following people and domains from my email sync" /> - <SettingsAccountsEmailsBlocklistInput + <SettingsAccountsBlocklistInput blockedEmailOrDomainList={blocklist.map((item) => item.handle)} updateBlockedEmailList={updateBlockedEmailList} /> - <SettingsAccountsEmailsBlocklistTable + <SettingsAccountsBlocklistTable blocklist={blocklist} handleBlockedEmailRemove={handleBlockedEmailRemove} /> diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistTable.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistTable.tsx similarity index 79% rename from packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistTable.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistTable.tsx index a238ef6e5e03..a4c9f5306fad 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistTable.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistTable.tsx @@ -1,13 +1,13 @@ import styled from '@emotion/styled'; import { BlocklistItem } from '@/accounts/types/BlocklistItem'; -import { SettingsAccountsEmailsBlocklistTableRow } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistTableRow'; +import { SettingsAccountsBlocklistTableRow } from '@/settings/accounts/components/SettingsAccountsBlocklistTableRow'; import { Table } from '@/ui/layout/table/components/Table'; import { TableBody } from '@/ui/layout/table/components/TableBody'; import { TableHeader } from '@/ui/layout/table/components/TableHeader'; import { TableRow } from '@/ui/layout/table/components/TableRow'; -type SettingsAccountsEmailsBlocklistTableProps = { +type SettingsAccountsBlocklistTableProps = { blocklist: BlocklistItem[]; handleBlockedEmailRemove: (id: string) => void; }; @@ -20,10 +20,10 @@ const StyledTableBody = styled(TableBody)` border-bottom: 1px solid ${({ theme }) => theme.border.color.light}; `; -export const SettingsAccountsEmailsBlocklistTable = ({ +export const SettingsAccountsBlocklistTable = ({ blocklist, handleBlockedEmailRemove, -}: SettingsAccountsEmailsBlocklistTableProps) => { +}: SettingsAccountsBlocklistTableProps) => { return ( <> {blocklist.length > 0 && ( @@ -35,7 +35,7 @@ export const SettingsAccountsEmailsBlocklistTable = ({ </TableRow> <StyledTableBody> {blocklist.map((blocklistItem) => ( - <SettingsAccountsEmailsBlocklistTableRow + <SettingsAccountsBlocklistTableRow key={blocklistItem.id} blocklistItem={blocklistItem} onRemove={handleBlockedEmailRemove} diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistTableRow.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistTableRow.tsx similarity index 85% rename from packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistTableRow.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistTableRow.tsx index a09a092ed234..9a1148447a17 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsEmailsBlocklistTableRow.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsBlocklistTableRow.tsx @@ -6,15 +6,15 @@ import { TableCell } from '@/ui/layout/table/components/TableCell'; import { TableRow } from '@/ui/layout/table/components/TableRow'; import { formatToHumanReadableDate } from '~/utils/date-utils'; -type SettingsAccountsEmailsBlocklistTableRowProps = { +type SettingsAccountsBlocklistTableRowProps = { blocklistItem: BlocklistItem; onRemove: (id: string) => void; }; -export const SettingsAccountsEmailsBlocklistTableRow = ({ +export const SettingsAccountsBlocklistTableRow = ({ blocklistItem, onRemove, -}: SettingsAccountsEmailsBlocklistTableRowProps) => { +}: SettingsAccountsBlocklistTableRowProps) => { return ( <TableRow key={blocklistItem.id}> <TableCell>{blocklistItem.handle}</TableCell> diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelDetails.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelDetails.tsx new file mode 100644 index 000000000000..3955904289ab --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelDetails.tsx @@ -0,0 +1,79 @@ +import { CalendarChannel } from '@/accounts/types/CalendarChannel'; +import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; +import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; +import { SettingsAccountsEventVisibilitySettingsCard } from '@/settings/accounts/components/SettingsAccountsCalendarVisibilitySettingsCard'; +import { SettingsAccountsToggleSettingCard } from '@/settings/accounts/components/SettingsAccountsToggleSettingCard'; +import styled from '@emotion/styled'; +import { Section } from '@react-email/components'; +import { H2Title } from 'twenty-ui'; +import { CalendarChannelVisibility } from '~/generated-metadata/graphql'; + +const StyledDetailsContainer = styled.div` + display: flex; + flex-direction: column; + gap: ${({ theme }) => theme.spacing(6)}; +`; + +type SettingsAccountsCalendarChannelDetailsProps = { + calendarChannel: Pick< + CalendarChannel, + 'id' | 'visibility' | 'isContactAutoCreationEnabled' | 'isSyncEnabled' + >; +}; + +export const SettingsAccountsCalendarChannelDetails = ({ + calendarChannel, +}: SettingsAccountsCalendarChannelDetailsProps) => { + const { updateOneRecord } = useUpdateOneRecord<CalendarChannel>({ + objectNameSingular: CoreObjectNameSingular.CalendarChannel, + }); + + const handleVisibilityChange = (value: CalendarChannelVisibility) => { + updateOneRecord({ + idToUpdate: calendarChannel.id, + updateOneRecordInput: { + visibility: value, + }, + }); + }; + + const handleContactAutoCreationToggle = (value: boolean) => { + updateOneRecord({ + idToUpdate: calendarChannel.id, + updateOneRecordInput: { + isContactAutoCreationEnabled: value, + }, + }); + }; + + return ( + <StyledDetailsContainer> + <Section> + <H2Title + title="Event visibility" + description="Define what will be visible to other users in your workspace" + /> + <SettingsAccountsEventVisibilitySettingsCard + value={calendarChannel.visibility} + onChange={handleVisibilityChange} + /> + </Section> + <Section> + <H2Title + title="Contact auto-creation" + description="Automatically create contacts for people you've participated in an event with." + /> + <SettingsAccountsToggleSettingCard + parameters={[ + { + value: !!calendarChannel.isContactAutoCreationEnabled, + title: 'Auto-creation', + description: 'Automatically create contacts for people.', + onToggle: handleContactAutoCreationToggle, + }, + ]} + /> + </Section> + </StyledDetailsContainer> + ); +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsContainer.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsContainer.tsx new file mode 100644 index 000000000000..10e7017d501d --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsContainer.tsx @@ -0,0 +1,85 @@ +import styled from '@emotion/styled'; +import { useRecoilValue } from 'recoil'; + +import { CalendarChannel } from '@/accounts/types/CalendarChannel'; +import { ConnectedAccount } from '@/accounts/types/ConnectedAccount'; +import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; +import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; +import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; +import { SettingsAccountsCalendarChannelDetails } from '@/settings/accounts/components/SettingsAccountsCalendarChannelDetails'; +import { SettingsAccountsCalendarChannelsGeneral } from '@/settings/accounts/components/SettingsAccountsCalendarChannelsGeneral'; +import { SettingsAccountsListEmptyStateCard } from '@/settings/accounts/components/SettingsAccountsListEmptyStateCard'; +import { SETTINGS_ACCOUNT_CALENDAR_CHANNELS_TAB_LIST_COMPONENT_ID } from '@/settings/accounts/constants/SettingsAccountCalendarChannelsTabListComponentId'; +import { TabList } from '@/ui/layout/tab/components/TabList'; +import { useTabList } from '@/ui/layout/tab/hooks/useTabList'; +import React from 'react'; + +const StyledCalenderContainer = styled.div` + padding-bottom: ${({ theme }) => theme.spacing(6)}; +`; + +export const SettingsAccountsCalendarChannelsContainer = () => { + const { activeTabIdState } = useTabList( + SETTINGS_ACCOUNT_CALENDAR_CHANNELS_TAB_LIST_COMPONENT_ID, + ); + const activeTabId = useRecoilValue(activeTabIdState); + const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); + + const { records: accounts } = useFindManyRecords<ConnectedAccount>({ + objectNameSingular: CoreObjectNameSingular.ConnectedAccount, + filter: { + accountOwnerId: { + eq: currentWorkspaceMember?.id, + }, + }, + }); + + const { records: calendarChannels } = useFindManyRecords< + CalendarChannel & { + connectedAccount: ConnectedAccount; + } + >({ + objectNameSingular: CoreObjectNameSingular.CalendarChannel, + filter: { + connectedAccountId: { + in: accounts.map((account) => account.id), + }, + }, + }); + + const tabs = [ + ...calendarChannels.map((calendarChannel) => ({ + id: calendarChannel.id, + title: calendarChannel.handle, + })), + ]; + + if (!calendarChannels.length) { + return <SettingsAccountsListEmptyStateCard />; + } + + return ( + <> + {tabs.length > 1 && ( + <StyledCalenderContainer> + <TabList + tabListId={SETTINGS_ACCOUNT_CALENDAR_CHANNELS_TAB_LIST_COMPONENT_ID} + tabs={tabs} + /> + </StyledCalenderContainer> + )} + {calendarChannels.map((calendarChannel) => ( + <React.Fragment key={calendarChannel.id}> + {calendarChannel.id === activeTabId && ( + <SettingsAccountsCalendarChannelDetails + calendarChannel={calendarChannel} + /> + )} + </React.Fragment> + ))} + {false && activeTabId === 'general' && ( + <SettingsAccountsCalendarChannelsGeneral /> + )} + </> + ); +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsGeneral.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsGeneral.tsx new file mode 100644 index 000000000000..14ffba7842a9 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsGeneral.tsx @@ -0,0 +1,93 @@ +import { CalendarMonthCard } from '@/activities/calendar/components/CalendarMonthCard'; +import { CalendarContext } from '@/activities/calendar/contexts/CalendarContext'; +import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; +import { SettingsAccountsCalendarDisplaySettings } from '@/settings/accounts/components/SettingsAccountsCalendarDisplaySettings'; +import styled from '@emotion/styled'; +import { Section } from '@react-email/components'; +import { addMinutes, endOfDay, min, startOfDay } from 'date-fns'; +import { useRecoilValue } from 'recoil'; +import { H2Title } from 'twenty-ui'; +import { + CalendarChannelVisibility, + TimelineCalendarEvent, +} from '~/generated-metadata/graphql'; + +const StyledGeneralContainer = styled.div` + display: flex; + flex-direction: column; + gap: ${({ theme }) => theme.spacing(6)}; + padding-top: ${({ theme }) => theme.spacing(6)}; +`; + +export const SettingsAccountsCalendarChannelsGeneral = () => { + const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); + + const exampleStartDate = new Date(); + const exampleEndDate = min([ + addMinutes(exampleStartDate, 30), + endOfDay(exampleStartDate), + ]); + const exampleDayTime = startOfDay(exampleStartDate).getTime(); + const exampleCalendarEvent: TimelineCalendarEvent = { + id: '', + participants: [ + { + firstName: currentWorkspaceMember?.name.firstName || '', + lastName: currentWorkspaceMember?.name.lastName || '', + displayName: currentWorkspaceMember + ? [ + currentWorkspaceMember.name.firstName, + currentWorkspaceMember.name.lastName, + ].join(' ') + : '', + avatarUrl: currentWorkspaceMember?.avatarUrl || '', + handle: '', + personId: '', + workspaceMemberId: currentWorkspaceMember?.id || '', + }, + ], + endsAt: exampleEndDate.toISOString(), + isFullDay: false, + startsAt: exampleStartDate.toISOString(), + conferenceSolution: '', + conferenceLink: { + label: '', + url: '', + }, + description: '', + isCanceled: false, + location: '', + title: 'Onboarding call', + visibility: CalendarChannelVisibility.ShareEverything, + }; + + return ( + <StyledGeneralContainer> + <Section> + <H2Title + title="Display" + description="Configure how we should display your events in your calendar" + /> + <SettingsAccountsCalendarDisplaySettings /> + </Section> + <Section> + <H2Title + title="Color code" + description="Events you participated in are displayed in red." + /> + <CalendarContext.Provider + value={{ + currentCalendarEvent: exampleCalendarEvent, + calendarEventsByDayTime: { + [exampleDayTime]: [exampleCalendarEvent], + }, + getNextCalendarEvent: () => undefined, + updateCurrentCalendarEvent: () => {}, + }} + > + <CalendarMonthCard dayTimes={[exampleDayTime]} /> + </CalendarContext.Provider> + </Section> + </StyledGeneralContainer> + ); +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsListCard.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsListCard.tsx deleted file mode 100644 index e00f32398c76..000000000000 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarChannelsListCard.tsx +++ /dev/null @@ -1,95 +0,0 @@ -import { useNavigate } from 'react-router-dom'; -import styled from '@emotion/styled'; -import { useRecoilValue } from 'recoil'; -import { IconChevronRight, IconGoogleCalendar } from 'twenty-ui'; - -import { CalendarChannel } from '@/accounts/types/CalendarChannel'; -import { ConnectedAccount } from '@/accounts/types/ConnectedAccount'; -import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; -import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; -import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; -import { generateDepthOneRecordGqlFields } from '@/object-record/graphql/utils/generateDepthOneRecordGqlFields'; -import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; -import { SettingsAccountsListEmptyStateCard } from '@/settings/accounts/components/SettingsAccountsListEmptyStateCard'; -import { - SettingsAccountsSynchronizationStatus, - SettingsAccountsSynchronizationStatusProps, -} from '@/settings/accounts/components/SettingsAccountsSynchronizationStatus'; -import { SettingsListCard } from '@/settings/components/SettingsListCard'; -import { LightIconButton } from '@/ui/input/button/components/LightIconButton'; - -const StyledRowRightContainer = styled.div` - align-items: center; - display: flex; - gap: ${({ theme }) => theme.spacing(1)}; -`; - -export const SettingsAccountsCalendarChannelsListCard = () => { - const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); - const navigate = useNavigate(); - const { objectMetadataItem } = useObjectMetadataItem({ - objectNameSingular: CoreObjectNameSingular.CalendarChannel, - }); - - const { records: accounts, loading: accountsLoading } = - useFindManyRecords<ConnectedAccount>({ - objectNameSingular: CoreObjectNameSingular.ConnectedAccount, - filter: { - accountOwnerId: { - eq: currentWorkspaceMember?.id, - }, - }, - }); - - const { records: calendarChannels, loading: calendarChannelsLoading } = - useFindManyRecords< - CalendarChannel & { - connectedAccount: ConnectedAccount; - } - >({ - objectNameSingular: CoreObjectNameSingular.CalendarChannel, - skip: !accounts.length, - filter: { - connectedAccountId: { - in: accounts.map((account) => account.id), - }, - }, - recordGqlFields: generateDepthOneRecordGqlFields({ objectMetadataItem }), - }); - - if (!calendarChannels.length) { - return <SettingsAccountsListEmptyStateCard />; - } - - const calendarChannelsWithSyncStatus: (CalendarChannel & { - connectedAccount: ConnectedAccount; - } & SettingsAccountsSynchronizationStatusProps)[] = calendarChannels.map( - (calendarChannel) => ({ - ...calendarChannel, - syncStatus: calendarChannel.connectedAccount?.authFailedAt - ? 'FAILED' - : 'SUCCEEDED', - }), - ); - - return ( - <SettingsListCard - items={calendarChannelsWithSyncStatus} - getItemLabel={(calendarChannel) => calendarChannel.handle} - isLoading={accountsLoading || calendarChannelsLoading} - onRowClick={(calendarChannel) => - navigate(`/settings/accounts/calendars/${calendarChannel.id}`) - } - RowIcon={IconGoogleCalendar} - RowRightComponent={({ item: calendarChannel }) => ( - <StyledRowRightContainer> - <SettingsAccountsSynchronizationStatus - syncStatus={calendarChannel.syncStatus} - isSyncEnabled={calendarChannel.isSyncEnabled} - /> - <LightIconButton Icon={IconChevronRight} accent="tertiary" /> - </StyledRowRightContainer> - )} - /> - ); -}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarVisibilitySettingsCard.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarVisibilitySettingsCard.tsx index b1a26e3ca493..20344671bb85 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarVisibilitySettingsCard.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsCalendarVisibilitySettingsCard.tsx @@ -1,7 +1,7 @@ import styled from '@emotion/styled'; import { SettingsAccountsRadioSettingsCard } from '@/settings/accounts/components/SettingsAccountsRadioSettingsCard'; -import { SettingsAccountsVisibilitySettingCardMedia } from '@/settings/accounts/components/SettingsAccountsVisibilitySettingCardMedia'; +import { SettingsAccountsVisibilityIcon } from '@/settings/accounts/components/SettingsAccountsVisibilityIcon'; import { CalendarChannelVisibility } from '~/generated/graphql'; type SettingsAccountsEventVisibilitySettingsCardProps = { @@ -9,7 +9,7 @@ type SettingsAccountsEventVisibilitySettingsCardProps = { value?: CalendarChannelVisibility; }; -const StyledCardMedia = styled(SettingsAccountsVisibilitySettingCardMedia)` +const StyledCardMedia = styled(SettingsAccountsVisibilityIcon)` height: ${({ theme }) => theme.spacing(6)}; `; @@ -33,6 +33,7 @@ export const SettingsAccountsEventVisibilitySettingsCard = ({ value = CalendarChannelVisibility.ShareEverything, }: SettingsAccountsEventVisibilitySettingsCardProps) => ( <SettingsAccountsRadioSettingsCard + name="event-visibility" options={eventSettingsVisibilityOptions} value={value} onChange={onChange} diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageAutoCreationCard.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageAutoCreationCard.tsx new file mode 100644 index 000000000000..ea01199a02e4 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageAutoCreationCard.tsx @@ -0,0 +1,48 @@ +import { MessageChannelContactAutoCreationPolicy } from '@/accounts/types/MessageChannel'; +import { SettingsAccountsMessageAutoCreationIcon } from '@/settings/accounts/components/SettingsAccountsMessageAutoCreationIcon'; +import { SettingsAccountsRadioSettingsCard } from '@/settings/accounts/components/SettingsAccountsRadioSettingsCard'; + +type SettingsAccountsMessageAutoCreationCardProps = { + onChange: (nextValue: MessageChannelContactAutoCreationPolicy) => void; + value?: MessageChannelContactAutoCreationPolicy; +}; + +const autoCreationOptions = [ + { + title: 'Send and Received', + description: 'People I’ve sent emails to and received emails from.', + value: MessageChannelContactAutoCreationPolicy.SENT_AND_RECEIVED, + cardMedia: ( + <SettingsAccountsMessageAutoCreationIcon isSentActive isReceivedActive /> + ), + }, + { + title: 'Sent', + description: 'People I’ve sent emails to.', + value: MessageChannelContactAutoCreationPolicy.SENT, + cardMedia: <SettingsAccountsMessageAutoCreationIcon isSentActive />, + }, + { + title: 'None', + description: 'Don’t auto-create contacts.', + value: MessageChannelContactAutoCreationPolicy.NONE, + cardMedia: ( + <SettingsAccountsMessageAutoCreationIcon + isSentActive={false} + isReceivedActive={false} + /> + ), + }, +]; + +export const SettingsAccountsMessageAutoCreationCard = ({ + onChange, + value = MessageChannelContactAutoCreationPolicy.SENT_AND_RECEIVED, +}: SettingsAccountsMessageAutoCreationCardProps) => ( + <SettingsAccountsRadioSettingsCard + name="message-auto-creation" + options={autoCreationOptions} + value={value} + onChange={onChange} + /> +); diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageAutoCreationIcon.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageAutoCreationIcon.tsx new file mode 100644 index 000000000000..d628f347c9d6 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageAutoCreationIcon.tsx @@ -0,0 +1,31 @@ +import styled from '@emotion/styled'; + +import { SettingsAccountsCardMedia } from '@/settings/accounts/components/SettingsAccountsCardMedia'; + +type SettingsAccountsMessageAutoCreationIconProps = { + className?: string; + isSentActive?: boolean; + isReceivedActive?: boolean; +}; + +const StyledIconContainer = styled(SettingsAccountsCardMedia)` + align-items: stretch; +`; + +const StyledDirectionSkeleton = styled.div<{ isActive?: boolean }>` + background-color: ${({ isActive, theme }) => + isActive ? theme.accent.accent4060 : theme.background.quaternary}; + border-radius: 1px; + height: 24px; +`; + +export const SettingsAccountsMessageAutoCreationIcon = ({ + className, + isSentActive, + isReceivedActive, +}: SettingsAccountsMessageAutoCreationIconProps) => ( + <StyledIconContainer className={className}> + <StyledDirectionSkeleton isActive={isSentActive} /> + <StyledDirectionSkeleton isActive={isReceivedActive} /> + </StyledIconContainer> +); diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelDetails.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelDetails.tsx new file mode 100644 index 000000000000..930d749abd39 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelDetails.tsx @@ -0,0 +1,121 @@ +import styled from '@emotion/styled'; +import { H2Title } from 'twenty-ui'; + +import { + MessageChannel, + MessageChannelContactAutoCreationPolicy, +} from '@/accounts/types/MessageChannel'; +import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; +import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; +import { SettingsAccountsMessageAutoCreationCard } from '@/settings/accounts/components/SettingsAccountsMessageAutoCreationCard'; +import { SettingsAccountsMessageVisibilityCard } from '@/settings/accounts/components/SettingsAccountsMessageVisibilityCard'; +import { SettingsAccountsToggleSettingCard } from '@/settings/accounts/components/SettingsAccountsToggleSettingCard'; +import { Section } from '@/ui/layout/section/components/Section'; +import { MessageChannelVisibility } from '~/generated-metadata/graphql'; + +type SettingsAccountsMessageChannelDetailsProps = { + messageChannel: Pick< + MessageChannel, + | 'id' + | 'visibility' + | 'contactAutoCreationPolicy' + | 'excludeNonProfessionalEmails' + | 'excludeGroupEmails' + | 'isSyncEnabled' + >; +}; + +const StyledDetailsContainer = styled.div` + display: flex; + flex-direction: column; + gap: ${({ theme }) => theme.spacing(6)}; +`; + +export const SettingsAccountsMessageChannelDetails = ({ + messageChannel, +}: SettingsAccountsMessageChannelDetailsProps) => { + const { updateOneRecord } = useUpdateOneRecord<MessageChannel>({ + objectNameSingular: CoreObjectNameSingular.MessageChannel, + }); + + const handleVisibilityChange = (value: MessageChannelVisibility) => { + updateOneRecord({ + idToUpdate: messageChannel.id, + updateOneRecordInput: { + visibility: value, + }, + }); + }; + + const handleContactAutoCreationChange = ( + value: MessageChannelContactAutoCreationPolicy, + ) => { + updateOneRecord({ + idToUpdate: messageChannel.id, + updateOneRecordInput: { + contactAutoCreationPolicy: value, + }, + }); + }; + + const handleIsGroupEmailExcludedToggle = (value: boolean) => { + updateOneRecord({ + idToUpdate: messageChannel.id, + updateOneRecordInput: { + excludeGroupEmails: value, + }, + }); + }; + + const handleIsNonProfessionalEmailExcludedToggle = (value: boolean) => { + updateOneRecord({ + idToUpdate: messageChannel.id, + updateOneRecordInput: { + excludeNonProfessionalEmails: value, + }, + }); + }; + + return ( + <StyledDetailsContainer> + <Section> + <H2Title + title="Visibility" + description="Define what will be visible to other users in your workspace" + /> + <SettingsAccountsMessageVisibilityCard + value={messageChannel.visibility} + onChange={handleVisibilityChange} + /> + </Section> + <Section> + <H2Title + title="Contact auto-creation" + description="Automatically create People records when receiving or sending emails" + /> + <SettingsAccountsMessageAutoCreationCard + value={messageChannel.contactAutoCreationPolicy} + onChange={handleContactAutoCreationChange} + /> + </Section> + <Section> + <SettingsAccountsToggleSettingCard + parameters={[ + { + title: 'Exclude non-professional emails', + description: 'Don’t sync emails from/to Gmail, Outlook...', + value: !!messageChannel.excludeNonProfessionalEmails, + onToggle: handleIsNonProfessionalEmailExcludedToggle, + }, + { + title: 'Exclude group emails', + description: 'Don’t sync emails from team@ support@ noreply@...', + value: !!messageChannel.excludeGroupEmails, + onToggle: handleIsGroupEmailExcludedToggle, + }, + ]} + /> + </Section> + </StyledDetailsContainer> + ); +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelsContainer.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelsContainer.tsx new file mode 100644 index 000000000000..24a5f62fa1b0 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelsContainer.tsx @@ -0,0 +1,81 @@ +import styled from '@emotion/styled'; +import { useRecoilValue } from 'recoil'; + +import { ConnectedAccount } from '@/accounts/types/ConnectedAccount'; +import { MessageChannel } from '@/accounts/types/MessageChannel'; +import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; +import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; +import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; +import { SettingsAccountsListEmptyStateCard } from '@/settings/accounts/components/SettingsAccountsListEmptyStateCard'; +import { SettingsAccountsMessageChannelDetails } from '@/settings/accounts/components/SettingsAccountsMessageChannelDetails'; +import { SETTINGS_ACCOUNT_MESSAGE_CHANNELS_TAB_LIST_COMPONENT_ID } from '@/settings/accounts/constants/SettingsAccountMessageChannelsTabListComponentId'; +import { TabList } from '@/ui/layout/tab/components/TabList'; +import { useTabList } from '@/ui/layout/tab/hooks/useTabList'; +import React from 'react'; + +const StyledMessageContainer = styled.div` + padding-bottom: ${({ theme }) => theme.spacing(6)}; +`; + +export const SettingsAccountsMessageChannelsContainer = () => { + const { activeTabIdState } = useTabList( + SETTINGS_ACCOUNT_MESSAGE_CHANNELS_TAB_LIST_COMPONENT_ID, + ); + const activeTabId = useRecoilValue(activeTabIdState); + const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); + + const { records: accounts } = useFindManyRecords<ConnectedAccount>({ + objectNameSingular: CoreObjectNameSingular.ConnectedAccount, + filter: { + accountOwnerId: { + eq: currentWorkspaceMember?.id, + }, + }, + }); + + const { records: messageChannels } = useFindManyRecords< + MessageChannel & { + connectedAccount: ConnectedAccount; + } + >({ + objectNameSingular: CoreObjectNameSingular.MessageChannel, + filter: { + connectedAccountId: { + in: accounts.map((account) => account.id), + }, + }, + }); + + const tabs = [ + ...messageChannels.map((messageChannel) => ({ + id: messageChannel.id, + title: messageChannel.handle, + })), + ]; + + if (!messageChannels.length) { + return <SettingsAccountsListEmptyStateCard />; + } + + return ( + <> + {tabs.length > 1 && ( + <StyledMessageContainer> + <TabList + tabListId={SETTINGS_ACCOUNT_MESSAGE_CHANNELS_TAB_LIST_COMPONENT_ID} + tabs={tabs} + /> + </StyledMessageContainer> + )} + {messageChannels.map((messageChannel) => ( + <React.Fragment key={messageChannel.id}> + {messageChannel.id === activeTabId && ( + <SettingsAccountsMessageChannelDetails + messageChannel={messageChannel} + /> + )} + </React.Fragment> + ))} + </> + ); +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelsListCard.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelsListCard.tsx deleted file mode 100644 index addc376b4d12..000000000000 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageChannelsListCard.tsx +++ /dev/null @@ -1,86 +0,0 @@ -import { useNavigate } from 'react-router-dom'; -import styled from '@emotion/styled'; -import { useRecoilValue } from 'recoil'; -import { IconChevronRight, IconGmail } from 'twenty-ui'; - -import { ConnectedAccount } from '@/accounts/types/ConnectedAccount'; -import { MessageChannel } from '@/accounts/types/MessageChannel'; -import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; -import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; -import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; -import { SettingsAccountsListEmptyStateCard } from '@/settings/accounts/components/SettingsAccountsListEmptyStateCard'; -import { - SettingsAccountsSynchronizationStatus, - SettingsAccountsSynchronizationStatusProps, -} from '@/settings/accounts/components/SettingsAccountsSynchronizationStatus'; -import { SettingsListCard } from '@/settings/components/SettingsListCard'; -import { LightIconButton } from '@/ui/input/button/components/LightIconButton'; - -const StyledRowRightContainer = styled.div` - align-items: center; - display: flex; - gap: ${({ theme }) => theme.spacing(1)}; -`; - -export const SettingsAccountsMessageChannelsListCard = () => { - const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); - const navigate = useNavigate(); - - const { records: accounts, loading: accountsLoading } = - useFindManyRecords<ConnectedAccount>({ - objectNameSingular: CoreObjectNameSingular.ConnectedAccount, - filter: { - accountOwnerId: { - eq: currentWorkspaceMember?.id, - }, - }, - }); - - const { records: messageChannels, loading: messageChannelsLoading } = - useFindManyRecords< - MessageChannel & { - connectedAccount: ConnectedAccount; - } - >({ - objectNameSingular: CoreObjectNameSingular.MessageChannel, - filter: { - connectedAccountId: { - in: accounts.map((account) => account.id), - }, - }, - }); - - const messageChannelsWithSyncedEmails: (MessageChannel & { - connectedAccount: ConnectedAccount; - } & SettingsAccountsSynchronizationStatusProps)[] = messageChannels.map( - (messageChannel) => ({ - ...messageChannel, - syncStatus: messageChannel.syncStatus, - }), - ); - - if (!messageChannelsWithSyncedEmails.length) { - return <SettingsAccountsListEmptyStateCard />; - } - - return ( - <SettingsListCard - items={messageChannelsWithSyncedEmails} - getItemLabel={(messageChannel) => messageChannel.handle} - isLoading={accountsLoading || messageChannelsLoading} - onRowClick={(messageChannel) => - navigate(`/settings/accounts/emails/${messageChannel.id}`) - } - RowIcon={IconGmail} - RowRightComponent={({ item: messageChannel }) => ( - <StyledRowRightContainer> - <SettingsAccountsSynchronizationStatus - syncStatus={messageChannel.syncStatus} - isSyncEnabled={messageChannel.isSyncEnabled} - /> - <LightIconButton Icon={IconChevronRight} accent="tertiary" /> - </StyledRowRightContainer> - )} - /> - ); -}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsInboxVisibilitySettingsCard.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageVisibilityCard.tsx similarity index 74% rename from packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsInboxVisibilitySettingsCard.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageVisibilityCard.tsx index 9f9469f46d2c..e0aa3172fd00 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsInboxVisibilitySettingsCard.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsMessageVisibilityCard.tsx @@ -1,8 +1,8 @@ import { SettingsAccountsRadioSettingsCard } from '@/settings/accounts/components/SettingsAccountsRadioSettingsCard'; -import { SettingsAccountsVisibilitySettingCardMedia } from '@/settings/accounts/components/SettingsAccountsVisibilitySettingCardMedia'; +import { SettingsAccountsVisibilityIcon } from '@/settings/accounts/components/SettingsAccountsVisibilityIcon'; import { MessageChannelVisibility } from '~/generated/graphql'; -type SettingsAccountsInboxVisibilitySettingsCardProps = { +type SettingsAccountsMessageVisibilityCardProps = { onChange: (nextValue: MessageChannelVisibility) => void; value?: MessageChannelVisibility; }; @@ -13,7 +13,7 @@ const inboxSettingsVisibilityOptions = [ description: 'Subject, body and attachments will be shared with your team.', value: MessageChannelVisibility.ShareEverything, cardMedia: ( - <SettingsAccountsVisibilitySettingCardMedia + <SettingsAccountsVisibilityIcon metadata="active" subject="active" body="active" @@ -25,7 +25,7 @@ const inboxSettingsVisibilityOptions = [ description: 'Subject and metadata will be shared with your team.', value: MessageChannelVisibility.Subject, cardMedia: ( - <SettingsAccountsVisibilitySettingCardMedia + <SettingsAccountsVisibilityIcon metadata="active" subject="active" body="inactive" @@ -37,7 +37,7 @@ const inboxSettingsVisibilityOptions = [ description: 'Timestamp and participants will be shared with your team.', value: MessageChannelVisibility.Metadata, cardMedia: ( - <SettingsAccountsVisibilitySettingCardMedia + <SettingsAccountsVisibilityIcon metadata="active" subject="inactive" body="inactive" @@ -46,11 +46,12 @@ const inboxSettingsVisibilityOptions = [ }, ]; -export const SettingsAccountsInboxVisibilitySettingsCard = ({ +export const SettingsAccountsMessageVisibilityCard = ({ onChange, value = MessageChannelVisibility.ShareEverything, -}: SettingsAccountsInboxVisibilitySettingsCardProps) => ( +}: SettingsAccountsMessageVisibilityCardProps) => ( <SettingsAccountsRadioSettingsCard + name="message-visibility" options={inboxSettingsVisibilityOptions} value={value} onChange={onChange} diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRadioSettingsCard.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRadioSettingsCard.tsx index 4592b42b4c2a..aaf856456fcf 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRadioSettingsCard.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRadioSettingsCard.tsx @@ -1,5 +1,5 @@ -import { ReactNode } from 'react'; import styled from '@emotion/styled'; +import { ReactNode } from 'react'; import { Radio } from '@/ui/input/components/Radio'; import { Card } from '@/ui/layout/card/components/Card'; @@ -10,6 +10,7 @@ type SettingsAccountsRadioSettingsCardProps<Option extends { value: string }> = onChange: (nextValue: Option['value']) => void; options: Option[]; value: Option['value']; + name: string; }; const StyledCardContent = styled(CardContent)` @@ -49,6 +50,7 @@ export const SettingsAccountsRadioSettingsCard = < onChange, options, value, + name, }: SettingsAccountsRadioSettingsCardProps<Option>) => ( <Card rounded> {options.map((option, index) => ( @@ -63,6 +65,7 @@ export const SettingsAccountsRadioSettingsCard = < <StyledDescription>{option.description}</StyledDescription> </div> <StyledRadio + name={name} value={option.value} onCheckedChange={() => onChange(option.value)} checked={value === option.value} diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRowDropdownMenu.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRowDropdownMenu.tsx index 8e5afcec6125..515dce50a610 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRowDropdownMenu.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsRowDropdownMenu.tsx @@ -54,9 +54,7 @@ export const SettingsAccountsRowDropdownMenu = ({ LeftIcon={IconMail} text="Emails settings" onClick={() => { - navigate( - `/settings/accounts/emails/${account.messageChannels[0].id}`, - ); + navigate(`/settings/accounts/emails`); closeDropdown(); }} /> @@ -64,9 +62,7 @@ export const SettingsAccountsRowDropdownMenu = ({ LeftIcon={IconCalendarEvent} text="Calendar settings" onClick={() => { - navigate( - `/settings/accounts/calendars/${account.calendarChannels[0].id}`, - ); + navigate(`/settings/accounts/calendars`); closeDropdown(); }} /> diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsSynchronizationStatus.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsSynchronizationStatus.tsx deleted file mode 100644 index b0f59f6a4acc..000000000000 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsSynchronizationStatus.tsx +++ /dev/null @@ -1,31 +0,0 @@ -import { useGetSyncStatusOptions } from '@/settings/accounts/hooks//useGetSyncStatusOptions'; -import { Status } from '@/ui/display/status/components/Status'; - -export type SettingsAccountsSynchronizationStatusProps = { - syncStatus: string; - isSyncEnabled?: boolean; -}; - -export const SettingsAccountsSynchronizationStatus = ({ - syncStatus, - isSyncEnabled, -}: SettingsAccountsSynchronizationStatusProps) => { - const syncStatusOptions = useGetSyncStatusOptions(); - - const syncStatusOption = syncStatusOptions?.find( - (option) => option.value === syncStatus, - ); - - if (!isSyncEnabled) { - return <Status color="gray" text="Not synced" weight="medium" />; - } - - return ( - <Status - color={syncStatusOption?.color ?? 'gray'} - isLoaderVisible={syncStatus === 'ONGOING' || syncStatus === 'PENDING'} - text={syncStatusOption?.label ?? 'Not synced'} - weight="medium" - /> - ); -}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsToggleSettingCard.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsToggleSettingCard.tsx index 7dee8347d9f2..62bfbaa40b8e 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsToggleSettingCard.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsToggleSettingCard.tsx @@ -1,22 +1,24 @@ -import { ReactNode } from 'react'; import styled from '@emotion/styled'; import { Toggle } from '@/ui/input/components/Toggle'; import { Card } from '@/ui/layout/card/components/Card'; import { CardContent } from '@/ui/layout/card/components/CardContent'; -type SettingsAccountsToggleSettingCardProps = { - cardMedia: ReactNode; +type Parameter = { value: boolean; - onToggle: (value: boolean) => void; title: string; + description: string; + onToggle: (value: boolean) => void; +}; + +type SettingsAccountsToggleSettingCardProps = { + parameters: Parameter[]; }; const StyledCardContent = styled(CardContent)` align-items: center; display: flex; gap: ${({ theme }) => theme.spacing(4)}; - padding: ${({ theme }) => theme.spacing(2, 4)}; cursor: pointer; &:hover { @@ -24,23 +26,37 @@ const StyledCardContent = styled(CardContent)` } `; -const StyledTitle = styled.span` +const StyledTitle = styled.div` color: ${({ theme }) => theme.font.color.primary}; font-weight: ${({ theme }) => theme.font.weight.medium}; - margin-right: auto; + margin-bottom: ${({ theme }) => theme.spacing(2)}; +`; + +const StyledDescription = styled.div` + color: ${({ theme }) => theme.font.color.tertiary}; + font-size: ${({ theme }) => theme.font.size.sm}; +`; + +const StyledToggle = styled(Toggle)` + margin-left: auto; `; export const SettingsAccountsToggleSettingCard = ({ - cardMedia, - value, - onToggle, - title, + parameters, }: SettingsAccountsToggleSettingCardProps) => ( <Card rounded> - <StyledCardContent onClick={() => onToggle(!value)}> - {cardMedia} - <StyledTitle>{title}</StyledTitle> - <Toggle value={value} onChange={onToggle} /> - </StyledCardContent> + {parameters.map((parameter, index) => ( + <StyledCardContent + key={index} + divider={index < parameters.length - 1} + onClick={() => parameter.onToggle(!parameter.value)} + > + <div> + <StyledTitle>{parameter.title}</StyledTitle> + <StyledDescription>{parameter.description}</StyledDescription> + </div> + <StyledToggle value={parameter.value} onChange={parameter.onToggle} /> + </StyledCardContent> + ))} </Card> ); diff --git a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsVisibilitySettingCardMedia.tsx b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsVisibilityIcon.tsx similarity index 87% rename from packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsVisibilitySettingCardMedia.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsVisibilityIcon.tsx index 119fb26e0c67..5931d828ec00 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsVisibilitySettingCardMedia.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/SettingsAccountsVisibilityIcon.tsx @@ -4,7 +4,7 @@ import { SettingsAccountsCardMedia } from '@/settings/accounts/components/Settin type VisibilityElementState = 'active' | 'inactive'; -type SettingsAccountsVisibilitySettingCardMediaProps = { +type SettingsAccountsVisibilityIconProps = { className?: string; metadata?: VisibilityElementState; subject?: VisibilityElementState; @@ -31,12 +31,12 @@ const StyledBodySkeleton = styled(StyledSubjectSkeleton)` flex: 1 0 auto; `; -export const SettingsAccountsVisibilitySettingCardMedia = ({ +export const SettingsAccountsVisibilityIcon = ({ className, metadata, subject, body, -}: SettingsAccountsVisibilitySettingCardMediaProps) => ( +}: SettingsAccountsVisibilityIconProps) => ( <StyledCardMedia className={className}> {!!metadata && <StyledMetadataSkeleton isActive={metadata === 'active'} />} {!!subject && <StyledSubjectSkeleton isActive={subject === 'active'} />} diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistInput.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistInput.stories.tsx similarity index 77% rename from packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistInput.stories.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistInput.stories.tsx index 3ceef7405d88..48a52896b524 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistInput.stories.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistInput.stories.tsx @@ -2,7 +2,7 @@ import { Decorator, Meta, StoryObj } from '@storybook/react'; import { expect, fn, userEvent, within } from '@storybook/test'; import { ComponentDecorator } from 'twenty-ui'; -import { SettingsAccountsEmailsBlocklistInput } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistInput'; +import { SettingsAccountsBlocklistInput } from '@/settings/accounts/components/SettingsAccountsBlocklistInput'; const updateBlockedEmailListJestFn = fn(); @@ -13,10 +13,9 @@ const ClearMocksDecorator: Decorator = (Story, context) => { return <Story />; }; -const meta: Meta<typeof SettingsAccountsEmailsBlocklistInput> = { - title: - 'Modules/Settings/Accounts/Blocklist/SettingsAccountsEmailsBlocklistInput', - component: SettingsAccountsEmailsBlocklistInput, +const meta: Meta<typeof SettingsAccountsBlocklistInput> = { + title: 'Modules/Settings/Accounts/Blocklist/SettingsAccountsBlocklistInput', + component: SettingsAccountsBlocklistInput, decorators: [ComponentDecorator, ClearMocksDecorator], args: { updateBlockedEmailList: updateBlockedEmailListJestFn, @@ -31,7 +30,7 @@ const meta: Meta<typeof SettingsAccountsEmailsBlocklistInput> = { }; export default meta; -type Story = StoryObj<typeof SettingsAccountsEmailsBlocklistInput>; +type Story = StoryObj<typeof SettingsAccountsBlocklistInput>; export const Default: Story = {}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistSection.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistSection.stories.tsx new file mode 100644 index 000000000000..bc3f9cb721e1 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistSection.stories.tsx @@ -0,0 +1,16 @@ +import { Meta, StoryObj } from '@storybook/react'; +import { ComponentDecorator } from 'twenty-ui'; + +import { SettingsAccountsBlocklistInput } from '@/settings/accounts/components/SettingsAccountsBlocklistInput'; +import { SettingsAccountsBlocklistSection } from '@/settings/accounts/components/SettingsAccountsBlocklistSection'; + +const meta: Meta<typeof SettingsAccountsBlocklistSection> = { + title: 'Modules/Settings/Accounts/Blocklist/SettingsAccountsBlocklistSection', + component: SettingsAccountsBlocklistInput, + decorators: [ComponentDecorator], +}; + +export default meta; +type Story = StoryObj<typeof SettingsAccountsBlocklistSection>; + +export const Default: Story = {}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistTable.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistTable.stories.tsx similarity index 81% rename from packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistTable.stories.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistTable.stories.tsx index 7ab7f1c9b051..71126ba4e285 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistTable.stories.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistTable.stories.tsx @@ -3,7 +3,7 @@ import { expect, fn, userEvent, within } from '@storybook/test'; import { ComponentDecorator } from 'twenty-ui'; import { mockedBlocklist } from '@/settings/accounts/components/__stories__/mockedBlocklist'; -import { SettingsAccountsEmailsBlocklistTable } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistTable'; +import { SettingsAccountsBlocklistTable } from '@/settings/accounts/components/SettingsAccountsBlocklistTable'; import { formatToHumanReadableDate } from '~/utils/date-utils'; const handleBlockedEmailRemoveJestFn = fn(); @@ -15,10 +15,9 @@ const ClearMocksDecorator: Decorator = (Story, context) => { return <Story />; }; -const meta: Meta<typeof SettingsAccountsEmailsBlocklistTable> = { - title: - 'Modules/Settings/Accounts/Blocklist/SettingsAccountsEmailsBlocklistTable', - component: SettingsAccountsEmailsBlocklistTable, +const meta: Meta<typeof SettingsAccountsBlocklistTable> = { + title: 'Modules/Settings/Accounts/Blocklist/SettingsAccountsBlocklistTable', + component: SettingsAccountsBlocklistTable, decorators: [ComponentDecorator, ClearMocksDecorator], args: { blocklist: mockedBlocklist, @@ -34,7 +33,7 @@ const meta: Meta<typeof SettingsAccountsEmailsBlocklistTable> = { }; export default meta; -type Story = StoryObj<typeof SettingsAccountsEmailsBlocklistTable>; +type Story = StoryObj<typeof SettingsAccountsBlocklistTable>; export const Default: Story = { play: async ({ canvasElement }) => { diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistTableRow.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistTableRow.stories.tsx similarity index 80% rename from packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistTableRow.stories.tsx rename to packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistTableRow.stories.tsx index f24b0033f5ae..b4a1991e65b4 100644 --- a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistTableRow.stories.tsx +++ b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsBlocklistTableRow.stories.tsx @@ -3,7 +3,7 @@ import { expect, fn, userEvent, within } from '@storybook/test'; import { ComponentDecorator } from 'twenty-ui'; import { mockedBlocklist } from '@/settings/accounts/components/__stories__/mockedBlocklist'; -import { SettingsAccountsEmailsBlocklistTableRow } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistTableRow'; +import { SettingsAccountsBlocklistTableRow } from '@/settings/accounts/components/SettingsAccountsBlocklistTableRow'; import { formatToHumanReadableDate } from '~/utils/date-utils'; const onRemoveJestFn = fn(); @@ -15,10 +15,10 @@ const ClearMocksDecorator: Decorator = (Story, context) => { return <Story />; }; -const meta: Meta<typeof SettingsAccountsEmailsBlocklistTableRow> = { +const meta: Meta<typeof SettingsAccountsBlocklistTableRow> = { title: - 'Modules/Settings/Accounts/Blocklist/SettingsAccountsEmailsBlocklistTableRow', - component: SettingsAccountsEmailsBlocklistTableRow, + 'Modules/Settings/Accounts/Blocklist/SettingsAccountsBlocklistTableRow', + component: SettingsAccountsBlocklistTableRow, decorators: [ComponentDecorator, ClearMocksDecorator], args: { blocklistItem: mockedBlocklist[0], @@ -34,7 +34,7 @@ const meta: Meta<typeof SettingsAccountsEmailsBlocklistTableRow> = { }; export default meta; -type Story = StoryObj<typeof SettingsAccountsEmailsBlocklistTableRow>; +type Story = StoryObj<typeof SettingsAccountsBlocklistTableRow>; export const Default: Story = { play: async ({ canvasElement }) => { diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsCalendarChannelDetails.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsCalendarChannelDetails.stories.tsx new file mode 100644 index 000000000000..5ef756baf4c6 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsCalendarChannelDetails.stories.tsx @@ -0,0 +1,30 @@ +import { Meta, StoryObj } from '@storybook/react'; +import { ComponentDecorator } from 'twenty-ui'; + +import { SettingsAccountsCalendarChannelDetails } from '@/settings/accounts/components/SettingsAccountsCalendarChannelDetails'; +import { CalendarChannelVisibility } from '~/generated/graphql'; + +const meta: Meta<typeof SettingsAccountsCalendarChannelDetails> = { + title: + 'Modules/Settings/Accounts/CalendarChannels/SettingsAccountsCalendarChannelDetails', + component: SettingsAccountsCalendarChannelDetails, + decorators: [ComponentDecorator], + args: { + calendarChannel: { + id: '20202020-ef5a-4822-9e08-ce6e6a4dcb6a', + isContactAutoCreationEnabled: true, + isSyncEnabled: true, + visibility: CalendarChannelVisibility.ShareEverything, + }, + }, + argTypes: { + calendarChannel: { control: false }, + }, +}; + +export default meta; +type Story = StoryObj<typeof SettingsAccountsCalendarChannelDetails>; + +export const Default: Story = { + play: async () => {}, +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsCalendarChannelsGeneral.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsCalendarChannelsGeneral.stories.tsx new file mode 100644 index 000000000000..49f279316908 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsCalendarChannelsGeneral.stories.tsx @@ -0,0 +1,18 @@ +import { Meta, StoryObj } from '@storybook/react'; +import { ComponentDecorator } from 'twenty-ui'; + +import { SettingsAccountsCalendarChannelsGeneral } from '@/settings/accounts/components/SettingsAccountsCalendarChannelsGeneral'; + +const meta: Meta<typeof SettingsAccountsCalendarChannelsGeneral> = { + title: + 'Modules/Settings/Accounts/CalendarChannels/SettingsAccountsCalendarChannelsGeneral', + component: SettingsAccountsCalendarChannelsGeneral, + decorators: [ComponentDecorator], +}; + +export default meta; +type Story = StoryObj<typeof SettingsAccountsCalendarChannelsGeneral>; + +export const Default: Story = { + play: async () => {}, +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistSection.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistSection.stories.tsx deleted file mode 100644 index 7fa2183aae4f..000000000000 --- a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsEmailsBlocklistSection.stories.tsx +++ /dev/null @@ -1,17 +0,0 @@ -import { Meta, StoryObj } from '@storybook/react'; -import { ComponentDecorator } from 'twenty-ui'; - -import { SettingsAccountsEmailsBlocklistInput } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistInput'; -import { SettingsAccountsEmailsBlocklistSection } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistSection'; - -const meta: Meta<typeof SettingsAccountsEmailsBlocklistSection> = { - title: - 'Modules/Settings/Accounts/Blocklist/SettingsAccountsEmailsBlocklistSection', - component: SettingsAccountsEmailsBlocklistInput, - decorators: [ComponentDecorator], -}; - -export default meta; -type Story = StoryObj<typeof SettingsAccountsEmailsBlocklistSection>; - -export const Default: Story = {}; diff --git a/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsMessageChannelDetails.stories.tsx b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsMessageChannelDetails.stories.tsx new file mode 100644 index 000000000000..02264d8e66fe --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/components/__stories__/SettingsAccountsMessageChannelDetails.stories.tsx @@ -0,0 +1,33 @@ +import { Meta, StoryObj } from '@storybook/react'; +import { ComponentDecorator } from 'twenty-ui'; + +import { MessageChannelContactAutoCreationPolicy } from '@/accounts/types/MessageChannel'; +import { SettingsAccountsMessageChannelDetails } from '@/settings/accounts/components/SettingsAccountsMessageChannelDetails'; +import { MessageChannelVisibility } from '~/generated/graphql'; + +const meta: Meta<typeof SettingsAccountsMessageChannelDetails> = { + title: + 'Modules/Settings/Accounts/MessageChannels/SettingsAccountsMessageChannelDetails', + component: SettingsAccountsMessageChannelDetails, + decorators: [ComponentDecorator], + args: { + messageChannel: { + id: '20202020-ef5a-4822-9e08-ce6e6a4dcb6a', + contactAutoCreationPolicy: MessageChannelContactAutoCreationPolicy.SENT, + excludeNonProfessionalEmails: true, + excludeGroupEmails: false, + isSyncEnabled: true, + visibility: MessageChannelVisibility.ShareEverything, + }, + }, + argTypes: { + messageChannel: { control: false }, + }, +}; + +export default meta; +type Story = StoryObj<typeof SettingsAccountsMessageChannelDetails>; + +export const Default: Story = { + play: async () => {}, +}; diff --git a/packages/twenty-front/src/modules/settings/accounts/constants/SettingsAccountCalendarChannelsTabListComponentId.ts b/packages/twenty-front/src/modules/settings/accounts/constants/SettingsAccountCalendarChannelsTabListComponentId.ts new file mode 100644 index 000000000000..5d414496d292 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/constants/SettingsAccountCalendarChannelsTabListComponentId.ts @@ -0,0 +1,2 @@ +export const SETTINGS_ACCOUNT_CALENDAR_CHANNELS_TAB_LIST_COMPONENT_ID = + 'settings-account-calendar-channels-tab-list'; diff --git a/packages/twenty-front/src/modules/settings/accounts/constants/SettingsAccountMessageChannelsTabListComponentId.ts b/packages/twenty-front/src/modules/settings/accounts/constants/SettingsAccountMessageChannelsTabListComponentId.ts new file mode 100644 index 000000000000..31f4638d767f --- /dev/null +++ b/packages/twenty-front/src/modules/settings/accounts/constants/SettingsAccountMessageChannelsTabListComponentId.ts @@ -0,0 +1,2 @@ +export const SETTINGS_ACCOUNT_MESSAGE_CHANNELS_TAB_LIST_COMPONENT_ID = + 'settings-account-message-channels-tab-list'; diff --git a/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/CancelButton.tsx b/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/CancelButton.tsx index f20d8737d7d7..e93d27a836d2 100644 --- a/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/CancelButton.tsx +++ b/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/CancelButton.tsx @@ -2,8 +2,19 @@ import { LightButton } from '@/ui/input/button/components/LightButton'; type CancelButtonProps = { onCancel?: () => void; + disabled?: boolean; }; -export const CancelButton = ({ onCancel }: CancelButtonProps) => { - return <LightButton title="Cancel" accent="tertiary" onClick={onCancel} />; +export const CancelButton = ({ + onCancel, + disabled = false, +}: CancelButtonProps) => { + return ( + <LightButton + title="Cancel" + accent="tertiary" + onClick={onCancel} + disabled={disabled} + /> + ); }; diff --git a/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/SaveAndCancelButtons.tsx b/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/SaveAndCancelButtons.tsx index 069e8aa201c4..5cb0e3677d6f 100644 --- a/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/SaveAndCancelButtons.tsx +++ b/packages/twenty-front/src/modules/settings/components/SaveAndCancelButtons/SaveAndCancelButtons.tsx @@ -13,16 +13,18 @@ type SaveAndCancelButtonsProps = { onSave?: () => void; onCancel?: () => void; isSaveDisabled?: boolean; + isCancelDisabled?: boolean; }; export const SaveAndCancelButtons = ({ onSave, onCancel, isSaveDisabled, + isCancelDisabled, }: SaveAndCancelButtonsProps) => { return ( <StyledContainer> - <CancelButton onCancel={onCancel} /> + <CancelButton onCancel={onCancel} disabled={isCancelDisabled} /> <SaveButton onSave={onSave} disabled={isSaveDisabled} /> </StyledContainer> ); diff --git a/packages/twenty-front/src/modules/settings/data-model/constants/DatabaseIdentifierMaximumLength.ts b/packages/twenty-front/src/modules/settings/data-model/constants/DatabaseIdentifierMaximumLength.ts new file mode 100644 index 000000000000..d7ac7dd9a27f --- /dev/null +++ b/packages/twenty-front/src/modules/settings/data-model/constants/DatabaseIdentifierMaximumLength.ts @@ -0,0 +1 @@ +export const DATABASE_IDENTIFIER_MAXIMUM_LENGTH = 63; diff --git a/packages/twenty-front/src/modules/settings/data-model/constants/FieldNameMaximumLength.ts b/packages/twenty-front/src/modules/settings/data-model/constants/FieldNameMaximumLength.ts new file mode 100644 index 000000000000..bd647b2cbee5 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/data-model/constants/FieldNameMaximumLength.ts @@ -0,0 +1,3 @@ +import { DATABASE_IDENTIFIER_MAXIMUM_LENGTH } from '@/settings/data-model/constants/DatabaseIdentifierMaximumLength'; + +export const FIELD_NAME_MAXIMUM_LENGTH = DATABASE_IDENTIFIER_MAXIMUM_LENGTH; diff --git a/packages/twenty-front/src/modules/settings/data-model/constants/ObjectNameMaximumLength.ts b/packages/twenty-front/src/modules/settings/data-model/constants/ObjectNameMaximumLength.ts new file mode 100644 index 000000000000..6ba3692374d2 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/data-model/constants/ObjectNameMaximumLength.ts @@ -0,0 +1,3 @@ +import { DATABASE_IDENTIFIER_MAXIMUM_LENGTH } from '@/settings/data-model/constants/DatabaseIdentifierMaximumLength'; + +export const OBJECT_NAME_MAXIMUM_LENGTH = DATABASE_IDENTIFIER_MAXIMUM_LENGTH; diff --git a/packages/twenty-front/src/modules/settings/data-model/constants/OptionValueMaximumLength.ts b/packages/twenty-front/src/modules/settings/data-model/constants/OptionValueMaximumLength.ts new file mode 100644 index 000000000000..97f5cc7726c7 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/data-model/constants/OptionValueMaximumLength.ts @@ -0,0 +1,3 @@ +import { DATABASE_IDENTIFIER_MAXIMUM_LENGTH } from '@/settings/data-model/constants/DatabaseIdentifierMaximumLength'; + +export const OPTION_VALUE_MAXIMUM_LENGTH = DATABASE_IDENTIFIER_MAXIMUM_LENGTH; diff --git a/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldCurrencyCodes.ts b/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldCurrencyCodes.ts index c84df4323d90..5e013b50ca26 100644 --- a/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldCurrencyCodes.ts +++ b/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldCurrencyCodes.ts @@ -1,5 +1,6 @@ import { IconComponent, + IconCurrencyBaht, IconCurrencyDirham, IconCurrencyDollar, IconCurrencyEuro, @@ -7,7 +8,9 @@ import { IconCurrencyKroneCzech, IconCurrencyKroneSwedish, IconCurrencyPound, + IconCurrencyReal, IconCurrencyRiyal, + IconCurrencyWon, IconCurrencyYen, IconCurrencyYuan, } from 'twenty-ui'; @@ -62,6 +65,10 @@ export const SETTINGS_FIELD_CURRENCY_CODES: Record< label: 'Swedish krona', Icon: IconCurrencyKroneSwedish, }, + BHT: { + label: 'Thai Baht', + Icon: IconCurrencyBaht, + }, MAD: { label: 'Moroccan dirham', Icon: IconCurrencyDirham, @@ -74,4 +81,16 @@ export const SETTINGS_FIELD_CURRENCY_CODES: Record< label: 'UAE dirham', Icon: IconCurrencyDirham, }, + KRW: { + label: 'South Korean won', + Icon: IconCurrencyWon, + }, + BRL: { + label: 'Brazilian real', + Icon: IconCurrencyReal, + }, + AUD: { + label: 'Australian dollar', + Icon: IconCurrencyDollar, + }, }; diff --git a/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldTypeConfigs.ts b/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldTypeConfigs.ts index a1b7135a89f4..9cc38edabd0c 100644 --- a/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldTypeConfigs.ts +++ b/packages/twenty-front/src/modules/settings/data-model/constants/SettingsFieldTypeConfigs.ts @@ -102,11 +102,6 @@ export const SETTINGS_FIELD_TYPE_CONFIGS = { Icon: IconPhone, defaultValue: '+1234-567-890', }, - [FieldMetadataType.Probability]: { - label: 'Rating', - Icon: IconTwentyStar, - defaultValue: '3', - }, [FieldMetadataType.Rating]: { label: 'Rating', Icon: IconTwentyStar, diff --git a/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/SettingsDataModelFieldAboutForm.tsx b/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/SettingsDataModelFieldAboutForm.tsx index e4c5e2141d73..01411114ce1d 100644 --- a/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/SettingsDataModelFieldAboutForm.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/SettingsDataModelFieldAboutForm.tsx @@ -22,6 +22,7 @@ type SettingsDataModelFieldAboutFormValues = z.infer< type SettingsDataModelFieldAboutFormProps = { disabled?: boolean; fieldMetadataItem?: FieldMetadataItem; + maxLength?: number; }; const StyledInputsContainer = styled.div` @@ -34,6 +35,7 @@ const StyledInputsContainer = styled.div` export const SettingsDataModelFieldAboutForm = ({ disabled, fieldMetadataItem, + maxLength, }: SettingsDataModelFieldAboutFormProps) => { const { control } = useFormContext<SettingsDataModelFieldAboutFormValues>(); @@ -63,6 +65,7 @@ export const SettingsDataModelFieldAboutForm = ({ value={value} onChange={onChange} disabled={disabled} + maxLength={maxLength} fullWidth /> )} diff --git a/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/__stories__/SettingsDataModelFieldTypeSelect.stories.tsx b/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/__stories__/SettingsDataModelFieldTypeSelect.stories.tsx index 6292d9522323..01b5164d1f8d 100644 --- a/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/__stories__/SettingsDataModelFieldTypeSelect.stories.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/fields/forms/components/__stories__/SettingsDataModelFieldTypeSelect.stories.tsx @@ -31,8 +31,8 @@ export const Disabled: Story = { }; export const WithOpenSelect: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const inputField = await canvas.findByText('Text'); @@ -49,8 +49,8 @@ export const WithExcludedFieldTypes: Story = { args: { excludedFieldTypes: [FieldMetadataType.Uuid, FieldMetadataType.Numeric], }, - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const inputField = await canvas.findByText('Text'); diff --git a/packages/twenty-front/src/modules/settings/data-model/fields/forms/relation/components/SettingsDataModelFieldRelationForm.tsx b/packages/twenty-front/src/modules/settings/data-model/fields/forms/relation/components/SettingsDataModelFieldRelationForm.tsx index f41b48bbf5fa..4415bc64aa72 100644 --- a/packages/twenty-front/src/modules/settings/data-model/fields/forms/relation/components/SettingsDataModelFieldRelationForm.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/fields/forms/relation/components/SettingsDataModelFieldRelationForm.tsx @@ -7,6 +7,7 @@ import { useFilteredObjectMetadataItems } from '@/object-metadata/hooks/useFilte import { FieldMetadataItem } from '@/object-metadata/types/FieldMetadataItem'; import { isObjectMetadataAvailableForRelation } from '@/object-metadata/utils/isObjectMetadataAvailableForRelation'; import { fieldMetadataItemSchema } from '@/object-metadata/validation-schemas/fieldMetadataItemSchema'; +import { FIELD_NAME_MAXIMUM_LENGTH } from '@/settings/data-model/constants/FieldNameMaximumLength'; import { RELATION_TYPES } from '@/settings/data-model/constants/RelationTypes'; import { useRelationSettingsFormInitialValues } from '@/settings/data-model/fields/forms/relation/hooks/useRelationSettingsFormInitialValues'; import { RelationType } from '@/settings/data-model/types/RelationType'; @@ -163,6 +164,7 @@ export const SettingsDataModelFieldRelationForm = ({ value={value} onChange={onChange} fullWidth + maxLength={FIELD_NAME_MAXIMUM_LENGTH} /> )} /> diff --git a/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectForm.tsx b/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectForm.tsx index 0f30a88b6caa..d23690fc3ff3 100644 --- a/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectForm.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectForm.tsx @@ -1,6 +1,8 @@ +import { useState } from 'react'; import { Controller, useFormContext } from 'react-hook-form'; import styled from '@emotion/styled'; import { DropResult } from '@hello-pangea/dnd'; +import { Key } from 'ts-key-enum'; import { IconPlus } from 'twenty-ui'; import { z } from 'zod'; @@ -19,6 +21,8 @@ import { CardContent } from '@/ui/layout/card/components/CardContent'; import { CardFooter } from '@/ui/layout/card/components/CardFooter'; import { DraggableItem } from '@/ui/layout/draggable-list/components/DraggableItem'; import { DraggableList } from '@/ui/layout/draggable-list/components/DraggableList'; +import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; +import { AppHotkeyScope } from '@/ui/utilities/hotkey/types/AppHotkeyScope'; import { FieldMetadataType } from '~/generated-metadata/graphql'; import { moveArrayItem } from '~/utils/array/moveArrayItem'; import { toSpliced } from '~/utils/array/toSpliced'; @@ -78,6 +82,7 @@ const StyledButton = styled(LightButton)` export const SettingsDataModelFieldSelectForm = ({ fieldMetadataItem, }: SettingsDataModelFieldSelectFormProps) => { + const [focusedOptionId, setFocusedOptionId] = useState(''); const { initialDefaultValue, initialOptions } = useSelectSettingsFormInitialValues({ fieldMetadataItem }); @@ -167,6 +172,37 @@ export const SettingsDataModelFieldSelectForm = ({ } }; + const getOptionsWithNewOption = () => { + const currentOptions = getValues('options'); + + const newOptions = [ + ...currentOptions, + generateNewSelectOption(currentOptions), + ]; + + return newOptions; + }; + + const handleAddOption = () => { + const newOptions = getOptionsWithNewOption(); + + setFormValue('options', newOptions); + }; + + useScopedHotkeys( + Key.Enter, + () => { + const newOptions = getOptionsWithNewOption(); + + setFormValue('options', newOptions); + + const lastOptionId = newOptions[newOptions.length - 1].id; + + setFocusedOptionId(lastOptionId); + }, + AppHotkeyScope.App, + ); + return ( <> <Controller @@ -197,6 +233,7 @@ export const SettingsDataModelFieldSelectForm = ({ <SettingsDataModelFieldSelectFormOptionRow key={option.id} option={option} + focused={focusedOptionId === option.id} onChange={(nextOption) => { const nextOptions = toSpliced( options, @@ -245,9 +282,7 @@ export const SettingsDataModelFieldSelectForm = ({ <StyledButton title="Add option" Icon={IconPlus} - onClick={() => - onChange([...options, generateNewSelectOption(options)]) - } + onClick={handleAddOption} /> </StyledFooter> </> diff --git a/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectFormOptionRow.tsx b/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectFormOptionRow.tsx index fdc99c137005..0cb96fc764df 100644 --- a/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectFormOptionRow.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/fields/forms/select/components/SettingsDataModelFieldSelectFormOptionRow.tsx @@ -1,4 +1,4 @@ -import { useMemo } from 'react'; +import { useEffect, useMemo, useRef } from 'react'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { @@ -13,6 +13,7 @@ import { import { v4 } from 'uuid'; import { FieldMetadataItemOption } from '@/object-metadata/types/FieldMetadataItem'; +import { OPTION_VALUE_MAXIMUM_LENGTH } from '@/settings/data-model/constants/OptionValueMaximumLength'; import { getOptionValueFromLabel } from '@/settings/data-model/fields/forms/select/utils/getOptionValueFromLabel'; import { LightIconButton } from '@/ui/input/button/components/LightIconButton'; import { TextInput } from '@/ui/input/components/TextInput'; @@ -31,6 +32,7 @@ type SettingsDataModelFieldSelectFormOptionRowProps = { onSetAsDefault?: () => void; onRemoveAsDefault?: () => void; option: FieldMetadataItemOption; + focused?: boolean; }; const StyledRow = styled.div` @@ -63,12 +65,18 @@ export const SettingsDataModelFieldSelectFormOptionRow = ({ onSetAsDefault, onRemoveAsDefault, option, + focused, }: SettingsDataModelFieldSelectFormOptionRowProps) => { + const inputRef = useRef<HTMLInputElement>(null); + const theme = useTheme(); const dropdownIds = useMemo(() => { const baseScopeId = `select-field-option-row-${v4()}`; - return { color: `${baseScopeId}-color`, actions: `${baseScopeId}-actions` }; + return { + color: `${baseScopeId}-color`, + actions: `${baseScopeId}-actions`, + }; }, []); const { closeDropdown: closeColorDropdown } = useDropdown(dropdownIds.color); @@ -76,6 +84,12 @@ export const SettingsDataModelFieldSelectFormOptionRow = ({ dropdownIds.actions, ); + useEffect(() => { + if (focused === true) { + inputRef.current?.focus(); + } + }, [focused]); + return ( <StyledRow className={className}> <IconGripVertical @@ -109,11 +123,18 @@ export const SettingsDataModelFieldSelectFormOptionRow = ({ } /> <StyledOptionInput + ref={inputRef} + disableHotkeys value={option.label} onChange={(label) => - onChange({ ...option, label, value: getOptionValueFromLabel(label) }) + onChange({ + ...option, + label, + value: getOptionValueFromLabel(label), + }) } RightIcon={isDefault ? IconCheck : undefined} + maxLength={OPTION_VALUE_MAXIMUM_LENGTH} /> <Dropdown dropdownId={dropdownIds.actions} diff --git a/packages/twenty-front/src/modules/settings/data-model/fields/preview/components/SettingsDataModelSetRecordEffect.tsx b/packages/twenty-front/src/modules/settings/data-model/fields/preview/components/SettingsDataModelSetRecordEffect.tsx index c34a07f773d3..7ef2d28c71ad 100644 --- a/packages/twenty-front/src/modules/settings/data-model/fields/preview/components/SettingsDataModelSetRecordEffect.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/fields/preview/components/SettingsDataModelSetRecordEffect.tsx @@ -1,6 +1,6 @@ import { useEffect } from 'react'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; import { ObjectRecord } from '@/object-record/types/ObjectRecord'; type SettingsDataModelSetRecordEffectProps = { @@ -10,11 +10,11 @@ type SettingsDataModelSetRecordEffectProps = { export const SettingsDataModelSetRecordEffect = ({ record, }: SettingsDataModelSetRecordEffectProps) => { - const { setRecords: setRecordsInStore } = useSetRecordInStore(); + const { upsertRecords: upsertRecordsInStore } = useUpsertRecordsInStore(); useEffect(() => { - setRecordsInStore([record]); - }, [record, setRecordsInStore]); + upsertRecordsInStore([record]); + }, [record, upsertRecordsInStore]); return null; }; diff --git a/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverview.tsx b/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverview.tsx index dd098c46bb79..e5bd42d0b4e3 100644 --- a/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverview.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverview.tsx @@ -203,7 +203,6 @@ export const SettingsDataModelOverview = () => { proOptions={{ hideAttribution: true }} > <Background /> - <IconButtonGroup className="react-flow__panel react-flow__controls bottom left" size="small" diff --git a/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverviewObject.tsx b/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverviewObject.tsx index 2c01bcb9f029..ff80c147e2fb 100644 --- a/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverviewObject.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/graph-overview/components/SettingsDataModelOverviewObject.tsx @@ -5,6 +5,7 @@ import styled from '@emotion/styled'; import { IconChevronDown, useIcons } from 'twenty-ui'; import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; +import { getObjectSlug } from '@/object-metadata/utils/getObjectSlug'; import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; import { ObjectFieldRow } from '@/settings/data-model/graph-overview/components/SettingsDataModelOverviewField'; import { SettingsDataModelObjectTypeTag } from '@/settings/data-model/objects/SettingsDataModelObjectTypeTag'; @@ -111,7 +112,7 @@ export const SettingsDataModelOverviewObject = ({ <StyledNode> <StyledHeader> <StyledObjectName onMouseEnter={() => {}} onMouseLeave={() => {}}> - <StyledObjectLink to={'/settings/objects/' + data.namePlural}> + <StyledObjectLink to={`/settings/objects/${getObjectSlug(data)}`}> {Icon && <Icon size={theme.icon.size.md} />} {capitalize(data.namePlural)} </StyledObjectLink> diff --git a/packages/twenty-front/src/modules/settings/data-model/objects/SettingsObjectCoverImage.tsx b/packages/twenty-front/src/modules/settings/data-model/objects/SettingsObjectCoverImage.tsx index 25886edaa279..23be86ff8ca0 100644 --- a/packages/twenty-front/src/modules/settings/data-model/objects/SettingsObjectCoverImage.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/objects/SettingsObjectCoverImage.tsx @@ -1,7 +1,7 @@ import styled from '@emotion/styled'; import { IconEye } from 'twenty-ui'; -import { Button } from '@/ui/input/button/components/Button'; +import { FloatingButton } from '@/ui/input/button/components/FloatingButton'; import { Card } from '@/ui/layout/card/components/Card'; import DarkCoverImage from '../assets/cover-dark.png'; @@ -30,12 +30,12 @@ export const SettingsObjectCoverImage = () => { return ( <StyledCoverImageContainer> <StyledButtonContainer> - <Button + <FloatingButton Icon={IconEye} title="Visualize" size="small" to="/settings/objects/overview" - ></Button> + /> </StyledButtonContainer> </StyledCoverImageContainer> ); diff --git a/packages/twenty-front/src/modules/settings/data-model/objects/__stories__/SettingsObjectInactiveMenuDropDown.stories.tsx b/packages/twenty-front/src/modules/settings/data-model/objects/__stories__/SettingsObjectInactiveMenuDropDown.stories.tsx index c67607f7ccb7..8401f576c858 100644 --- a/packages/twenty-front/src/modules/settings/data-model/objects/__stories__/SettingsObjectInactiveMenuDropDown.stories.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/objects/__stories__/SettingsObjectInactiveMenuDropDown.stories.tsx @@ -35,8 +35,8 @@ type Story = StoryObj<typeof SettingsObjectInactiveMenuDropDown>; export const Default: Story = {}; export const Open: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const dropdownButton = await canvas.getByRole('button'); @@ -45,8 +45,8 @@ export const Open: Story = { }; export const WithActivate: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const dropdownButton = await canvas.getByRole('button'); @@ -66,8 +66,8 @@ export const WithActivate: Story = { export const WithDelete: Story = { args: { isCustomObject: true }, - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const dropdownButton = await canvas.getByRole('button'); diff --git a/packages/twenty-front/src/modules/settings/data-model/objects/forms/components/SettingsDataModelObjectAboutForm.tsx b/packages/twenty-front/src/modules/settings/data-model/objects/forms/components/SettingsDataModelObjectAboutForm.tsx index 11ddfc5b5787..7a40b61e7ab6 100644 --- a/packages/twenty-front/src/modules/settings/data-model/objects/forms/components/SettingsDataModelObjectAboutForm.tsx +++ b/packages/twenty-front/src/modules/settings/data-model/objects/forms/components/SettingsDataModelObjectAboutForm.tsx @@ -1,9 +1,10 @@ -import { Controller, useFormContext } from 'react-hook-form'; import styled from '@emotion/styled'; +import { Controller, useFormContext } from 'react-hook-form'; import { z } from 'zod'; import { ObjectMetadataItem } from '@/object-metadata/types/ObjectMetadataItem'; import { objectMetadataItemSchema } from '@/object-metadata/validation-schemas/objectMetadataItemSchema'; +import { OBJECT_NAME_MAXIMUM_LENGTH } from '@/settings/data-model/constants/ObjectNameMaximumLength'; import { IconPicker } from '@/ui/input/components/IconPicker'; import { TextArea } from '@/ui/input/components/TextArea'; import { TextInput } from '@/ui/input/components/TextInput'; @@ -97,6 +98,7 @@ export const SettingsDataModelObjectAboutForm = ({ onChange={onChange} disabled={disabled || disableNameEdit} fullWidth + maxLength={OBJECT_NAME_MAXIMUM_LENGTH} /> )} /> diff --git a/packages/twenty-front/src/modules/settings/integrations/utils/__tests__/getSettingsIntegrationAll.test.ts b/packages/twenty-front/src/modules/settings/integrations/utils/__tests__/getSettingsIntegrationAll.test.ts new file mode 100644 index 000000000000..f8815dab20f1 --- /dev/null +++ b/packages/twenty-front/src/modules/settings/integrations/utils/__tests__/getSettingsIntegrationAll.test.ts @@ -0,0 +1,48 @@ +import { getSettingsIntegrationAll } from '../getSettingsIntegrationAll'; + +describe('getSettingsIntegrationAll', () => { + it('should return null if imageUrl is null', () => { + expect( + getSettingsIntegrationAll({ + isAirtableIntegrationActive: true, + isAirtableIntegrationEnabled: true, + isPostgresqlIntegrationActive: true, + isPostgresqlIntegrationEnabled: true, + isStripeIntegrationActive: true, + isStripeIntegrationEnabled: true, + }), + ).toStrictEqual({ + integrations: [ + { + from: { + image: '/images/integrations/airtable-logo.png', + key: 'airtable', + }, + link: '/settings/integrations/airtable', + text: 'Airtable', + type: 'Active', + }, + { + from: { + image: '/images/integrations/postgresql-logo.png', + key: 'postgresql', + }, + link: '/settings/integrations/postgresql', + text: 'PostgreSQL', + type: 'Active', + }, + { + from: { + image: '/images/integrations/stripe-logo.png', + key: 'stripe', + }, + link: '/settings/integrations/stripe', + text: 'Stripe', + type: 'Active', + }, + ], + key: 'all', + title: 'All', + }); + }); +}); diff --git a/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockColumnDefinitions.ts b/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockColumnDefinitions.ts index d6072c52b178..cb0634772c95 100644 --- a/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockColumnDefinitions.ts +++ b/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockColumnDefinitions.ts @@ -64,7 +64,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COLUMN_DEFINITIONS = ( type: FieldMetadataType.Relation, metadata: { fieldName: 'favorites', - placeHolder: 'Favorites', relationType: 'FROM_MANY_OBJECTS', relationObjectMetadataNameSingular: '', relationObjectMetadataNamePlural: '', @@ -79,7 +78,7 @@ export const SIGN_IN_BACKGROUND_MOCK_COLUMN_DEFINITIONS = ( fieldMetadataId: '20202020-ad10-4117-a039-3f04b7a5f939', label: 'Address', size: 100, - type: FieldMetadataType.Text, + type: FieldMetadataType.Address, metadata: { fieldName: 'address', placeHolder: 'Address', @@ -99,7 +98,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COLUMN_DEFINITIONS = ( type: FieldMetadataType.Relation, metadata: { fieldName: 'accountOwner', - placeHolder: 'Account Owner', relationType: 'TO_ONE_OBJECT', relationObjectMetadataNameSingular: 'workspaceMember', relationObjectMetadataNamePlural: 'workspaceMembers', @@ -117,7 +115,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COLUMN_DEFINITIONS = ( type: FieldMetadataType.Relation, metadata: { fieldName: 'people', - placeHolder: 'People', relationType: 'FROM_MANY_OBJECTS', relationObjectMetadataNameSingular: '', relationObjectMetadataNamePlural: '', @@ -135,7 +132,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COLUMN_DEFINITIONS = ( type: FieldMetadataType.Relation, metadata: { fieldName: 'attachments', - placeHolder: 'Attachments', relationType: 'FROM_MANY_OBJECTS', relationObjectMetadataNameSingular: '', relationObjectMetadataNamePlural: '', @@ -204,7 +200,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COLUMN_DEFINITIONS = ( type: FieldMetadataType.Relation, metadata: { fieldName: 'opportunities', - placeHolder: 'Opportunities', relationType: 'FROM_MANY_OBJECTS', relationObjectMetadataNameSingular: '', relationObjectMetadataNamePlural: '', @@ -239,7 +234,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COLUMN_DEFINITIONS = ( type: FieldMetadataType.Relation, metadata: { fieldName: 'activityTargets', - placeHolder: 'Activities', relationType: 'FROM_MANY_OBJECTS', relationObjectMetadataNameSingular: '', relationObjectMetadataNamePlural: '', diff --git a/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockCompanies.ts b/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockCompanies.ts index f89118b90f17..c1bd907325af 100644 --- a/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockCompanies.ts +++ b/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockCompanies.ts @@ -10,7 +10,12 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: 'OLINDA SAS. 18 rue de Navarin, 75009 Paris', + address: { + addressStreet1: 'OLINDA SAS', + addressStreet2: '18 rue de Navarin', + addressCity: 'Paris', + addressPostcode: '75009', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -115,7 +120,12 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '1600 Amphitheatre Pkwy, Mountain View, CA 94043', + address: { + addressStreet1: '1600 Amphitheatre Pkwy', + addressStreet2: 'Mountain View', + addressState: 'CA', + addressPostcode: '94043', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -314,7 +324,12 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '1 Hacker Way, Menlo Park, CA 94025', + address: { + addressStreet1: '1 Hacker Way', + addressStreet2: 'Menlo Park', + addressState: 'CA', + addressPostcode: '94025', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -400,7 +415,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ node: { __typename: 'Opportunity', id: '53f66647-0543-4cc2-9f96-95cc699960f2', - probability: '0.5', pointOfContactId: '93c72d2e-f517-42fd-80ae-14173b3b70ae', stage: 'NEW', amount: { @@ -444,7 +458,12 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '121 Albright Way, Los Gatos, CA 95032', + address: { + addressStreet1: '121 Albright Way', + addressStreet2: 'Los Gatos', + addressState: 'CA', + addressPostcode: '95032', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -492,7 +511,12 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '1 Microsoft Way, Redmond, WA 98052', + address: { + addressStreet1: '1 Microsoft Way', + addressStreet2: 'Redmond', + addressState: 'WA', + addressPostcode: '98052', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -608,7 +632,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ node: { __typename: 'Opportunity', id: '81ab695d-2f89-406f-90ea-180f433b2445', - probability: '0.5', stage: 'NEW', pointOfContactId: '9b324a88-6784-4449-afdf-dc62cb8702f2', amount: { @@ -628,7 +651,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ node: { __typename: 'Opportunity', id: '9b059852-35b1-4045-9cde-42f715148954', - probability: '0.5', stage: 'NEW', pointOfContactId: '98406e26-80f1-4dff-b570-a74942528de3', amount: { @@ -672,7 +694,11 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '42 Rue de Paradis, 75010 Paris', + address: { + addressStreet1: '42 rue de paradis', + addressCity: 'Paris', + addressPostcode: '75010', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -720,7 +746,12 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '888 Brannan Street San Francisco, CA 94103', + address: { + addressStreet1: '888 Brannan Street', + addressCity: 'San Francisco', + addressState: 'CA', + addressPostcode: '75010', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -768,7 +799,13 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '901 Fifth Avenue; Suite 1200; Seattle, WA 98164', + address: { + addressStreet1: '901 Fifth Avenue', + addressStreet2: 'Suite 1200', + addressCity: 'Seattle', + addressState: 'WA', + addressPostcode: '98164', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -816,7 +853,13 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '3790 El Camino Real, Unit 518, Palo Alto, CA 94306', + address: { + addressStreet1: '3790 El Camino Real', + addressStreet2: 'Unit 518', + addressCity: 'Palo Alto', + addressState: 'CA', + addressPostcode: '94306', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -895,7 +938,11 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '129, Samsung-ro, Yeongtong-gu, Suwon-si, Gyeonggi-do', + address: { + addressStreet1: '129, Samsung-ro', + addressStreet2: 'Yeongtong-gu, Suwon-si', + addressCity: 'Gyeonggi-do', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -974,7 +1021,13 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '576 Folsom St., Floor 3, San Francisco, CA 94105', + address: { + addressStreet1: '576 Folsom St.', + addressStreet2: 'Floor 3', + addressCity: 'San Francisco', + addressState: 'CA', + addressPostcode: '94105', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -1022,7 +1075,10 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '575 Lexington Ave 16th Floor, New York', + address: { + addressStreet1: '575 Lexington Ave 16th Floor', + addressCity: 'New York', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -1070,7 +1126,13 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ __typename: 'FavoriteConnection', edges: [], }, - address: '315 Montgomery St, 13th Fl. San Francisco, CA 94104', + address: { + addressStreet1: '315 Montgomery St', + addressStreet2: '13th Fl.', + addressCity: 'San Francisco', + addressState: 'CA', + addressPostcode: '94104', + }, accountOwner: null, people: { __typename: 'PersonConnection', @@ -1156,7 +1218,6 @@ export const SIGN_IN_BACKGROUND_MOCK_COMPANIES = [ node: { __typename: 'Opportunity', id: '7c887ee3-be10-412b-a663-16bd3c2228e1', - probability: '0.5', stage: 'NEW', pointOfContactId: '86083141-1c0e-494c-a1b6-85b1c6fefaa5', amount: { diff --git a/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockFilterDefinitions.ts b/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockFilterDefinitions.ts index 15a87498e30d..f53451ad7266 100644 --- a/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockFilterDefinitions.ts +++ b/packages/twenty-front/src/modules/sign-in-background-mock/constants/SignInBackgroundMockFilterDefinitions.ts @@ -23,7 +23,7 @@ export const SIGN_IN_BACKGROUND_MOCK_FILTER_DEFINITIONS = [ fieldMetadataId: '20202020-ad10-4117-a039-3f04b7a5f939', label: 'Address', iconName: 'IconMap', - type: 'TEXT', + type: 'ADDRESS', }, { fieldMetadataId: '20202020-0739-495d-8e70-c0807f6b2268', diff --git a/packages/twenty-front/src/modules/spreadsheet-import/components/MatchColumnSelect.tsx b/packages/twenty-front/src/modules/spreadsheet-import/components/MatchColumnSelect.tsx index f4dbc49f0bcf..9908d66a94e9 100644 --- a/packages/twenty-front/src/modules/spreadsheet-import/components/MatchColumnSelect.tsx +++ b/packages/twenty-front/src/modules/spreadsheet-import/components/MatchColumnSelect.tsx @@ -66,7 +66,9 @@ export const MatchColumnSelect = ({ const handleSearchFilterChange = useCallback( (text: string) => { setOptions( - initialOptions.filter((option) => option.label.includes(text)), + initialOptions.filter((option) => + option.label.toLowerCase().includes(text.toLowerCase()), + ), ); }, [initialOptions], @@ -123,7 +125,7 @@ export const MatchColumnSelect = ({ <DropdownMenu data-select-disable ref={dropdownContainerRef} - width={refs.domReference.current?.clientWidth} + // width={refs.domReference.current?.clientWidth} > <DropdownMenuSearchInput value={searchFilter} diff --git a/packages/twenty-front/src/modules/spreadsheet-import/steps/components/UploadFlow.tsx b/packages/twenty-front/src/modules/spreadsheet-import/steps/components/UploadFlow.tsx index 7e79ed33fd5c..052082fe2941 100644 --- a/packages/twenty-front/src/modules/spreadsheet-import/steps/components/UploadFlow.tsx +++ b/packages/twenty-front/src/modules/spreadsheet-import/steps/components/UploadFlow.tsx @@ -12,7 +12,7 @@ import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/Snac import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; import { Modal } from '@/ui/layout/modal/components/Modal'; -import { MatchColumnsStep } from './MatchColumnsStep/MatchColumnsStep'; +import { Columns, MatchColumnsStep } from './MatchColumnsStep/MatchColumnsStep'; import { SelectHeaderStep } from './SelectHeaderStep/SelectHeaderStep'; import { SelectSheetStep } from './SelectSheetStep/SelectSheetStep'; import { UploadStep } from './UploadStep/UploadStep'; @@ -52,6 +52,7 @@ export type StepState = | { type: StepType.validateData; data: any[]; + importedColumns: Columns<string>; } | { type: StepType.loading; @@ -216,6 +217,7 @@ export const UploadFlow = ({ nextStep, prevStep }: UploadFlowProps) => { setState({ type: StepType.validateData, data, + importedColumns: columns, }); setPreviousState(state); nextStep(); @@ -233,6 +235,7 @@ export const UploadFlow = ({ nextStep, prevStep }: UploadFlowProps) => { return ( <ValidationStep initialData={state.data} + importedColumns={state.importedColumns} file={uploadedFile} onSubmitStart={() => setState({ diff --git a/packages/twenty-front/src/modules/spreadsheet-import/steps/components/ValidationStep/ValidationStep.tsx b/packages/twenty-front/src/modules/spreadsheet-import/steps/components/ValidationStep/ValidationStep.tsx index e9b6ebe8bbf0..ccf38c5be32f 100644 --- a/packages/twenty-front/src/modules/spreadsheet-import/steps/components/ValidationStep/ValidationStep.tsx +++ b/packages/twenty-front/src/modules/spreadsheet-import/steps/components/ValidationStep/ValidationStep.tsx @@ -8,6 +8,10 @@ import { Heading } from '@/spreadsheet-import/components/Heading'; import { StepNavigationButton } from '@/spreadsheet-import/components/StepNavigationButton'; import { Table } from '@/spreadsheet-import/components/Table'; import { useSpreadsheetImportInternal } from '@/spreadsheet-import/hooks/useSpreadsheetImportInternal'; +import { + Columns, + ColumnType, +} from '@/spreadsheet-import/steps/components/MatchColumnsStep/MatchColumnsStep'; import { Data } from '@/spreadsheet-import/types'; import { addErrorsAndRunHooks } from '@/spreadsheet-import/utils/dataMutations'; import { useDialogManager } from '@/ui/feedback/dialog-manager/hooks/useDialogManager'; @@ -62,6 +66,7 @@ const StyledNoRowsContainer = styled.div` type ValidationStepProps<T extends string> = { initialData: Data<T>[]; + importedColumns: Columns<string>; file: File; onSubmitStart?: () => void; onBack: () => void; @@ -69,6 +74,7 @@ type ValidationStepProps<T extends string> = { export const ValidationStep = <T extends string>({ initialData, + importedColumns, file, onSubmitStart, onBack, @@ -88,6 +94,7 @@ export const ValidationStep = <T extends string>({ ReadonlySet<number | string> >(new Set()); const [filterByErrors, setFilterByErrors] = useState(false); + const [showUnmatchedColumns, setShowUnmatchedColumns] = useState(false); const updateData = useCallback( (rows: typeof data) => { @@ -127,7 +134,30 @@ export const ValidationStep = <T extends string>({ [data, updateData], ); - const columns = useMemo(() => generateColumns(fields), [fields]); + const columns = useMemo( + () => + generateColumns(fields) + .map((column) => { + const hasBeenImported = + importedColumns.filter( + (importColumn) => + (importColumn.type === ColumnType.matched && + importColumn.value === column.key) || + (importColumn.type === ColumnType.matchedSelect && + importColumn.value === column.key) || + (importColumn.type === ColumnType.matchedSelectOptions && + importColumn.value === column.key) || + (importColumn.type === ColumnType.matchedCheckbox && + importColumn.value === column.key) || + column.key === 'select-row', + ).length > 0; + + if (!hasBeenImported && !showUnmatchedColumns) return null; + return column; + }) + .filter(Boolean), + [fields, importedColumns, showUnmatchedColumns], + ); const tableData = useMemo(() => { if (filterByErrors) { @@ -212,6 +242,15 @@ export const ValidationStep = <T extends string>({ Show only rows with errors </StyledErrorToggleDescription> </StyledErrorToggle> + <StyledErrorToggle> + <Toggle + value={showUnmatchedColumns} + onChange={() => setShowUnmatchedColumns(!showUnmatchedColumns)} + /> + <StyledErrorToggleDescription> + Show unmatched columns + </StyledErrorToggleDescription> + </StyledErrorToggle> <Button Icon={IconTrash} title="Remove" diff --git a/packages/twenty-front/src/modules/spreadsheet-import/steps/components/__stories__/Validation.stories.tsx b/packages/twenty-front/src/modules/spreadsheet-import/steps/components/__stories__/Validation.stories.tsx index e2d79644ab47..1a5adabc0346 100644 --- a/packages/twenty-front/src/modules/spreadsheet-import/steps/components/__stories__/Validation.stories.tsx +++ b/packages/twenty-front/src/modules/spreadsheet-import/steps/components/__stories__/Validation.stories.tsx @@ -5,6 +5,7 @@ import { Providers } from '@/spreadsheet-import/components/Providers'; import { ValidationStep } from '@/spreadsheet-import/steps/components/ValidationStep/ValidationStep'; import { editableTableInitialData, + importedColums, mockRsiValues, } from '@/spreadsheet-import/tests/mockRsiValues'; import { DialogManagerScope } from '@/ui/feedback/dialog-manager/scopes/DialogManagerScope'; @@ -28,6 +29,7 @@ export const Default = () => ( <ValidationStep initialData={editableTableInitialData} file={file} + importedColumns={importedColums} onBack={() => Promise.resolve()} /> </ModalWrapper> diff --git a/packages/twenty-front/src/modules/spreadsheet-import/tests/mockRsiValues.ts b/packages/twenty-front/src/modules/spreadsheet-import/tests/mockRsiValues.ts index 814dae7a29e3..27dc64dabd8d 100644 --- a/packages/twenty-front/src/modules/spreadsheet-import/tests/mockRsiValues.ts +++ b/packages/twenty-front/src/modules/spreadsheet-import/tests/mockRsiValues.ts @@ -1,4 +1,5 @@ import { defaultSpreadsheetImportProps } from '@/spreadsheet-import/provider/components/SpreadsheetImport'; +import { Columns } from '@/spreadsheet-import/steps/components/MatchColumnsStep/MatchColumnsStep'; import { Fields, SpreadsheetOptions } from '@/spreadsheet-import/types'; import { sleep } from '~/utils/sleep'; @@ -88,6 +89,33 @@ const fields = [ }, ] as Fields<string>; +export const importedColums: Columns<string> = [ + { + header: 'Name', + index: 0, + type: 2, + value: 'name', + }, + { + header: 'Surname', + index: 1, + type: 2, + value: 'surname', + }, + { + header: 'Age', + index: 2, + type: 2, + value: 'age', + }, + { + header: 'Team', + index: 3, + type: 2, + value: 'team', + }, +]; + const mockComponentBehaviourForTypes = <T extends string>( props: SpreadsheetOptions<T>, ) => props; diff --git a/packages/twenty-front/src/modules/spreadsheet-import/utils/__tests__/setColumn.test.ts b/packages/twenty-front/src/modules/spreadsheet-import/utils/__tests__/setColumn.test.ts index 012dfbabfc31..e71aaddfa869 100644 --- a/packages/twenty-front/src/modules/spreadsheet-import/utils/__tests__/setColumn.test.ts +++ b/packages/twenty-front/src/modules/spreadsheet-import/utils/__tests__/setColumn.test.ts @@ -20,10 +20,13 @@ describe('setColumn', () => { value: 'oldValue', }; - it('should return a matchedSelect column if field type is "select"', () => { + it('should return a matchedSelectOptions column if field type is "select"', () => { const field = { ...defaultField, - fieldType: { type: 'select' }, + fieldType: { + type: 'select', + options: [{ value: 'John' }, { value: 'Alice' }], + }, } as Field<'Name'>; const data = [['John'], ['Alice']]; @@ -32,14 +35,16 @@ describe('setColumn', () => { expect(result).toEqual({ index: 0, header: 'Name', - type: ColumnType.matchedSelect, + type: ColumnType.matchedSelectOptions, value: 'Name', matchedOptions: [ { entry: 'John', + value: 'John', }, { entry: 'Alice', + value: 'Alice', }, ], }); diff --git a/packages/twenty-front/src/modules/spreadsheet-import/utils/setColumn.ts b/packages/twenty-front/src/modules/spreadsheet-import/utils/setColumn.ts index 0b428321872c..191cdf2081cc 100644 --- a/packages/twenty-front/src/modules/spreadsheet-import/utils/setColumn.ts +++ b/packages/twenty-front/src/modules/spreadsheet-import/utils/setColumn.ts @@ -2,6 +2,7 @@ import { Column, ColumnType, MatchColumnsStepProps, + MatchedOptions, } from '@/spreadsheet-import/steps/components/MatchColumnsStep/MatchColumnsStep'; import { Field } from '@/spreadsheet-import/types'; @@ -12,33 +13,56 @@ export const setColumn = <T extends string>( field?: Field<T>, data?: MatchColumnsStepProps<T>['data'], ): Column<T> => { - switch (field?.fieldType.type) { - case 'select': - return { - ...oldColumn, - type: ColumnType.matchedSelect, - value: field.key, - matchedOptions: uniqueEntries(data || [], oldColumn.index), - }; - case 'checkbox': - return { - index: oldColumn.index, - type: ColumnType.matchedCheckbox, - value: field.key, - header: oldColumn.header, - }; - case 'input': - return { - index: oldColumn.index, - type: ColumnType.matched, - value: field.key, - header: oldColumn.header, - }; - default: - return { - index: oldColumn.index, - header: oldColumn.header, - type: ColumnType.empty, - }; + if (field?.fieldType.type === 'select') { + const fieldOptions = field.fieldType.options; + const uniqueData = uniqueEntries( + data || [], + oldColumn.index, + ) as MatchedOptions<T>[]; + const matchedOptions = uniqueData.map((record) => { + const value = fieldOptions.find( + (fieldOption) => + fieldOption.value === record.entry || + fieldOption.label === record.entry, + )?.value; + return value + ? ({ ...record, value } as MatchedOptions<T>) + : (record as MatchedOptions<T>); + }); + const allMatched = + matchedOptions.filter((o) => o.value).length === uniqueData?.length; + + return { + ...oldColumn, + type: allMatched + ? ColumnType.matchedSelectOptions + : ColumnType.matchedSelect, + value: field.key, + matchedOptions, + }; + } + + if (field?.fieldType.type === 'checkbox') { + return { + index: oldColumn.index, + type: ColumnType.matchedCheckbox, + value: field.key, + header: oldColumn.header, + }; } + + if (field?.fieldType.type === 'input') { + return { + index: oldColumn.index, + type: ColumnType.matched, + value: field.key, + header: oldColumn.header, + }; + } + + return { + index: oldColumn.index, + header: oldColumn.header, + type: ColumnType.empty, + }; }; diff --git a/packages/twenty-front/src/modules/support/components/__stories__/SupportChat.stories.tsx b/packages/twenty-front/src/modules/support/components/__stories__/SupportChat.stories.tsx index 7da022306500..da0c12334d18 100644 --- a/packages/twenty-front/src/modules/support/components/__stories__/SupportChat.stories.tsx +++ b/packages/twenty-front/src/modules/support/components/__stories__/SupportChat.stories.tsx @@ -10,7 +10,7 @@ import { supportChatState } from '@/client-config/states/supportChatState'; import { graphqlMocks } from '~/testing/graphqlMocks'; import { mockDefaultWorkspace, - mockedUsersData, + mockedUserData, mockedWorkspaceMemberData, } from '~/testing/mock-data/users'; @@ -30,7 +30,7 @@ const meta: Meta<typeof SupportChat> = { setCurrentWorkspace(mockDefaultWorkspace); setCurrentWorkspaceMember(mockedWorkspaceMemberData); - setCurrentUser(mockedUsersData[0]); + setCurrentUser(mockedUserData); setSupportChat({ supportDriver: 'front', supportFrontChatId: '1234' }); return <Story />; diff --git a/packages/twenty-front/src/modules/ui/display/info/components/Info.tsx b/packages/twenty-front/src/modules/ui/display/info/components/Info.tsx index 822c193e8d15..e97dbbb4388d 100644 --- a/packages/twenty-front/src/modules/ui/display/info/components/Info.tsx +++ b/packages/twenty-front/src/modules/ui/display/info/components/Info.tsx @@ -1,8 +1,10 @@ import React from 'react'; +import { Link } from 'react-router-dom'; import { css, useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { IconInfoCircle } from 'twenty-ui'; +import { AppPath } from '@/types/AppPath'; import { Button } from '@/ui/input/button/components/Button'; export type InfoAccent = 'blue' | 'danger'; @@ -11,6 +13,7 @@ export type InfoProps = { text: string; buttonTitle?: string; onClick?: (event: React.MouseEvent<HTMLButtonElement>) => void; + to?: AppPath; }; const StyledTextContainer = styled.div` @@ -30,6 +33,7 @@ const StyledInfo = styled.div<Pick<InfoProps, 'accent'>>` font-weight: ${({ theme }) => theme.font.weight.medium}; justify-content: space-between; max-width: 512px; + gap: ${({ theme }) => theme.spacing(2)}; padding: ${({ theme }) => theme.spacing(2)}; ${({ theme, accent }) => { switch (accent) { @@ -46,11 +50,17 @@ const StyledInfo = styled.div<Pick<InfoProps, 'accent'>>` } }} `; + +const StyledLink = styled(Link)` + text-decoration: none; +`; + export const Info = ({ accent = 'blue', text, buttonTitle, onClick, + to, }: InfoProps) => { const theme = useTheme(); return ( @@ -59,12 +69,23 @@ export const Info = ({ <StyledIconInfoCircle size={theme.icon.size.md} /> {text} </StyledTextContainer> - {buttonTitle && onClick && ( + {buttonTitle && to && ( + <StyledLink to={to}> + <Button + title={buttonTitle} + size={'small'} + variant={'secondary'} + accent={accent} + /> + </StyledLink> + )} + {buttonTitle && onClick && !to && ( <Button title={buttonTitle} onClick={onClick} size={'small'} variant={'secondary'} + accent={accent} /> )} </StyledInfo> diff --git a/packages/twenty-front/src/modules/ui/feedback/dialog-manager/components/Dialog.tsx b/packages/twenty-front/src/modules/ui/feedback/dialog-manager/components/Dialog.tsx index 905bbe033638..90e0a4d22480 100644 --- a/packages/twenty-front/src/modules/ui/feedback/dialog-manager/components/Dialog.tsx +++ b/packages/twenty-front/src/modules/ui/feedback/dialog-manager/components/Dialog.tsx @@ -1,6 +1,6 @@ -import { useCallback } from 'react'; import styled from '@emotion/styled'; import { motion } from 'framer-motion'; +import { useCallback } from 'react'; import { Key } from 'ts-key-enum'; import { Button } from '@/ui/input/button/components/Button'; @@ -13,7 +13,7 @@ const StyledDialogOverlay = styled(motion.div)` align-items: center; background: ${({ theme }) => theme.background.overlay}; display: flex; - height: 100vh; + height: 100dvh; justify-content: center; left: 0; position: fixed; diff --git a/packages/twenty-front/src/modules/ui/field/display/components/LinksDisplay.tsx b/packages/twenty-front/src/modules/ui/field/display/components/LinksDisplay.tsx index ee58d124fbda..339859dfe62a 100644 --- a/packages/twenty-front/src/modules/ui/field/display/components/LinksDisplay.tsx +++ b/packages/twenty-front/src/modules/ui/field/display/components/LinksDisplay.tsx @@ -59,7 +59,7 @@ export const LinksDisplay = ({ value, isFocused }: LinksDisplayProps) => { ); return isFocused ? ( - <ExpandableList isChipCountDisplayed={isFocused}> + <ExpandableList isChipCountDisplayed> {links.map(({ url, label, type }, index) => type === LinkType.LinkedIn || type === LinkType.Twitter ? ( <SocialLink key={index} href={url} type={type} label={label} /> diff --git a/packages/twenty-front/src/modules/ui/field/input/components/DateInput.tsx b/packages/twenty-front/src/modules/ui/field/input/components/DateInput.tsx index 4b2131ad5ae5..1e74e447b5ac 100644 --- a/packages/twenty-front/src/modules/ui/field/input/components/DateInput.tsx +++ b/packages/twenty-front/src/modules/ui/field/input/components/DateInput.tsx @@ -33,6 +33,7 @@ export type DateInputProps = { onChange?: (newDate: Nullable<Date>) => void; isDateTimeInput?: boolean; onClear?: () => void; + onSubmit?: (newDate: Nullable<Date>) => void; }; export const DateInput = ({ @@ -44,6 +45,7 @@ export const DateInput = ({ onChange, isDateTimeInput, onClear, + onSubmit, }: DateInputProps) => { const [internalValue, setInternalValue] = useState(value); @@ -59,6 +61,11 @@ export const DateInput = ({ onClear?.(); }; + const handleMouseSelect = (newDate: Date | null) => { + setInternalValue(newDate); + onSubmit?.(newDate); + }; + const { closeDropdown } = useDropdown(MONTH_AND_YEAR_DROPDOWN_ID); const { closeDropdown: closeDropdownMonthSelect } = useDropdown( MONTH_AND_YEAR_DROPDOWN_MONTH_SELECT_ID, @@ -86,9 +93,7 @@ export const DateInput = ({ <InternalDatePicker date={internalValue ?? new Date()} onChange={handleChange} - onMouseSelect={(newDate: Date | null) => { - onEnter(newDate); - }} + onMouseSelect={handleMouseSelect} clearable={clearable ? clearable : false} isDateTimeInput={isDateTimeInput} onEnter={onEnter} diff --git a/packages/twenty-front/src/modules/ui/field/input/components/FieldTextAreaOverlay.tsx b/packages/twenty-front/src/modules/ui/field/input/components/FieldTextAreaOverlay.tsx index 6ebabfc23d5a..6be66c368982 100644 --- a/packages/twenty-front/src/modules/ui/field/input/components/FieldTextAreaOverlay.tsx +++ b/packages/twenty-front/src/modules/ui/field/input/components/FieldTextAreaOverlay.tsx @@ -2,10 +2,12 @@ import styled from '@emotion/styled'; import { OVERLAY_BACKGROUND } from 'twenty-ui'; const StyledFieldTextAreaOverlay = styled.div` + position: absolute; + top: 0; border-radius: ${({ theme }) => theme.border.radius.sm}; align-items: center; display: flex; - height: 32px; + max-height: 420px; margin: -1px; width: 100%; ${OVERLAY_BACKGROUND} diff --git a/packages/twenty-front/src/modules/ui/field/input/components/PhoneInput.tsx b/packages/twenty-front/src/modules/ui/field/input/components/PhoneInput.tsx index 2b8a406c8765..7094123dd18b 100644 --- a/packages/twenty-front/src/modules/ui/field/input/components/PhoneInput.tsx +++ b/packages/twenty-front/src/modules/ui/field/input/components/PhoneInput.tsx @@ -28,7 +28,7 @@ const StyledCustomPhoneInput = styled(ReactPhoneNumberInput)` padding: 0; .PhoneInputInput { - background: ${({ theme }) => theme.background.transparent.secondary}; + background: none; border: none; color: ${({ theme }) => theme.font.color.primary}; diff --git a/packages/twenty-front/src/modules/ui/field/input/components/RatingInput.tsx b/packages/twenty-front/src/modules/ui/field/input/components/RatingInput.tsx index 58d1fd060b06..c5dd404e59c9 100644 --- a/packages/twenty-front/src/modules/ui/field/input/components/RatingInput.tsx +++ b/packages/twenty-front/src/modules/ui/field/input/components/RatingInput.tsx @@ -1,7 +1,6 @@ -import { useState } from 'react'; -import { useTheme } from '@emotion/react'; -import styled from '@emotion/styled'; -import { IconTwentyStarFilled } from 'twenty-ui'; +import { useContext, useState } from 'react'; +import { styled } from '@linaria/react'; +import { IconTwentyStarFilled, THEME_COMMON, ThemeContext } from 'twenty-ui'; import { RATING_VALUES } from '@/object-record/record-field/meta-types/constants/RatingValues'; import { FieldRatingValue } from '@/object-record/record-field/types/FieldMetadata'; @@ -11,29 +10,38 @@ const StyledContainer = styled.div` display: flex; `; -const StyledRatingIconContainer = styled.div<{ isActive?: boolean }>` - color: ${({ isActive, theme }) => - isActive ? theme.font.color.secondary : theme.background.quaternary}; +const StyledRatingIconContainer = styled.div<{ + color: string; +}>` + color: ${({ color }) => color}; display: inline-flex; `; type RatingInputProps = { - onChange: (newValue: FieldRatingValue) => void; + onChange?: (newValue: FieldRatingValue) => void; value: FieldRatingValue; readonly?: boolean; }; +const iconSizeMd = THEME_COMMON.icon.size.md; + export const RatingInput = ({ onChange, value, readonly, }: RatingInputProps) => { - const theme = useTheme(); + const { theme } = useContext(ThemeContext); + + const activeColor = theme.font.color.secondary; + const inactiveColor = theme.background.quaternary; + const [hoveredValue, setHoveredValue] = useState<FieldRatingValue | null>( null, ); const currentValue = hoveredValue ?? value; + const selectedIndex = RATING_VALUES.indexOf(currentValue); + return ( <StyledContainer role="slider" @@ -44,17 +52,17 @@ export const RatingInput = ({ tabIndex={0} > {RATING_VALUES.map((value, index) => { - const currentIndex = RATING_VALUES.indexOf(currentValue); + const isActive = index <= selectedIndex; return ( <StyledRatingIconContainer key={index} - isActive={index <= currentIndex} - onClick={readonly ? undefined : () => onChange(value)} + color={isActive ? activeColor : inactiveColor} + onClick={readonly ? undefined : () => onChange?.(value)} onMouseEnter={readonly ? undefined : () => setHoveredValue(value)} onMouseLeave={readonly ? undefined : () => setHoveredValue(null)} > - <IconTwentyStarFilled size={theme.icon.size.md} /> + <IconTwentyStarFilled size={iconSizeMd} /> </StyledRatingIconContainer> ); })} diff --git a/packages/twenty-front/src/modules/ui/field/input/components/TextAreaInput.tsx b/packages/twenty-front/src/modules/ui/field/input/components/TextAreaInput.tsx index d14efa233d55..b9a08ef1f87c 100644 --- a/packages/twenty-front/src/modules/ui/field/input/components/TextAreaInput.tsx +++ b/packages/twenty-front/src/modules/ui/field/input/components/TextAreaInput.tsx @@ -30,6 +30,7 @@ const StyledTextArea = styled(TextareaAutosize)` display: flex; justify-content: center; resize: none; + max-height: 400px; width: calc(100% - ${({ theme }) => theme.spacing(7)}); `; diff --git a/packages/twenty-front/src/modules/ui/input/button/components/Button.tsx b/packages/twenty-front/src/modules/ui/input/button/components/Button.tsx index 375b55fdf3ed..a90daab5bb9d 100644 --- a/packages/twenty-front/src/modules/ui/input/button/components/Button.tsx +++ b/packages/twenty-front/src/modules/ui/input/button/components/Button.tsx @@ -1,8 +1,8 @@ -import React from 'react'; -import { Link } from 'react-router-dom'; import isPropValid from '@emotion/is-prop-valid'; import { css, useTheme } from '@emotion/react'; import styled from '@emotion/styled'; +import React from 'react'; +import { Link } from 'react-router-dom'; import { IconComponent, Pill } from 'twenty-ui'; export type ButtonSize = 'medium' | 'small'; @@ -27,6 +27,7 @@ export type ButtonProps = { onClick?: (event: React.MouseEvent<HTMLButtonElement>) => void; to?: string; target?: string; + dataTestId?: string; } & React.ComponentProps<'button'>; const StyledButton = styled('button', { @@ -374,6 +375,7 @@ export const Button = ({ onClick, to, target, + dataTestId, }: ButtonProps) => { const theme = useTheme(); @@ -393,6 +395,7 @@ export const Button = ({ to={to} as={to ? Link : 'button'} target={target} + data-testid={dataTestId} > {Icon && <Icon size={theme.icon.size.sm} />} {title} diff --git a/packages/twenty-front/src/modules/ui/input/button/components/FloatingButton.tsx b/packages/twenty-front/src/modules/ui/input/button/components/FloatingButton.tsx index 7ccecc0ffe6f..4f0dbcaf6c35 100644 --- a/packages/twenty-front/src/modules/ui/input/button/components/FloatingButton.tsx +++ b/packages/twenty-front/src/modules/ui/input/button/components/FloatingButton.tsx @@ -1,6 +1,6 @@ -import React from 'react'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; +import { Link } from 'react-router-dom'; import { IconComponent } from 'twenty-ui'; export type FloatingButtonSize = 'small' | 'medium'; @@ -16,12 +16,19 @@ export type FloatingButtonProps = { applyBlur?: boolean; disabled?: boolean; focus?: boolean; + to?: string; }; const StyledButton = styled.button< Pick< FloatingButtonProps, - 'size' | 'focus' | 'position' | 'applyBlur' | 'applyShadow' | 'position' + | 'size' + | 'focus' + | 'position' + | 'applyBlur' + | 'applyShadow' + | 'position' + | 'to' > >` align-items: center; @@ -87,6 +94,7 @@ const StyledButton = styled.button< &:focus { outline: none; } + text-decoration: none; `; export const FloatingButton = ({ @@ -99,6 +107,7 @@ export const FloatingButton = ({ applyShadow = true, disabled = false, focus = false, + to, }: FloatingButtonProps) => { const theme = useTheme(); return ( @@ -110,6 +119,8 @@ export const FloatingButton = ({ applyShadow={applyShadow} position={position} className={className} + to={to} + as={to ? Link : 'button'} > {Icon && <Icon size={theme.icon.size.sm} />} {title} diff --git a/packages/twenty-front/src/modules/ui/input/button/components/LightIconButton.tsx b/packages/twenty-front/src/modules/ui/input/button/components/LightIconButton.tsx index 5f03c700e0c9..868ed0c534a7 100644 --- a/packages/twenty-front/src/modules/ui/input/button/components/LightIconButton.tsx +++ b/packages/twenty-front/src/modules/ui/input/button/components/LightIconButton.tsx @@ -105,7 +105,7 @@ export const LightIconButton = ({ active={active} title={title} > - {Icon && <Icon size={theme.icon.size.sm} stroke={theme.icon.stroke.sm} />} + {Icon && <Icon size={theme.icon.size.sm} />} </StyledButton> ); }; diff --git a/packages/twenty-front/src/modules/ui/input/components/AutosizeTextInput.tsx b/packages/twenty-front/src/modules/ui/input/components/AutosizeTextInput.tsx index b827b8e08fd5..eb9afd68a398 100644 --- a/packages/twenty-front/src/modules/ui/input/components/AutosizeTextInput.tsx +++ b/packages/twenty-front/src/modules/ui/input/components/AutosizeTextInput.tsx @@ -30,6 +30,8 @@ type AutosizeTextInputProps = { value?: string; className?: string; onBlur?: () => void; + autoFocus?: boolean; + disabled?: boolean; }; const StyledContainer = styled.div` @@ -123,6 +125,8 @@ export const AutosizeTextInput = ({ value = '', className, onBlur, + autoFocus, + disabled, }: AutosizeTextInputProps) => { const [isFocused, setIsFocused] = useState(false); const [isHidden, setIsHidden] = useState( @@ -212,7 +216,9 @@ export const AutosizeTextInput = ({ {!isHidden && ( <StyledTextArea ref={textInputRef} - autoFocus={variant === AutosizeTextInputVariant.Button} + autoFocus={ + autoFocus || variant === AutosizeTextInputVariant.Button + } placeholder={placeholder ?? 'Write a comment'} maxRows={MAX_ROWS} minRows={computedMinRows} @@ -221,6 +227,7 @@ export const AutosizeTextInput = ({ onFocus={handleFocus} onBlur={handleBlur} variant={variant} + disabled={disabled} /> )} {variant === AutosizeTextInputVariant.Icon && ( diff --git a/packages/twenty-front/src/modules/ui/input/components/Radio.tsx b/packages/twenty-front/src/modules/ui/input/components/Radio.tsx index 1626e1321400..5530aaa3a332 100644 --- a/packages/twenty-front/src/modules/ui/input/components/Radio.tsx +++ b/packages/twenty-front/src/modules/ui/input/components/Radio.tsx @@ -1,8 +1,9 @@ -import * as React from 'react'; import styled from '@emotion/styled'; import { motion } from 'framer-motion'; +import * as React from 'react'; import { RGBA } from 'twenty-ui'; +import { v4 } from 'uuid'; import { RadioGroup } from './RadioGroup'; export enum RadioSize { @@ -105,6 +106,7 @@ const StyledLabel = styled.label<LabelProps>` export type RadioProps = { checked?: boolean; className?: string; + name?: string; disabled?: boolean; label?: string; labelPosition?: LabelPosition; @@ -118,6 +120,7 @@ export type RadioProps = { export const Radio = ({ checked, className, + name = 'input-radio', disabled = false, label, labelPosition = LabelPosition.Right, @@ -131,12 +134,14 @@ export const Radio = ({ onCheckedChange?.(event.target.checked); }; + const optionId = v4(); + return ( <StyledContainer className={className} labelPosition={labelPosition}> <StyledRadioInput type="radio" - id="input-radio" - name="input-radio" + id={optionId} + name={name} data-testid="input-radio" checked={checked} value={value || label} @@ -149,7 +154,7 @@ export const Radio = ({ /> {label && ( <StyledLabel - htmlFor="input-radio" + htmlFor={optionId} labelPosition={labelPosition} disabled={disabled} > diff --git a/packages/twenty-front/src/modules/ui/input/components/TextInputV2.tsx b/packages/twenty-front/src/modules/ui/input/components/TextInputV2.tsx index 19a0b92ada0a..530f27a7c1aa 100644 --- a/packages/twenty-front/src/modules/ui/input/components/TextInputV2.tsx +++ b/packages/twenty-front/src/modules/ui/input/components/TextInputV2.tsx @@ -145,6 +145,7 @@ const TextInputV2Component = ( RightIcon, LeftIcon, autoComplete, + maxLength, }: TextInputV2ComponentProps, // eslint-disable-next-line @nx/workspace-component-props-naming ref: ForwardedRef<HTMLInputElement>, @@ -182,7 +183,15 @@ const TextInputV2Component = ( onChange?.(event.target.value); }} onKeyDown={onKeyDown} - {...{ autoFocus, disabled, placeholder, required, value, LeftIcon }} + {...{ + autoFocus, + disabled, + placeholder, + required, + value, + LeftIcon, + maxLength, + }} /> <StyledTrailingIconContainer> {error && ( diff --git a/packages/twenty-front/src/modules/ui/input/components/__stories__/IconPicker.stories.tsx b/packages/twenty-front/src/modules/ui/input/components/__stories__/IconPicker.stories.tsx index 16499b234406..fc8faa2a7705 100644 --- a/packages/twenty-front/src/modules/ui/input/components/__stories__/IconPicker.stories.tsx +++ b/packages/twenty-front/src/modules/ui/input/components/__stories__/IconPicker.stories.tsx @@ -37,8 +37,8 @@ type Story = StoryObj<typeof IconPicker>; export const Default: Story = {}; export const WithOpen: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const iconPickerButton = await canvas.findByRole('button', { name: 'Click to select icon (no icon selected)', @@ -54,8 +54,8 @@ export const WithSelectedIcon: Story = { export const WithOpenAndSelectedIcon: Story = { ...WithSelectedIcon, - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const iconPickerButton = await canvas.findByRole('button', { name: 'Click to select icon (selected: IconCalendarEvent)', @@ -67,8 +67,8 @@ export const WithOpenAndSelectedIcon: Story = { export const WithSearch: Story = { ...WithSelectedIcon, - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const iconPickerButton = await canvas.findByRole('button', { name: 'Click to select icon (selected: IconCalendarEvent)', @@ -92,8 +92,8 @@ export const WithSearch: Story = { export const WithSearchAndClose: Story = { ...WithSelectedIcon, - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); let iconPickerButton = await canvas.findByRole('button', { name: 'Click to select icon (selected: IconCalendarEvent)', diff --git a/packages/twenty-front/src/modules/ui/input/components/internal/date/components/DateTimeInput.tsx b/packages/twenty-front/src/modules/ui/input/components/internal/date/components/DateTimeInput.tsx index 5d103375368a..239f8050922f 100644 --- a/packages/twenty-front/src/modules/ui/input/components/internal/date/components/DateTimeInput.tsx +++ b/packages/twenty-front/src/modules/ui/input/components/internal/date/components/DateTimeInput.tsx @@ -12,10 +12,13 @@ import { MAX_DATE } from '@/ui/input/components/internal/date/constants/MaxDate' import { MIN_DATE } from '@/ui/input/components/internal/date/constants/MinDate'; const StyledInputContainer = styled.div` - width: 100%; - display: flex; + align-items: center; border-bottom: 1px solid ${({ theme }) => theme.border.color.light}; + border-top-left-radius: ${({ theme }) => theme.border.radius.md}; + border-top-right-radius: ${({ theme }) => theme.border.radius.md}; + display: flex; height: ${({ theme }) => theme.spacing(8)}; + width: 100%; `; const StyledInput = styled.input<{ hasError?: boolean }>` @@ -23,7 +26,7 @@ const StyledInput = styled.input<{ hasError?: boolean }>` border: none; color: ${({ theme }) => theme.font.color.primary}; outline: none; - padding: 8px; + padding: 4px 8px 4px 8px; font-weight: 500; font-size: ${({ theme }) => theme.font.size.md}; width: 100%; @@ -54,11 +57,25 @@ export const DateTimeInput = ({ (date: any) => { const dateParsed = DateTime.fromJSDate(date); - const formattedDate = dateParsed.toFormat(parsingFormat); + const dateWithoutTime = DateTime.fromJSDate(date) + .toLocal() + .set({ + day: date.getUTCDate(), + month: date.getUTCMonth() + 1, + year: date.getUTCFullYear(), + hour: 0, + minute: 0, + second: 0, + millisecond: 0, + }); + + const formattedDate = isDateTimeInput + ? dateParsed.toFormat(parsingFormat) + : dateWithoutTime.toFormat(parsingFormat); return formattedDate; }, - [parsingFormat], + [parsingFormat, isDateTimeInput], ); const parseStringToDate = (str: string) => { diff --git a/packages/twenty-front/src/modules/ui/input/components/internal/date/components/InternalDatePicker.tsx b/packages/twenty-front/src/modules/ui/input/components/internal/date/components/InternalDatePicker.tsx index a45855074b33..95a1164cf79f 100644 --- a/packages/twenty-front/src/modules/ui/input/components/internal/date/components/InternalDatePicker.tsx +++ b/packages/twenty-front/src/modules/ui/input/components/internal/date/components/InternalDatePicker.tsx @@ -1,6 +1,6 @@ -import ReactDatePicker from 'react-datepicker'; import styled from '@emotion/styled'; import { DateTime } from 'luxon'; +import ReactDatePicker from 'react-datepicker'; import { Key } from 'ts-key-enum'; import { IconCalendarX, @@ -298,7 +298,7 @@ const StyledCustomDatePickerHeader = styled.div` `; export type InternalDatePickerProps = { - date: Date; + date: Date | null; onMouseSelect?: (date: Date | null) => void; onChange?: (date: Date | null) => void; clearable?: boolean; @@ -372,30 +372,47 @@ export const InternalDatePicker = ({ }; const handleChangeMonth = (month: number) => { - const newDate = new Date(date); + const newDate = new Date(internalDate); newDate.setMonth(month); onChange?.(newDate); }; const handleChangeYear = (year: number) => { - const newDate = new Date(date); + const newDate = new Date(internalDate); newDate.setFullYear(year); onChange?.(newDate); }; + const dateWithoutTime = DateTime.fromJSDate(internalDate) + .toLocal() + .set({ + day: internalDate.getUTCDate(), + month: internalDate.getUTCMonth() + 1, + year: internalDate.getUTCFullYear(), + hour: 0, + minute: 0, + second: 0, + millisecond: 0, + }) + .toJSDate(); + const dateToUse = isDateTimeInput ? date : dateWithoutTime; + return ( <StyledContainer onKeyDown={handleKeyDown}> <div className={clearable ? 'clearable ' : ''}> <ReactDatePicker open={true} - selected={internalDate} - openToDate={internalDate} + selected={dateToUse} + openToDate={isDefined(dateToUse) ? dateToUse : undefined} + disabledKeyboardNavigation onChange={(newDate) => { + newDate?.setHours(internalDate.getUTCHours()); + newDate?.setUTCMinutes(internalDate.getUTCMinutes()); onChange?.(newDate); }} customInput={ <DateTimeInput - date={internalDate} + date={dateToUse} isDateTimeInput={isDateTimeInput} onChange={onChange} /> @@ -424,13 +441,13 @@ export const InternalDatePicker = ({ options={months} disableBlur onChange={handleChangeMonth} - value={date.getMonth()} + value={internalDate.getUTCMonth()} fullWidth /> <Select dropdownId={MONTH_AND_YEAR_DROPDOWN_YEAR_SELECT_ID} onChange={handleChangeYear} - value={date.getFullYear()} + value={internalDate.getUTCFullYear()} options={years} disableBlur fullWidth @@ -450,16 +467,24 @@ export const InternalDatePicker = ({ </StyledCustomDatePickerHeader> </> )} - onSelect={(date: Date, event) => { - const dateUTC = DateTime.fromJSDate(date, { - zone: 'utc', - }).toJSDate(); - - if (event?.type === 'click') { - handleMouseSelect?.(dateUTC); - } else { - onChange?.(dateUTC); - } + onSelect={(date: Date) => { + const dateParsedWithoutTime = DateTime.fromObject( + { + day: date.getDate(), + month: date.getMonth() + 1, + year: date.getFullYear(), + hour: 0, + minute: 0, + second: 0, + }, + { zone: 'utc' }, + ).toJSDate(); + + const dateForUpdate = isDateTimeInput + ? date + : dateParsedWithoutTime; + + handleMouseSelect?.(dateForUpdate); }} /> </div> diff --git a/packages/twenty-front/src/modules/ui/input/components/internal/date/components/__stories__/DatePicker.stories.tsx b/packages/twenty-front/src/modules/ui/input/components/internal/date/components/__stories__/DatePicker.stories.tsx deleted file mode 100644 index 2d7360cd6b00..000000000000 --- a/packages/twenty-front/src/modules/ui/input/components/internal/date/components/__stories__/DatePicker.stories.tsx +++ /dev/null @@ -1,46 +0,0 @@ -import { Meta, StoryObj } from '@storybook/react'; -import { expect, userEvent, within } from '@storybook/test'; -import { ComponentDecorator } from 'twenty-ui'; - -import { InternalDatePicker } from '../InternalDatePicker'; - -const meta: Meta<typeof InternalDatePicker> = { - title: 'UI/Input/Internal/InternalDatePicker', - component: InternalDatePicker, - decorators: [ComponentDecorator], - argTypes: { - date: { control: 'date' }, - }, - args: { date: new Date('January 1, 2023 00:00:00') }, -}; - -export default meta; -type Story = StoryObj<typeof InternalDatePicker>; - -export const Default: Story = {}; - -export const WithOpenMonthSelect: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); - - const monthSelect = await canvas.findByText('January'); - - await userEvent.click(monthSelect); - - [ - 'February', - 'March', - 'April', - 'May', - 'June', - 'July', - 'August', - 'September', - 'October', - 'November', - 'December', - ].forEach((monthLabel) => - expect(canvas.getByText(monthLabel)).toBeInTheDocument(), - ); - }, -}; diff --git a/packages/twenty-front/src/modules/ui/input/components/internal/date/components/__stories__/InternalDatePicker.stories.tsx b/packages/twenty-front/src/modules/ui/input/components/internal/date/components/__stories__/InternalDatePicker.stories.tsx new file mode 100644 index 000000000000..9d107b1eb0d3 --- /dev/null +++ b/packages/twenty-front/src/modules/ui/input/components/internal/date/components/__stories__/InternalDatePicker.stories.tsx @@ -0,0 +1,80 @@ +import { useArgs } from '@storybook/preview-api'; +import { Meta, StoryObj } from '@storybook/react'; +import { expect, userEvent, within } from '@storybook/test'; +import { ComponentDecorator } from 'twenty-ui'; + +import { isDefined } from '~/utils/isDefined'; +import { InternalDatePicker } from '../InternalDatePicker'; + +const meta: Meta<typeof InternalDatePicker> = { + title: 'UI/Input/Internal/InternalDatePicker', + component: InternalDatePicker, + decorators: [ComponentDecorator], + argTypes: { + date: { control: 'date' }, + }, + render: ({ date }) => { + // eslint-disable-next-line react-hooks/rules-of-hooks + const [, updateArgs] = useArgs(); + return ( + <InternalDatePicker + date={isDefined(date) ? new Date(date) : new Date()} + onChange={(newDate) => updateArgs({ date: newDate })} + /> + ); + }, + args: { date: new Date('January 1, 2023 02:00:00') }, +}; + +export default meta; +type Story = StoryObj<typeof InternalDatePicker>; + +export const Default: Story = {}; + +export const WithOpenMonthSelect: Story = { + play: async () => { + const canvas = within(document.body); + + const monthSelect = await canvas.findByText('January'); + + await userEvent.click(monthSelect); + + [ + 'February', + 'March', + 'April', + 'May', + 'June', + 'July', + 'August', + 'September', + 'October', + 'November', + 'December', + ].forEach((monthLabel) => + expect(canvas.getByText(monthLabel)).toBeInTheDocument(), + ); + + await userEvent.click(canvas.getByText('February')); + + expect(canvas.getByText('February')).toBeInTheDocument(); + }, +}; + +export const WithOpenYearSelect: Story = { + play: async () => { + const canvas = within(document.body); + + const yearSelect = await canvas.findByText('2023'); + + await userEvent.click(yearSelect); + + ['2024', '2025', '2026'].forEach((yearLabel) => + expect(canvas.getByText(yearLabel)).toBeInTheDocument(), + ); + + await userEvent.click(canvas.getByText('2024')); + + expect(canvas.getByText('2024')).toBeInTheDocument(); + }, +}; diff --git a/packages/twenty-front/src/modules/ui/input/components/internal/phone/components/PhoneCountryPickerDropdownButton.tsx b/packages/twenty-front/src/modules/ui/input/components/internal/phone/components/PhoneCountryPickerDropdownButton.tsx index df439a4862b1..5932bccabcb5 100644 --- a/packages/twenty-front/src/modules/ui/input/components/internal/phone/components/PhoneCountryPickerDropdownButton.tsx +++ b/packages/twenty-front/src/modules/ui/input/components/internal/phone/components/PhoneCountryPickerDropdownButton.tsx @@ -21,7 +21,7 @@ type StyledDropdownButtonProps = { export const StyledDropdownButtonContainer = styled.div<StyledDropdownButtonProps>` align-items: center; - background: ${({ theme }) => theme.background.primary}; + background: none; border-radius: ${({ theme }) => theme.border.radius.xs} 0 0 ${({ theme }) => theme.border.radius.xs}; color: ${({ color }) => color ?? 'none'}; diff --git a/packages/twenty-front/src/modules/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled.tsx b/packages/twenty-front/src/modules/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled.tsx index 9d98e821f6bc..d088ec7e53a2 100644 --- a/packages/twenty-front/src/modules/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled.tsx +++ b/packages/twenty-front/src/modules/ui/layout/animated-placeholder/components/EmptyPlaceholderStyled.tsx @@ -1,6 +1,7 @@ import styled from '@emotion/styled'; +import { motion } from 'framer-motion'; -const StyledEmptyContainer = styled.div` +const StyledEmptyContainer = styled(motion.div)` align-items: center; width: 100%; height: 100%; @@ -13,6 +14,14 @@ const StyledEmptyContainer = styled.div` export { StyledEmptyContainer as AnimatedPlaceholderEmptyContainer }; +export const EMPTY_PLACEHOLDER_TRANSITION_PROPS = { + initial: { opacity: 0 }, + animate: { opacity: 1 }, + transition: { + duration: 0.15, + }, +}; + const StyledEmptyTextContainer = styled.div` align-items: center; display: flex; diff --git a/packages/twenty-front/src/modules/ui/layout/banner/components/Banner.tsx b/packages/twenty-front/src/modules/ui/layout/banner/components/Banner.tsx new file mode 100644 index 000000000000..f08b15ce0249 --- /dev/null +++ b/packages/twenty-front/src/modules/ui/layout/banner/components/Banner.tsx @@ -0,0 +1,22 @@ +import styled from '@emotion/styled'; + +const StyledBanner = styled.div` + align-items: center; + backdrop-filter: blur(5px); + background: ${({ theme }) => theme.color.blue}; + display: flex; + gap: ${({ theme }) => theme.spacing(3)}; + height: 40px; + justify-content: center; + padding: ${({ theme }) => theme.spacing(2) + ' ' + theme.spacing(3)}; + width: 100%; + color: ${({ theme }) => theme.font.color.inverted}; + font-family: Inter; + font-size: ${({ theme }) => theme.font.size.md}; + font-style: normal; + font-weight: ${({ theme }) => theme.font.weight.medium}; + line-height: 150%; + box-sizing: border-box; +`; + +export { StyledBanner as Banner }; diff --git a/packages/twenty-front/src/modules/ui/layout/banner/components/__stories__/Banner.stories.tsx b/packages/twenty-front/src/modules/ui/layout/banner/components/__stories__/Banner.stories.tsx new file mode 100644 index 000000000000..2becd60a9a1f --- /dev/null +++ b/packages/twenty-front/src/modules/ui/layout/banner/components/__stories__/Banner.stories.tsx @@ -0,0 +1,34 @@ +import { Meta, StoryObj } from '@storybook/react'; +import { ComponentDecorator, IconRefresh } from 'twenty-ui'; + +import { Button } from '@/ui/input/button/components/Button'; + +import { Banner } from '../Banner'; + +const meta: Meta<typeof Banner> = { + title: 'UI/Layout/Banner/Banner', + component: Banner, + decorators: [ComponentDecorator], + render: (args) => ( + // eslint-disable-next-line react/jsx-props-no-spreading + <Banner {...args}> + Sync lost with mailbox hello@twenty.com. Please reconnect for updates: + <Button + variant="secondary" + title="Reconnect" + Icon={IconRefresh} + size="small" + inverted + /> + </Banner> + ), + argTypes: { + as: { control: false }, + theme: { control: false }, + }, +}; + +export default meta; +type Story = StoryObj<typeof Banner>; + +export const Default: Story = {}; diff --git a/packages/twenty-front/src/modules/ui/layout/dropdown/components/Dropdown.tsx b/packages/twenty-front/src/modules/ui/layout/dropdown/components/Dropdown.tsx index 5964f92cc798..14ba3644193d 100644 --- a/packages/twenty-front/src/modules/ui/layout/dropdown/components/Dropdown.tsx +++ b/packages/twenty-front/src/modules/ui/layout/dropdown/components/Dropdown.tsx @@ -1,12 +1,13 @@ -import { useRef } from 'react'; -import { Keys } from 'react-hotkeys-hook'; import { autoUpdate, flip, + FloatingPortal, offset, Placement, useFloating, } from '@floating-ui/react'; +import { useRef } from 'react'; +import { Keys } from 'react-hotkeys-hook'; import { Key } from 'ts-key-enum'; import { DropdownScope } from '@/ui/layout/dropdown/scopes/DropdownScope'; @@ -39,6 +40,7 @@ type DropdownProps = { dropdownStrategy?: 'fixed' | 'absolute'; disableBlur?: boolean; onClickOutside?: () => void; + usePortal?: boolean; onClose?: () => void; onOpen?: () => void; }; @@ -55,6 +57,7 @@ export const Dropdown = ({ dropdownStrategy = 'absolute', dropdownOffset = { x: 0, y: 0 }, disableBlur = false, + usePortal = false, onClickOutside, onClose, onOpen, @@ -85,7 +88,7 @@ export const Dropdown = ({ }; useListenClickOutside({ - refs: [containerRef], + refs: [refs.floating], callback: () => { onClickOutside?.(); @@ -130,7 +133,20 @@ export const Dropdown = ({ onHotkeyTriggered={handleHotkeyTriggered} /> )} - {isDropdownOpen && ( + {isDropdownOpen && usePortal && ( + <FloatingPortal> + <DropdownMenu + disableBlur={disableBlur} + width={dropdownMenuWidth ?? dropdownWidth} + data-select-disable + ref={refs.setFloating} + style={floatingStyles} + > + {dropdownComponents} + </DropdownMenu> + </FloatingPortal> + )} + {isDropdownOpen && !usePortal && ( <DropdownMenu disableBlur={disableBlur} width={dropdownMenuWidth ?? dropdownWidth} diff --git a/packages/twenty-front/src/modules/ui/layout/dropdown/components/DropdownMenu.tsx b/packages/twenty-front/src/modules/ui/layout/dropdown/components/DropdownMenu.tsx index 5689b9391162..5ac402f98e8e 100644 --- a/packages/twenty-front/src/modules/ui/layout/dropdown/components/DropdownMenu.tsx +++ b/packages/twenty-front/src/modules/ui/layout/dropdown/components/DropdownMenu.tsx @@ -24,7 +24,7 @@ const StyledDropdownMenu = styled.div<{ display: flex; flex-direction: column; - z-index: 1; + z-index: 30; width: ${({ width = 160 }) => typeof width === 'number' ? `${width}px` : width}; `; diff --git a/packages/twenty-front/src/modules/ui/layout/dropdown/components/__stories__/DropdownMenu.stories.tsx b/packages/twenty-front/src/modules/ui/layout/dropdown/components/__stories__/DropdownMenu.stories.tsx index 3d939b5be6f0..69c9c7b9ccb0 100644 --- a/packages/twenty-front/src/modules/ui/layout/dropdown/components/__stories__/DropdownMenu.stories.tsx +++ b/packages/twenty-front/src/modules/ui/layout/dropdown/components/__stories__/DropdownMenu.stories.tsx @@ -1,8 +1,8 @@ -import { useState } from 'react'; import styled from '@emotion/styled'; import { Decorator, Meta, StoryObj } from '@storybook/react'; import { expect, userEvent, waitFor, within } from '@storybook/test'; import { PlayFunction } from '@storybook/types'; +import { useState } from 'react'; import { Avatar, ComponentDecorator } from 'twenty-ui'; import { Button } from '@/ui/input/button/components/Button'; @@ -76,8 +76,8 @@ export const Empty: Story = { <StyledEmptyDropdownContent data-testid="dropdown-content" /> ), }, - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const button = await canvas.findByRole('button'); userEvent.click(button); @@ -199,8 +199,8 @@ const FakeCheckableMenuItemList = ({ hasAvatar }: { hasAvatar?: boolean }) => { ); }; -const playInteraction: PlayFunction<any, any> = async ({ canvasElement }) => { - const canvas = within(canvasElement); +const playInteraction: PlayFunction<any, any> = async () => { + const canvas = within(document.body); const button = await canvas.findByRole('button'); userEvent.click(button); @@ -251,8 +251,8 @@ export const SearchWithLoadingMenu: Story = { </> ), }, - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const button = await canvas.findByRole('button'); diff --git a/packages/twenty-front/src/modules/ui/layout/expandable-list/components/ExpandableList.tsx b/packages/twenty-front/src/modules/ui/layout/expandable-list/components/ExpandableList.tsx index 6e7acc7b3ead..9f669598b61e 100644 --- a/packages/twenty-front/src/modules/ui/layout/expandable-list/components/ExpandableList.tsx +++ b/packages/twenty-front/src/modules/ui/layout/expandable-list/components/ExpandableList.tsx @@ -1,10 +1,10 @@ -import { ReactElement, useCallback, useEffect, useRef, useState } from 'react'; import styled from '@emotion/styled'; +import { ReactElement, useCallback, useEffect, useRef, useState } from 'react'; import { Chip, ChipVariant } from 'twenty-ui'; -import { AnimatedContainer } from '@/object-record/record-table/components/AnimatedContainer'; import { ExpandedListDropdown } from '@/ui/layout/expandable-list/components/ExpandedListDropdown'; import { isFirstOverflowingChildElement } from '@/ui/layout/expandable-list/utils/isFirstOverflowingChildElement'; +import { AnimatedContainer } from '@/ui/utilities/animation/components/AnimatedContainer'; import { useListenClickOutside } from '@/ui/utilities/pointer-event/hooks/useListenClickOutside'; import { isDefined } from '~/utils/isDefined'; diff --git a/packages/twenty-front/src/modules/ui/layout/hooks/__tests__/useShowAuthModal.test.tsx b/packages/twenty-front/src/modules/ui/layout/hooks/__tests__/useShowAuthModal.test.tsx index 375a5b7ff49d..270ab80b1aae 100644 --- a/packages/twenty-front/src/modules/ui/layout/hooks/__tests__/useShowAuthModal.test.tsx +++ b/packages/twenty-front/src/modules/ui/layout/hooks/__tests__/useShowAuthModal.test.tsx @@ -1,18 +1,34 @@ import { renderHook } from '@testing-library/react'; import { RecoilRoot, useSetRecoilState } from 'recoil'; -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; -import { OnboardingStatus } from '@/auth/utils/getOnboardingStatus'; +import { useIsLogged } from '@/auth/hooks/useIsLogged'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; import { AppPath } from '@/types/AppPath'; import { useShowAuthModal } from '@/ui/layout/hooks/useShowAuthModal'; import { isDefaultLayoutAuthModalVisibleState } from '@/ui/layout/states/isDefaultLayoutAuthModalVisibleState'; +import { useSubscriptionStatus } from '@/workspace/hooks/useSubscriptionStatus'; +import { OnboardingStatus, SubscriptionStatus } from '~/generated/graphql'; import { useIsMatchingLocation } from '~/hooks/useIsMatchingLocation'; -jest.mock('@/auth/hooks/useOnboardingStatus'); -const setupMockOnboardingStatus = (onboardingStatus: OnboardingStatus) => { +jest.mock('@/onboarding/hooks/useOnboardingStatus'); +const setupMockOnboardingStatus = ( + onboardingStatus: OnboardingStatus | undefined, +) => { jest.mocked(useOnboardingStatus).mockReturnValueOnce(onboardingStatus); }; +jest.mock('@/workspace/hooks/useSubscriptionStatus'); +const setupMockSubscriptionStatus = ( + subscriptionStatus: SubscriptionStatus | undefined, +) => { + jest.mocked(useSubscriptionStatus).mockReturnValueOnce(subscriptionStatus); +}; + +jest.mock('@/auth/hooks/useIsLogged'); +const setupMockIsLogged = (isLogged: boolean) => { + jest.mocked(useIsLogged).mockReturnValueOnce(isLogged); +}; + jest.mock('~/hooks/useIsMatchingLocation'); const mockUseIsMatchingLocation = jest.mocked(useIsMatchingLocation); @@ -39,264 +55,245 @@ const getResult = (isDefaultLayoutAuthModalVisible = true) => // prettier-ignore const testCases = [ - { loc: AppPath.Verify, status: OnboardingStatus.Incomplete, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingUserCreation, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingWorkspaceActivation, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingProfileCreation, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingSyncEmail, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.OngoingInviteTeam, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.Verify, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.SignInUp, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.SignInUp, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.SignInUp, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.SignInUp, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.SignInUp, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.SignInUp, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.SignInUp, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.Invite, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.Canceled, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.Unpaid, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.PastDue, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.Completed, res: true }, - { loc: AppPath.Invite, status: OnboardingStatus.CompletedWithoutSubscription, res: true }, - - { loc: AppPath.ResetPassword, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.Canceled, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.Unpaid, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.PastDue, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.Completed, res: true }, - { loc: AppPath.ResetPassword, status: OnboardingStatus.CompletedWithoutSubscription, res: true }, - - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.CreateWorkspace, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.CreateProfile, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.CreateProfile, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.SyncEmails, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.SyncEmails, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.InviteTeam, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.InviteTeam, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.PlanRequired, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.Canceled, res: true }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.PlanRequired, status: OnboardingStatus.CompletedWithoutSubscription, res: true }, - - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.PlanRequiredSuccess, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.Index, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.Index, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.Index, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.Index, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.Index, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.Index, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.Index, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.TasksPage, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.TasksPage, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.TasksPage, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.TasksPage, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.TasksPage, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.TasksPage, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.TasksPage, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.OpportunitiesPage, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.RecordIndexPage, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.RecordShowPage, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.SettingsCatchAll, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.DevelopersCatchAll, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.Impersonate, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.Impersonate, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.Impersonate, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.Impersonate, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.Impersonate, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.Impersonate, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.Impersonate, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.Authorize, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.Authorize, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.Authorize, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.Authorize, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.Authorize, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.Authorize, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.Authorize, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.NotFoundWildcard, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, - - { loc: AppPath.NotFound, status: OnboardingStatus.Incomplete, res: true }, - { loc: AppPath.NotFound, status: OnboardingStatus.Canceled, res: false }, - { loc: AppPath.NotFound, status: OnboardingStatus.Unpaid, res: false }, - { loc: AppPath.NotFound, status: OnboardingStatus.PastDue, res: false }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingUserCreation, res: true }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingWorkspaceActivation, res: true }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingProfileCreation, res: true }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingSyncEmail, res: true }, - { loc: AppPath.NotFound, status: OnboardingStatus.OngoingInviteTeam, res: true }, - { loc: AppPath.NotFound, status: OnboardingStatus.Completed, res: false }, - { loc: AppPath.NotFound, status: OnboardingStatus.CompletedWithoutSubscription, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Verify, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: false }, + { loc: AppPath.Verify, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SignInUp, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.SignInUp, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: true }, + { loc: AppPath.Invite, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.Invite, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: true }, + + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: true }, + { loc: AppPath.ResetPassword, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.ResetPassword, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: true }, + + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.CreateWorkspace, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.CreateWorkspace, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.CreateProfile, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.CreateProfile, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SyncEmails, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.SyncEmails, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.InviteTeam, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.InviteTeam, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: true }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.PlanRequired, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.PlanRequired, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.PlanRequiredSuccess, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.PlanRequiredSuccess, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.Index, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Index, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.Index, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.TasksPage, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.TasksPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.OpportunitiesPage, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.OpportunitiesPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.RecordIndexPage, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.RecordIndexPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.RecordShowPage, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.RecordShowPage, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.SettingsCatchAll, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.SettingsCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.DevelopersCatchAll, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.DevelopersCatchAll, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Impersonate, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.Impersonate, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.Authorize, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.Authorize, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.NotFoundWildcard, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.NotFoundWildcard, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, + + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: undefined, onboardingStatus: OnboardingStatus.PlanRequired, res: true }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.Canceled, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.Unpaid, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.PastDue, onboardingStatus: OnboardingStatus.Completed, res: false }, + { loc: AppPath.NotFound, isLogged: false, subscriptionStatus: undefined, onboardingStatus: undefined, res: true }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.WorkspaceActivation, res: true }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.ProfileCreation, res: true }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.SyncEmail, res: true }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.InviteTeam, res: true }, + { loc: AppPath.NotFound, isLogged: true, subscriptionStatus: SubscriptionStatus.Active, onboardingStatus: OnboardingStatus.Completed, res: false }, ]; describe('useShowAuthModal', () => { testCases.forEach((testCase) => { - it(`testCase for location ${testCase.loc} with onboardingStatus ${testCase.status} should return ${testCase.res}`, () => { - setupMockOnboardingStatus(testCase.status); + it(`testCase for location ${testCase.loc} with onboardingStatus ${testCase.onboardingStatus} should return ${testCase.res}`, () => { + setupMockOnboardingStatus(testCase.onboardingStatus); + setupMockSubscriptionStatus(testCase.subscriptionStatus); setupMockIsMatchingLocation(testCase.loc); + setupMockIsLogged(testCase.isLogged); const { result } = getResult(); if (testCase.res) { expect(result.current).toBeTruthy(); @@ -309,13 +306,17 @@ describe('useShowAuthModal', () => { describe('test with token validation loading', () => { it(`with appPath ${AppPath.Invite} and isDefaultLayoutAuthModalVisible=false`, () => { setupMockOnboardingStatus(OnboardingStatus.Completed); + setupMockSubscriptionStatus(SubscriptionStatus.Active); setupMockIsMatchingLocation(AppPath.Invite); + setupMockIsLogged(true); const { result } = getResult(false); expect(result.current).toBeFalsy(); }); it(`with appPath ${AppPath.ResetPassword} and isDefaultLayoutAuthModalVisible=false`, () => { setupMockOnboardingStatus(OnboardingStatus.Completed); + setupMockSubscriptionStatus(SubscriptionStatus.Active); setupMockIsMatchingLocation(AppPath.ResetPassword); + setupMockIsLogged(true); const { result } = getResult(false); expect(result.current).toBeFalsy(); }); @@ -323,8 +324,17 @@ describe('useShowAuthModal', () => { describe('tests should be exhaustive', () => { it('all location and onboarding status should be tested', () => { + const untestedSubscriptionStatus = [ + SubscriptionStatus.Active, + SubscriptionStatus.IncompleteExpired, + SubscriptionStatus.Paused, + SubscriptionStatus.Trialing, + ]; expect(testCases.length).toEqual( - Object.keys(AppPath).length * Object.keys(OnboardingStatus).length, + Object.keys(AppPath).length * + (Object.keys(OnboardingStatus).length + + (Object.keys(SubscriptionStatus).length - + untestedSubscriptionStatus.length)), ); }); }); diff --git a/packages/twenty-front/src/modules/ui/layout/hooks/useShowAuthModal.ts b/packages/twenty-front/src/modules/ui/layout/hooks/useShowAuthModal.ts index 47a59bc29b8d..fc241a6655f7 100644 --- a/packages/twenty-front/src/modules/ui/layout/hooks/useShowAuthModal.ts +++ b/packages/twenty-front/src/modules/ui/layout/hooks/useShowAuthModal.ts @@ -1,15 +1,20 @@ import { useMemo } from 'react'; import { useRecoilValue } from 'recoil'; -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; -import { OnboardingStatus } from '@/auth/utils/getOnboardingStatus'; +import { useIsLogged } from '@/auth/hooks/useIsLogged'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; import { AppPath } from '@/types/AppPath'; import { isDefaultLayoutAuthModalVisibleState } from '@/ui/layout/states/isDefaultLayoutAuthModalVisibleState'; +import { useSubscriptionStatus } from '@/workspace/hooks/useSubscriptionStatus'; +import { OnboardingStatus, SubscriptionStatus } from '~/generated/graphql'; import { useIsMatchingLocation } from '~/hooks/useIsMatchingLocation'; +import { isDefined } from '~/utils/isDefined'; export const useShowAuthModal = () => { const isMatchingLocation = useIsMatchingLocation(); + const isLoggedIn = useIsLogged(); const onboardingStatus = useOnboardingStatus(); + const subscriptionStatus = useSubscriptionStatus(); const isDefaultLayoutAuthModalVisible = useRecoilValue( isDefaultLayoutAuthModalVisibleState, ); @@ -24,21 +29,28 @@ export const useShowAuthModal = () => { return isDefaultLayoutAuthModalVisible; } if ( - OnboardingStatus.Incomplete === onboardingStatus || - OnboardingStatus.OngoingUserCreation === onboardingStatus || - OnboardingStatus.OngoingProfileCreation === onboardingStatus || - OnboardingStatus.OngoingWorkspaceActivation === onboardingStatus || - OnboardingStatus.OngoingSyncEmail === onboardingStatus || - OnboardingStatus.OngoingInviteTeam === onboardingStatus + !isLoggedIn || + onboardingStatus === OnboardingStatus.PlanRequired || + onboardingStatus === OnboardingStatus.ProfileCreation || + onboardingStatus === OnboardingStatus.WorkspaceActivation || + onboardingStatus === OnboardingStatus.SyncEmail || + onboardingStatus === OnboardingStatus.InviteTeam ) { return true; } if (isMatchingLocation(AppPath.PlanRequired)) { return ( - OnboardingStatus.CompletedWithoutSubscription === onboardingStatus || - OnboardingStatus.Canceled === onboardingStatus + (onboardingStatus === OnboardingStatus.Completed && + !isDefined(subscriptionStatus)) || + subscriptionStatus === SubscriptionStatus.Canceled ); } return false; - }, [isDefaultLayoutAuthModalVisible, isMatchingLocation, onboardingStatus]); + }, [ + isLoggedIn, + isDefaultLayoutAuthModalVisible, + isMatchingLocation, + onboardingStatus, + subscriptionStatus, + ]); }; diff --git a/packages/twenty-front/src/modules/ui/layout/modal/components/ConfirmationModal.tsx b/packages/twenty-front/src/modules/ui/layout/modal/components/ConfirmationModal.tsx index c6775ba1e849..402b94c51c0c 100644 --- a/packages/twenty-front/src/modules/ui/layout/modal/components/ConfirmationModal.tsx +++ b/packages/twenty-front/src/modules/ui/layout/modal/components/ConfirmationModal.tsx @@ -1,6 +1,6 @@ -import { ReactNode, useState } from 'react'; import styled from '@emotion/styled'; import { AnimatePresence, LayoutGroup } from 'framer-motion'; +import { ReactNode, useState } from 'react'; import { H1Title, H1TitleFontColor } from 'twenty-ui'; import { useDebouncedCallback } from 'use-debounce'; @@ -130,6 +130,7 @@ export const ConfirmationModal = ({ title={deleteButtonText} disabled={!isValidValue} fullWidth + dataTestId="confirmation-modal-confirm-button" /> </StyledConfirmationModal> </LayoutGroup> diff --git a/packages/twenty-front/src/modules/ui/layout/modal/components/Modal.tsx b/packages/twenty-front/src/modules/ui/layout/modal/components/Modal.tsx index 445086b52123..d9fd4f181c1f 100644 --- a/packages/twenty-front/src/modules/ui/layout/modal/components/Modal.tsx +++ b/packages/twenty-front/src/modules/ui/layout/modal/components/Modal.tsx @@ -1,4 +1,4 @@ -import React, { useEffect, useRef } from 'react'; +import { useEffect, useRef } from 'react'; import { Key } from 'ts-key-enum'; import { @@ -7,11 +7,8 @@ import { } from '@/ui/layout/modal/components/ModalLayout'; import { usePreviousHotkeyScope } from '@/ui/utilities/hotkey/hooks/usePreviousHotkeyScope'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; -import { - ClickOutsideMode, - useListenClickOutside, -} from '@/ui/utilities/pointer-event/hooks/useListenClickOutside'; +import { useListenClickOutsideV2 } from '@/ui/utilities/pointer-event/hooks/useListenClickOutsideV2'; import { ModalHotkeyScope } from './types/ModalHotkeyScope'; type ModalProps = ModalLayoutProps & { @@ -68,10 +65,10 @@ export const Modal = ({ const modalRef = useRef<HTMLDivElement>(null); - useListenClickOutside({ + useListenClickOutsideV2({ refs: [modalRef], + listenerId: 'MODAL_CLICK_OUTSIDE_LISTENER_ID', callback: () => onClose?.(), - mode: ClickOutsideMode.comparePixels, }); return isOpen ? ( diff --git a/packages/twenty-front/src/modules/ui/layout/page/BlankLayout.tsx b/packages/twenty-front/src/modules/ui/layout/page/BlankLayout.tsx index 7836cd39d70f..ab2e3eba7f97 100644 --- a/packages/twenty-front/src/modules/ui/layout/page/BlankLayout.tsx +++ b/packages/twenty-front/src/modules/ui/layout/page/BlankLayout.tsx @@ -1,12 +1,12 @@ -import { Outlet } from 'react-router-dom'; import { css, Global, useTheme } from '@emotion/react'; import styled from '@emotion/styled'; +import { Outlet } from 'react-router-dom'; const StyledLayout = styled.div` background: ${({ theme }) => theme.background.noisy}; display: flex; flex-direction: column; - height: 100vh; + height: 100dvh; position: relative; scrollbar-width: 4px; width: 100%; diff --git a/packages/twenty-front/src/modules/ui/layout/page/DefaultLayout.tsx b/packages/twenty-front/src/modules/ui/layout/page/DefaultLayout.tsx index d347a23d9976..a41e7914249e 100644 --- a/packages/twenty-front/src/modules/ui/layout/page/DefaultLayout.tsx +++ b/packages/twenty-front/src/modules/ui/layout/page/DefaultLayout.tsx @@ -1,7 +1,7 @@ -import { Outlet } from 'react-router-dom'; import { css, Global, useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { AnimatePresence, LayoutGroup, motion } from 'framer-motion'; +import { Outlet } from 'react-router-dom'; import { AuthModal } from '@/auth/components/Modal'; import { CommandMenu } from '@/command-menu/components/CommandMenu'; @@ -21,7 +21,7 @@ const StyledLayout = styled.div` background: ${({ theme }) => theme.background.noisy}; display: flex; flex-direction: column; - height: 100vh; + height: 100dvh; position: relative; scrollbar-color: ${({ theme }) => theme.border.color.medium}; scrollbar-width: 4px; diff --git a/packages/twenty-front/src/modules/ui/layout/page/PageHeader.tsx b/packages/twenty-front/src/modules/ui/layout/page/PageHeader.tsx index ae92a178319c..49e7ba382dcb 100644 --- a/packages/twenty-front/src/modules/ui/layout/page/PageHeader.tsx +++ b/packages/twenty-front/src/modules/ui/layout/page/PageHeader.tsx @@ -4,14 +4,15 @@ import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { useRecoilValue } from 'recoil'; import { - IconChevronLeft, + IconChevronDown, + IconChevronUp, IconComponent, + IconX, MOBILE_VIEWPORT, OverflowingTextWithTooltip, } from 'twenty-ui'; import { IconButton } from '@/ui/input/button/components/IconButton'; -import { UndecoratedLink } from '@/ui/navigation/link/components/UndecoratedLink'; import { NavigationDrawerCollapseButton } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerCollapseButton'; import { isNavigationDrawerOpenState } from '@/ui/navigation/states/isNavigationDrawerOpenState'; import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; @@ -53,6 +54,7 @@ const StyledLeftContainer = styled.div` const StyledTitleContainer = styled.div` display: flex; font-size: ${({ theme }) => theme.font.size.md}; + font-weight: ${({ theme }) => theme.font.weight.medium}; margin-left: ${({ theme }) => theme.spacing(1)}; max-width: 50%; `; @@ -61,6 +63,7 @@ const StyledTopBarIconStyledTitleContainer = styled.div` align-items: center; display: flex; flex: 1 0 auto; + gap: ${({ theme }) => theme.spacing(1)}; flex-direction: row; `; @@ -89,7 +92,13 @@ const StyledSkeletonLoader = () => { type PageHeaderProps = ComponentProps<'div'> & { title: string; - hasBackButton?: boolean; + hasClosePageButton?: boolean; + onClosePage?: () => void; + hasPaginationButtons?: boolean; + hasPreviousRecord?: boolean; + hasNextRecord?: boolean; + navigateToPreviousRecord?: () => void; + navigateToNextRecord?: () => void; Icon: IconComponent; children?: ReactNode; loading?: boolean; @@ -97,7 +106,13 @@ type PageHeaderProps = ComponentProps<'div'> & { export const PageHeader = ({ title, - hasBackButton, + hasClosePageButton, + onClosePage, + hasPaginationButtons, + hasPreviousRecord, + hasNextRecord, + navigateToPreviousRecord, + navigateToNextRecord, Icon, children, loading, @@ -114,19 +129,36 @@ export const PageHeader = ({ <NavigationDrawerCollapseButton direction="right" /> </StyledTopBarButtonContainer> )} - {hasBackButton && ( - <UndecoratedLink to={-1}> - <IconButton - Icon={IconChevronLeft} - size="small" - variant="tertiary" - /> - </UndecoratedLink> + {hasClosePageButton && ( + <IconButton + Icon={IconX} + size="small" + variant="tertiary" + onClick={() => onClosePage?.()} + /> )} {loading ? ( <StyledSkeletonLoader /> ) : ( <StyledTopBarIconStyledTitleContainer> + {hasPaginationButtons && ( + <> + <IconButton + Icon={IconChevronUp} + size="small" + variant="secondary" + disabled={!hasPreviousRecord} + onClick={() => navigateToPreviousRecord?.()} + /> + <IconButton + Icon={IconChevronDown} + size="small" + variant="secondary" + disabled={!hasNextRecord} + onClick={() => navigateToNextRecord?.()} + /> + </> + )} {Icon && <Icon size={theme.icon.size.md} />} <StyledTitleContainer data-testid="top-bar-title"> <OverflowingTextWithTooltip text={title} /> diff --git a/packages/twenty-front/src/modules/ui/layout/page/ShowPageContainer.tsx b/packages/twenty-front/src/modules/ui/layout/page/ShowPageContainer.tsx index 2014dc960f80..ab9f308c5cd7 100644 --- a/packages/twenty-front/src/modules/ui/layout/page/ShowPageContainer.tsx +++ b/packages/twenty-front/src/modules/ui/layout/page/ShowPageContainer.tsx @@ -8,8 +8,7 @@ const StyledOuterContainer = styled.div` display: flex; gap: ${({ theme }) => (useIsMobile() ? theme.spacing(3) : '0')}; - height: ${() => (useIsMobile() ? '100%' : '100%')}; - overflow-x: ${() => (useIsMobile() ? 'hidden' : 'auto')}; + height: 100%; width: 100%; `; diff --git a/packages/twenty-front/src/modules/ui/layout/right-drawer/components/RightDrawerRouter.tsx b/packages/twenty-front/src/modules/ui/layout/right-drawer/components/RightDrawerRouter.tsx index c9b4ab44bed7..20c931f86a81 100644 --- a/packages/twenty-front/src/modules/ui/layout/right-drawer/components/RightDrawerRouter.tsx +++ b/packages/twenty-front/src/modules/ui/layout/right-drawer/components/RightDrawerRouter.tsx @@ -2,6 +2,7 @@ import styled from '@emotion/styled'; import { useRecoilState, useRecoilValue } from 'recoil'; import { RightDrawerCalendarEvent } from '@/activities/calendar/right-drawer/components/RightDrawerCalendarEvent'; +import { RightDrawerAIChat } from '@/activities/copilot/right-drawer/components/RightDrawerAIChat'; import { RightDrawerEmailThread } from '@/activities/emails/right-drawer/components/RightDrawerEmailThread'; import { RightDrawerCreateActivity } from '@/activities/right-drawer/components/create/RightDrawerCreateActivity'; import { RightDrawerEditActivity } from '@/activities/right-drawer/components/edit/RightDrawerEditActivity'; @@ -50,6 +51,10 @@ const RIGHT_DRAWER_PAGES_CONFIG = { page: <RightDrawerRecord />, topBar: <RightDrawerTopBar page={RightDrawerPages.ViewRecord} />, }, + [RightDrawerPages.Copilot]: { + page: <RightDrawerAIChat />, + topBar: <RightDrawerTopBar page={RightDrawerPages.Copilot} />, + }, }; export const RightDrawerRouter = () => { diff --git a/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageIcons.ts b/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageIcons.ts index f1c46ba194eb..429436b66038 100644 --- a/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageIcons.ts +++ b/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageIcons.ts @@ -6,4 +6,5 @@ export const RIGHT_DRAWER_PAGE_ICONS = { [RightDrawerPages.ViewEmailThread]: 'IconMail', [RightDrawerPages.ViewCalendarEvent]: 'IconCalendarEvent', [RightDrawerPages.ViewRecord]: 'Icon123', + [RightDrawerPages.Copilot]: 'IconSparkles', }; diff --git a/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageTitles.ts b/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageTitles.ts index 6edb8fec2116..18517c80dd02 100644 --- a/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageTitles.ts +++ b/packages/twenty-front/src/modules/ui/layout/right-drawer/constants/RightDrawerPageTitles.ts @@ -6,4 +6,5 @@ export const RIGHT_DRAWER_PAGE_TITLES = { [RightDrawerPages.ViewEmailThread]: 'Email Thread', [RightDrawerPages.ViewCalendarEvent]: 'Calendar Event', [RightDrawerPages.ViewRecord]: 'Record Editor', + [RightDrawerPages.Copilot]: 'Copilot', }; diff --git a/packages/twenty-front/src/modules/ui/layout/right-drawer/types/RightDrawerPages.ts b/packages/twenty-front/src/modules/ui/layout/right-drawer/types/RightDrawerPages.ts index 487b1a16f841..e579b65448f1 100644 --- a/packages/twenty-front/src/modules/ui/layout/right-drawer/types/RightDrawerPages.ts +++ b/packages/twenty-front/src/modules/ui/layout/right-drawer/types/RightDrawerPages.ts @@ -4,4 +4,5 @@ export enum RightDrawerPages { ViewEmailThread = 'view-email-thread', ViewCalendarEvent = 'view-calendar-event', ViewRecord = 'view-record', + Copilot = 'copilot', } diff --git a/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageRightContainer.tsx b/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageRightContainer.tsx index be7c067f3ae8..2ef8c280b9a7 100644 --- a/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageRightContainer.tsx +++ b/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageRightContainer.tsx @@ -28,8 +28,7 @@ const StyledShowPageRightContainer = styled.div<{ isMobile: boolean }>` flex: 1 0 0; flex-direction: column; justify-content: start; - overflow: ${(isMobile) => (isMobile ? 'none' : 'hidden')}; - width: calc(100% + 4px); + width: 100%; `; const StyledTabListContainer = styled.div` @@ -67,9 +66,7 @@ export const ShowPageRightContainer = ({ summary, isRightDrawer = false, }: ShowPageRightContainerProps) => { - const { activeTabIdState } = useTabList( - TAB_LIST_COMPONENT_ID + isRightDrawer, - ); + const { activeTabIdState } = useTabList(TAB_LIST_COMPONENT_ID); const activeTabId = useRecoilValue(activeTabIdState); const targetObjectNameSingular = @@ -148,7 +145,7 @@ export const ShowPageRightContainer = ({ <StyledTabListContainer> <TabList loading={loading} - tabListId={TAB_LIST_COMPONENT_ID + isRightDrawer} + tabListId={TAB_LIST_COMPONENT_ID} tabs={tabs} /> </StyledTabListContainer> diff --git a/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageSummaryCard.tsx b/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageSummaryCard.tsx index 30b9085af956..6a6b26cc591c 100644 --- a/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageSummaryCard.tsx +++ b/packages/twenty-front/src/modules/ui/layout/show-page/components/ShowPageSummaryCard.tsx @@ -1,9 +1,8 @@ -import { ChangeEvent, ReactNode, useRef } from 'react'; -import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; -import { Tooltip } from 'react-tooltip'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; -import { Avatar, AvatarType } from 'twenty-ui'; +import { ChangeEvent, ReactNode, useRef } from 'react'; +import Skeleton, { SkeletonTheme } from 'react-loading-skeleton'; +import { AppTooltip, Avatar, AvatarType } from 'twenty-ui'; import { v4 as uuidV4 } from 'uuid'; import { @@ -55,17 +54,6 @@ const StyledTitle = styled.div` justify-content: center; `; -const StyledTooltip = styled(Tooltip)` - background-color: ${({ theme }) => theme.background.primary}; - box-shadow: ${({ theme }) => theme.boxShadow.light}; - - color: ${({ theme }) => theme.font.color.primary}; - - font-size: ${({ theme }) => theme.font.size.sm}; - font-weight: ${({ theme }) => theme.font.weight.regular}; - padding: ${({ theme }) => theme.spacing(2)}; -`; - const StyledAvatarWrapper = styled.div` cursor: pointer; `; @@ -136,7 +124,7 @@ export const ShowPageSummaryCard = ({ avatarUrl={logoOrAvatar} onClick={onUploadPicture ? handleAvatarClick : undefined} size="xl" - entityId={id} + placeholderColorSeed={id} placeholder={avatarPlaceholder} type={avatarType} /> @@ -153,7 +141,7 @@ export const ShowPageSummaryCard = ({ Added {beautifiedCreatedAt} </StyledDate> )} - <StyledTooltip + <AppTooltip anchorSelect={`#${dateElementId}`} content={exactCreatedAt} clickable diff --git a/packages/twenty-front/src/modules/ui/layout/tab/components/Tab.tsx b/packages/twenty-front/src/modules/ui/layout/tab/components/Tab.tsx index ff024d20f9e0..e723f0ae84ad 100644 --- a/packages/twenty-front/src/modules/ui/layout/tab/components/Tab.tsx +++ b/packages/twenty-front/src/modules/ui/layout/tab/components/Tab.tsx @@ -1,4 +1,3 @@ -import * as React from 'react'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; import { IconComponent, Pill } from 'twenty-ui'; @@ -11,7 +10,7 @@ type TabProps = { className?: string; onClick?: () => void; disabled?: boolean; - hasBetaPill?: boolean; + pill?: string; }; const StyledTab = styled.div<{ active?: boolean; disabled?: boolean }>` @@ -41,6 +40,7 @@ const StyledHover = styled.span` padding: ${({ theme }) => theme.spacing(1)}; padding-left: ${({ theme }) => theme.spacing(2)}; padding-right: ${({ theme }) => theme.spacing(2)}; + font-weight: ${({ theme }) => theme.font.weight.medium}; &:hover { background: ${({ theme }) => theme.background.tertiary}; @@ -59,7 +59,7 @@ export const Tab = ({ onClick, className, disabled, - hasBetaPill, + pill, }: TabProps) => { const theme = useTheme(); return ( @@ -73,7 +73,7 @@ export const Tab = ({ <StyledHover> {Icon && <Icon size={theme.icon.size.md} />} {title} - {hasBetaPill && <Pill label="Beta" />} + {pill && <Pill label={pill} />} </StyledHover> </StyledTab> ); diff --git a/packages/twenty-front/src/modules/ui/layout/tab/components/TabList.tsx b/packages/twenty-front/src/modules/ui/layout/tab/components/TabList.tsx index ba93d808f9d9..0cd5b4bccc11 100644 --- a/packages/twenty-front/src/modules/ui/layout/tab/components/TabList.tsx +++ b/packages/twenty-front/src/modules/ui/layout/tab/components/TabList.tsx @@ -15,7 +15,7 @@ type SingleTabProps = { id: string; hide?: boolean; disabled?: boolean; - hasBetaPill?: boolean; + pill?: string; }; type TabListProps = { @@ -62,7 +62,7 @@ export const TabList = ({ tabs, tabListId, loading }: TabListProps) => { setActiveTabId(tab.id); }} disabled={tab.disabled ?? loading} - hasBetaPill={tab.hasBetaPill} + pill={tab.pill} /> ))} </StyledContainer> diff --git a/packages/twenty-front/src/modules/ui/navigation/action-bar/components/ActionBar.tsx b/packages/twenty-front/src/modules/ui/navigation/action-bar/components/ActionBar.tsx index 97e565844f5c..1cb6a4af4f80 100644 --- a/packages/twenty-front/src/modules/ui/navigation/action-bar/components/ActionBar.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/action-bar/components/ActionBar.tsx @@ -1,15 +1,17 @@ -import { useEffect, useRef } from 'react'; import styled from '@emotion/styled'; +import { useEffect, useRef } from 'react'; import { useRecoilValue, useSetRecoilState } from 'recoil'; import { actionBarEntriesState } from '@/ui/navigation/action-bar/states/actionBarEntriesState'; import { contextMenuIsOpenState } from '@/ui/navigation/context-menu/states/contextMenuIsOpenState'; import SharedNavigationModal from '@/ui/navigation/shared/components/NavigationModal'; +import { isDefined } from '~/utils/isDefined'; import { ActionBarItem } from './ActionBarItem'; type ActionBarProps = { selectedIds?: string[]; + totalNumberOfSelectedRecords?: number; }; const StyledContainerActionBar = styled.div` @@ -40,7 +42,10 @@ const StyledLabel = styled.div` padding-right: ${({ theme }) => theme.spacing(2)}; `; -export const ActionBar = ({ selectedIds = [] }: ActionBarProps) => { +export const ActionBar = ({ + selectedIds = [], + totalNumberOfSelectedRecords, +}: ActionBarProps) => { const setContextMenuOpenState = useSetRecoilState(contextMenuIsOpenState); useEffect(() => { @@ -57,6 +62,12 @@ export const ActionBar = ({ selectedIds = [] }: ActionBarProps) => { return null; } + const selectedNumberLabel = + totalNumberOfSelectedRecords ?? selectedIds?.length; + + const showSelectedNumberLabel = + isDefined(totalNumberOfSelectedRecords) || Array.isArray(selectedIds); + return ( <> <StyledContainerActionBar @@ -64,8 +75,8 @@ export const ActionBar = ({ selectedIds = [] }: ActionBarProps) => { className="action-bar" ref={wrapperRef} > - {selectedIds && ( - <StyledLabel>{selectedIds.length} selected:</StyledLabel> + {showSelectedNumberLabel && ( + <StyledLabel>{selectedNumberLabel} selected:</StyledLabel> )} {actionBarEntries.map((item, index) => ( <ActionBarItem key={index} item={item} /> diff --git a/packages/twenty-front/src/modules/ui/navigation/action-bar/types/ActionBarEntry.ts b/packages/twenty-front/src/modules/ui/navigation/action-bar/types/ActionBarEntry.ts index f37a9b2e84cb..a276736cfb1c 100644 --- a/packages/twenty-front/src/modules/ui/navigation/action-bar/types/ActionBarEntry.ts +++ b/packages/twenty-front/src/modules/ui/navigation/action-bar/types/ActionBarEntry.ts @@ -1,12 +1,5 @@ -import { IconComponent } from 'twenty-ui'; +import { ContextMenuEntry } from '@/ui/navigation/context-menu/types/ContextMenuEntry'; -import { MenuItemAccent } from '@/ui/navigation/menu-item/types/MenuItemAccent'; - -export type ActionBarEntry = { - label: string; - Icon: IconComponent; - accent?: MenuItemAccent; - onClick?: () => void; +export type ActionBarEntry = ContextMenuEntry & { subActions?: ActionBarEntry[]; - ConfirmationModal?: JSX.Element; }; diff --git a/packages/twenty-front/src/modules/ui/navigation/context-menu/types/ContextMenuEntry.ts b/packages/twenty-front/src/modules/ui/navigation/context-menu/types/ContextMenuEntry.ts index d56820d6ebe1..416a41419f62 100644 --- a/packages/twenty-front/src/modules/ui/navigation/context-menu/types/ContextMenuEntry.ts +++ b/packages/twenty-front/src/modules/ui/navigation/context-menu/types/ContextMenuEntry.ts @@ -1,3 +1,4 @@ +import { MouseEvent, ReactNode } from 'react'; import { IconComponent } from 'twenty-ui'; import { MenuItemAccent } from '@/ui/navigation/menu-item/types/MenuItemAccent'; @@ -6,5 +7,6 @@ export type ContextMenuEntry = { label: string; Icon: IconComponent; accent?: MenuItemAccent; - onClick: () => void; + onClick?: (event?: MouseEvent<HTMLElement>) => void; + ConfirmationModal?: ReactNode; }; diff --git a/packages/twenty-front/src/modules/ui/navigation/link/components/RoundedLink.tsx b/packages/twenty-front/src/modules/ui/navigation/link/components/RoundedLink.tsx index 4b63ca09029b..61480f17fd2a 100644 --- a/packages/twenty-front/src/modules/ui/navigation/link/components/RoundedLink.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/link/components/RoundedLink.tsx @@ -1,7 +1,7 @@ -import { MouseEvent } from 'react'; +import { MouseEvent, useContext } from 'react'; import { styled } from '@linaria/react'; import { isNonEmptyString } from '@sniptt/guards'; -import { FONT_COMMON, THEME_COMMON } from 'twenty-ui'; +import { FONT_COMMON, THEME_COMMON, ThemeContext } from 'twenty-ui'; type RoundedLinkProps = { href: string; @@ -11,17 +11,23 @@ type RoundedLinkProps = { const fontSizeMd = FONT_COMMON.size.md; const spacing1 = THEME_COMMON.spacing(1); -const spacing3 = THEME_COMMON.spacing(3); +const spacing2 = THEME_COMMON.spacing(2); const spacingMultiplicator = THEME_COMMON.spacingMultiplicator; -const StyledLink = styled.a` +const StyledLink = styled.a<{ + color: string; + background: string; + backgroundHover: string; + backgroundActive: string; + border: string; +}>` align-items: center; - background-color: var(--twentycrm-background-transparent-light); - border: 1px solid var(--twentycrm-border-color-medium); + background-color: ${({ background }) => background}; + border: 1px solid ${({ border }) => border}; border-radius: 50px; - color: var(--twentycrm-font-color-primary); + color: ${({ color }) => color}; cursor: pointer; display: inline-flex; @@ -29,25 +35,39 @@ const StyledLink = styled.a` gap: ${spacing1}; - height: ${spacing3}; + height: 10px; justify-content: center; max-width: calc(100% - ${spacingMultiplicator} * 2px); - max-width: 100%; - min-width: fit-content; overflow: hidden; - padding: ${spacing1} ${spacing1}; + padding: ${spacing1} ${spacing2}; text-decoration: none; text-overflow: ellipsis; user-select: none; white-space: nowrap; + + &:hover { + background-color: ${({ backgroundHover }) => backgroundHover}; + } + + &:active { + background-color: ${({ backgroundActive }) => backgroundActive}; + } `; export const RoundedLink = ({ label, href, onClick }: RoundedLinkProps) => { + const { theme } = useContext(ThemeContext); + + const background = theme.background.transparent.lighter; + const backgroundHover = theme.background.transparent.light; + const backgroundActive = theme.background.transparent.medium; + const border = theme.border.color.strong; + const color = theme.font.color.primary; + if (!isNonEmptyString(label)) { return <></>; } @@ -64,6 +84,11 @@ export const RoundedLink = ({ label, href, onClick }: RoundedLinkProps) => { target="_blank" rel="noreferrer" onClick={handleClick} + color={color} + background={background} + backgroundHover={backgroundHover} + backgroundActive={backgroundActive} + border={border} > {label} </StyledLink> diff --git a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItem.tsx b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItem.tsx index 4e40759f9750..a8571280723c 100644 --- a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItem.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItem.tsx @@ -1,5 +1,6 @@ import { FunctionComponent, MouseEvent, ReactElement, ReactNode } from 'react'; -import { IconComponent } from 'twenty-ui'; +import { useTheme } from '@emotion/react'; +import { IconChevronRight, IconComponent } from 'twenty-ui'; import { LightIconButtonProps } from '@/ui/input/button/components/LightIconButton'; import { LightIconButtonGroup } from '@/ui/input/button/components/LightIconButtonGroup'; @@ -30,6 +31,7 @@ export type MenuItemProps = { onMouseLeave?: (event: MouseEvent<HTMLDivElement>) => void; testId?: string; text: ReactNode; + hasSubMenu?: boolean; }; export const MenuItem = ({ @@ -43,7 +45,9 @@ export const MenuItem = ({ onMouseLeave, testId, text, + hasSubMenu = false, }: MenuItemProps) => { + const theme = useTheme(); const showIconButtons = Array.isArray(iconButtons) && iconButtons.length > 0; const handleMenuItemClick = (event: MouseEvent<HTMLDivElement>) => { @@ -72,6 +76,13 @@ export const MenuItem = ({ <LightIconButtonGroup iconButtons={iconButtons} size="small" /> )} </div> + + {hasSubMenu && ( + <IconChevronRight + size={theme.icon.size.sm} + color={theme.font.color.tertiary} + /> + )} </StyledHoverableMenuItemBase> ); }; diff --git a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemDraggable.tsx b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemDraggable.tsx index 9643b8665948..98c01586b93f 100644 --- a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemDraggable.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemDraggable.tsx @@ -3,7 +3,10 @@ import { IconComponent } from 'twenty-ui'; import { LightIconButtonGroup } from '@/ui/input/button/components/LightIconButtonGroup'; import { MenuItemLeftContent } from '../internals/components/MenuItemLeftContent'; -import { StyledHoverableMenuItemBase } from '../internals/components/StyledMenuItemBase'; +import { + StyledHoverableMenuItemBase, + StyledMenuItemBase, +} from '../internals/components/StyledMenuItemBase'; import { MenuItemAccent } from '../types/MenuItemAccent'; import { MenuItemIconButton } from './MenuItem'; @@ -15,9 +18,11 @@ export type MenuItemDraggableProps = { isTooltipOpen?: boolean; onClick?: () => void; text: string; - isDragDisabled?: boolean; className?: string; isIconDisplayedOnHoverOnly?: boolean; + showGrip?: boolean; + isDragDisabled?: boolean; + isHoverDisabled?: boolean; }; export const MenuItemDraggable = ({ LeftIcon, @@ -28,9 +33,24 @@ export const MenuItemDraggable = ({ isDragDisabled = false, className, isIconDisplayedOnHoverOnly = true, + showGrip = false, + isHoverDisabled = false, }: MenuItemDraggableProps) => { const showIconButtons = Array.isArray(iconButtons) && iconButtons.length > 0; + if (isHoverDisabled) { + return ( + <StyledMenuItemBase accent={accent} isHoverBackgroundDisabled> + <MenuItemLeftContent + LeftIcon={LeftIcon} + text={text} + isDisabled={isDragDisabled} + showGrip={showGrip} + /> + </StyledMenuItemBase> + ); + } + return ( <StyledHoverableMenuItemBase onClick={onClick} @@ -41,7 +61,8 @@ export const MenuItemDraggable = ({ <MenuItemLeftContent LeftIcon={LeftIcon} text={text} - showGrip={!isDragDisabled} + isDisabled={isDragDisabled} + showGrip={showGrip} /> {showIconButtons && ( <LightIconButtonGroup diff --git a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemNavigate.tsx b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemNavigate.tsx index 07f5ff1e03a6..dd847b3048d6 100644 --- a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemNavigate.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemNavigate.tsx @@ -27,7 +27,10 @@ export const MenuItemNavigate = ({ <StyledMenuItemLeftContent> <MenuItemLeftContent LeftIcon={LeftIcon} text={text} /> </StyledMenuItemLeftContent> - <IconChevronRight size={theme.icon.size.md} /> + <IconChevronRight + size={theme.icon.size.sm} + color={theme.font.color.tertiary} + /> </StyledMenuItemBase> ); }; diff --git a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemSelectTag.tsx b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemSelectTag.tsx index ce429c8cf0cf..8c3d2ef23d9d 100644 --- a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemSelectTag.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/MenuItemSelectTag.tsx @@ -9,8 +9,9 @@ type MenuItemSelectTagProps = { selected: boolean; className?: string; onClick?: () => void; - color: ThemeColor; + color: ThemeColor | 'transparent'; text: string; + variant?: 'solid' | 'outline'; }; export const MenuItemSelectTag = ({ @@ -19,6 +20,7 @@ export const MenuItemSelectTag = ({ className, onClick, text, + variant = 'solid', }: MenuItemSelectTagProps) => { const theme = useTheme(); @@ -29,7 +31,7 @@ export const MenuItemSelectTag = ({ selected={selected} > <StyledMenuItemLeftContent> - <Tag color={color} text={text} /> + <Tag variant={variant} color={color} text={text} /> </StyledMenuItemLeftContent> {selected && <IconCheck size={theme.icon.size.sm} />} </StyledMenuItemSelect> diff --git a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/__stories__/MenuItemDraggable.stories.tsx b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/__stories__/MenuItemDraggable.stories.tsx index b7d6dba3be91..05885c00a02b 100644 --- a/packages/twenty-front/src/modules/ui/navigation/menu-item/components/__stories__/MenuItemDraggable.stories.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/menu-item/components/__stories__/MenuItemDraggable.stories.tsx @@ -104,3 +104,11 @@ export const Catalog: Story = { }, decorators: [CatalogDecorator], }; + +export const Grip: Story = { + args: { ...Default.args, showGrip: true, isDragDisabled: false }, +}; + +export const HoverDisabled: Story = { + args: { ...Default.args, isHoverDisabled: true }, +}; diff --git a/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/MenuItemLeftContent.tsx b/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/MenuItemLeftContent.tsx index ce6b2a188f4a..d2281aea7dad 100644 --- a/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/MenuItemLeftContent.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/MenuItemLeftContent.tsx @@ -8,6 +8,7 @@ import { } from 'twenty-ui'; import { + StyledDraggableItem, StyledMenuItemLabel, StyledMenuItemLeftContent, } from './StyledMenuItemBase'; @@ -16,6 +17,7 @@ type MenuItemLeftContentProps = { className?: string; LeftIcon: IconComponent | null | undefined; showGrip?: boolean; + isDisabled?: boolean; text: ReactNode; }; @@ -24,18 +26,34 @@ export const MenuItemLeftContent = ({ LeftIcon, text, showGrip = false, + isDisabled = false, }: MenuItemLeftContentProps) => { const theme = useTheme(); return ( <StyledMenuItemLeftContent className={className}> - {showGrip && ( - <IconGripVertical - size={theme.icon.size.md} - stroke={theme.icon.stroke.sm} - color={theme.font.color.extraLight} - /> - )} + {showGrip && + (isDisabled ? ( + <IconGripVertical + size={theme.icon.size.md} + stroke={theme.icon.stroke.sm} + color={ + isDisabled ? theme.font.color.extraLight : theme.font.color.light + } + /> + ) : ( + <StyledDraggableItem> + <IconGripVertical + size={theme.icon.size.md} + stroke={theme.icon.stroke.sm} + color={ + isDisabled + ? theme.font.color.extraLight + : theme.font.color.light + } + /> + </StyledDraggableItem> + ))} {LeftIcon && ( <LeftIcon size={theme.icon.size.md} stroke={theme.icon.stroke.sm} /> )} diff --git a/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/StyledMenuItemBase.tsx b/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/StyledMenuItemBase.tsx index 71cd27eae09e..a2489def35b9 100644 --- a/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/StyledMenuItemBase.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/menu-item/internals/components/StyledMenuItemBase.tsx @@ -7,6 +7,7 @@ import { MenuItemAccent } from '../../types/MenuItemAccent'; export type MenuItemBaseProps = { accent?: MenuItemAccent; isKeySelected?: boolean; + isHoverBackgroundDisabled?: boolean; }; export const StyledMenuItemBase = styled.div<MenuItemBaseProps>` @@ -31,7 +32,8 @@ export const StyledMenuItemBase = styled.div<MenuItemBaseProps>` padding: var(--vertical-padding) var(--horizontal-padding); - ${HOVER_BACKGROUND}; + ${({ isHoverBackgroundDisabled }) => + isHoverBackgroundDisabled ?? HOVER_BACKGROUND}; ${({ theme, accent }) => { switch (accent) { @@ -99,6 +101,10 @@ export const StyledMenuItemRightContent = styled.div` flex-direction: row; `; +export const StyledDraggableItem = styled.div` + cursor: grab; +`; + export const StyledHoverableMenuItemBase = styled(StyledMenuItemBase)<{ isIconDisplayedOnHoverOnly?: boolean; }>` diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/MultiWorkspaceDropdownButton.tsx b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/MultiWorkspaceDropdownButton.tsx index 0249de11c093..23d356aa4fea 100644 --- a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/MultiWorkspaceDropdownButton.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/MultiWorkspaceDropdownButton.tsx @@ -33,10 +33,12 @@ const StyledContainer = styled.div` border: 1px solid transparent; display: flex; justify-content: space-between; - height: ${({ theme }) => theme.spacing(7)}; - padding: 0 ${({ theme }) => theme.spacing(2)}; + height: ${({ theme }) => theme.spacing(5)}; + padding: calc(${({ theme }) => theme.spacing(1)} - 1px); width: 100%; + gap: ${({ theme }) => theme.spacing(1)}; + &:hover { background-color: ${({ theme }) => theme.background.transparent.lighter}; border: 1px solid ${({ theme }) => theme.border.color.medium}; @@ -46,7 +48,6 @@ const StyledContainer = styled.div` const StyledLabel = styled.div` align-items: center; display: flex; - gap: ${({ theme }) => theme.spacing(2)}; `; const StyledIconChevronDown = styled(IconChevronDown)<{ disabled?: boolean }>` diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawer.tsx b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawer.tsx index 524e87947691..406402377646 100644 --- a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawer.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawer.tsx @@ -53,7 +53,7 @@ const StyledContainer = styled.div<{ isSubMenu?: boolean }>` const StyledItemsContainer = styled.div` display: flex; flex-direction: column; - gap: ${({ theme }) => theme.spacing(8)}; + gap: ${({ theme }) => theme.spacing(3)}; margin-bottom: auto; overflow-y: auto; `; diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerCollapseButton.tsx b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerCollapseButton.tsx index 856cb3041a88..d555a43599d9 100644 --- a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerCollapseButton.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerCollapseButton.tsx @@ -14,7 +14,7 @@ const StyledCollapseButton = styled.div` color: ${({ theme }) => theme.font.color.light}; cursor: pointer; display: flex; - height: ${({ theme }) => theme.spacing(6)}; + height: ${({ theme }) => theme.spacing(5)}; justify-content: center; user-select: none; width: ${({ theme }) => theme.spacing(6)}; diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerHeader.tsx b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerHeader.tsx index eae27e8c73d3..9ba91a1c5935 100644 --- a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerHeader.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerHeader.tsx @@ -9,12 +9,14 @@ import { useIsMobile } from '@/ui/utilities/responsive/hooks/useIsMobile'; import { NavigationDrawerCollapseButton } from './NavigationDrawerCollapseButton'; -const StyledContainer = styled.div` +const StyledContainer = styled.div<{ isMultiWorkspace: boolean }>` align-items: center; display: flex; - gap: ${({ theme }) => theme.spacing(2)}; - height: ${({ theme }) => theme.spacing(6)}; - padding: ${({ theme }) => theme.spacing(1)}; + gap: ${({ theme, isMultiWorkspace }) => + !isMultiWorkspace ? theme.spacing(2) : null}; + padding: ${({ theme, isMultiWorkspace }) => + !isMultiWorkspace ? theme.spacing(1) : null}; + height: ${({ theme }) => theme.spacing(7)}; user-select: none; `; @@ -55,10 +57,11 @@ export const NavigationDrawerHeader = ({ }: NavigationDrawerHeaderProps) => { const isMobile = useIsMobile(); const workspaces = useRecoilValue(workspacesState); + const isMultiWorkspace = workspaces !== null && workspaces.length > 1; return ( - <StyledContainer> - {workspaces !== null && workspaces.length > 1 ? ( + <StyledContainer isMultiWorkspace={isMultiWorkspace}> + {isMultiWorkspace ? ( <MultiWorkspaceDropdownButton workspaces={workspaces} /> ) : ( <> diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerItem.tsx b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerItem.tsx index 8f0dd6ed2f8b..d4f33fb82a1b 100644 --- a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerItem.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerItem.tsx @@ -39,6 +39,7 @@ const StyledItem = styled('div', { align-items: center; background: ${(props) => props.active ? props.theme.background.transparent.light : 'inherit'}; + height: ${({ theme }) => theme.spacing(5)}; border: none; border-radius: ${({ theme }) => theme.border.radius.sm}; text-decoration: none; diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitle.tsx b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitle.tsx index 39052b65782d..4f8386e12d0f 100644 --- a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitle.tsx +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitle.tsx @@ -2,21 +2,34 @@ import styled from '@emotion/styled'; import { useIsPrefetchLoading } from '@/prefetch/hooks/useIsPrefetchLoading'; import { NavigationDrawerSectionTitleSkeletonLoader } from '@/ui/navigation/navigation-drawer/components/NavigationDrawerSectionTitleSkeletonLoader'; +import { isUndefinedOrNull } from '~/utils/isUndefinedOrNull'; type NavigationDrawerSectionTitleProps = { + onClick?: () => void; label: string; }; -const StyledTitle = styled.div` +const StyledTitle = styled.div<{ onClick?: () => void }>` + align-items: center; + border-radius: ${({ theme }) => theme.border.radius.sm}; color: ${({ theme }) => theme.font.color.light}; display: flex; font-size: ${({ theme }) => theme.font.size.xs}; font-weight: ${({ theme }) => theme.font.weight.semiBold}; + height: ${({ theme }) => theme.spacing(5)}; padding: ${({ theme }) => theme.spacing(1)}; - padding-top: 0; + + ${({ onClick, theme }) => + !isUndefinedOrNull(onClick) + ? `&:hover { + cursor: pointer; + background-color:${theme.background.transparent.light}; + }` + : ''} `; export const NavigationDrawerSectionTitle = ({ + onClick, label, }: NavigationDrawerSectionTitleProps) => { const loading = useIsPrefetchLoading(); @@ -24,5 +37,5 @@ export const NavigationDrawerSectionTitle = ({ if (loading) { return <NavigationDrawerSectionTitleSkeletonLoader />; } - return <StyledTitle>{label}</StyledTitle>; + return <StyledTitle onClick={onClick}>{label}</StyledTitle>; }; diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/hooks/useNavigationSection.ts b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/hooks/useNavigationSection.ts new file mode 100644 index 000000000000..b497319a09ac --- /dev/null +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/hooks/useNavigationSection.ts @@ -0,0 +1,60 @@ +import { useRecoilCallback } from 'recoil'; + +import { isNavigationSectionOpenComponentState } from '@/ui/navigation/navigation-drawer/states/isNavigationSectionOpenComponentState'; +import { extractComponentState } from '@/ui/utilities/state/component-state/utils/extractComponentState'; + +export const useNavigationSection = (scopeId: string) => { + const closeNavigationSection = useRecoilCallback( + ({ set }) => + () => { + set( + isNavigationSectionOpenComponentState({ + scopeId, + }), + false, + ); + }, + [scopeId], + ); + + const openNavigationSection = useRecoilCallback( + ({ set }) => + () => { + set( + isNavigationSectionOpenComponentState({ + scopeId, + }), + true, + ); + }, + [scopeId], + ); + + const toggleNavigationSection = useRecoilCallback( + ({ snapshot }) => + () => { + const isNavigationSectionOpen = snapshot + .getLoadable(isNavigationSectionOpenComponentState({ scopeId })) + .getValue(); + + if (isNavigationSectionOpen) { + closeNavigationSection(); + } else { + openNavigationSection(); + } + }, + [closeNavigationSection, openNavigationSection, scopeId], + ); + + const isNavigationSectionOpenState = extractComponentState( + isNavigationSectionOpenComponentState, + scopeId, + ); + + return { + isNavigationSectionOpenState, + closeNavigationSection, + openNavigationSection, + toggleNavigationSection, + }; +}; diff --git a/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/states/isNavigationSectionOpenComponentState.ts b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/states/isNavigationSectionOpenComponentState.ts new file mode 100644 index 000000000000..cd0fec7a0c99 --- /dev/null +++ b/packages/twenty-front/src/modules/ui/navigation/navigation-drawer/states/isNavigationSectionOpenComponentState.ts @@ -0,0 +1,9 @@ +import { createComponentState } from '@/ui/utilities/state/component-state/utils/createComponentState'; +import { localStorageEffect } from '~/utils/recoil-effects'; + +export const isNavigationSectionOpenComponentState = + createComponentState<boolean>({ + key: 'isNavigationSectionOpenComponentState', + defaultValue: true, + effects: [localStorageEffect()], + }); diff --git a/packages/twenty-front/src/modules/object-record/record-table/components/AnimatedContainer.tsx b/packages/twenty-front/src/modules/ui/utilities/animation/components/AnimatedContainer.tsx similarity index 65% rename from packages/twenty-front/src/modules/object-record/record-table/components/AnimatedContainer.tsx rename to packages/twenty-front/src/modules/ui/utilities/animation/components/AnimatedContainer.tsx index c821abb3794f..581e03cb7bc7 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/components/AnimatedContainer.tsx +++ b/packages/twenty-front/src/modules/ui/utilities/animation/components/AnimatedContainer.tsx @@ -1,20 +1,17 @@ -import React from 'react'; -import styled from '@emotion/styled'; import { motion } from 'framer-motion'; - -const StyledAnimatedChipContainer = styled(motion.div)``; +import React from 'react'; export const AnimatedContainer = ({ children, }: { children: React.ReactNode; }) => ( - <StyledAnimatedChipContainer + <motion.div initial={{ opacity: 0 }} animate={{ opacity: 1 }} transition={{ duration: 0.1 }} whileHover={{ scale: 1.04 }} > {children} - </StyledAnimatedChipContainer> + </motion.div> ); diff --git a/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useContextScopeId.test.tsx b/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useContextScopeId.test.tsx index 83ad28a3f371..b14092d3b311 100644 --- a/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useContextScopeId.test.tsx +++ b/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useContextScopeId.test.tsx @@ -20,8 +20,8 @@ describe('useContextScopeId', () => { ), }); - const scopedId = result.current; - expect(scopedId).toBe(mockedContextValue); + const scopeId = result.current; + expect(scopeId).toBe(mockedContextValue); }); it('Should throw an error when used outside of the specified context', () => { diff --git a/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useRecoilScopeId.test.tsx b/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useRecoilScopeId.test.tsx index 80e690b24ab8..a8d5eafd19d8 100644 --- a/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useRecoilScopeId.test.tsx +++ b/packages/twenty-front/src/modules/ui/utilities/recoil-scope/hooks/__tests__/useRecoilScopeId.test.tsx @@ -20,8 +20,8 @@ describe('useRecoilScopeId', () => { ), }); - const scopedId = result.current; - expect(scopedId).toBe(mockedContextValue); + const scopeId = result.current; + expect(scopeId).toBe(mockedContextValue); }); it('Should throw an error when used outside of the specified context', () => { diff --git a/packages/twenty-front/src/modules/ui/utilities/state/component-state/hooks/useRecoilComponentValue.ts b/packages/twenty-front/src/modules/ui/utilities/state/component-state/hooks/useRecoilComponentValue.ts new file mode 100644 index 000000000000..d72ee3cae3f2 --- /dev/null +++ b/packages/twenty-front/src/modules/ui/utilities/state/component-state/hooks/useRecoilComponentValue.ts @@ -0,0 +1,29 @@ +import { useRecoilValue } from 'recoil'; + +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; +import { getScopeIdOrUndefinedFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdOrUndefinedFromComponentId'; +import { ComponentState } from '@/ui/utilities/state/component-state/types/ComponentState'; + +export const useRecoilComponentValue = <StateType>( + componentState: ComponentState<StateType>, + componentId?: string, +) => { + const componentContext = (window as any).componentContextStateMap?.get( + componentState.key, + ); + + if (!componentContext) { + throw new Error( + `Component context for key "${componentState.key}" is not defined`, + ); + } + + const internalComponentId = useAvailableScopeIdOrThrow( + componentContext, + getScopeIdOrUndefinedFromComponentId(componentId), + ); + + return useRecoilValue( + componentState.atomFamily({ scopeId: internalComponentId }), + ); +}; diff --git a/packages/twenty-front/src/modules/ui/utilities/state/component-state/hooks/useSetRecoilComponentState.ts b/packages/twenty-front/src/modules/ui/utilities/state/component-state/hooks/useSetRecoilComponentState.ts new file mode 100644 index 000000000000..938d699b3f51 --- /dev/null +++ b/packages/twenty-front/src/modules/ui/utilities/state/component-state/hooks/useSetRecoilComponentState.ts @@ -0,0 +1,29 @@ +import { useSetRecoilState } from 'recoil'; + +import { useAvailableScopeIdOrThrow } from '@/ui/utilities/recoil-scope/scopes-internal/hooks/useAvailableScopeId'; +import { getScopeIdOrUndefinedFromComponentId } from '@/ui/utilities/recoil-scope/utils/getScopeIdOrUndefinedFromComponentId'; +import { ComponentState } from '@/ui/utilities/state/component-state/types/ComponentState'; + +export const useSetRecoilComponentState = <StateType>( + componentState: ComponentState<StateType>, + componentId?: string, +) => { + const componentContext = (window as any).componentContextStateMap?.get( + componentState.key, + ); + + if (!componentContext) { + throw new Error( + `Component context for key "${componentState.key}" is not defined`, + ); + } + + const internalComponentId = useAvailableScopeIdOrThrow( + componentContext, + getScopeIdOrUndefinedFromComponentId(componentId), + ); + + return useSetRecoilState( + componentState.atomFamily({ scopeId: internalComponentId }), + ); +}; diff --git a/packages/twenty-front/src/modules/ui/utilities/state/component-state/types/ComponentState.ts b/packages/twenty-front/src/modules/ui/utilities/state/component-state/types/ComponentState.ts new file mode 100644 index 000000000000..8fc43e8a291b --- /dev/null +++ b/packages/twenty-front/src/modules/ui/utilities/state/component-state/types/ComponentState.ts @@ -0,0 +1,8 @@ +import { RecoilState } from 'recoil'; + +import { ComponentStateKey } from '@/ui/utilities/state/component-state/types/ComponentStateKey'; + +export type ComponentState<StateType> = { + key: string; + atomFamily: (componentStateKey: ComponentStateKey) => RecoilState<StateType>; +}; diff --git a/packages/twenty-front/src/modules/ui/utilities/state/component-state/utils/createComponentState.ts b/packages/twenty-front/src/modules/ui/utilities/state/component-state/utils/createComponentState.ts index 2e2cd8ada9d9..ffbd9dd7eb48 100644 --- a/packages/twenty-front/src/modules/ui/utilities/state/component-state/utils/createComponentState.ts +++ b/packages/twenty-front/src/modules/ui/utilities/state/component-state/utils/createComponentState.ts @@ -1,16 +1,21 @@ -import { atomFamily } from 'recoil'; +import { AtomEffect, atomFamily } from 'recoil'; import { ComponentStateKey } from '@/ui/utilities/state/component-state/types/ComponentStateKey'; +type CreateComponentStateType<ValueType> = { + key: string; + defaultValue: ValueType; + effects?: AtomEffect<ValueType>[]; +}; + export const createComponentState = <ValueType>({ key, defaultValue, -}: { - key: string; - defaultValue: ValueType; -}) => { + effects, +}: CreateComponentStateType<ValueType>) => { return atomFamily<ValueType, ComponentStateKey>({ key, default: defaultValue, + effects: effects, }); }; diff --git a/packages/twenty-front/src/modules/ui/utilities/state/component-state/utils/createComponentStateV2.ts b/packages/twenty-front/src/modules/ui/utilities/state/component-state/utils/createComponentStateV2.ts new file mode 100644 index 000000000000..1fda1db8210e --- /dev/null +++ b/packages/twenty-front/src/modules/ui/utilities/state/component-state/utils/createComponentStateV2.ts @@ -0,0 +1,37 @@ +import { AtomEffect, atomFamily } from 'recoil'; + +import { ScopeInternalContext } from '@/ui/utilities/recoil-scope/scopes-internal/types/ScopeInternalContext'; +import { ComponentState } from '@/ui/utilities/state/component-state/types/ComponentState'; +import { ComponentStateKey } from '@/ui/utilities/state/component-state/types/ComponentStateKey'; +import { isDefined } from '~/utils/isDefined'; + +type CreateComponentStateV2Type<ValueType> = { + key: string; + defaultValue: ValueType; + componentContext?: ScopeInternalContext<any> | null; + effects?: AtomEffect<ValueType>[]; +}; + +export const createComponentStateV2 = <ValueType>({ + key, + defaultValue, + componentContext, + effects, +}: CreateComponentStateV2Type<ValueType>): ComponentState<ValueType> => { + if (isDefined(componentContext)) { + if (!isDefined((window as any).componentContextStateMap)) { + (window as any).componentContextStateMap = new Map(); + } + + (window as any).componentContextStateMap.set(key, componentContext); + } + + return { + key, + atomFamily: atomFamily<ValueType, ComponentStateKey>({ + key, + default: defaultValue, + effects: effects, + }), + }; +}; diff --git a/packages/twenty-front/src/modules/users/components/UserChip.tsx b/packages/twenty-front/src/modules/users/components/UserChip.tsx index 82beb9da53be..5f4fbc94ebf3 100644 --- a/packages/twenty-front/src/modules/users/components/UserChip.tsx +++ b/packages/twenty-front/src/modules/users/components/UserChip.tsx @@ -1,4 +1,4 @@ -import { EntityChip } from 'twenty-ui'; +import { AvatarChip } from 'twenty-ui'; import { getImageAbsoluteURIOrBase64 } from '~/utils/image/getImageAbsoluteURIOrBase64'; @@ -9,8 +9,8 @@ export type UserChipProps = { }; export const UserChip = ({ id, name, avatarUrl }: UserChipProps) => ( - <EntityChip - entityId={id} + <AvatarChip + placeholderColorSeed={id} name={name} avatarType="rounded" avatarUrl={getImageAbsoluteURIOrBase64(avatarUrl) || ''} diff --git a/packages/twenty-front/src/modules/users/graphql/fragments/userQueryFragment.ts b/packages/twenty-front/src/modules/users/graphql/fragments/userQueryFragment.ts index 32d358c64571..14dcf557a294 100644 --- a/packages/twenty-front/src/modules/users/graphql/fragments/userQueryFragment.ts +++ b/packages/twenty-front/src/modules/users/graphql/fragments/userQueryFragment.ts @@ -8,7 +8,7 @@ export const USER_QUERY_FRAGMENT = gql` email canImpersonate supportUserHash - onboardingStep + onboardingStatus workspaceMember { id name { @@ -26,7 +26,6 @@ export const USER_QUERY_FRAGMENT = gql` domainName inviteHash allowImpersonation - subscriptionStatus activationStatus featureFlags { id @@ -40,6 +39,7 @@ export const USER_QUERY_FRAGMENT = gql` status interval } + workspaceMembersCount } workspaces { workspace { diff --git a/packages/twenty-front/src/modules/views/components/EditableFilterChip.tsx b/packages/twenty-front/src/modules/views/components/EditableFilterChip.tsx index c7f727d8b0f6..86c413798fa5 100644 --- a/packages/twenty-front/src/modules/views/components/EditableFilterChip.tsx +++ b/packages/twenty-front/src/modules/views/components/EditableFilterChip.tsx @@ -16,8 +16,8 @@ export const EditableFilterChip = ({ const { getIcon } = useIcons(); return ( <SortOrFilterChip - key={viewFilter.fieldMetadataId} - testId={viewFilter.fieldMetadataId} + key={viewFilter.id} + testId={viewFilter.id} labelKey={viewFilter.definition.label} labelValue={`${getOperandLabelShort(viewFilter.operand)} ${ viewFilter.displayValue diff --git a/packages/twenty-front/src/modules/views/components/EditableFilterDropdownButton.tsx b/packages/twenty-front/src/modules/views/components/EditableFilterDropdownButton.tsx index ee7f4dedb1f7..ee5f17968ad3 100644 --- a/packages/twenty-front/src/modules/views/components/EditableFilterDropdownButton.tsx +++ b/packages/twenty-front/src/modules/views/components/EditableFilterDropdownButton.tsx @@ -62,13 +62,13 @@ export const EditableFilterDropdownButton = ({ const handleRemove = () => { closeDropdown(); - removeCombinedViewFilter(viewFilter.fieldMetadataId); + removeCombinedViewFilter(viewFilter.id); }; const handleDropdownClickOutside = useCallback(() => { - const { value, fieldMetadataId } = viewFilter; + const { id: fieldId, value } = viewFilter; if (!value) { - removeCombinedViewFilter(fieldMetadataId); + removeCombinedViewFilter(fieldId); } }, [viewFilter, removeCombinedViewFilter]); diff --git a/packages/twenty-front/src/modules/views/components/ViewBar.tsx b/packages/twenty-front/src/modules/views/components/ViewBar.tsx index d090b1f8c382..3c600daec05f 100644 --- a/packages/twenty-front/src/modules/views/components/ViewBar.tsx +++ b/packages/twenty-front/src/modules/views/components/ViewBar.tsx @@ -41,7 +41,6 @@ export const ViewBar = ({ const sortDropdownId = 'view-sort'; const loading = useIsPrefetchLoading(); - if (!objectNamePlural) { return; } diff --git a/packages/twenty-front/src/modules/views/components/ViewBarDetails.tsx b/packages/twenty-front/src/modules/views/components/ViewBarDetails.tsx index 6a0a2882dd4a..e51137f74461 100644 --- a/packages/twenty-front/src/modules/views/components/ViewBarDetails.tsx +++ b/packages/twenty-front/src/modules/views/components/ViewBarDetails.tsx @@ -153,19 +153,17 @@ export const ViewBarDetails = ({ availableFilterDefinitions, ).map((viewFilter) => ( <ObjectFilterDropdownScope - key={viewFilter.fieldMetadataId} - filterScopeId={viewFilter.fieldMetadataId} + key={viewFilter.id} + filterScopeId={viewFilter.id} > - <DropdownScope dropdownScopeId={viewFilter.fieldMetadataId}> - <ViewBarFilterEffect - filterDropdownId={viewFilter.fieldMetadataId} - /> + <DropdownScope dropdownScopeId={viewFilter.id}> + <ViewBarFilterEffect filterDropdownId={viewFilter.id} /> <EditableFilterDropdownButton viewFilter={viewFilter} hotkeyScope={{ - scope: viewFilter.fieldMetadataId, + scope: viewFilter.id, }} - viewFilterDropdownId={viewFilter.fieldMetadataId} + viewFilterDropdownId={viewFilter.id} /> </DropdownScope> </ObjectFilterDropdownScope> diff --git a/packages/twenty-front/src/modules/views/components/ViewBarEffect.tsx b/packages/twenty-front/src/modules/views/components/ViewBarEffect.tsx index 518ed1590b46..a23b73922d8b 100644 --- a/packages/twenty-front/src/modules/views/components/ViewBarEffect.tsx +++ b/packages/twenty-front/src/modules/views/components/ViewBarEffect.tsx @@ -1,5 +1,5 @@ -import { useEffect, useState } from 'react'; import { isUndefined } from '@sniptt/guards'; +import { useEffect, useState } from 'react'; import { useRecoilValue } from 'recoil'; import { useViewStates } from '@/views/hooks/internal/useViewStates'; diff --git a/packages/twenty-front/src/modules/views/components/ViewBarFilterEffect.tsx b/packages/twenty-front/src/modules/views/components/ViewBarFilterEffect.tsx index 1f19e48103da..544e3466fbb9 100644 --- a/packages/twenty-front/src/modules/views/components/ViewBarFilterEffect.tsx +++ b/packages/twenty-front/src/modules/views/components/ViewBarFilterEffect.tsx @@ -6,6 +6,7 @@ import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/ import { Filter } from '@/object-record/object-filter-dropdown/types/Filter'; import { useViewStates } from '@/views/hooks/internal/useViewStates'; import { useCombinedViewFilters } from '@/views/hooks/useCombinedViewFilters'; +import { useGetCurrentView } from '@/views/hooks/useGetCurrentView'; import { isDefined } from '~/utils/isDefined'; type ViewBarFilterEffectProps = { @@ -15,11 +16,12 @@ type ViewBarFilterEffectProps = { export const ViewBarFilterEffect = ({ filterDropdownId, }: ViewBarFilterEffectProps) => { - const { availableFilterDefinitionsState, unsavedToUpsertViewFiltersState } = - useViewStates(); + const { availableFilterDefinitionsState } = useViewStates(); const { upsertCombinedViewFilter } = useCombinedViewFilters(); + const { currentViewWithCombinedFiltersAndSorts } = useGetCurrentView(); + const availableFilterDefinitions = useRecoilValue( availableFilterDefinitionsState, ); @@ -51,47 +53,41 @@ export const ViewBarFilterEffect = ({ upsertCombinedViewFilter, ]); - const unsavedToUpsertViewFilters = useRecoilValue( - unsavedToUpsertViewFiltersState, - ); - useEffect(() => { if (filterDefinitionUsedInDropdown?.type === 'RELATION') { - const viewFilterUsedInDropdown = unsavedToUpsertViewFilters.find( - (filter) => - filter.fieldMetadataId === - filterDefinitionUsedInDropdown.fieldMetadataId, - ); + const viewFilterUsedInDropdown = + currentViewWithCombinedFiltersAndSorts?.viewFilters.find( + (filter) => + filter.fieldMetadataId === + filterDefinitionUsedInDropdown?.fieldMetadataId, + ); - const viewFilterSelectedRecordIds = isNonEmptyString( + const viewFilterSelectedRecords = isNonEmptyString( viewFilterUsedInDropdown?.value, ) ? JSON.parse(viewFilterUsedInDropdown.value) : []; - - setObjectFilterDropdownSelectedRecordIds(viewFilterSelectedRecordIds); + setObjectFilterDropdownSelectedRecordIds(viewFilterSelectedRecords); } else if (filterDefinitionUsedInDropdown?.type === 'SELECT') { - const viewFilterUsedInDropdown = unsavedToUpsertViewFilters.find( - (filter) => - filter.fieldMetadataId === - filterDefinitionUsedInDropdown.fieldMetadataId, - ); + const viewFilterUsedInDropdown = + currentViewWithCombinedFiltersAndSorts?.viewFilters.find( + (filter) => + filter.fieldMetadataId === + filterDefinitionUsedInDropdown?.fieldMetadataId, + ); - const viewFilterSelectedOptionValues = isNonEmptyString( + const viewFilterSelectedRecords = isNonEmptyString( viewFilterUsedInDropdown?.value, ) ? JSON.parse(viewFilterUsedInDropdown.value) : []; - - setObjectFilterDropdownSelectedOptionValues( - viewFilterSelectedOptionValues, - ); + setObjectFilterDropdownSelectedOptionValues(viewFilterSelectedRecords); } }, [ filterDefinitionUsedInDropdown, setObjectFilterDropdownSelectedRecordIds, setObjectFilterDropdownSelectedOptionValues, - unsavedToUpsertViewFilters, + currentViewWithCombinedFiltersAndSorts, ]); return <></>; diff --git a/packages/twenty-front/src/modules/views/components/ViewFieldsVisibilityDropdownSection.tsx b/packages/twenty-front/src/modules/views/components/ViewFieldsVisibilityDropdownSection.tsx index 9bc30f5a2359..edbbdf3c0a8d 100644 --- a/packages/twenty-front/src/modules/views/components/ViewFieldsVisibilityDropdownSection.tsx +++ b/packages/twenty-front/src/modules/views/components/ViewFieldsVisibilityDropdownSection.tsx @@ -19,7 +19,6 @@ import { DraggableItem } from '@/ui/layout/draggable-list/components/DraggableIt import { DraggableList } from '@/ui/layout/draggable-list/components/DraggableList'; import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer'; import { StyledDropdownMenuSubheader } from '@/ui/layout/dropdown/components/StyledDropdownMenuSubheader'; -import { MenuItem } from '@/ui/navigation/menu-item/components/MenuItem'; import { MenuItemDraggable } from '@/ui/navigation/menu-item/components/MenuItemDraggable'; import { useListenClickOutside } from '@/ui/utilities/pointer-event/hooks/useListenClickOutside'; import { groupArrayItemsBy } from '~/utils/array/groupArrayItemsBy'; @@ -101,13 +100,17 @@ export const ViewFieldsVisibilityDropdownSection = ({ )} <DropdownMenuItemsContainer> {nonDraggableItems.map((field, fieldIndex) => ( - <MenuItem + <MenuItemDraggable key={field.fieldMetadataId} LeftIcon={getIcon(field.iconName)} iconButtons={getIconButtons(fieldIndex, field)} isTooltipOpen={openToolTipIndex === fieldIndex} text={field.label} className={`${title}-fixed-item-tooltip-anchor-${fieldIndex}`} + accent={'placeholder'} + isHoverDisabled={field.isVisible} + showGrip + isDragDisabled /> ))} {!!draggableItems.length && ( @@ -131,6 +134,7 @@ export const ViewFieldsVisibilityDropdownSection = ({ isTooltipOpen={openToolTipIndex === fieldIndex} text={field.label} className={`${title}-draggable-item-tooltip-anchor-${fieldIndex}`} + showGrip /> } /> diff --git a/packages/twenty-front/src/modules/views/hooks/useCombinedViewFilters.ts b/packages/twenty-front/src/modules/views/hooks/useCombinedViewFilters.ts index e3eceeed3ebd..042ef8f33ab2 100644 --- a/packages/twenty-front/src/modules/views/hooks/useCombinedViewFilters.ts +++ b/packages/twenty-front/src/modules/views/hooks/useCombinedViewFilters.ts @@ -1,5 +1,4 @@ import { useRecoilCallback } from 'recoil'; -import { v4 } from 'uuid'; import { Filter } from '@/object-record/object-filter-dropdown/types/Filter'; import { getSnapshotValue } from '@/ui/utilities/recoil-scope/utils/getSnapshotValue'; @@ -43,13 +42,11 @@ export const useCombinedViewFilters = (viewBarComponentId?: string) => { } const matchingFilterInCurrentView = currentView.viewFilters.find( - (viewFilter) => - viewFilter.fieldMetadataId === upsertedFilter.fieldMetadataId, + (viewFilter) => viewFilter.id === upsertedFilter.id, ); const matchingFilterInUnsavedFilters = unsavedToUpsertViewFilters.find( - (viewFilter) => - viewFilter.fieldMetadataId === upsertedFilter.fieldMetadataId, + (viewFilter) => viewFilter.id === upsertedFilter.id, ); if (isDefined(matchingFilterInUnsavedFilters)) { @@ -81,7 +78,6 @@ export const useCombinedViewFilters = (viewBarComponentId?: string) => { ...unsavedToUpsertViewFilters, { ...upsertedFilter, - id: v4(), __typename: 'ViewFilter', } satisfies ViewFilter, ]); @@ -95,7 +91,7 @@ export const useCombinedViewFilters = (viewBarComponentId?: string) => { ); const removeCombinedViewFilter = useRecoilCallback( ({ snapshot, set }) => - async (fieldMetadataId: string) => { + async (fieldId: string) => { const unsavedToUpsertViewFilters = getSnapshotValue( snapshot, unsavedToUpsertViewFiltersState, @@ -119,18 +115,18 @@ export const useCombinedViewFilters = (viewBarComponentId?: string) => { } const matchingFilterInCurrentView = currentView.viewFilters.find( - (viewFilter) => viewFilter.fieldMetadataId === fieldMetadataId, + (viewFilter) => viewFilter.id === fieldId, ); const matchingFilterInUnsavedFilters = unsavedToUpsertViewFilters.find( - (viewFilter) => viewFilter.fieldMetadataId === fieldMetadataId, + (viewFilter) => viewFilter.id === fieldId, ); if (isDefined(matchingFilterInUnsavedFilters)) { set( unsavedToUpsertViewFiltersState, unsavedToUpsertViewFilters.filter( - (viewFilter) => viewFilter.fieldMetadataId !== fieldMetadataId, + (viewFilter) => viewFilter.id !== fieldId, ), ); } diff --git a/packages/twenty-front/src/modules/views/types/ViewFilterOperand.ts b/packages/twenty-front/src/modules/views/types/ViewFilterOperand.ts index fa1a27d68179..025d0085d49d 100644 --- a/packages/twenty-front/src/modules/views/types/ViewFilterOperand.ts +++ b/packages/twenty-front/src/modules/views/types/ViewFilterOperand.ts @@ -6,4 +6,6 @@ export enum ViewFilterOperand { GreaterThan = 'greaterThan', Contains = 'contains', DoesNotContain = 'doesNotContain', + IsEmpty = 'isEmpty', + IsNotEmpty = 'isNotEmpty', } diff --git a/packages/twenty-front/src/modules/views/utils/__tests__/viewMapFunctions.test.ts b/packages/twenty-front/src/modules/views/utils/__tests__/viewMapFunctions.test.ts index 8175b44d3cf3..68acdb054655 100644 --- a/packages/twenty-front/src/modules/views/utils/__tests__/viewMapFunctions.test.ts +++ b/packages/twenty-front/src/modules/views/utils/__tests__/viewMapFunctions.test.ts @@ -55,6 +55,7 @@ describe('mapViewFiltersToFilters', () => { ]; const expectedFilters: Filter[] = [ { + id: 'id', fieldMetadataId: '05731f68-6e7a-4903-8374-c0b6a9063482', value: 'testValue', displayValue: 'Test Display Value', diff --git a/packages/twenty-front/src/modules/views/utils/combinedViewFilters.ts b/packages/twenty-front/src/modules/views/utils/combinedViewFilters.ts index 84eefa8f5bc8..09dda1fd232d 100644 --- a/packages/twenty-front/src/modules/views/utils/combinedViewFilters.ts +++ b/packages/twenty-front/src/modules/views/utils/combinedViewFilters.ts @@ -28,9 +28,6 @@ export const combinedViewFilters = ( .concat(toCreateViewFilters); return Object.values( - combinedViewFilters.reduce( - (acc, obj) => ({ ...acc, [obj.fieldMetadataId]: obj }), - {}, - ), + combinedViewFilters.reduce((acc, obj) => ({ ...acc, [obj.id]: obj }), {}), ); }; diff --git a/packages/twenty-front/src/modules/views/utils/mapViewFiltersToFilters.ts b/packages/twenty-front/src/modules/views/utils/mapViewFiltersToFilters.ts index 4df8f1e993fb..104ba6afdaae 100644 --- a/packages/twenty-front/src/modules/views/utils/mapViewFiltersToFilters.ts +++ b/packages/twenty-front/src/modules/views/utils/mapViewFiltersToFilters.ts @@ -18,6 +18,7 @@ export const mapViewFiltersToFilters = ( if (!availableFilterDefinition) return null; return { + id: viewFilter.id, fieldMetadataId: viewFilter.fieldMetadataId, value: viewFilter.value, displayValue: viewFilter.displayValue, diff --git a/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerCreateOrEditContentEffect.tsx b/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerCreateOrEditContentEffect.tsx index 6124063e002c..45ba1209c18d 100644 --- a/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerCreateOrEditContentEffect.tsx +++ b/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerCreateOrEditContentEffect.tsx @@ -1,5 +1,5 @@ import { useEffect } from 'react'; -import { useRecoilValue, useSetRecoilState } from 'recoil'; +import { useRecoilState, useRecoilValue, useSetRecoilState } from 'recoil'; import { useGetCurrentView } from '@/views/hooks/useGetCurrentView'; import { useGetAvailableFieldsForKanban } from '@/views/view-picker/hooks/useGetAvailableFieldsForKanban'; @@ -22,9 +22,8 @@ export const ViewPickerCreateOrEditContentEffect = () => { ); const setViewPickerInputName = useSetRecoilState(viewPickerInputNameState); - const setViewPickerKanbanFieldMetadataId = useSetRecoilState( - viewPickerKanbanFieldMetadataIdState, - ); + const [viewPickerKanbanFieldMetadataId, setViewPickerKanbanFieldMetadataId] = + useRecoilState(viewPickerKanbanFieldMetadataIdState); const setViewPickerType = useSetRecoilState(viewPickerTypeState); const viewPickerReferenceViewId = useRecoilValue( @@ -50,13 +49,11 @@ export const ViewPickerCreateOrEditContentEffect = () => { ) { setViewPickerSelectedIcon(referenceView.icon); setViewPickerInputName(referenceView.name); - setViewPickerKanbanFieldMetadataId(referenceView.kanbanFieldMetadataId); setViewPickerType(referenceView.type); } }, [ referenceView, setViewPickerInputName, - setViewPickerKanbanFieldMetadataId, setViewPickerSelectedIcon, setViewPickerType, viewPickerIsPersisting, @@ -64,13 +61,22 @@ export const ViewPickerCreateOrEditContentEffect = () => { ]); useEffect(() => { - if (availableFieldsForKanban.length > 0 && !viewPickerIsDirty) { - setViewPickerKanbanFieldMetadataId(availableFieldsForKanban[0].id); + if ( + isDefined(referenceView) && + availableFieldsForKanban.length > 0 && + viewPickerKanbanFieldMetadataId === '' + ) { + setViewPickerKanbanFieldMetadataId( + referenceView.kanbanFieldMetadataId !== '' + ? referenceView.kanbanFieldMetadataId + : availableFieldsForKanban[0].id, + ); } }, [ + referenceView, availableFieldsForKanban, + viewPickerKanbanFieldMetadataId, setViewPickerKanbanFieldMetadataId, - viewPickerIsDirty, ]); return <></>; diff --git a/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerListContent.tsx b/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerListContent.tsx index ef0a258b0dfd..91b18694dd2c 100644 --- a/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerListContent.tsx +++ b/packages/twenty-front/src/modules/views/view-picker/components/ViewPickerListContent.tsx @@ -117,7 +117,7 @@ export const ViewPickerListContent = () => { )} /> {indexView && ( - <MenuItem + <MenuItemDraggable key={indexView.id} iconButtons={[ { @@ -128,6 +128,8 @@ export const ViewPickerListContent = () => { onClick={() => handleViewSelect(indexView.id)} LeftIcon={getIcon(indexView.icon)} text={indexView.name} + accent="placeholder" + isDragDisabled /> )} </DropdownMenuItemsContainer> diff --git a/packages/twenty-front/src/modules/views/view-picker/hooks/useGetAvailableFieldsForKanban.ts b/packages/twenty-front/src/modules/views/view-picker/hooks/useGetAvailableFieldsForKanban.ts index 21df6d23c257..0f47fcc78f15 100644 --- a/packages/twenty-front/src/modules/views/view-picker/hooks/useGetAvailableFieldsForKanban.ts +++ b/packages/twenty-front/src/modules/views/view-picker/hooks/useGetAvailableFieldsForKanban.ts @@ -1,16 +1,23 @@ import { useCallback } from 'react'; -import { useNavigate } from 'react-router-dom'; -import { useRecoilValue } from 'recoil'; +import { useLocation, useNavigate } from 'react-router-dom'; +import { useRecoilValue, useSetRecoilState } from 'recoil'; import { objectMetadataItemsState } from '@/object-metadata/states/objectMetadataItemsState'; +import { getObjectSlug } from '@/object-metadata/utils/getObjectSlug'; +import { navigationMemorizedUrlState } from '@/ui/navigation/states/navigationMemorizedUrlState'; import { useViewStates } from '@/views/hooks/internal/useViewStates'; import { FieldMetadataType } from '~/generated-metadata/graphql'; +import { isDefined } from '~/utils/isDefined'; export const useGetAvailableFieldsForKanban = () => { const { viewObjectMetadataIdState } = useViewStates(); const viewObjectMetadataId = useRecoilValue(viewObjectMetadataIdState); const objectMetadataItems = useRecoilValue(objectMetadataItemsState); + const setNavigationMemorizedUrl = useSetRecoilState( + navigationMemorizedUrlState, + ); + const location = useLocation(); const objectMetadataItem = objectMetadataItems.find( (objectMetadata) => objectMetadata.id === viewObjectMetadataId, @@ -24,8 +31,24 @@ export const useGetAvailableFieldsForKanban = () => { const navigate = useNavigate(); const navigateToSelectSettings = useCallback(() => { - navigate(`/settings/objects/${objectMetadataItem?.namePlural}`); - }, [navigate, objectMetadataItem?.namePlural]); + setNavigationMemorizedUrl(location.pathname + location.search); + + if (isDefined(objectMetadataItem?.namePlural)) { + navigate( + `/settings/objects/${getObjectSlug( + objectMetadataItem, + )}/new-field/step-2`, + ); + } else { + navigate(`/settings/objects`); + } + }, [ + setNavigationMemorizedUrl, + location.pathname, + location.search, + objectMetadataItem, + navigate, + ]); return { availableFieldsForKanban, diff --git a/packages/twenty-front/src/modules/workspace/components/WorkspaceMemberCard.tsx b/packages/twenty-front/src/modules/workspace/components/WorkspaceMemberCard.tsx index cae6dff291c6..2fc774905ae0 100644 --- a/packages/twenty-front/src/modules/workspace/components/WorkspaceMemberCard.tsx +++ b/packages/twenty-front/src/modules/workspace/components/WorkspaceMemberCard.tsx @@ -40,7 +40,7 @@ export const WorkspaceMemberCard = ({ <StyledContainer> <Avatar avatarUrl={getImageAbsoluteURIOrBase64(workspaceMember.avatarUrl)} - entityId={workspaceMember.id} + placeholderColorSeed={workspaceMember.id} placeholder={workspaceMember.name.firstName || ''} type="squared" size="xl" @@ -53,7 +53,6 @@ export const WorkspaceMemberCard = ({ /> <StyledEmailText>{workspaceMember.userEmail}</StyledEmailText> </StyledContent> - {accessory} </StyledContainer> ); diff --git a/packages/twenty-front/src/modules/workspace/graphql/mutations/updateWorkspace.ts b/packages/twenty-front/src/modules/workspace/graphql/mutations/updateWorkspace.ts index c244ce041097..1d9a9b9fbed2 100644 --- a/packages/twenty-front/src/modules/workspace/graphql/mutations/updateWorkspace.ts +++ b/packages/twenty-front/src/modules/workspace/graphql/mutations/updateWorkspace.ts @@ -8,7 +8,6 @@ export const UPDATE_WORKSPACE = gql` displayName logo allowImpersonation - subscriptionStatus } } `; diff --git a/packages/twenty-front/src/modules/workspace/hooks/__tests__/useSubscriptionStatus.test.ts b/packages/twenty-front/src/modules/workspace/hooks/__tests__/useSubscriptionStatus.test.ts new file mode 100644 index 000000000000..3ba84ddb77dc --- /dev/null +++ b/packages/twenty-front/src/modules/workspace/hooks/__tests__/useSubscriptionStatus.test.ts @@ -0,0 +1,57 @@ +import { act } from 'react-dom/test-utils'; +import { renderHook } from '@testing-library/react'; +import { RecoilRoot, useSetRecoilState } from 'recoil'; +import { v4 } from 'uuid'; + +import { + CurrentWorkspace, + currentWorkspaceState, +} from '@/auth/states/currentWorkspaceState'; +import { useSubscriptionStatus } from '@/workspace/hooks/useSubscriptionStatus'; +import { SubscriptionStatus } from '~/generated/graphql'; + +const currentWorkspace = { + id: '1', + currentBillingSubscription: { status: SubscriptionStatus.Incomplete }, + activationStatus: 'active', + allowImpersonation: true, +} as CurrentWorkspace; + +const renderHooks = () => { + const { result } = renderHook( + () => { + const subscriptionStatus = useSubscriptionStatus(); + const setCurrentWorkspace = useSetRecoilState(currentWorkspaceState); + + return { + subscriptionStatus, + setCurrentWorkspace, + }; + }, + { + wrapper: RecoilRoot, + }, + ); + return { result }; +}; + +describe('useSubscriptionStatus', () => { + Object.values(SubscriptionStatus).forEach((subscriptionStatus) => { + it(`should return "${subscriptionStatus}"`, async () => { + const { result } = renderHooks(); + const { setCurrentWorkspace } = result.current; + + act(() => { + setCurrentWorkspace({ + ...currentWorkspace, + currentBillingSubscription: { + id: v4(), + status: subscriptionStatus, + }, + }); + }); + + expect(result.current.subscriptionStatus).toBe(subscriptionStatus); + }); + }); +}); diff --git a/packages/twenty-front/src/modules/workspace/hooks/useSubscriptionStatus.ts b/packages/twenty-front/src/modules/workspace/hooks/useSubscriptionStatus.ts new file mode 100644 index 000000000000..332675639a07 --- /dev/null +++ b/packages/twenty-front/src/modules/workspace/hooks/useSubscriptionStatus.ts @@ -0,0 +1,9 @@ +import { useRecoilValue } from 'recoil'; + +import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; +import { SubscriptionStatus } from '~/generated/graphql'; + +export const useSubscriptionStatus = (): SubscriptionStatus | undefined => { + const currentWorkspace = useRecoilValue(currentWorkspaceState); + return currentWorkspace?.currentBillingSubscription?.status; +}; diff --git a/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts b/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts index cb6262c2d008..d9b76577cbce 100644 --- a/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts +++ b/packages/twenty-front/src/modules/workspace/types/FeatureFlagKey.ts @@ -4,4 +4,5 @@ export type FeatureFlagKey = | 'IS_EVENT_OBJECT_ENABLED' | 'IS_AIRTABLE_INTEGRATION_ENABLED' | 'IS_POSTGRESQL_INTEGRATION_ENABLED' - | 'IS_STRIPE_INTEGRATION_ENABLED'; + | 'IS_STRIPE_INTEGRATION_ENABLED' + | 'IS_COPILOT_ENABLED'; diff --git a/packages/twenty-front/src/pages/auth/Authorize.tsx b/packages/twenty-front/src/pages/auth/Authorize.tsx index 8fa99481f4fa..8a247aad2235 100644 --- a/packages/twenty-front/src/pages/auth/Authorize.tsx +++ b/packages/twenty-front/src/pages/auth/Authorize.tsx @@ -1,6 +1,6 @@ +import styled from '@emotion/styled'; import { useEffect, useState } from 'react'; import { useNavigate, useSearchParams } from 'react-router-dom'; -import styled from '@emotion/styled'; import { AppPath } from '@/types/AppPath'; import { MainButton } from '@/ui/input/button/components/MainButton'; @@ -14,7 +14,7 @@ const StyledContainer = styled.div` display: flex; align-items: center; flex-direction: column; - height: 100vh; + height: 100dvh; justify-content: center; width: 100%; `; diff --git a/packages/twenty-front/src/pages/auth/Invite.tsx b/packages/twenty-front/src/pages/auth/Invite.tsx index 5c62531ef6f8..e643654519ae 100644 --- a/packages/twenty-front/src/pages/auth/Invite.tsx +++ b/packages/twenty-front/src/pages/auth/Invite.tsx @@ -1,5 +1,5 @@ -import { useMemo } from 'react'; import styled from '@emotion/styled'; +import { useMemo } from 'react'; import { useRecoilValue } from 'recoil'; import { Logo } from '@/auth/components/Logo'; @@ -17,8 +17,8 @@ import { useAddUserToWorkspaceMutation } from '~/generated/graphql'; import { isDefined } from '~/utils/isDefined'; const StyledContentContainer = styled.div` - margin-bottom: ${({ theme }) => theme.spacing(8)}; - margin-top: ${({ theme }) => theme.spacing(4)}; + margin-bottom: ${({ theme }) => theme.spacing(8)}; + margin-top: ${({ theme }) => theme.spacing(4)}; `; export const Invite = () => { @@ -74,8 +74,23 @@ export const Invite = () => { /> </StyledContentContainer> <FooterNote> - By using Funnelmink, you agree to the Terms of Service and Privacy - Policy. + By using Funnelmink, you agree to the{' '} + <a + href="https://funnelmink.com/legal/terms" + target="_blank" + rel="noopener noreferrer" + > + Terms of Service + </a>{' '} + and{' '} + <a + href="https://funnelmink.com/legal/privacy" + target="_blank" + rel="noopener noreferrer" + > + Privacy Policy + </a> + . </FooterNote> </> ) : ( diff --git a/packages/twenty-front/src/pages/auth/__stories__/Invite.stories.tsx b/packages/twenty-front/src/pages/auth/__stories__/Invite.stories.tsx new file mode 100644 index 000000000000..08369b84fb52 --- /dev/null +++ b/packages/twenty-front/src/pages/auth/__stories__/Invite.stories.tsx @@ -0,0 +1,84 @@ +import { getOperationName } from '@apollo/client/utilities'; +import { Meta, StoryObj } from '@storybook/react'; +import { fireEvent, within } from '@storybook/test'; +import { HttpResponse, graphql } from 'msw'; + +import { AppPath } from '@/types/AppPath'; +import { GET_CURRENT_USER } from '@/users/graphql/queries/getCurrentUser'; +import { GET_WORKSPACE_FROM_INVITE_HASH } from '@/workspace/graphql/queries/getWorkspaceFromInviteHash'; +import { + PageDecorator, + PageDecoratorArgs, +} from '~/testing/decorators/PageDecorator'; +import { graphqlMocks } from '~/testing/graphqlMocks'; + +import { Invite } from '../Invite'; + +const meta: Meta<PageDecoratorArgs> = { + title: 'Pages/Auth/Invite', + component: Invite, + decorators: [PageDecorator], + args: { + routePath: AppPath.Invite, + routeParams: { ':workspaceInviteHash': 'my-hash' }, + }, + parameters: { + msw: { + handlers: [ + graphql.query( + getOperationName(GET_WORKSPACE_FROM_INVITE_HASH) ?? '', + () => { + return HttpResponse.json({ + data: { + findWorkspaceFromInviteHash: { + __typename: 'Workspace', + id: '20202020-91f0-46d0-acab-cb5afef3cc3b', + displayName: 'Twenty dev', + logo: null, + allowImpersonation: false, + }, + }, + }); + }, + ), + graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { + return HttpResponse.json({ + data: null, + errors: [ + { + message: 'Unauthorized', + extensions: { + code: 'UNAUTHENTICATED', + response: { + statusCode: 401, + message: 'Unauthorized', + }, + }, + }, + ], + }); + }), + graphqlMocks.handlers, + ], + }, + cookie: '', + }, +}; + +export default meta; + +export type Story = StoryObj<typeof Invite>; + +export const Default: Story = { + play: async ({ canvasElement }) => { + const canvas = within(canvasElement); + + await canvas.findByText('Join Twenty dev team'); + + const continueWithEmailButton = await canvas.findByText( + 'Continue With Email', + ); + + await fireEvent.click(continueWithEmailButton); + }, +}; diff --git a/packages/twenty-front/src/pages/auth/__stories__/PasswordReset.stories.tsx b/packages/twenty-front/src/pages/auth/__stories__/PasswordReset.stories.tsx index 234b06e795df..ea98157a1e4a 100644 --- a/packages/twenty-front/src/pages/auth/__stories__/PasswordReset.stories.tsx +++ b/packages/twenty-front/src/pages/auth/__stories__/PasswordReset.stories.tsx @@ -4,14 +4,21 @@ import { within } from '@storybook/test'; import { graphql, HttpResponse } from 'msw'; import { GET_CURRENT_USER } from '@/users/graphql/queries/getCurrentUser'; -import { ValidatePasswordResetTokenDocument } from '~/generated/graphql'; +import { + OnboardingStatus, + ValidatePasswordResetTokenDocument, +} from '~/generated/graphql'; import { PasswordReset } from '~/pages/auth/PasswordReset'; import { PageDecorator, PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedOnboardingUsersData } from '~/testing/mock-data/users'; +import { mockedOnboardingUserData } from '~/testing/mock-data/users'; + +const mockedOnboardingUsersData = mockedOnboardingUserData( + OnboardingStatus.Completed, +); const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Auth/PasswordReset', @@ -30,8 +37,8 @@ const meta: Meta<PageDecoratorArgs> = { return HttpResponse.json({ data: { validatePasswordResetToken: { - id: mockedOnboardingUsersData[0].id, - email: mockedOnboardingUsersData[0].email, + id: mockedOnboardingUsersData.id, + email: mockedOnboardingUsersData.email, }, }, }); @@ -40,7 +47,7 @@ const meta: Meta<PageDecoratorArgs> = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: mockedOnboardingUsersData[0], + currentUser: mockedOnboardingUsersData, }, }); }), diff --git a/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx b/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx index bcb3508dc359..2e4a0268ddac 100644 --- a/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx +++ b/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx @@ -1,5 +1,5 @@ -import { useParams } from 'react-router-dom'; import styled from '@emotion/styled'; +import { useParams } from 'react-router-dom'; import { v4 } from 'uuid'; import { RecordIndexContainer } from '@/object-record/record-index/components/RecordIndexContainer'; diff --git a/packages/twenty-front/src/pages/object-record/RecordShowPage.tsx b/packages/twenty-front/src/pages/object-record/RecordShowPage.tsx index e8d5bd90247c..1037467db9ff 100644 --- a/packages/twenty-front/src/pages/object-record/RecordShowPage.tsx +++ b/packages/twenty-front/src/pages/object-record/RecordShowPage.tsx @@ -3,6 +3,7 @@ import { useParams } from 'react-router-dom'; import { TimelineActivityContext } from '@/activities/timelineActivities/contexts/TimelineActivityContext'; import { RecordShowContainer } from '@/object-record/record-show/components/RecordShowContainer'; import { useRecordShowPage } from '@/object-record/record-show/hooks/useRecordShowPage'; +import { useRecordShowPagePagination } from '@/object-record/record-show/hooks/useRecordShowPagePagination'; import { RecordValueSetterEffect } from '@/object-record/record-store/components/RecordValueSetterEffect'; import { RecordFieldValueSelectorContextProvider } from '@/object-record/record-store/contexts/RecordFieldValueSelectorContext'; import { PageBody } from '@/ui/layout/page/PageBody'; @@ -35,16 +36,35 @@ export const RecordShowPage = () => { parameters.objectRecordId ?? '', ); + const { + viewName, + hasPreviousRecord, + hasNextRecord, + navigateToPreviousRecord, + navigateToNextRecord, + navigateToIndexView, + isLoadingPagination, + } = useRecordShowPagePagination( + parameters.objectNameSingular ?? '', + parameters.objectRecordId ?? '', + ); + return ( <RecordFieldValueSelectorContextProvider> <RecordValueSetterEffect recordId={objectRecordId} /> <PageContainer> <PageTitle title={pageTitle} /> <PageHeader - title={pageName ?? ''} - hasBackButton + title={viewName} + hasPaginationButtons + hasClosePageButton + onClosePage={navigateToIndexView} + hasPreviousRecord={hasPreviousRecord} + navigateToPreviousRecord={navigateToPreviousRecord} + hasNextRecord={hasNextRecord} + navigateToNextRecord={navigateToNextRecord} Icon={headerIcon} - loading={loading} + loading={loading || isLoadingPagination} > <> <PageFavoriteButton diff --git a/packages/twenty-front/src/pages/object-record/__stories__/RecordShowPage.stories.tsx b/packages/twenty-front/src/pages/object-record/__stories__/RecordShowPage.stories.tsx index efec731eefbd..2674fd4bdef7 100644 --- a/packages/twenty-front/src/pages/object-record/__stories__/RecordShowPage.stories.tsx +++ b/packages/twenty-front/src/pages/object-record/__stories__/RecordShowPage.stories.tsx @@ -8,7 +8,7 @@ import { PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { getPeopleMock } from '~/testing/mock-data/people'; +import { getPeopleMock, peopleQueryResult } from '~/testing/mock-data/people'; import { mockedWorkspaceMemberData } from '~/testing/mock-data/users'; import { RecordShowPage } from '../RecordShowPage'; @@ -22,12 +22,17 @@ const meta: Meta<PageDecoratorArgs> = { routePath: '/object/:objectNameSingular/:objectRecordId', routeParams: { ':objectNameSingular': 'person', - ':objectRecordId': '1234', + ':objectRecordId': peopleMock[0].id, }, }, parameters: { msw: { handlers: [ + graphql.query('FindManyPeople', () => { + return HttpResponse.json({ + data: peopleQueryResult, + }); + }), graphql.query('FindOnePerson', () => { return HttpResponse.json({ data: { @@ -64,8 +69,8 @@ const meta: Meta<PageDecoratorArgs> = { edges: [], pageInfo: { hasNextPage: false, - startCursor: '1234', - endCursor: '1234', + startCursor: peopleMock[0].id, + endCursor: peopleMock[0].id, }, totalCount: 0, }, diff --git a/packages/twenty-front/src/pages/onboarding/ChooseYourPlan.tsx b/packages/twenty-front/src/pages/onboarding/ChooseYourPlan.tsx index 1c24f58aaeb6..1a85d31434c7 100644 --- a/packages/twenty-front/src/pages/onboarding/ChooseYourPlan.tsx +++ b/packages/twenty-front/src/pages/onboarding/ChooseYourPlan.tsx @@ -19,6 +19,7 @@ import { ActionLink } from '@/ui/navigation/link/components/ActionLink'; import { CAL_LINK } from '@/ui/navigation/link/constants/Cal'; import { ProductPriceEntity, + SubscriptionInterval, useCheckoutSessionMutation, useGetProductPricesQuery, } from '~/generated/graphql'; @@ -75,7 +76,7 @@ const benefits = [ export const ChooseYourPlan = () => { const billing = useRecoilValue(billingState); - const [planSelected, setPlanSelected] = useState('month'); + const [planSelected, setPlanSelected] = useState(SubscriptionInterval.Month); const [isSubmitting, setIsSubmitting] = useState(false); @@ -87,7 +88,7 @@ export const ChooseYourPlan = () => { const [checkoutSession] = useCheckoutSessionMutation(); - const handlePlanChange = (type?: string) => { + const handlePlanChange = (type?: SubscriptionInterval) => { return () => { if (isNonEmptyString(type) && planSelected !== type) { setPlanSelected(type); @@ -101,11 +102,11 @@ export const ChooseYourPlan = () => { price: ProductPriceEntity, prices: ProductPriceEntity[], ): string => { - if (price.recurringInterval !== 'year') { + if (price.recurringInterval !== SubscriptionInterval.Year) { return 'Cancel anytime'; } const monthPrice = prices.filter( - (price) => price.recurringInterval === 'month', + (price) => price.recurringInterval === SubscriptionInterval.Month, )?.[0]; if ( isDefined(monthPrice) && diff --git a/packages/twenty-front/src/pages/onboarding/CreateProfile.tsx b/packages/twenty-front/src/pages/onboarding/CreateProfile.tsx index fa588b34ee97..f5faa4ff4b19 100644 --- a/packages/twenty-front/src/pages/onboarding/CreateProfile.tsx +++ b/packages/twenty-front/src/pages/onboarding/CreateProfile.tsx @@ -9,11 +9,11 @@ import { z } from 'zod'; import { SubTitle } from '@/auth/components/SubTitle'; import { Title } from '@/auth/components/Title'; -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; -import { OnboardingStatus } from '@/auth/utils/getOnboardingStatus'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; +import { useSetNextOnboardingStatus } from '@/onboarding/hooks/useSetNextOnboardingStatus'; import { ProfilePictureUploader } from '@/settings/profile/components/ProfilePictureUploader'; import { PageHotkeyScope } from '@/types/PageHotkeyScope'; import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; @@ -22,6 +22,8 @@ import { MainButton } from '@/ui/input/button/components/MainButton'; import { TextInputV2 } from '@/ui/input/components/TextInputV2'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; import { WorkspaceMember } from '@/workspace-member/types/WorkspaceMember'; +import { OnboardingStatus } from '~/generated/graphql'; +import { isDefined } from '~/utils/isDefined'; const StyledContentContainer = styled.div` width: 100%; @@ -55,11 +57,11 @@ type Form = z.infer<typeof validationSchema>; export const CreateProfile = () => { const onboardingStatus = useOnboardingStatus(); + const setNextOnboardingStatus = useSetNextOnboardingStatus(); const { enqueueSnackBar } = useSnackBar(); const [currentWorkspaceMember, setCurrentWorkspaceMember] = useRecoilState( currentWorkspaceMemberState, ); - const { updateOneRecord } = useUpdateOneRecord<WorkspaceMember>({ objectNameSingular: CoreObjectNameSingular.WorkspaceMember, }); @@ -100,17 +102,20 @@ export const CreateProfile = () => { }, }); - setCurrentWorkspaceMember( - (current) => - ({ + setCurrentWorkspaceMember((current) => { + if (isDefined(current)) { + return { ...current, name: { firstName: data.firstName, lastName: data.lastName, }, colorScheme: 'System', - }) as any, - ); + }; + } + return current; + }); + setNextOnboardingStatus(); } catch (error: any) { enqueueSnackBar(error?.message, { variant: SnackBarVariant.Error, @@ -119,6 +124,7 @@ export const CreateProfile = () => { }, [ currentWorkspaceMember?.id, + setNextOnboardingStatus, enqueueSnackBar, setCurrentWorkspaceMember, updateOneRecord, @@ -137,7 +143,7 @@ export const CreateProfile = () => { PageHotkeyScope.CreateProfile, ); - if (onboardingStatus !== OnboardingStatus.OngoingProfileCreation) { + if (onboardingStatus !== OnboardingStatus.ProfileCreation) { return null; } diff --git a/packages/twenty-front/src/pages/onboarding/CreateWorkspace.tsx b/packages/twenty-front/src/pages/onboarding/CreateWorkspace.tsx index 5d37f13d27cb..b703ef13b0fd 100644 --- a/packages/twenty-front/src/pages/onboarding/CreateWorkspace.tsx +++ b/packages/twenty-front/src/pages/onboarding/CreateWorkspace.tsx @@ -9,18 +9,20 @@ import { z } from 'zod'; import { SubTitle } from '@/auth/components/SubTitle'; import { Title } from '@/auth/components/Title'; -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; import { isCurrentUserLoadedState } from '@/auth/states/isCurrentUserLoadingState'; -import { OnboardingStatus } from '@/auth/utils/getOnboardingStatus'; import { FIND_MANY_OBJECT_METADATA_ITEMS } from '@/object-metadata/graphql/queries'; import { useApolloMetadataClient } from '@/object-metadata/hooks/useApolloMetadataClient'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; import { WorkspaceLogoUploader } from '@/settings/workspace/components/WorkspaceLogoUploader'; import { Loader } from '@/ui/feedback/loader/components/Loader'; import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; import { useSnackBar } from '@/ui/feedback/snack-bar-manager/hooks/useSnackBar'; import { MainButton } from '@/ui/input/button/components/MainButton'; import { TextInputV2 } from '@/ui/input/components/TextInputV2'; -import { useActivateWorkspaceMutation } from '~/generated/graphql'; +import { + OnboardingStatus, + useActivateWorkspaceMutation, +} from '~/generated/graphql'; import { isDefined } from '~/utils/isDefined'; const StyledContentContainer = styled.div` @@ -105,7 +107,7 @@ export const CreateWorkspace = () => { } }; - if (onboardingStatus !== OnboardingStatus.OngoingWorkspaceActivation) { + if (onboardingStatus !== OnboardingStatus.WorkspaceActivation) { return null; } diff --git a/packages/twenty-front/src/pages/onboarding/InviteTeam.tsx b/packages/twenty-front/src/pages/onboarding/InviteTeam.tsx index 2395c4bb27d3..53a7b212494c 100644 --- a/packages/twenty-front/src/pages/onboarding/InviteTeam.tsx +++ b/packages/twenty-front/src/pages/onboarding/InviteTeam.tsx @@ -17,7 +17,7 @@ import { SubTitle } from '@/auth/components/SubTitle'; import { Title } from '@/auth/components/Title'; import { currentUserState } from '@/auth/states/currentUserState'; import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; -import { useSetNextOnboardingStep } from '@/onboarding/hooks/useSetNextOnboardingStep'; +import { useSetNextOnboardingStatus } from '@/onboarding/hooks/useSetNextOnboardingStatus'; import { PageHotkeyScope } from '@/types/PageHotkeyScope'; import { SeparatorLineText } from '@/ui/display/text/components/SeparatorLineText'; import { SnackBarVariant } from '@/ui/feedback/snack-bar-manager/components/SnackBar'; @@ -27,7 +27,10 @@ import { MainButton } from '@/ui/input/button/components/MainButton'; import { TextInputV2 } from '@/ui/input/components/TextInputV2'; import { AnimatedTranslation } from '@/ui/utilities/animation/components/AnimatedTranslation'; import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; -import { OnboardingStep, useSendInviteLinkMutation } from '~/generated/graphql'; +import { + OnboardingStatus, + useSendInviteLinkMutation, +} from '~/generated/graphql'; import { isDefined } from '~/utils/isDefined'; const StyledAnimatedContainer = styled.div` @@ -63,7 +66,7 @@ export const InviteTeam = () => { const theme = useTheme(); const { enqueueSnackBar } = useSnackBar(); const [sendInviteLink] = useSendInviteLinkMutation(); - const setNextOnboardingStep = useSetNextOnboardingStep(); + const setNextOnboardingStatus = useSetNextOnboardingStatus(); const currentUser = useRecoilValue(currentUserState); const currentWorkspace = useRecoilValue(currentWorkspaceState); const { @@ -133,7 +136,7 @@ export const InviteTeam = () => { ); const result = await sendInviteLink({ variables: { emails } }); - setNextOnboardingStep(OnboardingStep.InviteTeam); + setNextOnboardingStatus(); if (isDefined(result.errors)) { throw result.errors; @@ -145,7 +148,7 @@ export const InviteTeam = () => { }); } }, - [enqueueSnackBar, sendInviteLink, setNextOnboardingStep], + [enqueueSnackBar, sendInviteLink, setNextOnboardingStatus], ); useScopedHotkeys( @@ -157,7 +160,7 @@ export const InviteTeam = () => { [handleSubmit], ); - if (currentUser?.onboardingStep !== OnboardingStep.InviteTeam) { + if (currentUser?.onboardingStatus !== OnboardingStatus.InviteTeam) { return <></>; } diff --git a/packages/twenty-front/src/pages/onboarding/PaymentSuccess.tsx b/packages/twenty-front/src/pages/onboarding/PaymentSuccess.tsx index f77d7360e65b..ce57c50795ad 100644 --- a/packages/twenty-front/src/pages/onboarding/PaymentSuccess.tsx +++ b/packages/twenty-front/src/pages/onboarding/PaymentSuccess.tsx @@ -1,14 +1,17 @@ import React from 'react'; import { useTheme } from '@emotion/react'; import styled from '@emotion/styled'; +import { useRecoilValue } from 'recoil'; import { IconCheck, RGBA } from 'twenty-ui'; import { SubTitle } from '@/auth/components/SubTitle'; import { Title } from '@/auth/components/Title'; +import { currentUserState } from '@/auth/states/currentUserState'; import { AppPath } from '@/types/AppPath'; import { MainButton } from '@/ui/input/button/components/MainButton'; import { UndecoratedLink } from '@/ui/navigation/link/components/UndecoratedLink'; import { AnimatedEaseIn } from '@/ui/utilities/animation/components/AnimatedEaseIn'; +import { OnboardingStatus } from '~/generated/graphql'; const StyledCheckContainer = styled.div` align-items: center; @@ -29,8 +32,14 @@ const StyledButtonContainer = styled.div` export const PaymentSuccess = () => { const theme = useTheme(); + const currentUser = useRecoilValue(currentUserState); const color = theme.name === 'light' ? theme.grayScale.gray90 : theme.grayScale.gray10; + + if (currentUser?.onboardingStatus === OnboardingStatus.Completed) { + return <></>; + } + return ( <> <AnimatedEaseIn> diff --git a/packages/twenty-front/src/pages/onboarding/SyncEmails.tsx b/packages/twenty-front/src/pages/onboarding/SyncEmails.tsx index 44828c33a837..4fd138ec7541 100644 --- a/packages/twenty-front/src/pages/onboarding/SyncEmails.tsx +++ b/packages/twenty-front/src/pages/onboarding/SyncEmails.tsx @@ -9,7 +9,7 @@ import { SubTitle } from '@/auth/components/SubTitle'; import { Title } from '@/auth/components/Title'; import { currentUserState } from '@/auth/states/currentUserState'; import { OnboardingSyncEmailsSettingsCard } from '@/onboarding/components/OnboardingSyncEmailsSettingsCard'; -import { useSetNextOnboardingStep } from '@/onboarding/hooks/useSetNextOnboardingStep'; +import { useSetNextOnboardingStatus } from '@/onboarding/hooks/useSetNextOnboardingStatus'; import { useTriggerGoogleApisOAuth } from '@/settings/accounts/hooks/useTriggerGoogleApisOAuth'; import { AppPath } from '@/types/AppPath'; import { PageHotkeyScope } from '@/types/PageHotkeyScope'; @@ -19,7 +19,7 @@ import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys'; import { CalendarChannelVisibility, MessageChannelVisibility, - OnboardingStep, + OnboardingStatus, useSkipSyncEmailOnboardingStepMutation, } from '~/generated/graphql'; @@ -40,12 +40,12 @@ const StyledActionLinkContainer = styled.div` export const SyncEmails = () => { const theme = useTheme(); const { triggerGoogleApisOAuth } = useTriggerGoogleApisOAuth(); - const setNextOnboardingStep = useSetNextOnboardingStep(); + const setNextOnboardingStatus = useSetNextOnboardingStatus(); const currentUser = useRecoilValue(currentUserState); const [visibility, setVisibility] = useState<MessageChannelVisibility>( MessageChannelVisibility.ShareEverything, ); - const [skipSyncEmailOnboardingStepMutation] = + const [skipSyncEmailOnboardingStatusMutation] = useSkipSyncEmailOnboardingStepMutation(); const handleButtonClick = async () => { @@ -62,8 +62,8 @@ export const SyncEmails = () => { }; const continueWithoutSync = async () => { - await skipSyncEmailOnboardingStepMutation(); - setNextOnboardingStep(OnboardingStep.SyncEmail); + await skipSyncEmailOnboardingStatusMutation(); + setNextOnboardingStatus(); }; useScopedHotkeys( @@ -75,7 +75,7 @@ export const SyncEmails = () => { [continueWithoutSync], ); - if (currentUser?.onboardingStep !== OnboardingStep.SyncEmail) { + if (currentUser?.onboardingStatus !== OnboardingStatus.SyncEmail) { return <></>; } diff --git a/packages/twenty-front/src/pages/onboarding/__stories__/ChooseYourPlan.stories.tsx b/packages/twenty-front/src/pages/onboarding/__stories__/ChooseYourPlan.stories.tsx index 7325383bfd54..b42177fedd01 100644 --- a/packages/twenty-front/src/pages/onboarding/__stories__/ChooseYourPlan.stories.tsx +++ b/packages/twenty-front/src/pages/onboarding/__stories__/ChooseYourPlan.stories.tsx @@ -5,13 +5,14 @@ import { graphql, HttpResponse } from 'msw'; import { AppPath } from '@/types/AppPath'; import { GET_CURRENT_USER } from '@/users/graphql/queries/getCurrentUser'; +import { OnboardingStatus } from '~/generated/graphql'; import { ChooseYourPlan } from '~/pages/onboarding/ChooseYourPlan'; import { PageDecorator, PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedOnboardingUsersData } from '~/testing/mock-data/users'; +import { mockedOnboardingUserData } from '~/testing/mock-data/users'; import { sleep } from '~/utils/sleep'; const meta: Meta<PageDecoratorArgs> = { @@ -25,7 +26,9 @@ const meta: Meta<PageDecoratorArgs> = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: mockedOnboardingUsersData[0], + currentUser: mockedOnboardingUserData( + OnboardingStatus.PlanRequired, + ), }, }); }), diff --git a/packages/twenty-front/src/pages/onboarding/__stories__/CreateProfile.stories.tsx b/packages/twenty-front/src/pages/onboarding/__stories__/CreateProfile.stories.tsx index 026ef4c60332..19512ebe183e 100644 --- a/packages/twenty-front/src/pages/onboarding/__stories__/CreateProfile.stories.tsx +++ b/packages/twenty-front/src/pages/onboarding/__stories__/CreateProfile.stories.tsx @@ -5,13 +5,14 @@ import { graphql, HttpResponse } from 'msw'; import { AppPath } from '@/types/AppPath'; import { GET_CURRENT_USER } from '@/users/graphql/queries/getCurrentUser'; +import { OnboardingStatus } from '~/generated/graphql'; import { CreateProfile } from '~/pages/onboarding/CreateProfile'; import { PageDecorator, PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedOnboardingUsersData } from '~/testing/mock-data/users'; +import { mockedOnboardingUserData } from '~/testing/mock-data/users'; const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Onboarding/CreateProfile', @@ -24,7 +25,9 @@ const meta: Meta<PageDecoratorArgs> = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: mockedOnboardingUsersData[0], + currentUser: mockedOnboardingUserData( + OnboardingStatus.ProfileCreation, + ), }, }); }), diff --git a/packages/twenty-front/src/pages/onboarding/__stories__/CreateWorkspace.stories.tsx b/packages/twenty-front/src/pages/onboarding/__stories__/CreateWorkspace.stories.tsx index f881ef40ab3e..baccc13b2c80 100644 --- a/packages/twenty-front/src/pages/onboarding/__stories__/CreateWorkspace.stories.tsx +++ b/packages/twenty-front/src/pages/onboarding/__stories__/CreateWorkspace.stories.tsx @@ -2,30 +2,22 @@ import { getOperationName } from '@apollo/client/utilities'; import { Meta, StoryObj } from '@storybook/react'; import { within } from '@storybook/test'; import { graphql, HttpResponse } from 'msw'; -import { useSetRecoilState } from 'recoil'; -import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; import { AppPath } from '@/types/AppPath'; import { GET_CURRENT_USER } from '@/users/graphql/queries/getCurrentUser'; +import { OnboardingStatus } from '~/generated/graphql'; import { CreateWorkspace } from '~/pages/onboarding/CreateWorkspace'; import { PageDecorator, PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedOnboardingUsersData } from '~/testing/mock-data/users'; +import { mockedOnboardingUserData } from '~/testing/mock-data/users'; const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Onboarding/CreateWorkspace', component: CreateWorkspace, - decorators: [ - (Story) => { - const setCurrentWorkspace = useSetRecoilState(currentWorkspaceState); - setCurrentWorkspace(mockedOnboardingUsersData[1].defaultWorkspace); - return <Story />; - }, - PageDecorator, - ], + decorators: [PageDecorator], args: { routePath: AppPath.CreateWorkspace }, parameters: { msw: { @@ -33,7 +25,9 @@ const meta: Meta<PageDecoratorArgs> = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: mockedOnboardingUsersData[1], + currentUser: mockedOnboardingUserData( + OnboardingStatus.WorkspaceActivation, + ), }, }); }), diff --git a/packages/twenty-front/src/pages/onboarding/__stories__/InviteTeam.stories.tsx b/packages/twenty-front/src/pages/onboarding/__stories__/InviteTeam.stories.tsx index 799daa507379..e174fc517073 100644 --- a/packages/twenty-front/src/pages/onboarding/__stories__/InviteTeam.stories.tsx +++ b/packages/twenty-front/src/pages/onboarding/__stories__/InviteTeam.stories.tsx @@ -3,7 +3,7 @@ import { Meta, StoryObj } from '@storybook/react'; import { within } from '@storybook/test'; import { graphql, HttpResponse } from 'msw'; -import { OnboardingStep } from '~/generated/graphql'; +import { OnboardingStatus } from '~/generated/graphql'; import { AppPath } from '~/modules/types/AppPath'; import { GET_CURRENT_USER } from '~/modules/users/graphql/queries/getCurrentUser'; import { InviteTeam } from '~/pages/onboarding/InviteTeam'; @@ -12,7 +12,7 @@ import { PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedOnboardingUsersData } from '~/testing/mock-data/users'; +import { mockedOnboardingUserData } from '~/testing/mock-data/users'; const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Onboarding/InviteTeam', @@ -25,10 +25,9 @@ const meta: Meta<PageDecoratorArgs> = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: { - ...mockedOnboardingUsersData[0], - onboardingStep: OnboardingStep.InviteTeam, - }, + currentUser: mockedOnboardingUserData( + OnboardingStatus.InviteTeam, + ), }, }); }), diff --git a/packages/twenty-front/src/pages/onboarding/__stories__/PaymentSuccess.stories.tsx b/packages/twenty-front/src/pages/onboarding/__stories__/PaymentSuccess.stories.tsx index 79ad41ee12a8..fccb1e3c64eb 100644 --- a/packages/twenty-front/src/pages/onboarding/__stories__/PaymentSuccess.stories.tsx +++ b/packages/twenty-front/src/pages/onboarding/__stories__/PaymentSuccess.stories.tsx @@ -5,13 +5,14 @@ import { graphql, HttpResponse } from 'msw'; import { AppPath } from '@/types/AppPath'; import { GET_CURRENT_USER } from '@/users/graphql/queries/getCurrentUser'; +import { OnboardingStatus } from '~/generated/graphql'; import { PaymentSuccess } from '~/pages/onboarding/PaymentSuccess'; import { PageDecorator, PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedOnboardingUsersData } from '~/testing/mock-data/users'; +import { mockedOnboardingUserData } from '~/testing/mock-data/users'; const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Onboarding/PaymentSuccess', @@ -24,7 +25,9 @@ const meta: Meta<PageDecoratorArgs> = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: mockedOnboardingUsersData[0], + currentUser: mockedOnboardingUserData( + OnboardingStatus.WorkspaceActivation, + ), }, }); }), diff --git a/packages/twenty-front/src/pages/onboarding/__stories__/SyncEmails.stories.tsx b/packages/twenty-front/src/pages/onboarding/__stories__/SyncEmails.stories.tsx index d6f7ba36dd89..8d9b4261ef0b 100644 --- a/packages/twenty-front/src/pages/onboarding/__stories__/SyncEmails.stories.tsx +++ b/packages/twenty-front/src/pages/onboarding/__stories__/SyncEmails.stories.tsx @@ -3,7 +3,7 @@ import { Meta, StoryObj } from '@storybook/react'; import { within } from '@storybook/test'; import { graphql, HttpResponse } from 'msw'; -import { OnboardingStep } from '~/generated/graphql'; +import { OnboardingStatus } from '~/generated/graphql'; import { AppPath } from '~/modules/types/AppPath'; import { GET_CURRENT_USER } from '~/modules/users/graphql/queries/getCurrentUser'; import { SyncEmails } from '~/pages/onboarding/SyncEmails'; @@ -12,7 +12,7 @@ import { PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedOnboardingUsersData } from '~/testing/mock-data/users'; +import { mockedOnboardingUserData } from '~/testing/mock-data/users'; const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Onboarding/SyncEmails', @@ -25,10 +25,7 @@ const meta: Meta<PageDecoratorArgs> = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: { - ...mockedOnboardingUsersData[0], - onboardingStep: OnboardingStep.SyncEmail, - }, + currentUser: mockedOnboardingUserData(OnboardingStatus.SyncEmail), }, }); }), diff --git a/packages/twenty-front/src/pages/settings/SettingsBilling.tsx b/packages/twenty-front/src/pages/settings/SettingsBilling.tsx index 77763f68bce2..2e738952b376 100644 --- a/packages/twenty-front/src/pages/settings/SettingsBilling.tsx +++ b/packages/twenty-front/src/pages/settings/SettingsBilling.tsx @@ -10,10 +10,9 @@ import { IconCurrencyDollar, } from 'twenty-ui'; -import { useOnboardingStatus } from '@/auth/hooks/useOnboardingStatus'; import { currentWorkspaceState } from '@/auth/states/currentWorkspaceState'; -import { OnboardingStatus } from '@/auth/utils/getOnboardingStatus'; import { SettingsBillingCoverImage } from '@/billing/components/SettingsBillingCoverImage'; +import { useOnboardingStatus } from '@/onboarding/hooks/useOnboardingStatus'; import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; import { SupportChat } from '@/support/components/SupportChat'; import { AppPath } from '@/types/AppPath'; @@ -24,8 +23,11 @@ import { Button } from '@/ui/input/button/components/Button'; import { ConfirmationModal } from '@/ui/layout/modal/components/ConfirmationModal'; import { SubMenuTopBarContainer } from '@/ui/layout/page/SubMenuTopBarContainer'; import { Section } from '@/ui/layout/section/components/Section'; -import { UndecoratedLink } from '@/ui/navigation/link/components/UndecoratedLink'; +import { useSubscriptionStatus } from '@/workspace/hooks/useSubscriptionStatus'; import { + OnboardingStatus, + SubscriptionInterval, + SubscriptionStatus, useBillingPortalSessionQuery, useUpdateBillingSubscriptionMutation, } from '~/generated/graphql'; @@ -40,21 +42,21 @@ const StyledInvisibleChat = styled.div` `; type SwitchInfo = { - newInterval: string; + newInterval: SubscriptionInterval; to: string; from: string; impact: string; }; const MONTHLY_SWITCH_INFO: SwitchInfo = { - newInterval: 'year', + newInterval: SubscriptionInterval.Year, to: 'to yearly', from: 'from monthly to yearly', impact: 'You will be charged immediately for the full year.', }; const YEARLY_SWITCH_INFO: SwitchInfo = { - newInterval: 'month', + newInterval: SubscriptionInterval.Month, to: 'to monthly', from: 'from yearly to monthly', impact: 'Your credit balance will be used to pay the monthly bills.', @@ -68,10 +70,12 @@ const SWITCH_INFOS = { export const SettingsBilling = () => { const { enqueueSnackBar } = useSnackBar(); const onboardingStatus = useOnboardingStatus(); + const subscriptionStatus = useSubscriptionStatus(); const currentWorkspace = useRecoilValue(currentWorkspaceState); const setCurrentWorkspace = useSetRecoilState(currentWorkspaceState); const switchingInfo = - currentWorkspace?.currentBillingSubscription?.interval === 'year' + currentWorkspace?.currentBillingSubscription?.interval === + SubscriptionInterval.Year ? SWITCH_INFOS.year : SWITCH_INFOS.month; const [isSwitchingIntervalModalOpen, setIsSwitchingIntervalModalOpen] = @@ -94,14 +98,15 @@ export const SettingsBilling = () => { onboardingStatus !== OnboardingStatus.Completed; const displayPaymentFailInfo = - onboardingStatus === OnboardingStatus.PastDue || - onboardingStatus === OnboardingStatus.Unpaid; + subscriptionStatus === SubscriptionStatus.PastDue || + subscriptionStatus === SubscriptionStatus.Unpaid; const displaySubscriptionCanceledInfo = - onboardingStatus === OnboardingStatus.Canceled; + subscriptionStatus === SubscriptionStatus.Canceled; const displaySubscribeInfo = - onboardingStatus === OnboardingStatus.CompletedWithoutSubscription; + onboardingStatus === OnboardingStatus.Completed && + !isDefined(subscriptionStatus); const openBillingPortal = () => { if (isDefined(data) && isDefined(data.billingPortalSession.url)) { @@ -153,22 +158,20 @@ export const SettingsBilling = () => { /> )} {displaySubscriptionCanceledInfo && ( - <UndecoratedLink to={AppPath.PlanRequired}> - <Info - text={'Subscription canceled. Please start a new one'} - buttonTitle={'Subscribe'} - accent={'danger'} - /> - </UndecoratedLink> + <Info + text={'Subscription canceled. Please start a new one'} + buttonTitle={'Subscribe'} + accent={'danger'} + to={AppPath.PlanRequired} + /> )} {displaySubscribeInfo ? ( - <UndecoratedLink to={AppPath.PlanRequired}> - <Info - text={'Your workspace does not have an active subscription'} - buttonTitle={'Subscribe'} - accent={'danger'} - /> - </UndecoratedLink> + <Info + text={'Your workspace does not have an active subscription'} + buttonTitle={'Subscribe'} + accent={'danger'} + to={AppPath.PlanRequired} + /> ) : ( <> <Section> diff --git a/packages/twenty-front/src/pages/settings/accounts/SettingsAccounts.tsx b/packages/twenty-front/src/pages/settings/accounts/SettingsAccounts.tsx index 484a0a681742..0a46dfbe95d9 100644 --- a/packages/twenty-front/src/pages/settings/accounts/SettingsAccounts.tsx +++ b/packages/twenty-front/src/pages/settings/accounts/SettingsAccounts.tsx @@ -8,8 +8,8 @@ import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSi import { generateDepthOneRecordGqlFields } from '@/object-record/graphql/utils/generateDepthOneRecordGqlFields'; import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; import { SettingsAccountLoader } from '@/settings/accounts/components/SettingsAccountLoader'; +import { SettingsAccountsBlocklistSection } from '@/settings/accounts/components/SettingsAccountsBlocklistSection'; import { SettingsAccountsConnectedAccountsListCard } from '@/settings/accounts/components/SettingsAccountsConnectedAccountsListCard'; -import { SettingsAccountsEmailsBlocklistSection } from '@/settings/accounts/components/SettingsAccountsEmailsBlocklistSection'; import { SettingsAccountsSettingsSection } from '@/settings/accounts/components/SettingsAccountsSettingsSection'; import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; import { SubMenuTopBarContainer } from '@/ui/layout/page/SubMenuTopBarContainer'; @@ -55,7 +55,7 @@ export const SettingsAccounts = () => { loading={loading} /> </Section> - {isBlocklistEnabled && <SettingsAccountsEmailsBlocklistSection />} + {isBlocklistEnabled && <SettingsAccountsBlocklistSection />} <SettingsAccountsSettingsSection /> </> )} diff --git a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsCalendars.tsx b/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsCalendars.tsx index 6ad0070f3699..6d3d1c3570e7 100644 --- a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsCalendars.tsx +++ b/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsCalendars.tsx @@ -1,85 +1,14 @@ -import { addMinutes, endOfDay, min, startOfDay } from 'date-fns'; -import { useRecoilValue } from 'recoil'; -import { H2Title, IconSettings } from 'twenty-ui'; +import { IconSettings } from 'twenty-ui'; -import { CalendarChannel } from '@/accounts/types/CalendarChannel'; -import { ConnectedAccount } from '@/accounts/types/ConnectedAccount'; -import { CalendarMonthCard } from '@/activities/calendar/components/CalendarMonthCard'; -import { CalendarContext } from '@/activities/calendar/contexts/CalendarContext'; -import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; -import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; -import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; -import { SettingsAccountsCalendarChannelsListCard } from '@/settings/accounts/components/SettingsAccountsCalendarChannelsListCard'; -import { SettingsAccountsCalendarDisplaySettings } from '@/settings/accounts/components/SettingsAccountsCalendarDisplaySettings'; +import { SettingsAccountsCalendarChannelsContainer } from '@/settings/accounts/components/SettingsAccountsCalendarChannelsContainer'; import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; import { getSettingsPagePath } from '@/settings/utils/getSettingsPagePath'; import { SettingsPath } from '@/types/SettingsPath'; import { SubMenuTopBarContainer } from '@/ui/layout/page/SubMenuTopBarContainer'; import { Section } from '@/ui/layout/section/components/Section'; import { Breadcrumb } from '@/ui/navigation/bread-crumb/components/Breadcrumb'; -import { CalendarChannelVisibility } from '~/generated/graphql'; -import { TimelineCalendarEvent } from '~/generated-metadata/graphql'; export const SettingsAccountsCalendars = () => { - const calendarSettingsEnabled = false; - const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState); - const { records: accounts } = useFindManyRecords<ConnectedAccount>({ - objectNameSingular: CoreObjectNameSingular.ConnectedAccount, - filter: { - accountOwnerId: { - eq: currentWorkspaceMember?.id, - }, - }, - }); - - const { records: calendarChannels } = useFindManyRecords<CalendarChannel>({ - objectNameSingular: CoreObjectNameSingular.CalendarChannel, - filter: { - connectedAccountId: { - in: accounts.map((account) => account.id), - }, - }, - }); - - const exampleStartDate = new Date(); - const exampleEndDate = min([ - addMinutes(exampleStartDate, 30), - endOfDay(exampleStartDate), - ]); - const exampleDayTime = startOfDay(exampleStartDate).getTime(); - const exampleCalendarEvent: TimelineCalendarEvent = { - id: '', - participants: [ - { - firstName: currentWorkspaceMember?.name.firstName || '', - lastName: currentWorkspaceMember?.name.lastName || '', - displayName: currentWorkspaceMember - ? [ - currentWorkspaceMember.name.firstName, - currentWorkspaceMember.name.lastName, - ].join(' ') - : '', - avatarUrl: currentWorkspaceMember?.avatarUrl || '', - handle: '', - personId: '', - workspaceMemberId: currentWorkspaceMember?.id || '', - }, - ], - endsAt: exampleEndDate.toISOString(), - isFullDay: false, - startsAt: exampleStartDate.toISOString(), - conferenceSolution: '', - conferenceLink: { - label: '', - url: '', - }, - description: '', - isCanceled: false, - location: '', - title: 'Onboarding call', - visibility: CalendarChannelVisibility.ShareEverything, - }; - return ( <SubMenuTopBarContainer Icon={IconSettings} title="Settings"> <SettingsPageContainer> @@ -93,41 +22,8 @@ export const SettingsAccountsCalendars = () => { ]} /> <Section> - <H2Title - title="Calendar settings" - description="Sync your calendars and set your preferences" - /> - <SettingsAccountsCalendarChannelsListCard /> + <SettingsAccountsCalendarChannelsContainer /> </Section> - {!!calendarChannels.length && calendarSettingsEnabled && ( - <> - <Section> - <H2Title - title="Display" - description="Configure how we should display your events in your calendar" - /> - <SettingsAccountsCalendarDisplaySettings /> - </Section> - <Section> - <H2Title - title="Color code" - description="Events you participated in are displayed in red." - /> - <CalendarContext.Provider - value={{ - currentCalendarEvent: exampleCalendarEvent, - calendarEventsByDayTime: { - [exampleDayTime]: [exampleCalendarEvent], - }, - getNextCalendarEvent: () => undefined, - updateCurrentCalendarEvent: () => {}, - }} - > - <CalendarMonthCard dayTimes={[exampleDayTime]} /> - </CalendarContext.Provider> - </Section> - </> - )} </SettingsPageContainer> </SubMenuTopBarContainer> ); diff --git a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsCalendarsSettings.tsx b/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsCalendarsSettings.tsx deleted file mode 100644 index 589b89acc52c..000000000000 --- a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsCalendarsSettings.tsx +++ /dev/null @@ -1,143 +0,0 @@ -import { useEffect } from 'react'; -import { useNavigate, useParams } from 'react-router-dom'; -import { useTheme } from '@emotion/react'; -import styled from '@emotion/styled'; -import { H2Title, IconRefresh, IconSettings, IconUser } from 'twenty-ui'; - -import { CalendarChannel } from '@/accounts/types/CalendarChannel'; -import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; -import { useFindOneRecord } from '@/object-record/hooks/useFindOneRecord'; -import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; -import { SettingsAccountsEventVisibilitySettingsCard } from '@/settings/accounts/components/SettingsAccountsCalendarVisibilitySettingsCard'; -import { SettingsAccountsCardMedia } from '@/settings/accounts/components/SettingsAccountsCardMedia'; -import { SettingsAccountsToggleSettingCard } from '@/settings/accounts/components/SettingsAccountsToggleSettingCard'; -import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; -import { getSettingsPagePath } from '@/settings/utils/getSettingsPagePath'; -import { AppPath } from '@/types/AppPath'; -import { SettingsPath } from '@/types/SettingsPath'; -import { SubMenuTopBarContainer } from '@/ui/layout/page/SubMenuTopBarContainer'; -import { Section } from '@/ui/layout/section/components/Section'; -import { Breadcrumb } from '@/ui/navigation/bread-crumb/components/Breadcrumb'; -import { CalendarChannelVisibility } from '~/generated/graphql'; - -const StyledCardMedia = styled(SettingsAccountsCardMedia)` - height: ${({ theme }) => theme.spacing(6)}; -`; - -export const SettingsAccountsCalendarsSettings = () => { - const theme = useTheme(); - const navigate = useNavigate(); - - const { accountUuid: calendarChannelId = '' } = useParams(); - - const { record: calendarChannel, loading } = - useFindOneRecord<CalendarChannel>({ - objectNameSingular: CoreObjectNameSingular.CalendarChannel, - objectRecordId: calendarChannelId, - }); - - const { updateOneRecord } = useUpdateOneRecord<CalendarChannel>({ - objectNameSingular: CoreObjectNameSingular.CalendarChannel, - }); - - const handleVisibilityChange = (value: CalendarChannelVisibility) => { - updateOneRecord({ - idToUpdate: calendarChannelId, - updateOneRecordInput: { - visibility: value, - }, - }); - }; - - const handleContactAutoCreationToggle = (value: boolean) => { - updateOneRecord({ - idToUpdate: calendarChannelId, - updateOneRecordInput: { - isContactAutoCreationEnabled: value, - }, - }); - }; - - const handleSyncEventsToggle = (value: boolean) => { - updateOneRecord({ - idToUpdate: calendarChannelId, - updateOneRecordInput: { - isSyncEnabled: value, - }, - }); - }; - - useEffect(() => { - if (!loading && !calendarChannel) navigate(AppPath.NotFound); - }, [loading, calendarChannel, navigate]); - - if (!calendarChannel) return null; - - return ( - <SubMenuTopBarContainer Icon={IconSettings} title="Settings"> - <SettingsPageContainer> - <Breadcrumb - links={[ - { - children: 'Accounts', - href: getSettingsPagePath(SettingsPath.Accounts), - }, - { - children: 'Calendars', - href: getSettingsPagePath(SettingsPath.AccountsCalendars), - }, - { children: calendarChannel?.handle || '' }, - ]} - /> - <Section> - <H2Title - title="Event visibility" - description="Define what will be visible to other users in your workspace" - /> - <SettingsAccountsEventVisibilitySettingsCard - value={calendarChannel.visibility} - onChange={handleVisibilityChange} - /> - </Section> - <Section> - <H2Title - title="Contact auto-creation" - description="Automatically create contacts for people you've participated in an event with." - /> - <SettingsAccountsToggleSettingCard - cardMedia={ - <StyledCardMedia> - <IconUser - size={theme.icon.size.sm} - stroke={theme.icon.stroke.lg} - /> - </StyledCardMedia> - } - title="Auto-creation" - value={!!calendarChannel.isContactAutoCreationEnabled} - onToggle={handleContactAutoCreationToggle} - /> - </Section> - <Section> - <H2Title - title="Synchronization" - description="Past and future calendar events will automatically be synced to this workspace" - /> - <SettingsAccountsToggleSettingCard - cardMedia={ - <StyledCardMedia> - <IconRefresh - size={theme.icon.size.sm} - stroke={theme.icon.stroke.lg} - /> - </StyledCardMedia> - } - title="Sync events" - value={!!calendarChannel.isSyncEnabled} - onToggle={handleSyncEventsToggle} - /> - </Section> - </SettingsPageContainer> - </SubMenuTopBarContainer> - ); -}; diff --git a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsEmails.tsx b/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsEmails.tsx index d2827187f75c..b972ad215e87 100644 --- a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsEmails.tsx +++ b/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsEmails.tsx @@ -1,6 +1,6 @@ -import { H2Title, IconSettings } from 'twenty-ui'; +import { IconSettings } from 'twenty-ui'; -import { SettingsAccountsMessageChannelsListCard } from '@/settings/accounts/components/SettingsAccountsMessageChannelsListCard'; +import { SettingsAccountsMessageChannelsContainer } from '@/settings/accounts/components/SettingsAccountsMessageChannelsContainer'; import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; import { SubMenuTopBarContainer } from '@/ui/layout/page/SubMenuTopBarContainer'; import { Section } from '@/ui/layout/section/components/Section'; @@ -16,11 +16,7 @@ export const SettingsAccountsEmails = () => ( ]} /> <Section> - <H2Title - title="Emails sync" - description="Sync your inboxes and set your privacy settings" - /> - <SettingsAccountsMessageChannelsListCard /> + <SettingsAccountsMessageChannelsContainer /> </Section> </SettingsPageContainer> </SubMenuTopBarContainer> diff --git a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsEmailsInboxSettings.tsx b/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsEmailsInboxSettings.tsx deleted file mode 100644 index afef0c498d28..000000000000 --- a/packages/twenty-front/src/pages/settings/accounts/SettingsAccountsEmailsInboxSettings.tsx +++ /dev/null @@ -1,128 +0,0 @@ -import { useEffect } from 'react'; -import { useNavigate, useParams } from 'react-router-dom'; -import { useTheme } from '@emotion/react'; -import { H2Title, IconRefresh, IconSettings, IconUser } from 'twenty-ui'; - -import { MessageChannel } from '@/accounts/types/MessageChannel'; -import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; -import { useFindOneRecord } from '@/object-record/hooks/useFindOneRecord'; -import { useUpdateOneRecord } from '@/object-record/hooks/useUpdateOneRecord'; -import { SettingsAccountsCardMedia } from '@/settings/accounts/components/SettingsAccountsCardMedia'; -import { SettingsAccountsInboxVisibilitySettingsCard } from '@/settings/accounts/components/SettingsAccountsInboxVisibilitySettingsCard'; -import { SettingsAccountsToggleSettingCard } from '@/settings/accounts/components/SettingsAccountsToggleSettingCard'; -import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; -import { AppPath } from '@/types/AppPath'; -import { SubMenuTopBarContainer } from '@/ui/layout/page/SubMenuTopBarContainer'; -import { Section } from '@/ui/layout/section/components/Section'; -import { Breadcrumb } from '@/ui/navigation/bread-crumb/components/Breadcrumb'; -import { MessageChannelVisibility } from '~/generated/graphql'; - -export const SettingsAccountsEmailsInboxSettings = () => { - const theme = useTheme(); - const navigate = useNavigate(); - const { accountUuid: messageChannelId = '' } = useParams(); - - const { record: messageChannel, loading } = useFindOneRecord<MessageChannel>({ - objectNameSingular: CoreObjectNameSingular.MessageChannel, - objectRecordId: messageChannelId, - }); - - const { updateOneRecord } = useUpdateOneRecord<MessageChannel>({ - objectNameSingular: CoreObjectNameSingular.MessageChannel, - }); - - const handleVisibilityChange = (value: MessageChannelVisibility) => { - updateOneRecord({ - idToUpdate: messageChannelId, - updateOneRecordInput: { - visibility: value, - }, - }); - }; - - const handleContactAutoCreationToggle = (value: boolean) => { - updateOneRecord({ - idToUpdate: messageChannelId, - updateOneRecordInput: { - isContactAutoCreationEnabled: value, - }, - }); - }; - - const handleIsSyncEnabledToggle = (value: boolean) => { - updateOneRecord({ - idToUpdate: messageChannelId, - updateOneRecordInput: { - isSyncEnabled: value, - }, - }); - }; - - useEffect(() => { - if (!loading && !messageChannel) navigate(AppPath.NotFound); - }, [loading, messageChannel, navigate]); - - if (!messageChannel) return null; - - return ( - <SubMenuTopBarContainer Icon={IconSettings} title="Settings"> - <SettingsPageContainer> - <Breadcrumb - links={[ - { children: 'Accounts', href: '/settings/accounts' }, - { children: 'Emails', href: '/settings/accounts/emails' }, - { children: messageChannel.handle || '' }, - ]} - /> - <Section> - <H2Title - title="Email visibility" - description="Define what will be visible to other users in your workspace" - /> - <SettingsAccountsInboxVisibilitySettingsCard - value={messageChannel.visibility} - onChange={handleVisibilityChange} - /> - </Section> - <Section> - <H2Title - title="Contact auto-creation" - description="Automatically create contacts for people you’ve sent emails to" - /> - <SettingsAccountsToggleSettingCard - cardMedia={ - <SettingsAccountsCardMedia> - <IconUser - size={theme.icon.size.sm} - stroke={theme.icon.stroke.lg} - /> - </SettingsAccountsCardMedia> - } - title="Auto-creation" - value={!!messageChannel.isContactAutoCreationEnabled} - onToggle={handleContactAutoCreationToggle} - /> - </Section> - <Section> - <H2Title - title="Synchronization" - description="Past and future emails will automatically be synced to this workspace" - /> - <SettingsAccountsToggleSettingCard - cardMedia={ - <SettingsAccountsCardMedia> - <IconRefresh - size={theme.icon.size.sm} - stroke={theme.icon.stroke.lg} - /> - </SettingsAccountsCardMedia> - } - title="Sync emails" - value={!!messageChannel.isSyncEnabled} - onToggle={handleIsSyncEnabledToggle} - /> - </Section> - </SettingsPageContainer> - </SubMenuTopBarContainer> - ); -}; diff --git a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsCalendars.stories.tsx b/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsCalendars.stories.tsx index d2134d2f4264..fc184ea50d93 100644 --- a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsCalendars.stories.tsx +++ b/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsCalendars.stories.tsx @@ -1,23 +1,19 @@ import { Meta, StoryObj } from '@storybook/react'; -import { within } from '@storybook/test'; +import { HttpResponse, graphql } from 'msw'; +import { SettingsAccountsCalendars } from '~/pages/settings/accounts/SettingsAccountsCalendars'; -import { getSettingsPagePath } from '@/settings/utils/getSettingsPagePath'; -import { SettingsPath } from '@/types/SettingsPath'; import { PageDecorator, PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; -import { sleep } from '~/utils/sleep'; - -import { SettingsAccountsCalendars } from '../SettingsAccountsCalendars'; const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Settings/Accounts/SettingsAccountsCalendars', component: SettingsAccountsCalendars, decorators: [PageDecorator], args: { - routePath: getSettingsPagePath(SettingsPath.AccountsCalendars), + routePath: '/settings/accounts/calendars', }, parameters: { layout: 'fullscreen', @@ -29,11 +25,120 @@ export default meta; export type Story = StoryObj<typeof SettingsAccountsCalendars>; -export const Default: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); - sleep(100); +export const NoConnectedAccount: Story = {}; - await canvas.findByText('Calendar settings'); +export const TwoConnectedAccounts: Story = { + parameters: { + msw: { + handlers: [ + ...graphqlMocks.handlers, + graphql.query('FindManyConnectedAccounts', () => { + return HttpResponse.json({ + data: { + connectedAccounts: { + __typename: 'ConnectedAccountConnection', + totalCount: 1, + pageInfo: { + __typename: 'PageInfo', + hasNextPage: false, + startCursor: '', + endCursor: '', + }, + edges: [ + { + __typename: 'ConnectedAccountEdge', + cursor: '', + node: { + __typename: 'ConnectedAccount', + accessToken: '', + refreshToken: '', + updatedAt: '2024-07-03T20:03:35.064Z', + createdAt: '2024-07-03T20:03:35.064Z', + id: '20202020-954c-4d76-9a87-e5f072d4b7ef', + provider: 'google', + accountOwnerId: '20202020-03f2-4d83-b0d5-2ec2bcee72d4', + lastSyncHistoryId: '', + handleAliases: '', + handle: 'test.test@gmail.com', + authFailedAt: null, + }, + }, + ], + }, + }, + }); + }), + graphql.query('FindManyCalendarChannels', () => { + return HttpResponse.json({ + data: { + calendarChannels: { + __typename: 'CalendarChannelConnection', + totalCount: 2, + pageInfo: { + __typename: 'PageInfo', + hasNextPage: false, + startCursor: '', + endCursor: '', + }, + edges: [ + { + __typename: 'CalendarChannelEdge', + cursor: '', + node: { + __typename: 'CalendarChannel', + handle: 'test.test@gmail.com', + excludeNonProfessionalEmails: true, + syncStageStartedAt: null, + id: '20202020-ef5a-4822-9e08-ce6e6a4dcb6f', + updatedAt: '2024-07-03T20:03:11.903Z', + createdAt: '2024-07-03T20:03:11.903Z', + connectedAccountId: + '20202020-954c-4d76-9a87-e5f072d4b7ef', + contactAutoCreationPolicy: 'SENT', + syncStage: 'PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING', + type: 'email', + isContactAutoCreationEnabled: true, + syncCursor: '1562764', + excludeGroupEmails: true, + throttleFailureCount: 0, + isSyncEnabled: true, + visibility: 'SHARE_EVERYTHING', + syncStatus: 'COMPLETED', + syncedAt: '2024-07-04T16:25:04.960Z', + }, + }, + { + __typename: 'CalendarChannelEdge', + cursor: '', + node: { + __typename: 'CalendarChannel', + handle: 'test.test2@gmail.com', + excludeNonProfessionalEmails: true, + syncStageStartedAt: null, + id: '20202020-ef5a-4822-9e08-ce6e6a4dcb6a', + updatedAt: '2024-07-03T20:03:11.903Z', + createdAt: '2024-07-03T20:03:11.903Z', + connectedAccountId: + '20202020-954c-4d76-9a87-e5f072d4b7ef', + contactAutoCreationPolicy: 'SENT', + syncStage: 'PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING', + type: 'email', + isContactAutoCreationEnabled: true, + syncCursor: '1562764', + excludeGroupEmails: true, + throttleFailureCount: 0, + isSyncEnabled: true, + visibility: 'SHARE_EVERYTHING', + syncStatus: 'COMPLETED', + syncedAt: '2024-07-04T16:25:04.960Z', + }, + }, + ], + }, + }, + }); + }), + ], + }, }, }; diff --git a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsCalendarsSettings.stories.tsx b/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsCalendarsSettings.stories.tsx deleted file mode 100644 index 1e3ee5934d4d..000000000000 --- a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsCalendarsSettings.stories.tsx +++ /dev/null @@ -1,61 +0,0 @@ -import { Meta, StoryObj } from '@storybook/react'; -import { within } from '@storybook/test'; -import { graphql, HttpResponse } from 'msw'; - -import { getSettingsPagePath } from '@/settings/utils/getSettingsPagePath'; -import { SettingsPath } from '@/types/SettingsPath'; -import { - PageDecorator, - PageDecoratorArgs, -} from '~/testing/decorators/PageDecorator'; -import { graphqlMocks } from '~/testing/graphqlMocks'; -import { mockedConnectedAccounts } from '~/testing/mock-data/accounts'; -import { sleep } from '~/utils/sleep'; - -import { SettingsAccountsCalendarsSettings } from '../SettingsAccountsCalendarsSettings'; - -const meta: Meta<PageDecoratorArgs> = { - title: 'Pages/Settings/Accounts/SettingsAccountsCalendarsSettings', - component: SettingsAccountsCalendarsSettings, - decorators: [PageDecorator], - args: { - routePath: getSettingsPagePath(SettingsPath.AccountsCalendarsSettings), - routeParams: { ':accountUuid': mockedConnectedAccounts[0].id }, - }, - parameters: { - layout: 'fullscreen', - msw: { - handlers: [ - ...graphqlMocks.handlers, - graphql.query('FindOneCalendarChannel', () => { - return HttpResponse.json({ - data: { - calendarChannel: { - edges: [], - pageInfo: { - hasNextPage: false, - hasPreviousPage: false, - startCursor: null, - endCursor: null, - }, - }, - }, - }); - }), - ], - }, - }, -}; - -export default meta; - -export type Story = StoryObj<typeof SettingsAccountsCalendarsSettings>; - -export const Default: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); - sleep(100); - - await canvas.findByText('Event visibility'); - }, -}; diff --git a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsEmails.stories.tsx b/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsEmails.stories.tsx index 21bede0eb563..f0ebac280c23 100644 --- a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsEmails.stories.tsx +++ b/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsEmails.stories.tsx @@ -1,4 +1,5 @@ import { Meta, StoryObj } from '@storybook/react'; +import { graphql, HttpResponse } from 'msw'; import { PageDecorator, @@ -25,4 +26,120 @@ export default meta; export type Story = StoryObj<typeof SettingsAccountsEmails>; -export const Default: Story = {}; +export const NoConnectedAccount: Story = {}; + +export const TwoConnectedAccounts: Story = { + parameters: { + msw: { + handlers: [ + ...graphqlMocks.handlers, + graphql.query('FindManyConnectedAccounts', () => { + return HttpResponse.json({ + data: { + connectedAccounts: { + __typename: 'ConnectedAccountConnection', + totalCount: 1, + pageInfo: { + __typename: 'PageInfo', + hasNextPage: false, + startCursor: '', + endCursor: '', + }, + edges: [ + { + __typename: 'ConnectedAccountEdge', + cursor: '', + node: { + __typename: 'ConnectedAccount', + accessToken: '', + refreshToken: '', + updatedAt: '2024-07-03T20:03:35.064Z', + createdAt: '2024-07-03T20:03:35.064Z', + id: '20202020-954c-4d76-9a87-e5f072d4b7ef', + provider: 'google', + accountOwnerId: '20202020-03f2-4d83-b0d5-2ec2bcee72d4', + lastSyncHistoryId: '', + handleAliases: '', + handle: 'test.test@gmail.com', + authFailedAt: null, + }, + }, + ], + }, + }, + }); + }), + graphql.query('FindManyMessageChannels', () => { + return HttpResponse.json({ + data: { + messageChannels: { + __typename: 'MessageChannelConnection', + totalCount: 2, + pageInfo: { + __typename: 'PageInfo', + hasNextPage: false, + startCursor: '', + endCursor: '', + }, + edges: [ + { + __typename: 'MessageChannelEdge', + cursor: '', + node: { + __typename: 'MessageChannel', + handle: 'test.test@gmail.com', + excludeNonProfessionalEmails: true, + syncStageStartedAt: null, + id: '20202020-ef5a-4822-9e08-ce6e6a4dcb6f', + updatedAt: '2024-07-03T20:03:11.903Z', + createdAt: '2024-07-03T20:03:11.903Z', + connectedAccountId: + '20202020-954c-4d76-9a87-e5f072d4b7ef', + contactAutoCreationPolicy: 'SENT', + syncStage: 'PARTIAL_MESSAGE_LIST_FETCH_PENDING', + type: 'email', + isContactAutoCreationEnabled: true, + syncCursor: '1562764', + excludeGroupEmails: true, + throttleFailureCount: 0, + isSyncEnabled: true, + visibility: 'SHARE_EVERYTHING', + syncStatus: 'COMPLETED', + syncedAt: '2024-07-04T16:25:04.960Z', + }, + }, + { + __typename: 'MessageChannelEdge', + cursor: '', + node: { + __typename: 'MessageChannel', + handle: 'test.test2@gmail.com', + excludeNonProfessionalEmails: true, + syncStageStartedAt: null, + id: '20202020-ef5a-4822-9e08-ce6e6a4dcb6a', + updatedAt: '2024-07-03T20:03:11.903Z', + createdAt: '2024-07-03T20:03:11.903Z', + connectedAccountId: + '20202020-954c-4d76-9a87-e5f072d4b7ef', + contactAutoCreationPolicy: 'SENT', + syncStage: 'PARTIAL_MESSAGE_LIST_FETCH_PENDING', + type: 'email', + isContactAutoCreationEnabled: true, + syncCursor: '1562764', + excludeGroupEmails: true, + throttleFailureCount: 0, + isSyncEnabled: true, + visibility: 'SHARE_EVERYTHING', + syncStatus: 'COMPLETED', + syncedAt: '2024-07-04T16:25:04.960Z', + }, + }, + ], + }, + }, + }); + }), + ], + }, + }, +}; diff --git a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsEmailsInboxSettings.stories.tsx b/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsEmailsInboxSettings.stories.tsx deleted file mode 100644 index 6c23dd71c0ca..000000000000 --- a/packages/twenty-front/src/pages/settings/accounts/__stories__/SettingsAccountsEmailsInboxSettings.stories.tsx +++ /dev/null @@ -1,79 +0,0 @@ -import { Meta, StoryObj } from '@storybook/react'; -import { within } from '@storybook/test'; -import { graphql, HttpResponse } from 'msw'; - -import { MessageChannelVisibility } from '~/generated/graphql'; -import { SettingsAccountsEmailsInboxSettings } from '~/pages/settings/accounts/SettingsAccountsEmailsInboxSettings'; -import { - PageDecorator, - PageDecoratorArgs, -} from '~/testing/decorators/PageDecorator'; -import { graphqlMocks } from '~/testing/graphqlMocks'; - -const meta: Meta<PageDecoratorArgs> = { - title: 'Pages/Settings/Accounts/SettingsAccountsEmailsInboxSettings', - component: SettingsAccountsEmailsInboxSettings, - decorators: [PageDecorator], - args: { - routePath: '/settings/accounts/emails/:accountUuid', - routeParams: { ':accountUuid': '123' }, - }, - parameters: { - layout: 'fullscreen', - msw: { - handlers: [ - graphql.query('FindOneMessageChannel', () => { - return HttpResponse.json({ - data: { - messageChannel: { - id: '1', - visibility: MessageChannelVisibility.ShareEverything, - messageThreads: { edges: [] }, - createdAt: '2021-08-27T12:00:00Z', - type: 'email', - updatedAt: '2021-08-27T12:00:00Z', - targetUrl: 'https://example.com/webhook', - connectedAccountId: '1', - handle: 'handle', - connectedAccount: { - id: '1', - handle: 'handle', - updatedAt: '2021-08-27T12:00:00Z', - accessToken: 'accessToken', - messageChannels: { edges: [] }, - refreshToken: 'refreshToken', - __typename: 'ConnectedAccount', - accountOwner: { id: '1', __typename: 'WorkspaceMember' }, - provider: 'provider', - createdAt: '2021-08-27T12:00:00Z', - accountOwnerId: '1', - }, - __typename: 'MessageChannel', - }, - }, - }); - }), - graphqlMocks.handlers, - ], - }, - }, -}; - -export default meta; - -export type Story = StoryObj<typeof SettingsAccountsEmailsInboxSettings>; - -export const Default: Story = { - play: async ({ canvasElement }) => { - const canvas = within(canvasElement); - - await canvas.findByText('Email visibility'); - await canvas.findByText( - 'Define what will be visible to other users in your workspace', - ); - await canvas.findByText('Contact auto-creation'); - await canvas.findByText( - 'Automatically create contacts for people you’ve sent emails to', - ); - }, -}; diff --git a/packages/twenty-front/src/pages/settings/data-model/SettingsNewObject.tsx b/packages/twenty-front/src/pages/settings/data-model/SettingsNewObject.tsx index cfb6f203b04d..7adb8ee74f2c 100644 --- a/packages/twenty-front/src/pages/settings/data-model/SettingsNewObject.tsx +++ b/packages/twenty-front/src/pages/settings/data-model/SettingsNewObject.tsx @@ -1,6 +1,6 @@ +import { zodResolver } from '@hookform/resolvers/zod'; import { FormProvider, useForm } from 'react-hook-form'; import { useNavigate } from 'react-router-dom'; -import { zodResolver } from '@hookform/resolvers/zod'; import { H2Title, IconSettings } from 'twenty-ui'; import { z } from 'zod'; @@ -40,8 +40,8 @@ export const SettingsNewObject = () => { resolver: zodResolver(newObjectFormSchema), }); - const canSave = - formConfig.formState.isValid && !formConfig.formState.isSubmitting; + const { isValid, isSubmitting } = formConfig.formState; + const canSave = isValid && !isSubmitting; const handleSave = async ( formValues: SettingsDataModelNewObjectFormValues, @@ -84,6 +84,7 @@ export const SettingsNewObject = () => { /> <SaveAndCancelButtons isSaveDisabled={!canSave} + isCancelDisabled={isSubmitting} onCancel={() => navigate(settingsObjectsPagePath)} onSave={formConfig.handleSubmit(handleSave)} /> diff --git a/packages/twenty-front/src/pages/settings/data-model/SettingsObjectEdit.tsx b/packages/twenty-front/src/pages/settings/data-model/SettingsObjectEdit.tsx index e12103ee9f8e..82fdb67b8768 100644 --- a/packages/twenty-front/src/pages/settings/data-model/SettingsObjectEdit.tsx +++ b/packages/twenty-front/src/pages/settings/data-model/SettingsObjectEdit.tsx @@ -1,8 +1,8 @@ +import { zodResolver } from '@hookform/resolvers/zod'; +import pick from 'lodash.pick'; import { useEffect } from 'react'; import { FormProvider, useForm } from 'react-hook-form'; import { useNavigate, useParams } from 'react-router-dom'; -import { zodResolver } from '@hookform/resolvers/zod'; -import pick from 'lodash.pick'; import { H2Title, IconArchive, IconSettings } from 'twenty-ui'; import { z } from 'zod'; @@ -81,7 +81,12 @@ export const SettingsObjectEdit = () => { ), }); - navigate(`${settingsObjectsPagePath}/${getObjectSlug(formValues)}`); + navigate( + `${settingsObjectsPagePath}/${getObjectSlug({ + ...formValues, + namePlural: formValues.labelPlural, + })}`, + ); } catch (error) { enqueueSnackBar((error as Error).message, { variant: SnackBarVariant.Error, @@ -119,6 +124,7 @@ export const SettingsObjectEdit = () => { {activeObjectMetadataItem.isCustom && ( <SaveAndCancelButtons isSaveDisabled={!canSave} + isCancelDisabled={isSubmitting} onCancel={() => navigate(`${settingsObjectsPagePath}/${objectSlug}`) } diff --git a/packages/twenty-front/src/pages/settings/data-model/SettingsObjectFieldEdit.tsx b/packages/twenty-front/src/pages/settings/data-model/SettingsObjectFieldEdit.tsx index bb2ac5e637e4..fdc02bcdd33e 100644 --- a/packages/twenty-front/src/pages/settings/data-model/SettingsObjectFieldEdit.tsx +++ b/packages/twenty-front/src/pages/settings/data-model/SettingsObjectFieldEdit.tsx @@ -1,11 +1,11 @@ -import { useEffect } from 'react'; -import { FormProvider, useForm } from 'react-hook-form'; -import { useNavigate, useParams } from 'react-router-dom'; import { useApolloClient } from '@apollo/client'; import styled from '@emotion/styled'; import { zodResolver } from '@hookform/resolvers/zod'; import omit from 'lodash.omit'; import pick from 'lodash.pick'; +import { useEffect } from 'react'; +import { FormProvider, useForm } from 'react-hook-form'; +import { useNavigate, useParams } from 'react-router-dom'; import { H2Title, IconArchive, IconSettings } from 'twenty-ui'; import { z } from 'zod'; @@ -22,6 +22,7 @@ import { RecordFieldValueSelectorContextProvider } from '@/object-record/record- import { SaveAndCancelButtons } from '@/settings/components/SaveAndCancelButtons/SaveAndCancelButtons'; import { SettingsHeaderContainer } from '@/settings/components/SettingsHeaderContainer'; import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; +import { FIELD_NAME_MAXIMUM_LENGTH } from '@/settings/data-model/constants/FieldNameMaximumLength'; import { SettingsDataModelFieldAboutForm } from '@/settings/data-model/fields/forms/components/SettingsDataModelFieldAboutForm'; import { SettingsDataModelFieldSettingsFormCard } from '@/settings/data-model/fields/forms/components/SettingsDataModelFieldSettingsFormCard'; import { SettingsDataModelFieldTypeSelect } from '@/settings/data-model/fields/forms/components/SettingsDataModelFieldTypeSelect'; @@ -103,10 +104,8 @@ export const SettingsObjectFieldEdit = () => { if (!activeObjectMetadataItem || !activeMetadataField) return null; - const canSave = - formConfig.formState.isValid && - formConfig.formState.isDirty && - !formConfig.formState.isSubmitting; + const { isDirty, isValid, isSubmitting } = formConfig.formState; + const canSave = isDirty && isValid && !isSubmitting; const isLabelIdentifier = isLabelIdentifierField({ fieldMetadataItem: activeMetadataField, @@ -189,6 +188,7 @@ export const SettingsObjectFieldEdit = () => { {shouldDisplaySaveAndCancel && ( <SaveAndCancelButtons isSaveDisabled={!canSave} + isCancelDisabled={isSubmitting} onCancel={() => navigate(`/settings/objects/${objectSlug}`)} onSave={formConfig.handleSubmit(handleSave)} /> @@ -202,6 +202,7 @@ export const SettingsObjectFieldEdit = () => { <SettingsDataModelFieldAboutForm disabled={!activeMetadataField.isCustom} fieldMetadataItem={activeMetadataField} + maxLength={FIELD_NAME_MAXIMUM_LENGTH} /> </Section> <Section> @@ -212,6 +213,7 @@ export const SettingsObjectFieldEdit = () => { <StyledSettingsObjectFieldTypeSelect disabled fieldMetadataItem={activeMetadataField} + excludedFieldTypes={[FieldMetadataType.Link]} /> <SettingsDataModelFieldSettingsFormCard disableCurrencyForm diff --git a/packages/twenty-front/src/pages/settings/data-model/SettingsObjectNewField/SettingsObjectNewFieldStep2.tsx b/packages/twenty-front/src/pages/settings/data-model/SettingsObjectNewField/SettingsObjectNewFieldStep2.tsx index da7fa13d2559..a2dca70beab8 100644 --- a/packages/twenty-front/src/pages/settings/data-model/SettingsObjectNewField/SettingsObjectNewFieldStep2.tsx +++ b/packages/twenty-front/src/pages/settings/data-model/SettingsObjectNewField/SettingsObjectNewFieldStep2.tsx @@ -1,10 +1,10 @@ -import { useEffect, useState } from 'react'; -import { FormProvider, useForm } from 'react-hook-form'; -import { useNavigate, useParams } from 'react-router-dom'; import { useApolloClient } from '@apollo/client'; import styled from '@emotion/styled'; import { zodResolver } from '@hookform/resolvers/zod'; import pick from 'lodash.pick'; +import { useEffect, useState } from 'react'; +import { FormProvider, useForm } from 'react-hook-form'; +import { useNavigate, useParams } from 'react-router-dom'; import { H2Title, IconSettings } from 'twenty-ui'; import { z } from 'zod'; @@ -17,6 +17,7 @@ import { RecordFieldValueSelectorContextProvider } from '@/object-record/record- import { SaveAndCancelButtons } from '@/settings/components/SaveAndCancelButtons/SaveAndCancelButtons'; import { SettingsHeaderContainer } from '@/settings/components/SettingsHeaderContainer'; import { SettingsPageContainer } from '@/settings/components/SettingsPageContainer'; +import { FIELD_NAME_MAXIMUM_LENGTH } from '@/settings/data-model/constants/FieldNameMaximumLength'; import { SettingsDataModelFieldAboutForm } from '@/settings/data-model/fields/forms/components/SettingsDataModelFieldAboutForm'; import { SettingsDataModelFieldSettingsFormCard } from '@/settings/data-model/fields/forms/components/SettingsDataModelFieldSettingsFormCard'; import { SettingsDataModelFieldTypeSelect } from '@/settings/data-model/fields/forms/components/SettingsDataModelFieldTypeSelect'; @@ -108,8 +109,8 @@ export const SettingsObjectNewFieldStep2 = () => { if (!activeObjectMetadataItem) return null; - const canSave = - formConfig.formState.isValid && !formConfig.formState.isSubmitting; + const { isValid, isSubmitting } = formConfig.formState; + const canSave = isValid && !isSubmitting; const handleSave = async ( formValues: SettingsDataModelNewFieldFormValues, @@ -133,26 +134,20 @@ export const SettingsObjectNewFieldStep2 = () => { objectMetadataId: relationFormValues.objectMetadataId, }, }); - - // TODO: fix optimistic update logic - // Forcing a refetch for now but it's not ideal - await apolloClient.refetchQueries({ - include: ['FindManyViews', 'CombinedFindManyRecords'], - }); } else { await createMetadataField({ ...formValues, objectMetadataId: activeObjectMetadataItem.id, }); - - // TODO: fix optimistic update logic - // Forcing a refetch for now but it's not ideal - await apolloClient.refetchQueries({ - include: ['FindManyViews', 'CombinedFindManyRecords'], - }); } navigate(`/settings/objects/${objectSlug}`); + + // TODO: fix optimistic update logic + // Forcing a refetch for now but it's not ideal + await apolloClient.refetchQueries({ + include: ['FindManyViews', 'CombinedFindManyRecords'], + }); } catch (error) { enqueueSnackBar((error as Error).message, { variant: SnackBarVariant.Error, @@ -164,9 +159,9 @@ export const SettingsObjectNewFieldStep2 = () => { [ // FieldMetadataType.Email, // FieldMetadataType.FullName, - // FieldMetadataType.Link, + FieldMetadataType.Link, FieldMetadataType.Numeric, - FieldMetadataType.Probability, + // FieldMetadataType.Probability, // FieldMetadataType.Uuid, // FieldMetadataType.Phone, ] as const @@ -192,6 +187,7 @@ export const SettingsObjectNewFieldStep2 = () => { {!activeObjectMetadataItem.isRemote && ( <SaveAndCancelButtons isSaveDisabled={!canSave} + isCancelDisabled={isSubmitting} onCancel={() => navigate(`/settings/objects/${objectSlug}`)} onSave={formConfig.handleSubmit(handleSave)} /> @@ -202,7 +198,9 @@ export const SettingsObjectNewFieldStep2 = () => { title="Name and description" description="The name and description of this field" /> - <SettingsDataModelFieldAboutForm /> + <SettingsDataModelFieldAboutForm + maxLength={FIELD_NAME_MAXIMUM_LENGTH} + /> </Section> <Section> <H2Title diff --git a/packages/twenty-front/src/pages/settings/data-model/__stories__/SettingsObjectDetail.stories.tsx b/packages/twenty-front/src/pages/settings/data-model/__stories__/SettingsObjectDetail.stories.tsx index 0e4b4eeac4a1..687cf19a2c9b 100644 --- a/packages/twenty-front/src/pages/settings/data-model/__stories__/SettingsObjectDetail.stories.tsx +++ b/packages/twenty-front/src/pages/settings/data-model/__stories__/SettingsObjectDetail.stories.tsx @@ -35,15 +35,13 @@ export const StandardObject: Story = { export const CustomObject: Story = { args: { - routeParams: { ':objectSlug': 'workspaces' }, + routeParams: { ':objectSlug': 'my-customs' }, }, }; export const ObjectDropdownMenu: Story = { - play: async ({ canvasElement }) => { - await sleep(100); - - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const objectSummaryVerticalDotsIconButton = await canvas.findByRole( 'button', { @@ -59,10 +57,8 @@ export const ObjectDropdownMenu: Story = { }; export const FieldDropdownMenu: Story = { - play: async ({ canvasElement }) => { - await sleep(100); - - const canvas = within(canvasElement); + play: async () => { + const canvas = within(document.body); const [fieldVerticalDotsIconButton] = await canvas.findAllByRole('button', { name: 'Active Field Options', }); diff --git a/packages/twenty-front/src/pages/settings/developers/__stories__/api-keys/SettingsDevelopersApiKeysDetail.stories.tsx b/packages/twenty-front/src/pages/settings/developers/__stories__/api-keys/SettingsDevelopersApiKeysDetail.stories.tsx index cb5902051781..4428ce4e1bd5 100644 --- a/packages/twenty-front/src/pages/settings/developers/__stories__/api-keys/SettingsDevelopersApiKeysDetail.stories.tsx +++ b/packages/twenty-front/src/pages/settings/developers/__stories__/api-keys/SettingsDevelopersApiKeysDetail.stories.tsx @@ -1,6 +1,6 @@ import { Meta, StoryObj } from '@storybook/react'; -import { within } from '@storybook/test'; -import { graphql, HttpResponse } from 'msw'; +import { userEvent, within } from '@storybook/test'; +import { HttpResponse, graphql } from 'msw'; import { SettingsDevelopersApiKeyDetail } from '~/pages/settings/developers/api-keys/SettingsDevelopersApiKeyDetail'; import { @@ -8,6 +8,7 @@ import { PageDecoratorArgs, } from '~/testing/decorators/PageDecorator'; import { graphqlMocks } from '~/testing/graphqlMocks'; +import { sleep } from '~/utils/sleep'; const meta: Meta<PageDecoratorArgs> = { title: 'Pages/Settings/Developers/ApiKeys/SettingsDevelopersApiKeyDetail', @@ -50,6 +51,56 @@ export const Default: Story = { play: async ({ canvasElement }) => { const canvas = within(canvasElement); await canvas.findByText('Settings'); - await canvas.findByText('Regenerate an Api key'); + await canvas.findByText('sfsfdsf API Key'); + }, +}; + +export const RegenerateApiKey: Story = { + play: async ({ canvasElement, step }) => { + const canvas = within(canvasElement); + await canvas.findByText('Settings'); + + await userEvent.click(await canvas.findByText('Regenerate Key')); + + await canvas.findByText('Cancel'); + const confirmationInput = await canvas.findByPlaceholderText('yes'); + await userEvent.click(confirmationInput); + await userEvent.keyboard('y'); + await userEvent.keyboard('e'); + await userEvent.keyboard('s'); + + const confirmButton = await canvas.findByTestId( + 'confirmation-modal-confirm-button', + ); + + await step('Click on confirm button', async () => { + await sleep(1000); + await userEvent.click(confirmButton); + }); + }, +}; + +export const DeleteApiKey: Story = { + play: async ({ canvasElement, step }) => { + const canvas = within(canvasElement); + await canvas.findByText('Settings'); + + await userEvent.click(await canvas.findByText('Delete')); + + await canvas.findByText('Cancel'); + const confirmationInput = await canvas.findByPlaceholderText('yes'); + await userEvent.click(confirmationInput); + await userEvent.keyboard('y'); + await userEvent.keyboard('e'); + await userEvent.keyboard('s'); + + const confirmButton = await canvas.findByTestId( + 'confirmation-modal-confirm-button', + ); + + await step('Click on confirm button', async () => { + await sleep(1000); + await userEvent.click(confirmButton); + }); }, }; diff --git a/packages/twenty-front/src/pages/settings/developers/api-keys/SettingsDevelopersApiKeyDetail.tsx b/packages/twenty-front/src/pages/settings/developers/api-keys/SettingsDevelopersApiKeyDetail.tsx index 0a718cf7697f..2f3987e81b83 100644 --- a/packages/twenty-front/src/pages/settings/developers/api-keys/SettingsDevelopersApiKeyDetail.tsx +++ b/packages/twenty-front/src/pages/settings/developers/api-keys/SettingsDevelopersApiKeyDetail.tsx @@ -1,8 +1,8 @@ -import { useEffect, useState } from 'react'; -import { useNavigate, useParams } from 'react-router-dom'; import styled from '@emotion/styled'; import { isNonEmptyString } from '@sniptt/guards'; import { DateTime } from 'luxon'; +import { useEffect, useState } from 'react'; +import { useNavigate, useParams } from 'react-router-dom'; import { useRecoilState } from 'recoil'; import { H2Title, IconRepeat, IconSettings, IconTrash } from 'twenty-ui'; diff --git a/packages/twenty-front/src/pages/tasks/TasksEffect.tsx b/packages/twenty-front/src/pages/tasks/TasksEffect.tsx index c3d5e2daf743..9476c755f285 100644 --- a/packages/twenty-front/src/pages/tasks/TasksEffect.tsx +++ b/packages/twenty-front/src/pages/tasks/TasksEffect.tsx @@ -1,5 +1,6 @@ import { useEffect } from 'react'; import { useRecoilValue } from 'recoil'; +import { v4 } from 'uuid'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown'; @@ -29,6 +30,7 @@ export const TasksEffect = ({ filterDropdownId }: TasksEffectProps) => { useEffect(() => { if (isDefined(currentWorkspaceMember)) { setSelectedFilter({ + id: v4(), fieldMetadataId: 'assigneeId', value: JSON.stringify(currentWorkspaceMember.id), operand: ViewFilterOperand.Is, diff --git a/packages/twenty-front/src/testing/decorators/ChipGeneratorsDecorator.tsx b/packages/twenty-front/src/testing/decorators/ChipGeneratorsDecorator.tsx index 6f9ac5d305bd..283e7046e0ab 100644 --- a/packages/twenty-front/src/testing/decorators/ChipGeneratorsDecorator.tsx +++ b/packages/twenty-front/src/testing/decorators/ChipGeneratorsDecorator.tsx @@ -1,21 +1,21 @@ -import { useMemo } from 'react'; import { Decorator } from '@storybook/react'; +import { useMemo } from 'react'; import { PreComputedChipGeneratorsContext } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; -import { getRecordChipGeneratorPerObjectPerField } from '@/object-record/utils/getRecordChipGeneratorPerObjectPerField'; +import { getRecordChipGenerators } from '@/object-record/utils/getRecordChipGenerators'; import { generatedMockObjectMetadataItems } from '~/testing/mock-data/objectMetadataItems'; export const ChipGeneratorsDecorator: Decorator = (Story) => { - const chipGeneratorPerObjectPerField = useMemo(() => { - return getRecordChipGeneratorPerObjectPerField( - generatedMockObjectMetadataItems, - ); - }, []); + const { chipGeneratorPerObjectPerField, identifierChipGeneratorPerObject } = + useMemo(() => { + return getRecordChipGenerators(generatedMockObjectMetadataItems); + }, []); return ( <PreComputedChipGeneratorsContext.Provider value={{ chipGeneratorPerObjectPerField, + identifierChipGeneratorPerObject, }} > <Story /> diff --git a/packages/twenty-front/src/testing/decorators/ObjectMetadataItemsDecorator.tsx b/packages/twenty-front/src/testing/decorators/ObjectMetadataItemsDecorator.tsx index 9c27a82e354f..c1582edbae07 100644 --- a/packages/twenty-front/src/testing/decorators/ObjectMetadataItemsDecorator.tsx +++ b/packages/twenty-front/src/testing/decorators/ObjectMetadataItemsDecorator.tsx @@ -1,14 +1,13 @@ -import { useEffect, useMemo } from 'react'; import { Decorator } from '@storybook/react'; +import { useEffect } from 'react'; import { useRecoilValue, useSetRecoilState } from 'recoil'; import { currentUserState } from '@/auth/states/currentUserState'; import { currentWorkspaceMemberState } from '@/auth/states/currentWorkspaceMemberState'; import { ObjectMetadataItemsLoadEffect } from '@/object-metadata/components/ObjectMetadataItemsLoadEffect'; -import { PreComputedChipGeneratorsContext } from '@/object-metadata/context/PreComputedChipGeneratorsContext'; +import { PreComputedChipGeneratorsProvider } from '@/object-metadata/components/PreComputedChipGeneratorsProvider'; import { objectMetadataItemsState } from '@/object-metadata/states/objectMetadataItemsState'; -import { getRecordChipGeneratorPerObjectPerField } from '@/object-record/utils/getRecordChipGeneratorPerObjectPerField'; -import { mockedUsersData } from '~/testing/mock-data/users'; +import { mockedUserData } from '~/testing/mock-data/users'; import { mockWorkspaceMembers } from '~/testing/mock-data/workspace-members'; export const ObjectMetadataItemsDecorator: Decorator = (Story) => { @@ -20,23 +19,15 @@ export const ObjectMetadataItemsDecorator: Decorator = (Story) => { useEffect(() => { setCurrentWorkspaceMember(mockWorkspaceMembers[0]); - setCurrentUser(mockedUsersData[0]); + setCurrentUser(mockedUserData); }, [setCurrentUser, setCurrentWorkspaceMember]); - const chipGeneratorPerObjectPerField = useMemo(() => { - return getRecordChipGeneratorPerObjectPerField(objectMetadataItems); - }, [objectMetadataItems]); - return ( <> <ObjectMetadataItemsLoadEffect /> - <PreComputedChipGeneratorsContext.Provider - value={{ - chipGeneratorPerObjectPerField, - }} - > + <PreComputedChipGeneratorsProvider> {!!objectMetadataItems.length && <Story />} - </PreComputedChipGeneratorsContext.Provider> + </PreComputedChipGeneratorsProvider> </> ); }; diff --git a/packages/twenty-front/src/testing/decorators/PageDecorator.tsx b/packages/twenty-front/src/testing/decorators/PageDecorator.tsx index b6e842d26c52..19453453c664 100644 --- a/packages/twenty-front/src/testing/decorators/PageDecorator.tsx +++ b/packages/twenty-front/src/testing/decorators/PageDecorator.tsx @@ -1,3 +1,6 @@ +import { ApolloProvider } from '@apollo/client'; +import { loadDevMessages } from '@apollo/client/dev'; +import { Decorator } from '@storybook/react'; import { HelmetProvider } from 'react-helmet-async'; import { createMemoryRouter, @@ -6,9 +9,6 @@ import { Route, RouterProvider, } from 'react-router-dom'; -import { ApolloProvider } from '@apollo/client'; -import { loadDevMessages } from '@apollo/client/dev'; -import { Decorator } from '@storybook/react'; import { RecoilRoot } from 'recoil'; import { ClientConfigProviderEffect } from '@/client-config/components/ClientConfigProviderEffect'; @@ -21,6 +21,7 @@ import { DefaultLayout } from '~/modules/ui/layout/page/DefaultLayout'; import { UserProvider } from '~/modules/users/components/UserProvider'; import { mockedApolloClient } from '~/testing/mockedApolloClient'; +import { PrefetchDataProvider } from '@/prefetch/components/PrefetchDataProvider'; import { FullHeightStorybookLayout } from '../FullHeightStorybookLayout'; export type PageDecoratorArgs = { @@ -73,7 +74,9 @@ const Providers = () => { <HelmetProvider> <SnackBarProviderScope snackBarManagerScopeId="snack-bar-manager"> <ObjectMetadataItemsProvider> - <Outlet /> + <PrefetchDataProvider> + <Outlet /> + </PrefetchDataProvider> </ObjectMetadataItemsProvider> </SnackBarProviderScope> </HelmetProvider> diff --git a/packages/twenty-front/src/testing/decorators/RecordStoreDecorator.tsx b/packages/twenty-front/src/testing/decorators/RecordStoreDecorator.tsx index 96fcda4ab443..77705e775418 100644 --- a/packages/twenty-front/src/testing/decorators/RecordStoreDecorator.tsx +++ b/packages/twenty-front/src/testing/decorators/RecordStoreDecorator.tsx @@ -1,15 +1,15 @@ import { useEffect } from 'react'; import { Decorator } from '@storybook/react'; -import { useSetRecordInStore } from '@/object-record/record-store/hooks/useSetRecordInStore'; +import { useUpsertRecordsInStore } from '@/object-record/record-store/hooks/useUpsertRecordsInStore'; export const RecordStoreDecorator: Decorator = (Story, context) => { const { records } = context.parameters; - const { setRecords } = useSetRecordInStore(); + const { upsertRecords } = useUpsertRecordsInStore(); useEffect(() => { - setRecords(records); + upsertRecords(records); }); return <Story />; diff --git a/packages/twenty-front/src/testing/graphqlMocks.ts b/packages/twenty-front/src/testing/graphqlMocks.ts index c55061831824..696ba7eac5b4 100644 --- a/packages/twenty-front/src/testing/graphqlMocks.ts +++ b/packages/twenty-front/src/testing/graphqlMocks.ts @@ -15,7 +15,7 @@ import { mockedClientConfig } from '~/testing/mock-data/config'; import { mockedObjectMetadataItemsQueryResult } from '~/testing/mock-data/metadata'; import { getPeopleMock } from '~/testing/mock-data/people'; import { mockedRemoteTables } from '~/testing/mock-data/remote-tables'; -import { mockedUsersData } from '~/testing/mock-data/users'; +import { mockedUserData } from '~/testing/mock-data/users'; import { mockedViewsData } from '~/testing/mock-data/views'; import { mockWorkspaceMembers } from '~/testing/mock-data/workspace-members'; @@ -35,7 +35,7 @@ export const graphqlMocks = { graphql.query(getOperationName(GET_CURRENT_USER) ?? '', () => { return HttpResponse.json({ data: { - currentUser: mockedUsersData[0], + currentUser: mockedUserData, }, }); }), @@ -110,6 +110,57 @@ export const graphqlMocks = { }, }); }), + graphql.query('CombinedFindManyRecords', () => { + return HttpResponse.json({ + data: { + views: { + edges: mockedViewsData.map((view) => ({ + node: { + ...view, + viewFilters: { + edges: [], + totalCount: 0, + }, + viewSorts: { + edges: [], + totalCount: 0, + }, + viewFields: { + edges: mockedViewFieldsData + .filter((viewField) => viewField.viewId === view.id) + .map((viewField) => ({ + node: viewField, + cursor: null, + })), + totalCount: mockedViewFieldsData.filter( + (viewField) => viewField.viewId === view.id, + ).length, + }, + }, + cursor: null, + })), + pageInfo: { + hasNextPage: false, + hasPreviousPage: false, + startCursor: null, + endCursor: null, + totalCount: mockedViewsData.length, + }, + totalCount: mockedViewsData.length, + }, + }, + favorites: { + edges: [], + totalCount: 0, + pageInfo: { + hasNextPage: false, + hasPreviousPage: false, + startCursor: null, + endCursor: null, + }, + }, + }); + }), graphql.query('FindManyCompanies', ({ variables }) => { const mockedData = variables.limit ? companiesMock.slice(0, variables.limit) @@ -157,43 +208,44 @@ export const graphqlMocks = { graphql.query('FindDuplicateCompany', () => { return HttpResponse.json({ data: { - companyDuplicates: { - edges: [ - { - node: { - ...duplicateCompanyMock, - favorites: { - edges: [], - __typename: 'FavoriteConnection', - }, - attachments: { - edges: [], - __typename: 'AttachmentConnection', - }, - people: { - edges: [], - __typename: 'PersonConnection', - }, - opportunities: { - edges: [], - __typename: 'OpportunityConnection', - }, - activityTargets: { - edges: [], - __typename: 'ActivityTargetConnection', + companyDuplicates: [ + { + edges: [ + { + node: { + ...duplicateCompanyMock, + favorites: { + edges: [], + __typename: 'FavoriteConnection', + }, + attachments: { + edges: [], + __typename: 'AttachmentConnection', + }, + people: { + edges: [], + __typename: 'PersonConnection', + }, + opportunities: { + edges: [], + __typename: 'OpportunityConnection', + }, + activityTargets: { + edges: [], + __typename: 'ActivityTargetConnection', + }, }, + cursor: null, }, - cursor: null, + ], + pageInfo: { + hasNextPage: false, + hasPreviousPage: false, + startCursor: null, + endCursor: null, }, - ], - pageInfo: { - hasNextPage: false, - hasPreviousPage: false, - startCursor: null, - endCursor: null, }, - totalCount: 1, - }, + ], }, }); }), diff --git a/packages/twenty-front/src/testing/mock-data/companies.ts b/packages/twenty-front/src/testing/mock-data/companies.ts index a2fc79b6f370..41e7751cc802 100644 --- a/packages/twenty-front/src/testing/mock-data/companies.ts +++ b/packages/twenty-front/src/testing/mock-data/companies.ts @@ -47,7 +47,9 @@ export const companiesQueryResult = { id: '20202020-3ec3-4fe3-8997-b76aa0bfa408', employees: 100, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Linkedin', + accountOwnerId: null, accountOwner: null, domainName: 'linkedin.com', address: '', @@ -58,6 +60,16 @@ export const companiesQueryResult = { label: '', url: '', }, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, previousEmployees: { __typename: 'Person', id: '20202020-2d40-4e49-8df4-9c6a049191de', @@ -132,13 +144,25 @@ export const companiesQueryResult = { id: '20202020-5d81-46d6-bf83-f7fd33ea6102', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Facebook', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'facebook.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 2, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -154,13 +178,25 @@ export const companiesQueryResult = { id: '20202020-0713-40a5-8216-82802401d33e', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Qonto', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'qonto.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 3, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -176,13 +212,25 @@ export const companiesQueryResult = { id: '20202020-ed89-413a-b31a-962986e67bb4', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Microsoft', idealCustomerProfile: true, accountOwner: null, + accountOwnerId: null, domainName: 'microsoft.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 4, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -198,13 +246,25 @@ export const companiesQueryResult = { id: '20202020-171e-4bcc-9cf7-43448d6fb278', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Airbnb', idealCustomerProfile: true, accountOwner: null, + accountOwnerId: null, domainName: 'airbnb.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 5, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -220,13 +280,25 @@ export const companiesQueryResult = { id: '20202020-c21e-4ec2-873b-de4264d89025', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Google', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'google.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 6, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -242,13 +314,25 @@ export const companiesQueryResult = { id: '20202020-707e-44dc-a1d2-30030bf1a944', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Netflix', idealCustomerProfile: true, accountOwner: null, + accountOwnerId: null, domainName: 'netflix.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 7, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -264,13 +348,25 @@ export const companiesQueryResult = { id: '20202020-3f74-492d-a101-2a70f50a1645', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Libeo', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'libeo.io', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 8, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -286,13 +382,25 @@ export const companiesQueryResult = { id: '20202020-cfbf-4156-a790-e39854dcd4eb', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Claap', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'claap.io', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 9, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -308,13 +416,25 @@ export const companiesQueryResult = { id: '20202020-f86b-419f-b794-02319abe8637', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Hasura', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'hasura.io', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 10, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -330,13 +450,25 @@ export const companiesQueryResult = { id: '20202020-5518-4553-9433-42d8eb82834b', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Wework', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'wework.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 11, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -352,13 +484,25 @@ export const companiesQueryResult = { id: '20202020-f79e-40dd-bd06-c36e6abb4678', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Samsung', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'samsung.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 12, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', @@ -374,13 +518,25 @@ export const companiesQueryResult = { id: '20202020-1455-4c57-afaf-dd5dc086361d', employees: null, createdAt: '2024-06-05T09:00:20.412Z', + updatedAt: '2024-06-05T09:00:20.412Z', name: 'Algolia', idealCustomerProfile: false, accountOwner: null, + accountOwnerId: null, domainName: 'algolia.com', address: '', previousEmployees: null, + annualRecurringRevenue: { + __typename: 'Currency', + amountMicros: null, + currencyCode: '', + }, position: 13, + xLink: { + __typename: 'Link', + label: '', + url: '', + }, linkedinLink: { __typename: 'Link', label: '', diff --git a/packages/twenty-front/src/testing/mock-data/config.ts b/packages/twenty-front/src/testing/mock-data/config.ts index b3eb74af804c..a14ea35bf85e 100644 --- a/packages/twenty-front/src/testing/mock-data/config.ts +++ b/packages/twenty-front/src/testing/mock-data/config.ts @@ -34,8 +34,9 @@ export const mockedClientConfig: ClientConfig = { __typename: 'Billing', }, captcha: { - provider: CaptchaDriverType.GoogleRecatpcha, + provider: CaptchaDriverType.GoogleRecaptcha, siteKey: 'MOCKED_SITE_KEY', __typename: 'Captcha', }, + api: { mutationMaximumAffectedRecords: 100 }, }; diff --git a/packages/twenty-front/src/testing/mock-data/generated/standard-metadata-query-result.ts b/packages/twenty-front/src/testing/mock-data/generated/standard-metadata-query-result.ts index c31eaaa65788..b537a84c2b0d 100644 --- a/packages/twenty-front/src/testing/mock-data/generated/standard-metadata-query-result.ts +++ b/packages/twenty-front/src/testing/mock-data/generated/standard-metadata-query-result.ts @@ -1,10 +1,6 @@ -import { CurrencyCode } from '@/object-record/record-field/types/CurrencyCode'; import { - FieldMetadataType, - ObjectEdge, - ObjectMetadataItemsQuery, + ObjectMetadataItemsQuery } from '~/generated-metadata/graphql'; -import { CalendarChannelVisibility, MessageChannelVisibility } from "~/generated/graphql"; // This file is not designed to be manually edited. // It's an extract from the dev seeded environment metadata call @@ -2229,6 +2225,60 @@ export const mockedStandardObjectMetadataQueryResult: ObjectMetadataItemsQuery = "toRelationMetadata": null } }, + { + "__typename": "fieldEdge", + "node": { + "__typename": "field", + "id": "3715c0ac-c16f-4db3-b9be-e908b787929e", + "type": "RATING", + "name": "testRating", + "label": "Test rating", + "description": null, + "icon": "IconUsers", + "isCustom": true, + "isActive": true, + "isSystem": false, + "isNullable": true, + "createdAt": "2024-06-17T13:03:52.175Z", + "updatedAt": "2024-06-17T13:03:52.175Z", + "defaultValue": null, + "options": [ + { + "id": "9876aaeb-91ac-4e02-b521-356ff0c0a6f9", + "label": "1", + "value": "RATING_1", + "position": 0 + }, + { + "id": "4651d042-0804-465b-8265-5fae554de3a8", + "label": "2", + "value": "RATING_2", + "position": 1 + }, + { + "id": "a6942bdd-a8c8-44f9-87fc-b9a7f64ee5dd", + "label": "3", + "value": "RATING_3", + "position": 2 + }, + { + "id": "a838666f-cd2f-4feb-a72f-d3447b23ad42", + "label": "4", + "value": "RATING_4", + "position": 3 + }, + { + "id": "428f765e-4792-4cea-8270-9dba60f45fd9", + "label": "5", + "value": "RATING_5", + "position": 4 + } + ], + "relationDefinition": null, + "fromRelationMetadata": null, + "toRelationMetadata": null + } + }, { "__typename": "fieldEdge", "node": { @@ -5517,29 +5567,6 @@ export const mockedStandardObjectMetadataQueryResult: ObjectMetadataItemsQuery = "toRelationMetadata": null } }, - { - "__typename": "fieldEdge", - "node": { - "__typename": "field", - "id": "dd99cf8d-a10c-4d41-8469-6b5e03e5ae2e", - "type": "TEXT", - "name": "probability", - "label": "Probability", - "description": "Opportunity probability", - "icon": "IconProgressCheck", - "isCustom": false, - "isActive": true, - "isSystem": false, - "isNullable": false, - "createdAt": "2024-06-07T09:05:12.599Z", - "updatedAt": "2024-06-07T09:05:12.599Z", - "defaultValue": "'0'", - "options": null, - "relationDefinition": null, - "fromRelationMetadata": null, - "toRelationMetadata": null - } - }, { "__typename": "fieldEdge", "node": { diff --git a/packages/twenty-front/src/testing/mock-data/metadata.ts b/packages/twenty-front/src/testing/mock-data/metadata.ts index 94b5f910c24c..d6ec68da1d70 100644 --- a/packages/twenty-front/src/testing/mock-data/metadata.ts +++ b/packages/twenty-front/src/testing/mock-data/metadata.ts @@ -7,6 +7,7 @@ import { } from '~/generated-metadata/graphql'; import { mockedStandardObjectMetadataQueryResult } from '~/testing/mock-data/generated/standard-metadata-query-result'; +// TODO: replace with new mock const customObjectMetadataItemEdge: ObjectEdge = { __typename: 'objectEdge', node: { diff --git a/packages/twenty-front/src/testing/mock-data/people.ts b/packages/twenty-front/src/testing/mock-data/people.ts index e0aaa78486c3..89bff03b013e 100644 --- a/packages/twenty-front/src/testing/mock-data/people.ts +++ b/packages/twenty-front/src/testing/mock-data/people.ts @@ -1,3 +1,5 @@ +import { RecordGqlConnection } from '@/object-record/graphql/types/RecordGqlConnection'; + export const getPeopleMock = () => { const peopleMock = peopleQueryResult.people.edges.map((edge) => edge.node); @@ -22,7 +24,7 @@ export const mockedEmptyPersonData = { __typename: 'Person', }; -export const peopleQueryResult = { +export const peopleQueryResult: { people: RecordGqlConnection } = { people: { __typename: 'PersonConnection', totalCount: 15, diff --git a/packages/twenty-front/src/testing/mock-data/remote-servers.ts b/packages/twenty-front/src/testing/mock-data/remote-servers.ts index fee3c13cae5b..12948b099210 100644 --- a/packages/twenty-front/src/testing/mock-data/remote-servers.ts +++ b/packages/twenty-front/src/testing/mock-data/remote-servers.ts @@ -16,6 +16,7 @@ export const mockedRemoteServers = [ }, updatedAt: '2024-04-30T13:41:25.858Z', schema: 'public', + label: 'postgres DB', }, { __typename: 'RemoteServer', @@ -27,5 +28,6 @@ export const mockedRemoteServers = [ }, foreignDataWrapperType: 'stripe_fdw', updatedAt: '2024-04-30T13:41:25.858Z', + label: 'stripe DB', }, ]; diff --git a/packages/twenty-front/src/testing/mock-data/users.ts b/packages/twenty-front/src/testing/mock-data/users.ts index 92630930a647..d8067e6cca62 100644 --- a/packages/twenty-front/src/testing/mock-data/users.ts +++ b/packages/twenty-front/src/testing/mock-data/users.ts @@ -1,5 +1,11 @@ import { WorkspaceMember } from '@/workspace-member/types/WorkspaceMember'; -import { User, Workspace } from '~/generated/graphql'; +import { + OnboardingStatus, + SubscriptionInterval, + SubscriptionStatus, + User, + Workspace, +} from '~/generated/graphql'; type MockedUser = Pick< User, @@ -10,7 +16,7 @@ type MockedUser = Pick< | 'canImpersonate' | '__typename' | 'supportUserHash' - | 'onboardingStep' + | 'onboardingStatus' > & { workspaceMember: WorkspaceMember | null; locale: string; @@ -30,7 +36,6 @@ export const mockDefaultWorkspace: Workspace = { inviteHash: 'twenty.com-invite-hash', logo: workspaceLogoUrl, allowImpersonation: true, - subscriptionStatus: 'active', activationStatus: 'active', featureFlags: [ { @@ -58,9 +63,10 @@ export const mockDefaultWorkspace: Workspace = { currentBillingSubscription: { __typename: 'BillingSubscription', id: '7efbc3f7-6e5e-4128-957e-8d86808cdf6a', - interval: 'month', - status: 'active', + interval: SubscriptionInterval.Month, + status: SubscriptionStatus.Active, }, + workspaceMembersCount: 1, }; export const mockedWorkspaceMemberData: WorkspaceMember = { @@ -79,49 +85,26 @@ export const mockedWorkspaceMemberData: WorkspaceMember = { userEmail: 'charles@test.com', }; -export const mockedUsersData: Array<MockedUser> = [ - { - id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6d', - __typename: 'User', - email: 'charles@test.com', - firstName: 'Charles', - lastName: 'Test', - canImpersonate: false, - supportUserHash: - 'a95afad9ff6f0b364e2a3fd3e246a1a852c22b6e55a3ca33745a86c201f9c10d', - workspaceMember: mockedWorkspaceMemberData, - defaultWorkspace: mockDefaultWorkspace, - locale: 'en', - workspaces: [{ workspace: mockDefaultWorkspace }], - onboardingStep: null, - }, - { - id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6c', - __typename: 'User', - email: 'felix@test.com', - firstName: 'Felix', - lastName: 'Test', - canImpersonate: false, - supportUserHash: - '54ac3986035961724cdb9a7a30c70e6463a4b68f0ecd2014c727171a82144b74', - workspaceMember: { - ...mockedWorkspaceMemberData, - id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6c', - name: { - firstName: 'Felix', - lastName: 'Test', - }, - userId: '81aeb270-d689-4515-bd5d-35dbe956da3b', - }, - defaultWorkspace: mockDefaultWorkspace, - locale: 'en', - workspaces: [{ workspace: mockDefaultWorkspace }], - onboardingStep: null, - }, -]; +export const mockedUserData: MockedUser = { + id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6d', + __typename: 'User', + email: 'charles@test.com', + firstName: 'Charles', + lastName: 'Test', + canImpersonate: false, + supportUserHash: + 'a95afad9ff6f0b364e2a3fd3e246a1a852c22b6e55a3ca33745a86c201f9c10d', + workspaceMember: mockedWorkspaceMemberData, + defaultWorkspace: mockDefaultWorkspace, + locale: 'en', + workspaces: [{ workspace: mockDefaultWorkspace }], + onboardingStatus: OnboardingStatus.Completed, +}; -export const mockedOnboardingUsersData: Array<MockedUser> = [ - { +export const mockedOnboardingUserData = ( + onboardingStatus?: OnboardingStatus, +) => { + return { id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6d', __typename: 'User', email: 'workspace-onboarding@test.com', @@ -130,35 +113,10 @@ export const mockedOnboardingUsersData: Array<MockedUser> = [ canImpersonate: false, supportUserHash: '4fb61d34ed3a4aeda2476d4b308b5162db9e1809b2b8277e6fdc6efc4a609254', - workspaceMember: { - ...mockedWorkspaceMemberData, - id: 'd454f075-c72f-4ebe-bac7-d28e75e74a23', - name: { - firstName: '', - lastName: '', - }, - - userId: '7f793378-b939-43b7-8642-292c9510754c', - }, - defaultWorkspace: mockDefaultWorkspace, - locale: 'en', - workspaces: [{ workspace: mockDefaultWorkspace }], - onboardingStep: null, - }, - { - id: '7dfbc3f7-6e5e-4128-957e-8d86808cdf6d', - __typename: 'User', - email: 'profile-onboarding@test.com', - firstName: '', - lastName: '', - canImpersonate: false, workspaceMember: null, - defaultWorkspace: { - ...mockDefaultWorkspace, - activationStatus: 'inactive', - }, + defaultWorkspace: mockDefaultWorkspace, locale: 'en', workspaces: [{ workspace: mockDefaultWorkspace }], - onboardingStep: null, - }, -]; + onboardingStatus: onboardingStatus || null, + }; +}; diff --git a/packages/twenty-front/src/testing/mock-data/view-fields.ts b/packages/twenty-front/src/testing/mock-data/view-fields.ts index c644510c90f5..f8af8db65787 100644 --- a/packages/twenty-front/src/testing/mock-data/view-fields.ts +++ b/packages/twenty-front/src/testing/mock-data/view-fields.ts @@ -9,6 +9,9 @@ export const mockedViewFieldsData = [ position: 0, isVisible: true, size: 180, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '2a96bbc8-d86d-439a-8e50-4b07ebd27750', @@ -17,6 +20,9 @@ export const mockedViewFieldsData = [ position: 1, isVisible: true, size: 100, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '0c1b4c7b-6a3d-4fb0-bf2b-5d7c8fb844ed', @@ -25,6 +31,9 @@ export const mockedViewFieldsData = [ position: 2, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: 'cc7f9560-32b5-4b82-8fd9-b05fe77c8cf7', @@ -33,6 +42,9 @@ export const mockedViewFieldsData = [ position: 3, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '3de4d078-3396-4480-be2d-6f3b1a228b0d', @@ -41,6 +53,9 @@ export const mockedViewFieldsData = [ position: 4, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '4650c8fb-0f1e-4342-88dc-adedae1445f9', @@ -49,6 +64,9 @@ export const mockedViewFieldsData = [ position: 5, isVisible: true, size: 170, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '727430bf-6ff8-4c85-9828-cbe72ac0fc27', @@ -57,6 +75,9 @@ export const mockedViewFieldsData = [ position: 6, isVisible: true, size: 170, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, // People @@ -67,6 +88,9 @@ export const mockedViewFieldsData = [ position: 0, isVisible: true, size: 210, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: 'e1e24864-8601-4cd8-8a63-09c1285f2e39', @@ -75,6 +99,9 @@ export const mockedViewFieldsData = [ position: 1, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '5a1df716-7211-445a-9f16-9783a00998a7', @@ -83,6 +110,9 @@ export const mockedViewFieldsData = [ position: 2, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: 'a6e1197a-7e84-4d92-ace2-367c0bc46c49', @@ -91,6 +121,9 @@ export const mockedViewFieldsData = [ position: 3, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: 'c9343097-d14b-4559-a5fa-626c1527d39f', @@ -99,6 +132,9 @@ export const mockedViewFieldsData = [ position: 4, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: 'a873e5f0-fed6-47e9-a712-6854eab3ec77', @@ -107,6 +143,9 @@ export const mockedViewFieldsData = [ position: 5, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '66f134b8-5329-422f-b88e-83e6bb707eb5', @@ -115,6 +154,9 @@ export const mockedViewFieldsData = [ position: 6, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '648faa24-cabb-482a-8578-ba3f09906017', @@ -123,6 +165,9 @@ export const mockedViewFieldsData = [ position: 7, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '3a9e7f0d-a4ce-4ad5-aac7-3a24eb1a412d', @@ -131,6 +176,9 @@ export const mockedViewFieldsData = [ position: 8, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, // Opportunities @@ -141,14 +189,9 @@ export const mockedViewFieldsData = [ position: 0, isVisible: true, size: 180, - }, - { - id: 'e5a731bb-82b9-4abe-ad22-1ddea94722f9', - fieldMetadataId: 'probability', - viewId: mockedViewsData[2].id, - position: 1, - isVisible: true, - size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '3159acd8-463f-458d-bf9a-af8ac6f57dc0', @@ -157,6 +200,9 @@ export const mockedViewFieldsData = [ position: 2, isVisible: true, size: 100, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: 'afc0819d-b694-4e3c-a2e6-25261aa3ed2c', @@ -165,6 +211,9 @@ export const mockedViewFieldsData = [ position: 3, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: 'ec0507bb-aedc-4695-ba96-d81bdeb9db83', @@ -173,6 +222,9 @@ export const mockedViewFieldsData = [ position: 4, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, { id: '3f1585f6-44f6-45c5-b840-bc05af5d0008', @@ -181,5 +233,8 @@ export const mockedViewFieldsData = [ position: 5, isVisible: true, size: 150, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + __typename: 'ViewField', }, ]; diff --git a/packages/twenty-front/src/testing/mock-data/views.ts b/packages/twenty-front/src/testing/mock-data/views.ts index c58e50846ec1..555e0a7c5d28 100644 --- a/packages/twenty-front/src/testing/mock-data/views.ts +++ b/packages/twenty-front/src/testing/mock-data/views.ts @@ -2,25 +2,58 @@ export const mockedViewsData = [ { id: '37a8a866-eb17-4e76-9382-03143a2f6a80', name: 'All companies', - objectMetadataId: 'company', + objectMetadataId: 'f9fd99a8-108f-4066-9675-cde753cf5de9', type: 'table', + icon: 'IconSkyline', + key: 'INDEX', + kanbanFieldMetadataId: null, + position: 0, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + isCompact: false, + + __typename: 'View', }, { id: '6095799e-b48f-4e00-b071-10818083593a', name: 'All people', objectMetadataId: 'person', type: 'table', + icon: 'IconPerson', + key: 'INDEX', + kanbanFieldMetadataId: null, + position: 0, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + isCompact: false, + __typename: 'View', }, { id: 'e26f66b7-f890-4a5c-b4d2-ec09987b5308', name: 'All opportunities', objectMetadataId: 'company', type: 'kanban', + icon: 'IconOpportunity', + key: 'INDEX', + kanbanFieldMetadataId: null, + position: 0, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + isCompact: false, + __typename: 'View', }, { id: '5c307222-1dd5-4ff3-ab06-8d990e9b3c74', name: 'All companies (v2)', - objectMetadataId: 'a3195559-cc20-4749-9565-572a2f506581', + objectMetadataId: 'f9fd99a8-108f-4066-9675-cde753cf5de9', type: 'table', + icon: 'IconSkyline', + key: 'INDEX', + kanbanFieldMetadataId: null, + position: 0, + createdAt: '2021-09-01T00:00:00.000Z', + updatedAt: '2021-09-01T00:00:00.000Z', + isCompact: false, + __typename: 'View', }, ]; diff --git a/packages/twenty-front/src/types/WithNarrowedStringLiteralProperty.ts b/packages/twenty-front/src/types/WithNarrowedStringLiteralProperty.ts new file mode 100644 index 000000000000..7e0ff52b4062 --- /dev/null +++ b/packages/twenty-front/src/types/WithNarrowedStringLiteralProperty.ts @@ -0,0 +1,7 @@ +export type WithNarrowedStringLiteralProperty< + T, + K extends keyof T, + Sub extends T[K], +> = Omit<T, K> & { + [P in K]: Extract<T[K], Sub>; +}; diff --git a/packages/twenty-front/src/utils/array/generateILikeFiltersForCompositeFields.ts b/packages/twenty-front/src/utils/array/generateILikeFiltersForCompositeFields.ts index 4d92976934b1..a5a2883e74e7 100644 --- a/packages/twenty-front/src/utils/array/generateILikeFiltersForCompositeFields.ts +++ b/packages/twenty-front/src/utils/array/generateILikeFiltersForCompositeFields.ts @@ -4,7 +4,31 @@ export const generateILikeFiltersForCompositeFields = ( filterString: string, baseFieldName: string, subFields: string[], + emptyCheck = false, ) => { + if (emptyCheck) { + return subFields.map((subField) => { + return { + or: [ + { + [baseFieldName]: { + [subField]: { + is: 'NULL', + }, + }, + }, + { + [baseFieldName]: { + [subField]: { + ilike: '', + }, + }, + }, + ], + }; + }); + } + return filterString .split(' ') .reduce((previousValue: RecordGqlOperationFilter[], currentValue) => { diff --git a/packages/twenty-front/src/utils/createApolloStoreFieldName.ts b/packages/twenty-front/src/utils/createApolloStoreFieldName.ts new file mode 100644 index 000000000000..7959a7f2a938 --- /dev/null +++ b/packages/twenty-front/src/utils/createApolloStoreFieldName.ts @@ -0,0 +1,9 @@ +export const createApolloStoreFieldName = ({ + fieldName, + fieldVariables, +}: { + fieldName: string; + fieldVariables: Record<string, any>; +}) => { + return `${fieldName}(${JSON.stringify(fieldVariables)})`; +}; diff --git a/packages/twenty-front/src/utils/createEventContext.ts b/packages/twenty-front/src/utils/createEventContext.ts new file mode 100644 index 000000000000..c69425003def --- /dev/null +++ b/packages/twenty-front/src/utils/createEventContext.ts @@ -0,0 +1,12 @@ +import { Context, createContext } from 'react'; + +type ObjectOfFunctions = { + [key: string]: (...args: any[]) => void; +}; + +export type EventContext<T extends ObjectOfFunctions> = + T extends ObjectOfFunctions ? T : never; + +export const createEventContext = <T extends ObjectOfFunctions>(): Context< + EventContext<T> +> => createContext<EventContext<T>>({} as EventContext<T>); diff --git a/packages/twenty-front/src/utils/format/__tests__/formatDate.test.ts b/packages/twenty-front/src/utils/format/__tests__/formatDate.test.ts index a80c11d3e7f5..60431e17b267 100644 --- a/packages/twenty-front/src/utils/format/__tests__/formatDate.test.ts +++ b/packages/twenty-front/src/utils/format/__tests__/formatDate.test.ts @@ -24,6 +24,7 @@ describe('formatToHumanReadableTime', () => { it('should format the date to a human-readable time', () => { const date = new Date('2022-01-01T12:30:00'); const result = formatToHumanReadableTime(date); - expect(result).toBe('12:30 PM'); + // it seems when running locally on MacOS the space is not the same + expect(['12:30 PM', '12:30 PM']).toContain(result); }); }); diff --git a/packages/twenty-front/src/utils/image/__tests__/getImageAbsoluteURIOrBase64.test.ts b/packages/twenty-front/src/utils/image/__tests__/getImageAbsoluteURIOrBase64.test.ts new file mode 100644 index 000000000000..226aa6a0e598 --- /dev/null +++ b/packages/twenty-front/src/utils/image/__tests__/getImageAbsoluteURIOrBase64.test.ts @@ -0,0 +1,27 @@ +import { getImageAbsoluteURIOrBase64 } from '../getImageAbsoluteURIOrBase64'; + +describe('getImageAbsoluteURIOrBase64', () => { + it('should return null if imageUrl is null', () => { + const imageUrl = null; + const result = getImageAbsoluteURIOrBase64(imageUrl); + expect(result).toBeNull(); + }); + + it('should return base64 encoded string if prefixed with data', () => { + const imageUrl = 'data:XXX'; + const result = getImageAbsoluteURIOrBase64(imageUrl); + expect(result).toBe(imageUrl); + }); + + it('should return absolute url if the imageUrl is an absolute url', () => { + const imageUrl = 'https://XXX'; + const result = getImageAbsoluteURIOrBase64(imageUrl); + expect(result).toBe(imageUrl); + }); + + it('should return fully formed url if imageUrl is a relative url', () => { + const imageUrl = 'XXX'; + const result = getImageAbsoluteURIOrBase64(imageUrl); + expect(result).toBe('http://localhost:3000/files/XXX'); + }); +}); diff --git a/packages/twenty-front/src/utils/isDeeplyEqual.ts b/packages/twenty-front/src/utils/isDeeplyEqual.ts index 1a887595abc0..62f10026352b 100644 --- a/packages/twenty-front/src/utils/isDeeplyEqual.ts +++ b/packages/twenty-front/src/utils/isDeeplyEqual.ts @@ -1,3 +1,4 @@ import deepEqual from 'deep-equal'; -export const isDeeplyEqual = <T>(a: T, b: T) => deepEqual(a, b); +export const isDeeplyEqual = <T>(a: T, b: T, options?: { strict: boolean }) => + deepEqual(a, b, options); diff --git a/packages/twenty-front/src/utils/recoil-effects.ts b/packages/twenty-front/src/utils/recoil-effects.ts index d7c0261b955f..7f20e24c5062 100644 --- a/packages/twenty-front/src/utils/recoil-effects.ts +++ b/packages/twenty-front/src/utils/recoil-effects.ts @@ -5,17 +5,17 @@ import { cookieStorage } from '~/utils/cookie-storage'; import { isDefined } from './isDefined'; export const localStorageEffect = - <T>(key: string): AtomEffect<T> => - ({ setSelf, onSet }) => { - const savedValue = localStorage.getItem(key); + <T>(key?: string): AtomEffect<T> => + ({ setSelf, onSet, node }) => { + const savedValue = localStorage.getItem(key ?? node.key); if (savedValue != null) { setSelf(JSON.parse(savedValue)); } onSet((newValue, _, isReset) => { isReset - ? localStorage.removeItem(key) - : localStorage.setItem(key, JSON.stringify(newValue)); + ? localStorage.removeItem(key ?? node.key) + : localStorage.setItem(key ?? node.key, JSON.stringify(newValue)); }); }; diff --git a/packages/twenty-front/src/utils/validation-schemas/__tests__/absoluteUrlSchema.test.ts b/packages/twenty-front/src/utils/validation-schemas/__tests__/absoluteUrlSchema.test.ts index d20f3bcf6393..ffb83cfc8ef5 100644 --- a/packages/twenty-front/src/utils/validation-schemas/__tests__/absoluteUrlSchema.test.ts +++ b/packages/twenty-front/src/utils/validation-schemas/__tests__/absoluteUrlSchema.test.ts @@ -28,7 +28,6 @@ describe('absoluteUrlSchema', () => { it('fails for invalid urls', () => { expect(absoluteUrlSchema.safeParse('?o').success).toBe(false); - expect(absoluteUrlSchema.safeParse('').success).toBe(false); expect(absoluteUrlSchema.safeParse('\\').success).toBe(false); }); }); diff --git a/packages/twenty-front/src/utils/validation-schemas/absoluteUrlSchema.ts b/packages/twenty-front/src/utils/validation-schemas/absoluteUrlSchema.ts index e0c376c7403f..a0deb4bfab25 100644 --- a/packages/twenty-front/src/utils/validation-schemas/absoluteUrlSchema.ts +++ b/packages/twenty-front/src/utils/validation-schemas/absoluteUrlSchema.ts @@ -8,4 +8,5 @@ export const absoluteUrlSchema = z .string() .transform((value) => `https://${value}`) .pipe(z.string().url()), - ); + ) + .or(z.literal('')); diff --git a/packages/twenty-front/vite.config.ts b/packages/twenty-front/vite.config.ts index 34055bdb14ba..705c2dcc51f5 100644 --- a/packages/twenty-front/vite.config.ts +++ b/packages/twenty-front/vite.config.ts @@ -63,6 +63,15 @@ export default defineConfig(({ command, mode }) => { '**/Chip.tsx', '**/Tag.tsx', '**/MultiSelectFieldDisplay.tsx', + '**/RatingInput.tsx', + '**/RecordTableCellContainer.tsx', + '**/RecordTableCellDisplayContainer.tsx', + '**/Avatar.tsx', + '**/RecordTableBodyDroppable.tsx', + '**/RecordTableCellBaseContainer.tsx', + '**/RecordTableCellTd.tsx', + '**/RecordTableTd.tsx', + '**/RecordTableHeaderDragDropColumn.tsx', ], babelOptions: { presets: ['@babel/preset-typescript', '@babel/preset-react'], diff --git a/packages/twenty-postgres/linux/Dockerfile b/packages/twenty-postgres/linux/Dockerfile index c8996796c3c7..871f87dff40c 100644 --- a/packages/twenty-postgres/linux/Dockerfile +++ b/packages/twenty-postgres/linux/Dockerfile @@ -3,7 +3,7 @@ ARG IMAGE_TAG='15.5.0-debian-11-r15' FROM bitnami/postgresql:${IMAGE_TAG} ARG PG_MAIN_VERSION=15 -ARG PG_GRAPHQL_VERSION=1.5.1 +ARG PG_GRAPHQL_VERSION=1.5.6 ARG WRAPPERS_VERSION=0.2.0 ARG TARGETARCH diff --git a/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql b/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql new file mode 100644 index 000000000000..b5489e7cd65c --- /dev/null +++ b/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql @@ -0,0 +1,116 @@ +/* +This file is auto generated by pgrx. + +The ordering of items is not stable, it is driven by a dependency graph. +*/ + +-- src/lib.rs:27 +-- pg_graphql::_internal_resolve +CREATE FUNCTION graphql."_internal_resolve"( + "query" TEXT, /* &str */ + "variables" jsonb DEFAULT '{}', /* core::option::Option<pgrx::datum::json::JsonB> */ + "operationName" TEXT DEFAULT null, /* core::option::Option<alloc::string::String> */ + "extensions" jsonb DEFAULT null /* core::option::Option<pgrx::datum::json::JsonB> */ +) RETURNS jsonb /* pgrx::datum::json::JsonB */ + +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'resolve_wrapper'; + +-- src/lib.rs:22 +create or replace function graphql.exception(message text) + returns text + language plpgsql +as $$ +begin + raise exception using errcode='22000', message=message; +end; +$$; + + +-- src/lib.rs:23 +-- requires: +-- resolve + +create or replace function graphql.resolve( + "query" text, + "variables" jsonb default '{}', + "operationName" text default null, + "extensions" jsonb default null +) + returns jsonb + language plpgsql +as $$ +declare + res jsonb; + message_text text; +begin + begin + select graphql._internal_resolve("query" := "query", + "variables" := "variables", + "operationName" := "operationName", + "extensions" := "extensions") into res; + return res; + exception + when others then + get stacked diagnostics message_text = message_text; + return + jsonb_build_object('data', null, + 'errors', jsonb_build_array(jsonb_build_object('message', message_text))); + end; +end; +$$; + + +-- src/lib.rs:21 +create function graphql.comment_directive(comment_ text) + returns jsonb + language sql + immutable +as $$ + /* + comment on column public.account.name is '@graphql.name: myField' + */ + select + coalesce( + ( + regexp_match( + comment_, + '@graphql\((.+?)\)' + ) + )[1]::jsonb, + jsonb_build_object() + ) +$$; + + +-- src/lib.rs:20 +-- Is updated every time the schema changes +create sequence if not exists graphql.seq_schema_version as int cycle; + +create or replace function graphql.increment_schema_version() + returns event_trigger + security definer + language plpgsql +as $$ +begin + perform nextval('graphql.seq_schema_version'); +end; +$$; + +create or replace function graphql.get_schema_version() + returns int + security definer + language sql +as $$ + select last_value from graphql.seq_schema_version; +$$; + +-- On DDL event, increment the schema version number +create event trigger graphql_watch_ddl + on ddl_command_end + execute procedure graphql.increment_schema_version(); + +create event trigger graphql_watch_drop + on sql_drop + execute procedure graphql.increment_schema_version(); + diff --git a/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql.control b/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql.control new file mode 100644 index 000000000000..56c20ea629f1 --- /dev/null +++ b/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql.control @@ -0,0 +1,6 @@ +comment = 'pg_graphql: GraphQL support' +default_version = '1.5.6' +module_pathname = '$libdir/pg_graphql' +relocatable = false +superuser = true +schema = 'graphql' diff --git a/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql.so b/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql.so new file mode 100755 index 000000000000..9d1667369cb1 Binary files /dev/null and b/packages/twenty-postgres/linux/amd64/15/pg_graphql/1.5.6/pg_graphql.so differ diff --git a/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql b/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql new file mode 100644 index 000000000000..35a393c223bc --- /dev/null +++ b/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql @@ -0,0 +1,116 @@ +/* +This file is auto generated by pgrx. + +The ordering of items is not stable, it is driven by a dependency graph. +*/ + +-- src/lib.rs:27 +-- pg_graphql::_internal_resolve +CREATE FUNCTION graphql."_internal_resolve"( + "query" TEXT, /* &str */ + "variables" jsonb DEFAULT '{}', /* core::option::Option<pgrx::datum::json::JsonB> */ + "operationName" TEXT DEFAULT null, /* core::option::Option<alloc::string::String> */ + "extensions" jsonb DEFAULT null /* core::option::Option<pgrx::datum::json::JsonB> */ +) RETURNS jsonb /* pgrx::datum::json::JsonB */ + +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'resolve_wrapper'; + +-- src/lib.rs:21 +create function graphql.comment_directive(comment_ text) + returns jsonb + language sql + immutable +as $$ + /* + comment on column public.account.name is '@graphql.name: myField' + */ + select + coalesce( + ( + regexp_match( + comment_, + '@graphql\((.+?)\)' + ) + )[1]::jsonb, + jsonb_build_object() + ) +$$; + + +-- src/lib.rs:22 +create or replace function graphql.exception(message text) + returns text + language plpgsql +as $$ +begin + raise exception using errcode='22000', message=message; +end; +$$; + + +-- src/lib.rs:20 +-- Is updated every time the schema changes +create sequence if not exists graphql.seq_schema_version as int cycle; + +create or replace function graphql.increment_schema_version() + returns event_trigger + security definer + language plpgsql +as $$ +begin + perform nextval('graphql.seq_schema_version'); +end; +$$; + +create or replace function graphql.get_schema_version() + returns int + security definer + language sql +as $$ + select last_value from graphql.seq_schema_version; +$$; + +-- On DDL event, increment the schema version number +create event trigger graphql_watch_ddl + on ddl_command_end + execute procedure graphql.increment_schema_version(); + +create event trigger graphql_watch_drop + on sql_drop + execute procedure graphql.increment_schema_version(); + + +-- src/lib.rs:23 +-- requires: +-- resolve + +create or replace function graphql.resolve( + "query" text, + "variables" jsonb default '{}', + "operationName" text default null, + "extensions" jsonb default null +) + returns jsonb + language plpgsql +as $$ +declare + res jsonb; + message_text text; +begin + begin + select graphql._internal_resolve("query" := "query", + "variables" := "variables", + "operationName" := "operationName", + "extensions" := "extensions") into res; + return res; + exception + when others then + get stacked diagnostics message_text = message_text; + return + jsonb_build_object('data', null, + 'errors', jsonb_build_array(jsonb_build_object('message', message_text))); + end; +end; +$$; + diff --git a/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql.control b/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql.control new file mode 100644 index 000000000000..56c20ea629f1 --- /dev/null +++ b/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql.control @@ -0,0 +1,6 @@ +comment = 'pg_graphql: GraphQL support' +default_version = '1.5.6' +module_pathname = '$libdir/pg_graphql' +relocatable = false +superuser = true +schema = 'graphql' diff --git a/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql.so b/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql.so new file mode 100755 index 000000000000..462b794534c9 Binary files /dev/null and b/packages/twenty-postgres/linux/arm64/15/pg_graphql/1.5.6/pg_graphql.so differ diff --git a/packages/twenty-postgres/linux/build-postgres-linux.sh b/packages/twenty-postgres/linux/build-postgres-linux.sh index 11744e774d7f..4b6e3b21831e 100755 --- a/packages/twenty-postgres/linux/build-postgres-linux.sh +++ b/packages/twenty-postgres/linux/build-postgres-linux.sh @@ -48,7 +48,7 @@ EOF echo_header $BLUE " DATABASE SETUP" PG_MAIN_VERSION=15 -PG_GRAPHQL_VERSION=1.5.1 +PG_GRAPHQL_VERSION=1.5.6 CARGO_PGRX_VERSION=0.11.2 TARGETARCH=$(dpkg --print-architecture) diff --git a/packages/twenty-postgres/linux/build_postgres.md b/packages/twenty-postgres/linux/build_postgres.md index 12c931125edc..c966b1ba5350 100644 --- a/packages/twenty-postgres/linux/build_postgres.md +++ b/packages/twenty-postgres/linux/build_postgres.md @@ -2,21 +2,23 @@ This doc explains how to build postgresql for Funnelmink Build .control, .so and .pg_graphql--version.sql -> docker buildx create --name mybuilder -> docker buildx use mybuilder +``` +docker buildx create --name mybuilder +docker buildx use mybuilder +``` Do the same for <PLATFORM> in ['amd64', 'arm64'] ('amd64' builds faster) -> cd packages/twenty-postgres -> docker buildx build --platform linux/<PLATFORM> --load -t twenty-bitnami-postgres-<PLATFORM> linux -> docker run --name twenty-bitnami-<PLATFORM> -v ~/Desktop/twenty/packages/twenty-postgres:/twenty <IMAGE_TAG> +``` +cd packages/twenty-postgres +docker buildx build --platform linux/<PLATFORM> --load -t twenty-bitnami-postgres-<PLATFORM> linux +docker run --name twenty-bitnami-<PLATFORM> -v ~/Desktop/twenty/packages/twenty-postgres:/twenty <IMAGE_TAG> +``` In another terminal -> docker exec -it <CONTAINER_TAG> sh -> sh twenty/linux/build-postgres-linux.sh -> cp opt/bitnami/postgresql/lib/pg_graphql.so twenty/linux/<PLATFORM>/15/pg_graphql/<PG_GRAPHQL_VERSION> -> cp opt/bitnami/postgresql/share/extension/pg_graphql.control twenty/linux/<PLATFORM>/15/pg_graphql/<PG_GRAPHQL_VERSION> -> cp opt/bitnami/postgresql/share/extension/pg_graphql--<PG_GRAPHQL_VERSION>.sql twenty/linux/<PLATFORM>/15/pg_graphql/<PG_GRAPHQL_VERSION> - -Then -> prod-server-build -> prod-server-run +``` +docker exec -it <CONTAINER_TAG> sh +sh twenty/linux/build-postgres-linux.sh +cp opt/bitnami/postgresql/lib/pg_graphql.so twenty/linux/<PLATFORM>/15/pg_graphql/<PG_GRAPHQL_VERSION> +cp opt/bitnami/postgresql/share/extension/pg_graphql.control twenty/linux/<PLATFORM>/15/pg_graphql/<PG_GRAPHQL_VERSION> +cp opt/bitnami/postgresql/share/extension/pg_graphql--<PG_GRAPHQL_VERSION>.sql twenty/linux/<PLATFORM>/15/pg_graphql/<PG_GRAPHQL_VERSION> +``` diff --git a/packages/twenty-postgres/linux/provision-postgres-linux.sh b/packages/twenty-postgres/linux/provision-postgres-linux.sh index 651224284e05..620d3e887fee 100755 --- a/packages/twenty-postgres/linux/provision-postgres-linux.sh +++ b/packages/twenty-postgres/linux/provision-postgres-linux.sh @@ -48,7 +48,7 @@ EOF echo_header $BLUE " DATABASE SETUP" PG_MAIN_VERSION=15 -PG_GRAPHQL_VERSION=1.5.1 +PG_GRAPHQL_VERSION=1.5.6 TARGETARCH=$(dpkg --print-architecture) # Install PostgresSQL diff --git a/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql b/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql new file mode 100644 index 000000000000..f7687883c020 --- /dev/null +++ b/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql @@ -0,0 +1,116 @@ +/* +This file is auto generated by pgrx. + +The ordering of items is not stable, it is driven by a dependency graph. +*/ + +-- src/lib.rs:27 +-- pg_graphql::_internal_resolve +CREATE FUNCTION graphql."_internal_resolve"( + "query" TEXT, /* &str */ + "variables" jsonb DEFAULT '{}', /* core::option::Option<pgrx::datum::json::JsonB> */ + "operationName" TEXT DEFAULT null, /* core::option::Option<alloc::string::String> */ + "extensions" jsonb DEFAULT null /* core::option::Option<pgrx::datum::json::JsonB> */ +) RETURNS jsonb /* pgrx::datum::json::JsonB */ + +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'resolve_wrapper'; + +-- src/lib.rs:22 +create or replace function graphql.exception(message text) + returns text + language plpgsql +as $$ +begin + raise exception using errcode='22000', message=message; +end; +$$; + + +-- src/lib.rs:20 +-- Is updated every time the schema changes +create sequence if not exists graphql.seq_schema_version as int cycle; + +create or replace function graphql.increment_schema_version() + returns event_trigger + security definer + language plpgsql +as $$ +begin + perform nextval('graphql.seq_schema_version'); +end; +$$; + +create or replace function graphql.get_schema_version() + returns int + security definer + language sql +as $$ + select last_value from graphql.seq_schema_version; +$$; + +-- On DDL event, increment the schema version number +create event trigger graphql_watch_ddl + on ddl_command_end + execute procedure graphql.increment_schema_version(); + +create event trigger graphql_watch_drop + on sql_drop + execute procedure graphql.increment_schema_version(); + + +-- src/lib.rs:21 +create function graphql.comment_directive(comment_ text) + returns jsonb + language sql + immutable +as $$ + /* + comment on column public.account.name is '@graphql.name: myField' + */ + select + coalesce( + ( + regexp_match( + comment_, + '@graphql\((.+?)\)' + ) + )[1]::jsonb, + jsonb_build_object() + ) +$$; + + +-- src/lib.rs:23 +-- requires: +-- resolve + +create or replace function graphql.resolve( + "query" text, + "variables" jsonb default '{}', + "operationName" text default null, + "extensions" jsonb default null +) + returns jsonb + language plpgsql +as $$ +declare + res jsonb; + message_text text; +begin + begin + select graphql._internal_resolve("query" := "query", + "variables" := "variables", + "operationName" := "operationName", + "extensions" := "extensions") into res; + return res; + exception + when others then + get stacked diagnostics message_text = message_text; + return + jsonb_build_object('data', null, + 'errors', jsonb_build_array(jsonb_build_object('message', message_text))); + end; +end; +$$; + diff --git a/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql.control b/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql.control new file mode 100644 index 000000000000..56c20ea629f1 --- /dev/null +++ b/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql.control @@ -0,0 +1,6 @@ +comment = 'pg_graphql: GraphQL support' +default_version = '1.5.6' +module_pathname = '$libdir/pg_graphql' +relocatable = false +superuser = true +schema = 'graphql' diff --git a/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql.so b/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql.so new file mode 100755 index 000000000000..c87cdbd4c66f Binary files /dev/null and b/packages/twenty-postgres/macos/arm/15/pg_graphql/1.5.6/pg_graphql.so differ diff --git a/packages/twenty-postgres/macos/arm/build-postgres-macos-arm.sh b/packages/twenty-postgres/macos/arm/build-postgres-macos-arm.sh index a2738797f821..3decabfd7c5c 100755 --- a/packages/twenty-postgres/macos/arm/build-postgres-macos-arm.sh +++ b/packages/twenty-postgres/macos/arm/build-postgres-macos-arm.sh @@ -48,7 +48,7 @@ EOF echo_header $BLUE " DATABASE SETUP" PG_MAIN_VERSION=15 -PG_GRAPHQL_VERSION=1.5.1 +PG_GRAPHQL_VERSION=1.5.6 CARGO_PGRX_VERSION=0.11.2 current_directory=$(pwd) diff --git a/packages/twenty-postgres/macos/arm/provision-postgres-macos-arm.sh b/packages/twenty-postgres/macos/arm/provision-postgres-macos-arm.sh index 24e98760867b..201a4abb2ec0 100755 --- a/packages/twenty-postgres/macos/arm/provision-postgres-macos-arm.sh +++ b/packages/twenty-postgres/macos/arm/provision-postgres-macos-arm.sh @@ -1,7 +1,7 @@ #!/bin/bash PG_MAIN_VERSION=15 -PG_GRAPHQL_VERSION=1.5.1 +PG_GRAPHQL_VERSION=1.5.6 current_directory=$(pwd) diff --git a/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql b/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql new file mode 100644 index 000000000000..a24d69a1aef6 --- /dev/null +++ b/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql--1.5.6.sql @@ -0,0 +1,116 @@ +/* +This file is auto generated by pgrx. + +The ordering of items is not stable, it is driven by a dependency graph. +*/ + +-- src/lib.rs:27 +-- pg_graphql::_internal_resolve +CREATE FUNCTION graphql."_internal_resolve"( + "query" TEXT, /* &str */ + "variables" jsonb DEFAULT '{}', /* core::option::Option<pgrx::datum::json::JsonB> */ + "operationName" TEXT DEFAULT null, /* core::option::Option<alloc::string::String> */ + "extensions" jsonb DEFAULT null /* core::option::Option<pgrx::datum::json::JsonB> */ +) RETURNS jsonb /* pgrx::datum::json::JsonB */ + +LANGUAGE c /* Rust */ +AS 'MODULE_PATHNAME', 'resolve_wrapper'; + +-- src/lib.rs:20 +-- Is updated every time the schema changes +create sequence if not exists graphql.seq_schema_version as int cycle; + +create or replace function graphql.increment_schema_version() + returns event_trigger + security definer + language plpgsql +as $$ +begin + perform nextval('graphql.seq_schema_version'); +end; +$$; + +create or replace function graphql.get_schema_version() + returns int + security definer + language sql +as $$ + select last_value from graphql.seq_schema_version; +$$; + +-- On DDL event, increment the schema version number +create event trigger graphql_watch_ddl + on ddl_command_end + execute procedure graphql.increment_schema_version(); + +create event trigger graphql_watch_drop + on sql_drop + execute procedure graphql.increment_schema_version(); + + +-- src/lib.rs:22 +create or replace function graphql.exception(message text) + returns text + language plpgsql +as $$ +begin + raise exception using errcode='22000', message=message; +end; +$$; + + +-- src/lib.rs:21 +create function graphql.comment_directive(comment_ text) + returns jsonb + language sql + immutable +as $$ + /* + comment on column public.account.name is '@graphql.name: myField' + */ + select + coalesce( + ( + regexp_match( + comment_, + '@graphql\((.+?)\)' + ) + )[1]::jsonb, + jsonb_build_object() + ) +$$; + + +-- src/lib.rs:23 +-- requires: +-- resolve + +create or replace function graphql.resolve( + "query" text, + "variables" jsonb default '{}', + "operationName" text default null, + "extensions" jsonb default null +) + returns jsonb + language plpgsql +as $$ +declare + res jsonb; + message_text text; +begin + begin + select graphql._internal_resolve("query" := "query", + "variables" := "variables", + "operationName" := "operationName", + "extensions" := "extensions") into res; + return res; + exception + when others then + get stacked diagnostics message_text = message_text; + return + jsonb_build_object('data', null, + 'errors', jsonb_build_array(jsonb_build_object('message', message_text))); + end; +end; +$$; + diff --git a/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql.control b/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql.control new file mode 100644 index 000000000000..56c20ea629f1 --- /dev/null +++ b/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql.control @@ -0,0 +1,6 @@ +comment = 'pg_graphql: GraphQL support' +default_version = '1.5.6' +module_pathname = '$libdir/pg_graphql' +relocatable = false +superuser = true +schema = 'graphql' diff --git a/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql.so b/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql.so new file mode 100644 index 000000000000..c6f712ec040d Binary files /dev/null and b/packages/twenty-postgres/macos/intel/15/pg_graphql/1.5.6/pg_graphql.so differ diff --git a/packages/twenty-postgres/macos/intel/build-postgres-macos-intel.sh b/packages/twenty-postgres/macos/intel/build-postgres-macos-intel.sh index a5636eac08d2..944f64d72326 100755 --- a/packages/twenty-postgres/macos/intel/build-postgres-macos-intel.sh +++ b/packages/twenty-postgres/macos/intel/build-postgres-macos-intel.sh @@ -48,7 +48,7 @@ EOF echo_header $BLUE " DATABASE SETUP" PG_MAIN_VERSION=15 -PG_GRAPHQL_VERSION=1.5.1 +PG_GRAPHQL_VERSION=1.5.6 CARGO_PGRX_VERSION=0.11.2 current_directory=$(pwd) diff --git a/packages/twenty-postgres/macos/intel/provision-postgres-macos-intel.sh b/packages/twenty-postgres/macos/intel/provision-postgres-macos-intel.sh index c6a49fa430e7..72091af18798 100755 --- a/packages/twenty-postgres/macos/intel/provision-postgres-macos-intel.sh +++ b/packages/twenty-postgres/macos/intel/provision-postgres-macos-intel.sh @@ -1,7 +1,7 @@ #!/bin/bash PG_MAIN_VERSION=15 -PG_GRAPHQL_VERSION=1.5.1 +PG_GRAPHQL_VERSION=1.5.6 current_directory=$(pwd) diff --git a/packages/twenty-server/.env.example b/packages/twenty-server/.env.example index bfe76f8a49da..8b2c6cac2f93 100644 --- a/packages/twenty-server/.env.example +++ b/packages/twenty-server/.env.example @@ -70,6 +70,6 @@ SIGN_IN_PREFILLED=true # CAPTCHA_SECRET_KEY= # API_RATE_LIMITING_TTL= # API_RATE_LIMITING_LIMIT= -# MUTATION_MAXIMUM_RECORD_AFFECTED=100 +# MUTATION_MAXIMUM_AFFECTED_RECORDS=100 # CHROME_EXTENSION_ID=bggmipldbceihilonnbpgoeclgbkblkp -# PG_SSL_ALLOW_SELF_SIGNED=true +# PG_SSL_ALLOW_SELF_SIGNED=true \ No newline at end of file diff --git a/packages/twenty-server/.env.test b/packages/twenty-server/.env.test index 20bd13ac8ba3..fc51ab9e1e22 100644 --- a/packages/twenty-server/.env.test +++ b/packages/twenty-server/.env.test @@ -23,4 +23,4 @@ FILE_TOKEN_SECRET=secret_file_token # MESSAGING_PROVIDER_GMAIL_ENABLED=false # STORAGE_TYPE=local # STORAGE_LOCAL_PATH=.local-storage -# MUTATION_MAXIMUM_RECORD_AFFECTED=100 +# MUTATION_MAXIMUM_AFFECTED_RECORDS=100 diff --git a/packages/twenty-server/@types/express.d.ts b/packages/twenty-server/@types/express.d.ts index 9f5ffd96da42..09c6bfbfef80 100644 --- a/packages/twenty-server/@types/express.d.ts +++ b/packages/twenty-server/@types/express.d.ts @@ -5,6 +5,7 @@ declare module 'express-serve-static-core' { interface Request { user?: User; workspace?: Workspace; + workspaceId?: string; cacheVersion?: string | null; } } diff --git a/packages/twenty-server/package.json b/packages/twenty-server/package.json index b8d7cc58f722..c97dcfd3fb56 100644 --- a/packages/twenty-server/package.json +++ b/packages/twenty-server/package.json @@ -1,6 +1,6 @@ { "name": "twenty-server", - "version": "0.20.0", + "version": "0.22.0", "description": "", "author": "", "private": true, @@ -15,6 +15,8 @@ }, "dependencies": { "@graphql-yoga/nestjs": "patch:@graphql-yoga/nestjs@2.1.0#./patches/@graphql-yoga-nestjs-npm-2.1.0-cb509e6047.patch", + "@langchain/mistralai": "^0.0.24", + "@langchain/openai": "^0.1.3", "@nestjs/cache-manager": "^2.2.1", "@nestjs/devtools-integration": "^0.1.6", "@nestjs/graphql": "patch:@nestjs/graphql@12.1.1#./patches/@nestjs+graphql+12.1.1.patch", @@ -25,12 +27,16 @@ "graphql-middleware": "^6.1.35", "jsdom": "~22.1.0", "jwt-decode": "^4.0.0", + "langchain": "^0.2.6", + "langfuse-langchain": "^3.11.2", "lodash.differencewith": "^4.5.0", + "lodash.omitby": "^4.6.0", "lodash.uniq": "^4.5.0", "lodash.uniqby": "^4.7.0", "passport": "^0.7.0", "psl": "^1.9.0", - "tsconfig-paths": "^4.2.0" + "tsconfig-paths": "^4.2.0", + "zod-to-json-schema": "^3.23.1" }, "devDependencies": { "@nestjs/cli": "10.3.0", @@ -40,6 +46,7 @@ "@types/lodash.isequal": "^4.5.8", "@types/lodash.isobject": "^3.0.7", "@types/lodash.omit": "^4.5.9", + "@types/lodash.omitby": "^4.6.9", "@types/lodash.snakecase": "^4.1.7", "@types/lodash.uniq": "^4.5.9", "@types/lodash.uniqby": "^4.7.9", diff --git a/packages/twenty-server/scripts/render-worker.sh b/packages/twenty-server/scripts/render-worker.sh new file mode 100755 index 000000000000..1a34c6f7c3d7 --- /dev/null +++ b/packages/twenty-server/scripts/render-worker.sh @@ -0,0 +1,3 @@ +#!/bin/sh +export PG_DATABASE_URL=postgres://twenty:twenty@$PG_DATABASE_HOST:$PG_DATABASE_PORT/default +node dist/src/queue-worker/queue-worker diff --git a/packages/twenty-server/src/app.module.ts b/packages/twenty-server/src/app.module.ts index 95e8b31a0199..914b5b295f6c 100644 --- a/packages/twenty-server/src/app.module.ts +++ b/packages/twenty-server/src/app.module.ts @@ -21,9 +21,11 @@ import { GraphQLConfigModule } from 'src/engine/api/graphql/graphql-config/graph import { GraphQLConfigService } from 'src/engine/api/graphql/graphql-config/graphql-config.service'; import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; import { GraphQLHydrateRequestFromTokenMiddleware } from 'src/engine/middlewares/graphql-hydrate-request-from-token.middleware'; +import { MessageQueueModule } from 'src/engine/integrations/message-queue/message-queue.module'; +import { MessageQueueDriverType } from 'src/engine/integrations/message-queue/interfaces'; -import { CoreEngineModule } from './engine/core-modules/core-engine.module'; import { IntegrationsModule } from './engine/integrations/integrations.module'; +import { CoreEngineModule } from './engine/core-modules/core-engine.module'; @Module({ imports: [ @@ -72,6 +74,13 @@ export class AppModule { ); } + // Messaque Queue explorer only for sync driver + // Maybe we don't need to conditionaly register the explorer, because we're creating a jobs module + // that will expose classes that are only used in the queue worker + if (process.env.MESSAGE_QUEUE_TYPE === MessageQueueDriverType.Sync) { + modules.push(MessageQueueModule.registerExplorer()); + } + return modules; } diff --git a/packages/twenty-server/src/command/command.module.ts b/packages/twenty-server/src/command/command.module.ts index 86c629f92898..70fa379446f6 100644 --- a/packages/twenty-server/src/command/command.module.ts +++ b/packages/twenty-server/src/command/command.module.ts @@ -3,20 +3,15 @@ import { Module } from '@nestjs/common'; import { DatabaseCommandModule } from 'src/database/commands/database-command.module'; import { WorkspaceHealthCommandModule } from 'src/engine/workspace-manager/workspace-health/commands/workspace-health-command.module'; import { WorkspaceCleanerModule } from 'src/engine/workspace-manager/workspace-cleaner/workspace-cleaner.module'; -import { CalendarCronCommandsModule } from 'src/modules/calendar/crons/commands/calendar-cron-commands.module'; import { AppModule } from 'src/app.module'; import { WorkspaceMigrationRunnerCommandsModule } from 'src/engine/workspace-manager/workspace-migration-runner/commands/workspace-sync-metadata-commands.module'; import { WorkspaceSyncMetadataCommandsModule } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/workspace-sync-metadata-commands.module'; -import { CalendarCommandsModule } from 'src/modules/calendar/commands/calendar-commands.module'; @Module({ imports: [ AppModule, WorkspaceSyncMetadataCommandsModule, DatabaseCommandModule, - // MessagingCronCommandsModule, - CalendarCronCommandsModule, - CalendarCommandsModule, WorkspaceCleanerModule, WorkspaceHealthCommandModule, WorkspaceMigrationRunnerCommandsModule, diff --git a/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/start-data-seed-demo-workspace.cron.command.ts b/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/start-data-seed-demo-workspace.cron.command.ts index e9a8df6d35f2..bdf120095b23 100644 --- a/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/start-data-seed-demo-workspace.cron.command.ts +++ b/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/start-data-seed-demo-workspace.cron.command.ts @@ -1,9 +1,8 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner } from 'nest-commander'; import { dataSeedDemoWorkspaceCronPattern } from 'src/database/commands/data-seed-demo-workspace/crons/data-seed-demo-workspace-cron-pattern'; import { DataSeedDemoWorkspaceJob } from 'src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; @@ -13,7 +12,7 @@ import { MessageQueueService } from 'src/engine/integrations/message-queue/servi }) export class StartDataSeedDemoWorkspaceCronCommand extends CommandRunner { constructor( - @Inject(MessageQueue.cronQueue) + @InjectMessageQueue(MessageQueue.cronQueue) private readonly messageQueueService: MessageQueueService, ) { super(); diff --git a/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/stop-data-seed-demo-workspace.cron.command.ts b/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/stop-data-seed-demo-workspace.cron.command.ts index 84cd4a28df16..99dd1558bf7f 100644 --- a/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/stop-data-seed-demo-workspace.cron.command.ts +++ b/packages/twenty-server/src/database/commands/data-seed-demo-workspace/crons/stop-data-seed-demo-workspace.cron.command.ts @@ -1,9 +1,8 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner } from 'nest-commander'; import { dataSeedDemoWorkspaceCronPattern } from 'src/database/commands/data-seed-demo-workspace/crons/data-seed-demo-workspace-cron-pattern'; import { DataSeedDemoWorkspaceJob } from 'src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; @@ -13,7 +12,7 @@ import { MessageQueueService } from 'src/engine/integrations/message-queue/servi }) export class StopDataSeedDemoWorkspaceCronCommand extends CommandRunner { constructor( - @Inject(MessageQueue.cronQueue) + @InjectMessageQueue(MessageQueue.cronQueue) private readonly messageQueueService: MessageQueueService, ) { super(); diff --git a/packages/twenty-server/src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job.ts b/packages/twenty-server/src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job.ts index e7b963d62008..14655209c009 100644 --- a/packages/twenty-server/src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job.ts +++ b/packages/twenty-server/src/database/commands/data-seed-demo-workspace/jobs/data-seed-demo-workspace.job.ts @@ -1,15 +1,15 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { DataSeedDemoWorkspaceService } from 'src/database/commands/data-seed-demo-workspace/services/data-seed-demo-workspace.service'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; -@Injectable() -export class DataSeedDemoWorkspaceJob implements MessageQueueJob<undefined> { +@Processor(MessageQueue.cronQueue) +export class DataSeedDemoWorkspaceJob { constructor( private readonly dataSeedDemoWorkspaceService: DataSeedDemoWorkspaceService, ) {} + @Process(DataSeedDemoWorkspaceJob.name) async handle(): Promise<void> { await this.dataSeedDemoWorkspaceService.seedDemo(); } diff --git a/packages/twenty-server/src/database/commands/database-command.module.ts b/packages/twenty-server/src/database/commands/database-command.module.ts index d829bcf75221..eaaaa8624a28 100644 --- a/packages/twenty-server/src/database/commands/database-command.module.ts +++ b/packages/twenty-server/src/database/commands/database-command.module.ts @@ -1,33 +1,39 @@ import { Module } from '@nestjs/common'; import { TypeOrmModule } from '@nestjs/typeorm'; -import { ConfirmationQuestion } from 'src/database/commands/questions/confirmation.question'; -import { WorkspaceManagerModule } from 'src/engine/workspace-manager/workspace-manager.module'; -import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; -import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; -import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; -import { DataSeedWorkspaceCommand } from 'src/database/commands/data-seed-dev-workspace.command'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; -import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; import { StartDataSeedDemoWorkspaceCronCommand } from 'src/database/commands/data-seed-demo-workspace/crons/start-data-seed-demo-workspace.cron.command'; import { StopDataSeedDemoWorkspaceCronCommand } from 'src/database/commands/data-seed-demo-workspace/crons/stop-data-seed-demo-workspace.cron.command'; -import { WorkspaceAddTotalCountCommand } from 'src/database/commands/workspace-add-total-count.command'; import { DataSeedDemoWorkspaceCommand } from 'src/database/commands/data-seed-demo-workspace/data-seed-demo-workspace-command'; import { DataSeedDemoWorkspaceModule } from 'src/database/commands/data-seed-demo-workspace/data-seed-demo-workspace.module'; +import { DataSeedWorkspaceCommand } from 'src/database/commands/data-seed-dev-workspace.command'; +import { ConfirmationQuestion } from 'src/database/commands/questions/confirmation.question'; +import { UpdateMessageChannelVisibilityEnumCommand } from 'src/database/commands/upgrade-version/0-20/0-20-update-message-channel-visibility-enum.command'; +import { UpgradeTo0_22CommandModule } from 'src/database/commands/upgrade-version/0-22/0-22-upgrade-version.module'; +import { WorkspaceAddTotalCountCommand } from 'src/database/commands/workspace-add-total-count.command'; +import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; +import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; -import { UpdateMessageChannelVisibilityEnumCommand } from 'src/database/commands/update-message-channel-visibility-enum.command'; +import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; +import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; -import { UpdateMessageChannelSyncStatusEnumCommand } from 'src/database/commands/0-20-update-message-channel-sync-status-enum.command'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; +import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { WorkspaceManagerModule } from 'src/engine/workspace-manager/workspace-manager.module'; +import { WorkspaceStatusModule } from 'src/engine/workspace-manager/workspace-status/workspace-manager.module'; +import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; @Module({ imports: [ WorkspaceManagerModule, DataSourceModule, TypeORMModule, - TypeOrmModule.forFeature([Workspace], 'core'), + TypeOrmModule.forFeature( + [Workspace, BillingSubscription, FeatureFlagEntity], + 'core', + ), TypeOrmModule.forFeature( [FieldMetadataEntity, ObjectMetadataEntity], 'metadata', @@ -35,9 +41,12 @@ import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadat WorkspaceModule, WorkspaceDataSourceModule, WorkspaceSyncMetadataModule, + WorkspaceStatusModule, ObjectMetadataModule, DataSeedDemoWorkspaceModule, WorkspaceCacheVersionModule, + // Upgrades + UpgradeTo0_22CommandModule, ], providers: [ DataSeedWorkspaceCommand, @@ -47,7 +56,6 @@ import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadat StartDataSeedDemoWorkspaceCronCommand, StopDataSeedDemoWorkspaceCronCommand, UpdateMessageChannelVisibilityEnumCommand, - UpdateMessageChannelSyncStatusEnumCommand, ], }) export class DatabaseCommandModule {} diff --git a/packages/twenty-server/src/database/commands/update-message-channel-visibility-enum.command.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-20/0-20-update-message-channel-visibility-enum.command.ts similarity index 100% rename from packages/twenty-server/src/database/commands/update-message-channel-visibility-enum.command.ts rename to packages/twenty-server/src/database/commands/upgrade-version/0-20/0-20-update-message-channel-visibility-enum.command.ts diff --git a/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-add-new-address-field-to-views-with-deprecated-address.command.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-add-new-address-field-to-views-with-deprecated-address.command.ts new file mode 100644 index 000000000000..4f4522b036e8 --- /dev/null +++ b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-add-new-address-field-to-views-with-deprecated-address.command.ts @@ -0,0 +1,180 @@ +import { Logger } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; + +import chalk from 'chalk'; +import isEmpty from 'lodash.isempty'; +import { Command, CommandRunner, Option } from 'nest-commander'; +import { Repository } from 'typeorm'; + +import { TypeORMService } from 'src/database/typeorm/typeorm.service'; +import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; +import { TwentyORMManager } from 'src/engine/twenty-orm/twenty-orm.manager'; +import { WorkspaceStatusService } from 'src/engine/workspace-manager/workspace-status/services/workspace-status.service'; +import { COMPANY_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; +import { ViewFieldWorkspaceEntity } from 'src/modules/view/standard-objects/view-field.workspace-entity'; + +interface AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommandOptions { + workspaceId?: string; +} + +@Command({ + name: 'migrate-0.22:add-new-address-field-to-views-with-deprecated-address-field', + description: 'Adding new field Address to views containing old address field', +}) +export class AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommand extends CommandRunner { + private readonly logger = new Logger( + AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommand.name, + ); + constructor( + @InjectRepository(FieldMetadataEntity, 'metadata') + private readonly fieldMetadataRepository: Repository<FieldMetadataEntity>, + private readonly typeORMService: TypeORMService, + private readonly dataSourceService: DataSourceService, + private readonly workspaceCacheVersionService: WorkspaceCacheVersionService, + private readonly twentyORMManager: TwentyORMManager, + private readonly workspaceStatusService: WorkspaceStatusService, + ) { + super(); + } + + @Option({ + flags: '-w, --workspace-id [workspace_id]', + description: 'workspace id. Command runs on all workspaces if not provided', + required: false, + }) + parseWorkspaceId(value: string): string { + return value; + } + + async run( + _passedParam: string[], + options: AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommandOptions, + ): Promise<void> { + // This command can be generic-ified turning the below consts in options + const deprecatedFieldStandardId = + COMPANY_STANDARD_FIELD_IDS.address_deprecated; + const newFieldStandardId = COMPANY_STANDARD_FIELD_IDS.address; + + this.logger.log('running'); + let workspaceIds: string[] = []; + + if (options.workspaceId) { + workspaceIds = [options.workspaceId]; + } else { + const activeWorkspaceIds = + await this.workspaceStatusService.getActiveWorkspaceIds(); + + workspaceIds = activeWorkspaceIds; + } + + if (!workspaceIds.length) { + this.logger.log(chalk.yellow('No workspace found')); + + return; + } else { + this.logger.log( + chalk.green(`Running command on ${workspaceIds.length} workspaces`), + ); + } + + for (const workspaceId of workspaceIds) { + this.logger.log(`Running command for workspace ${workspaceId}`); + try { + const viewFieldRepository = + await this.twentyORMManager.getRepositoryForWorkspace( + workspaceId, + ViewFieldWorkspaceEntity, + ); + + const dataSourceMetadatas = + await this.dataSourceService.getDataSourcesMetadataFromWorkspaceId( + workspaceId, + ); + + for (const dataSourceMetadata of dataSourceMetadatas) { + const workspaceDataSource = + await this.typeORMService.connectToDataSource(dataSourceMetadata); + + if (workspaceDataSource) { + const newAddressField = await this.fieldMetadataRepository.findBy({ + workspaceId, + standardId: newFieldStandardId, + }); + + if (isEmpty(newAddressField)) { + this.logger.log( + `Error - missing new Address standard field of type Address, please run workspace-sync-metadata on your workspace (${workspaceId}) before running this command`, + ); + continue; + } + + const addressDeprecatedField = + await this.fieldMetadataRepository.findOneBy({ + workspaceId, + standardId: deprecatedFieldStandardId, + }); + + if (isEmpty(addressDeprecatedField)) { + continue; + } + + const viewsWithAddressDeprecatedField = + await viewFieldRepository.find({ + where: { + fieldMetadataId: addressDeprecatedField.id, + isVisible: true, + }, + }); + + for (const viewWithAddressDeprecatedField of viewsWithAddressDeprecatedField) { + const viewId = viewWithAddressDeprecatedField.viewId; + + const newAddressFieldInThisView = + await viewFieldRepository.findBy({ + fieldMetadataId: newAddressField[0].id, + viewId: viewWithAddressDeprecatedField.viewId as string, + isVisible: true, + }); + + if (!isEmpty(newAddressFieldInThisView)) { + continue; + } + + this.logger.log( + `Adding new address field to view ${viewId} for workspace ${workspaceId}...`, + ); + const newViewField = viewFieldRepository.create({ + viewId: viewWithAddressDeprecatedField.viewId, + fieldMetadataId: newAddressField[0].id, + position: viewWithAddressDeprecatedField.position - 0.5, + isVisible: true, + }); + + await viewFieldRepository.save(newViewField); + this.logger.log( + `New address field successfully added to view ${viewId} for workspace ${workspaceId}`, + ); + } + } + } + + await this.workspaceCacheVersionService.incrementVersion(workspaceId); + + this.logger.log( + chalk.green(`Running command on workspace ${workspaceId} done`), + ); + } catch (error) { + this.logger.log( + chalk.red( + `Running command on workspace ${workspaceId} failed with error: ${error}`, + ), + ); + continue; + } + + this.logger.log(chalk.green(`Command completed!`)); + } + } +} diff --git a/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-fix-object-metadata-id-standard-id.command.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-fix-object-metadata-id-standard-id.command.ts new file mode 100644 index 000000000000..213892637d95 --- /dev/null +++ b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-fix-object-metadata-id-standard-id.command.ts @@ -0,0 +1,101 @@ +import { Logger } from '@nestjs/common'; +import { InjectDataSource, InjectRepository } from '@nestjs/typeorm'; + +import chalk from 'chalk'; +import { Command, CommandRunner, Option } from 'nest-commander'; +import { DataSource, Repository } from 'typeorm'; + +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; +import { AUDIT_LOGS_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; + +interface FixObjectMetadataIdStandardIdCommandOptions { + workspaceId?: string; +} + +@Command({ + name: 'upgrade-0.22:fix-object-metadata-id-standard-id', + description: 'Fix object metadata id standard id', +}) +export class FixObjectMetadataIdStandardIdCommand extends CommandRunner { + private readonly logger = new Logger( + FixObjectMetadataIdStandardIdCommand.name, + ); + constructor( + @InjectRepository(Workspace, 'core') + private readonly workspaceRepository: Repository<Workspace>, + private readonly workspaceCacheVersionService: WorkspaceCacheVersionService, + @InjectDataSource('metadata') + private readonly metadataDataSource: DataSource, + ) { + super(); + } + + @Option({ + flags: '-w, --workspace-id [workspace_id]', + description: 'workspace id. Command runs on all workspaces if not provided', + required: false, + }) + parseWorkspaceId(value: string): string { + return value; + } + + async run( + _passedParam: string[], + options: FixObjectMetadataIdStandardIdCommandOptions, + ): Promise<void> { + const workspaceIds = options.workspaceId + ? [options.workspaceId] + : (await this.workspaceRepository.find()).map( + (workspace) => workspace.id, + ); + + if (!workspaceIds.length) { + this.logger.log(chalk.yellow('No workspace found')); + + return; + } + + this.logger.log( + chalk.green(`Running command on ${workspaceIds.length} workspaces`), + ); + + const metadataQueryRunner = this.metadataDataSource.createQueryRunner(); + + await metadataQueryRunner.connect(); + + const fieldMetadataRepository = + metadataQueryRunner.manager.getRepository(FieldMetadataEntity); + + for (const workspaceId of workspaceIds) { + try { + await metadataQueryRunner.startTransaction(); + + await fieldMetadataRepository.delete({ + workspaceId, + standardId: AUDIT_LOGS_STANDARD_FIELD_IDS.objectName, + name: 'objectMetadataId', + }); + + await metadataQueryRunner.commitTransaction(); + } catch (error) { + await metadataQueryRunner.rollbackTransaction(); + this.logger.log( + chalk.red(`Running command on workspace ${workspaceId} failed`), + ); + throw error; + } + + await this.workspaceCacheVersionService.incrementVersion(workspaceId); + + this.logger.log( + chalk.green(`Running command on workspace ${workspaceId} done`), + ); + } + + await metadataQueryRunner.release(); + + this.logger.log(chalk.green(`Command completed!`)); + } +} diff --git a/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-boolean-fields-null-default-values-and-null-values.command.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-boolean-fields-null-default-values-and-null-values.command.ts new file mode 100644 index 000000000000..3809600b515a --- /dev/null +++ b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-boolean-fields-null-default-values-and-null-values.command.ts @@ -0,0 +1,165 @@ +import { Logger } from '@nestjs/common'; +import { InjectDataSource, InjectRepository } from '@nestjs/typeorm'; + +import chalk from 'chalk'; +import { Command, CommandRunner, Option } from 'nest-commander'; +import { DataSource, IsNull, Repository } from 'typeorm'; + +import { TypeORMService } from 'src/database/typeorm/typeorm.service'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; +import { + FieldMetadataEntity, + FieldMetadataType, +} from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; +import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; + +interface UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommandOptions { + workspaceId?: string; +} + +@Command({ + name: 'upgrade-0.22:update-boolean-field-null-default-values-and-null-values', + description: + 'Update boolean fields null default values and null values to false', +}) +export class UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommand extends CommandRunner { + private readonly logger = new Logger( + UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommand.name, + ); + constructor( + @InjectRepository(Workspace, 'core') + private readonly workspaceRepository: Repository<Workspace>, + private readonly typeORMService: TypeORMService, + private readonly dataSourceService: DataSourceService, + private readonly workspaceCacheVersionService: WorkspaceCacheVersionService, + @InjectDataSource('metadata') + private readonly metadataDataSource: DataSource, + ) { + super(); + } + + @Option({ + flags: '-w, --workspace-id [workspace_id]', + description: 'workspace id. Command runs on all workspaces if not provided', + required: false, + }) + parseWorkspaceId(value: string): string { + return value; + } + + async run( + _passedParam: string[], + options: UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommandOptions, + ): Promise<void> { + const workspaceIds = options.workspaceId + ? [options.workspaceId] + : (await this.workspaceRepository.find()).map( + (workspace) => workspace.id, + ); + + if (!workspaceIds.length) { + this.logger.log(chalk.yellow('No workspace found')); + + return; + } + + this.logger.log( + chalk.green(`Running command on ${workspaceIds.length} workspaces`), + ); + + for (const workspaceId of workspaceIds) { + const dataSourceMetadata = + await this.dataSourceService.getLastDataSourceMetadataFromWorkspaceId( + workspaceId, + ); + + if (!dataSourceMetadata) { + this.logger.log( + `Could not find dataSourceMetadata for workspace ${workspaceId}`, + ); + continue; + } + + const workspaceDataSource = + await this.typeORMService.connectToDataSource(dataSourceMetadata); + + if (!workspaceDataSource) { + throw new Error( + `Could not connect to dataSource for workspace ${workspaceId}`, + ); + } + + const workspaceQueryRunner = workspaceDataSource.createQueryRunner(); + const metadataQueryRunner = this.metadataDataSource.createQueryRunner(); + + await workspaceQueryRunner.connect(); + await metadataQueryRunner.connect(); + + await workspaceQueryRunner.startTransaction(); + await metadataQueryRunner.startTransaction(); + + try { + const fieldMetadataRepository = + metadataQueryRunner.manager.getRepository(FieldMetadataEntity); + + const booleanFieldsWithoutDefaultValue = + await fieldMetadataRepository.find({ + where: { + workspaceId, + type: FieldMetadataType.BOOLEAN, + defaultValue: IsNull(), + }, + relations: ['object'], + }); + + for (const booleanField of booleanFieldsWithoutDefaultValue) { + if (!booleanField.object) { + this.logger.log( + `Could not find objectMetadataItem for field ${booleanField.id}`, + ); + continue; + } + + // Could be done via a batch update but it's safer in this context to run it sequentially with the ALTER TABLE + await fieldMetadataRepository.update(booleanField.id, { + defaultValue: false, + }); + + const fieldName = booleanField.name; + const tableName = computeObjectTargetTable(booleanField.object); + + await workspaceQueryRunner.query( + `UPDATE "${dataSourceMetadata.schema}"."${tableName}" SET "${fieldName}" = 'false' WHERE "${fieldName}" IS NULL`, + ); + + await workspaceQueryRunner.query( + `ALTER TABLE "${dataSourceMetadata.schema}"."${tableName}" ALTER COLUMN "${fieldName}" SET DEFAULT false;`, + ); + } + + await workspaceQueryRunner.commitTransaction(); + await metadataQueryRunner.commitTransaction(); + } catch (error) { + await workspaceQueryRunner.rollbackTransaction(); + await metadataQueryRunner.rollbackTransaction(); + this.logger.log( + chalk.red(`Running command on workspace ${workspaceId} failed`), + ); + throw error; + } finally { + await workspaceQueryRunner.release(); + await metadataQueryRunner.release(); + } + + await this.workspaceCacheVersionService.incrementVersion(workspaceId); + + this.logger.log( + chalk.green(`Running command on workspace ${workspaceId} done`), + ); + } + + this.logger.log(chalk.green(`Command completed!`)); + } +} diff --git a/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-stage-enum.command.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-stage-enum.command.ts new file mode 100644 index 000000000000..e9de0891e37d --- /dev/null +++ b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-stage-enum.command.ts @@ -0,0 +1,233 @@ +import { Logger } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; + +import chalk from 'chalk'; +import { Command, CommandRunner, Option } from 'nest-commander'; +import { Repository } from 'typeorm'; +import { v4 } from 'uuid'; + +import { TypeORMService } from 'src/database/typeorm/typeorm.service'; +import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; +import { WorkspaceStatusService } from 'src/engine/workspace-manager/workspace-status/services/workspace-status.service'; +import { MessageChannelSyncStage } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; + +interface UpdateMessageChannelSyncStageEnumCommandOptions { + workspaceId?: string; +} + +@Command({ + name: 'migrate-0.22:update-message-channel-sync-stage-enum', + description: 'Update messageChannel syncStage', +}) +export class UpdateMessageChannelSyncStageEnumCommand extends CommandRunner { + private readonly logger = new Logger( + UpdateMessageChannelSyncStageEnumCommand.name, + ); + constructor( + private readonly workspaceStatusService: WorkspaceStatusService, + @InjectRepository(FieldMetadataEntity, 'metadata') + private readonly fieldMetadataRepository: Repository<FieldMetadataEntity>, + @InjectRepository(ObjectMetadataEntity, 'metadata') + private readonly objectMetadataRepository: Repository<ObjectMetadataEntity>, + private readonly typeORMService: TypeORMService, + private readonly dataSourceService: DataSourceService, + private readonly workspaceCacheVersionService: WorkspaceCacheVersionService, + ) { + super(); + } + + @Option({ + flags: '-w, --workspace-id [workspace_id]', + description: 'workspace id. Command runs on all workspaces if not provided', + required: false, + }) + parseWorkspaceId(value: string): string { + return value; + } + + async run( + _passedParam: string[], + options: UpdateMessageChannelSyncStageEnumCommandOptions, + ): Promise<void> { + let workspaceIds: string[] = []; + + if (options.workspaceId) { + workspaceIds = [options.workspaceId]; + } else { + workspaceIds = await this.workspaceStatusService.getActiveWorkspaceIds(); + } + + if (!workspaceIds.length) { + this.logger.log(chalk.yellow('No workspace found')); + + return; + } else { + this.logger.log( + chalk.green(`Running command on ${workspaceIds.length} workspaces`), + ); + } + + for (const workspaceId of workspaceIds) { + try { + const dataSourceMetadatas = + await this.dataSourceService.getDataSourcesMetadataFromWorkspaceId( + workspaceId, + ); + + for (const dataSourceMetadata of dataSourceMetadatas) { + const workspaceDataSource = + await this.typeORMService.connectToDataSource(dataSourceMetadata); + + if (workspaceDataSource) { + const queryRunner = workspaceDataSource.createQueryRunner(); + + await queryRunner.connect(); + await queryRunner.startTransaction(); + + try { + await queryRunner.query( + `ALTER TYPE "${dataSourceMetadata.schema}"."messageChannel_syncStage_enum" RENAME TO "messageChannel_syncStage_enum_old"`, + ); + await queryRunner.query( + `CREATE TYPE "${dataSourceMetadata.schema}"."messageChannel_syncStage_enum" AS ENUM ( + 'FULL_MESSAGE_LIST_FETCH_PENDING', + 'PARTIAL_MESSAGE_LIST_FETCH_PENDING', + 'MESSAGE_LIST_FETCH_ONGOING', + 'MESSAGES_IMPORT_PENDING', + 'MESSAGES_IMPORT_ONGOING', + 'FAILED' + )`, + ); + + await queryRunner.query( + `ALTER TABLE "${dataSourceMetadata.schema}"."messageChannel" ALTER COLUMN "syncStage" DROP DEFAULT`, + ); + await queryRunner.query( + `ALTER TABLE "${dataSourceMetadata.schema}"."messageChannel" ALTER COLUMN "syncStage" TYPE text`, + ); + + await queryRunner.query( + `ALTER TABLE "${dataSourceMetadata.schema}"."messageChannel" ALTER COLUMN "syncStage" TYPE "${dataSourceMetadata.schema}"."messageChannel_syncStage_enum" USING "syncStage"::text::"${dataSourceMetadata.schema}"."messageChannel_syncStage_enum"`, + ); + + await queryRunner.query( + `ALTER TABLE "${dataSourceMetadata.schema}"."messageChannel" ALTER COLUMN "syncStage" SET DEFAULT 'FULL_MESSAGE_LIST_FETCH_PENDING'`, + ); + + await queryRunner.query( + `DROP TYPE "${dataSourceMetadata.schema}"."messageChannel_syncStage_enum_old"`, + ); + await queryRunner.commitTransaction(); + } catch (error) { + await queryRunner.rollbackTransaction(); + this.logger.log( + chalk.red(`Running command on workspace ${workspaceId} failed`), + ); + throw error; + } finally { + await queryRunner.release(); + } + } + } + + const messageChannelObjectMetadata = + await this.objectMetadataRepository.findOne({ + where: { nameSingular: 'messageChannel', workspaceId }, + }); + + if (!messageChannelObjectMetadata) { + this.logger.log( + chalk.yellow( + `Object metadata for messageChannel not found in workspace ${workspaceId}`, + ), + ); + + continue; + } + + const syncStageFieldMetadata = + await this.fieldMetadataRepository.findOne({ + where: { + name: 'syncStage', + workspaceId, + objectMetadataId: messageChannelObjectMetadata.id, + }, + }); + + if (!syncStageFieldMetadata) { + this.logger.log( + chalk.yellow( + `Field metadata for syncStage not found in workspace ${workspaceId}`, + ), + ); + + continue; + } + + const newOptions = [ + { + id: v4(), + value: MessageChannelSyncStage.FULL_MESSAGE_LIST_FETCH_PENDING, + label: 'Full messages list fetch pending', + position: 0, + color: 'blue', + }, + { + id: v4(), + value: MessageChannelSyncStage.PARTIAL_MESSAGE_LIST_FETCH_PENDING, + label: 'Partial messages list fetch pending', + position: 1, + color: 'blue', + }, + { + id: v4(), + value: MessageChannelSyncStage.MESSAGE_LIST_FETCH_ONGOING, + label: 'Messages list fetch ongoing', + position: 2, + color: 'orange', + }, + { + id: v4(), + value: MessageChannelSyncStage.MESSAGES_IMPORT_PENDING, + label: 'Messages import pending', + position: 3, + color: 'blue', + }, + { + id: v4(), + value: MessageChannelSyncStage.MESSAGES_IMPORT_ONGOING, + label: 'Messages import ongoing', + position: 4, + color: 'orange', + }, + { + id: v4(), + value: MessageChannelSyncStage.FAILED, + label: 'Failed', + position: 5, + color: 'red', + }, + ]; + + await this.fieldMetadataRepository.update(syncStageFieldMetadata.id, { + options: newOptions, + }); + + await this.workspaceCacheVersionService.incrementVersion(workspaceId); + + this.logger.log( + chalk.green(`Running command on workspace ${workspaceId} done`), + ); + } catch (error) { + this.logger.error( + `Migration failed for workspace ${workspaceId}: ${error.message}`, + ); + } + } + + this.logger.log(chalk.green(`Command completed!`)); + } +} diff --git a/packages/twenty-server/src/database/commands/0-20-update-message-channel-sync-status-enum.command.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-status-enum.command.ts similarity index 91% rename from packages/twenty-server/src/database/commands/0-20-update-message-channel-sync-status-enum.command.ts rename to packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-status-enum.command.ts index 1a1e345ff232..d31a0fcd2e48 100644 --- a/packages/twenty-server/src/database/commands/0-20-update-message-channel-sync-status-enum.command.ts +++ b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-status-enum.command.ts @@ -1,17 +1,17 @@ -import { InjectRepository } from '@nestjs/typeorm'; import { Logger } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; +import chalk from 'chalk'; import { Command, CommandRunner, Option } from 'nest-commander'; import { Repository } from 'typeorm'; -import chalk from 'chalk'; import { v4 } from 'uuid'; -import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { TypeORMService } from 'src/database/typeorm/typeorm.service'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; import { MessageChannelSyncStatus } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; interface UpdateMessageChannelSyncStatusEnumCommandOptions { @@ -19,7 +19,7 @@ interface UpdateMessageChannelSyncStatusEnumCommandOptions { } @Command({ - name: 'migrate-0.20:update-message-channel-sync-status-enum', + name: 'migrate-0.22:update-message-channel-sync-status-enum', description: 'Update messageChannel syncStatus', }) export class UpdateMessageChannelSyncStatusEnumCommand extends CommandRunner { @@ -94,9 +94,7 @@ export class UpdateMessageChannelSyncStatusEnumCommand extends CommandRunner { `ALTER TYPE "${dataSourceMetadata.schema}"."messageChannel_syncStatus_enum" RENAME TO "messageChannel_syncStatus_enum_old"`, ); await queryRunner.query( - `CREATE TYPE "${dataSourceMetadata.schema}"."messageChannel_syncStatus_enum" AS ENUM ('PENDING', - 'SUCCEEDED', - 'FAILED', + `CREATE TYPE "${dataSourceMetadata.schema}"."messageChannel_syncStatus_enum" AS ENUM ( 'ONGOING', 'NOT_SYNCED', 'COMPLETED', @@ -166,27 +164,6 @@ export class UpdateMessageChannelSyncStatusEnumCommand extends CommandRunner { } const newOptions = [ - { - id: v4(), - value: MessageChannelSyncStatus.PENDING, - label: 'Pending', - position: 0, - color: 'blue', - }, - { - id: v4(), - value: MessageChannelSyncStatus.SUCCEEDED, - label: 'Succeeded', - position: 2, - color: 'green', - }, - { - id: v4(), - value: MessageChannelSyncStatus.FAILED, - label: 'Failed', - position: 3, - color: 'red', - }, { id: v4(), value: MessageChannelSyncStatus.ONGOING, diff --git a/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-upgrade-version.command.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-upgrade-version.command.ts new file mode 100644 index 000000000000..f42b3fd39cc9 --- /dev/null +++ b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-upgrade-version.command.ts @@ -0,0 +1,59 @@ +import { Command, CommandRunner, Option } from 'nest-commander'; + +import { AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommand } from 'src/database/commands/upgrade-version/0-22/0-22-add-new-address-field-to-views-with-deprecated-address.command'; +import { FixObjectMetadataIdStandardIdCommand } from 'src/database/commands/upgrade-version/0-22/0-22-fix-object-metadata-id-standard-id.command'; +import { UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommand } from 'src/database/commands/upgrade-version/0-22/0-22-update-boolean-fields-null-default-values-and-null-values.command'; +import { UpdateMessageChannelSyncStageEnumCommand } from 'src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-stage-enum.command'; +import { UpdateMessageChannelSyncStatusEnumCommand } from 'src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-status-enum.command'; + +interface UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommandOptions { + workspaceId?: string; +} + +@Command({ + name: 'upgrade-0.22', + description: 'Upgrade to 0.22', +}) +export class UpgradeTo0_22Command extends CommandRunner { + constructor( + private readonly fixObjectMetadataIdStandardIdCommand: FixObjectMetadataIdStandardIdCommand, + private readonly updateBooleanFieldsNullDefaultValuesAndNullValuesCommand: UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommand, + private readonly addNewAddressFieldToViewsWithDeprecatedAddressFieldCommand: AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommand, + private readonly updateMessageChannelSyncStatusEnumCommand: UpdateMessageChannelSyncStatusEnumCommand, + private readonly updateMessageChannelSyncStageEnumCommand: UpdateMessageChannelSyncStageEnumCommand, + ) { + super(); + } + + @Option({ + flags: '-w, --workspace-id [workspace_id]', + description: 'workspace id. Command runs on all workspaces if not provided', + required: false, + }) + parseWorkspaceId(value: string): string { + return value; + } + + async run( + _passedParam: string[], + options: UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommandOptions, + ): Promise<void> { + await this.fixObjectMetadataIdStandardIdCommand.run(_passedParam, options); + await this.updateBooleanFieldsNullDefaultValuesAndNullValuesCommand.run( + _passedParam, + options, + ); + await this.addNewAddressFieldToViewsWithDeprecatedAddressFieldCommand.run( + _passedParam, + options, + ); + await this.updateMessageChannelSyncStatusEnumCommand.run( + _passedParam, + options, + ); + await this.updateMessageChannelSyncStageEnumCommand.run( + _passedParam, + options, + ); + } +} diff --git a/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-upgrade-version.module.ts b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-upgrade-version.module.ts new file mode 100644 index 000000000000..898f720bdb10 --- /dev/null +++ b/packages/twenty-server/src/database/commands/upgrade-version/0-22/0-22-upgrade-version.module.ts @@ -0,0 +1,56 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { DataSeedDemoWorkspaceModule } from 'src/database/commands/data-seed-demo-workspace/data-seed-demo-workspace.module'; +import { AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommand } from 'src/database/commands/upgrade-version/0-22/0-22-add-new-address-field-to-views-with-deprecated-address.command'; +import { FixObjectMetadataIdStandardIdCommand } from 'src/database/commands/upgrade-version/0-22/0-22-fix-object-metadata-id-standard-id.command'; +import { UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommand } from 'src/database/commands/upgrade-version/0-22/0-22-update-boolean-fields-null-default-values-and-null-values.command'; +import { UpdateMessageChannelSyncStageEnumCommand } from 'src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-stage-enum.command'; +import { UpdateMessageChannelSyncStatusEnumCommand } from 'src/database/commands/upgrade-version/0-22/0-22-update-message-channel-sync-status-enum.command'; +import { UpgradeTo0_22Command } from 'src/database/commands/upgrade-version/0-22/0-22-upgrade-version.command'; +import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; +import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; +import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; +import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { WorkspaceManagerModule } from 'src/engine/workspace-manager/workspace-manager.module'; +import { WorkspaceStatusModule } from 'src/engine/workspace-manager/workspace-status/workspace-manager.module'; +import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; + +@Module({ + imports: [ + WorkspaceManagerModule, + DataSourceModule, + TypeORMModule, + TypeOrmModule.forFeature( + [Workspace, BillingSubscription, FeatureFlagEntity], + 'core', + ), + TypeOrmModule.forFeature( + [FieldMetadataEntity, ObjectMetadataEntity], + 'metadata', + ), + WorkspaceModule, + WorkspaceDataSourceModule, + WorkspaceSyncMetadataModule, + WorkspaceStatusModule, + ObjectMetadataModule, + DataSeedDemoWorkspaceModule, + WorkspaceCacheVersionModule, + ], + providers: [ + FixObjectMetadataIdStandardIdCommand, + UpdateBooleanFieldsNullDefaultValuesAndNullValuesCommand, + UpdateMessageChannelSyncStatusEnumCommand, + UpdateMessageChannelSyncStageEnumCommand, + AddNewAddressFieldToViewsWithDeprecatedAddressFieldCommand, + UpgradeTo0_22Command, + ], +}) +export class UpgradeTo0_22CommandModule {} diff --git a/packages/twenty-server/src/database/typeorm-seeds/core/demo/workspaces.ts b/packages/twenty-server/src/database/typeorm-seeds/core/demo/workspaces.ts index adf4f886e9fa..8778e721a39d 100644 --- a/packages/twenty-server/src/database/typeorm-seeds/core/demo/workspaces.ts +++ b/packages/twenty-server/src/database/typeorm-seeds/core/demo/workspaces.ts @@ -16,7 +16,6 @@ export const seedWorkspaces = async ( 'domainName', 'inviteHash', 'logo', - 'subscriptionStatus', ]) .orIgnore() .values([ @@ -26,7 +25,6 @@ export const seedWorkspaces = async ( domainName: 'demo.dev', inviteHash: 'demo.dev-invite-hash', logo: '', - subscriptionStatus: 'incomplete', }, ]) .execute(); diff --git a/packages/twenty-server/src/database/typeorm-seeds/core/feature-flags.ts b/packages/twenty-server/src/database/typeorm-seeds/core/feature-flags.ts index 3a36ad753ff7..a030ad88f8a8 100644 --- a/packages/twenty-server/src/database/typeorm-seeds/core/feature-flags.ts +++ b/packages/twenty-server/src/database/typeorm-seeds/core/feature-flags.ts @@ -41,7 +41,12 @@ export const seedFeatureFlags = async ( value: true, }, { - key: FeatureFlagKeys.IsContactCreationForSentAndReceivedEmailsEnabled, + key: FeatureFlagKeys.IsMessagingAliasFetchingEnabled, + workspaceId: workspaceId, + value: true, + }, + { + key: FeatureFlagKeys.IsGoogleCalendarSyncV2Enabled, workspaceId: workspaceId, value: true, }, diff --git a/packages/twenty-server/src/database/typeorm-seeds/core/workspaces.ts b/packages/twenty-server/src/database/typeorm-seeds/core/workspaces.ts index 58830c3c9d47..196746e8bb4d 100644 --- a/packages/twenty-server/src/database/typeorm-seeds/core/workspaces.ts +++ b/packages/twenty-server/src/database/typeorm-seeds/core/workspaces.ts @@ -15,12 +15,7 @@ export const seedWorkspaces = async ( const workspaces: { [key: string]: Pick< Workspace, - | 'id' - | 'displayName' - | 'domainName' - | 'inviteHash' - | 'logo' - | 'subscriptionStatus' + 'id' | 'displayName' | 'domainName' | 'inviteHash' | 'logo' >; } = { [SEED_APPLE_WORKSPACE_ID]: { @@ -29,7 +24,6 @@ export const seedWorkspaces = async ( domainName: 'apple.dev', inviteHash: 'apple.dev-invite-hash', logo: '', - subscriptionStatus: 'incomplete', }, [SEED_TWENTY_WORKSPACE_ID]: { id: workspaceId, @@ -37,7 +31,6 @@ export const seedWorkspaces = async ( domainName: 'twenty.dev', inviteHash: 'twenty.dev-invite-hash', logo: '', - subscriptionStatus: 'incomplete', }, }; @@ -50,7 +43,6 @@ export const seedWorkspaces = async ( 'domainName', 'inviteHash', 'logo', - 'subscriptionStatus', ]) .orIgnore() .values(workspaces[workspaceId]) diff --git a/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-channel.ts b/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-channel.ts index 1c217c07cf8e..14a4117ee3c9 100644 --- a/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-channel.ts +++ b/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-channel.ts @@ -1,7 +1,7 @@ import { EntityManager } from 'typeorm'; import { DEV_SEED_CONNECTED_ACCOUNT_IDS } from 'src/database/typeorm-seeds/workspace/connected-account'; -import { CalendarChannelVisibility } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; +import { CalendarChannelVisibility } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; const tableName = 'calendarChannel'; diff --git a/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-event-participants.ts b/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-event-participants.ts index 784ad55af836..4efc44406120 100644 --- a/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-event-participants.ts +++ b/packages/twenty-server/src/database/typeorm-seeds/workspace/calendar-event-participants.ts @@ -2,7 +2,7 @@ import { EntityManager } from 'typeorm'; import { DEV_SEED_PERSON_IDS } from 'src/database/typeorm-seeds/workspace/people'; import { DEV_SEED_WORKSPACE_MEMBER_IDS } from 'src/database/typeorm-seeds/workspace/workspace-members'; -import { CalendarEventParticipantResponseStatus } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventParticipantResponseStatus } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; const tableName = 'calendarEventParticipant'; diff --git a/packages/twenty-server/src/database/typeorm-seeds/workspace/companies.ts b/packages/twenty-server/src/database/typeorm-seeds/workspace/companies.ts index 2762a29777c3..0d552b95e72e 100644 --- a/packages/twenty-server/src/database/typeorm-seeds/workspace/companies.ts +++ b/packages/twenty-server/src/database/typeorm-seeds/workspace/companies.ts @@ -29,7 +29,12 @@ export const seedCompanies = async ( 'id', 'name', 'domainName', - 'address', + 'addressAddressStreet1', + 'addressAddressStreet2', + 'addressAddressCity', + 'addressAddressState', + 'addressAddressPostcode', + 'addressAddressCountry', 'position', ]) .orIgnore() @@ -38,91 +43,156 @@ export const seedCompanies = async ( id: DEV_SEED_COMPANY_IDS.LINKEDIN, name: 'Linkedin', domainName: 'linkedin.com', - address: '', + addressAddressStreet1: 'Eutaw Street', + addressAddressStreet2: null, + addressAddressCity: 'Dublin', + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: 'Ireland', position: 1, }, { id: DEV_SEED_COMPANY_IDS.FACEBOOK, name: 'Facebook', domainName: 'facebook.com', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 2, }, { id: DEV_SEED_COMPANY_IDS.QONTO, name: 'Qonto', domainName: 'qonto.com', - address: '', + addressAddressStreet1: '18 rue de navarrin', + addressAddressStreet2: null, + addressAddressCity: 'Paris', + addressAddressState: null, + addressAddressPostcode: '75009', + addressAddressCountry: 'France', position: 3, }, { id: DEV_SEED_COMPANY_IDS.MICROSOFT, name: 'Microsoft', domainName: 'microsoft.com', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 4, }, { id: DEV_SEED_COMPANY_IDS.AIRBNB, name: 'Airbnb', domainName: 'airbnb.com', - address: '', + addressAddressStreet1: '888 Brannan St', + addressAddressStreet2: null, + addressAddressCity: 'San Francisco', + addressAddressState: 'CA', + addressAddressPostcode: '94103', + addressAddressCountry: 'United States', position: 5, }, { id: DEV_SEED_COMPANY_IDS.GOOGLE, name: 'Google', domainName: 'google.com', - address: '', + addressAddressStreet1: '760 Market St', + addressAddressStreet2: 'Floor 10', + addressAddressCity: 'San Francisco', + addressAddressState: null, + addressAddressPostcode: '94102', + addressAddressCountry: 'United States', position: 6, }, { id: DEV_SEED_COMPANY_IDS.NETFLIX, name: 'Netflix', domainName: 'netflix.com', - address: '', + addressAddressStreet1: '2300 Harrison St', + addressAddressStreet2: null, + addressAddressCity: 'San Francisco', + addressAddressState: 'CA', + addressAddressPostcode: '94110', + addressAddressCountry: 'United States', position: 7, }, { id: DEV_SEED_COMPANY_IDS.LIBEO, name: 'Libeo', domainName: 'libeo.io', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 8, }, { id: DEV_SEED_COMPANY_IDS.CLAAP, name: 'Claap', domainName: 'claap.io', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 9, }, { id: DEV_SEED_COMPANY_IDS.HASURA, name: 'Hasura', domainName: 'hasura.io', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 10, }, { id: DEV_SEED_COMPANY_IDS.WEWORK, name: 'Wework', domainName: 'wework.com', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 11, }, { id: DEV_SEED_COMPANY_IDS.SAMSUNG, name: 'Samsung', domainName: 'samsung.com', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 12, }, { id: DEV_SEED_COMPANY_IDS.ALGOLIA, name: 'Algolia', domainName: 'algolia.com', - address: '', + addressAddressStreet1: null, + addressAddressStreet2: null, + addressAddressCity: null, + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: null, position: 13, }, ]) diff --git a/packages/twenty-server/src/database/typeorm-seeds/workspace/opportunities.ts b/packages/twenty-server/src/database/typeorm-seeds/workspace/opportunities.ts index b314205ef7bb..b1c00b54dffc 100644 --- a/packages/twenty-server/src/database/typeorm-seeds/workspace/opportunities.ts +++ b/packages/twenty-server/src/database/typeorm-seeds/workspace/opportunities.ts @@ -1,7 +1,7 @@ import { EntityManager } from 'typeorm'; -import { DEV_SEED_PERSON_IDS } from 'src/database/typeorm-seeds/workspace/people'; import { DEV_SEED_COMPANY_IDS } from 'src/database/typeorm-seeds/workspace/companies'; +import { DEV_SEED_PERSON_IDS } from 'src/database/typeorm-seeds/workspace/people'; const tableName = 'opportunity'; @@ -25,7 +25,6 @@ export const seedOpportunity = async ( 'amountAmountMicros', 'amountCurrencyCode', 'closeDate', - 'probability', 'stage', 'position', 'pointOfContactId', @@ -39,7 +38,6 @@ export const seedOpportunity = async ( amountAmountMicros: 100000, amountCurrencyCode: 'USD', closeDate: new Date(), - probability: 0.5, stage: 'NEW', position: 1, pointOfContactId: DEV_SEED_PERSON_IDS.CHRISTOPH, @@ -51,7 +49,6 @@ export const seedOpportunity = async ( amountAmountMicros: 2000000, amountCurrencyCode: 'USD', closeDate: new Date(), - probability: 0.5, stage: 'MEETING', position: 2, pointOfContactId: DEV_SEED_PERSON_IDS.CHRISTOPHER_G, @@ -63,7 +60,6 @@ export const seedOpportunity = async ( amountAmountMicros: 300000, amountCurrencyCode: 'USD', closeDate: new Date(), - probability: 0.5, stage: 'PROPOSAL', position: 3, pointOfContactId: DEV_SEED_PERSON_IDS.NICHOLAS, @@ -75,7 +71,6 @@ export const seedOpportunity = async ( amountAmountMicros: 4000000, amountCurrencyCode: 'USD', closeDate: new Date(), - probability: 0.5, stage: 'PROPOSAL', position: 4, pointOfContactId: DEV_SEED_PERSON_IDS.MATTHEW, diff --git a/packages/twenty-server/src/database/typeorm/core/migrations/1719327438923-useEnumForSubscriptionStatusInterval.ts b/packages/twenty-server/src/database/typeorm/core/migrations/1719327438923-useEnumForSubscriptionStatusInterval.ts new file mode 100644 index 000000000000..c2894f0fd48a --- /dev/null +++ b/packages/twenty-server/src/database/typeorm/core/migrations/1719327438923-useEnumForSubscriptionStatusInterval.ts @@ -0,0 +1,61 @@ +import { MigrationInterface, QueryRunner } from 'typeorm'; + +export class UseEnumForSubscriptionStatusInterval1719327438923 + implements MigrationInterface +{ + name = 'UseEnumForSubscriptionStatusInterval1719327438923'; + + public async up(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query( + `CREATE TYPE "core"."billingSubscription_status_enum" AS ENUM('active', 'canceled', 'incomplete', 'incomplete_expired', 'past_due', 'paused', 'trialing', 'unpaid')`, + ); + await queryRunner.query( + `ALTER TABLE "core"."billingSubscription" ALTER COLUMN "status" TYPE "core"."billingSubscription_status_enum" USING "status"::"core"."billingSubscription_status_enum"`, + ); + await queryRunner.query( + `ALTER TABLE "core"."billingSubscription" ALTER COLUMN "status" SET NOT NULL`, + ); + await queryRunner.query( + `CREATE TYPE "core"."billingSubscription_interval_enum" AS ENUM('day', 'month', 'week', 'year')`, + ); + await queryRunner.query( + `ALTER TABLE "core"."billingSubscription" ALTER COLUMN "interval" TYPE "core"."billingSubscription_interval_enum" USING "interval"::"core"."billingSubscription_interval_enum"`, + ); + await queryRunner.query( + `CREATE TYPE "core"."workspace_subscriptionstatus_enum" AS ENUM('active', 'canceled', 'incomplete', 'incomplete_expired', 'past_due', 'paused', 'trialing', 'unpaid')`, + ); + await queryRunner.query( + `ALTER TABLE "core"."workspace" ALTER COLUMN "subscriptionStatus" DROP DEFAULT`, + ); + await queryRunner.query( + `ALTER TABLE "core"."workspace" ALTER COLUMN "subscriptionStatus" TYPE "core"."workspace_subscriptionstatus_enum" USING "subscriptionStatus"::"core"."workspace_subscriptionstatus_enum"`, + ); + await queryRunner.query( + `ALTER TABLE "core"."workspace" ALTER COLUMN "subscriptionStatus" SET NOT NULL`, + ); + await queryRunner.query( + `ALTER TABLE "core"."workspace" ALTER COLUMN "subscriptionStatus" SET DEFAULT 'incomplete'::"core"."workspace_subscriptionstatus_enum"`, + ); + } + + public async down(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query( + `ALTER TABLE "core"."workspace" ALTER COLUMN "subscriptionStatus" TYPE text`, + ); + await queryRunner.query( + `DROP TYPE "core"."workspace_subscriptionstatus_enum"`, + ); + await queryRunner.query( + `ALTER TABLE "core"."billingSubscription" ALTER COLUMN "interval" TYPE text`, + ); + await queryRunner.query( + `DROP TYPE "core"."billingSubscription_interval_enum"`, + ); + await queryRunner.query( + `ALTER TABLE "core"."billingSubscription" ALTER COLUMN "status" TYPE text`, + ); + await queryRunner.query( + `DROP TYPE "core"."billingSubscription_status_enum"`, + ); + } +} diff --git a/packages/twenty-server/src/database/typeorm/core/migrations/1719494707738-removeSubscriptionStatusFromCoreWorkspace.ts b/packages/twenty-server/src/database/typeorm/core/migrations/1719494707738-removeSubscriptionStatusFromCoreWorkspace.ts new file mode 100644 index 000000000000..85e64b408c4c --- /dev/null +++ b/packages/twenty-server/src/database/typeorm/core/migrations/1719494707738-removeSubscriptionStatusFromCoreWorkspace.ts @@ -0,0 +1,25 @@ +import { MigrationInterface, QueryRunner } from 'typeorm'; + +export class RemoveSubscriptionStatusFromCoreWorkspace1719494707738 + implements MigrationInterface +{ + name = 'RemoveSubscriptionStatusFromCoreWorkspace1719494707738'; + + public async up(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query( + `ALTER TABLE "core"."workspace" DROP COLUMN "subscriptionStatus"`, + ); + await queryRunner.query( + `DROP TYPE "core"."workspace_subscriptionstatus_enum"`, + ); + } + + public async down(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query( + `CREATE TYPE "core"."workspace_subscriptionstatus_enum" AS ENUM('active', 'canceled', 'incomplete', 'incomplete_expired', 'past_due', 'paused', 'trialing', 'unpaid')`, + ); + await queryRunner.query( + `ALTER TABLE "core"."workspace" ADD "subscriptionStatus" "core"."workspace_subscriptionstatus_enum" NOT NULL DEFAULT 'incomplete'`, + ); + } +} diff --git a/packages/twenty-server/src/database/typeorm/metadata/migrations/1718985664968-addIndexMetadataTable.ts b/packages/twenty-server/src/database/typeorm/metadata/migrations/1718985664968-addIndexMetadataTable.ts new file mode 100644 index 000000000000..c4d5afca948d --- /dev/null +++ b/packages/twenty-server/src/database/typeorm/metadata/migrations/1718985664968-addIndexMetadataTable.ts @@ -0,0 +1,28 @@ +import { MigrationInterface, QueryRunner } from 'typeorm'; + +export class AddIndexMetadataTable1718985664968 implements MigrationInterface { + name = 'AddIndexMetadataTable1718985664968'; + + public async up(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query( + `CREATE TABLE "metadata"."indexMetadata" ("id" uuid NOT NULL DEFAULT uuid_generate_v4(), "name" character varying NOT NULL, "workspaceId" character varying, "objectMetadataId" uuid NOT NULL, CONSTRAINT "PK_f73bb3c3678aee204e341f0ca4e" PRIMARY KEY ("id"))`, + ); + await queryRunner.query( + `CREATE TABLE "metadata"."indexFieldMetadata" ("id" uuid NOT NULL DEFAULT uuid_generate_v4(), "indexMetadataId" uuid NOT NULL, "fieldMetadataId" uuid NOT NULL, "order" integer NOT NULL, CONSTRAINT "PK_5928f67e43eff7d95aa79fd96fd" PRIMARY KEY ("id"))`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexMetadata" ADD CONSTRAINT "FK_051487e9b745cb175950130b63f" FOREIGN KEY ("objectMetadataId") REFERENCES "metadata"."objectMetadata"("id") ON DELETE CASCADE ON UPDATE NO ACTION`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexFieldMetadata" ADD CONSTRAINT "FK_b20192c432612eb710801dd5664" FOREIGN KEY ("indexMetadataId") REFERENCES "metadata"."indexMetadata"("id") ON DELETE CASCADE ON UPDATE NO ACTION`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexFieldMetadata" ADD CONSTRAINT "FK_be0950612a54b58c72bd62d629e" FOREIGN KEY ("fieldMetadataId") REFERENCES "metadata"."fieldMetadata"("id") ON DELETE CASCADE ON UPDATE NO ACTION`, + ); + } + + public async down(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query(`DROP TABLE "metadata"."indexFieldMetadata"`); + await queryRunner.query(`DROP TABLE "metadata"."indexMetadata"`); + } +} diff --git a/packages/twenty-server/src/database/typeorm/metadata/migrations/1720524654925-addDateColumnsToIndexMetadata.ts b/packages/twenty-server/src/database/typeorm/metadata/migrations/1720524654925-addDateColumnsToIndexMetadata.ts new file mode 100644 index 000000000000..6753c5c017ed --- /dev/null +++ b/packages/twenty-server/src/database/typeorm/metadata/migrations/1720524654925-addDateColumnsToIndexMetadata.ts @@ -0,0 +1,37 @@ +import { MigrationInterface, QueryRunner } from 'typeorm'; + +export class AddDateColumnsToIndexMetadata1720524654925 + implements MigrationInterface +{ + name = 'AddDateColumnsToIndexMetadata1720524654925'; + + public async up(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query( + `ALTER TABLE "metadata"."indexMetadata" ADD "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT now()`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexMetadata" ADD "updatedAt" TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT now()`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexFieldMetadata" ADD "createdAt" TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT now()`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexFieldMetadata" ADD "updatedAt" TIMESTAMP WITH TIME ZONE NOT NULL DEFAULT now()`, + ); + } + + public async down(queryRunner: QueryRunner): Promise<void> { + await queryRunner.query( + `ALTER TABLE "metadata"."indexFieldMetadata" DROP COLUMN "updatedAt"`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexFieldMetadata" DROP COLUMN "createdAt"`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexMetadata" DROP COLUMN "updatedAt"`, + ); + await queryRunner.query( + `ALTER TABLE "metadata"."indexMetadata" DROP COLUMN "createdAt"`, + ); + } +} diff --git a/packages/twenty-server/src/database/typeorm/typeorm.module.ts b/packages/twenty-server/src/database/typeorm/typeorm.module.ts index b24fe05bad94..bda843eae6f4 100644 --- a/packages/twenty-server/src/database/typeorm/typeorm.module.ts +++ b/packages/twenty-server/src/database/typeorm/typeorm.module.ts @@ -3,6 +3,7 @@ import { TypeOrmModule, TypeOrmModuleOptions } from '@nestjs/typeorm'; import { typeORMCoreModuleOptions } from 'src/database/typeorm/core/core.datasource'; import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; import { TypeORMService } from './typeorm.service'; @@ -28,6 +29,9 @@ const coreTypeORMFactory = async (): Promise<TypeOrmModuleOptions> => ({ useFactory: coreTypeORMFactory, name: 'core', }), + TwentyORMModule.register({ + workspaceEntities: ['dist/src/**/*.workspace-entity{.ts,.js}'], + }), EnvironmentModule, ], providers: [TypeORMService], diff --git a/packages/twenty-server/src/engine/api/__mocks__/object-metadata-item.mock.ts b/packages/twenty-server/src/engine/api/__mocks__/object-metadata-item.mock.ts index 8767507e5e57..d988d614a298 100644 --- a/packages/twenty-server/src/engine/api/__mocks__/object-metadata-item.mock.ts +++ b/packages/twenty-server/src/engine/api/__mocks__/object-metadata-item.mock.ts @@ -75,7 +75,7 @@ const fieldMultiSelectMock = { ], }; -const fieldRelationMock = { +export const fieldRelationMock = { name: 'fieldRelation', type: FieldMetadataType.RELATION, fromRelationMetadata: { @@ -145,13 +145,6 @@ const fieldNumericMock = { defaultValue: null, }; -const fieldProbabilityMock = { - name: 'fieldProbability', - type: FieldMetadataType.PROBABILITY, - isNullable: true, - defaultValue: null, -}; - const fieldFullNameMock = { name: 'fieldFullName', type: FieldMetadataType.FULL_NAME, @@ -206,7 +199,6 @@ export const fields = [ fieldBooleanMock, fieldNumberMock, fieldNumericMock, - fieldProbabilityMock, fieldLinkMock, fieldLinksMock, fieldCurrencyMock, diff --git a/packages/twenty-server/src/engine/api/graphql/graphql-config/graphql-config.service.ts b/packages/twenty-server/src/engine/api/graphql/graphql-config/graphql-config.service.ts index b71ff69d4dd8..266dbf2ce48c 100644 --- a/packages/twenty-server/src/engine/api/graphql/graphql-config/graphql-config.service.ts +++ b/packages/twenty-server/src/engine/api/graphql/graphql-config/graphql-config.service.ts @@ -141,8 +141,10 @@ export class GraphQLConfigService // Create a new contextId for each request const contextId = ContextIdFactory.create(); - // Register the request in the contextId - this.moduleRef.registerRequestByContextId(context.req, contextId); + if (this.moduleRef.registerRequestByContextId) { + // Register the request in the contextId + this.moduleRef.registerRequestByContextId(context.req, contextId); + } // Resolve the WorkspaceSchemaFactory for the contextId const workspaceFactory = await this.moduleRef.resolve( diff --git a/packages/twenty-server/src/engine/api/graphql/metadata.module-factory.ts b/packages/twenty-server/src/engine/api/graphql/metadata.module-factory.ts index 5c28d63f3947..3d2e2b7e1584 100644 --- a/packages/twenty-server/src/engine/api/graphql/metadata.module-factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/metadata.module-factory.ts @@ -1,15 +1,15 @@ import { YogaDriverConfig } from '@graphql-yoga/nestjs'; import GraphQLJSON from 'graphql-type-json'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service'; -import { useExceptionHandler } from 'src/engine/integrations/exception-handler/hooks/use-exception-handler.hook'; +import { useCachedMetadata } from 'src/engine/api/graphql/graphql-config/hooks/use-cached-metadata'; import { useThrottler } from 'src/engine/api/graphql/graphql-config/hooks/use-throttler'; import { MetadataGraphQLApiModule } from 'src/engine/api/graphql/metadata-graphql-api.module'; -import { renderApolloPlayground } from 'src/engine/utils/render-apollo-playground.util'; +import { useGraphQLErrorHandlerHook } from 'src/engine/core-modules/graphql/hooks/use-graphql-error-handler.hook'; import { DataloaderService } from 'src/engine/dataloaders/dataloader.service'; -import { useCachedMetadata } from 'src/engine/api/graphql/graphql-config/hooks/use-cached-metadata'; import { CacheStorageService } from 'src/engine/integrations/cache-storage/cache-storage.service'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; +import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service'; +import { renderApolloPlayground } from 'src/engine/utils/render-apollo-playground.util'; export const metadataModuleFactory = async ( environmentService: EnvironmentService, @@ -32,7 +32,7 @@ export const metadataModuleFactory = async ( return context.req.user?.id ?? context.req.ip ?? 'anonymous'; }, }), - useExceptionHandler({ + useGraphQLErrorHandlerHook({ exceptionHandlerService, }), useCachedMetadata({ diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/__tests__/args-string.factory.spec.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/__tests__/args-string.factory.spec.ts index 4cc9394e1adb..f4be0b8fb204 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/__tests__/args-string.factory.spec.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/__tests__/args-string.factory.spec.ts @@ -68,10 +68,7 @@ describe('ArgsStringFactory', () => { it('when orderBy is present, should return an array of objects', () => { const args = { - orderBy: { - id: 'AscNullsFirst', - name: 'AscNullsFirst', - }, + orderBy: [{ id: 'AscNullsFirst' }, { name: 'AscNullsFirst' }], }; argsAliasCreate.mockReturnValue(args); @@ -85,11 +82,11 @@ describe('ArgsStringFactory', () => { it('when orderBy is present with position criteria, should return position at the end of the list', () => { const args = { - orderBy: { - position: 'AscNullsFirst', - id: 'AscNullsFirst', - name: 'AscNullsFirst', - }, + orderBy: [ + { position: 'AscNullsFirst' }, + { id: 'AscNullsFirst' }, + { name: 'AscNullsFirst' }, + ], }; argsAliasCreate.mockReturnValue(args); @@ -103,11 +100,11 @@ describe('ArgsStringFactory', () => { it('when orderBy is present with position in the middle, should return position at the end of the list', () => { const args = { - orderBy: { - id: 'AscNullsFirst', - position: 'AscNullsFirst', - name: 'AscNullsFirst', - }, + orderBy: [ + { id: 'AscNullsFirst' }, + { position: 'AscNullsFirst' }, + { name: 'AscNullsFirst' }, + ], }; argsAliasCreate.mockReturnValue(args); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/__tests__/find-duplicates-query.factory.spec.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/__tests__/find-duplicates-query.factory.spec.ts deleted file mode 100644 index b2423c9207f7..000000000000 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/__tests__/find-duplicates-query.factory.spec.ts +++ /dev/null @@ -1,206 +0,0 @@ -import { Test, TestingModule } from '@nestjs/testing'; - -import { RecordFilter } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; -import { FindDuplicatesResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; - -import { ArgsAliasFactory } from 'src/engine/api/graphql/workspace-query-builder/factories/args-alias.factory'; -import { FieldsStringFactory } from 'src/engine/api/graphql/workspace-query-builder/factories/fields-string.factory'; -import { FindDuplicatesQueryFactory } from 'src/engine/api/graphql/workspace-query-builder/factories/find-duplicates-query.factory'; -import { workspaceQueryBuilderOptionsMock } from 'src/engine/api/graphql/workspace-query-builder/__mocks__/workspace-query-builder-options.mock'; - -describe('FindDuplicatesQueryFactory', () => { - let service: FindDuplicatesQueryFactory; - const argAliasCreate = jest.fn(); - - beforeEach(async () => { - jest.resetAllMocks(); - - const module: TestingModule = await Test.createTestingModule({ - providers: [ - FindDuplicatesQueryFactory, - { - provide: FieldsStringFactory, - useValue: { - create: jest.fn().mockResolvedValue('fieldsString'), - // Mock implementation of FieldsStringFactory methods if needed - }, - }, - { - provide: ArgsAliasFactory, - useValue: { - create: argAliasCreate, - // Mock implementation of ArgsAliasFactory methods if needed - }, - }, - ], - }).compile(); - - service = module.get<FindDuplicatesQueryFactory>( - FindDuplicatesQueryFactory, - ); - }); - - it('should be defined', () => { - expect(service).toBeDefined(); - }); - - describe('create', () => { - it('should return (first: 0) as a filter when args are missing', async () => { - const args: FindDuplicatesResolverArgs<RecordFilter> = {}; - - const query = await service.create( - args, - workspaceQueryBuilderOptionsMock, - ); - - expect(query.trim()).toEqual(`query { - objectNameCollection(first: 0) { - fieldsString - } - }`); - }); - - it('should use firstName and lastName as a filter when both args are present', async () => { - argAliasCreate.mockReturnValue({ - nameFirstName: 'John', - nameLastName: 'Doe', - }); - - const args: FindDuplicatesResolverArgs<RecordFilter> = { - data: { - name: { - firstName: 'John', - lastName: 'Doe', - }, - } as unknown as RecordFilter, - }; - - const query = await service.create(args, { - ...workspaceQueryBuilderOptionsMock, - objectMetadataItem: { - ...workspaceQueryBuilderOptionsMock.objectMetadataItem, - nameSingular: 'person', - }, - }); - - expect(query.trim()).toEqual(`query { - personCollection(filter: {or:[{nameFirstName:{eq:"John"},nameLastName:{eq:"Doe"}}]}) { - fieldsString - } - }`); - }); - - it('should ignore an argument if the string length is less than 3', async () => { - argAliasCreate.mockReturnValue({ - linkedinLinkUrl: 'ab', - email: 'test@test.com', - }); - - const args: FindDuplicatesResolverArgs<RecordFilter> = { - data: { - linkedinLinkUrl: 'ab', - email: 'test@test.com', - } as unknown as RecordFilter, - }; - - const query = await service.create(args, { - ...workspaceQueryBuilderOptionsMock, - objectMetadataItem: { - ...workspaceQueryBuilderOptionsMock.objectMetadataItem, - nameSingular: 'person', - }, - }); - - expect(query.trim()).toEqual(`query { - personCollection(filter: {or:[{email:{eq:"test@test.com"}}]}) { - fieldsString - } - }`); - }); - - it('should return (first: 0) as a filter when only firstName is present', async () => { - argAliasCreate.mockReturnValue({ - nameFirstName: 'John', - }); - - const args: FindDuplicatesResolverArgs<RecordFilter> = { - data: { - name: { - firstName: 'John', - }, - } as unknown as RecordFilter, - }; - - const query = await service.create(args, { - ...workspaceQueryBuilderOptionsMock, - objectMetadataItem: { - ...workspaceQueryBuilderOptionsMock.objectMetadataItem, - nameSingular: 'person', - }, - }); - - expect(query.trim()).toEqual(`query { - personCollection(first: 0) { - fieldsString - } - }`); - }); - - it('should use "currentRecord" as query args when its present', async () => { - argAliasCreate.mockReturnValue({ - nameFirstName: 'John', - }); - - const args: FindDuplicatesResolverArgs<RecordFilter> = { - id: 'uuid', - }; - - const query = await service.create( - args, - { - ...workspaceQueryBuilderOptionsMock, - objectMetadataItem: { - ...workspaceQueryBuilderOptionsMock.objectMetadataItem, - nameSingular: 'person', - }, - }, - { - nameFirstName: 'Peter', - nameLastName: 'Parker', - }, - ); - - expect(query.trim()).toEqual(`query { - personCollection(filter: {id:{neq:"uuid"},or:[{nameFirstName:{eq:"Peter"},nameLastName:{eq:"Parker"}}]}) { - fieldsString - } - }`); - }); - }); - - describe('buildQueryForExistingRecord', () => { - it(`should include all the fields that exist for person inside "duplicateCriteriaCollection" constant`, async () => { - const query = service.buildQueryForExistingRecord('uuid', { - ...workspaceQueryBuilderOptionsMock, - objectMetadataItem: { - ...workspaceQueryBuilderOptionsMock.objectMetadataItem, - nameSingular: 'person', - }, - }); - - expect(query.trim()).toEqual(`query { - personCollection(filter: { id: { eq: "uuid" }}){ - edges { - node { - __typename - nameFirstName -nameLastName -linkedinLinkUrl -email - } - } - } - }`); - }); - }); -}); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/args-string.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/args-string.factory.ts index d28c124868a3..d4b8efd77ed6 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/args-string.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/args-string.factory.ts @@ -36,11 +36,16 @@ export class ArgsStringFactory { typeof computedArgs[key] === 'object' && computedArgs[key] !== null ) { - // If it's an object (and not null), stringify it - argsString += `${key}: ${this.buildStringifiedObject( - key, - computedArgs[key], - )}, `; + if (key === 'orderBy') { + argsString += `${key}: ${this.buildStringifiedOrderBy( + computedArgs[key], + )}, `; + } else { + // If it's an object (and not null), stringify it + argsString += `${key}: ${stringifyWithoutKeyQuote( + computedArgs[key], + )}, `; + } } else { // For other types (number, boolean), add as is argsString += `${key}: ${computedArgs[key]}, `; @@ -55,22 +60,30 @@ export class ArgsStringFactory { return argsString; } - private buildStringifiedObject( - key: string, - obj: Record<string, any>, + private buildStringifiedOrderBy( + keyValuePairArray: Array<Record<string, any>>, ): string { - // PgGraphql is expecting the orderBy argument to be an array of objects - if (key === 'orderBy') { - const orderByString = Object.keys(obj) - .sort((_, b) => { - return b === 'position' ? -1 : 0; - }) - .map((orderByKey) => `{${orderByKey}: ${obj[orderByKey]}}`) - .join(', '); + if ( + keyValuePairArray.length !== 0 && + Object.keys(keyValuePairArray[0]).length === 0 + ) { + return `[]`; + } + // if position argument is present we want to put it at the very last + let orderByString = keyValuePairArray + .sort((_, obj) => (Object.hasOwnProperty.call(obj, 'position') ? -1 : 0)) + .map((obj) => { + const [key] = Object.keys(obj); + const value = obj[key]; + + return `{${key}: ${value}}`; + }) + .join(', '); - return `[${orderByString}]`; + if (orderByString.endsWith(', ')) { + orderByString = orderByString.slice(0, -2); } - return stringifyWithoutKeyQuote(obj); + return `[${orderByString}]`; } } diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/create-many-query.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/create-many-query.factory.ts index b667472a3d3e..f44d1fbcb7c8 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/create-many-query.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/create-many-query.factory.ts @@ -22,7 +22,7 @@ export class CreateManyQueryFactory { ) {} async create<Record extends IRecord = IRecord>( - args: CreateManyResolverArgs<Record>, + args: CreateManyResolverArgs<Partial<Record>>, options: WorkspaceQueryBuilderOptions, ) { const fieldsString = await this.fieldsStringFactory.create( diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/fields-string.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/fields-string.factory.ts index e3ae2dfde33f..1acc3778d15f 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/fields-string.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/fields-string.factory.ts @@ -6,6 +6,7 @@ import isEmpty from 'lodash.isempty'; import { FieldMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/field-metadata.interface'; import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; +import { Record } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; import { isRelationFieldMetadataType } from 'src/engine/utils/is-relation-field-metadata-type.util'; @@ -26,7 +27,7 @@ export class FieldsStringFactory { fieldMetadataCollection: FieldMetadataInterface[], objectMetadataCollection: ObjectMetadataInterface[], ): Promise<string> { - const selectedFields: Record<string, any> = graphqlFields(info); + const selectedFields: Partial<Record> = graphqlFields(info); return this.createFieldsStringRecursive( info, @@ -38,7 +39,7 @@ export class FieldsStringFactory { async createFieldsStringRecursive( info: GraphQLResolveInfo, - selectedFields: Record<string, any>, + selectedFields: Partial<Record>, fieldMetadataCollection: FieldMetadataInterface[], objectMetadataCollection: ObjectMetadataInterface[], accumulator = '', diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/find-duplicates-query.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/find-duplicates-query.factory.ts index 03655d4aeafe..5281dc7be5e7 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/find-duplicates-query.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/find-duplicates-query.factory.ts @@ -3,15 +3,13 @@ import { Injectable, Logger } from '@nestjs/common'; import isEmpty from 'lodash.isempty'; import { WorkspaceQueryBuilderOptions } from 'src/engine/api/graphql/workspace-query-builder/interfaces/workspace-query-builder-options.interface'; -import { RecordFilter } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; +import { Record } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; import { FindDuplicatesResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; -import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; import { stringifyWithoutKeyQuote } from 'src/engine/api/graphql/workspace-query-builder/utils/stringify-without-key-quote.util'; import { ArgsAliasFactory } from 'src/engine/api/graphql/workspace-query-builder/factories/args-alias.factory'; -import { DUPLICATE_CRITERIA_COLLECTION } from 'src/engine/api/graphql/workspace-resolver-builder/constants/duplicate-criteria.constants'; -import { settings } from 'src/engine/constants/settings'; +import { DuplicateService } from 'src/engine/core-modules/duplicate/duplicate.service'; import { FieldsStringFactory } from './fields-string.factory'; @@ -22,12 +20,13 @@ export class FindDuplicatesQueryFactory { constructor( private readonly fieldsStringFactory: FieldsStringFactory, private readonly argsAliasFactory: ArgsAliasFactory, + private readonly duplicateService: DuplicateService, ) {} - async create<Filter extends RecordFilter = RecordFilter>( - args: FindDuplicatesResolverArgs<Filter>, + async create( + args: FindDuplicatesResolverArgs, options: WorkspaceQueryBuilderOptions, - currentRecord?: Record<string, unknown>, + existingRecords?: Record[], ) { const fieldsString = await this.fieldsStringFactory.create( options.info, @@ -35,121 +34,66 @@ export class FindDuplicatesQueryFactory { options.objectMetadataCollection, ); - const argsData = this.getFindDuplicateBy<Filter>( - args, - options, - currentRecord, - ); - - const duplicateCondition = this.buildDuplicateCondition( - options.objectMetadataItem, - argsData, - args.id, - ); - - const filters = stringifyWithoutKeyQuote(duplicateCondition); - - return ` - query { - ${computeObjectTargetTable(options.objectMetadataItem)}Collection${ - isEmpty(duplicateCondition?.or) - ? '(first: 0)' - : `(filter: ${filters})` - } { - ${fieldsString} - } - } - `; - } + if (existingRecords) { + const query = existingRecords.reduce((acc, record, index) => { + return ( + acc + this.buildQuery(fieldsString, options, undefined, record, index) + ); + }, ''); - getFindDuplicateBy<Filter extends RecordFilter = RecordFilter>( - args: FindDuplicatesResolverArgs<Filter>, - options: WorkspaceQueryBuilderOptions, - currentRecord?: Record<string, unknown>, - ) { - if (currentRecord) { - return currentRecord; + return `query { + ${query} + }`; } - return this.argsAliasFactory.create( - args.data ?? {}, - options.fieldMetadataCollection, - ); + const query = args.data?.reduce((acc, dataItem, index) => { + const argsData = this.argsAliasFactory.create( + dataItem ?? {}, + options.fieldMetadataCollection, + ); + + return ( + acc + + this.buildQuery( + fieldsString, + options, + argsData as Record, + undefined, + index, + ) + ); + }, ''); + + return `query { + ${query} + }`; } - buildQueryForExistingRecord( - id: string | number, + buildQuery( + fieldsString: string, options: WorkspaceQueryBuilderOptions, + data?: Record, + existingRecord?: Record, + index?: number, ) { - const idQueryField = typeof id === 'string' ? `"${id}"` : id; + const duplicateCondition = + this.duplicateService.buildDuplicateConditionForGraphQL( + options.objectMetadataItem, + data ?? existingRecord, + existingRecord?.id, + ); - return ` - query { - ${computeObjectTargetTable( - options.objectMetadataItem, - )}Collection(filter: { id: { eq: ${idQueryField} }}){ - edges { - node { - __typename - ${this.getApplicableDuplicateCriteriaCollection( - options.objectMetadataItem, - ) - .flatMap((dc) => dc.columnNames) - .join('\n')} - } - } - } - } - `; - } + const filters = stringifyWithoutKeyQuote(duplicateCondition); - private buildDuplicateCondition( - objectMetadataItem: ObjectMetadataInterface, - argsData?: Record<string, unknown>, - filteringByExistingRecordId?: string, - ) { - if (!argsData) { - return; + return `${computeObjectTargetTable( + options.objectMetadataItem, + )}Collection${index}: ${computeObjectTargetTable( + options.objectMetadataItem, + )}Collection${ + isEmpty(duplicateCondition?.or) ? '(first: 0)' : `(filter: ${filters})` + } { + ${fieldsString} } - - const criteriaCollection = - this.getApplicableDuplicateCriteriaCollection(objectMetadataItem); - - const criteriaWithMatchingArgs = criteriaCollection.filter((criteria) => - criteria.columnNames.every((columnName) => { - const value = argsData[columnName] as string | undefined; - - return ( - !!value && value.length >= settings.minLengthOfStringForDuplicateCheck - ); - }), - ); - - const filterCriteria = criteriaWithMatchingArgs.map((criteria) => - Object.fromEntries( - criteria.columnNames.map((columnName) => [ - columnName, - { eq: argsData[columnName] }, - ]), - ), - ); - - return { - // when filtering by an existing record, we need to filter that explicit record out - ...(filteringByExistingRecordId && { - id: { neq: filteringByExistingRecordId }, - }), - // keep condition as "or" to get results by more duplicate criteria - or: filterCriteria, - }; - } - - private getApplicableDuplicateCriteriaCollection( - objectMetadataItem: ObjectMetadataInterface, - ) { - return DUPLICATE_CRITERIA_COLLECTION.filter( - (duplicateCriteria) => - duplicateCriteria.objectName === objectMetadataItem.nameSingular, - ); + `; } } diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-many-query.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-many-query.factory.ts index 936013110d3d..f55e512bac2d 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-many-query.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-many-query.factory.ts @@ -28,7 +28,7 @@ export class UpdateManyQueryFactory { Record extends IRecord = IRecord, Filter extends RecordFilter = RecordFilter, >( - args: UpdateManyResolverArgs<Record, Filter>, + args: UpdateManyResolverArgs<Partial<Record>, Filter>, options: UpdateManyQueryFactoryOptions, ) { const fieldsString = await this.fieldsStringFactory.create( diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-one-query.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-one-query.factory.ts index ba86a60fcdb7..57df907d95e6 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-one-query.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/factories/update-one-query.factory.ts @@ -20,7 +20,7 @@ export class UpdateOneQueryFactory { ) {} async create<Record extends IRecord = IRecord>( - args: UpdateOneResolverArgs<Record>, + args: UpdateOneResolverArgs<Partial<Record>>, options: WorkspaceQueryBuilderOptions, ) { const fieldsString = await this.fieldsStringFactory.create( @@ -35,6 +35,7 @@ export class UpdateOneQueryFactory { const argsData = { ...computedArgs.data, + id: undefined, // do not allow updating an existing object's id updatedAt: new Date().toISOString(), }; diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/record.interface.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/record.interface.ts index b9847baed1da..32ee60f567c6 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/record.interface.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/interfaces/record.interface.ts @@ -16,9 +16,9 @@ export enum OrderByDirection { DescNullsLast = 'DescNullsLast', } -export type RecordOrderBy = { +export type RecordOrderBy = Array<{ [Property in keyof Record]?: OrderByDirection; -}; +}>; export interface RecordDuplicateCriteria { objectName: string; diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.factory.ts index a563be43c4bd..fd49160eb5ab 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.factory.ts @@ -64,37 +64,27 @@ export class WorkspaceQueryBuilderFactory { return this.findOneQueryFactory.create<Filter>(args, options); } - findDuplicates<Filter extends RecordFilter = RecordFilter>( - args: FindDuplicatesResolverArgs<Filter>, + findDuplicates( + args: FindDuplicatesResolverArgs, options: WorkspaceQueryBuilderOptions, - existingRecord?: Record<string, unknown>, + existingRecords?: IRecord[], ): Promise<string> { - return this.findDuplicatesQueryFactory.create<Filter>( + return this.findDuplicatesQueryFactory.create( args, options, - existingRecord, - ); - } - - findDuplicatesExistingRecord( - id: string | number, - options: WorkspaceQueryBuilderOptions, - ): string { - return this.findDuplicatesQueryFactory.buildQueryForExistingRecord( - id, - options, + existingRecords, ); } createMany<Record extends IRecord = IRecord>( - args: CreateManyResolverArgs<Record>, + args: CreateManyResolverArgs<Partial<Record>>, options: WorkspaceQueryBuilderOptions, ): Promise<string> { return this.createManyQueryFactory.create<Record>(args, options); } updateOne<Record extends IRecord = IRecord>( - initialArgs: UpdateOneResolverArgs<Record>, + initialArgs: UpdateOneResolverArgs<Partial<Record>>, options: WorkspaceQueryBuilderOptions, ): Promise<string> { return this.updateOneQueryFactory.create<Record>(initialArgs, options); @@ -111,7 +101,7 @@ export class WorkspaceQueryBuilderFactory { Record extends IRecord = IRecord, Filter extends RecordFilter = RecordFilter, >( - args: UpdateManyResolverArgs<Record, Filter>, + args: UpdateManyResolverArgs<Partial<Record>, Filter>, options: UpdateManyQueryFactoryOptions, ): Promise<string> { return this.updateManyQueryFactory.create(args, options); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.module.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.module.ts index 46367dd98e71..0db5486c63e7 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.module.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-builder/workspace-query-builder.module.ts @@ -3,13 +3,14 @@ import { Module } from '@nestjs/common'; import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; import { FieldsStringFactory } from 'src/engine/api/graphql/workspace-query-builder/factories/fields-string.factory'; import { RecordPositionQueryFactory } from 'src/engine/api/graphql/workspace-query-builder/factories/record-position-query.factory'; +import { DuplicateModule } from 'src/engine/core-modules/duplicate/duplicate.module'; import { WorkspaceQueryBuilderFactory } from './workspace-query-builder.factory'; import { workspaceQueryBuilderFactories } from './factories/factories'; @Module({ - imports: [ObjectMetadataModule], + imports: [ObjectMetadataModule, DuplicateModule], providers: [...workspaceQueryBuilderFactories, WorkspaceQueryBuilderFactory], exports: [ WorkspaceQueryBuilderFactory, diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/commands/0-20-record-position-backfill.command.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/commands/0-20-record-position-backfill.command.ts index b5709a895481..ddc2f06a4dbf 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/commands/0-20-record-position-backfill.command.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/commands/0-20-record-position-backfill.command.ts @@ -1,11 +1,10 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner, Option } from 'nest-commander'; import { RecordPositionBackfillJob, RecordPositionBackfillJobData, } from 'src/engine/api/graphql/workspace-query-runner/jobs/record-position-backfill.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; @@ -20,7 +19,7 @@ export type RecordPositionBackfillCommandOptions = { }) export class RecordPositionBackfillCommand extends CommandRunner { constructor( - @Inject(MessageQueue.recordPositionBackfillQueue) + @InjectMessageQueue(MessageQueue.recordPositionBackfillQueue) private readonly messageQueueService: MessageQueueService, ) { super(); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/__tests__/query-runner-args.factory.spec.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/__tests__/query-runner-args.factory.spec.ts index 71e26b268146..962c8fcc9d95 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/__tests__/query-runner-args.factory.spec.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/__tests__/query-runner-args.factory.spec.ts @@ -152,8 +152,8 @@ describe('QueryRunnerArgsFactory', () => { } as WorkspaceQueryRunnerOptions; const args = { - id: '123', - data: { testNumber: '1', otherField: 'test' }, + ids: ['123'], + data: [{ testNumber: '1', otherField: 'test' }], }; const result = await factory.create( @@ -163,8 +163,8 @@ describe('QueryRunnerArgsFactory', () => { ); expect(result).toEqual({ - id: 123, - data: { testNumber: 1, otherField: 'test' }, + ids: [123], + data: [{ testNumber: 1, position: 2, otherField: 'test' }], }); }); }); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/query-runner-args.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/query-runner-args.factory.ts index 4a07ee358c5d..d8299e1cbfef 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/query-runner-args.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/factories/query-runner-args.factory.ts @@ -10,7 +10,10 @@ import { ResolverArgs, ResolverArgsType, } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; -import { RecordFilter } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; +import { + Record, + RecordFilter, +} from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { hasPositionField } from 'src/engine/metadata-modules/object-metadata/utils/has-position-field.util'; @@ -47,12 +50,12 @@ export class QueryRunnerArgsFactory { return { ...args, data: await Promise.all( - (args as CreateManyResolverArgs).data.map((arg, index) => + (args as CreateManyResolverArgs).data?.map((arg, index) => this.overrideDataByFieldMetadata(arg, options, fieldMetadataMap, { argIndex: index, shouldBackfillPosition, }), - ), + ) ?? [], ), } satisfies CreateManyResolverArgs; case ResolverArgsType.FindOne: @@ -75,25 +78,27 @@ export class QueryRunnerArgsFactory { case ResolverArgsType.FindDuplicates: return { ...args, - id: await this.overrideValueByFieldMetadata( - 'id', - (args as FindDuplicatesResolverArgs).id, - fieldMetadataMap, - ), - data: await this.overrideDataByFieldMetadata( - (args as FindDuplicatesResolverArgs).data, - options, - fieldMetadataMap, - { shouldBackfillPosition: false }, + ids: (await Promise.all( + (args as FindDuplicatesResolverArgs).ids?.map((id) => + this.overrideValueByFieldMetadata('id', id, fieldMetadataMap), + ) ?? [], + )) as string[], + data: await Promise.all( + (args as FindDuplicatesResolverArgs).data?.map((arg, index) => + this.overrideDataByFieldMetadata(arg, options, fieldMetadataMap, { + argIndex: index, + shouldBackfillPosition, + }), + ) ?? [], ), - }; + } satisfies FindDuplicatesResolverArgs; default: return args; } } private async overrideDataByFieldMetadata( - data: Record<string, any> | undefined, + data: Partial<Record> | undefined, options: WorkspaceQueryRunnerOptions, fieldMetadataMap: Map<string, FieldMetadataInterface>, argPositionBackfillInput: ArgPositionBackfillInput, @@ -199,14 +204,20 @@ export class QueryRunnerArgsFactory { if (!fieldMetadata) { return value; } + switch (fieldMetadata.type) { - case 'NUMBER': - return Object.fromEntries( - Object.entries(value).map(([filterKey, filterValue]) => [ - filterKey, - Number(filterValue), - ]), - ); + case 'NUMBER': { + if (value?.is === 'NULL') { + return value; + } else { + return Object.fromEntries( + Object.entries(value).map(([filterKey, filterValue]) => [ + filterKey, + Number(filterValue), + ]), + ); + } + } default: return value; } diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook-jobs.job.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook-jobs.job.ts index e3b7e68a1982..818be2a60d65 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook-jobs.job.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook-jobs.job.ts @@ -1,6 +1,5 @@ -import { Inject, Injectable, Logger } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; @@ -11,6 +10,9 @@ import { CallWebhookJob, CallWebhookJobData, } from 'src/engine/api/graphql/workspace-query-runner/jobs/call-webhook.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; export enum CallWebhookJobsJobOperation { create = 'create', @@ -25,19 +27,18 @@ export type CallWebhookJobsJobData = { operation: CallWebhookJobsJobOperation; }; -@Injectable() -export class CallWebhookJobsJob - implements MessageQueueJob<CallWebhookJobsJobData> -{ +@Processor(MessageQueue.webhookQueue) +export class CallWebhookJobsJob { private readonly logger = new Logger(CallWebhookJobsJob.name); constructor( private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly dataSourceService: DataSourceService, - @Inject(MessageQueue.webhookQueue) + @InjectMessageQueue(MessageQueue.webhookQueue) private readonly messageQueueService: MessageQueueService, ) {} + @Process(CallWebhookJobsJob.name) async handle(data: CallWebhookJobsJobData): Promise<void> { const dataSourceMetadata = await this.dataSourceService.getLastDataSourceMetadataFromWorkspaceIdOrFail( diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook.job.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook.job.ts index 5ce75b717f4b..d18686e96aba 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook.job.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/call-webhook.job.ts @@ -1,7 +1,9 @@ -import { Injectable, Logger } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; import { HttpService } from '@nestjs/axios'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; export type CallWebhookJobData = { targetUrl: string; @@ -13,12 +15,13 @@ export type CallWebhookJobData = { record: any; }; -@Injectable() -export class CallWebhookJob implements MessageQueueJob<CallWebhookJobData> { +@Processor(MessageQueue.webhookQueue) +export class CallWebhookJob { private readonly logger = new Logger(CallWebhookJob.name); constructor(private readonly httpService: HttpService) {} + @Process(CallWebhookJob.name) async handle(data: CallWebhookJobData): Promise<void> { try { await this.httpService.axiosRef.post(data.targetUrl, data); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/record-position-backfill.job.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/record-position-backfill.job.ts index e0c589d9bdff..8619f1182814 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/record-position-backfill.job.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/record-position-backfill.job.ts @@ -1,22 +1,20 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { RecordPositionBackfillService } from 'src/engine/api/graphql/workspace-query-runner/services/record-position-backfill-service'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; export type RecordPositionBackfillJobData = { workspaceId: string; dryRun: boolean; }; -@Injectable() -export class RecordPositionBackfillJob - implements MessageQueueJob<RecordPositionBackfillJobData> -{ +@Processor(MessageQueue.recordPositionBackfillQueue) +export class RecordPositionBackfillJob { constructor( private readonly recordPositionBackfillService: RecordPositionBackfillService, ) {} + @Process(RecordPositionBackfillJob.name) async handle(data: RecordPositionBackfillJobData): Promise<void> { this.recordPositionBackfillService.backfill(data.workspaceId, data.dryRun); } diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/workspace-query-runner-job.module.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/workspace-query-runner-job.module.ts index fc4b7c6d52cb..8104cf075cf3 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/workspace-query-runner-job.module.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/jobs/workspace-query-runner-job.module.ts @@ -15,19 +15,6 @@ import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/works RecordPositionBackfillModule, HttpModule, ], - providers: [ - { - provide: CallWebhookJobsJob.name, - useClass: CallWebhookJobsJob, - }, - { - provide: CallWebhookJob.name, - useClass: CallWebhookJob, - }, - { - provide: RecordPositionBackfillJob.name, - useClass: RecordPositionBackfillJob, - }, - ], + providers: [CallWebhookJobsJob, CallWebhookJob, RecordPositionBackfillJob], }) export class WorkspaceQueryRunnerJobModule {} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/listeners/entity-events-to-db.listener.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/listeners/entity-events-to-db.listener.ts index 755173ba63e1..d9a018f86ceb 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/listeners/entity-events-to-db.listener.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/listeners/entity-events-to-db.listener.ts @@ -1,19 +1,20 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; -import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; -import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; -import { objectRecordChangedValues } from 'src/engine/integrations/event-emitter/utils/object-record-changed-values'; import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; import { ObjectRecordBaseEvent } from 'src/engine/integrations/event-emitter/types/object-record.base.event'; -import { UpsertTimelineActivityFromInternalEvent } from 'src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job'; +import { objectRecordChangedValues } from 'src/engine/integrations/event-emitter/utils/object-record-changed-values'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { CreateAuditLogFromInternalEvent } from 'src/modules/timeline/jobs/create-audit-log-from-internal-event'; +import { UpsertTimelineActivityFromInternalEvent } from 'src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job'; @Injectable() export class EntityEventsToDbListener { constructor( - @Inject(MessageQueue.entityEventsToDbQueue) + @InjectMessageQueue(MessageQueue.entityEventsToDbQueue) private readonly messageQueueService: MessageQueueService, ) {} @@ -40,7 +41,7 @@ export class EntityEventsToDbListener { // .... private async handle(payload: ObjectRecordBaseEvent) { - if (!payload.objectMetadata.isAuditLogged) { + if (!payload.objectMetadata?.isAuditLogged) { return; } diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.config.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.config.ts deleted file mode 100644 index 518e147241a1..000000000000 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.config.ts +++ /dev/null @@ -1,31 +0,0 @@ -import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/types/workspace-query-hook.type'; -import { CalendarEventFindManyPreQueryHook } from 'src/modules/calendar/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook'; -import { CalendarEventFindOnePreQueryHook } from 'src/modules/calendar/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook'; -import { BlocklistCreateManyPreQueryHook } from 'src/modules/connected-account/query-hooks/blocklist/blocklist-create-many.pre-query.hook'; -import { BlocklistUpdateManyPreQueryHook } from 'src/modules/connected-account/query-hooks/blocklist/blocklist-update-many.pre-query.hook'; -import { BlocklistUpdateOnePreQueryHook } from 'src/modules/connected-account/query-hooks/blocklist/blocklist-update-one.pre-query.hook'; -import { WorkspaceMemberDeleteOnePreQueryHook } from 'src/modules/workspace-member/query-hooks/workspace-member-delete-one.pre-query.hook'; -import { WorkspaceMemberDeleteManyPreQueryHook } from 'src/modules/workspace-member/query-hooks/workspace-member-delete-many.pre-query.hook'; -import { MessageFindManyPreQueryHook } from 'src/modules/messaging/common/query-hooks/message/message-find-many.pre-query.hook'; -import { MessageFindOnePreQueryHook } from 'src/modules/messaging/common/query-hooks/message/message-find-one.pre-query-hook'; - -// TODO: move to a decorator -export const workspacePreQueryHooks: WorkspaceQueryHook = { - message: { - findOne: [MessageFindOnePreQueryHook.name], - findMany: [MessageFindManyPreQueryHook.name], - }, - calendarEvent: { - findOne: [CalendarEventFindOnePreQueryHook.name], - findMany: [CalendarEventFindManyPreQueryHook.name], - }, - blocklist: { - createMany: [BlocklistCreateManyPreQueryHook.name], - updateMany: [BlocklistUpdateManyPreQueryHook.name], - updateOne: [BlocklistUpdateOnePreQueryHook.name], - }, - workspaceMember: { - deleteOne: [WorkspaceMemberDeleteOnePreQueryHook.name], - deleteMany: [WorkspaceMemberDeleteManyPreQueryHook.name], - }, -}; diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.module.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.module.ts deleted file mode 100644 index 8ea6b6a6e11b..000000000000 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.module.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { WorkspacePreQueryHookService } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.service'; -import { CalendarQueryHookModule } from 'src/modules/calendar/query-hooks/calendar-query-hook.module'; -import { ConnectedAccountQueryHookModule } from 'src/modules/connected-account/query-hooks/connected-account-query-hook.module'; -import { MessagingQueryHookModule } from 'src/modules/messaging/common/query-hooks/messaging-query-hook.module'; -import { WorkspaceMemberQueryHookModule } from 'src/modules/workspace-member/query-hooks/workspace-member-query-hook.module'; - -@Module({ - imports: [ - MessagingQueryHookModule, - CalendarQueryHookModule, - ConnectedAccountQueryHookModule, - WorkspaceMemberQueryHookModule, - ], - providers: [WorkspacePreQueryHookService], - exports: [WorkspacePreQueryHookService], -}) -export class WorkspacePreQueryHookModule {} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.service.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.service.ts deleted file mode 100644 index 28681e0b965b..000000000000 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.service.ts +++ /dev/null @@ -1,34 +0,0 @@ -import { Injectable } from '@nestjs/common'; -import { ModuleRef } from '@nestjs/core'; - -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; - -import { - ExecutePreHookMethod, - WorkspacePreQueryHookPayload, -} from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/types/workspace-query-hook.type'; -import { workspacePreQueryHooks } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.config'; - -@Injectable() -export class WorkspacePreQueryHookService { - constructor(private readonly workspaceQueryHookModuleRef: ModuleRef) {} - - public async executePreHooks<T extends ExecutePreHookMethod>( - userId: string | undefined, - workspaceId: string, - objectName: string, - method: T, - payload: WorkspacePreQueryHookPayload<T>, - ): Promise<void> { - const hooks = workspacePreQueryHooks[objectName] || []; - - for (const hookName of Object.values(hooks[method] ?? [])) { - const hook: WorkspacePreQueryHook = - await this.workspaceQueryHookModuleRef.get(hookName, { - strict: false, - }); - - await hook.execute(userId, workspaceId, payload); - } - } -} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator.ts new file mode 100644 index 000000000000..d87979418638 --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator.ts @@ -0,0 +1,40 @@ +import { Scope, SetMetadata } from '@nestjs/common'; +import { SCOPE_OPTIONS_METADATA } from '@nestjs/common/constants'; + +import { WorkspaceResolverBuilderMethodNames } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; + +import { WORKSPACE_QUERY_HOOK_METADATA } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.constants'; +import { WorkspaceQueryHookType } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/types/workspace-query-hook.type'; + +export type WorkspaceQueryHookKey = + `${string}.${WorkspaceResolverBuilderMethodNames}`; + +export interface WorkspaceQueryHookOptions { + key: WorkspaceQueryHookKey; + type?: WorkspaceQueryHookType; + scope?: Scope; +} + +export function WorkspaceQueryHook(key: WorkspaceQueryHookKey): ClassDecorator; +export function WorkspaceQueryHook( + options: WorkspaceQueryHookOptions, +): ClassDecorator; +export function WorkspaceQueryHook( + keyOrOptions: WorkspaceQueryHookKey | WorkspaceQueryHookOptions, +): ClassDecorator { + const options: WorkspaceQueryHookOptions = + keyOrOptions && typeof keyOrOptions === 'object' + ? keyOrOptions + : { key: keyOrOptions }; + + // Default to PreHook + if (!options.type) { + options.type = WorkspaceQueryHookType.PreHook; + } + + // eslint-disable-next-line @typescript-eslint/ban-types + return (target: Function) => { + SetMetadata(SCOPE_OPTIONS_METADATA, options)(target); + SetMetadata(WORKSPACE_QUERY_HOOK_METADATA, options)(target); + }; +} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface.ts similarity index 84% rename from packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface.ts rename to packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface.ts index 60a782e8c443..93c3964c83b5 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface.ts @@ -1,6 +1,6 @@ import { ResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; -export interface WorkspacePreQueryHook { +export interface WorkspaceQueryHookInstance { execute( userId: string | undefined, workspaceId: string, diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/storage/workspace-query-hook.storage.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/storage/workspace-query-hook.storage.ts new file mode 100644 index 000000000000..18b25d1d964d --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/storage/workspace-query-hook.storage.ts @@ -0,0 +1,59 @@ +// hook-registry.service.ts +import { Injectable } from '@nestjs/common'; +import { Module } from '@nestjs/core/injector/module'; + +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; + +import { WorkspaceQueryHookKey } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; + +interface WorkspaceQueryHookData<T> { + instance: T; + host: Module; + isRequestScoped: boolean; +} + +@Injectable() +export class WorkspaceQueryHookStorage { + private preHookInstances = new Map< + WorkspaceQueryHookKey, + WorkspaceQueryHookData<WorkspaceQueryHookInstance>[] + >(); + private postHookInstances = new Map< + WorkspaceQueryHookKey, + WorkspaceQueryHookData<WorkspaceQueryHookInstance>[] + >(); + + registerWorkspaceQueryPreHookInstance( + key: WorkspaceQueryHookKey, + data: WorkspaceQueryHookData<WorkspaceQueryHookInstance>, + ) { + if (!this.preHookInstances.has(key)) { + this.preHookInstances.set(key, []); + } + + this.preHookInstances.get(key)?.push(data); + } + + getWorkspaceQueryPreHookInstances( + key: WorkspaceQueryHookKey, + ): WorkspaceQueryHookData<WorkspaceQueryHookInstance>[] | undefined { + return this.preHookInstances.get(key); + } + + registerWorkspaceQueryPostHookInstance( + key: WorkspaceQueryHookKey, + data: WorkspaceQueryHookData<WorkspaceQueryHookInstance>, + ) { + if (!this.postHookInstances.has(key)) { + this.postHookInstances.set(key, []); + } + + this.postHookInstances.get(key)?.push(data); + } + + getWorkspaceQueryPostHookInstances( + key: WorkspaceQueryHookKey, + ): WorkspaceQueryHookData<WorkspaceQueryHookInstance>[] | undefined { + return this.postHookInstances.get(key); + } +} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/types/workspace-query-hook.type.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/types/workspace-query-hook.type.ts similarity index 73% rename from packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/types/workspace-query-hook.type.ts rename to packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/types/workspace-query-hook.type.ts index 84d75b1358b9..40ec1df6b4d4 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/types/workspace-query-hook.type.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/types/workspace-query-hook.type.ts @@ -10,26 +10,10 @@ import { UpdateOneResolverArgs, } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; -export type ExecutePreHookMethod = - | 'createMany' - | 'createOne' - | 'deleteMany' - | 'deleteOne' - | 'findMany' - | 'findOne' - | 'findDuplicates' - | 'updateMany' - | 'updateOne'; - -export type ObjectName = string; - -export type HookName = string; - -export type WorkspaceQueryHook = { - [key in ObjectName]: { - [key in ExecutePreHookMethod]?: HookName[]; - }; -}; +export enum WorkspaceQueryHookType { + PreHook = 'PreHook', + PostHook = 'PostHook', +} export type WorkspacePreQueryHookPayload<T> = T extends 'createMany' ? CreateManyResolverArgs diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook-metadata.accessor.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook-metadata.accessor.ts new file mode 100644 index 000000000000..229e0b0bf3db --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook-metadata.accessor.ts @@ -0,0 +1,25 @@ +/* eslint-disable @typescript-eslint/ban-types */ +import { Injectable, Type } from '@nestjs/common'; +import { Reflector } from '@nestjs/core'; + +import { WORKSPACE_QUERY_HOOK_METADATA } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.constants'; +import { WorkspaceQueryHookOptions } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; + +@Injectable() +export class WorkspaceQueryHookMetadataAccessor { + constructor(private readonly reflector: Reflector) {} + + isWorkspaceQueryHook(target: Type<any> | Function): boolean { + if (!target) { + return false; + } + + return !!this.reflector.get(WORKSPACE_QUERY_HOOK_METADATA, target); + } + + getWorkspaceQueryHookMetadata( + target: Type<any> | Function, + ): WorkspaceQueryHookOptions | undefined { + return this.reflector.get(WORKSPACE_QUERY_HOOK_METADATA, target); + } +} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.constants.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.constants.ts new file mode 100644 index 000000000000..ea1aadc49006 --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.constants.ts @@ -0,0 +1,3 @@ +export const WORKSPACE_QUERY_HOOK_METADATA = Symbol( + 'workspace-query-hook:query-hook-metadata', +); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.explorer.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.explorer.ts new file mode 100644 index 000000000000..87bce8248bd1 --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.explorer.ts @@ -0,0 +1,139 @@ +import { Injectable, Logger, OnModuleInit } from '@nestjs/common'; +import { DiscoveryService, ModuleRef, createContextId } from '@nestjs/core'; +import { Module } from '@nestjs/core/injector/module'; +import { Injector } from '@nestjs/core/injector/injector'; + +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; + +import { WorkspaceQueryHookMetadataAccessor } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook-metadata.accessor'; +import { WorkspaceQueryHookStorage } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/storage/workspace-query-hook.storage'; +import { WorkspaceQueryHookKey } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; +import { WorkspaceQueryHookType } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/types/workspace-query-hook.type'; + +@Injectable() +export class WorkspaceQueryHookExplorer implements OnModuleInit { + private readonly logger = new Logger('WorkspaceQueryHookModule'); + private readonly injector = new Injector(); + + constructor( + private readonly moduleRef: ModuleRef, + private readonly discoveryService: DiscoveryService, + private readonly metadataAccessor: WorkspaceQueryHookMetadataAccessor, + private readonly workspaceQueryHookStorage: WorkspaceQueryHookStorage, + ) {} + + onModuleInit() { + this.explore(); + } + + async explore() { + const hooks = this.discoveryService + .getProviders() + .filter((wrapper) => + this.metadataAccessor.isWorkspaceQueryHook( + !wrapper.metatype || wrapper.inject + ? wrapper.instance?.constructor + : wrapper.metatype, + ), + ); + + for (const hook of hooks) { + const { instance, metatype } = hook; + + const { key, type } = + this.metadataAccessor.getWorkspaceQueryHookMetadata( + instance.constructor || metatype, + ) ?? {}; + + if (!key || !type) { + this.logger.error( + `PreHook ${hook.name} is missing key or type metadata`, + ); + continue; + } + + if (!hook.host) { + this.logger.error(`PreHook ${hook.name} is missing host metadata`); + + continue; + } + + this.registerWorkspaceQueryHook( + key, + type, + instance, + hook.host, + !hook.isDependencyTreeStatic(), + ); + } + } + + async handleHook( + payload: Parameters<WorkspaceQueryHookInstance['execute']>, + instance: object, + host: Module, + isRequestScoped: boolean, + ) { + const methodName = 'execute'; + + if (isRequestScoped) { + const contextId = createContextId(); + + if (this.moduleRef.registerRequestByContextId) { + this.moduleRef.registerRequestByContextId( + { + req: { + workspaceId: payload?.[1], + }, + }, + contextId, + ); + } + + const contextInstance = await this.injector.loadPerContext( + instance, + host, + host.providers, + contextId, + ); + + await contextInstance[methodName].call(contextInstance, ...payload); + } else { + await instance[methodName].call(instance, ...payload); + } + } + + private registerWorkspaceQueryHook( + key: WorkspaceQueryHookKey, + type: WorkspaceQueryHookType, + instance: object, + host: Module, + isRequestScoped: boolean, + ) { + switch (type) { + case WorkspaceQueryHookType.PreHook: + this.workspaceQueryHookStorage.registerWorkspaceQueryPreHookInstance( + key, + { + instance: instance as WorkspaceQueryHookInstance, + host, + isRequestScoped, + }, + ); + break; + case WorkspaceQueryHookType.PostHook: + this.workspaceQueryHookStorage.registerWorkspaceQueryPostHookInstance( + key, + { + instance: instance as WorkspaceQueryHookInstance, + host, + isRequestScoped, + }, + ); + break; + default: + this.logger.error(`Unknown WorkspaceQueryHookType: ${type}`); + break; + } + } +} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.module.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.module.ts new file mode 100644 index 000000000000..04e48b8c9da7 --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.module.ts @@ -0,0 +1,31 @@ +import { Module } from '@nestjs/common'; +import { DiscoveryModule } from '@nestjs/core'; + +import { WorkspaceQueryHookStorage } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/storage/workspace-query-hook.storage'; +import { WorkspaceQueryHookMetadataAccessor } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook-metadata.accessor'; +import { WorkspaceQueryHookExplorer } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.explorer'; +import { WorkspaceQueryHookService } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.service'; +import { BlocklistQueryHookModule } from 'src/modules/blocklist/query-hooks/blocklist-query-hook.module'; +import { CalendarQueryHookModule } from 'src/modules/calendar/common/query-hooks/calendar-query-hook.module'; +import { ConnectedAccountQueryHookModule } from 'src/modules/connected-account/query-hooks/connected-account-query-hook.module'; +import { MessagingQueryHookModule } from 'src/modules/messaging/common/query-hooks/messaging-query-hook.module'; +import { WorkspaceMemberQueryHookModule } from 'src/modules/workspace-member/query-hooks/workspace-member-query-hook.module'; + +@Module({ + imports: [ + MessagingQueryHookModule, + CalendarQueryHookModule, + ConnectedAccountQueryHookModule, + BlocklistQueryHookModule, + WorkspaceMemberQueryHookModule, + DiscoveryModule, + ], + providers: [ + WorkspaceQueryHookService, + WorkspaceQueryHookExplorer, + WorkspaceQueryHookMetadataAccessor, + WorkspaceQueryHookStorage, + ], + exports: [WorkspaceQueryHookService], +}) +export class WorkspaceQueryHookModule {} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.service.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.service.ts new file mode 100644 index 000000000000..adfd90f4261d --- /dev/null +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.service.ts @@ -0,0 +1,43 @@ +import { Injectable } from '@nestjs/common'; + +import { WorkspaceResolverBuilderMethodNames } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; + +import { WorkspaceQueryHookStorage } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/storage/workspace-query-hook.storage'; +import { WorkspaceQueryHookKey } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; +import { WorkspaceQueryHookExplorer } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.explorer'; +import { WorkspacePreQueryHookPayload } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/types/workspace-query-hook.type'; + +@Injectable() +export class WorkspaceQueryHookService { + constructor( + private readonly workspaceQueryHookStorage: WorkspaceQueryHookStorage, + private readonly workspaceQueryHookExplorer: WorkspaceQueryHookExplorer, + ) {} + + public async executePreQueryHooks< + T extends WorkspaceResolverBuilderMethodNames, + >( + userId: string | undefined, + workspaceId: string, + objectName: string, + methodName: T, + payload: WorkspacePreQueryHookPayload<T>, + ): Promise<void> { + const key: WorkspaceQueryHookKey = `${objectName}.${methodName}`; + const preHookInstances = + this.workspaceQueryHookStorage.getWorkspaceQueryPreHookInstances(key); + + if (!preHookInstances) { + return; + } + + for (const preHookInstance of preHookInstances) { + await this.workspaceQueryHookExplorer.handleHook( + [userId, workspaceId, payload], + preHookInstance.instance, + preHookInstance.host, + preHookInstance.isRequestScoped, + ); + } + } +} diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.module.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.module.ts index f7358021b849..99c9b7a6e3ba 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.module.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.module.ts @@ -2,7 +2,7 @@ import { Module } from '@nestjs/common'; import { WorkspaceQueryBuilderModule } from 'src/engine/api/graphql/workspace-query-builder/workspace-query-builder.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { WorkspacePreQueryHookModule } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.module'; +import { WorkspaceQueryHookModule } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.module'; import { workspaceQueryRunnerFactories } from 'src/engine/api/graphql/workspace-query-runner/factories'; import { AuthModule } from 'src/engine/core-modules/auth/auth.module'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; @@ -10,6 +10,7 @@ import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repos import { TelemetryListener } from 'src/engine/api/graphql/workspace-query-runner/listeners/telemetry.listener'; import { AnalyticsModule } from 'src/engine/core-modules/analytics/analytics.module'; import { RecordPositionBackfillCommand } from 'src/engine/api/graphql/workspace-query-runner/commands/0-20-record-position-backfill.command'; +import { DuplicateModule } from 'src/engine/core-modules/duplicate/duplicate.module'; import { WorkspaceQueryRunnerService } from './workspace-query-runner.service'; @@ -20,9 +21,10 @@ import { EntityEventsToDbListener } from './listeners/entity-events-to-db.listen AuthModule, WorkspaceQueryBuilderModule, WorkspaceDataSourceModule, - WorkspacePreQueryHookModule, + WorkspaceQueryHookModule, ObjectMetadataRepositoryModule.forFeature([WorkspaceMemberWorkspaceEntity]), AnalyticsModule, + DuplicateModule, ], providers: [ WorkspaceQueryRunnerService, diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.service.ts b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.service.ts index a1cab54beeba..1ab7764f5549 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.service.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-query-runner/workspace-query-runner.service.ts @@ -1,6 +1,5 @@ import { BadRequestException, - Inject, Injectable, Logger, RequestTimeoutException, @@ -8,13 +7,14 @@ import { import { EventEmitter2 } from '@nestjs/event-emitter'; import isEmpty from 'lodash.isempty'; +import { DataSource } from 'typeorm'; -import { IConnection } from 'src/engine/api/graphql/workspace-query-runner/interfaces/connection.interface'; import { Record as IRecord, RecordFilter, RecordOrderBy, } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; +import { IConnection } from 'src/engine/api/graphql/workspace-query-runner/interfaces/connection.interface'; import { CreateManyResolverArgs, CreateOneResolverArgs, @@ -30,34 +30,36 @@ import { import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; import { WorkspaceQueryBuilderFactory } from 'src/engine/api/graphql/workspace-query-builder/workspace-query-builder.factory'; -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { QueryResultGettersFactory } from 'src/engine/api/graphql/workspace-query-runner/factories/query-result-getters.factory'; +import { QueryRunnerArgsFactory } from 'src/engine/api/graphql/workspace-query-runner/factories/query-runner-args.factory'; import { CallWebhookJobsJob, CallWebhookJobsJobData, CallWebhookJobsJobOperation, } from 'src/engine/api/graphql/workspace-query-runner/jobs/call-webhook-jobs.job'; +import { assertIsValidUuid } from 'src/engine/api/graphql/workspace-query-runner/utils/assert-is-valid-uuid.util'; import { parseResult } from 'src/engine/api/graphql/workspace-query-runner/utils/parse-result.util'; -import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; -import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; +import { WorkspaceQueryHookService } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/workspace-query-hook.service'; +import { DuplicateService } from 'src/engine/core-modules/duplicate/duplicate.service'; +import { NotFoundError } from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; +import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; -import { WorkspacePreQueryHookService } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/workspace-pre-query-hook.service'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { NotFoundError } from 'src/engine/utils/graphql-errors.util'; -import { QueryRunnerArgsFactory } from 'src/engine/api/graphql/workspace-query-runner/factories/query-runner-args.factory'; -import { QueryResultGettersFactory } from 'src/engine/api/graphql/workspace-query-runner/factories/query-result-getters.factory'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { assertMutationNotOnRemoteObject } from 'src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util'; -import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { assertIsValidUuid } from 'src/engine/api/graphql/workspace-query-runner/utils/assert-is-valid-uuid.util'; +import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; import { isQueryTimeoutError } from 'src/engine/utils/query-timeout.util'; +import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; +import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { WorkspaceQueryRunnerOptions } from './interfaces/query-runner-option.interface'; import { PGGraphQLMutation, PGGraphQLResult, } from './interfaces/pg-graphql.interface'; +import { WorkspaceQueryRunnerOptions } from './interfaces/query-runner-option.interface'; import { PgGraphQLConfig, computePgGraphQLError, @@ -72,11 +74,12 @@ export class WorkspaceQueryRunnerService { private readonly workspaceDataSourceService: WorkspaceDataSourceService, private readonly queryRunnerArgsFactory: QueryRunnerArgsFactory, private readonly queryResultGettersFactory: QueryResultGettersFactory, - @Inject(MessageQueue.webhookQueue) + @InjectMessageQueue(MessageQueue.webhookQueue) private readonly messageQueueService: MessageQueueService, private readonly eventEmitter: EventEmitter2, - private readonly workspacePreQueryHookService: WorkspacePreQueryHookService, + private readonly workspaceQueryHookService: WorkspaceQueryHookService, private readonly environmentService: EnvironmentService, + private readonly duplicateService: DuplicateService, ) {} async findMany< @@ -101,7 +104,7 @@ export class WorkspaceQueryRunnerService { options, ); - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -148,7 +151,7 @@ export class WorkspaceQueryRunnerService { options, ); - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -167,16 +170,16 @@ export class WorkspaceQueryRunnerService { } async findDuplicates<TRecord extends IRecord = IRecord>( - args: FindDuplicatesResolverArgs<TRecord>, + args: FindDuplicatesResolverArgs<Partial<TRecord>>, options: WorkspaceQueryRunnerOptions, ): Promise<IConnection<TRecord> | undefined> { - if (!args.data && !args.id) { + if (!args.data && !args.ids) { throw new BadRequestException( 'You have to provide either "data" or "id" argument', ); } - if (!args.id && isEmpty(args.data)) { + if (!args.ids && isEmpty(args.data)) { throw new BadRequestException( 'The "data" condition can not be empty when ID input not provided', ); @@ -190,40 +193,27 @@ export class WorkspaceQueryRunnerService { ResolverArgsType.FindDuplicates, )) as FindDuplicatesResolverArgs<TRecord>; - let existingRecord: Record<string, unknown> | undefined; - - if (computedArgs.id) { - const existingRecordQuery = - this.workspaceQueryBuilderFactory.findDuplicatesExistingRecord( - computedArgs.id, - options, - ); - - const existingRecordResult = await this.execute( - existingRecordQuery, - workspaceId, - ); + let existingRecords: IRecord[] | undefined = undefined; - const parsedResult = await this.parseResult<Record<string, unknown>>( - existingRecordResult, + if (computedArgs.ids && computedArgs.ids.length > 0) { + existingRecords = await this.duplicateService.findExistingRecords( + computedArgs.ids, objectMetadataItem, - '', + workspaceId, ); - existingRecord = parsedResult?.edges?.[0]?.node; - - if (!existingRecord) { - throw new NotFoundError(`Object with id ${args.id} not found`); + if (!existingRecords || existingRecords.length === 0) { + throw new NotFoundError(`Object with id ${args.ids} not found`); } } const query = await this.workspaceQueryBuilderFactory.findDuplicates( computedArgs, options, - existingRecord, + existingRecords, ); - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -237,17 +227,22 @@ export class WorkspaceQueryRunnerService { result, objectMetadataItem, '', + true, ); } async createMany<Record extends IRecord = IRecord>( - args: CreateManyResolverArgs<Record>, + args: CreateManyResolverArgs<Partial<Record>>, options: WorkspaceQueryRunnerOptions, ): Promise<Record[] | undefined> { const { workspaceId, userId, objectMetadataItem } = options; assertMutationNotOnRemoteObject(objectMetadataItem); + if (args.upsert) { + return await this.upsertMany(args, options); + } + args.data.forEach((record) => { if (record?.id) { assertIsValidUuid(record.id); @@ -260,7 +255,7 @@ export class WorkspaceQueryRunnerService { ResolverArgsType.CreateMany, )) as CreateManyResolverArgs<Record>; - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -305,17 +300,73 @@ export class WorkspaceQueryRunnerService { return parsedResults; } + async upsertMany<Record extends IRecord = IRecord>( + args: CreateManyResolverArgs<Partial<Record>>, + options: WorkspaceQueryRunnerOptions, + ): Promise<Record[] | undefined> { + const ids = args.data + .map((item) => item.id) + .filter((id) => id !== undefined); + + const existingRecords = + ids.length > 0 + ? await this.duplicateService.findExistingRecords( + ids as string[], + options.objectMetadataItem, + options.workspaceId, + ) + : []; + + const existingRecordsMap = new Map( + existingRecords.map((record) => [record.id, record]), + ); + + const results: Record[] = []; + const recordsToCreate: Partial<Record>[] = []; + + for (const payload of args.data) { + if (payload.id && existingRecordsMap.has(payload.id)) { + const result = await this.updateOne( + { id: payload.id, data: payload }, + options, + ); + + if (result) { + results.push(result); + } + } else { + recordsToCreate.push(payload); + } + } + + if (recordsToCreate.length > 0) { + const createResults = await this.createMany( + { data: recordsToCreate } as CreateManyResolverArgs<Partial<Record>>, + options, + ); + + if (createResults) { + results.push(...createResults); + } + } + + return results; + } + async createOne<Record extends IRecord = IRecord>( - args: CreateOneResolverArgs<Record>, + args: CreateOneResolverArgs<Partial<Record>>, options: WorkspaceQueryRunnerOptions, ): Promise<Record | undefined> { - const results = await this.createMany({ data: [args.data] }, options); + const results = await this.createMany( + { data: [args.data], upsert: args.upsert }, + options, + ); return results?.[0]; } async updateOne<Record extends IRecord = IRecord>( - args: UpdateOneResolverArgs<Record>, + args: UpdateOneResolverArgs<Partial<Record>>, options: WorkspaceQueryRunnerOptions, ): Promise<Record | undefined> { const { workspaceId, userId, objectMetadataItem } = options; @@ -333,7 +384,7 @@ export class WorkspaceQueryRunnerService { options, ); - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -373,7 +424,7 @@ export class WorkspaceQueryRunnerService { } async updateMany<Record extends IRecord = IRecord>( - args: UpdateManyResolverArgs<Record>, + args: UpdateManyResolverArgs<Partial<Record>>, options: WorkspaceQueryRunnerOptions, ): Promise<Record[] | undefined> { const { userId, workspaceId, objectMetadataItem } = options; @@ -382,14 +433,14 @@ export class WorkspaceQueryRunnerService { args.filter?.id?.in?.forEach((id) => assertIsValidUuid(id)); const maximumRecordAffected = this.environmentService.get( - 'MUTATION_MAXIMUM_RECORD_AFFECTED', + 'MUTATION_MAXIMUM_AFFECTED_RECORDS', ); const query = await this.workspaceQueryBuilderFactory.updateMany(args, { ...options, atMost: maximumRecordAffected, }); - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -434,14 +485,14 @@ export class WorkspaceQueryRunnerService { assertMutationNotOnRemoteObject(objectMetadataItem); const maximumRecordAffected = this.environmentService.get( - 'MUTATION_MAXIMUM_RECORD_AFFECTED', + 'MUTATION_MAXIMUM_AFFECTED_RECORDS', ); const query = await this.workspaceQueryBuilderFactory.deleteMany(args, { ...options, atMost: maximumRecordAffected, }); - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -509,7 +560,7 @@ export class WorkspaceQueryRunnerService { ); // TODO END - await this.workspacePreQueryHookService.executePreHooks( + await this.workspaceQueryHookService.executePreQueryHooks( userId, workspaceId, objectMetadataItem.nameSingular, @@ -570,15 +621,12 @@ export class WorkspaceQueryRunnerService { return sanitizedRecord; } - async execute( - query: string, + async executeSQL( + workspaceDataSource: DataSource, workspaceId: string, - ): Promise<PGGraphQLResult | undefined> { - const workspaceDataSource = - await this.workspaceDataSourceService.connectToWorkspaceDataSource( - workspaceId, - ); - + sqlQuery: string, + parameters?: any[], + ) { try { return await workspaceDataSource?.transaction( async (transactionManager) => { @@ -588,10 +636,7 @@ export class WorkspaceQueryRunnerService { )}; `); - const results = transactionManager.query<PGGraphQLResult>( - `SELECT graphql.resolve($1);`, - [query], - ); + const results = transactionManager.query(sqlQuery, parameters); return results; }, @@ -605,15 +650,42 @@ export class WorkspaceQueryRunnerService { } } + async execute( + query: string, + workspaceId: string, + ): Promise<PGGraphQLResult | undefined> { + const workspaceDataSource = + await this.workspaceDataSourceService.connectToWorkspaceDataSource( + workspaceId, + ); + + return this.executeSQL( + workspaceDataSource, + workspaceId, + `SELECT graphql.resolve($1);`, + [query], + ); + } + private async parseResult<Result>( graphqlResult: PGGraphQLResult | undefined, objectMetadataItem: ObjectMetadataInterface, command: string, + isMultiQuery = false, ): Promise<Result> { const entityKey = `${command}${computeObjectTargetTable( objectMetadataItem, )}Collection`; - const result = graphqlResult?.[0]?.resolve?.data?.[entityKey]; + const result = !isMultiQuery + ? graphqlResult?.[0]?.resolve?.data?.[entityKey] + : Object.keys(graphqlResult?.[0]?.resolve?.data).reduce( + (acc: IRecord[], dataItem, index) => { + acc.push(graphqlResult?.[0]?.resolve?.data[`${entityKey}${index}`]); + + return acc; + }, + [], + ); const errors = graphqlResult?.[0]?.resolve?.errors; if ( @@ -631,7 +703,7 @@ export class WorkspaceQueryRunnerService { errors, { atMost: this.environmentService.get( - 'MUTATION_MAXIMUM_RECORD_AFFECTED', + 'MUTATION_MAXIMUM_AFFECTED_RECORDS', ), } satisfies PgGraphQLConfig, ); diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface.ts b/packages/twenty-server/src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface.ts index aa31be0bf763..a2db909472bc 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface.ts @@ -39,26 +39,36 @@ export interface FindOneResolverArgs<Filter = any> { filter?: Filter; } -export interface FindDuplicatesResolverArgs<Data extends Record = Record> { - id?: string; - data?: Data; +export interface FindDuplicatesResolverArgs< + Data extends Partial<Record> = Partial<Record>, +> { + ids?: string[]; + data?: Data[]; } -export interface CreateOneResolverArgs<Data extends Record = Record> { +export interface CreateOneResolverArgs< + Data extends Partial<Record> = Partial<Record>, +> { data: Data; + upsert?: boolean; } -export interface CreateManyResolverArgs<Data extends Record = Record> { +export interface CreateManyResolverArgs< + Data extends Partial<Record> = Partial<Record>, +> { data: Data[]; + upsert?: boolean; } -export interface UpdateOneResolverArgs<Data extends Record = Record> { +export interface UpdateOneResolverArgs< + Data extends Partial<Record> = Partial<Record>, +> { id: string; data: Data; } export interface UpdateManyResolverArgs< - Data extends Record = Record, + Data extends Partial<Record> = Partial<Record>, Filter = any, > { filter: Filter; diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/factories/root-type.factory.ts b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/factories/root-type.factory.ts index 66a031420ab8..780cf2985af7 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/factories/root-type.factory.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/factories/root-type.factory.ts @@ -102,9 +102,12 @@ export class RootTypeFactory { } const outputType = this.typeMapperService.mapToGqlType(objectType, { - isArray: ['updateMany', 'deleteMany', 'createMany'].includes( - methodName, - ), + isArray: [ + 'updateMany', + 'deleteMany', + 'createMany', + 'findDuplicates', + ].includes(methodName), }); fieldConfigMap[name] = { diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/services/type-mapper.service.ts b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/services/type-mapper.service.ts index cb04e530021f..531939cf4c7b 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/services/type-mapper.service.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/services/type-mapper.service.ts @@ -73,7 +73,6 @@ export class TypeMapperService { ), ], [FieldMetadataType.NUMERIC, BigFloatScalarType], - [FieldMetadataType.PROBABILITY, GraphQLFloat], [FieldMetadataType.POSITION, PositionScalarType], [FieldMetadataType.RAW_JSON, RawJSONScalar], ]); @@ -109,7 +108,6 @@ export class TypeMapperService { ), ], [FieldMetadataType.NUMERIC, BigFloatFilterType], - [FieldMetadataType.PROBABILITY, FloatFilterType], [FieldMetadataType.POSITION, FloatFilterType], [FieldMetadataType.RAW_JSON, RawJsonFilterType], ]); @@ -130,7 +128,6 @@ export class TypeMapperService { [FieldMetadataType.BOOLEAN, OrderByDirectionType], [FieldMetadataType.NUMBER, OrderByDirectionType], [FieldMetadataType.NUMERIC, OrderByDirectionType], - [FieldMetadataType.PROBABILITY, OrderByDirectionType], [FieldMetadataType.RATING, OrderByDirectionType], [FieldMetadataType.SELECT, OrderByDirectionType], [FieldMetadataType.MULTI_SELECT, OrderByDirectionType], diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/__tests__/get-resolver-args.spec.ts b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/__tests__/get-resolver-args.spec.ts index b66087fe89f2..26652d04a69a 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/__tests__/get-resolver-args.spec.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/__tests__/get-resolver-args.spec.ts @@ -1,4 +1,4 @@ -import { GraphQLID, GraphQLInt, GraphQLString } from 'graphql'; +import { GraphQLBoolean, GraphQLID, GraphQLInt, GraphQLString } from 'graphql'; import { WorkspaceResolverBuilderMethodNames } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; @@ -13,7 +13,11 @@ describe('getResolverArgs', () => { before: { type: GraphQLString, isNullable: true }, after: { type: GraphQLString, isNullable: true }, filter: { kind: InputTypeDefinitionKind.Filter, isNullable: true }, - orderBy: { kind: InputTypeDefinitionKind.OrderBy, isNullable: true }, + orderBy: { + kind: InputTypeDefinitionKind.OrderBy, + isNullable: true, + isArray: true, + }, limit: { type: GraphQLInt, isNullable: true }, }, findOne: { @@ -25,9 +29,19 @@ describe('getResolverArgs', () => { isNullable: false, isArray: true, }, + upsert: { + isArray: false, + isNullable: true, + type: GraphQLBoolean, + }, }, createOne: { data: { kind: InputTypeDefinitionKind.Create, isNullable: false }, + upsert: { + isArray: false, + isNullable: true, + type: GraphQLBoolean, + }, }, updateOne: { id: { type: GraphQLID, isNullable: false }, diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/get-resolver-args.util.ts b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/get-resolver-args.util.ts index 5eaa8f314ce1..609f268fd81f 100644 --- a/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/get-resolver-args.util.ts +++ b/packages/twenty-server/src/engine/api/graphql/workspace-schema-builder/utils/get-resolver-args.util.ts @@ -1,4 +1,4 @@ -import { GraphQLString, GraphQLInt, GraphQLID } from 'graphql'; +import { GraphQLString, GraphQLInt, GraphQLID, GraphQLBoolean } from 'graphql'; import { WorkspaceResolverBuilderMethodNames } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; import { ArgMetadata } from 'src/engine/api/graphql/workspace-schema-builder/interfaces/param-metadata.interface'; @@ -38,6 +38,7 @@ export const getResolverArgs = ( orderBy: { kind: InputTypeDefinitionKind.OrderBy, isNullable: true, + isArray: true, }, }; case 'findOne': @@ -55,6 +56,11 @@ export const getResolverArgs = ( isNullable: false, isArray: true, }, + upsert: { + type: GraphQLBoolean, + isNullable: true, + isArray: false, + }, }; case 'createOne': return { @@ -62,6 +68,11 @@ export const getResolverArgs = ( kind: InputTypeDefinitionKind.Create, isNullable: false, }, + upsert: { + type: GraphQLBoolean, + isNullable: true, + isArray: false, + }, }; case 'updateOne': return { @@ -76,13 +87,15 @@ export const getResolverArgs = ( }; case 'findDuplicates': return { - id: { + ids: { type: GraphQLID, isNullable: true, + isArray: true, }, data: { kind: InputTypeDefinitionKind.Create, isNullable: true, + isArray: true, }, }; case 'deleteOne': diff --git a/packages/twenty-server/src/engine/api/rest/controllers/rest-api-core-batch.controller.ts b/packages/twenty-server/src/engine/api/rest/core/controllers/rest-api-core-batch.controller.ts similarity index 86% rename from packages/twenty-server/src/engine/api/rest/controllers/rest-api-core-batch.controller.ts rename to packages/twenty-server/src/engine/api/rest/core/controllers/rest-api-core-batch.controller.ts index 6f35ed834d29..e77ae4c8912f 100644 --- a/packages/twenty-server/src/engine/api/rest/controllers/rest-api-core-batch.controller.ts +++ b/packages/twenty-server/src/engine/api/rest/core/controllers/rest-api-core-batch.controller.ts @@ -2,7 +2,7 @@ import { Controller, Post, Req, Res } from '@nestjs/common'; import { Request, Response } from 'express'; -import { RestApiCoreService } from 'src/engine/api/rest/services/rest-api-core.service'; +import { RestApiCoreService } from 'src/engine/api/rest/core/rest-api-core.service'; import { cleanGraphQLResponse } from 'src/engine/api/rest/utils/clean-graphql-response.utils'; @Controller('rest/batch/*') diff --git a/packages/twenty-server/src/engine/api/rest/controllers/rest-api-core.controller.ts b/packages/twenty-server/src/engine/api/rest/core/controllers/rest-api-core.controller.ts similarity index 94% rename from packages/twenty-server/src/engine/api/rest/controllers/rest-api-core.controller.ts rename to packages/twenty-server/src/engine/api/rest/core/controllers/rest-api-core.controller.ts index fe2dfacaeaf0..5fd46c49a269 100644 --- a/packages/twenty-server/src/engine/api/rest/controllers/rest-api-core.controller.ts +++ b/packages/twenty-server/src/engine/api/rest/core/controllers/rest-api-core.controller.ts @@ -11,7 +11,7 @@ import { import { Request, Response } from 'express'; -import { RestApiCoreService } from 'src/engine/api/rest/services/rest-api-core.service'; +import { RestApiCoreService } from 'src/engine/api/rest/core/rest-api-core.service'; import { cleanGraphQLResponse } from 'src/engine/api/rest/utils/clean-graphql-response.utils'; @Controller('rest/*') diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/core-query-builder.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/core-query-builder.factory.ts similarity index 76% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/core-query-builder.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/core-query-builder.factory.ts index d7f0d6d185da..bd2bda31d9cf 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/core-query-builder.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/core-query-builder.factory.ts @@ -2,24 +2,24 @@ import { BadRequestException, Injectable } from '@nestjs/common'; import { Request } from 'express'; -import { DeleteQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/delete-query.factory'; +import { DeleteQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/delete-query.factory'; import { ObjectMetadataService } from 'src/engine/metadata-modules/object-metadata/object-metadata.service'; import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; -import { CreateOneQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-one-query.factory'; -import { UpdateQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/update-query.factory'; -import { FindOneQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/find-one-query.factory'; -import { FindManyQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/find-many-query.factory'; -import { DeleteVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/delete-variables.factory'; -import { CreateVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-variables.factory'; -import { UpdateVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/update-variables.factory'; -import { GetVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/get-variables.factory'; -import { parseCorePath } from 'src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-path.utils'; -import { computeDepth } from 'src/engine/api/rest/rest-api-core-query-builder/utils/compute-depth.utils'; +import { CreateOneQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/create-one-query.factory'; +import { UpdateQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/update-query.factory'; +import { FindOneQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/find-one-query.factory'; +import { FindManyQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/find-many-query.factory'; +import { DeleteVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/delete-variables.factory'; +import { CreateVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/create-variables.factory'; +import { UpdateVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/update-variables.factory'; +import { GetVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/get-variables.factory'; +import { parseCorePath } from 'src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-path.utils'; +import { computeDepth } from 'src/engine/api/rest/core/query-builder/utils/compute-depth.utils'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; -import { Query } from 'src/engine/api/rest/types/query.type'; +import { Query } from 'src/engine/api/rest/core/types/query.type'; import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { CreateManyQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-many-query.factory'; -import { parseCoreBatchPath } from 'src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-batch-path.utils'; +import { CreateManyQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/create-many-query.factory'; +import { parseCoreBatchPath } from 'src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-batch-path.utils'; @Injectable() export class CoreQueryBuilderFactory { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/core-query-builder.module.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/core-query-builder.module.ts similarity index 64% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/core-query-builder.module.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/core-query-builder.module.ts index 8e91f11742d4..38b5ec398c34 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/core-query-builder.module.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/core-query-builder.module.ts @@ -1,7 +1,7 @@ import { Module } from '@nestjs/common'; -import { CoreQueryBuilderFactory } from 'src/engine/api/rest/rest-api-core-query-builder/core-query-builder.factory'; -import { coreQueryBuilderFactories } from 'src/engine/api/rest/rest-api-core-query-builder/factories/factories'; +import { CoreQueryBuilderFactory } from 'src/engine/api/rest/core/query-builder/core-query-builder.factory'; +import { coreQueryBuilderFactories } from 'src/engine/api/rest/core/query-builder/factories/factories'; import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; import { AuthModule } from 'src/engine/core-modules/auth/auth.module'; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-many-query.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-many-query.factory.ts similarity index 92% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-many-query.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-many-query.factory.ts index 02c8714f44e6..b02bf194bbcc 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-many-query.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-many-query.factory.ts @@ -1,7 +1,7 @@ import { Injectable } from '@nestjs/common'; import { capitalize } from 'src/utils/capitalize'; -import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils'; +import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils'; @Injectable() export class CreateManyQueryFactory { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-one-query.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-one-query.factory.ts similarity index 91% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-one-query.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-one-query.factory.ts index a7135605eb41..e8078898935d 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-one-query.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-one-query.factory.ts @@ -1,7 +1,7 @@ import { Injectable } from '@nestjs/common'; import { capitalize } from 'src/utils/capitalize'; -import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils'; +import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils'; @Injectable() export class CreateOneQueryFactory { diff --git a/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-variables.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-variables.factory.ts new file mode 100644 index 000000000000..cb907164a258 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/create-variables.factory.ts @@ -0,0 +1,14 @@ +import { Injectable } from '@nestjs/common'; + +import { Request } from 'express'; + +import { QueryVariables } from 'src/engine/api/rest/core/types/query-variables.type'; + +@Injectable() +export class CreateVariablesFactory { + create(request: Request): QueryVariables { + return { + data: request.body, + }; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/delete-query.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/delete-query.factory.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/delete-query.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/delete-query.factory.ts diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/delete-variables.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/delete-variables.factory.ts similarity index 67% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/delete-variables.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/delete-variables.factory.ts index c41c2073c2df..a1a5a40bd221 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/delete-variables.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/delete-variables.factory.ts @@ -1,6 +1,6 @@ import { Injectable } from '@nestjs/common'; -import { QueryVariables } from 'src/engine/api/rest/types/query-variables.type'; +import { QueryVariables } from 'src/engine/api/rest/core/types/query-variables.type'; @Injectable() export class DeleteVariablesFactory { diff --git a/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/factories.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/factories.ts new file mode 100644 index 000000000000..00faf41ca5e9 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/factories.ts @@ -0,0 +1,25 @@ +import { DeleteQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/delete-query.factory'; +import { CreateOneQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/create-one-query.factory'; +import { UpdateQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/update-query.factory'; +import { FindOneQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/find-one-query.factory'; +import { FindManyQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/find-many-query.factory'; +import { DeleteVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/delete-variables.factory'; +import { CreateVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/create-variables.factory'; +import { UpdateVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/update-variables.factory'; +import { GetVariablesFactory } from 'src/engine/api/rest/core/query-builder/factories/get-variables.factory'; +import { CreateManyQueryFactory } from 'src/engine/api/rest/core/query-builder/factories/create-many-query.factory'; +import { inputFactories } from 'src/engine/api/rest/input-factories/factories'; + +export const coreQueryBuilderFactories = [ + DeleteQueryFactory, + CreateOneQueryFactory, + CreateManyQueryFactory, + UpdateQueryFactory, + FindOneQueryFactory, + FindManyQueryFactory, + DeleteVariablesFactory, + CreateVariablesFactory, + UpdateVariablesFactory, + GetVariablesFactory, + ...inputFactories, +]; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/find-many-query.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/find-many-query.factory.ts similarity index 78% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/find-many-query.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/find-many-query.factory.ts index bd6bd1e458c1..401721162d83 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/find-many-query.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/find-many-query.factory.ts @@ -1,7 +1,7 @@ import { Injectable } from '@nestjs/common'; import { capitalize } from 'src/utils/capitalize'; -import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils'; +import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils'; @Injectable() export class FindManyQueryFactory { @@ -14,13 +14,19 @@ export class FindManyQueryFactory { return ` query FindMany${capitalize(objectNamePlural)}( $filter: ${objectNameSingular}FilterInput, - $orderBy: ${objectNameSingular}OrderByInput, + $orderBy: [${objectNameSingular}OrderByInput], $startingAfter: String, $endingBefore: String, - $limit: Int = 60 + $first: Int, + $last: Int ) { ${objectNamePlural}( - filter: $filter, orderBy: $orderBy, first: $limit, after: $startingAfter, before: $endingBefore + filter: $filter, + orderBy: $orderBy, + first: $first, + last: $last, + after: $startingAfter, + before: $endingBefore ) { edges { node { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/find-one-query.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/find-one-query.factory.ts similarity index 91% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/find-one-query.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/find-one-query.factory.ts index 9286080be123..cc2b498710bc 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/find-one-query.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/find-one-query.factory.ts @@ -1,7 +1,7 @@ import { Injectable } from '@nestjs/common'; import { capitalize } from 'src/utils/capitalize'; -import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils'; +import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils'; @Injectable() export class FindOneQueryFactory { diff --git a/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/get-variables.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/get-variables.factory.ts new file mode 100644 index 000000000000..5af552d48f70 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/get-variables.factory.ts @@ -0,0 +1,44 @@ +import { Injectable } from '@nestjs/common'; + +import { Request } from 'express'; + +import { LimitInputFactory } from 'src/engine/api/rest/input-factories/limit-input.factory'; +import { OrderByInputFactory } from 'src/engine/api/rest/input-factories/order-by-input.factory'; +import { FilterInputFactory } from 'src/engine/api/rest/input-factories/filter-input.factory'; +import { QueryVariables } from 'src/engine/api/rest/core/types/query-variables.type'; +import { EndingBeforeInputFactory } from 'src/engine/api/rest/input-factories/ending-before-input.factory'; +import { StartingAfterInputFactory } from 'src/engine/api/rest/input-factories/starting-after-input.factory'; + +@Injectable() +export class GetVariablesFactory { + constructor( + private readonly startingAfterInputFactory: StartingAfterInputFactory, + private readonly endingBeforeInputFactory: EndingBeforeInputFactory, + private readonly limitInputFactory: LimitInputFactory, + private readonly orderByInputFactory: OrderByInputFactory, + private readonly filterInputFactory: FilterInputFactory, + ) {} + + create( + id: string | undefined, + request: Request, + objectMetadata, + ): QueryVariables { + if (id) { + return { filter: { id: { eq: id } } }; + } + + const limit = this.limitInputFactory.create(request); + const endingBefore = this.endingBeforeInputFactory.create(request); + const startingAfter = this.startingAfterInputFactory.create(request); + + return { + filter: this.filterInputFactory.create(request, objectMetadata), + orderBy: this.orderByInputFactory.create(request, objectMetadata), + first: !endingBefore ? limit : undefined, + last: endingBefore ? limit : undefined, + startingAfter, + endingBefore, + }; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/update-query.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/update-query.factory.ts similarity index 91% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/update-query.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/update-query.factory.ts index ca7e9cd03d3d..f578b2d18e49 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/update-query.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/update-query.factory.ts @@ -1,7 +1,7 @@ import { Injectable } from '@nestjs/common'; import { capitalize } from 'src/utils/capitalize'; -import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils'; +import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils'; @Injectable() export class UpdateQueryFactory { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/update-variables.factory.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/update-variables.factory.ts similarity index 74% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/update-variables.factory.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/factories/update-variables.factory.ts index 588288bf92d1..c911faa03a0b 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/update-variables.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/factories/update-variables.factory.ts @@ -2,7 +2,7 @@ import { Injectable } from '@nestjs/common'; import { Request } from 'express'; -import { QueryVariables } from 'src/engine/api/rest/types/query-variables.type'; +import { QueryVariables } from 'src/engine/api/rest/core/types/query-variables.type'; @Injectable() export class UpdateVariablesFactory { diff --git a/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/check-fields.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/check-fields.utils.spec.ts new file mode 100644 index 000000000000..865eac4bc2b8 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/check-fields.utils.spec.ts @@ -0,0 +1,34 @@ +import { objectMetadataItemMock } from 'src/engine/api/__mocks__/object-metadata-item.mock'; +import { checkFields } from 'src/engine/api/rest/core/query-builder/utils/check-fields.utils'; +import { checkArrayFields } from 'src/engine/api/rest/core/query-builder/utils/check-order-by.utils'; + +describe('checkFields', () => { + it('should check field types', () => { + expect(() => + checkFields(objectMetadataItemMock, ['fieldNumber']), + ).not.toThrow(); + + expect(() => checkFields(objectMetadataItemMock, ['wrongField'])).toThrow(); + + expect(() => + checkFields(objectMetadataItemMock, ['fieldNumber', 'wrongField']), + ).toThrow(); + }); + + it('should check field types from array of fields', () => { + expect(() => + checkArrayFields(objectMetadataItemMock, [{ fieldNumber: undefined }]), + ).not.toThrow(); + + expect(() => + checkArrayFields(objectMetadataItemMock, [{ wrongField: undefined }]), + ).toThrow(); + + expect(() => + checkArrayFields(objectMetadataItemMock, [ + { fieldNumber: undefined }, + { wrongField: undefined }, + ]), + ).toThrow(); + }); +}); diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/compute-depth.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/compute-depth.utils.spec.ts similarity index 54% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/compute-depth.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/compute-depth.utils.spec.ts index a6909574a6cc..7a2f51f18ad0 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/compute-depth.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/compute-depth.utils.spec.ts @@ -1,12 +1,14 @@ -import { computeDepth } from 'src/engine/api/rest/rest-api-core-query-builder/utils/compute-depth.utils'; +import { computeDepth } from 'src/engine/api/rest/core/query-builder/utils/compute-depth.utils'; describe('computeDepth', () => { - it('should compute depth from query', () => { - const request: any = { - query: { depth: '1' }, - }; + [0, 1, 2].forEach((depth) => { + it('should compute depth from query', () => { + const request: any = { + query: { depth: `${depth}` }, + }; - expect(computeDepth(request)).toEqual(1); + expect(computeDepth(request)).toEqual(depth); + }); }); it('should return default depth if missing', () => { @@ -19,7 +21,7 @@ describe('computeDepth', () => { expect(() => computeDepth(request)).toThrow(); - request.query.depth = '0'; + request.query.depth = '-1'; expect(() => computeDepth(request)).toThrow(); }); diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/get-field-type.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/get-field-type.utils.spec.ts similarity index 78% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/get-field-type.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/get-field-type.utils.spec.ts index 6c09709c205f..ec6854bc482f 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/get-field-type.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/get-field-type.utils.spec.ts @@ -1,6 +1,6 @@ import { objectMetadataItemMock } from 'src/engine/api/__mocks__/object-metadata-item.mock'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { getFieldType } from 'src/engine/api/rest/rest-api-core-query-builder/utils/get-field-type.utils'; +import { getFieldType } from 'src/engine/api/rest/core/query-builder/utils/get-field-type.utils'; describe('getFieldType', () => { it('should get field type', () => { diff --git a/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/map-field-metadata-to-graphql-query.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/map-field-metadata-to-graphql-query.utils.spec.ts new file mode 100644 index 000000000000..473c6f8408d8 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/__tests__/map-field-metadata-to-graphql-query.utils.spec.ts @@ -0,0 +1,59 @@ +import { + fieldCurrencyMock, + fieldLinkMock, + fieldNumberMock, + fieldTextMock, + objectMetadataItemMock, +} from 'src/engine/api/__mocks__/object-metadata-item.mock'; +import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils'; +import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; + +describe('mapFieldMetadataToGraphqlQuery', () => { + it('should map properly', () => { + expect( + mapFieldMetadataToGraphqlQuery([objectMetadataItemMock], fieldNumberMock), + ).toEqual('fieldNumber'); + expect( + mapFieldMetadataToGraphqlQuery([objectMetadataItemMock], fieldTextMock), + ).toEqual('fieldText'); + expect( + mapFieldMetadataToGraphqlQuery([objectMetadataItemMock], fieldLinkMock), + ).toEqual(` + fieldLink + { + label + url + } + `); + expect( + mapFieldMetadataToGraphqlQuery( + [objectMetadataItemMock], + fieldCurrencyMock, + ), + ).toEqual(` + fieldCurrency + { + amountMicros + currencyCode + } + `); + }); + describe('should handle all field metadata types', () => { + Object.values(FieldMetadataType).forEach((fieldMetadataType) => { + it(`with field type ${fieldMetadataType}`, () => { + const field = { + type: fieldMetadataType, + name: 'toObjectMetadataName', + fromRelationMetadata: { + relationType: RelationMetadataType.ONE_TO_MANY, + }, + }; + + expect( + mapFieldMetadataToGraphqlQuery([objectMetadataItemMock], field), + ).toBeDefined(); + }); + }); + }); +}); diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/check-fields.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/check-fields.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/check-fields.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/check-fields.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/check-order-by.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/check-order-by.utils.ts new file mode 100644 index 000000000000..ce14a02bd276 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/check-order-by.utils.ts @@ -0,0 +1,48 @@ +import { BadRequestException } from '@nestjs/common'; + +import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; +import { Record } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; + +import { compositeTypeDefintions } from 'src/engine/metadata-modules/field-metadata/composite-types'; +import { isCompositeFieldMetadataType } from 'src/engine/metadata-modules/field-metadata/utils/is-composite-field-metadata-type.util'; +import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; + +export const checkArrayFields = ( + objectMetadata: ObjectMetadataInterface, + fields: Array<Partial<Record>>, +): void => { + const fieldMetadataNames = objectMetadata.fields + .map((field) => { + if (isCompositeFieldMetadataType(field.type)) { + const compositeType = compositeTypeDefintions.get(field.type); + + if (!compositeType) { + throw new BadRequestException( + `Composite type '${field.type}' not found`, + ); + } + + return [ + field.name, + compositeType.properties.map( + (compositeProperty) => compositeProperty.name, + ), + ].flat(); + } + + return field.name; + }) + .flat(); + + for (const fieldObj of fields) { + for (const fieldName in fieldObj) { + if (!fieldMetadataNames.includes(fieldName)) { + throw new BadRequestException( + `field '${fieldName}' does not exist in '${computeObjectTargetTable( + objectMetadata, + )}' object`, + ); + } + } + } +}; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/compute-depth.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/compute-depth.utils.ts similarity index 93% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/compute-depth.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/compute-depth.utils.ts index 6fdee1b30d9f..ecd5060cfb45 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/compute-depth.utils.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/compute-depth.utils.ts @@ -2,7 +2,7 @@ import { BadRequestException } from '@nestjs/common'; import { Request } from 'express'; -const ALLOWED_DEPTH_VALUES = [1, 2]; +const ALLOWED_DEPTH_VALUES = [0, 1, 2]; export const computeDepth = (request: Request): number | undefined => { if (!request.query.depth) { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/add-default-conjunction.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/add-default-conjunction.utils.spec.ts similarity index 81% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/add-default-conjunction.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/add-default-conjunction.utils.spec.ts index 8adda483599e..1b949c855194 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/add-default-conjunction.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/add-default-conjunction.utils.spec.ts @@ -1,4 +1,4 @@ -import { addDefaultConjunctionIfMissing } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/add-default-conjunction.utils'; +import { addDefaultConjunctionIfMissing } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/add-default-conjunction.utils'; describe('addDefaultConjunctionIfMissing', () => { it('should add default conjunction if missing', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/check-filter-enum-values.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/check-filter-enum-values.spec.ts similarity index 84% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/check-filter-enum-values.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/check-filter-enum-values.spec.ts index 4e4694c11019..8c24bce85846 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/check-filter-enum-values.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/check-filter-enum-values.spec.ts @@ -1,4 +1,4 @@ -import { checkFilterEnumValues } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-enum-values'; +import { checkFilterEnumValues } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-enum-values'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { fieldSelectMock, diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/check-filter-query.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/check-filter-query.utils.spec.ts similarity index 86% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/check-filter-query.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/check-filter-query.utils.spec.ts index fc1f3e81870b..7cba92df6d0e 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/check-filter-query.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/check-filter-query.utils.spec.ts @@ -1,4 +1,4 @@ -import { checkFilterQuery } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-query.utils'; +import { checkFilterQuery } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-query.utils'; describe('checkFilterQuery', () => { it('should check filter query', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/format-field-values.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/format-field-values.utils.spec.ts similarity index 92% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/format-field-values.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/format-field-values.utils.spec.ts index eb94f88b310d..6787ccc3eec8 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/format-field-values.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/format-field-values.utils.spec.ts @@ -1,5 +1,5 @@ import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { formatFieldValue } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/format-field-values.utils'; +import { formatFieldValue } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/format-field-values.utils'; describe('formatFieldValue', () => { it('should format fieldNumber value', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-base-filter.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-base-filter.utils.spec.ts similarity index 90% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-base-filter.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-base-filter.utils.spec.ts index c522b00bde8d..dc41296b7ac7 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-base-filter.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-base-filter.utils.spec.ts @@ -1,4 +1,4 @@ -import { parseBaseFilter } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-base-filter.utils'; +import { parseBaseFilter } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-base-filter.utils'; describe('parseBaseFilter', () => { it('should parse simple filter string test 1', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-filter-content.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-filter-content.utils.spec.ts similarity index 91% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-filter-content.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-filter-content.utils.spec.ts index ed0efea1c25b..884c672395d8 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-filter-content.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-filter-content.utils.spec.ts @@ -1,4 +1,4 @@ -import { parseFilterContent } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter-content.utils'; +import { parseFilterContent } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter-content.utils'; describe('parseFilterContent', () => { it('should parse query filter test 1', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-filter.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-filter.utils.spec.ts similarity index 94% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-filter.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-filter.utils.spec.ts index 3dfdd2b75b89..b92b09c96b31 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/__tests__/parse-filter.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/__tests__/parse-filter.utils.spec.ts @@ -1,5 +1,5 @@ import { objectMetadataItemMock } from 'src/engine/api/__mocks__/object-metadata-item.mock'; -import { parseFilter } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils'; +import { parseFilter } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils'; describe('parseFilter', () => { it('should parse string filter test 1', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/add-default-conjunction.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/add-default-conjunction.utils.ts similarity index 67% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/add-default-conjunction.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/add-default-conjunction.utils.ts index 03f66456e802..f30bf375d2e9 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/add-default-conjunction.utils.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/add-default-conjunction.utils.ts @@ -1,4 +1,4 @@ -import { Conjunctions } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils'; +import { Conjunctions } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils'; export const DEFAULT_CONJUNCTION = Conjunctions.and; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-enum-values.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-enum-values.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-enum-values.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-enum-values.ts diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-query.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-query.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-query.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-query.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/format-field-values.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/format-field-values.utils.ts similarity index 93% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/format-field-values.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/format-field-values.utils.ts index eeab31d6745e..79e351f94136 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/format-field-values.utils.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/format-field-values.utils.ts @@ -1,7 +1,7 @@ import { BadRequestException } from '@nestjs/common'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { FieldValue } from 'src/engine/api/rest/types/field-value.type'; +import { FieldValue } from 'src/engine/api/rest/core/types/field-value.type'; export const formatFieldValue = ( value: string, diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-base-filter.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/parse-base-filter.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-base-filter.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/parse-base-filter.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter-content.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter-content.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter-content.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter-content.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils.ts similarity index 65% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils.ts index ac8307d873c3..0f8a043b3bf6 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils.ts @@ -2,13 +2,13 @@ import { BadRequestException } from '@nestjs/common'; import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; -import { parseFilterContent } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter-content.utils'; -import { parseBaseFilter } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-base-filter.utils'; -import { checkFields } from 'src/engine/api/rest/rest-api-core-query-builder/utils/check-fields.utils'; -import { formatFieldValue } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/format-field-values.utils'; -import { FieldValue } from 'src/engine/api/rest/types/field-value.type'; -import { checkFilterEnumValues } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-enum-values'; -import { getFieldType } from 'src/engine/api/rest/rest-api-core-query-builder/utils/get-field-type.utils'; +import { parseFilterContent } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter-content.utils'; +import { parseBaseFilter } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-base-filter.utils'; +import { checkFields } from 'src/engine/api/rest/core/query-builder/utils/check-fields.utils'; +import { formatFieldValue } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/format-field-values.utils'; +import { FieldValue } from 'src/engine/api/rest/core/types/field-value.type'; +import { checkFilterEnumValues } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-enum-values'; +import { getFieldType } from 'src/engine/api/rest/core/query-builder/utils/get-field-type.utils'; export enum Conjunctions { or = 'or', diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/get-field-type.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/get-field-type.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/get-field-type.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/get-field-type.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils.ts similarity index 94% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils.ts index c7112e0dfc5c..a34cc2123f36 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/map-field-metadata-to-graphql-query.utils.ts @@ -9,7 +9,7 @@ export const mapFieldMetadataToGraphqlQuery = ( field, maxDepthForRelations = DEFAULT_DEPTH_VALUE, ): string | undefined => { - if (maxDepthForRelations <= 0) { + if (maxDepthForRelations < 0) { return ''; } @@ -19,19 +19,23 @@ export const mapFieldMetadataToGraphqlQuery = ( FieldMetadataType.UUID, FieldMetadataType.TEXT, FieldMetadataType.PHONE, + FieldMetadataType.EMAIL, FieldMetadataType.DATE_TIME, FieldMetadataType.DATE, - FieldMetadataType.EMAIL, + FieldMetadataType.BOOLEAN, FieldMetadataType.NUMBER, - FieldMetadataType.SELECT, + FieldMetadataType.NUMERIC, FieldMetadataType.RATING, - FieldMetadataType.BOOLEAN, + FieldMetadataType.SELECT, + FieldMetadataType.MULTI_SELECT, FieldMetadataType.POSITION, + FieldMetadataType.RAW_JSON, ].includes(fieldType); if (fieldIsSimpleValue) { return field.name; } else if ( + maxDepthForRelations > 0 && fieldType === FieldMetadataType.RELATION && field.toRelationMetadata?.relationType === RelationMetadataType.ONE_TO_MANY ) { @@ -45,7 +49,6 @@ export const mapFieldMetadataToGraphqlQuery = ( { id ${(relationMetadataItem?.fields ?? []) - .filter((field) => field.type !== FieldMetadataType.RELATION) .map((field) => mapFieldMetadataToGraphqlQuery( objectMetadataItems, @@ -56,6 +59,7 @@ export const mapFieldMetadataToGraphqlQuery = ( .join('\n')} }`; } else if ( + maxDepthForRelations > 0 && fieldType === FieldMetadataType.RELATION && field.fromRelationMetadata?.relationType === RelationMetadataType.ONE_TO_MANY @@ -72,7 +76,6 @@ export const mapFieldMetadataToGraphqlQuery = ( node { id ${(relationMetadataItem?.fields ?? []) - .filter((field) => field.type !== FieldMetadataType.RELATION) .map((field) => mapFieldMetadataToGraphqlQuery( objectMetadataItems, diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-core-batch-path.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/__tests__/parse-core-batch-path.utils.spec.ts similarity index 65% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-core-batch-path.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/__tests__/parse-core-batch-path.utils.spec.ts index 04131521108f..a66400eecb00 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-core-batch-path.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/__tests__/parse-core-batch-path.utils.spec.ts @@ -1,4 +1,4 @@ -import { parseCoreBatchPath } from 'src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-batch-path.utils'; +import { parseCoreBatchPath } from 'src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-batch-path.utils'; describe('parseCoreBatchPath', () => { it('should parse object from request path', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-core-path.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/__tests__/parse-core-path.utils.spec.ts similarity index 86% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-core-path.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/__tests__/parse-core-path.utils.spec.ts index 0d20d453afc4..c5c9434160aa 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-core-path.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/__tests__/parse-core-path.utils.spec.ts @@ -1,4 +1,4 @@ -import { parseCorePath } from 'src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-path.utils'; +import { parseCorePath } from 'src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-path.utils'; describe('parseCorePath', () => { it('should parse object from request path', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-batch-path.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-batch-path.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-batch-path.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-batch-path.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-path.utils.ts b/packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-path.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-core-path.utils.ts rename to packages/twenty-server/src/engine/api/rest/core/query-builder/utils/path-parsers/parse-core-path.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/services/rest-api-core.service.ts b/packages/twenty-server/src/engine/api/rest/core/rest-api-core.service.ts similarity index 88% rename from packages/twenty-server/src/engine/api/rest/services/rest-api-core.service.ts rename to packages/twenty-server/src/engine/api/rest/core/rest-api-core.service.ts index 126c64e6a352..3b56c76e5148 100644 --- a/packages/twenty-server/src/engine/api/rest/services/rest-api-core.service.ts +++ b/packages/twenty-server/src/engine/api/rest/core/rest-api-core.service.ts @@ -2,11 +2,11 @@ import { Injectable } from '@nestjs/common'; import { Request } from 'express'; -import { CoreQueryBuilderFactory } from 'src/engine/api/rest/rest-api-core-query-builder/core-query-builder.factory'; +import { CoreQueryBuilderFactory } from 'src/engine/api/rest/core/query-builder/core-query-builder.factory'; import { GraphqlApiType, RestApiService, -} from 'src/engine/api/rest/services/rest-api.service'; +} from 'src/engine/api/rest/rest-api.service'; @Injectable() export class RestApiCoreService { diff --git a/packages/twenty-server/src/engine/api/rest/types/field-value.type.ts b/packages/twenty-server/src/engine/api/rest/core/types/field-value.type.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/types/field-value.type.ts rename to packages/twenty-server/src/engine/api/rest/core/types/field-value.type.ts diff --git a/packages/twenty-server/src/engine/api/rest/types/query-variables.type.ts b/packages/twenty-server/src/engine/api/rest/core/types/query-variables.type.ts similarity index 83% rename from packages/twenty-server/src/engine/api/rest/types/query-variables.type.ts rename to packages/twenty-server/src/engine/api/rest/core/types/query-variables.type.ts index 6907aed8d5ec..97a23c657a48 100644 --- a/packages/twenty-server/src/engine/api/rest/types/query-variables.type.ts +++ b/packages/twenty-server/src/engine/api/rest/core/types/query-variables.type.ts @@ -3,7 +3,8 @@ export type QueryVariables = { data?: object | null; filter?: object; orderBy?: object; - limit?: number; + last?: number; + first?: number; startingAfter?: string; endingBefore?: string; input?: object; diff --git a/packages/twenty-server/src/engine/api/rest/core/types/query.type.ts b/packages/twenty-server/src/engine/api/rest/core/types/query.type.ts new file mode 100644 index 000000000000..92a9cf27ce53 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/core/types/query.type.ts @@ -0,0 +1,6 @@ +import { QueryVariables } from 'src/engine/api/rest/core/types/query-variables.type'; + +export type Query = { + query: string; + variables: QueryVariables; +}; diff --git a/packages/twenty-server/src/engine/api/rest/errors/RestApiException.ts b/packages/twenty-server/src/engine/api/rest/errors/RestApiException.ts index 11a72c026232..2cc0b571b7ea 100644 --- a/packages/twenty-server/src/engine/api/rest/errors/RestApiException.ts +++ b/packages/twenty-server/src/engine/api/rest/errors/RestApiException.ts @@ -1,6 +1,6 @@ import { BadRequestException } from '@nestjs/common'; -import { BaseGraphQLError } from 'src/engine/utils/graphql-errors.util'; +import { BaseGraphQLError } from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; const formatMessage = (message: BaseGraphQLError) => { if (message.extensions) { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/ending-before-input.factory.spec.ts b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/ending-before-input.factory.spec.ts similarity index 85% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/ending-before-input.factory.spec.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/__tests__/ending-before-input.factory.spec.ts index 2de86ff7f382..9bef0dbfdc98 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/ending-before-input.factory.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/ending-before-input.factory.spec.ts @@ -1,6 +1,6 @@ import { Test, TestingModule } from '@nestjs/testing'; -import { EndingBeforeInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/ending-before-input.factory'; +import { EndingBeforeInputFactory } from 'src/engine/api/rest/input-factories/ending-before-input.factory'; describe('EndingBeforeInputFactory', () => { let service: EndingBeforeInputFactory; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/filter-input.factory.spec.ts b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/filter-input.factory.spec.ts similarity index 96% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/filter-input.factory.spec.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/__tests__/filter-input.factory.spec.ts index 12dd66750025..b6366ff7034d 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/filter-input.factory.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/filter-input.factory.spec.ts @@ -1,7 +1,7 @@ import { Test, TestingModule } from '@nestjs/testing'; import { objectMetadataItemMock } from 'src/engine/api/__mocks__/object-metadata-item.mock'; -import { FilterInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-input.factory'; +import { FilterInputFactory } from 'src/engine/api/rest/input-factories/filter-input.factory'; describe('FilterInputFactory', () => { const objectMetadata = { objectMetadataItem: objectMetadataItemMock }; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/limit-input.factory.spec.ts b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/limit-input.factory.spec.ts similarity index 90% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/limit-input.factory.spec.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/__tests__/limit-input.factory.spec.ts index 16ed18507a0f..2104250f1cda 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/limit-input.factory.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/limit-input.factory.spec.ts @@ -1,6 +1,6 @@ import { Test, TestingModule } from '@nestjs/testing'; -import { LimitInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/limit-input.factory'; +import { LimitInputFactory } from 'src/engine/api/rest/input-factories/limit-input.factory'; describe('LimitInputFactory', () => { let service: LimitInputFactory; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/order-by-input.factory.spec.ts b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/order-by-input.factory.spec.ts similarity index 72% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/order-by-input.factory.spec.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/__tests__/order-by-input.factory.spec.ts index ddd8b3eff533..63b3179ee68d 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/order-by-input.factory.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/order-by-input.factory.spec.ts @@ -3,7 +3,7 @@ import { Test, TestingModule } from '@nestjs/testing'; import { OrderByDirection } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; import { objectMetadataItemMock } from 'src/engine/api/__mocks__/object-metadata-item.mock'; -import { OrderByInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory'; +import { OrderByInputFactory } from 'src/engine/api/rest/input-factories/order-by-input.factory'; describe('OrderByInputFactory', () => { const objectMetadata = { objectMetadataItem: objectMetadataItemMock }; @@ -26,7 +26,7 @@ describe('OrderByInputFactory', () => { it('should return default if order by missing', () => { const request: any = { query: {} }; - expect(service.create(request, objectMetadata)).toEqual({}); + expect(service.create(request, objectMetadata)).toEqual([{}]); }); it('should create order by parser properly', () => { @@ -36,10 +36,10 @@ describe('OrderByInputFactory', () => { }, }; - expect(service.create(request, objectMetadata)).toEqual({ - fieldNumber: OrderByDirection.AscNullsFirst, - fieldText: OrderByDirection.DescNullsLast, - }); + expect(service.create(request, objectMetadata)).toEqual([ + { fieldNumber: OrderByDirection.AscNullsFirst }, + { fieldText: OrderByDirection.DescNullsLast }, + ]); }); it('should choose default direction if missing', () => { @@ -49,9 +49,9 @@ describe('OrderByInputFactory', () => { }, }; - expect(service.create(request, objectMetadata)).toEqual({ - fieldNumber: OrderByDirection.AscNullsFirst, - }); + expect(service.create(request, objectMetadata)).toEqual([ + { fieldNumber: OrderByDirection.AscNullsFirst }, + ]); }); it('should handler complex fields', () => { @@ -61,9 +61,9 @@ describe('OrderByInputFactory', () => { }, }; - expect(service.create(request, objectMetadata)).toEqual({ - fieldCurrency: { amountMicros: OrderByDirection.AscNullsFirst }, - }); + expect(service.create(request, objectMetadata)).toEqual([ + { fieldCurrency: { amountMicros: OrderByDirection.AscNullsFirst } }, + ]); }); it('should handler complex fields with direction', () => { @@ -73,9 +73,9 @@ describe('OrderByInputFactory', () => { }, }; - expect(service.create(request, objectMetadata)).toEqual({ - fieldCurrency: { amountMicros: OrderByDirection.DescNullsLast }, - }); + expect(service.create(request, objectMetadata)).toEqual([ + { fieldCurrency: { amountMicros: OrderByDirection.DescNullsLast } }, + ]); }); it('should handler multiple complex fields with direction', () => { @@ -86,10 +86,10 @@ describe('OrderByInputFactory', () => { }, }; - expect(service.create(request, objectMetadata)).toEqual({ - fieldCurrency: { amountMicros: OrderByDirection.DescNullsLast }, - fieldLink: { label: OrderByDirection.AscNullsLast }, - }); + expect(service.create(request, objectMetadata)).toEqual([ + { fieldCurrency: { amountMicros: OrderByDirection.DescNullsLast } }, + { fieldLink: { label: OrderByDirection.AscNullsLast } }, + ]); }); it('should throw if direction invalid', () => { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/starting-before-input.factory.spec.ts b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/starting-before-input.factory.spec.ts similarity index 85% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/starting-before-input.factory.spec.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/__tests__/starting-before-input.factory.spec.ts index 484cd8da636e..814739584210 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/__tests__/starting-before-input.factory.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/__tests__/starting-before-input.factory.spec.ts @@ -1,6 +1,6 @@ import { Test, TestingModule } from '@nestjs/testing'; -import { StartingAfterInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/starting-after-input.factory'; +import { StartingAfterInputFactory } from 'src/engine/api/rest/input-factories/starting-after-input.factory'; describe('StartingAfterInputFactory', () => { let service: StartingAfterInputFactory; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/ending-before-input.factory.ts b/packages/twenty-server/src/engine/api/rest/input-factories/ending-before-input.factory.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/ending-before-input.factory.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/ending-before-input.factory.ts diff --git a/packages/twenty-server/src/engine/api/rest/input-factories/factories.ts b/packages/twenty-server/src/engine/api/rest/input-factories/factories.ts new file mode 100644 index 000000000000..bbe17eeca1ba --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/input-factories/factories.ts @@ -0,0 +1,13 @@ +import { StartingAfterInputFactory } from 'src/engine/api/rest/input-factories/starting-after-input.factory'; +import { EndingBeforeInputFactory } from 'src/engine/api/rest/input-factories/ending-before-input.factory'; +import { LimitInputFactory } from 'src/engine/api/rest/input-factories/limit-input.factory'; +import { OrderByInputFactory } from 'src/engine/api/rest/input-factories/order-by-input.factory'; +import { FilterInputFactory } from 'src/engine/api/rest/input-factories/filter-input.factory'; + +export const inputFactories = [ + StartingAfterInputFactory, + EndingBeforeInputFactory, + LimitInputFactory, + OrderByInputFactory, + FilterInputFactory, +]; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-input.factory.ts b/packages/twenty-server/src/engine/api/rest/input-factories/filter-input.factory.ts similarity index 54% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-input.factory.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/filter-input.factory.ts index 1041212dfbbb..93ed93d51eb4 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-input.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/filter-input.factory.ts @@ -2,10 +2,10 @@ import { Injectable } from '@nestjs/common'; import { Request } from 'express'; -import { addDefaultConjunctionIfMissing } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/add-default-conjunction.utils'; -import { checkFilterQuery } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/check-filter-query.utils'; -import { parseFilter } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils'; -import { FieldValue } from 'src/engine/api/rest/types/field-value.type'; +import { addDefaultConjunctionIfMissing } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/add-default-conjunction.utils'; +import { checkFilterQuery } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/check-filter-query.utils'; +import { parseFilter } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils'; +import { FieldValue } from 'src/engine/api/rest/core/types/field-value.type'; @Injectable() export class FilterInputFactory { diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/limit-input.factory.ts b/packages/twenty-server/src/engine/api/rest/input-factories/limit-input.factory.ts similarity index 83% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/limit-input.factory.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/limit-input.factory.ts index 26d958d3a390..d7ccaddc0904 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/limit-input.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/limit-input.factory.ts @@ -4,9 +4,9 @@ import { Request } from 'express'; @Injectable() export class LimitInputFactory { - create(request: Request): number { + create(request: Request, defaultLimit = 60): number { if (!request.query.limit) { - return 60; + return defaultLimit; } const limit = +request.query.limit; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory.ts b/packages/twenty-server/src/engine/api/rest/input-factories/order-by-input.factory.ts similarity index 82% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/order-by-input.factory.ts index 2b44cd279a22..832de44c52f9 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory.ts +++ b/packages/twenty-server/src/engine/api/rest/input-factories/order-by-input.factory.ts @@ -7,7 +7,7 @@ import { RecordOrderBy, } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; -import { checkFields } from 'src/engine/api/rest/rest-api-core-query-builder/utils/check-fields.utils'; +import { checkArrayFields } from 'src/engine/api/rest/core/query-builder/utils/check-order-by.utils'; export const DEFAULT_ORDER_DIRECTION = OrderByDirection.AscNullsFirst; @@ -17,12 +17,12 @@ export class OrderByInputFactory { const orderByQuery = request.query.order_by; if (typeof orderByQuery !== 'string') { - return {}; + return [{}]; } //orderByQuery = field_1[AscNullsFirst],field_2[DescNullsLast],field_3 const orderByItems = orderByQuery.split(','); - let result = {}; + let result: Array<Record<string, OrderByDirection>> = []; let itemDirection = ''; let itemFields = ''; @@ -65,10 +65,14 @@ export class OrderByInputFactory { } }, itemDirection); - result = { ...result, ...fieldResult }; + const resultFields = Object.keys(fieldResult).map((key) => ({ + [key]: fieldResult[key], + })); + + result = [...result, ...resultFields]; } - checkFields(objectMetadata.objectMetadataItem, Object.keys(result)); + checkArrayFields(objectMetadata.objectMetadataItem, result); return result; } diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/starting-after-input.factory.ts b/packages/twenty-server/src/engine/api/rest/input-factories/starting-after-input.factory.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/starting-after-input.factory.ts rename to packages/twenty-server/src/engine/api/rest/input-factories/starting-after-input.factory.ts diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/create-metadata-query.factory.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/create-metadata-query.factory.ts new file mode 100644 index 000000000000..67333a146c3a --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/create-metadata-query.factory.ts @@ -0,0 +1,24 @@ +import { Injectable } from '@nestjs/common'; + +import { capitalize } from 'src/utils/capitalize'; +import { fetchMetadataFields } from 'src/engine/api/rest/metadata/query-builder/utils/fetch-metadata-fields.utils'; + +@Injectable() +export class CreateMetadataQueryFactory { + create(objectNameSingular: string, objectNamePlural: string): string { + const objectNameCapitalized = capitalize(objectNameSingular); + + const fields = fetchMetadataFields(objectNamePlural); + + return ` + mutation Create${objectNameCapitalized}($input: CreateOne${objectNameCapitalized}${ + objectNameSingular === 'field' ? 'Metadata' : '' + }Input!) { + createOne${objectNameCapitalized}(input: $input) { + id + ${fields} + } + } + `; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/delete-metadata-query.factory.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/delete-metadata-query.factory.ts new file mode 100644 index 000000000000..c1e09b05bdf9 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/delete-metadata-query.factory.ts @@ -0,0 +1,18 @@ +import { Injectable } from '@nestjs/common'; + +import { capitalize } from 'src/utils/capitalize'; + +@Injectable() +export class DeleteMetadataQueryFactory { + create(objectNameSingular: string): string { + const objectNameCapitalized = capitalize(objectNameSingular); + + return ` + mutation Delete${objectNameCapitalized}($input: DeleteOne${objectNameCapitalized}Input!) { + deleteOne${objectNameCapitalized}(input: $input) { + id + } + } + `; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/find-many-metadata-query.factory.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/find-many-metadata-query.factory.ts new file mode 100644 index 000000000000..7434b2eb30b7 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/find-many-metadata-query.factory.ts @@ -0,0 +1,33 @@ +import { Injectable } from '@nestjs/common'; + +import { capitalize } from 'src/utils/capitalize'; +import { fetchMetadataFields } from 'src/engine/api/rest/metadata/query-builder/utils/fetch-metadata-fields.utils'; + +@Injectable() +export class FindManyMetadataQueryFactory { + create(objectNamePlural): string { + const fields = fetchMetadataFields(objectNamePlural); + + return ` + query FindMany${capitalize(objectNamePlural)}( + $paging: CursorPaging! + ) { + ${objectNamePlural}( + paging: $paging + ) { + edges { + node { + id + ${fields} + } + } + pageInfo { + hasNextPage + startCursor + endCursor + } + } + } + `; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/find-one-metadata-query.factory.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/find-one-metadata-query.factory.ts new file mode 100644 index 000000000000..98078a29efa4 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/find-one-metadata-query.factory.ts @@ -0,0 +1,22 @@ +import { Injectable } from '@nestjs/common'; + +import { capitalize } from 'src/utils/capitalize'; +import { fetchMetadataFields } from 'src/engine/api/rest/metadata/query-builder/utils/fetch-metadata-fields.utils'; + +@Injectable() +export class FindOneMetadataQueryFactory { + create(objectNameSingular: string, objectNamePlural: string): string { + const fields = fetchMetadataFields(objectNamePlural); + + return ` + query FindOne${capitalize(objectNameSingular)}( + $id: UUID!, + ) { + ${objectNameSingular}(id: $id) { + id + ${fields} + } + } + `; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/get-metadata-variables.factory.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/get-metadata-variables.factory.ts new file mode 100644 index 000000000000..61149853b0f9 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/get-metadata-variables.factory.ts @@ -0,0 +1,42 @@ +import { BadRequestException, Injectable } from '@nestjs/common'; + +import { Request } from 'express'; + +import { LimitInputFactory } from 'src/engine/api/rest/input-factories/limit-input.factory'; +import { EndingBeforeInputFactory } from 'src/engine/api/rest/input-factories/ending-before-input.factory'; +import { StartingAfterInputFactory } from 'src/engine/api/rest/input-factories/starting-after-input.factory'; +import { MetadataQueryVariables } from 'src/engine/api/rest/metadata/types/metadata-query-variables.type'; + +@Injectable() +export class GetMetadataVariablesFactory { + constructor( + private readonly startingAfterInputFactory: StartingAfterInputFactory, + private readonly endingBeforeInputFactory: EndingBeforeInputFactory, + private readonly limitInputFactory: LimitInputFactory, + ) {} + + create(id: string | undefined, request: Request): MetadataQueryVariables { + if (id) { + return { id }; + } + + const limit = this.limitInputFactory.create(request, 1000); + const before = this.endingBeforeInputFactory.create(request); + const after = this.startingAfterInputFactory.create(request); + + if (before && after) { + throw new BadRequestException( + `Only one of 'endingBefore' and 'startingAfter' may be provided`, + ); + } + + return { + paging: { + first: !before ? limit : undefined, + last: before ? limit : undefined, + after, + before, + }, + }; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/metadata-factories.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/metadata-factories.ts new file mode 100644 index 000000000000..f8a7f485561c --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/metadata-factories.ts @@ -0,0 +1,17 @@ +import { FindOneMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/find-one-metadata-query.factory'; +import { FindManyMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/find-many-metadata-query.factory'; +import { GetMetadataVariablesFactory } from 'src/engine/api/rest/metadata/query-builder/factories/get-metadata-variables.factory'; +import { inputFactories } from 'src/engine/api/rest/input-factories/factories'; +import { CreateMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/create-metadata-query.factory'; +import { UpdateMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/update-metadata-query.factory'; +import { DeleteMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/delete-metadata-query.factory'; + +export const metadataQueryBuilderFactories = [ + FindOneMetadataQueryFactory, + FindManyMetadataQueryFactory, + CreateMetadataQueryFactory, + DeleteMetadataQueryFactory, + UpdateMetadataQueryFactory, + GetMetadataVariablesFactory, + ...inputFactories, +]; diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/update-metadata-query.factory.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/update-metadata-query.factory.ts new file mode 100644 index 000000000000..6be62fc48a9d --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/factories/update-metadata-query.factory.ts @@ -0,0 +1,24 @@ +import { Injectable } from '@nestjs/common'; + +import { capitalize } from 'src/utils/capitalize'; +import { fetchMetadataFields } from 'src/engine/api/rest/metadata/query-builder/utils/fetch-metadata-fields.utils'; + +@Injectable() +export class UpdateMetadataQueryFactory { + create(objectNameSingular: string, objectNamePlural: string): string { + const objectNameCapitalized = capitalize(objectNameSingular); + + const fields = fetchMetadataFields(objectNamePlural); + + return ` + mutation Update${objectNameCapitalized}($input: UpdateOne${objectNameCapitalized}${ + objectNameSingular === 'field' ? 'Metadata' : '' + }Input!) { + updateOne${objectNameCapitalized}(input: $input) { + id + ${fields} + } + } + `; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/metadata-query-builder.factory.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/metadata-query-builder.factory.ts new file mode 100644 index 000000000000..ce9b38ce8265 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/metadata-query-builder.factory.ts @@ -0,0 +1,95 @@ +import { BadRequestException, Injectable } from '@nestjs/common'; + +import { Request } from 'express'; + +import { GetMetadataVariablesFactory } from 'src/engine/api/rest/metadata/query-builder/factories/get-metadata-variables.factory'; +import { FindOneMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/find-one-metadata-query.factory'; +import { FindManyMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/find-many-metadata-query.factory'; +import { parseMetadataPath } from 'src/engine/api/rest/metadata/query-builder/utils/parse-metadata-path.utils'; +import { CreateMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/create-metadata-query.factory'; +import { UpdateMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/update-metadata-query.factory'; +import { DeleteMetadataQueryFactory } from 'src/engine/api/rest/metadata/query-builder/factories/delete-metadata-query.factory'; +import { MetadataQuery } from 'src/engine/api/rest/metadata/types/metadata-query.type'; + +@Injectable() +export class MetadataQueryBuilderFactory { + constructor( + private readonly findOneQueryFactory: FindOneMetadataQueryFactory, + private readonly findManyQueryFactory: FindManyMetadataQueryFactory, + private readonly createQueryFactory: CreateMetadataQueryFactory, + private readonly updateQueryFactory: UpdateMetadataQueryFactory, + private readonly deleteQueryFactory: DeleteMetadataQueryFactory, + private readonly getMetadataVariablesFactory: GetMetadataVariablesFactory, + ) {} + + async get(request: Request): Promise<MetadataQuery> { + const { id, objectNameSingular, objectNamePlural } = + parseMetadataPath(request); + + return { + query: id + ? this.findOneQueryFactory.create(objectNameSingular, objectNamePlural) + : this.findManyQueryFactory.create(objectNamePlural), + variables: this.getMetadataVariablesFactory.create(id, request), + }; + } + + async create(request: Request): Promise<MetadataQuery> { + const { objectNameSingular, objectNamePlural } = parseMetadataPath(request); + + return { + query: this.createQueryFactory.create( + objectNameSingular, + objectNamePlural, + ), + variables: { + input: { + [objectNameSingular]: request.body, + }, + }, + }; + } + + async update(request: Request): Promise<MetadataQuery> { + const { objectNameSingular, objectNamePlural, id } = + parseMetadataPath(request); + + if (!id) { + throw new BadRequestException( + `update ${objectNameSingular} query invalid. Id missing. eg: /rest/metadata/${objectNameSingular}/0d4389ef-ea9c-4ae8-ada1-1cddc440fb56`, + ); + } + + return { + query: this.updateQueryFactory.create( + objectNameSingular, + objectNamePlural, + ), + variables: { + input: { + update: request.body, + id, + }, + }, + }; + } + + async delete(request: Request): Promise<MetadataQuery> { + const { objectNameSingular, id } = parseMetadataPath(request); + + if (!id) { + throw new BadRequestException( + `delete ${objectNameSingular} query invalid. Id missing. eg: /rest/metadata/${objectNameSingular}/0d4389ef-ea9c-4ae8-ada1-1cddc440fb56`, + ); + } + + return { + query: this.deleteQueryFactory.create(objectNameSingular), + variables: { + input: { + id, + }, + }, + }; + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/metadata-query-builder.module.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/metadata-query-builder.module.ts new file mode 100644 index 000000000000..82bf0fb5674e --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/metadata-query-builder.module.ts @@ -0,0 +1,12 @@ +import { Module } from '@nestjs/common'; + +import { AuthModule } from 'src/engine/core-modules/auth/auth.module'; +import { MetadataQueryBuilderFactory } from 'src/engine/api/rest/metadata/query-builder/metadata-query-builder.factory'; +import { metadataQueryBuilderFactories } from 'src/engine/api/rest/metadata/query-builder/factories/metadata-factories'; + +@Module({ + imports: [AuthModule], + providers: [...metadataQueryBuilderFactories, MetadataQueryBuilderFactory], + exports: [MetadataQueryBuilderFactory], +}) +export class MetadataQueryBuilderModule {} diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-metadata-path.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/__tests__/parse-metadata-path.utils.spec.ts similarity index 90% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-metadata-path.utils.spec.ts rename to packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/__tests__/parse-metadata-path.utils.spec.ts index 28b1238c2468..ec6284da777d 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/__tests__/parse-metadata-path.utils.spec.ts +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/__tests__/parse-metadata-path.utils.spec.ts @@ -1,4 +1,4 @@ -import { parseMetadataPath } from 'src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-metadata-path.utils'; +import { parseMetadataPath } from 'src/engine/api/rest/metadata/query-builder/utils/parse-metadata-path.utils'; describe('parseMetadataPath', () => { it('should parse object from request path with uuid', () => { diff --git a/packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/fetch-metadata-fields.utils.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/fetch-metadata-fields.utils.ts new file mode 100644 index 000000000000..61899096d206 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/fetch-metadata-fields.utils.ts @@ -0,0 +1,93 @@ +export const fetchMetadataFields = (objectNamePlural: string) => { + const fields = ` + type + name + label + description + icon + isCustom + isActive + isSystem + isNullable + createdAt + updatedAt + fromRelationMetadata { + id + relationType + toObjectMetadata { + id + dataSourceId + nameSingular + namePlural + isSystem + } + toFieldMetadataId + } + toRelationMetadata { + id + relationType + fromObjectMetadata { + id + dataSourceId + nameSingular + namePlural + isSystem + } + fromFieldMetadataId + } + defaultValue + options + `; + + switch (objectNamePlural) { + case 'objects': + return ` + dataSourceId + nameSingular + namePlural + labelSingular + labelPlural + description + icon + isCustom + isActive + isSystem + createdAt + updatedAt + labelIdentifierFieldMetadataId + imageIdentifierFieldMetadataId + fields(paging: { first: 1000 }) { + edges { + node { + id + ${fields} + } + } + } + `; + case 'fields': + return fields; + case 'relations': + return ` + relationType + fromObjectMetadata { + id + dataSourceId + nameSingular + namePlural + isSystem + } + fromObjectMetadataId + toObjectMetadata { + id + dataSourceId + nameSingular + namePlural + isSystem + } + toObjectMetadataId + fromFieldMetadataId + toFieldMetadataId + `; + } +}; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-metadata-path.utils.ts b/packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/parse-metadata-path.utils.ts similarity index 100% rename from packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-metadata-path.utils.ts rename to packages/twenty-server/src/engine/api/rest/metadata/query-builder/utils/parse-metadata-path.utils.ts diff --git a/packages/twenty-server/src/engine/api/rest/controllers/rest-api-metadata.controller.ts b/packages/twenty-server/src/engine/api/rest/metadata/rest-api-metadata.controller.ts similarity index 76% rename from packages/twenty-server/src/engine/api/rest/controllers/rest-api-metadata.controller.ts rename to packages/twenty-server/src/engine/api/rest/metadata/rest-api-metadata.controller.ts index ac42dba52880..fc0154081f51 100644 --- a/packages/twenty-server/src/engine/api/rest/controllers/rest-api-metadata.controller.ts +++ b/packages/twenty-server/src/engine/api/rest/metadata/rest-api-metadata.controller.ts @@ -11,7 +11,7 @@ import { import { Request, Response } from 'express'; -import { RestApiMetadataService } from 'src/engine/api/rest/services/rest-api-metadata.service'; +import { RestApiMetadataService } from 'src/engine/api/rest/metadata/rest-api-metadata.service'; import { cleanGraphQLResponse } from 'src/engine/api/rest/utils/clean-graphql-response.utils'; @Controller('rest/metadata/*') @@ -24,28 +24,28 @@ export class RestApiMetadataController { async handleApiGet(@Req() request: Request, @Res() res: Response) { const result = await this.restApiMetadataService.get(request); - res.status(200).send(cleanGraphQLResponse(result.data)); + res.status(200).send(cleanGraphQLResponse(result.data.data)); } @Delete() async handleApiDelete(@Req() request: Request, @Res() res: Response) { const result = await this.restApiMetadataService.delete(request); - res.status(200).send(cleanGraphQLResponse(result.data)); + res.status(200).send(cleanGraphQLResponse(result.data.data)); } @Post() async handleApiPost(@Req() request: Request, @Res() res: Response) { const result = await this.restApiMetadataService.create(request); - res.status(201).send(cleanGraphQLResponse(result.data)); + res.status(201).send(cleanGraphQLResponse(result.data.data)); } @Patch() async handleApiPatch(@Req() request: Request, @Res() res: Response) { const result = await this.restApiMetadataService.update(request); - res.status(200).send(cleanGraphQLResponse(result.data)); + res.status(200).send(cleanGraphQLResponse(result.data.data)); } // This endpoint is not documented in the OpenAPI schema. @@ -55,6 +55,6 @@ export class RestApiMetadataController { async handleApiPut(@Req() request: Request, @Res() res: Response) { const result = await this.restApiMetadataService.update(request); - res.status(200).send(cleanGraphQLResponse(result.data)); + res.status(200).send(cleanGraphQLResponse(result.data.data)); } } diff --git a/packages/twenty-server/src/engine/api/rest/metadata/rest-api-metadata.service.ts b/packages/twenty-server/src/engine/api/rest/metadata/rest-api-metadata.service.ts new file mode 100644 index 000000000000..afd0e59efd77 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/rest-api-metadata.service.ts @@ -0,0 +1,63 @@ +import { Injectable } from '@nestjs/common'; + +import { Request } from 'express'; + +import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; +import { + GraphqlApiType, + RestApiService, +} from 'src/engine/api/rest/rest-api.service'; +import { MetadataQueryBuilderFactory } from 'src/engine/api/rest/metadata/query-builder/metadata-query-builder.factory'; + +@Injectable() +export class RestApiMetadataService { + constructor( + private readonly tokenService: TokenService, + private readonly metadataQueryBuilderFactory: MetadataQueryBuilderFactory, + private readonly restApiService: RestApiService, + ) {} + + async get(request: Request) { + await this.tokenService.validateToken(request); + const data = await this.metadataQueryBuilderFactory.get(request); + + return await this.restApiService.call( + GraphqlApiType.METADATA, + request, + data, + ); + } + + async create(request: Request) { + await this.tokenService.validateToken(request); + const data = await this.metadataQueryBuilderFactory.create(request); + + return await this.restApiService.call( + GraphqlApiType.METADATA, + request, + data, + ); + } + + async update(request: Request) { + await this.tokenService.validateToken(request); + const data = await this.metadataQueryBuilderFactory.update(request); + + return await this.restApiService.call( + GraphqlApiType.METADATA, + request, + data, + ); + } + + async delete(request: Request) { + await this.tokenService.validateToken(request); + const data = await this.metadataQueryBuilderFactory.delete(request); + + return await this.restApiService.call( + GraphqlApiType.METADATA, + request, + data, + ); + } +} diff --git a/packages/twenty-server/src/engine/api/rest/metadata/types/metadata-query-variables.type.ts b/packages/twenty-server/src/engine/api/rest/metadata/types/metadata-query-variables.type.ts new file mode 100644 index 000000000000..edc86f56fb1e --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/types/metadata-query-variables.type.ts @@ -0,0 +1,10 @@ +export type MetadataQueryVariables = { + id?: string; + input?: object; + paging?: { + first?: number; + last?: number; + after?: string; + before?: string; + }; +}; diff --git a/packages/twenty-server/src/engine/api/rest/metadata/types/metadata-query.type.ts b/packages/twenty-server/src/engine/api/rest/metadata/types/metadata-query.type.ts new file mode 100644 index 000000000000..6a3342482338 --- /dev/null +++ b/packages/twenty-server/src/engine/api/rest/metadata/types/metadata-query.type.ts @@ -0,0 +1,6 @@ +import { MetadataQueryVariables } from 'src/engine/api/rest/metadata/types/metadata-query-variables.type'; + +export type MetadataQuery = { + query: string; + variables: MetadataQueryVariables; +}; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/__tests__/core-query-builder.factory.spec.ts b/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/__tests__/core-query-builder.factory.spec.ts deleted file mode 100644 index b64197417129..000000000000 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/__tests__/core-query-builder.factory.spec.ts +++ /dev/null @@ -1,46 +0,0 @@ -import { Test, TestingModule } from '@nestjs/testing'; - -import { CoreQueryBuilderFactory } from 'src/engine/api/rest/rest-api-core-query-builder/core-query-builder.factory'; -import { DeleteQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/delete-query.factory'; -import { CreateOneQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-one-query.factory'; -import { CreateManyQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-many-query.factory'; -import { UpdateQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/update-query.factory'; -import { FindOneQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/find-one-query.factory'; -import { FindManyQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/find-many-query.factory'; -import { DeleteVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/delete-variables.factory'; -import { CreateVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-variables.factory'; -import { UpdateVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/update-variables.factory'; -import { GetVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/get-variables.factory'; -import { ObjectMetadataService } from 'src/engine/metadata-modules/object-metadata/object-metadata.service'; -import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; - -describe('CoreQueryBuilderFactory', () => { - let service: CoreQueryBuilderFactory; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - providers: [ - CoreQueryBuilderFactory, - { provide: DeleteQueryFactory, useValue: {} }, - { provide: CreateOneQueryFactory, useValue: {} }, - { provide: CreateManyQueryFactory, useValue: {} }, - { provide: UpdateQueryFactory, useValue: {} }, - { provide: FindOneQueryFactory, useValue: {} }, - { provide: FindManyQueryFactory, useValue: {} }, - { provide: DeleteVariablesFactory, useValue: {} }, - { provide: CreateVariablesFactory, useValue: {} }, - { provide: UpdateVariablesFactory, useValue: {} }, - { provide: GetVariablesFactory, useValue: {} }, - { provide: ObjectMetadataService, useValue: {} }, - { provide: TokenService, useValue: {} }, - { provide: EnvironmentService, useValue: {} }, - ], - }).compile(); - - service = module.get<CoreQueryBuilderFactory>(CoreQueryBuilderFactory); - }); - it('should be defined', () => { - expect(service).toBeDefined(); - }); -}); diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-variables.factory.ts b/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-variables.factory.ts deleted file mode 100644 index eb4b20ef1c85..000000000000 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/create-variables.factory.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { Request } from 'express'; - -import { QueryVariables } from 'src/engine/api/rest/types/query-variables.type'; - -@Injectable() -export class CreateVariablesFactory { - create(request: Request): QueryVariables { - const data = Array.isArray(request.body) - ? request.body.map((recordData) => { - return { position: 'first', ...recordData }; - }) - : { position: 'first', ...request.body }; - - return { - data, - }; - } -} diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/factories.ts b/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/factories.ts deleted file mode 100644 index d583d2dfdf84..000000000000 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/factories.ts +++ /dev/null @@ -1,33 +0,0 @@ -import { DeleteQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/delete-query.factory'; -import { CreateOneQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-one-query.factory'; -import { UpdateQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/update-query.factory'; -import { FindOneQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/find-one-query.factory'; -import { FindManyQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/find-many-query.factory'; -import { DeleteVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/delete-variables.factory'; -import { CreateVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-variables.factory'; -import { UpdateVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/update-variables.factory'; -import { GetVariablesFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/get-variables.factory'; -import { LimitInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/limit-input.factory'; -import { OrderByInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory'; -import { FilterInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-input.factory'; -import { CreateManyQueryFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/create-many-query.factory'; -import { StartingAfterInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/starting-after-input.factory'; -import { EndingBeforeInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/ending-before-input.factory'; - -export const coreQueryBuilderFactories = [ - DeleteQueryFactory, - CreateOneQueryFactory, - CreateManyQueryFactory, - UpdateQueryFactory, - FindOneQueryFactory, - FindManyQueryFactory, - DeleteVariablesFactory, - CreateVariablesFactory, - UpdateVariablesFactory, - GetVariablesFactory, - StartingAfterInputFactory, - EndingBeforeInputFactory, - LimitInputFactory, - OrderByInputFactory, - FilterInputFactory, -]; diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/get-variables.factory.ts b/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/get-variables.factory.ts deleted file mode 100644 index 8703e67f7f2f..000000000000 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/factories/get-variables.factory.ts +++ /dev/null @@ -1,39 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { Request } from 'express'; - -import { LimitInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/limit-input.factory'; -import { OrderByInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory'; -import { FilterInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-input.factory'; -import { QueryVariables } from 'src/engine/api/rest/types/query-variables.type'; -import { EndingBeforeInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/ending-before-input.factory'; -import { StartingAfterInputFactory } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/starting-after-input.factory'; - -@Injectable() -export class GetVariablesFactory { - constructor( - private readonly startingAfterInputFactory: StartingAfterInputFactory, - private readonly endingBeforeInputFactory: EndingBeforeInputFactory, - private readonly limitInputFactory: LimitInputFactory, - private readonly orderByInputFactory: OrderByInputFactory, - private readonly filterInputFactory: FilterInputFactory, - ) {} - - create( - id: string | undefined, - request: Request, - objectMetadata, - ): QueryVariables { - if (id) { - return { filter: { id: { eq: id } } }; - } - - return { - filter: this.filterInputFactory.create(request, objectMetadata), - orderBy: this.orderByInputFactory.create(request, objectMetadata), - limit: this.limitInputFactory.create(request), - startingAfter: this.startingAfterInputFactory.create(request), - endingBefore: this.endingBeforeInputFactory.create(request), - }; - } -} diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/check-fields.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/check-fields.utils.spec.ts deleted file mode 100644 index c54bf49bfcf6..000000000000 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/check-fields.utils.spec.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { objectMetadataItemMock } from 'src/engine/api/__mocks__/object-metadata-item.mock'; -import { checkFields } from 'src/engine/api/rest/rest-api-core-query-builder/utils/check-fields.utils'; - -describe('checkFields', () => { - it('should check field types', () => { - expect(() => - checkFields(objectMetadataItemMock, ['fieldNumber']), - ).not.toThrow(); - - expect(() => checkFields(objectMetadataItemMock, ['wrongField'])).toThrow(); - - expect(() => - checkFields(objectMetadataItemMock, ['fieldNumber', 'wrongField']), - ).toThrow(); - }); -}); diff --git a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/map-field-metadata-to-graphql-query.utils.spec.ts b/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/map-field-metadata-to-graphql-query.utils.spec.ts deleted file mode 100644 index 76dbe1b732ff..000000000000 --- a/packages/twenty-server/src/engine/api/rest/rest-api-core-query-builder/utils/__tests__/map-field-metadata-to-graphql-query.utils.spec.ts +++ /dev/null @@ -1,37 +0,0 @@ -import { - fieldCurrencyMock, - fieldLinkMock, - fieldNumberMock, - fieldTextMock, - objectMetadataItemMock, -} from 'src/engine/api/__mocks__/object-metadata-item.mock'; -import { mapFieldMetadataToGraphqlQuery } from 'src/engine/api/rest/rest-api-core-query-builder/utils/map-field-metadata-to-graphql-query.utils'; - -describe('mapFieldMetadataToGraphqlQuery', () => { - it('should map properly', () => { - expect( - mapFieldMetadataToGraphqlQuery(objectMetadataItemMock, fieldNumberMock), - ).toEqual('fieldNumber'); - expect( - mapFieldMetadataToGraphqlQuery(objectMetadataItemMock, fieldTextMock), - ).toEqual('fieldText'); - expect( - mapFieldMetadataToGraphqlQuery(objectMetadataItemMock, fieldLinkMock), - ).toEqual(` - fieldLink - { - label - url - } - `); - expect( - mapFieldMetadataToGraphqlQuery(objectMetadataItemMock, fieldCurrencyMock), - ).toEqual(` - fieldCurrency - { - amountMicros - currencyCode - } - `); - }); -}); diff --git a/packages/twenty-server/src/engine/api/rest/rest-api.module.ts b/packages/twenty-server/src/engine/api/rest/rest-api.module.ts index 16141805d5e9..c459f3286264 100644 --- a/packages/twenty-server/src/engine/api/rest/rest-api.module.ts +++ b/packages/twenty-server/src/engine/api/rest/rest-api.module.ts @@ -1,23 +1,39 @@ import { Module } from '@nestjs/common'; import { HttpModule } from '@nestjs/axios'; -import { RestApiCoreController } from 'src/engine/api/rest/controllers/rest-api-core.controller'; -import { RestApiCoreService } from 'src/engine/api/rest/services/rest-api-core.service'; -import { CoreQueryBuilderModule } from 'src/engine/api/rest/rest-api-core-query-builder/core-query-builder.module'; +import { RestApiCoreController } from 'src/engine/api/rest/core/controllers/rest-api-core.controller'; +import { RestApiCoreService } from 'src/engine/api/rest/core/rest-api-core.service'; +import { CoreQueryBuilderModule } from 'src/engine/api/rest/core/query-builder/core-query-builder.module'; import { AuthModule } from 'src/engine/core-modules/auth/auth.module'; -import { RestApiMetadataController } from 'src/engine/api/rest/controllers/rest-api-metadata.controller'; -import { RestApiMetadataService } from 'src/engine/api/rest/services/rest-api-metadata.service'; -import { RestApiCoreBatchController } from 'src/engine/api/rest/controllers/rest-api-core-batch.controller'; -import { RestApiService } from 'src/engine/api/rest/services/rest-api.service'; +import { RestApiMetadataController } from 'src/engine/api/rest/metadata/rest-api-metadata.controller'; +import { RestApiMetadataService } from 'src/engine/api/rest/metadata/rest-api-metadata.service'; +import { RestApiCoreBatchController } from 'src/engine/api/rest/core/controllers/rest-api-core-batch.controller'; +import { RestApiService } from 'src/engine/api/rest/rest-api.service'; +import { EndingBeforeInputFactory } from 'src/engine/api/rest/input-factories/ending-before-input.factory'; +import { LimitInputFactory } from 'src/engine/api/rest/input-factories/limit-input.factory'; +import { StartingAfterInputFactory } from 'src/engine/api/rest/input-factories/starting-after-input.factory'; +import { MetadataQueryBuilderModule } from 'src/engine/api/rest/metadata/query-builder/metadata-query-builder.module'; @Module({ - imports: [CoreQueryBuilderModule, AuthModule, HttpModule], + imports: [ + CoreQueryBuilderModule, + MetadataQueryBuilderModule, + AuthModule, + HttpModule, + ], controllers: [ RestApiMetadataController, RestApiCoreBatchController, RestApiCoreController, ], - providers: [RestApiMetadataService, RestApiCoreService, RestApiService], + providers: [ + RestApiMetadataService, + RestApiCoreService, + RestApiService, + StartingAfterInputFactory, + EndingBeforeInputFactory, + LimitInputFactory, + ], exports: [RestApiMetadataService], }) export class RestApiModule {} diff --git a/packages/twenty-server/src/engine/api/rest/services/rest-api.service.ts b/packages/twenty-server/src/engine/api/rest/rest-api.service.ts similarity index 95% rename from packages/twenty-server/src/engine/api/rest/services/rest-api.service.ts rename to packages/twenty-server/src/engine/api/rest/rest-api.service.ts index 1107a259812d..d6b4d21640c3 100644 --- a/packages/twenty-server/src/engine/api/rest/services/rest-api.service.ts +++ b/packages/twenty-server/src/engine/api/rest/rest-api.service.ts @@ -4,7 +4,7 @@ import { HttpService } from '@nestjs/axios'; import { Request } from 'express'; import { AxiosResponse } from 'axios'; -import { Query } from 'src/engine/api/rest/types/query.type'; +import { Query } from 'src/engine/api/rest/core/types/query.type'; import { getServerUrl } from 'src/utils/get-server-url'; import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { RestApiException } from 'src/engine/api/rest/errors/RestApiException'; diff --git a/packages/twenty-server/src/engine/api/rest/services/__tests__/core.service.spec.ts b/packages/twenty-server/src/engine/api/rest/services/__tests__/core.service.spec.ts deleted file mode 100644 index 45fde1dfdc34..000000000000 --- a/packages/twenty-server/src/engine/api/rest/services/__tests__/core.service.spec.ts +++ /dev/null @@ -1,30 +0,0 @@ -import { Test, TestingModule } from '@nestjs/testing'; - -import { RestApiCoreService } from 'src/engine/api/rest/services/rest-api-core.service'; -import { CoreQueryBuilderFactory } from 'src/engine/api/rest/rest-api-core-query-builder/core-query-builder.factory'; -import { RestApiService } from 'src/engine/api/rest/services/rest-api.service'; - -describe('RestApiCoreService', () => { - let service: RestApiCoreService; - - beforeEach(async () => { - const module: TestingModule = await Test.createTestingModule({ - providers: [ - RestApiCoreService, - { - provide: CoreQueryBuilderFactory, - useValue: {}, - }, - { - provide: RestApiService, - useValue: {}, - }, - ], - }).compile(); - - service = module.get<RestApiCoreService>(RestApiCoreService); - }); - it('should be defined', () => { - expect(service).toBeDefined(); - }); -}); diff --git a/packages/twenty-server/src/engine/api/rest/services/rest-api-metadata.service.ts b/packages/twenty-server/src/engine/api/rest/services/rest-api-metadata.service.ts deleted file mode 100644 index 007c67b2e5cc..000000000000 --- a/packages/twenty-server/src/engine/api/rest/services/rest-api-metadata.service.ts +++ /dev/null @@ -1,284 +0,0 @@ -import { BadRequestException, Injectable } from '@nestjs/common'; - -import { Query } from 'src/engine/api/rest/types/query.type'; -import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; -import { capitalize } from 'src/utils/capitalize'; -import { parseMetadataPath } from 'src/engine/api/rest/rest-api-core-query-builder/utils/path-parsers/parse-metadata-path.utils'; -import { - GraphqlApiType, - RestApiService, -} from 'src/engine/api/rest/services/rest-api.service'; - -@Injectable() -export class RestApiMetadataService { - constructor( - private readonly tokenService: TokenService, - private readonly restApiService: RestApiService, - ) {} - - fetchMetadataFields(objectNamePlural: string) { - const fields = ` - type - name - label - description - icon - isCustom - isActive - isSystem - isNullable - createdAt - updatedAt - fromRelationMetadata { - id - relationType - toObjectMetadata { - id - dataSourceId - nameSingular - namePlural - isSystem - } - toFieldMetadataId - } - toRelationMetadata { - id - relationType - fromObjectMetadata { - id - dataSourceId - nameSingular - namePlural - isSystem - } - fromFieldMetadataId - } - defaultValue - options - `; - - switch (objectNamePlural) { - case 'objects': - return ` - dataSourceId - nameSingular - namePlural - labelSingular - labelPlural - description - icon - isCustom - isActive - isSystem - createdAt - updatedAt - labelIdentifierFieldMetadataId - imageIdentifierFieldMetadataId - fields(paging: { first: 1000 }) { - edges { - node { - id - ${fields} - } - } - } - `; - case 'fields': - return fields; - case 'relations': - return ` - relationType - fromObjectMetadata { - id - dataSourceId - nameSingular - namePlural - isSystem - } - fromObjectMetadataId - toObjectMetadata { - id - dataSourceId - nameSingular - namePlural - isSystem - } - toObjectMetadataId - fromFieldMetadataId - toFieldMetadataId - `; - } - } - - generateFindManyQuery(objectNameSingular: string, objectNamePlural: string) { - const fields = this.fetchMetadataFields(objectNamePlural); - - return ` - query FindMany${capitalize(objectNamePlural)}( - $filter: ${objectNameSingular}Filter, - ) { - ${objectNamePlural}( - filter: $filter, - paging: { first: 1000 } - ) { - edges { - node { - id - ${fields} - } - } - } - } - `; - } - - generateFindOneQuery(objectNameSingular: string, objectNamePlural: string) { - const fields = this.fetchMetadataFields(objectNamePlural); - - return ` - query FindOne${capitalize(objectNameSingular)}( - $id: UUID!, - ) { - ${objectNameSingular}(id: $id) { - id - ${fields} - } - } - `; - } - - async get(request) { - await this.tokenService.validateToken(request); - - const { objectNameSingular, objectNamePlural, id } = - parseMetadataPath(request); - - const query = id - ? this.generateFindOneQuery(objectNameSingular, objectNamePlural) - : this.generateFindManyQuery(objectNameSingular, objectNamePlural); - - const data: Query = { - query, - variables: id ? { id } : request.body, - }; - - return await this.restApiService.call( - GraphqlApiType.METADATA, - request, - data, - ); - } - - async create(request) { - await this.tokenService.validateToken(request); - - const { objectNameSingular: objectName, objectNamePlural } = - parseMetadataPath(request); - const objectNameCapitalized = capitalize(objectName); - - const fields = this.fetchMetadataFields(objectNamePlural); - - const query = ` - mutation Create${objectNameCapitalized}($input: CreateOne${objectNameCapitalized}Input!) { - createOne${objectNameCapitalized}(input: $input) { - id - ${fields} - } - } - `; - - const data: Query = { - query, - variables: { - input: { - [objectName]: request.body, - }, - }, - }; - - return await this.restApiService.call( - GraphqlApiType.METADATA, - request, - data, - ); - } - - async update(request) { - await this.tokenService.validateToken(request); - - const { - objectNameSingular: objectName, - objectNamePlural, - id, - } = parseMetadataPath(request); - const objectNameCapitalized = capitalize(objectName); - - if (!id) { - throw new BadRequestException( - `update ${objectName} query invalid. Id missing. eg: /rest/metadata/${objectName}/0d4389ef-ea9c-4ae8-ada1-1cddc440fb56`, - ); - } - const fields = this.fetchMetadataFields(objectNamePlural); - - const query = ` - mutation Update${objectNameCapitalized}($input: UpdateOne${objectNameCapitalized}Input!) { - updateOne${objectNameCapitalized}(input: $input) { - id - ${fields} - } - } - `; - - const data: Query = { - query, - variables: { - input: { - update: request.body, - id, - }, - }, - }; - - return await this.restApiService.call( - GraphqlApiType.METADATA, - request, - data, - ); - } - - async delete(request) { - await this.tokenService.validateToken(request); - - const { objectNameSingular: objectName, id } = parseMetadataPath(request); - const objectNameCapitalized = capitalize(objectName); - - if (!id) { - throw new BadRequestException( - `delete ${objectName} query invalid. Id missing. eg: /rest/metadata/${objectName}/0d4389ef-ea9c-4ae8-ada1-1cddc440fb56`, - ); - } - - const query = ` - mutation Delete${objectNameCapitalized}($input: DeleteOne${objectNameCapitalized}Input!) { - deleteOne${objectNameCapitalized}(input: $input) { - id - } - } - `; - - const data: Query = { - query, - variables: { - input: { - id, - }, - }, - }; - - return await this.restApiService.call( - GraphqlApiType.METADATA, - request, - data, - ); - } -} diff --git a/packages/twenty-server/src/engine/api/rest/types/query.type.ts b/packages/twenty-server/src/engine/api/rest/types/query.type.ts deleted file mode 100644 index 1af013c8a13c..000000000000 --- a/packages/twenty-server/src/engine/api/rest/types/query.type.ts +++ /dev/null @@ -1,6 +0,0 @@ -import { QueryVariables } from 'src/engine/api/rest/types/query-variables.type'; - -export type Query = { - query: string; - variables: QueryVariables; -}; diff --git a/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.module.ts b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.module.ts new file mode 100644 index 000000000000..f416a0898787 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.module.ts @@ -0,0 +1,30 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { UserModule } from 'src/engine/core-modules/user/user.module'; +import { AISQLQueryResolver } from 'src/engine/core-modules/ai-sql-query/ai-sql-query.resolver'; +import { AISQLQueryService } from 'src/engine/core-modules/ai-sql-query/ai-sql-query.service'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { WorkspaceQueryRunnerModule } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-runner.module'; +import { LLMChatModelModule } from 'src/engine/integrations/llm-chat-model/llm-chat-model.module'; +import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module'; +import { LLMTracingModule } from 'src/engine/integrations/llm-tracing/llm-tracing.module'; +import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; +import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; +@Module({ + imports: [ + WorkspaceDataSourceModule, + WorkspaceQueryRunnerModule, + UserModule, + TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), + LLMChatModelModule, + LLMTracingModule, + EnvironmentModule, + ObjectMetadataModule, + WorkspaceSyncMetadataModule, + ], + exports: [], + providers: [AISQLQueryResolver, AISQLQueryService], +}) +export class AISQLQueryModule {} diff --git a/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.prompt-templates.ts b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.prompt-templates.ts new file mode 100644 index 000000000000..3b100c0b7711 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.prompt-templates.ts @@ -0,0 +1,14 @@ +import { PromptTemplate } from '@langchain/core/prompts'; + +export const sqlGenerationPromptTemplate = PromptTemplate.fromTemplate<{ + llmOutputJsonSchema: string; + sqlCreateTableStatements: string; + userQuestion: string; +}>(`Always respond following this JSON Schema: {llmOutputJsonSchema} + +Based on the table schema below, write a PostgreSQL query that would answer the user's question. All column names must be enclosed in double quotes. + +{sqlCreateTableStatements} + +Question: {userQuestion} +SQL Query:`); diff --git a/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.resolver.ts b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.resolver.ts new file mode 100644 index 000000000000..6aa38399a91c --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.resolver.ts @@ -0,0 +1,64 @@ +import { Args, Query, Resolver, ArgsType, Field } from '@nestjs/graphql'; +import { ForbiddenException, UseGuards } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; + +import { Repository } from 'typeorm'; + +import { User } from 'src/engine/core-modules/user/user.entity'; +import { JwtAuthGuard } from 'src/engine/guards/jwt.auth.guard'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { AuthWorkspace } from 'src/engine/decorators/auth/auth-workspace.decorator'; +import { + FeatureFlagEntity, + FeatureFlagKeys, +} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { AuthUser } from 'src/engine/decorators/auth/auth-user.decorator'; +import { AISQLQueryResult } from 'src/engine/core-modules/ai-sql-query/dtos/ai-sql-query-result.dto'; +import { AISQLQueryService } from 'src/engine/core-modules/ai-sql-query/ai-sql-query.service'; + +@ArgsType() +class GetAISQLQueryArgs { + @Field(() => String) + text: string; +} + +@UseGuards(JwtAuthGuard) +@Resolver(() => AISQLQueryResult) +export class AISQLQueryResolver { + constructor( + private readonly aiSqlQueryService: AISQLQueryService, + @InjectRepository(FeatureFlagEntity, 'core') + private readonly featureFlagRepository: Repository<FeatureFlagEntity>, + ) {} + + @Query(() => AISQLQueryResult) + async getAISQLQuery( + @AuthWorkspace() { id: workspaceId }: Workspace, + @AuthUser() user: User, + @Args() { text }: GetAISQLQueryArgs, + ) { + const isCopilotEnabledFeatureFlag = + await this.featureFlagRepository.findOneBy({ + workspaceId, + key: FeatureFlagKeys.IsCopilotEnabled, + value: true, + }); + + if (!isCopilotEnabledFeatureFlag?.value) { + throw new ForbiddenException( + `${FeatureFlagKeys.IsCopilotEnabled} feature flag is disabled`, + ); + } + + const traceMetadata = { + userId: user.id, + userEmail: user.email, + }; + + return this.aiSqlQueryService.generateAndExecute( + workspaceId, + text, + traceMetadata, + ); + } +} diff --git a/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.service.ts b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.service.ts new file mode 100644 index 000000000000..465f9a0a481e --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/ai-sql-query/ai-sql-query.service.ts @@ -0,0 +1,250 @@ +import { Injectable, Logger } from '@nestjs/common'; + +import { RunnableSequence } from '@langchain/core/runnables'; +import { StructuredOutputParser } from '@langchain/core/output_parsers'; +import { DataSource, QueryFailedError } from 'typeorm'; +import { z } from 'zod'; +import { zodToJsonSchema } from 'zod-to-json-schema'; +import { PostgresConnectionOptions } from 'typeorm/driver/postgres/PostgresConnectionOptions'; +import groupBy from 'lodash.groupby'; + +import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; +import { WorkspaceQueryRunnerService } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-runner.service'; +import { LLMChatModelService } from 'src/engine/integrations/llm-chat-model/llm-chat-model.service'; +import { LLMTracingService } from 'src/engine/integrations/llm-tracing/llm-tracing.service'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { DEFAULT_LABEL_IDENTIFIER_FIELD_NAME } from 'src/engine/metadata-modules/object-metadata/object-metadata.constants'; +import { StandardObjectFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory'; +import { AISQLQueryResult } from 'src/engine/core-modules/ai-sql-query/dtos/ai-sql-query-result.dto'; +import { sqlGenerationPromptTemplate } from 'src/engine/core-modules/ai-sql-query/ai-sql-query.prompt-templates'; + +@Injectable() +export class AISQLQueryService { + private readonly logger = new Logger(AISQLQueryService.name); + constructor( + private readonly workspaceDataSourceService: WorkspaceDataSourceService, + private readonly workspaceQueryRunnerService: WorkspaceQueryRunnerService, + private readonly llmChatModelService: LLMChatModelService, + private readonly llmTracingService: LLMTracingService, + private readonly standardObjectFactory: StandardObjectFactory, + ) {} + + private getLabelIdentifierName( + objectMetadata: ObjectMetadataEntity, + dataSourceId, + workspaceId, + workspaceFeatureFlagsMap, + ): string | undefined { + const customObjectLabelIdentifierFieldMetadata = objectMetadata.fields.find( + (fieldMetadata) => + fieldMetadata.id === objectMetadata.labelIdentifierFieldMetadataId, + ); + + /* const standardObjectMetadataCollection = this.standardObjectFactory.create( + standardObjectMetadataDefinitions, + { workspaceId, dataSourceId }, + workspaceFeatureFlagsMap, + ); + + const standardObjectLabelIdentifierFieldMetadata = + standardObjectMetadataCollection + .find( + (standardObjectMetadata) => + standardObjectMetadata.nameSingular === objectMetadata.nameSingular, + ) + ?.fields.find( + (field: PartialFieldMetadata) => + field.name === DEFAULT_LABEL_IDENTIFIER_FIELD_NAME, + ) as PartialFieldMetadata; */ + + const labelIdentifierFieldMetadata = + customObjectLabelIdentifierFieldMetadata; /*?? + standardObjectLabelIdentifierFieldMetadata*/ + + return ( + labelIdentifierFieldMetadata?.name ?? DEFAULT_LABEL_IDENTIFIER_FIELD_NAME + ); + } + + private async getColInfosByTableName(dataSource: DataSource) { + const { schema } = dataSource.options as PostgresConnectionOptions; + + // From LangChain sql_utils.ts + const sqlQuery = `SELECT + t.table_name, + c.* + FROM + information_schema.tables t + JOIN information_schema.columns c + ON t.table_name = c.table_name + WHERE + t.table_schema = '${schema}' + AND c.table_schema = '${schema}' + ORDER BY + t.table_name, + c.ordinal_position;`; + const colInfos = await dataSource.query< + { + table_name: string; + column_name: string; + data_type: string | undefined; + is_nullable: 'YES' | 'NO'; + }[] + >(sqlQuery); + + return groupBy(colInfos, (colInfo) => colInfo.table_name); + } + + private getCreateTableStatement(tableName: string, colInfos: any[]) { + return `${`CREATE TABLE ${tableName} (\n`} ${colInfos + .map( + (colInfo) => + `${colInfo.column_name} ${colInfo.data_type} ${ + colInfo.is_nullable === 'YES' ? '' : 'NOT NULL' + }`, + ) + .join(', ')});`; + } + + private getRelationDescriptions() { + // TODO - Construct sentences like the following: + // investorId: a foreign key referencing the person table, indicating the investor who owns this portfolio company. + return ''; + } + + private getTableDescription(tableName: string, colInfos: any[]) { + return [ + this.getCreateTableStatement(tableName, colInfos), + this.getRelationDescriptions(), + ].join('\n'); + } + + private async getWorkspaceSchemaDescription( + dataSource: DataSource, + ): Promise<string> { + const colInfoByTableName = await this.getColInfosByTableName(dataSource); + + return Object.entries(colInfoByTableName) + .map(([tableName, colInfos]) => + this.getTableDescription(tableName, colInfos), + ) + .join('\n\n'); + } + + private async generateWithDataSource( + workspaceId: string, + workspaceDataSource: DataSource, + userQuestion: string, + traceMetadata: Record<string, string> = {}, + ) { + const workspaceSchemaName = + this.workspaceDataSourceService.getSchemaName(workspaceId); + + workspaceDataSource.setOptions({ + schema: workspaceSchemaName, + }); + + const workspaceSchemaDescription = + await this.getWorkspaceSchemaDescription(workspaceDataSource); + + const llmOutputSchema = z.object({ + sqlQuery: z.string(), + }); + + const llmOutputJsonSchema = JSON.stringify( + zodToJsonSchema(llmOutputSchema), + ); + + const structuredOutputParser = + StructuredOutputParser.fromZodSchema(llmOutputSchema); + + const sqlQueryGeneratorChain = RunnableSequence.from([ + sqlGenerationPromptTemplate, + this.llmChatModelService.getJSONChatModel(), + structuredOutputParser, + ]); + + const metadata = { + workspaceId, + ...traceMetadata, + }; + const tracingCallbackHandler = + this.llmTracingService.getCallbackHandler(metadata); + + const { sqlQuery } = await sqlQueryGeneratorChain.invoke( + { + llmOutputJsonSchema, + sqlCreateTableStatements: workspaceSchemaDescription, + userQuestion, + }, + { + callbacks: [tracingCallbackHandler], + }, + ); + + return sqlQuery; + } + + async generate( + workspaceId: string, + userQuestion: string, + traceMetadata: Record<string, string> = {}, + ) { + const workspaceDataSource = + await this.workspaceDataSourceService.connectToWorkspaceDataSource( + workspaceId, + ); + + return this.generateWithDataSource( + workspaceId, + workspaceDataSource, + userQuestion, + traceMetadata, + ); + } + + async generateAndExecute( + workspaceId: string, + userQuestion: string, + traceMetadata: Record<string, string> = {}, + ): Promise<AISQLQueryResult> { + const workspaceDataSource = + await this.workspaceDataSourceService.connectToWorkspaceDataSource( + workspaceId, + ); + + const sqlQuery = await this.generateWithDataSource( + workspaceId, + workspaceDataSource, + userQuestion, + traceMetadata, + ); + + try { + const sqlQueryResult: Record<string, any>[] = + await this.workspaceQueryRunnerService.executeSQL( + workspaceDataSource, + workspaceId, + sqlQuery, + ); + + return { + sqlQuery, + sqlQueryResult: JSON.stringify(sqlQueryResult), + }; + } catch (error) { + if (error instanceof QueryFailedError) { + return { + sqlQuery, + queryFailedErrorMessage: error.message, + }; + } + + this.logger.error(error.message, error.stack); + + return { + sqlQuery, + }; + } + } +} diff --git a/packages/twenty-server/src/engine/core-modules/ai-sql-query/dtos/ai-sql-query-result.dto.ts b/packages/twenty-server/src/engine/core-modules/ai-sql-query/dtos/ai-sql-query-result.dto.ts new file mode 100644 index 000000000000..1046631f3253 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/ai-sql-query/dtos/ai-sql-query-result.dto.ts @@ -0,0 +1,17 @@ +import { Field, ObjectType } from '@nestjs/graphql'; + +import { IsOptional } from 'class-validator'; + +@ObjectType('AISQLQueryResult') +export class AISQLQueryResult { + @Field(() => String) + sqlQuery: string; + + @Field(() => String, { nullable: true }) + @IsOptional() + sqlQueryResult?: string; + + @Field(() => String, { nullable: true }) + @IsOptional() + queryFailedErrorMessage?: string; +} diff --git a/packages/twenty-server/src/engine/core-modules/analytics/analytics.service.ts b/packages/twenty-server/src/engine/core-modules/analytics/analytics.service.ts index fd685851c584..73851f71ea5b 100644 --- a/packages/twenty-server/src/engine/core-modules/analytics/analytics.service.ts +++ b/packages/twenty-server/src/engine/core-modules/analytics/analytics.service.ts @@ -19,8 +19,8 @@ export class AnalyticsService { async create( createEventInput: CreateEventInput, - userId: string | undefined, - workspaceId: string | undefined, + userId: string | null | undefined, + workspaceId: string | null | undefined, workspaceDisplayName: string | undefined, workspaceDomainName: string | undefined, hostName: string | undefined, diff --git a/packages/twenty-server/src/engine/core-modules/auth/auth.module.ts b/packages/twenty-server/src/engine/core-modules/auth/auth.module.ts index 999d3175074f..5f60af70e3db 100644 --- a/packages/twenty-server/src/engine/core-modules/auth/auth.module.ts +++ b/packages/twenty-server/src/engine/core-modules/auth/auth.module.ts @@ -25,9 +25,12 @@ import { MicrosoftAuthController } from 'src/engine/core-modules/auth/controller import { AppTokenService } from 'src/engine/core-modules/app-token/services/app-token.service'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { OnboardingModule } from 'src/engine/core-modules/onboarding/onboarding.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; import { AuthResolver } from './auth.resolver'; @@ -60,11 +63,13 @@ const jwtModule = JwtModule.registerAsync({ ObjectMetadataRepositoryModule.forFeature([ ConnectedAccountWorkspaceEntity, MessageChannelWorkspaceEntity, - CalendarChannelWorkspaceEntity, ]), HttpModule, UserWorkspaceModule, + WorkspaceModule, OnboardingModule, + TwentyORMModule.forFeature([CalendarChannelWorkspaceEntity]), + WorkspaceDataSourceModule, ], controllers: [ GoogleAuthController, diff --git a/packages/twenty-server/src/engine/core-modules/auth/auth.resolver.ts b/packages/twenty-server/src/engine/core-modules/auth/auth.resolver.ts index 9ce29e324c0b..13ecbec165dc 100644 --- a/packages/twenty-server/src/engine/core-modules/auth/auth.resolver.ts +++ b/packages/twenty-server/src/engine/core-modules/auth/auth.resolver.ts @@ -82,9 +82,15 @@ export class AuthResolver { async findWorkspaceFromInviteHash( @Args() workspaceInviteHashValidInput: WorkspaceInviteHashValidInput, ) { - return await this.workspaceRepository.findOneBy({ + const workspace = await this.workspaceRepository.findOneBy({ inviteHash: workspaceInviteHashValidInput.inviteHash, }); + + if (!workspace) { + throw new BadRequestException('Workspace does not exist'); + } + + return workspace; } @UseGuards(CaptchaGuard) @@ -132,6 +138,7 @@ export class AuthResolver { } const transientToken = await this.tokenService.generateTransientToken( workspaceMember.id, + user.id, user.defaultWorkspace.id, ); diff --git a/packages/twenty-server/src/engine/core-modules/auth/controllers/google-apis-auth.controller.ts b/packages/twenty-server/src/engine/core-modules/auth/controllers/google-apis-auth.controller.ts index 0a36fe5d0132..0715d31220a3 100644 --- a/packages/twenty-server/src/engine/core-modules/auth/controllers/google-apis-auth.controller.ts +++ b/packages/twenty-server/src/engine/core-modules/auth/controllers/google-apis-auth.controller.ts @@ -9,16 +9,14 @@ import { import { Response } from 'express'; -import { GoogleAPIsProviderEnabledGuard } from 'src/engine/core-modules/auth/guards/google-apis-provider-enabled.guard'; -import { GoogleAPIsOauthGuard } from 'src/engine/core-modules/auth/guards/google-apis-oauth.guard'; -import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/strategies/google-apis.auth.strategy'; +import { GoogleAPIsOauthRequestCodeGuard } from 'src/engine/core-modules/auth/guards/google-apis-oauth-request-code.guard'; import { GoogleAPIsService } from 'src/engine/core-modules/auth/services/google-apis.service'; import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { OnboardingService } from 'src/engine/core-modules/onboarding/onboarding.service'; -import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { LoadServiceWithWorkspaceContext } from 'src/engine/twenty-orm/context/load-service-with-workspace.context'; +import { GoogleAPIsOauthExchangeCodeForTokenGuard } from 'src/engine/core-modules/auth/guards/google-apis-oauth-exchange-code-for-token.guard'; +import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type'; @Controller('auth/google-apis') export class GoogleAPIsAuthController { @@ -27,19 +25,18 @@ export class GoogleAPIsAuthController { private readonly tokenService: TokenService, private readonly environmentService: EnvironmentService, private readonly onboardingService: OnboardingService, - @InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity) - private readonly workspaceMemberService: WorkspaceMemberRepository, + private readonly loadServiceWithWorkspaceContext: LoadServiceWithWorkspaceContext, ) {} @Get() - @UseGuards(GoogleAPIsProviderEnabledGuard, GoogleAPIsOauthGuard) + @UseGuards(GoogleAPIsOauthRequestCodeGuard) async googleAuth() { // As this method is protected by Google Auth guard, it will trigger Google SSO flow return; } @Get('get-access-token') - @UseGuards(GoogleAPIsProviderEnabledGuard, GoogleAPIsOauthGuard) + @UseGuards(GoogleAPIsOauthExchangeCodeForTokenGuard) async googleAuthGetAccessToken( @Req() req: GoogleAPIsRequest, @Res() res: Response, @@ -47,7 +44,7 @@ export class GoogleAPIsAuthController { const { user } = req; const { - email, + emails, accessToken, refreshToken, transientToken, @@ -56,7 +53,7 @@ export class GoogleAPIsAuthController { messageVisibility, } = user; - const { workspaceMemberId, workspaceId } = + const { workspaceMemberId, userId, workspaceId } = await this.tokenService.verifyTransientToken(transientToken); const demoWorkspaceIds = this.environmentService.get('DEMO_WORKSPACE_IDS'); @@ -71,8 +68,16 @@ export class GoogleAPIsAuthController { throw new Error('Workspace not found'); } - await this.googleAPIsService.refreshGoogleRefreshToken({ - handle: email, + const handle = emails[0].value; + + const googleAPIsServiceInstance = + await this.loadServiceWithWorkspaceContext.load( + this.googleAPIsService, + workspaceId, + ); + + await googleAPIsServiceInstance.refreshGoogleRefreshToken({ + handle, workspaceMemberId: workspaceMemberId, workspaceId: workspaceId, accessToken, @@ -81,12 +86,14 @@ export class GoogleAPIsAuthController { messageVisibility, }); - const userId = ( - await this.workspaceMemberService.find(workspaceMemberId, workspaceId) - )?.userId; - if (userId) { - await this.onboardingService.skipSyncEmailOnboardingStep( + const onboardingServiceInstance = + await this.loadServiceWithWorkspaceContext.load( + this.onboardingService, + workspaceId, + ); + + await onboardingServiceInstance.skipSyncEmailOnboardingStep( userId, workspaceId, ); diff --git a/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth-exchange-code-for-token.guard.ts b/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth-exchange-code-for-token.guard.ts new file mode 100644 index 000000000000..56780ecbcc20 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth-exchange-code-for-token.guard.ts @@ -0,0 +1,74 @@ +import { + ExecutionContext, + Injectable, + NotFoundException, +} from '@nestjs/common'; +import { AuthGuard } from '@nestjs/passport'; +import { InjectRepository } from '@nestjs/typeorm'; + +import { Repository } from 'typeorm'; + +import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; +import { + GoogleAPIScopeConfig, + GoogleAPIsOauthExchangeCodeForTokenStrategy, +} from 'src/engine/core-modules/auth/strategies/google-apis-oauth-exchange-code-for-token.auth.strategy'; +import { + FeatureFlagEntity, + FeatureFlagKeys, +} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; +import { setRequestExtraParams } from 'src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util'; + +@Injectable() +export class GoogleAPIsOauthExchangeCodeForTokenGuard extends AuthGuard( + 'google-apis', +) { + constructor( + private readonly environmentService: EnvironmentService, + private readonly tokenService: TokenService, + @InjectRepository(FeatureFlagEntity, 'core') + private readonly featureFlagRepository: Repository<FeatureFlagEntity>, + ) { + super(); + } + + async canActivate(context: ExecutionContext) { + const request = context.switchToHttp().getRequest(); + const state = JSON.parse(request.query.state); + + if ( + !this.environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED') && + !this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') + ) { + throw new NotFoundException('Google apis auth is not enabled'); + } + + const { workspaceId } = await this.tokenService.verifyTransientToken( + state.transientToken, + ); + + const scopeConfig: GoogleAPIScopeConfig = { + isMessagingAliasFetchingEnabled: + !!(await this.featureFlagRepository.findOneBy({ + workspaceId, + key: FeatureFlagKeys.IsMessagingAliasFetchingEnabled, + value: true, + })), + }; + + new GoogleAPIsOauthExchangeCodeForTokenStrategy( + this.environmentService, + scopeConfig, + ); + + setRequestExtraParams(request, { + transientToken: state.transientToken, + redirectLocation: state.redirectLocation, + calendarVisibility: state.calendarVisibility, + messageVisibility: state.messageVisibility, + }); + + return (await super.canActivate(context)) as boolean; + } +} diff --git a/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth-request-code.guard.ts b/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth-request-code.guard.ts new file mode 100644 index 000000000000..d34ea86180b8 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth-request-code.guard.ts @@ -0,0 +1,72 @@ +import { + ExecutionContext, + Injectable, + NotFoundException, +} from '@nestjs/common'; +import { AuthGuard } from '@nestjs/passport'; +import { InjectRepository } from '@nestjs/typeorm'; + +import { Repository } from 'typeorm'; + +import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; +import { GoogleAPIScopeConfig } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-exchange-code-for-token.auth.strategy'; +import { + FeatureFlagEntity, + FeatureFlagKeys, +} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; +import { GoogleAPIsOauthRequestCodeStrategy } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-request-code.auth.strategy'; +import { setRequestExtraParams } from 'src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util'; + +@Injectable() +export class GoogleAPIsOauthRequestCodeGuard extends AuthGuard('google-apis') { + constructor( + private readonly environmentService: EnvironmentService, + private readonly tokenService: TokenService, + @InjectRepository(FeatureFlagEntity, 'core') + private readonly featureFlagRepository: Repository<FeatureFlagEntity>, + ) { + super({ + prompt: 'select_account', + }); + } + + async canActivate(context: ExecutionContext) { + const request = context.switchToHttp().getRequest(); + + if ( + !this.environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED') && + !this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') + ) { + throw new NotFoundException('Google apis auth is not enabled'); + } + + const { workspaceId } = await this.tokenService.verifyTransientToken( + request.query.transientToken, + ); + + const scopeConfig: GoogleAPIScopeConfig = { + isMessagingAliasFetchingEnabled: + !!(await this.featureFlagRepository.findOneBy({ + workspaceId, + key: FeatureFlagKeys.IsMessagingAliasFetchingEnabled, + value: true, + })), + }; + + new GoogleAPIsOauthRequestCodeStrategy( + this.environmentService, + scopeConfig, + ); + setRequestExtraParams(request, { + transientToken: request.query.transientToken, + redirectLocation: request.query.redirectLocation, + calendarVisibility: request.query.calendarVisibility, + messageVisibility: request.query.messageVisibility, + }); + + const activate = (await super.canActivate(context)) as boolean; + + return activate; + } +} diff --git a/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth.guard.ts b/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth.guard.ts deleted file mode 100644 index a1214404fda4..000000000000 --- a/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-oauth.guard.ts +++ /dev/null @@ -1,39 +0,0 @@ -import { ExecutionContext, Injectable } from '@nestjs/common'; -import { AuthGuard } from '@nestjs/passport'; - -@Injectable() -export class GoogleAPIsOauthGuard extends AuthGuard('google-apis') { - constructor() { - super({ - prompt: 'select_account', - }); - } - - async canActivate(context: ExecutionContext) { - const request = context.switchToHttp().getRequest(); - const transientToken = request.query.transientToken; - const redirectLocation = request.query.redirectLocation; - const calendarVisibility = request.query.calendarVisibility; - const messageVisibility = request.query.messageVisibility; - - if (transientToken && typeof transientToken === 'string') { - request.params.transientToken = transientToken; - } - - if (redirectLocation && typeof redirectLocation === 'string') { - request.params.redirectLocation = redirectLocation; - } - - if (calendarVisibility && typeof calendarVisibility === 'string') { - request.params.calendarVisibility = calendarVisibility; - } - - if (messageVisibility && typeof messageVisibility === 'string') { - request.params.messageVisibility = messageVisibility; - } - - const activate = (await super.canActivate(context)) as boolean; - - return activate; - } -} diff --git a/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-provider-enabled.guard.ts b/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-provider-enabled.guard.ts deleted file mode 100644 index bbd094be4803..000000000000 --- a/packages/twenty-server/src/engine/core-modules/auth/guards/google-apis-provider-enabled.guard.ts +++ /dev/null @@ -1,41 +0,0 @@ -import { Injectable, CanActivate, NotFoundException } from '@nestjs/common'; -import { InjectRepository } from '@nestjs/typeorm'; - -import { Repository } from 'typeorm'; - -import { TokenService } from 'src/engine/core-modules/auth/services/token.service'; -import { - GoogleAPIScopeConfig, - GoogleAPIsStrategy, -} from 'src/engine/core-modules/auth/strategies/google-apis.auth.strategy'; -import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; - -@Injectable() -export class GoogleAPIsProviderEnabledGuard implements CanActivate { - constructor( - private readonly environmentService: EnvironmentService, - private readonly tokenService: TokenService, - @InjectRepository(FeatureFlagEntity, 'core') - private readonly featureFlagRepository: Repository<FeatureFlagEntity>, - ) {} - - async canActivate(): Promise<boolean> { - if ( - !this.environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED') && - !this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') - ) { - throw new NotFoundException('Google apis auth is not enabled'); - } - - const scopeConfig: GoogleAPIScopeConfig = { - isCalendarEnabled: !!this.environmentService.get( - 'MESSAGING_PROVIDER_GMAIL_ENABLED', - ), - }; - - new GoogleAPIsStrategy(this.environmentService, scopeConfig); - - return true; - } -} diff --git a/packages/twenty-server/src/engine/core-modules/auth/services/google-apis.service.ts b/packages/twenty-server/src/engine/core-modules/auth/services/google-apis.service.ts index ed4b58e59f9b..534ad1a6e1db 100644 --- a/packages/twenty-server/src/engine/core-modules/auth/services/google-apis.service.ts +++ b/packages/twenty-server/src/engine/core-modules/auth/services/google-apis.service.ts @@ -1,23 +1,12 @@ -import { Injectable, Inject } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { v4 } from 'uuid'; -import { TypeORMService } from 'src/database/typeorm/typeorm.service'; import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { - GoogleCalendarSyncJobData, - GoogleCalendarSyncJob, -} from 'src/modules/calendar/jobs/google-calendar-sync.job'; -import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; -import { - CalendarChannelWorkspaceEntity, - CalendarChannelVisibility, -} from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountWorkspaceEntity, @@ -34,23 +23,36 @@ import { MessagingMessageListFetchJob, MessagingMessageListFetchJobData, } from 'src/modules/messaging/message-import-manager/jobs/messaging-message-list-fetch.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { WorkspaceDataSource } from 'src/engine/twenty-orm/datasource/workspace.datasource'; +import { InjectWorkspaceDatasource } from 'src/engine/twenty-orm/decorators/inject-workspace-datasource.decorator'; +import { + CalendarChannelWorkspaceEntity, + CalendarChannelVisibility, +} from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { + CalendarEventsImportJobData, + CalendarEventListFetchJob, +} from 'src/modules/calendar/calendar-event-import-manager/jobs/calendar-event-list-fetch.job'; @Injectable() export class GoogleAPIsService { constructor( - private readonly dataSourceService: DataSourceService, - private readonly typeORMService: TypeORMService, - @Inject(MessageQueue.messagingQueue) + @InjectWorkspaceDatasource() + private readonly workspaceDataSource: WorkspaceDataSource, + @InjectMessageQueue(MessageQueue.messagingQueue) private readonly messageQueueService: MessageQueueService, - @Inject(MessageQueue.calendarQueue) + @InjectMessageQueue(MessageQueue.calendarQueue) private readonly calendarQueueService: MessageQueueService, private readonly environmentService: EnvironmentService, @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) private readonly connectedAccountRepository: ConnectedAccountRepository, @InjectObjectMetadataRepository(MessageChannelWorkspaceEntity) private readonly messageChannelRepository: MessageChannelRepository, - @InjectObjectMetadataRepository(CalendarChannelWorkspaceEntity) - private readonly calendarChannelRepository: CalendarChannelRepository, + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, ) {} async refreshGoogleRefreshToken(input: { @@ -70,14 +72,6 @@ export class GoogleAPIsService { messageVisibility, } = input; - const dataSourceMetadata = - await this.dataSourceService.getLastDataSourceMetadataFromWorkspaceIdOrFail( - workspaceId, - ); - - const workspaceDataSource = - await this.typeORMService.connectToDataSource(dataSourceMetadata); - const isCalendarEnabled = this.environmentService.get( 'CALENDAR_PROVIDER_GOOGLE_ENABLED', ); @@ -92,65 +86,67 @@ export class GoogleAPIsService { const existingAccountId = connectedAccounts?.[0]?.id; const newOrExistingConnectedAccountId = existingAccountId ?? v4(); - await workspaceDataSource?.transaction(async (manager: EntityManager) => { - if (!existingAccountId) { - await this.connectedAccountRepository.create( - { - id: newOrExistingConnectedAccountId, - handle, - provider: ConnectedAccountProvider.GOOGLE, - accessToken: input.accessToken, - refreshToken: input.refreshToken, - accountOwnerId: workspaceMemberId, - }, - workspaceId, - manager, - ); - - await this.messageChannelRepository.create( - { - id: v4(), - connectedAccountId: newOrExistingConnectedAccountId, - type: MessageChannelType.EMAIL, - handle, - visibility: - messageVisibility || MessageChannelVisibility.SHARE_EVERYTHING, - syncStatus: MessageChannelSyncStatus.ONGOING, - }, - workspaceId, - manager, - ); + await this.workspaceDataSource.transaction( + async (manager: EntityManager) => { + if (!existingAccountId) { + await this.connectedAccountRepository.create( + { + id: newOrExistingConnectedAccountId, + handle, + provider: ConnectedAccountProvider.GOOGLE, + accessToken: input.accessToken, + refreshToken: input.refreshToken, + accountOwnerId: workspaceMemberId, + }, + workspaceId, + manager, + ); - if (isCalendarEnabled) { - await this.calendarChannelRepository.create( + await this.messageChannelRepository.create( { id: v4(), connectedAccountId: newOrExistingConnectedAccountId, + type: MessageChannelType.EMAIL, handle, visibility: - calendarVisibility || - CalendarChannelVisibility.SHARE_EVERYTHING, + messageVisibility || MessageChannelVisibility.SHARE_EVERYTHING, + syncStatus: MessageChannelSyncStatus.ONGOING, }, workspaceId, manager, ); - } - } else { - await this.connectedAccountRepository.updateAccessTokenAndRefreshToken( - input.accessToken, - input.refreshToken, - newOrExistingConnectedAccountId, - workspaceId, - manager, - ); - await this.messageChannelRepository.resetSync( - newOrExistingConnectedAccountId, - workspaceId, - manager, - ); - } - }); + if (isCalendarEnabled) { + await this.calendarChannelRepository.save( + { + id: v4(), + connectedAccountId: newOrExistingConnectedAccountId, + handle, + visibility: + calendarVisibility || + CalendarChannelVisibility.SHARE_EVERYTHING, + }, + {}, + manager, + ); + } + } else { + await this.connectedAccountRepository.updateAccessTokenAndRefreshToken( + input.accessToken, + input.refreshToken, + newOrExistingConnectedAccountId, + workspaceId, + manager, + ); + + await this.messageChannelRepository.resetSync( + newOrExistingConnectedAccountId, + workspaceId, + manager, + ); + } + }, + ); if (this.environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED')) { const messageChannels = @@ -170,20 +166,22 @@ export class GoogleAPIsService { } } - if ( - this.environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') && - isCalendarEnabled - ) { - await this.calendarQueueService.add<GoogleCalendarSyncJobData>( - GoogleCalendarSyncJob.name, - { - workspaceId, + if (isCalendarEnabled) { + const calendarChannels = await this.calendarChannelRepository.find({ + where: { connectedAccountId: newOrExistingConnectedAccountId, }, - { - retryLimit: 2, - }, - ); + }); + + for (const calendarChannel of calendarChannels) { + await this.calendarQueueService.add<CalendarEventsImportJobData>( + CalendarEventListFetchJob.name, + { + calendarChannelId: calendarChannel.id, + workspaceId, + }, + ); + } } } } diff --git a/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.spec.ts b/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.spec.ts index 370bfeee8985..b6b381ec0d1e 100644 --- a/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.spec.ts +++ b/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.spec.ts @@ -8,6 +8,7 @@ import { EnvironmentService } from 'src/engine/integrations/environment/environm import { SignInUpService } from 'src/engine/core-modules/auth/services/sign-in-up.service'; import { FileUploadService } from 'src/engine/core-modules/file/file-upload/services/file-upload.service'; import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; +import { WorkspaceService } from 'src/engine/core-modules/workspace/services/workspace.service'; describe('SignInUpService', () => { let service: SignInUpService; @@ -40,6 +41,10 @@ describe('SignInUpService', () => { provide: HttpService, useValue: {}, }, + { + provide: WorkspaceService, + useValue: {}, + }, ], }).compile(); diff --git a/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.ts b/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.ts index 3397f2603365..4dafcf12666f 100644 --- a/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.ts +++ b/packages/twenty-server/src/engine/core-modules/auth/services/sign-in-up.service.ts @@ -24,6 +24,7 @@ import { FileUploadService } from 'src/engine/core-modules/file/file-upload/serv import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { getImageBufferFromUrl } from 'src/utils/image'; import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; +import { WorkspaceService } from 'src/engine/core-modules/workspace/services/workspace.service'; export type SignInUpServiceInput = { email: string; @@ -44,6 +45,7 @@ export class SignInUpService { @InjectRepository(User, 'core') private readonly userRepository: Repository<User>, private readonly userWorkspaceService: UserWorkspaceService, + private readonly workspaceService: WorkspaceService, private readonly httpService: HttpService, private readonly environmentService: EnvironmentService, ) {} @@ -142,10 +144,12 @@ export class SignInUpService { ForbiddenException, ); + const isWorkspaceActivated = + await this.workspaceService.isWorkspaceActivated(workspace.id); + assert( - !this.environmentService.get('IS_BILLING_ENABLED') || - workspace.subscriptionStatus !== 'incomplete', - 'Workspace subscription status is incomplete', + isWorkspaceActivated, + 'Workspace is not ready to welcome new members', ForbiddenException, ); @@ -199,7 +203,6 @@ export class SignInUpService { displayName: '', domainName: '', inviteHash: v4(), - subscriptionStatus: 'incomplete', }); const workspace = await this.workspaceRepository.save(workspaceToCreate); diff --git a/packages/twenty-server/src/engine/core-modules/auth/services/token.service.ts b/packages/twenty-server/src/engine/core-modules/auth/services/token.service.ts index c8a8a85dfe12..85bfcebc7974 100644 --- a/packages/twenty-server/src/engine/core-modules/auth/services/token.service.ts +++ b/packages/twenty-server/src/engine/core-modules/auth/services/token.service.ts @@ -147,6 +147,7 @@ export class TokenService { async generateTransientToken( workspaceMemberId: string, + userId: string, workspaceId: string, ): Promise<AuthToken> { const secret = this.environmentService.get('LOGIN_TOKEN_SECRET'); @@ -158,6 +159,7 @@ export class TokenService { const expiresAt = addMilliseconds(new Date().getTime(), ms(expiresIn)); const jwtPayload = { sub: workspaceMemberId, + userId, workspaceId, }; @@ -234,6 +236,7 @@ export class TokenService { async verifyTransientToken(transientToken: string): Promise<{ workspaceMemberId: string; + userId: string; workspaceId: string; }> { const transientTokenSecret = @@ -243,6 +246,7 @@ export class TokenService { return { workspaceMemberId: payload.sub, + userId: payload.userId, workspaceId: payload.workspaceId, }; } diff --git a/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-common.auth.strategy.ts b/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-common.auth.strategy.ts new file mode 100644 index 000000000000..99bd05cab01f --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-common.auth.strategy.ts @@ -0,0 +1,41 @@ +import { PassportStrategy } from '@nestjs/passport'; +import { Injectable } from '@nestjs/common'; + +import { Strategy } from 'passport-google-oauth20'; + +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; + +export type GoogleAPIScopeConfig = { + isCalendarEnabled?: boolean; + isMessagingAliasFetchingEnabled?: boolean; +}; + +@Injectable() +export class GoogleAPIsOauthCommonStrategy extends PassportStrategy( + Strategy, + 'google-apis', +) { + constructor( + environmentService: EnvironmentService, + scopeConfig: GoogleAPIScopeConfig, + ) { + const scopes = [ + 'email', + 'profile', + 'https://www.googleapis.com/auth/gmail.readonly', + 'https://www.googleapis.com/auth/calendar.events', + ]; + + if (scopeConfig?.isMessagingAliasFetchingEnabled) { + scopes.push('https://www.googleapis.com/auth/profile.emails.read'); + } + + super({ + clientID: environmentService.get('AUTH_GOOGLE_CLIENT_ID'), + clientSecret: environmentService.get('AUTH_GOOGLE_CLIENT_SECRET'), + callbackURL: environmentService.get('AUTH_GOOGLE_APIS_CALLBACK_URL'), + scope: scopes, + passReqToCallback: true, + }); + } +} diff --git a/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-exchange-code-for-token.auth.strategy.ts b/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-exchange-code-for-token.auth.strategy.ts new file mode 100644 index 000000000000..047a7f55fa01 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-exchange-code-for-token.auth.strategy.ts @@ -0,0 +1,52 @@ +import { Injectable } from '@nestjs/common'; + +import { VerifyCallback } from 'passport-google-oauth20'; + +import { GoogleAPIsOauthCommonStrategy } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-common.auth.strategy'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; +import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type'; + +export type GoogleAPIScopeConfig = { + isCalendarEnabled?: boolean; + isMessagingAliasFetchingEnabled?: boolean; +}; + +@Injectable() +export class GoogleAPIsOauthExchangeCodeForTokenStrategy extends GoogleAPIsOauthCommonStrategy { + constructor( + environmentService: EnvironmentService, + scopeConfig: GoogleAPIScopeConfig, + ) { + super(environmentService, scopeConfig); + } + + async validate( + request: GoogleAPIsRequest, + accessToken: string, + refreshToken: string, + profile: any, + done: VerifyCallback, + ): Promise<void> { + const { name, emails, photos } = profile; + + const state = + typeof request.query.state === 'string' + ? JSON.parse(request.query.state) + : undefined; + + const user: GoogleAPIsRequest['user'] = { + emails, + firstName: name.givenName, + lastName: name.familyName, + picture: photos?.[0]?.value, + accessToken, + refreshToken, + transientToken: state.transientToken, + redirectLocation: state.redirectLocation, + calendarVisibility: state.calendarVisibility, + messageVisibility: state.messageVisibility, + }; + + done(null, user); + } +} diff --git a/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-request-code.auth.strategy.ts b/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-request-code.auth.strategy.ts new file mode 100644 index 000000000000..128ba607cd45 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis-oauth-request-code.auth.strategy.ts @@ -0,0 +1,35 @@ +import { Injectable } from '@nestjs/common'; + +import { GoogleAPIsOauthCommonStrategy } from 'src/engine/core-modules/auth/strategies/google-apis-oauth-common.auth.strategy'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; + +export type GoogleAPIScopeConfig = { + isCalendarEnabled?: boolean; + isMessagingAliasFetchingEnabled?: boolean; +}; + +@Injectable() +export class GoogleAPIsOauthRequestCodeStrategy extends GoogleAPIsOauthCommonStrategy { + constructor( + environmentService: EnvironmentService, + scopeConfig: GoogleAPIScopeConfig, + ) { + super(environmentService, scopeConfig); + } + + authenticate(req: any, options: any) { + options = { + ...options, + accessType: 'offline', + prompt: 'consent', + state: JSON.stringify({ + transientToken: req.params.transientToken, + redirectLocation: req.params.redirectLocation, + calendarVisibility: req.params.calendarVisibility, + messageVisibility: req.params.messageVisibility, + }), + }; + + return super.authenticate(req, options); + } +} diff --git a/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis.auth.strategy.ts b/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis.auth.strategy.ts deleted file mode 100644 index f4e55d5c4298..000000000000 --- a/packages/twenty-server/src/engine/core-modules/auth/strategies/google-apis.auth.strategy.ts +++ /dev/null @@ -1,110 +0,0 @@ -import { PassportStrategy } from '@nestjs/passport'; -import { Injectable } from '@nestjs/common'; - -import { Strategy, VerifyCallback } from 'passport-google-oauth20'; -import { Request } from 'express'; - -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { CalendarChannelVisibility } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { MessageChannelVisibility } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; - -export type GoogleAPIsRequest = Omit< - Request, - 'user' | 'workspace' | 'cacheVersion' -> & { - user: { - firstName?: string | null; - lastName?: string | null; - email: string; - picture: string | null; - workspaceInviteHash?: string; - accessToken: string; - refreshToken: string; - transientToken: string; - redirectLocation?: string; - calendarVisibility?: CalendarChannelVisibility; - messageVisibility?: MessageChannelVisibility; - }; -}; - -export type GoogleAPIScopeConfig = { - isCalendarEnabled?: boolean; -}; - -@Injectable() -export class GoogleAPIsStrategy extends PassportStrategy( - Strategy, - 'google-apis', -) { - constructor( - environmentService: EnvironmentService, - scopeConfig: GoogleAPIScopeConfig, - ) { - const scope = ['email', 'profile']; - - if (environmentService.get('MESSAGING_PROVIDER_GMAIL_ENABLED')) { - scope.push('https://www.googleapis.com/auth/gmail.readonly'); - } - - if ( - environmentService.get('CALENDAR_PROVIDER_GOOGLE_ENABLED') && - scopeConfig?.isCalendarEnabled - ) { - scope.push('https://www.googleapis.com/auth/calendar.events'); - } - - super({ - clientID: environmentService.get('AUTH_GOOGLE_CLIENT_ID'), - clientSecret: environmentService.get('AUTH_GOOGLE_CLIENT_SECRET'), - callbackURL: environmentService.get('AUTH_GOOGLE_APIS_CALLBACK_URL'), - scope, - passReqToCallback: true, - }); - } - - authenticate(req: any, options: any) { - options = { - ...options, - accessType: 'offline', - prompt: 'consent', - state: JSON.stringify({ - transientToken: req.params.transientToken, - redirectLocation: req.params.redirectLocation, - calendarVisibility: req.params.calendarVisibility, - messageVisibility: req.params.messageVisibility, - }), - }; - - return super.authenticate(req, options); - } - - async validate( - request: GoogleAPIsRequest, - accessToken: string, - refreshToken: string, - profile: any, - done: VerifyCallback, - ): Promise<void> { - const { name, emails, photos } = profile; - - const state = - typeof request.query.state === 'string' - ? JSON.parse(request.query.state) - : undefined; - - const user: GoogleAPIsRequest['user'] = { - email: emails[0].value, - firstName: name.givenName, - lastName: name.familyName, - picture: photos?.[0]?.value, - accessToken, - refreshToken, - transientToken: state.transientToken, - redirectLocation: state.redirectLocation, - calendarVisibility: state.calendarVisibility, - messageVisibility: state.messageVisibility, - }; - - done(null, user); - } -} diff --git a/packages/twenty-server/src/engine/core-modules/auth/types/google-api-request.type.ts b/packages/twenty-server/src/engine/core-modules/auth/types/google-api-request.type.ts new file mode 100644 index 000000000000..f8d5f609d4d1 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/types/google-api-request.type.ts @@ -0,0 +1,23 @@ +import { Request } from 'express'; + +import { CalendarChannelVisibility } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { MessageChannelVisibility } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; + +export type GoogleAPIsRequest = Omit< + Request, + 'user' | 'workspace' | 'cacheVersion' +> & { + user: { + firstName?: string | null; + lastName?: string | null; + emails: { value: string }[]; + picture: string | null; + workspaceInviteHash?: string; + accessToken: string; + refreshToken: string; + transientToken: string; + redirectLocation?: string; + calendarVisibility?: CalendarChannelVisibility; + messageVisibility?: MessageChannelVisibility; + }; +}; diff --git a/packages/twenty-server/src/engine/core-modules/auth/utils/__tests__/google-apis-set-request-extra-params.util.spec.ts b/packages/twenty-server/src/engine/core-modules/auth/utils/__tests__/google-apis-set-request-extra-params.util.spec.ts new file mode 100644 index 000000000000..052f4ff04d3a --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/utils/__tests__/google-apis-set-request-extra-params.util.spec.ts @@ -0,0 +1,40 @@ +import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type'; +import { setRequestExtraParams } from 'src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util'; +import { CalendarChannelVisibility } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { MessageChannelVisibility } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; + +describe('googleApisSetRequestExtraParams', () => { + it('should set request extra params', () => { + const request = { + params: {}, + } as GoogleAPIsRequest; + + setRequestExtraParams(request, { + transientToken: 'abc', + redirectLocation: '/test', + calendarVisibility: CalendarChannelVisibility.SHARE_EVERYTHING, + messageVisibility: MessageChannelVisibility.SHARE_EVERYTHING, + }); + + expect(request.params).toEqual({ + transientToken: 'abc', + redirectLocation: '/test', + calendarVisibility: CalendarChannelVisibility.SHARE_EVERYTHING, + messageVisibility: MessageChannelVisibility.SHARE_EVERYTHING, + }); + }); + + it('should throw error if transientToken is not provided', () => { + const request = { + params: {}, + } as GoogleAPIsRequest; + + expect(() => { + setRequestExtraParams(request, { + redirectLocation: '/test', + calendarVisibility: CalendarChannelVisibility.SHARE_EVERYTHING, + messageVisibility: MessageChannelVisibility.SHARE_EVERYTHING, + }); + }).toThrow('transientToken is required'); + }); +}); diff --git a/packages/twenty-server/src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util.ts b/packages/twenty-server/src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util.ts new file mode 100644 index 000000000000..668426c23bc1 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/auth/utils/google-apis-set-request-extra-params.util.ts @@ -0,0 +1,38 @@ +import { GoogleAPIsRequest } from 'src/engine/core-modules/auth/types/google-api-request.type'; + +type GoogleAPIsRequestExtraParams = { + transientToken?: string; + redirectLocation?: string; + calendarVisibility?: string; + messageVisibility?: string; +}; + +export const setRequestExtraParams = ( + request: GoogleAPIsRequest, + params: GoogleAPIsRequestExtraParams, +): void => { + const { + transientToken, + redirectLocation, + calendarVisibility, + messageVisibility, + } = params; + + if (!transientToken) { + throw new Error('transientToken is required'); + } + + request.params.transientToken = transientToken; + + if (redirectLocation) { + request.params.redirectLocation = redirectLocation; + } + + if (calendarVisibility) { + request.params.calendarVisibility = calendarVisibility; + } + + if (messageVisibility) { + request.params.messageVisibility = messageVisibility; + } +}; diff --git a/packages/twenty-server/src/engine/core-modules/billing/billing.controller.ts b/packages/twenty-server/src/engine/core-modules/billing/billing.controller.ts index 775604f7d295..4e71805b9c93 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/billing.controller.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/billing.controller.ts @@ -11,9 +11,9 @@ import { import { Response } from 'express'; import { - BillingService, + BillingWorkspaceService, WebhookEvent, -} from 'src/engine/core-modules/billing/billing.service'; +} from 'src/engine/core-modules/billing/billing.workspace-service'; import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service'; @Controller('billing') @@ -22,7 +22,7 @@ export class BillingController { constructor( private readonly stripeService: StripeService, - private readonly billingService: BillingService, + private readonly billingWorkspaceService: BillingWorkspaceService, ) {} @Post('/webhooks') @@ -42,7 +42,7 @@ export class BillingController { ); if (event.type === WebhookEvent.SETUP_INTENT_SUCCEEDED) { - await this.billingService.handleUnpaidInvoices(event.data); + await this.billingWorkspaceService.handleUnpaidInvoices(event.data); } if ( @@ -58,7 +58,7 @@ export class BillingController { return; } - await this.billingService.upsertBillingSubscription( + await this.billingWorkspaceService.upsertBillingSubscription( workspaceId, event.data, ); diff --git a/packages/twenty-server/src/engine/core-modules/billing/billing.module.ts b/packages/twenty-server/src/engine/core-modules/billing/billing.module.ts index 4be921b78c04..a4462607a737 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/billing.module.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/billing.module.ts @@ -10,18 +10,30 @@ import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { BillingResolver } from 'src/engine/core-modules/billing/billing.resolver'; import { BillingWorkspaceMemberListener } from 'src/engine/core-modules/billing/listeners/billing-workspace-member.listener'; import { UserWorkspaceModule } from 'src/engine/core-modules/user-workspace/user-workspace.module'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service'; @Module({ imports: [ StripeModule, UserWorkspaceModule, TypeOrmModule.forFeature( - [BillingSubscription, BillingSubscriptionItem, Workspace], + [ + BillingSubscription, + BillingSubscriptionItem, + Workspace, + FeatureFlagEntity, + ], 'core', ), ], controllers: [BillingController], - providers: [BillingService, BillingResolver, BillingWorkspaceMemberListener], - exports: [BillingService], + providers: [ + BillingService, + BillingWorkspaceService, + BillingResolver, + BillingWorkspaceMemberListener, + ], + exports: [BillingService, BillingWorkspaceService], }) export class BillingModule {} diff --git a/packages/twenty-server/src/engine/core-modules/billing/billing.resolver.ts b/packages/twenty-server/src/engine/core-modules/billing/billing.resolver.ts index 8766c4adc37e..40c90fbc2cea 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/billing.resolver.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/billing.resolver.ts @@ -3,8 +3,8 @@ import { UseGuards } from '@nestjs/common'; import { AvailableProduct, - BillingService, -} from 'src/engine/core-modules/billing/billing.service'; + BillingWorkspaceService, +} from 'src/engine/core-modules/billing/billing.workspace-service'; import { ProductInput } from 'src/engine/core-modules/billing/dto/product.input'; import { assert } from 'src/utils/assert'; import { ProductPricesEntity } from 'src/engine/core-modules/billing/dto/product-prices.entity'; @@ -18,11 +18,14 @@ import { UpdateBillingEntity } from 'src/engine/core-modules/billing/dto/update- @Resolver() export class BillingResolver { - constructor(private readonly billingService: BillingService) {} + constructor( + private readonly billingWorkspaceService: BillingWorkspaceService, + ) {} @Query(() => ProductPricesEntity) async getProductPrices(@Args() { product }: ProductInput) { - const stripeProductId = this.billingService.getProductStripeId(product); + const stripeProductId = + this.billingWorkspaceService.getProductStripeId(product); assert( stripeProductId, @@ -32,7 +35,7 @@ export class BillingResolver { ); const productPrices = - await this.billingService.getProductPrices(stripeProductId); + await this.billingWorkspaceService.getProductPrices(stripeProductId); return { totalNumberOfPrices: productPrices.length, @@ -47,7 +50,7 @@ export class BillingResolver { @Args() { returnUrlPath }: BillingSessionInput, ) { return { - url: await this.billingService.computeBillingPortalSessionURL( + url: await this.billingWorkspaceService.computeBillingPortalSessionURL( user.defaultWorkspaceId, returnUrlPath, ), @@ -60,7 +63,7 @@ export class BillingResolver { @AuthUser() user: User, @Args() { recurringInterval, successUrlPath }: CheckoutSessionInput, ) { - const stripeProductId = this.billingService.getProductStripeId( + const stripeProductId = this.billingWorkspaceService.getProductStripeId( AvailableProduct.BasePlan, ); @@ -70,7 +73,7 @@ export class BillingResolver { ); const productPrices = - await this.billingService.getProductPrices(stripeProductId); + await this.billingWorkspaceService.getProductPrices(stripeProductId); const stripePriceId = productPrices.filter( (price) => price.recurringInterval === recurringInterval, @@ -82,7 +85,7 @@ export class BillingResolver { ); return { - url: await this.billingService.computeCheckoutSessionURL( + url: await this.billingWorkspaceService.computeCheckoutSessionURL( user, stripePriceId, successUrlPath, @@ -93,7 +96,7 @@ export class BillingResolver { @Mutation(() => UpdateBillingEntity) @UseGuards(JwtAuthGuard) async updateBillingSubscription(@AuthUser() user: User) { - await this.billingService.updateBillingSubscription(user); + await this.billingWorkspaceService.updateBillingSubscription(user); return { success: true }; } diff --git a/packages/twenty-server/src/engine/core-modules/billing/billing.service.ts b/packages/twenty-server/src/engine/core-modules/billing/billing.service.ts index 8228788f4637..34cae6a376e4 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/billing.service.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/billing.service.ts @@ -1,308 +1,71 @@ import { Injectable, Logger } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; -import Stripe from 'stripe'; -import { Not, Repository } from 'typeorm'; +import { In, Repository } from 'typeorm'; import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service'; -import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; -import { BillingSubscriptionItem } from 'src/engine/core-modules/billing/entities/billing-subscription-item.entity'; +import { + BillingSubscription, + SubscriptionStatus, +} from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; -import { ProductPriceEntity } from 'src/engine/core-modules/billing/dto/product-price.entity'; -import { User } from 'src/engine/core-modules/user/user.entity'; -import { assert } from 'src/utils/assert'; -import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; - -export enum AvailableProduct { - BasePlan = 'base-plan', -} - -export enum WebhookEvent { - CUSTOMER_SUBSCRIPTION_CREATED = 'customer.subscription.created', - CUSTOMER_SUBSCRIPTION_UPDATED = 'customer.subscription.updated', - CUSTOMER_SUBSCRIPTION_DELETED = 'customer.subscription.deleted', - SETUP_INTENT_SUCCEEDED = 'setup_intent.succeeded', -} +import { + FeatureFlagEntity, + FeatureFlagKeys, +} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; @Injectable() export class BillingService { protected readonly logger = new Logger(BillingService.name); constructor( - private readonly stripeService: StripeService, - private readonly userWorkspaceService: UserWorkspaceService, private readonly environmentService: EnvironmentService, @InjectRepository(BillingSubscription, 'core') private readonly billingSubscriptionRepository: Repository<BillingSubscription>, - @InjectRepository(BillingSubscriptionItem, 'core') - private readonly billingSubscriptionItemRepository: Repository<BillingSubscriptionItem>, + @InjectRepository(FeatureFlagEntity, 'core') + private readonly featureFlagRepository: Repository<FeatureFlagEntity>, @InjectRepository(Workspace, 'core') private readonly workspaceRepository: Repository<Workspace>, ) {} - getProductStripeId(product: AvailableProduct) { - if (product === AvailableProduct.BasePlan) { - return this.environmentService.get('BILLING_STRIPE_BASE_PLAN_PRODUCT_ID'); - } - } - - async getProductPrices(stripeProductId: string) { - const productPrices = - await this.stripeService.getProductPrices(stripeProductId); - - return this.formatProductPrices(productPrices.data); - } - - formatProductPrices(prices: Stripe.Price[]) { - const result: Record<string, ProductPriceEntity> = {}; - - prices.forEach((item) => { - const interval = item.recurring?.interval; - - if (!interval || !item.unit_amount) { - return; - } - if ( - !result[interval] || - item.created > (result[interval]?.created || 0) - ) { - result[interval] = { - unitAmount: item.unit_amount, - recurringInterval: interval, - created: item.created, - stripePriceId: item.id, - }; - } - }); - - return Object.values(result).sort((a, b) => a.unitAmount - b.unitAmount); - } - - async getCurrentBillingSubscription(criteria: { - workspaceId?: string; - stripeCustomerId?: string; - }) { - const notCanceledSubscriptions = - await this.billingSubscriptionRepository.find({ - where: { ...criteria, status: Not('canceled') }, - relations: ['billingSubscriptionItems'], - }); - - assert( - notCanceledSubscriptions.length <= 1, - `More than on not canceled subscription for workspace ${criteria.workspaceId}`, - ); - - return notCanceledSubscriptions?.[0]; - } - - async getBillingSubscription(stripeSubscriptionId: string) { - return this.billingSubscriptionRepository.findOneOrFail({ - where: { stripeSubscriptionId }, - }); - } - - async getStripeCustomerId(workspaceId: string) { - const subscriptions = await this.billingSubscriptionRepository.find({ - where: { workspaceId }, - }); - - return subscriptions?.[0]?.stripeCustomerId; - } - - async getBillingSubscriptionItem( - workspaceId: string, - stripeProductId = this.environmentService.get( - 'BILLING_STRIPE_BASE_PLAN_PRODUCT_ID', - ), - ) { - const billingSubscription = await this.getCurrentBillingSubscription({ - workspaceId, - }); - - if (!billingSubscription) { - throw new Error( - `Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`, - ); - } - - const billingSubscriptionItem = - billingSubscription.billingSubscriptionItems.filter( - (billingSubscriptionItem) => - billingSubscriptionItem.stripeProductId === stripeProductId, - )?.[0]; - - if (!billingSubscriptionItem) { - throw new Error( - `Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`, + async getActiveSubscriptionWorkspaceIds() { + if (!this.environmentService.get('IS_BILLING_ENABLED')) { + return (await this.workspaceRepository.find({ select: ['id'] })).map( + (workspace) => workspace.id, ); } - return billingSubscriptionItem; - } - - async computeBillingPortalSessionURL( - workspaceId: string, - returnUrlPath?: string, - ) { - const stripeCustomerId = await this.getStripeCustomerId(workspaceId); - - if (!stripeCustomerId) { - return; - } - - const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL'); - const returnUrl = returnUrlPath - ? frontBaseUrl + returnUrlPath - : frontBaseUrl; - - const session = await this.stripeService.createBillingPortalSession( - stripeCustomerId, - returnUrl, - ); - - assert(session.url, 'Error: missing billingPortal.session.url'); - - return session.url; - } - - async updateBillingSubscription(user: User) { - const billingSubscription = await this.getCurrentBillingSubscription({ - workspaceId: user.defaultWorkspaceId, - }); - const newInterval = - billingSubscription?.interval === 'year' ? 'month' : 'year'; - const billingSubscriptionItem = await this.getBillingSubscriptionItem( - user.defaultWorkspaceId, - ); - const stripeProductId = this.getProductStripeId(AvailableProduct.BasePlan); - - if (!stripeProductId) { - throw new Error('Stripe product id not found for basePlan'); - } - const productPrices = await this.getProductPrices(stripeProductId); - - const stripePriceId = productPrices.filter( - (price) => price.recurringInterval === newInterval, - )?.[0]?.stripePriceId; - - await this.stripeService.updateBillingSubscriptionItem( - billingSubscriptionItem, - stripePriceId, - ); - } - - async computeCheckoutSessionURL( - user: User, - priceId: string, - successUrlPath?: string, - ): Promise<string> { - const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL'); - const successUrl = successUrlPath - ? frontBaseUrl + successUrlPath - : frontBaseUrl; - - const quantity = - (await this.userWorkspaceService.getWorkspaceMemberCount( - user.defaultWorkspaceId, - )) || 1; - - const stripeCustomerId = ( - await this.billingSubscriptionRepository.findOneBy({ - workspaceId: user.defaultWorkspaceId, - }) - )?.stripeCustomerId; - - const session = await this.stripeService.createCheckoutSession( - user, - priceId, - quantity, - successUrl, - frontBaseUrl, - stripeCustomerId, - ); - - assert(session.url, 'Error: missing checkout.session.url'); - - return session.url; - } - - async deleteSubscription(workspaceId: string) { - const subscriptionToCancel = await this.getCurrentBillingSubscription({ - workspaceId, - }); - - if (subscriptionToCancel) { - await this.stripeService.cancelSubscription( - subscriptionToCancel.stripeSubscriptionId, - ); - await this.billingSubscriptionRepository.delete(subscriptionToCancel.id); - } - } - - async handleUnpaidInvoices(data: Stripe.SetupIntentSucceededEvent.Data) { - const billingSubscription = await this.getCurrentBillingSubscription({ - stripeCustomerId: data.object.customer as string, - }); - - if (billingSubscription?.status === 'unpaid') { - await this.stripeService.collectLastInvoice( - billingSubscription.stripeSubscriptionId, - ); - } - } - - async upsertBillingSubscription( - workspaceId: string, - data: - | Stripe.CustomerSubscriptionUpdatedEvent.Data - | Stripe.CustomerSubscriptionCreatedEvent.Data - | Stripe.CustomerSubscriptionDeletedEvent.Data, - ) { - const workspace = this.workspaceRepository.find({ - where: { id: workspaceId }, + const activeSubscriptions = await this.billingSubscriptionRepository.find({ + where: { + status: In([ + SubscriptionStatus.Active, + SubscriptionStatus.Trialing, + SubscriptionStatus.PastDue, + ]), + }, + select: ['workspaceId'], }); - if (!workspace) { - return; - } - - await this.workspaceRepository.update(workspaceId, { - subscriptionStatus: data.object.status, + const freeAccessFeatureFlags = await this.featureFlagRepository.find({ + where: { + key: FeatureFlagKeys.IsFreeAccessEnabled, + value: true, + }, + select: ['workspaceId'], }); - await this.billingSubscriptionRepository.upsert( - { - workspaceId: workspaceId, - stripeCustomerId: data.object.customer as string, - stripeSubscriptionId: data.object.id, - status: data.object.status, - interval: data.object.items.data[0].plan.interval, - }, - { - conflictPaths: ['stripeSubscriptionId'], - skipUpdateIfNoValuesChanged: true, - }, + const activeWorkspaceIdsBasedOnSubscriptions = activeSubscriptions.map( + (subscription) => subscription.workspaceId, ); - const billingSubscription = await this.getBillingSubscription( - data.object.id, + const activeWorkspaceIdsBasedOnFeatureFlags = freeAccessFeatureFlags.map( + (featureFlag) => featureFlag.workspaceId, ); - await this.billingSubscriptionItemRepository.upsert( - data.object.items.data.map((item) => { - return { - billingSubscriptionId: billingSubscription.id, - stripeProductId: item.price.product as string, - stripePriceId: item.price.id, - stripeSubscriptionItemId: item.id, - quantity: item.quantity, - }; - }), - { - conflictPaths: ['billingSubscriptionId', 'stripeProductId'], - skipUpdateIfNoValuesChanged: true, - }, + return Array.from( + new Set([ + ...activeWorkspaceIdsBasedOnSubscriptions, + ...activeWorkspaceIdsBasedOnFeatureFlags, + ]), ); } } diff --git a/packages/twenty-server/src/engine/core-modules/billing/billing.workspace-service.ts b/packages/twenty-server/src/engine/core-modules/billing/billing.workspace-service.ts new file mode 100644 index 000000000000..d912751ea7e6 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/billing/billing.workspace-service.ts @@ -0,0 +1,332 @@ +import { Injectable, Logger } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; + +import Stripe from 'stripe'; +import { Not, Repository } from 'typeorm'; + +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; +import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service'; +import { + BillingSubscription, + SubscriptionInterval, + SubscriptionStatus, +} from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; +import { BillingSubscriptionItem } from 'src/engine/core-modules/billing/entities/billing-subscription-item.entity'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { ProductPriceEntity } from 'src/engine/core-modules/billing/dto/product-price.entity'; +import { User } from 'src/engine/core-modules/user/user.entity'; +import { assert } from 'src/utils/assert'; +import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; +import { + FeatureFlagEntity, + FeatureFlagKeys, +} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { BillingService } from 'src/engine/core-modules/billing/billing.service'; + +export enum AvailableProduct { + BasePlan = 'base-plan', +} + +export enum WebhookEvent { + CUSTOMER_SUBSCRIPTION_CREATED = 'customer.subscription.created', + CUSTOMER_SUBSCRIPTION_UPDATED = 'customer.subscription.updated', + CUSTOMER_SUBSCRIPTION_DELETED = 'customer.subscription.deleted', + SETUP_INTENT_SUCCEEDED = 'setup_intent.succeeded', +} + +@Injectable() +export class BillingWorkspaceService { + protected readonly logger = new Logger(BillingService.name); + constructor( + private readonly stripeService: StripeService, + private readonly userWorkspaceService: UserWorkspaceService, + private readonly environmentService: EnvironmentService, + @InjectRepository(BillingSubscription, 'core') + private readonly billingSubscriptionRepository: Repository<BillingSubscription>, + @InjectRepository(FeatureFlagEntity, 'core') + private readonly featureFlagRepository: Repository<FeatureFlagEntity>, + @InjectRepository(BillingSubscriptionItem, 'core') + private readonly billingSubscriptionItemRepository: Repository<BillingSubscriptionItem>, + @InjectRepository(Workspace, 'core') + private readonly workspaceRepository: Repository<Workspace>, + ) {} + + async isBillingEnabledForWorkspace(workspaceId: string) { + const isFreeAccessEnabled = await this.featureFlagRepository.findOneBy({ + workspaceId, + key: FeatureFlagKeys.IsFreeAccessEnabled, + value: true, + }); + + return ( + !isFreeAccessEnabled && this.environmentService.get('IS_BILLING_ENABLED') + ); + } + + getProductStripeId(product: AvailableProduct) { + if (product === AvailableProduct.BasePlan) { + return this.environmentService.get('BILLING_STRIPE_BASE_PLAN_PRODUCT_ID'); + } + } + + async getProductPrices(stripeProductId: string) { + const productPrices = + await this.stripeService.getProductPrices(stripeProductId); + + return this.formatProductPrices(productPrices.data); + } + + formatProductPrices(prices: Stripe.Price[]) { + const result: Record<string, ProductPriceEntity> = {}; + + prices.forEach((item) => { + const interval = item.recurring?.interval; + + if (!interval || !item.unit_amount) { + return; + } + if ( + !result[interval] || + item.created > (result[interval]?.created || 0) + ) { + result[interval] = { + unitAmount: item.unit_amount, + recurringInterval: interval, + created: item.created, + stripePriceId: item.id, + }; + } + }); + + return Object.values(result).sort((a, b) => a.unitAmount - b.unitAmount); + } + + async getCurrentBillingSubscription(criteria: { + workspaceId?: string; + stripeCustomerId?: string; + }) { + const notCanceledSubscriptions = + await this.billingSubscriptionRepository.find({ + where: { ...criteria, status: Not(SubscriptionStatus.Canceled) }, + relations: ['billingSubscriptionItems'], + }); + + assert( + notCanceledSubscriptions.length <= 1, + `More than one not canceled subscription for workspace ${criteria.workspaceId}`, + ); + + return notCanceledSubscriptions?.[0]; + } + + async getBillingSubscription(stripeSubscriptionId: string) { + return this.billingSubscriptionRepository.findOneOrFail({ + where: { stripeSubscriptionId }, + }); + } + + async getStripeCustomerId(workspaceId: string) { + const subscriptions = await this.billingSubscriptionRepository.find({ + where: { workspaceId }, + }); + + return subscriptions?.[0]?.stripeCustomerId; + } + + async getBillingSubscriptionItem( + workspaceId: string, + stripeProductId = this.environmentService.get( + 'BILLING_STRIPE_BASE_PLAN_PRODUCT_ID', + ), + ) { + const billingSubscription = await this.getCurrentBillingSubscription({ + workspaceId, + }); + + if (!billingSubscription) { + throw new Error( + `Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`, + ); + } + + const billingSubscriptionItem = + billingSubscription.billingSubscriptionItems.filter( + (billingSubscriptionItem) => + billingSubscriptionItem.stripeProductId === stripeProductId, + )?.[0]; + + if (!billingSubscriptionItem) { + throw new Error( + `Cannot find billingSubscriptionItem for product ${stripeProductId} for workspace ${workspaceId}`, + ); + } + + return billingSubscriptionItem; + } + + async computeBillingPortalSessionURL( + workspaceId: string, + returnUrlPath?: string, + ) { + const stripeCustomerId = await this.getStripeCustomerId(workspaceId); + + if (!stripeCustomerId) { + return; + } + + const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL'); + const returnUrl = returnUrlPath + ? frontBaseUrl + returnUrlPath + : frontBaseUrl; + + const session = await this.stripeService.createBillingPortalSession( + stripeCustomerId, + returnUrl, + ); + + assert(session.url, 'Error: missing billingPortal.session.url'); + + return session.url; + } + + async updateBillingSubscription(user: User) { + const billingSubscription = await this.getCurrentBillingSubscription({ + workspaceId: user.defaultWorkspaceId, + }); + const newInterval = + billingSubscription?.interval === SubscriptionInterval.Year + ? SubscriptionInterval.Month + : SubscriptionInterval.Year; + const billingSubscriptionItem = await this.getBillingSubscriptionItem( + user.defaultWorkspaceId, + ); + const stripeProductId = this.getProductStripeId(AvailableProduct.BasePlan); + + if (!stripeProductId) { + throw new Error('Stripe product id not found for basePlan'); + } + const productPrices = await this.getProductPrices(stripeProductId); + + const stripePriceId = productPrices.filter( + (price) => price.recurringInterval === newInterval, + )?.[0]?.stripePriceId; + + await this.stripeService.updateBillingSubscriptionItem( + billingSubscriptionItem, + stripePriceId, + ); + } + + async computeCheckoutSessionURL( + user: User, + priceId: string, + successUrlPath?: string, + ): Promise<string> { + const frontBaseUrl = this.environmentService.get('FRONT_BASE_URL'); + const successUrl = successUrlPath + ? frontBaseUrl + successUrlPath + : frontBaseUrl; + + const quantity = + (await this.userWorkspaceService.getWorkspaceMemberCount()) || 1; + + const stripeCustomerId = ( + await this.billingSubscriptionRepository.findOneBy({ + workspaceId: user.defaultWorkspaceId, + }) + )?.stripeCustomerId; + + const session = await this.stripeService.createCheckoutSession( + user, + priceId, + quantity, + successUrl, + frontBaseUrl, + stripeCustomerId, + ); + + assert(session.url, 'Error: missing checkout.session.url'); + + return session.url; + } + + async deleteSubscription(workspaceId: string) { + const subscriptionToCancel = await this.getCurrentBillingSubscription({ + workspaceId, + }); + + if (subscriptionToCancel) { + await this.stripeService.cancelSubscription( + subscriptionToCancel.stripeSubscriptionId, + ); + await this.billingSubscriptionRepository.delete(subscriptionToCancel.id); + } + } + + async handleUnpaidInvoices(data: Stripe.SetupIntentSucceededEvent.Data) { + const billingSubscription = await this.getCurrentBillingSubscription({ + stripeCustomerId: data.object.customer as string, + }); + + if (billingSubscription?.status === 'unpaid') { + await this.stripeService.collectLastInvoice( + billingSubscription.stripeSubscriptionId, + ); + } + } + + async upsertBillingSubscription( + workspaceId: string, + data: + | Stripe.CustomerSubscriptionUpdatedEvent.Data + | Stripe.CustomerSubscriptionCreatedEvent.Data + | Stripe.CustomerSubscriptionDeletedEvent.Data, + ) { + const workspace = this.workspaceRepository.find({ + where: { id: workspaceId }, + }); + + if (!workspace) { + return; + } + + await this.billingSubscriptionRepository.upsert( + { + workspaceId: workspaceId, + stripeCustomerId: data.object.customer as string, + stripeSubscriptionId: data.object.id, + status: data.object.status, + interval: data.object.items.data[0].plan.interval, + }, + { + conflictPaths: ['stripeSubscriptionId'], + skipUpdateIfNoValuesChanged: true, + }, + ); + + const billingSubscription = await this.getBillingSubscription( + data.object.id, + ); + + await this.billingSubscriptionItemRepository.upsert( + data.object.items.data.map((item) => { + return { + billingSubscriptionId: billingSubscription.id, + stripeProductId: item.price.product as string, + stripePriceId: item.price.id, + stripeSubscriptionItemId: item.id, + quantity: item.quantity, + }; + }), + { + conflictPaths: ['billingSubscriptionId', 'stripeProductId'], + skipUpdateIfNoValuesChanged: true, + }, + ); + + await this.featureFlagRepository.delete({ + workspaceId, + key: FeatureFlagKeys.IsFreeAccessEnabled, + }); + } +} diff --git a/packages/twenty-server/src/engine/core-modules/billing/dto/checkout-session.input.ts b/packages/twenty-server/src/engine/core-modules/billing/dto/checkout-session.input.ts index 778f0671ee6c..48738b85f8dd 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/dto/checkout-session.input.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/dto/checkout-session.input.ts @@ -3,9 +3,11 @@ import { ArgsType, Field } from '@nestjs/graphql'; import { IsNotEmpty, IsOptional, IsString } from 'class-validator'; import Stripe from 'stripe'; +import { SubscriptionInterval } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; + @ArgsType() export class CheckoutSessionInput { - @Field(() => String) + @Field(() => SubscriptionInterval) @IsString() @IsNotEmpty() recurringInterval: Stripe.Price.Recurring.Interval; diff --git a/packages/twenty-server/src/engine/core-modules/billing/dto/product-price.entity.ts b/packages/twenty-server/src/engine/core-modules/billing/dto/product-price.entity.ts index 69fc80011f3c..d8c2d6d434f3 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/dto/product-price.entity.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/dto/product-price.entity.ts @@ -2,9 +2,11 @@ import { Field, ObjectType } from '@nestjs/graphql'; import Stripe from 'stripe'; +import { SubscriptionInterval } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; + @ObjectType() export class ProductPriceEntity { - @Field(() => String) + @Field(() => SubscriptionInterval) recurringInterval: Stripe.Price.Recurring.Interval; @Field(() => Number) diff --git a/packages/twenty-server/src/engine/core-modules/billing/dto/product.input.ts b/packages/twenty-server/src/engine/core-modules/billing/dto/product.input.ts index 089d18ba0e05..1bab951e2eb1 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/dto/product.input.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/dto/product.input.ts @@ -2,7 +2,7 @@ import { ArgsType, Field } from '@nestjs/graphql'; import { IsNotEmpty, IsString } from 'class-validator'; -import { AvailableProduct } from 'src/engine/core-modules/billing/billing.service'; +import { AvailableProduct } from 'src/engine/core-modules/billing/billing.workspace-service'; @ArgsType() export class ProductInput { diff --git a/packages/twenty-server/src/engine/core-modules/billing/entities/billing-subscription.entity.ts b/packages/twenty-server/src/engine/core-modules/billing/entities/billing-subscription.entity.ts index 4da2b3ef5688..509d5250b265 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/entities/billing-subscription.entity.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/entities/billing-subscription.entity.ts @@ -1,4 +1,4 @@ -import { Field, ObjectType } from '@nestjs/graphql'; +import { Field, ObjectType, registerEnumType } from '@nestjs/graphql'; import { Column, @@ -18,6 +18,27 @@ import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { BillingSubscriptionItem } from 'src/engine/core-modules/billing/entities/billing-subscription-item.entity'; import { UUIDScalarType } from 'src/engine/api/graphql/workspace-schema-builder/graphql-types/scalars'; +export enum SubscriptionStatus { + Active = 'active', + Canceled = 'canceled', + Incomplete = 'incomplete', + IncompleteExpired = 'incomplete_expired', + PastDue = 'past_due', + Paused = 'paused', + Trialing = 'trialing', + Unpaid = 'unpaid', +} + +export enum SubscriptionInterval { + Day = 'day', + Month = 'month', + Week = 'week', + Year = 'year', +} + +registerEnumType(SubscriptionStatus, { name: 'SubscriptionStatus' }); +registerEnumType(SubscriptionInterval, { name: 'SubscriptionInterval' }); + @Entity({ name: 'billingSubscription', schema: 'core' }) @ObjectType('BillingSubscription') export class BillingSubscription { @@ -49,12 +70,20 @@ export class BillingSubscription { @Column({ unique: true, nullable: false }) stripeSubscriptionId: string; - @Field(() => String) - @Column({ type: 'text', nullable: false }) + @Field(() => SubscriptionStatus) + @Column({ + type: 'enum', + enum: Object.values(SubscriptionStatus), + nullable: false, + }) status: Stripe.Subscription.Status; - @Field(() => String, { nullable: true }) - @Column({ type: 'text', nullable: true }) + @Field(() => SubscriptionInterval, { nullable: true }) + @Column({ + type: 'enum', + enum: Object.values(SubscriptionInterval), + nullable: true, + }) interval: Stripe.Price.Recurring.Interval; @OneToMany( diff --git a/packages/twenty-server/src/engine/core-modules/billing/jobs/update-subscription.job.ts b/packages/twenty-server/src/engine/core-modules/billing/jobs/update-subscription.job.ts index cf3ff0367a8d..9302c50f8d56 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/jobs/update-subscription.job.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/jobs/update-subscription.job.ts @@ -1,25 +1,30 @@ -import { Injectable, Logger } from '@nestjs/common'; +import { Logger, Scope } from '@nestjs/common'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { BillingService } from 'src/engine/core-modules/billing/billing.service'; +import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service'; import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; import { StripeService } from 'src/engine/core-modules/billing/stripe/stripe.service'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; export type UpdateSubscriptionJobData = { workspaceId: string }; -@Injectable() -export class UpdateSubscriptionJob - implements MessageQueueJob<UpdateSubscriptionJobData> -{ + +@Processor({ + queueName: MessageQueue.billingQueue, + scope: Scope.REQUEST, +}) +export class UpdateSubscriptionJob { protected readonly logger = new Logger(UpdateSubscriptionJob.name); + constructor( - private readonly billingService: BillingService, + private readonly billingWorkspaceService: BillingWorkspaceService, private readonly userWorkspaceService: UserWorkspaceService, private readonly stripeService: StripeService, ) {} + @Process(UpdateSubscriptionJob.name) async handle(data: UpdateSubscriptionJobData): Promise<void> { const workspaceMembersCount = - await this.userWorkspaceService.getWorkspaceMemberCount(data.workspaceId); + await this.userWorkspaceService.getWorkspaceMemberCount(); if (!workspaceMembersCount || workspaceMembersCount <= 0) { return; @@ -27,7 +32,9 @@ export class UpdateSubscriptionJob try { const billingSubscriptionItem = - await this.billingService.getBillingSubscriptionItem(data.workspaceId); + await this.billingWorkspaceService.getBillingSubscriptionItem( + data.workspaceId, + ); await this.stripeService.updateSubscriptionItem( billingSubscriptionItem.stripeSubscriptionItemId, diff --git a/packages/twenty-server/src/engine/core-modules/billing/listeners/billing-workspace-member.listener.ts b/packages/twenty-server/src/engine/core-modules/billing/listeners/billing-workspace-member.listener.ts index 77365b74aa8e..4fadaaf5ed7c 100644 --- a/packages/twenty-server/src/engine/core-modules/billing/listeners/billing-workspace-member.listener.ts +++ b/packages/twenty-server/src/engine/core-modules/billing/listeners/billing-workspace-member.listener.ts @@ -1,4 +1,4 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; @@ -9,14 +9,15 @@ import { UpdateSubscriptionJob, UpdateSubscriptionJobData, } from 'src/engine/core-modules/billing/jobs/update-subscription.job'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service'; @Injectable() export class BillingWorkspaceMemberListener { constructor( - @Inject(MessageQueue.billingQueue) + @InjectMessageQueue(MessageQueue.billingQueue) private readonly messageQueueService: MessageQueueService, - private readonly environmentService: EnvironmentService, + private readonly billingWorkspaceService: BillingWorkspaceService, ) {} @OnEvent('workspaceMember.created') @@ -24,7 +25,12 @@ export class BillingWorkspaceMemberListener { async handleCreateOrDeleteEvent( payload: ObjectRecordCreateEvent<WorkspaceMemberWorkspaceEntity>, ) { - if (!this.environmentService.get('IS_BILLING_ENABLED')) { + const isBillingEnabledForWorkspace = + await this.billingWorkspaceService.isBillingEnabledForWorkspace( + payload.workspaceId, + ); + + if (!isBillingEnabledForWorkspace) { return; } diff --git a/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event-participant.dto.ts b/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event-participant.dto.ts index 1d5d99720899..6e9a3189a285 100644 --- a/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event-participant.dto.ts +++ b/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event-participant.dto.ts @@ -5,10 +5,10 @@ import { UUIDScalarType } from 'src/engine/api/graphql/workspace-schema-builder/ @ObjectType('TimelineCalendarEventParticipant') export class TimelineCalendarEventParticipant { @Field(() => UUIDScalarType, { nullable: true }) - personId: string; + personId: string | null; @Field(() => UUIDScalarType, { nullable: true }) - workspaceMemberId: string; + workspaceMemberId: string | null; @Field() firstName: string; diff --git a/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event.dto.ts b/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event.dto.ts index 2c64d9226135..c9e579ccd30b 100644 --- a/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event.dto.ts +++ b/packages/twenty-server/src/engine/core-modules/calendar/dtos/timeline-calendar-event.dto.ts @@ -2,7 +2,7 @@ import { ObjectType, Field, registerEnumType } from '@nestjs/graphql'; import { UUIDScalarType } from 'src/engine/api/graphql/workspace-schema-builder/graphql-types/scalars'; import { TimelineCalendarEventParticipant } from 'src/engine/core-modules/calendar/dtos/timeline-calendar-event-participant.dto'; -import { CalendarChannelVisibility } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; +import { CalendarChannelVisibility } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; registerEnumType(CalendarChannelVisibility, { name: 'CalendarChannelVisibility', diff --git a/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.module.ts b/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.module.ts index c8a7d442a073..47bdee1888d4 100644 --- a/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.module.ts +++ b/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.module.ts @@ -4,8 +4,8 @@ import { UserModule } from 'src/engine/core-modules/user/user.module'; import { TimelineCalendarEventResolver } from 'src/engine/core-modules/calendar/timeline-calendar-event.resolver'; import { TimelineCalendarEventService } from 'src/engine/core-modules/calendar/timeline-calendar-event.service'; import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; @Module({ imports: [ diff --git a/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.service.ts b/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.service.ts index 810869e2a74e..1ef6370abc2e 100644 --- a/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.service.ts +++ b/packages/twenty-server/src/engine/core-modules/calendar/timeline-calendar-event.service.ts @@ -7,9 +7,9 @@ import { TIMELINE_CALENDAR_EVENTS_DEFAULT_PAGE_SIZE } from 'src/engine/core-modu import { TimelineCalendarEventsWithTotal } from 'src/engine/core-modules/calendar/dtos/timeline-calendar-events-with-total.dto'; import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { CalendarChannelVisibility } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; +import { CalendarChannelVisibility } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; @Injectable() export class TimelineCalendarEventService { @@ -31,9 +31,7 @@ export class TimelineCalendarEventService { const calendarEventIds = await this.calendarEventRepository.find({ where: { calendarEventParticipants: { - person: { - id: Any(personIds), - }, + personId: Any(personIds), }, }, select: { @@ -81,19 +79,19 @@ export class TimelineCalendarEventService { const participants = event.calendarEventParticipants.map( (participant) => ({ calendarEventId: event.id, - personId: participant.person?.id, - workspaceMemberId: participant.workspaceMember?.id, + personId: participant.personId ?? null, + workspaceMemberId: participant.workspaceMemberId ?? null, firstName: - participant.person?.name.firstName || + participant.person?.name?.firstName || participant.workspaceMember?.name.firstName || '', lastName: - participant.person?.name.lastName || + participant.person?.name?.lastName || participant.workspaceMember?.name.lastName || '', displayName: - participant.person?.name.firstName || - participant.person?.name.lastName || + participant.person?.name?.firstName || + participant.person?.name?.lastName || participant.workspaceMember?.name.firstName || participant.workspaceMember?.name.lastName || '', @@ -135,9 +133,7 @@ export class TimelineCalendarEventService { ): Promise<TimelineCalendarEventsWithTotal> { const personIds = await this.personRepository.find({ where: { - company: { - id: companyId, - }, + companyId, }, select: { id: true, diff --git a/packages/twenty-server/src/engine/core-modules/client-config/client-config.entity.ts b/packages/twenty-server/src/engine/core-modules/client-config/client-config.entity.ts index 8252db6ba1af..478c89da0cd7 100644 --- a/packages/twenty-server/src/engine/core-modules/client-config/client-config.entity.ts +++ b/packages/twenty-server/src/engine/core-modules/client-config/client-config.entity.ts @@ -65,6 +65,12 @@ class Captcha { siteKey: string | undefined; } +@ObjectType() +class ApiConfig { + @Field(() => Number, { nullable: false }) + mutationMaximumAffectedRecords: number; +} + @ObjectType() export class ClientConfig { @Field(() => AuthProviders, { nullable: false }) @@ -96,4 +102,7 @@ export class ClientConfig { @Field(() => String, { nullable: true }) chromeExtensionId: string | undefined; + + @Field(() => ApiConfig) + api: ApiConfig; } diff --git a/packages/twenty-server/src/engine/core-modules/client-config/client-config.resolver.ts b/packages/twenty-server/src/engine/core-modules/client-config/client-config.resolver.ts index 4ce91428ca74..c8222a3418f8 100644 --- a/packages/twenty-server/src/engine/core-modules/client-config/client-config.resolver.ts +++ b/packages/twenty-server/src/engine/core-modules/client-config/client-config.resolver.ts @@ -46,6 +46,11 @@ export class ClientConfigResolver { siteKey: this.environmentService.get('CAPTCHA_SITE_KEY'), }, chromeExtensionId: this.environmentService.get('CHROME_EXTENSION_ID'), + api: { + mutationMaximumAffectedRecords: this.environmentService.get( + 'MUTATION_MAXIMUM_AFFECTED_RECORDS', + ), + }, }; return Promise.resolve(clientConfig); diff --git a/packages/twenty-server/src/engine/core-modules/core-engine.module.ts b/packages/twenty-server/src/engine/core-modules/core-engine.module.ts index 2df9de73b593..cb63a112ccab 100644 --- a/packages/twenty-server/src/engine/core-modules/core-engine.module.ts +++ b/packages/twenty-server/src/engine/core-modules/core-engine.module.ts @@ -10,7 +10,7 @@ import { TimelineMessagingModule } from 'src/engine/core-modules/messaging/timel import { TimelineCalendarEventModule } from 'src/engine/core-modules/calendar/timeline-calendar-event.module'; import { BillingModule } from 'src/engine/core-modules/billing/billing.module'; import { HealthModule } from 'src/engine/core-modules/health/health.module'; -import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { AISQLQueryModule } from 'src/engine/core-modules/ai-sql-query/ai-sql-query.module'; import { PostgresCredentialsModule } from 'src/engine/core-modules/postgres-credentials/postgres-credentials.module'; import { AnalyticsModule } from './analytics/analytics.module'; @@ -19,9 +19,6 @@ import { ClientConfigModule } from './client-config/client-config.module'; @Module({ imports: [ - TwentyORMModule.register({ - workspaceEntities: ['dist/src/**/*.workspace-entity{.ts,.js}'], - }), HealthModule, AnalyticsModule, AuthModule, @@ -35,6 +32,7 @@ import { ClientConfigModule } from './client-config/client-config.module'; TimelineCalendarEventModule, UserModule, WorkspaceModule, + AISQLQueryModule, PostgresCredentialsModule, ], exports: [ diff --git a/packages/twenty-server/src/engine/api/graphql/workspace-resolver-builder/constants/duplicate-criteria.constants.ts b/packages/twenty-server/src/engine/core-modules/duplicate/constants/duplicate-criteria.constants.ts similarity index 100% rename from packages/twenty-server/src/engine/api/graphql/workspace-resolver-builder/constants/duplicate-criteria.constants.ts rename to packages/twenty-server/src/engine/core-modules/duplicate/constants/duplicate-criteria.constants.ts diff --git a/packages/twenty-server/src/engine/core-modules/duplicate/duplicate.module.ts b/packages/twenty-server/src/engine/core-modules/duplicate/duplicate.module.ts new file mode 100644 index 000000000000..c32a4fa598cf --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/duplicate/duplicate.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; + +import { DuplicateService } from 'src/engine/core-modules/duplicate/duplicate.service'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; + +@Module({ + imports: [WorkspaceDataSourceModule], + exports: [DuplicateService], + providers: [DuplicateService], +}) +export class DuplicateModule {} diff --git a/packages/twenty-server/src/engine/core-modules/duplicate/duplicate.service.ts b/packages/twenty-server/src/engine/core-modules/duplicate/duplicate.service.ts new file mode 100644 index 000000000000..d7ed6f87bd85 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/duplicate/duplicate.service.ts @@ -0,0 +1,173 @@ +import { Injectable } from '@nestjs/common'; + +import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; +import { + Record as IRecord, + Record, +} from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; + +import { settings } from 'src/engine/constants/settings'; +import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; +import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; +import { DUPLICATE_CRITERIA_COLLECTION } from 'src/engine/core-modules/duplicate/constants/duplicate-criteria.constants'; + +@Injectable() +export class DuplicateService { + constructor( + private readonly workspaceDataSourceService: WorkspaceDataSourceService, + ) {} + + async findExistingRecords( + recordIds: (string | number)[], + objectMetadata: ObjectMetadataInterface, + workspaceId: string, + ) { + const dataSourceSchema = + this.workspaceDataSourceService.getSchemaName(workspaceId); + + const results = await this.workspaceDataSourceService.executeRawQuery( + ` + SELECT + * + FROM + ${dataSourceSchema}."${computeObjectTargetTable( + objectMetadata, + )}" p + WHERE + p."id" IN (${recordIds + .map((_, index) => `$${index + 1}`) + .join(', ')}) + `, + recordIds, + workspaceId, + ); + + return results as IRecord[]; + } + + buildDuplicateConditionForGraphQL( + objectMetadata: ObjectMetadataInterface, + argsData?: Partial<Record>, + filteringByExistingRecordId?: string, + ) { + if (!argsData) { + return; + } + + const criteriaCollection = + this.getApplicableDuplicateCriteriaCollection(objectMetadata); + + const criteriaWithMatchingArgs = criteriaCollection.filter((criteria) => + criteria.columnNames.every((columnName) => { + const value = argsData[columnName] as string | undefined; + + return ( + !!value && value.length >= settings.minLengthOfStringForDuplicateCheck + ); + }), + ); + + const filterCriteria = criteriaWithMatchingArgs.map((criteria) => + Object.fromEntries( + criteria.columnNames.map((columnName) => [ + columnName, + { eq: argsData[columnName] }, + ]), + ), + ); + + return { + // when filtering by an existing record, we need to filter that explicit record out + ...(filteringByExistingRecordId && { + id: { neq: filteringByExistingRecordId }, + }), + // keep condition as "or" to get results by more duplicate criteria + or: filterCriteria, + }; + } + + private getApplicableDuplicateCriteriaCollection( + objectMetadataItem: ObjectMetadataInterface, + ) { + return DUPLICATE_CRITERIA_COLLECTION.filter( + (duplicateCriteria) => + duplicateCriteria.objectName === objectMetadataItem.nameSingular, + ); + } + + /** + * TODO: Remove this code by September 1st, 2024 if it isn't used + * It was build to be used by the upsertMany function, but it was not used. + * It's a re-implementation of the methods to findDuplicates, but done + * at the SQL layer instead of doing it at the GraphQL layer + * + async findDuplicate( + data: Partial<Record>, + objectMetadata: ObjectMetadataInterface, + workspaceId: string, + ) { + const dataSourceSchema = + this.workspaceDataSourceService.getSchemaName(workspaceId); + + const { duplicateWhereClause, duplicateWhereParameters } = + this.buildDuplicateConditionForUpsert(objectMetadata, data); + + const results = await this.workspaceDataSourceService.executeRawQuery( + ` + SELECT + * + FROM + ${dataSourceSchema}."${computeObjectTargetTable( + objectMetadata, + )}" p + WHERE + ${duplicateWhereClause} + `, + duplicateWhereParameters, + workspaceId, + ); + + return results.length > 0 ? results[0] : null; + } + + private buildDuplicateConditionForUpsert( + objectMetadata: ObjectMetadataInterface, + data: Partial<Record>, + ) { + const criteriaCollection = this.getApplicableDuplicateCriteriaCollection( + objectMetadata, + ).filter( + (duplicateCriteria) => duplicateCriteria.useAsUniqueKeyForUpsert === true, + ); + + const whereClauses: string[] = []; + const whereParameters: any[] = []; + let parameterIndex = 1; + + criteriaCollection.forEach((c) => { + const clauseParts: string[] = []; + + c.columnNames.forEach((column) => { + const dataKey = Object.keys(data).find( + (key) => key.toLowerCase() === column.toLowerCase(), + ); + + if (dataKey) { + clauseParts.push(`p."${column}" = $${parameterIndex}`); + whereParameters.push(data[dataKey]); + parameterIndex++; + } + }); + if (clauseParts.length > 0) { + whereClauses.push(`(${clauseParts.join(' AND ')})`); + } + }); + + const duplicateWhereClause = whereClauses.join(' OR '); + const duplicateWhereParameters = whereParameters; + + return { duplicateWhereClause, duplicateWhereParameters }; + } + * + */ +} diff --git a/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.entity.ts b/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.entity.ts index 954461abdc70..b064661900d4 100644 --- a/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.entity.ts +++ b/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.entity.ts @@ -1,19 +1,19 @@ import { Field, ObjectType } from '@nestjs/graphql'; +import { IDField } from '@ptc-org/nestjs-query-graphql'; import { - Entity, - Unique, - PrimaryGeneratedColumn, Column, CreateDateColumn, - UpdateDateColumn, + Entity, ManyToOne, + PrimaryGeneratedColumn, Relation, + Unique, + UpdateDateColumn, } from 'typeorm'; -import { IDField } from '@ptc-org/nestjs-query-graphql'; -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { UUIDScalarType } from 'src/engine/api/graphql/workspace-schema-builder/graphql-types/scalars'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; export enum FeatureFlagKeys { IsBlocklistEnabled = 'IS_BLOCKLIST_ENABLED', @@ -21,7 +21,10 @@ export enum FeatureFlagKeys { IsAirtableIntegrationEnabled = 'IS_AIRTABLE_INTEGRATION_ENABLED', IsPostgreSQLIntegrationEnabled = 'IS_POSTGRESQL_INTEGRATION_ENABLED', IsStripeIntegrationEnabled = 'IS_STRIPE_INTEGRATION_ENABLED', - IsContactCreationForSentAndReceivedEmailsEnabled = 'IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED', + IsCopilotEnabled = 'IS_COPILOT_ENABLED', + IsMessagingAliasFetchingEnabled = 'IS_MESSAGING_ALIAS_FETCHING_ENABLED', + IsGoogleCalendarSyncV2Enabled = 'IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED', + IsFreeAccessEnabled = 'IS_FREE_ACCESS_ENABLED', } @Entity({ name: 'featureFlag', schema: 'core' }) diff --git a/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.module.ts b/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.module.ts index 0b0f9bbd25ad..a79d3a72901e 100644 --- a/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.module.ts +++ b/packages/twenty-server/src/engine/core-modules/feature-flag/feature-flag.module.ts @@ -5,6 +5,7 @@ import { NestjsQueryTypeOrmModule } from '@ptc-org/nestjs-query-typeorm'; import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { IsFeatureEnabledService } from 'src/engine/core-modules/feature-flag/services/is-feature-enabled.service'; @Module({ imports: [ @@ -17,7 +18,7 @@ import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature- resolvers: [], }), ], - exports: [], - providers: [], + exports: [IsFeatureEnabledService], + providers: [IsFeatureEnabledService], }) export class FeatureFlagModule {} diff --git a/packages/twenty-server/src/engine/core-modules/feature-flag/services/is-feature-enabled.service.ts b/packages/twenty-server/src/engine/core-modules/feature-flag/services/is-feature-enabled.service.ts new file mode 100644 index 000000000000..33e3bcc8655c --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/feature-flag/services/is-feature-enabled.service.ts @@ -0,0 +1,30 @@ +import { Injectable } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; + +import { Repository } from 'typeorm'; + +import { + FeatureFlagEntity, + FeatureFlagKeys, +} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; + +@Injectable() +export class IsFeatureEnabledService { + constructor( + @InjectRepository(FeatureFlagEntity, 'core') + private readonly featureFlagRepository: Repository<FeatureFlagEntity>, + ) {} + + public async isFeatureEnabled( + key: FeatureFlagKeys, + workspaceId: string, + ): Promise<boolean> { + const featureFlag = await this.featureFlagRepository.findOneBy({ + workspaceId, + key, + value: true, + }); + + return !!featureFlag?.value; + } +} diff --git a/packages/twenty-server/src/engine/core-modules/graphql/engine-graphql.module.ts b/packages/twenty-server/src/engine/core-modules/graphql/engine-graphql.module.ts new file mode 100644 index 000000000000..ae592e22c802 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/graphql/engine-graphql.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; + +import { useGraphQLErrorHandlerHook } from 'src/engine/core-modules/graphql/hooks/use-graphql-error-handler.hook'; +import { ExceptionHandlerModule } from 'src/engine/integrations/exception-handler/exception-handler.module'; + +@Module({ + imports: [ExceptionHandlerModule], + exports: [useGraphQLErrorHandlerHook], + providers: [], +}) +export class EngineGraphQLModule {} diff --git a/packages/twenty-server/src/engine/core-modules/graphql/hooks/use-graphql-error-handler.hook.ts b/packages/twenty-server/src/engine/core-modules/graphql/hooks/use-graphql-error-handler.hook.ts new file mode 100644 index 000000000000..0f98cdcfdb42 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/graphql/hooks/use-graphql-error-handler.hook.ts @@ -0,0 +1,133 @@ +import { + OnExecuteDoneHookResultOnNextHook, + Plugin, + getDocumentString, + handleStreamOrSingleExecutionResult, +} from '@envelop/core'; +import { GraphQLError, Kind, OperationDefinitionNode, print } from 'graphql'; + +import { GraphQLContext } from 'src/engine/api/graphql/graphql-config/interfaces/graphql-context.interface'; + +import { generateGraphQLErrorFromError } from 'src/engine/core-modules/graphql/utils/generate-graphql-error-from-error.util'; +import { BaseGraphQLError } from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; +import { shouldCaptureException } from 'src/engine/core-modules/graphql/utils/should-capture-exception.util'; +import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service'; + +type GraphQLErrorHandlerHookOptions = { + /** + * The exception handler service to use. + */ + exceptionHandlerService: ExceptionHandlerService; + /** + * The key of the event id in the error's extension. `null` to disable. + * @default exceptionEventId + */ + eventIdKey?: string | null; +}; + +export const useGraphQLErrorHandlerHook = < + PluginContext extends GraphQLContext, +>( + options: GraphQLErrorHandlerHookOptions, +): Plugin<PluginContext> => { + const eventIdKey = options.eventIdKey === null ? null : 'exceptionEventId'; + + function addEventId( + err: GraphQLError, + eventId: string | undefined | null, + ): GraphQLError { + if (eventIdKey !== null && eventId) { + err.extensions[eventIdKey] = eventId; + } + + return err; + } + + return { + async onExecute({ args }) { + const exceptionHandlerService = options.exceptionHandlerService; + const rootOperation = args.document.definitions.find( + (o) => o.kind === Kind.OPERATION_DEFINITION, + ) as OperationDefinitionNode; + const operationType = rootOperation.operation; + const user = args.contextValue.req.user; + const document = getDocumentString(args.document, print); + const opName = + args.operationName || + rootOperation.name?.value || + 'Anonymous Operation'; + + return { + onExecuteDone(payload) { + const handleResult: OnExecuteDoneHookResultOnNextHook<object> = ({ + result, + setResult, + }) => { + if (result.errors && result.errors.length > 0) { + const errorsToCapture = result.errors.reduce<BaseGraphQLError[]>( + (acc, error) => { + if (!(error instanceof BaseGraphQLError)) { + error = generateGraphQLErrorFromError(error); + } + + if (shouldCaptureException(error)) { + acc.push(error); + } + + return acc; + }, + [], + ); + + if (errorsToCapture.length > 0) { + const eventIds = exceptionHandlerService.captureExceptions( + errorsToCapture, + { + operation: { + name: opName, + type: operationType, + }, + document, + user, + }, + ); + + errorsToCapture.map((err, i) => addEventId(err, eventIds?.[i])); + } + + const nonCapturedErrors = result.errors.filter( + (error) => !errorsToCapture.includes(error), + ); + + setResult({ + ...result, + errors: [...nonCapturedErrors, ...errorsToCapture], + }); + } + }; + + return handleStreamOrSingleExecutionResult(payload, handleResult); + }, + }; + }, + onValidate: ({ context, validateFn, params: { documentAST, schema } }) => { + const errors = validateFn(schema, documentAST); + + if (Array.isArray(errors) && errors.length > 0) { + const headers = context.req.headers; + const currentSchemaVersion = context.req.cacheVersion; + + const requestSchemaVersion = headers['x-schema-version']; + + if ( + requestSchemaVersion && + requestSchemaVersion !== currentSchemaVersion + ) { + throw new GraphQLError( + `Schema version mismatch, please refresh the page.`, + ); + } + } + }, + }; +}; diff --git a/packages/twenty-server/src/engine/core-modules/graphql/utils/generate-graphql-error-from-error.util.ts b/packages/twenty-server/src/engine/core-modules/graphql/utils/generate-graphql-error-from-error.util.ts new file mode 100644 index 000000000000..35201e31bd73 --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/graphql/utils/generate-graphql-error-from-error.util.ts @@ -0,0 +1,18 @@ +import { + BaseGraphQLError, + ErrorCode, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; + +export const generateGraphQLErrorFromError = (error: Error) => { + const graphqlError = new BaseGraphQLError( + error.name, + ErrorCode.INTERNAL_SERVER_ERROR, + ); + + if (process.env.NODE_ENV === 'development') { + graphqlError.stack = error.stack; + graphqlError.extensions['response'] = error.message; + } + + return error; +}; diff --git a/packages/twenty-server/src/engine/utils/graphql-errors.util.ts b/packages/twenty-server/src/engine/core-modules/graphql/utils/graphql-errors.util.ts similarity index 68% rename from packages/twenty-server/src/engine/utils/graphql-errors.util.ts rename to packages/twenty-server/src/engine/core-modules/graphql/utils/graphql-errors.util.ts index d198a9e6ee73..613225796cb3 100644 --- a/packages/twenty-server/src/engine/utils/graphql-errors.util.ts +++ b/packages/twenty-server/src/engine/core-modules/graphql/utils/graphql-errors.util.ts @@ -15,7 +15,22 @@ declare module 'graphql' { } } -export class BaseGraphQLError extends Error implements GraphQLError { +export enum ErrorCode { + GRAPHQL_PARSE_FAILED = 'GRAPHQL_PARSE_FAILED', + GRAPHQL_VALIDATION_FAILED = 'GRAPHQL_VALIDATION_FAILED', + UNAUTHENTICATED = 'UNAUTHENTICATED', + FORBIDDEN = 'FORBIDDEN', + PERSISTED_QUERY_NOT_FOUND = 'PERSISTED_QUERY_NOT_FOUND', + PERSISTED_QUERY_NOT_SUPPORTED = 'PERSISTED_QUERY_NOT_SUPPORTED', + BAD_USER_INPUT = 'BAD_USER_INPUT', + NOT_FOUND = 'NOT_FOUND', + METHOD_NOT_ALLOWED = 'METHOD_NOT_ALLOWED', + CONFLICT = 'CONFLICT', + TIMEOUT = 'TIMEOUT', + INTERNAL_SERVER_ERROR = 'INTERNAL_SERVER_ERROR', +} + +export class BaseGraphQLError extends GraphQLError { public extensions: Record<string, any>; override readonly name!: string; readonly locations: ReadonlyArray<SourceLocation> | undefined; @@ -76,7 +91,7 @@ function toGraphQLError(error: BaseGraphQLError): GraphQLError { export class SyntaxError extends BaseGraphQLError { constructor(message: string) { - super(message, 'GRAPHQL_PARSE_FAILED'); + super(message, ErrorCode.GRAPHQL_PARSE_FAILED); Object.defineProperty(this, 'name', { value: 'SyntaxError' }); } @@ -84,23 +99,23 @@ export class SyntaxError extends BaseGraphQLError { export class ValidationError extends BaseGraphQLError { constructor(message: string) { - super(message, 'GRAPHQL_VALIDATION_FAILED'); + super(message, ErrorCode.GRAPHQL_VALIDATION_FAILED); Object.defineProperty(this, 'name', { value: 'ValidationError' }); } } export class AuthenticationError extends BaseGraphQLError { - constructor(message: string, extensions?: Record<string, any>) { - super(message, 'UNAUTHENTICATED', extensions); + constructor(message: string) { + super(message, ErrorCode.UNAUTHENTICATED); Object.defineProperty(this, 'name', { value: 'AuthenticationError' }); } } export class ForbiddenError extends BaseGraphQLError { - constructor(message: string, extensions?: Record<string, any>) { - super(message, 'FORBIDDEN', extensions); + constructor(message: string) { + super(message, ErrorCode.FORBIDDEN); Object.defineProperty(this, 'name', { value: 'ForbiddenError' }); } @@ -108,7 +123,7 @@ export class ForbiddenError extends BaseGraphQLError { export class PersistedQueryNotFoundError extends BaseGraphQLError { constructor() { - super('PersistedQueryNotFound', 'PERSISTED_QUERY_NOT_FOUND'); + super('PersistedQueryNotFound', ErrorCode.PERSISTED_QUERY_NOT_FOUND); Object.defineProperty(this, 'name', { value: 'PersistedQueryNotFoundError', @@ -118,7 +133,10 @@ export class PersistedQueryNotFoundError extends BaseGraphQLError { export class PersistedQueryNotSupportedError extends BaseGraphQLError { constructor() { - super('PersistedQueryNotSupported', 'PERSISTED_QUERY_NOT_SUPPORTED'); + super( + 'PersistedQueryNotSupported', + ErrorCode.PERSISTED_QUERY_NOT_SUPPORTED, + ); Object.defineProperty(this, 'name', { value: 'PersistedQueryNotSupportedError', @@ -127,41 +145,49 @@ export class PersistedQueryNotSupportedError extends BaseGraphQLError { } export class UserInputError extends BaseGraphQLError { - constructor(message: string, extensions?: Record<string, any>) { - super(message, 'BAD_USER_INPUT', extensions); + constructor(message: string) { + super(message, ErrorCode.BAD_USER_INPUT); Object.defineProperty(this, 'name', { value: 'UserInputError' }); } } export class NotFoundError extends BaseGraphQLError { - constructor(message: string, extensions?: Record<string, any>) { - super(message, 'NOT_FOUND', extensions); + constructor(message: string) { + super(message, ErrorCode.NOT_FOUND); Object.defineProperty(this, 'name', { value: 'NotFoundError' }); } } export class MethodNotAllowedError extends BaseGraphQLError { - constructor(message: string, extensions?: Record<string, any>) { - super(message, 'METHOD_NOT_ALLOWED', extensions); + constructor(message: string) { + super(message, ErrorCode.METHOD_NOT_ALLOWED); Object.defineProperty(this, 'name', { value: 'MethodNotAllowedError' }); } } export class ConflictError extends BaseGraphQLError { - constructor(message: string, extensions?: Record<string, any>) { - super(message, 'CONFLICT', extensions); + constructor(message: string) { + super(message, ErrorCode.CONFLICT); Object.defineProperty(this, 'name', { value: 'ConflictError' }); } } export class TimeoutError extends BaseGraphQLError { - constructor(message: string, extensions?: Record<string, any>) { - super(message, 'TIMEOUT', extensions); + constructor(message: string) { + super(message, ErrorCode.TIMEOUT); Object.defineProperty(this, 'name', { value: 'TimeoutError' }); } } + +export class InternalServerError extends BaseGraphQLError { + constructor(message: string) { + super(message, ErrorCode.INTERNAL_SERVER_ERROR); + + Object.defineProperty(this, 'name', { value: 'InternalServerError' }); + } +} diff --git a/packages/twenty-server/src/engine/core-modules/graphql/utils/should-capture-exception.util.ts b/packages/twenty-server/src/engine/core-modules/graphql/utils/should-capture-exception.util.ts new file mode 100644 index 000000000000..f801e9be71ae --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/graphql/utils/should-capture-exception.util.ts @@ -0,0 +1,26 @@ +import { + BaseGraphQLError, + ErrorCode, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; + +export const graphQLErrorCodesToFilter = [ + ErrorCode.GRAPHQL_VALIDATION_FAILED, + ErrorCode.UNAUTHENTICATED, + ErrorCode.FORBIDDEN, + ErrorCode.NOT_FOUND, + ErrorCode.METHOD_NOT_ALLOWED, + ErrorCode.TIMEOUT, + ErrorCode.CONFLICT, + ErrorCode.BAD_USER_INPUT, +]; + +export const shouldCaptureException = (exception: Error): boolean => { + if ( + exception instanceof BaseGraphQLError && + graphQLErrorCodesToFilter.includes(exception?.extensions?.code) + ) { + return true; + } + + return false; +}; diff --git a/packages/twenty-server/src/engine/core-modules/onboarding/enums/onboarding-status.enum.ts b/packages/twenty-server/src/engine/core-modules/onboarding/enums/onboarding-status.enum.ts new file mode 100644 index 000000000000..8370714f2b7f --- /dev/null +++ b/packages/twenty-server/src/engine/core-modules/onboarding/enums/onboarding-status.enum.ts @@ -0,0 +1,8 @@ +export enum OnboardingStatus { + PLAN_REQUIRED = 'PLAN_REQUIRED', + WORKSPACE_ACTIVATION = 'WORKSPACE_ACTIVATION', + PROFILE_CREATION = 'PROFILE_CREATION', + SYNC_EMAIL = 'SYNC_EMAIL', + INVITE_TEAM = 'INVITE_TEAM', + COMPLETED = 'COMPLETED', +} diff --git a/packages/twenty-server/src/engine/core-modules/onboarding/enums/onboarding-step.enum.ts b/packages/twenty-server/src/engine/core-modules/onboarding/enums/onboarding-step.enum.ts deleted file mode 100644 index 55087ce5aad0..000000000000 --- a/packages/twenty-server/src/engine/core-modules/onboarding/enums/onboarding-step.enum.ts +++ /dev/null @@ -1,4 +0,0 @@ -export enum OnboardingStep { - SYNC_EMAIL = 'SYNC_EMAIL', - INVITE_TEAM = 'INVITE_TEAM', -} diff --git a/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.module.ts b/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.module.ts index 65f0156d3dd0..a036cafee618 100644 --- a/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.module.ts +++ b/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.module.ts @@ -5,9 +5,19 @@ import { OnboardingResolver } from 'src/engine/core-modules/onboarding/onboardin import { KeyValuePairModule } from 'src/engine/core-modules/key-value-pair/key-value-pair.module'; import { UserWorkspaceModule } from 'src/engine/core-modules/user-workspace/user-workspace.module'; import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; +import { WorkspaceManagerModule } from 'src/engine/workspace-manager/workspace-manager.module'; +import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module'; +import { BillingModule } from 'src/engine/core-modules/billing/billing.module'; @Module({ - imports: [DataSourceModule, UserWorkspaceModule, KeyValuePairModule], + imports: [ + DataSourceModule, + WorkspaceManagerModule, + UserWorkspaceModule, + KeyValuePairModule, + EnvironmentModule, + BillingModule, + ], exports: [OnboardingService], providers: [OnboardingService, OnboardingResolver], }) diff --git a/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.service.ts b/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.service.ts index 1709f3d789b9..f131e5bca20c 100644 --- a/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.service.ts +++ b/packages/twenty-server/src/engine/core-modules/onboarding/onboarding.service.ts @@ -1,13 +1,20 @@ import { Injectable } from '@nestjs/common'; import { KeyValuePairService } from 'src/engine/core-modules/key-value-pair/key-value-pair.service'; -import { OnboardingStep } from 'src/engine/core-modules/onboarding/enums/onboarding-step.enum'; +import { OnboardingStatus } from 'src/engine/core-modules/onboarding/enums/onboarding-status.enum'; import { User } from 'src/engine/core-modules/user/user.entity'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; +import { WorkspaceManagerService } from 'src/engine/workspace-manager/workspace-manager.service'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { SubscriptionStatus } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; +import { isDefined } from 'src/utils/is-defined'; +import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service'; enum OnboardingStepValues { SKIPPED = 'SKIPPED', @@ -26,29 +33,71 @@ type OnboardingKeyValueType = { @Injectable() export class OnboardingService { constructor( + private readonly billingWorkspaceService: BillingWorkspaceService, + private readonly workspaceManagerService: WorkspaceManagerService, private readonly userWorkspaceService: UserWorkspaceService, private readonly keyValuePairService: KeyValuePairService<OnboardingKeyValueType>, @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) private readonly connectedAccountRepository: ConnectedAccountRepository, + @InjectWorkspaceRepository(WorkspaceMemberWorkspaceEntity) + private readonly workspaceMemberRepository: WorkspaceRepository<WorkspaceMemberWorkspaceEntity>, ) {} - private async isSyncEmailOnboardingStep(user: User, workspace: Workspace) { + private async isSubscriptionIncompleteOnboardingStatus(user: User) { + const isBillingEnabledForWorkspace = + await this.billingWorkspaceService.isBillingEnabledForWorkspace( + user.defaultWorkspaceId, + ); + + if (!isBillingEnabledForWorkspace) { + return false; + } + + const currentBillingSubscription = + await this.billingWorkspaceService.getCurrentBillingSubscription({ + workspaceId: user.defaultWorkspaceId, + }); + + return ( + !isDefined(currentBillingSubscription) || + currentBillingSubscription?.status === SubscriptionStatus.Incomplete + ); + } + + private async isWorkspaceActivationOnboardingStatus(user: User) { + return !(await this.workspaceManagerService.doesDataSourceExist( + user.defaultWorkspaceId, + )); + } + + private async isProfileCreationOnboardingStatus(user: User) { + const workspaceMember = await this.workspaceMemberRepository.findOneBy({ + userId: user.id, + }); + + return ( + workspaceMember && + (!workspaceMember.name.firstName || !workspaceMember.name.lastName) + ); + } + + private async isSyncEmailOnboardingStatus(user: User) { const syncEmailValue = await this.keyValuePairService.get({ userId: user.id, - workspaceId: workspace.id, + workspaceId: user.defaultWorkspaceId, key: OnboardingStepKeys.SYNC_EMAIL_ONBOARDING_STEP, }); const isSyncEmailSkipped = syncEmailValue === OnboardingStepValues.SKIPPED; const connectedAccounts = await this.connectedAccountRepository.getAllByUserId( user.id, - workspace.id, + user.defaultWorkspaceId, ); return !isSyncEmailSkipped && !connectedAccounts?.length; } - private async isInviteTeamOnboardingStep(workspace: Workspace) { + private async isInviteTeamOnboardingStatus(workspace: Workspace) { const inviteTeamValue = await this.keyValuePairService.get({ workspaceId: workspace.id, key: OnboardingStepKeys.INVITE_TEAM_ONBOARDING_STEP, @@ -56,7 +105,7 @@ export class OnboardingService { const isInviteTeamSkipped = inviteTeamValue === OnboardingStepValues.SKIPPED; const workspaceMemberCount = - await this.userWorkspaceService.getWorkspaceMemberCount(workspace.id); + await this.userWorkspaceService.getWorkspaceMemberCount(); return ( !isInviteTeamSkipped && @@ -64,19 +113,28 @@ export class OnboardingService { ); } - async getOnboardingStep( - user: User, - workspace: Workspace, - ): Promise<OnboardingStep | null> { - if (await this.isSyncEmailOnboardingStep(user, workspace)) { - return OnboardingStep.SYNC_EMAIL; + async getOnboardingStatus(user: User) { + if (await this.isSubscriptionIncompleteOnboardingStatus(user)) { + return OnboardingStatus.PLAN_REQUIRED; + } + + if (await this.isWorkspaceActivationOnboardingStatus(user)) { + return OnboardingStatus.WORKSPACE_ACTIVATION; + } + + if (await this.isProfileCreationOnboardingStatus(user)) { + return OnboardingStatus.PROFILE_CREATION; + } + + if (await this.isSyncEmailOnboardingStatus(user)) { + return OnboardingStatus.SYNC_EMAIL; } - if (await this.isInviteTeamOnboardingStep(workspace)) { - return OnboardingStep.INVITE_TEAM; + if (await this.isInviteTeamOnboardingStatus(user.defaultWorkspace)) { + return OnboardingStatus.INVITE_TEAM; } - return null; + return OnboardingStatus.COMPLETED; } async skipInviteTeamOnboardingStep(workspaceId: string) { diff --git a/packages/twenty-server/src/engine/core-modules/open-api/open-api.service.ts b/packages/twenty-server/src/engine/core-modules/open-api/open-api.service.ts index c833377e1ae0..8a3738b2a695 100644 --- a/packages/twenty-server/src/engine/core-modules/open-api/open-api.service.ts +++ b/packages/twenty-server/src/engine/core-modules/open-api/open-api.service.ts @@ -133,9 +133,13 @@ export class OpenApiService { get: { tags: [item.namePlural], summary: `Find Many ${item.namePlural}`, - parameters: [{ $ref: '#/components/parameters/filter' }], + parameters: [ + { $ref: '#/components/parameters/limit' }, + { $ref: '#/components/parameters/startingAfter' }, + { $ref: '#/components/parameters/endingBefore' }, + ], responses: { - '200': getFindManyResponse200(item), + '200': getFindManyResponse200(item, true), '400': { $ref: '#/components/responses/400' }, '401': { $ref: '#/components/responses/401' }, }, @@ -158,7 +162,7 @@ export class OpenApiService { summary: `Find One ${item.nameSingular}`, parameters: [{ $ref: '#/components/parameters/idPath' }], responses: { - '200': getFindOneResponse200(item), + '200': getFindOneResponse200(item, true), '400': { $ref: '#/components/responses/400' }, '401': { $ref: '#/components/responses/401' }, }, @@ -174,18 +178,20 @@ export class OpenApiService { '401': { $ref: '#/components/responses/401' }, }, }, - patch: { - tags: [item.namePlural], - summary: `Update One ${item.namePlural}`, - operationId: `updateOne${capitalize(item.nameSingular)}`, - parameters: [{ $ref: '#/components/parameters/idPath' }], - requestBody: getRequestBody(capitalize(item.nameSingular)), - responses: { - '200': getUpdateOneResponse200(item, true), - '400': { $ref: '#/components/responses/400' }, - '401': { $ref: '#/components/responses/401' }, + ...(item.nameSingular !== 'relation' && { + patch: { + tags: [item.namePlural], + summary: `Update One ${item.nameSingular}`, + operationId: `updateOne${capitalize(item.nameSingular)}`, + parameters: [{ $ref: '#/components/parameters/idPath' }], + requestBody: getRequestBody(capitalize(item.nameSingular)), + responses: { + '200': getUpdateOneResponse200(item, true), + '400': { $ref: '#/components/responses/400' }, + '401': { $ref: '#/components/responses/401' }, + }, }, - }, + }), } as OpenAPIV3_1.PathItemObject; return path; @@ -194,7 +200,7 @@ export class OpenApiService { schema.components = { ...schema.components, // components.securitySchemes is defined in base Schema schemas: computeMetadataSchemaComponents(metadata), - parameters: computeParameterComponents(), + parameters: computeParameterComponents(true), responses: { '400': get400ErrorResponses(), '401': get401ErrorResponses(), diff --git a/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/components.utils.spec.ts b/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/components.utils.spec.ts index 8a953bc4540f..09d864377649 100644 --- a/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/components.utils.spec.ts +++ b/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/components.utils.spec.ts @@ -55,9 +55,6 @@ describe('computeSchemaComponents', () => { fieldNumeric: { type: 'number', }, - fieldProbability: { - type: 'number', - }, fieldLink: { properties: { label: { type: 'string' }, diff --git a/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/parameters.utils.spec.ts b/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/parameters.utils.spec.ts index 3fd99913c9c8..f91da6a750e6 100644 --- a/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/parameters.utils.spec.ts +++ b/packages/twenty-server/src/engine/core-modules/open-api/utils/__tests__/parameters.utils.spec.ts @@ -9,10 +9,10 @@ import { computeOrderByParameters, computeStartingAfterParameters, } from 'src/engine/core-modules/open-api/utils/parameters.utils'; -import { DEFAULT_ORDER_DIRECTION } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory'; -import { FilterComparators } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-base-filter.utils'; -import { Conjunctions } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils'; -import { DEFAULT_CONJUNCTION } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/add-default-conjunction.utils'; +import { DEFAULT_ORDER_DIRECTION } from 'src/engine/api/rest/input-factories/order-by-input.factory'; +import { FilterComparators } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-base-filter.utils'; +import { Conjunctions } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils'; +import { DEFAULT_CONJUNCTION } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/add-default-conjunction.utils'; describe('computeParameters', () => { describe('computeLimit', () => { @@ -68,7 +68,7 @@ describe('computeParameters', () => { required: false, schema: { type: 'integer', - enum: [1, 2], + enum: [0, 1, 2], }, }); }); diff --git a/packages/twenty-server/src/engine/core-modules/open-api/utils/components.utils.ts b/packages/twenty-server/src/engine/core-modules/open-api/utils/components.utils.ts index 62255806f168..6d65e82a26e6 100644 --- a/packages/twenty-server/src/engine/core-modules/open-api/utils/components.utils.ts +++ b/packages/twenty-server/src/engine/core-modules/open-api/utils/components.utils.ts @@ -35,7 +35,6 @@ const getFieldProperties = (type: FieldMetadataType): Property => { case FieldMetadataType.NUMBER: return { type: 'integer' }; case FieldMetadataType.NUMERIC: - case FieldMetadataType.PROBABILITY: case FieldMetadataType.RATING: case FieldMetadataType.POSITION: return { type: 'number' }; @@ -218,10 +217,9 @@ export const computeSchemaComponents = ( ); }; -export const computeParameterComponents = (): Record< - string, - OpenAPIV3_1.ParameterObject -> => { +export const computeParameterComponents = ( + fromMetadata = false, +): Record<string, OpenAPIV3_1.ParameterObject> => { return { idPath: computeIdPathParameter(), startingAfter: computeStartingAfterParameters(), @@ -229,7 +227,7 @@ export const computeParameterComponents = (): Record< filter: computeFilterParameters(), depth: computeDepthParameters(), orderBy: computeOrderByParameters(), - limit: computeLimitParameters(), + limit: computeLimitParameters(fromMetadata), }; }; diff --git a/packages/twenty-server/src/engine/core-modules/open-api/utils/parameters.utils.ts b/packages/twenty-server/src/engine/core-modules/open-api/utils/parameters.utils.ts index b07c297e0d8e..56034403a912 100644 --- a/packages/twenty-server/src/engine/core-modules/open-api/utils/parameters.utils.ts +++ b/packages/twenty-server/src/engine/core-modules/open-api/utils/parameters.utils.ts @@ -2,12 +2,14 @@ import { OpenAPIV3_1 } from 'openapi-types'; import { OrderByDirection } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; -import { FilterComparators } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-base-filter.utils'; -import { Conjunctions } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/parse-filter.utils'; -import { DEFAULT_ORDER_DIRECTION } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/order-by-input.factory'; -import { DEFAULT_CONJUNCTION } from 'src/engine/api/rest/rest-api-core-query-builder/factories/input-factories/filter-utils/add-default-conjunction.utils'; +import { FilterComparators } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-base-filter.utils'; +import { Conjunctions } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/parse-filter.utils'; +import { DEFAULT_ORDER_DIRECTION } from 'src/engine/api/rest/input-factories/order-by-input.factory'; +import { DEFAULT_CONJUNCTION } from 'src/engine/api/rest/core/query-builder/utils/filter-utils/add-default-conjunction.utils'; -export const computeLimitParameters = (): OpenAPIV3_1.ParameterObject => { +export const computeLimitParameters = ( + fromMetadata = false, +): OpenAPIV3_1.ParameterObject => { return { name: 'limit', in: 'query', @@ -16,8 +18,8 @@ export const computeLimitParameters = (): OpenAPIV3_1.ParameterObject => { schema: { type: 'integer', minimum: 0, - maximum: 60, - default: 60, + maximum: fromMetadata ? 1000 : 60, + default: fromMetadata ? 1000 : 60, }, }; }; @@ -57,7 +59,7 @@ export const computeDepthParameters = (): OpenAPIV3_1.ParameterObject => { required: false, schema: { type: 'integer', - enum: [1, 2], + enum: [0, 1, 2], }, }; }; diff --git a/packages/twenty-server/src/engine/core-modules/open-api/utils/path.utils.ts b/packages/twenty-server/src/engine/core-modules/open-api/utils/path.utils.ts index 37b1e217a415..fb73fb5ecb9e 100644 --- a/packages/twenty-server/src/engine/core-modules/open-api/utils/path.utils.ts +++ b/packages/twenty-server/src/engine/core-modules/open-api/utils/path.utils.ts @@ -105,7 +105,7 @@ export const computeSingleResultPath = ( }, patch: { tags: [item.namePlural], - summary: `Update One ${item.namePlural}`, + summary: `Update One ${item.nameSingular}`, operationId: `UpdateOne${capitalize(item.nameSingular)}`, parameters: [ { $ref: '#/components/parameters/idPath' }, diff --git a/packages/twenty-server/src/engine/core-modules/open-api/utils/responses.utils.ts b/packages/twenty-server/src/engine/core-modules/open-api/utils/responses.utils.ts index 7793a507f842..49264fee0e9f 100644 --- a/packages/twenty-server/src/engine/core-modules/open-api/utils/responses.utils.ts +++ b/packages/twenty-server/src/engine/core-modules/open-api/utils/responses.utils.ts @@ -3,6 +3,7 @@ import { capitalize } from 'src/utils/capitalize'; export const getFindManyResponse200 = ( item: Pick<ObjectMetadataEntity, 'nameSingular' | 'namePlural'>, + fromMetadata = false, ) => { return { description: 'Successful operation', @@ -19,7 +20,7 @@ export const getFindManyResponse200 = ( items: { $ref: `#/components/schemas/${capitalize( item.nameSingular, - )} with Relations`, + )}${!fromMetadata ? ' with Relations' : ''}`, }, }, }, @@ -32,9 +33,11 @@ export const getFindManyResponse200 = ( endCursor: { type: 'string' }, }, }, - totalCount: { - type: 'integer', - }, + ...(!fromMetadata && { + totalCount: { + type: 'integer', + }, + }), }, example: { data: { @@ -59,6 +62,7 @@ export const getFindManyResponse200 = ( export const getFindOneResponse200 = ( item: Pick<ObjectMetadataEntity, 'nameSingular'>, + fromMetadata = false, ) => { return { description: 'Successful operation', @@ -71,9 +75,9 @@ export const getFindOneResponse200 = ( type: 'object', properties: { [item.nameSingular]: { - $ref: `#/components/schemas/${capitalize( - item.nameSingular, - )} with Relations`, + $ref: `#/components/schemas/${capitalize(item.nameSingular)}${ + !fromMetadata ? ' with Relations' : '' + }`, }, }, }, diff --git a/packages/twenty-server/src/engine/core-modules/postgres-credentials/postgres-credentials.service.ts b/packages/twenty-server/src/engine/core-modules/postgres-credentials/postgres-credentials.service.ts index 6bdc67d43c99..790c36ab266e 100644 --- a/packages/twenty-server/src/engine/core-modules/postgres-credentials/postgres-credentials.service.ts +++ b/packages/twenty-server/src/engine/core-modules/postgres-credentials/postgres-credentials.service.ts @@ -1,5 +1,5 @@ -import { InjectRepository } from '@nestjs/typeorm'; import { BadRequestException } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; import { randomBytes } from 'crypto'; @@ -9,10 +9,10 @@ import { decryptText, encryptText, } from 'src/engine/core-modules/auth/auth.util'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { PostgresCredentials } from 'src/engine/core-modules/postgres-credentials/postgres-credentials.entity'; -import { NotFoundError } from 'src/engine/utils/graphql-errors.util'; +import { NotFoundError } from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; import { PostgresCredentialsDTO } from 'src/engine/core-modules/postgres-credentials/dtos/postgres-credentials.dto'; +import { PostgresCredentials } from 'src/engine/core-modules/postgres-credentials/postgres-credentials.entity'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; export class PostgresCredentialsService { constructor( diff --git a/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.module.ts b/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.module.ts index 09d3c621fc11..1fd86d27cbe2 100644 --- a/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.module.ts +++ b/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.module.ts @@ -9,6 +9,8 @@ import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/use import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { User } from 'src/engine/core-modules/user/user.entity'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; @Module({ imports: [ @@ -21,6 +23,7 @@ import { User } from 'src/engine/core-modules/user/user.entity'; ], services: [UserWorkspaceService], }), + TwentyORMModule.forFeature([WorkspaceMemberWorkspaceEntity]), ], exports: [UserWorkspaceService], providers: [UserWorkspaceService], diff --git a/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.service.ts b/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.service.ts index bdeaa7535735..99c68f81ddd5 100644 --- a/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.service.ts +++ b/packages/twenty-server/src/engine/core-modules/user-workspace/user-workspace.service.ts @@ -8,11 +8,12 @@ import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-works import { TypeORMService } from 'src/database/typeorm/typeorm.service'; import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; import { User } from 'src/engine/core-modules/user/user.entity'; -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; import { assert } from 'src/utils/assert'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; export class UserWorkspaceService extends TypeOrmQueryService<UserWorkspace> { constructor( @@ -20,9 +21,10 @@ export class UserWorkspaceService extends TypeOrmQueryService<UserWorkspace> { private readonly userWorkspaceRepository: Repository<UserWorkspace>, @InjectRepository(User, 'core') private readonly userRepository: Repository<User>, + @InjectWorkspaceRepository(WorkspaceMemberWorkspaceEntity) + private readonly workspaceMemberRepository: WorkspaceRepository<WorkspaceMemberWorkspaceEntity>, private readonly dataSourceService: DataSourceService, private readonly typeORMService: TypeORMService, - private readonly workspaceDataSourceService: WorkspaceDataSourceService, private eventEmitter: EventEmitter2, ) { super(userWorkspaceRepository); @@ -99,23 +101,13 @@ export class UserWorkspaceService extends TypeOrmQueryService<UserWorkspace> { }); } - public async getWorkspaceMemberCount( - workspaceId: string, - ): Promise<number | undefined> { - try { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return ( - await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."workspaceMember"`, - [], - workspaceId, - ) - ).length; - } catch { + public async getWorkspaceMemberCount(): Promise<number | undefined> { + // TODO: to refactor, this could happen today for the first signup since the workspace does not exist yet + if (!this.workspaceMemberRepository) { return undefined; } + + return await this.workspaceMemberRepository.count(); } async checkUserWorkspaceExists( diff --git a/packages/twenty-server/src/engine/core-modules/user/services/user.service.ts b/packages/twenty-server/src/engine/core-modules/user/services/user.service.ts index 47b90a4197a7..853bafed4156 100644 --- a/packages/twenty-server/src/engine/core-modules/user/services/user.service.ts +++ b/packages/twenty-server/src/engine/core-modules/user/services/user.service.ts @@ -12,7 +12,6 @@ import { TypeORMService } from 'src/database/typeorm/typeorm.service'; import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { WorkspaceService } from 'src/engine/core-modules/workspace/services/workspace.service'; export class UserService extends TypeOrmQueryService<User> { @@ -113,8 +112,7 @@ export class UserService extends TypeOrmQueryService<User> { `SELECT * FROM ${dataSourceMetadata.schema}."workspaceMember"`, ); const workspaceMember = workspaceMembers.filter( - (member: ObjectRecord<WorkspaceMemberWorkspaceEntity>) => - member.userId === userId, + (member: WorkspaceMemberWorkspaceEntity) => member.userId === userId, )?.[0]; assert(workspaceMember, 'WorkspaceMember not found'); diff --git a/packages/twenty-server/src/engine/core-modules/user/user.entity.ts b/packages/twenty-server/src/engine/core-modules/user/user.entity.ts index 4a855db1035a..54c371481933 100644 --- a/packages/twenty-server/src/engine/core-modules/user/user.entity.ts +++ b/packages/twenty-server/src/engine/core-modules/user/user.entity.ts @@ -18,11 +18,11 @@ import { WorkspaceMember } from 'src/engine/core-modules/user/dtos/workspace-mem import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity'; import { UUIDScalarType } from 'src/engine/api/graphql/workspace-schema-builder/graphql-types/scalars'; import { KeyValuePair } from 'src/engine/core-modules/key-value-pair/key-value-pair.entity'; -import { OnboardingStep } from 'src/engine/core-modules/onboarding/enums/onboarding-step.enum'; +import { OnboardingStatus } from 'src/engine/core-modules/onboarding/enums/onboarding-status.enum'; -registerEnumType(OnboardingStep, { - name: 'OnboardingStep', - description: 'Onboarding step', +registerEnumType(OnboardingStatus, { + name: 'OnboardingStatus', + description: 'Onboarding status', }); @Entity({ name: 'user', schema: 'core' }) @@ -119,6 +119,6 @@ export class User { @OneToMany(() => UserWorkspace, (userWorkspace) => userWorkspace.user) workspaces: Relation<UserWorkspace[]>; - @Field(() => OnboardingStep, { nullable: true }) - onboardingStep: OnboardingStep; + @Field(() => OnboardingStatus, { nullable: true }) + onboardingStatus: OnboardingStatus; } diff --git a/packages/twenty-server/src/engine/core-modules/user/user.resolver.ts b/packages/twenty-server/src/engine/core-modules/user/user.resolver.ts index ef596d19d0a5..93008f2502e1 100644 --- a/packages/twenty-server/src/engine/core-modules/user/user.resolver.ts +++ b/packages/twenty-server/src/engine/core-modules/user/user.resolver.ts @@ -27,8 +27,9 @@ import { DemoEnvGuard } from 'src/engine/guards/demo.env.guard'; import { JwtAuthGuard } from 'src/engine/guards/jwt.auth.guard'; import { User } from 'src/engine/core-modules/user/user.entity'; import { WorkspaceMember } from 'src/engine/core-modules/user/dtos/workspace-member.dto'; -import { OnboardingStep } from 'src/engine/core-modules/onboarding/enums/onboarding-step.enum'; +import { OnboardingStatus } from 'src/engine/core-modules/onboarding/enums/onboarding-status.enum'; import { OnboardingService } from 'src/engine/core-modules/onboarding/onboarding.service'; +import { LoadServiceWithWorkspaceContext } from 'src/engine/twenty-orm/context/load-service-with-workspace.context'; const getHMACKey = (email?: string, key?: string | null) => { if (!email || !key) return null; @@ -48,6 +49,7 @@ export class UserResolver { private readonly environmentService: EnvironmentService, private readonly fileUploadService: FileUploadService, private readonly onboardingService: OnboardingService, + private readonly loadServiceWithWorkspaceContext: LoadServiceWithWorkspaceContext, ) {} @Query(() => User) @@ -116,15 +118,13 @@ export class UserResolver { return this.userService.deleteUser(userId); } - @ResolveField(() => OnboardingStep) - async onboardingStep(@Parent() user: User): Promise<OnboardingStep | null> { - if (!user) { - return null; - } - - return this.onboardingService.getOnboardingStep( - user, - user.defaultWorkspace, + @ResolveField(() => OnboardingStatus) + async onboardingStatus(@Parent() user: User): Promise<OnboardingStatus> { + const contextInstance = await this.loadServiceWithWorkspaceContext.load( + this.onboardingService, + user.defaultWorkspaceId, ); + + return contextInstance.getOnboardingStatus(user); } } diff --git a/packages/twenty-server/src/engine/core-modules/workspace/handle-workspace-member-deleted.job.ts b/packages/twenty-server/src/engine/core-modules/workspace/handle-workspace-member-deleted.job.ts index 8ec5dad7ded7..7e3bce067751 100644 --- a/packages/twenty-server/src/engine/core-modules/workspace/handle-workspace-member-deleted.job.ts +++ b/packages/twenty-server/src/engine/core-modules/workspace/handle-workspace-member-deleted.job.ts @@ -1,19 +1,18 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { WorkspaceService } from 'src/engine/core-modules/workspace/services/workspace.service'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; export type HandleWorkspaceMemberDeletedJobData = { workspaceId: string; userId: string; }; -@Injectable() -export class HandleWorkspaceMemberDeletedJob - implements MessageQueueJob<HandleWorkspaceMemberDeletedJobData> -{ + +@Processor(MessageQueue.workspaceQueue) +export class HandleWorkspaceMemberDeletedJob { constructor(private readonly workspaceService: WorkspaceService) {} + @Process(HandleWorkspaceMemberDeletedJob.name) async handle(data: HandleWorkspaceMemberDeletedJobData): Promise<void> { const { workspaceId, userId } = data; diff --git a/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.spec.ts b/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.spec.ts index 87a10f16c6a2..696287311451 100644 --- a/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.spec.ts +++ b/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.spec.ts @@ -5,7 +5,7 @@ import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { WorkspaceManagerService } from 'src/engine/workspace-manager/workspace-manager.service'; import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity'; import { User } from 'src/engine/core-modules/user/user.entity'; -import { BillingService } from 'src/engine/core-modules/billing/billing.service'; +import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service'; import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; import { UserService } from 'src/engine/core-modules/user/services/user.service'; import { EmailService } from 'src/engine/integrations/email/email.service'; @@ -46,7 +46,7 @@ describe('WorkspaceService', () => { useValue: {}, }, { - provide: BillingService, + provide: BillingWorkspaceService, useValue: {}, }, { diff --git a/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.ts b/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.ts index 185926b5afef..b546aedc62ea 100644 --- a/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.ts +++ b/packages/twenty-server/src/engine/core-modules/workspace/services/workspace.service.ts @@ -1,24 +1,24 @@ -import { InjectRepository } from '@nestjs/typeorm'; import { BadRequestException } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; import assert from 'assert'; import { TypeOrmQueryService } from '@ptc-org/nestjs-query-typeorm'; -import { Repository } from 'typeorm'; -import { SendInviteLinkEmail } from 'twenty-emails'; import { render } from '@react-email/render'; +import { SendInviteLinkEmail } from 'twenty-emails'; +import { Repository } from 'typeorm'; -import { WorkspaceManagerService } from 'src/engine/workspace-manager/workspace-manager.service'; -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; -import { User } from 'src/engine/core-modules/user/user.entity'; -import { ActivateWorkspaceInput } from 'src/engine/core-modules/workspace/dtos/activate-workspace-input'; +import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service'; +import { OnboardingService } from 'src/engine/core-modules/onboarding/onboarding.service'; import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity'; import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; -import { BillingService } from 'src/engine/core-modules/billing/billing.service'; +import { User } from 'src/engine/core-modules/user/user.entity'; +import { ActivateWorkspaceInput } from 'src/engine/core-modules/workspace/dtos/activate-workspace-input'; import { SendInviteLink } from 'src/engine/core-modules/workspace/dtos/send-invite-link.entity'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { EmailService } from 'src/engine/integrations/email/email.service'; import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { OnboardingService } from 'src/engine/core-modules/onboarding/onboarding.service'; +import { WorkspaceManagerService } from 'src/engine/workspace-manager/workspace-manager.service'; export class WorkspaceService extends TypeOrmQueryService<Workspace> { constructor( @@ -30,7 +30,7 @@ export class WorkspaceService extends TypeOrmQueryService<Workspace> { private readonly userWorkspaceRepository: Repository<UserWorkspace>, private readonly workspaceManagerService: WorkspaceManagerService, private readonly userWorkspaceService: UserWorkspaceService, - private readonly billingService: BillingService, + private readonly billingWorkspaceService: BillingWorkspaceService, private readonly environmentService: EnvironmentService, private readonly emailService: EmailService, private readonly onboardingService: OnboardingService, @@ -64,7 +64,7 @@ export class WorkspaceService extends TypeOrmQueryService<Workspace> { assert(workspace, 'Workspace not found'); await this.userWorkspaceRepository.delete({ workspaceId: id }); - await this.billingService.deleteSubscription(workspace.id); + await this.billingWorkspaceService.deleteSubscription(workspace.id); await this.workspaceManagerService.delete(id); @@ -119,6 +119,7 @@ export class WorkspaceService extends TypeOrmQueryService<Workspace> { link: inviteLink, workspace: { name: workspace.displayName, logo: workspace.logo }, sender: { email: sender.email, firstName: sender.firstName }, + serverUrl: this.environmentService.get('SERVER_URL'), }; const emailTemplate = SendInviteLinkEmail(emailData); const html = render(emailTemplate, { diff --git a/packages/twenty-server/src/engine/core-modules/workspace/workspace-workspace-member.listener.ts b/packages/twenty-server/src/engine/core-modules/workspace/workspace-workspace-member.listener.ts index dbb717fcfd15..240b4eb1cf5f 100644 --- a/packages/twenty-server/src/engine/core-modules/workspace/workspace-workspace-member.listener.ts +++ b/packages/twenty-server/src/engine/core-modules/workspace/workspace-workspace-member.listener.ts @@ -1,4 +1,4 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; @@ -9,11 +9,12 @@ import { HandleWorkspaceMemberDeletedJob, HandleWorkspaceMemberDeletedJobData, } from 'src/engine/core-modules/workspace/handle-workspace-member-deleted.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; @Injectable() export class WorkspaceWorkspaceMemberListener { constructor( - @Inject(MessageQueue.workspaceQueue) + @InjectMessageQueue(MessageQueue.workspaceQueue) private readonly messageQueueService: MessageQueueService, ) {} diff --git a/packages/twenty-server/src/engine/core-modules/workspace/workspace.entity.ts b/packages/twenty-server/src/engine/core-modules/workspace/workspace.entity.ts index edc141b5580d..6b4d2b425343 100644 --- a/packages/twenty-server/src/engine/core-modules/workspace/workspace.entity.ts +++ b/packages/twenty-server/src/engine/core-modules/workspace/workspace.entity.ts @@ -10,7 +10,6 @@ import { Relation, UpdateDateColumn, } from 'typeorm'; -import Stripe from 'stripe'; import { User } from 'src/engine/core-modules/user/user.entity'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; @@ -85,12 +84,8 @@ export class Workspace { @OneToMany(() => FeatureFlagEntity, (featureFlag) => featureFlag.workspace) featureFlags: Relation<FeatureFlagEntity[]>; - @Field(() => String) - @Column({ type: 'text', default: 'incomplete' }) - subscriptionStatus: Stripe.Subscription.Status; - @Field({ nullable: true }) - currentBillingSubscription: BillingSubscription; + workspaceMembersCount: number; @Field() activationStatus: 'active' | 'inactive'; diff --git a/packages/twenty-server/src/engine/core-modules/workspace/workspace.module.ts b/packages/twenty-server/src/engine/core-modules/workspace/workspace.module.ts index 86f416fc4101..5e9668cabbd8 100644 --- a/packages/twenty-server/src/engine/core-modules/workspace/workspace.module.ts +++ b/packages/twenty-server/src/engine/core-modules/workspace/workspace.module.ts @@ -3,20 +3,20 @@ import { Module } from '@nestjs/common'; import { NestjsQueryGraphQLModule } from '@ptc-org/nestjs-query-graphql'; import { NestjsQueryTypeOrmModule } from '@ptc-org/nestjs-query-typeorm'; -import { WorkspaceManagerModule } from 'src/engine/workspace-manager/workspace-manager.module'; -import { WorkspaceResolver } from 'src/engine/core-modules/workspace/workspace.resolver'; import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; +import { BillingModule } from 'src/engine/core-modules/billing/billing.module'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { FileUploadModule } from 'src/engine/core-modules/file/file-upload/file-upload.module'; +import { OnboardingModule } from 'src/engine/core-modules/onboarding/onboarding.module'; import { UserWorkspace } from 'src/engine/core-modules/user-workspace/user-workspace.entity'; import { UserWorkspaceModule } from 'src/engine/core-modules/user-workspace/user-workspace.module'; -import { BillingModule } from 'src/engine/core-modules/billing/billing.module'; -import { FileUploadModule } from 'src/engine/core-modules/file/file-upload/file-upload.module'; -import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; +import { UserWorkspaceResolver } from 'src/engine/core-modules/user-workspace/user-workspace.resolver'; +import { User } from 'src/engine/core-modules/user/user.entity'; import { WorkspaceWorkspaceMemberListener } from 'src/engine/core-modules/workspace/workspace-workspace-member.listener'; +import { WorkspaceResolver } from 'src/engine/core-modules/workspace/workspace.resolver'; +import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; -import { User } from 'src/engine/core-modules/user/user.entity'; -import { UserWorkspaceResolver } from 'src/engine/core-modules/user-workspace/user-workspace.resolver'; -import { OnboardingModule } from 'src/engine/core-modules/onboarding/onboarding.module'; +import { WorkspaceManagerModule } from 'src/engine/workspace-manager/workspace-manager.module'; import { workspaceAutoResolverOpts } from './workspace.auto-resolver-opts'; import { Workspace } from './workspace.entity'; diff --git a/packages/twenty-server/src/engine/core-modules/workspace/workspace.resolver.ts b/packages/twenty-server/src/engine/core-modules/workspace/workspace.resolver.ts index a30853ac2c54..5ead7800d66d 100644 --- a/packages/twenty-server/src/engine/core-modules/workspace/workspace.resolver.ts +++ b/packages/twenty-server/src/engine/core-modules/workspace/workspace.resolver.ts @@ -22,11 +22,12 @@ import { User } from 'src/engine/core-modules/user/user.entity'; import { AuthUser } from 'src/engine/decorators/auth/auth-user.decorator'; import { ActivateWorkspaceInput } from 'src/engine/core-modules/workspace/dtos/activate-workspace-input'; import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; -import { BillingService } from 'src/engine/core-modules/billing/billing.service'; +import { BillingWorkspaceService } from 'src/engine/core-modules/billing/billing.workspace-service'; import { DemoEnvGuard } from 'src/engine/guards/demo.env.guard'; import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; import { SendInviteLink } from 'src/engine/core-modules/workspace/dtos/send-invite-link.entity'; import { SendInviteLinkInput } from 'src/engine/core-modules/workspace/dtos/send-invite-link.input'; +import { UserWorkspaceService } from 'src/engine/core-modules/user-workspace/user-workspace.service'; import { Workspace } from './workspace.entity'; @@ -38,8 +39,9 @@ export class WorkspaceResolver { constructor( private readonly workspaceService: WorkspaceService, private readonly workspaceCacheVersionService: WorkspaceCacheVersionService, + private readonly userWorkspaceService: UserWorkspaceService, private readonly fileUploadService: FileUploadService, - private readonly billingService: BillingService, + private readonly billingWorkspaceService: BillingWorkspaceService, ) {} @Query(() => Workspace) @@ -116,15 +118,20 @@ export class WorkspaceResolver { return this.workspaceCacheVersionService.getVersion(workspace.id); } - @ResolveField(() => BillingSubscription) + @ResolveField(() => BillingSubscription, { nullable: true }) async currentBillingSubscription( @Parent() workspace: Workspace, ): Promise<BillingSubscription | null> { - return this.billingService.getCurrentBillingSubscription({ + return this.billingWorkspaceService.getCurrentBillingSubscription({ workspaceId: workspace.id, }); } + @ResolveField(() => Number) + async workspaceMembersCount(): Promise<number | undefined> { + return await this.userWorkspaceService.getWorkspaceMemberCount(); + } + @Mutation(() => SendInviteLink) async sendInviteLink( @Args() sendInviteLinkInput: SendInviteLinkInput, diff --git a/packages/twenty-server/src/engine/integrations/cache-storage/types/cache-storage-namespace.enum.ts b/packages/twenty-server/src/engine/integrations/cache-storage/types/cache-storage-namespace.enum.ts index 5ef0221bc969..e89fb00b0396 100644 --- a/packages/twenty-server/src/engine/integrations/cache-storage/types/cache-storage-namespace.enum.ts +++ b/packages/twenty-server/src/engine/integrations/cache-storage/types/cache-storage-namespace.enum.ts @@ -1,4 +1,5 @@ export enum CacheStorageNamespace { Messaging = 'messaging', + Calendar = 'calendar', WorkspaceSchema = 'workspaceSchema', } diff --git a/packages/twenty-server/src/engine/integrations/captcha/captcha.module.ts b/packages/twenty-server/src/engine/integrations/captcha/captcha.module.ts index 891bed2e5e12..506e51f42b2f 100644 --- a/packages/twenty-server/src/engine/integrations/captcha/captcha.module.ts +++ b/packages/twenty-server/src/engine/integrations/captcha/captcha.module.ts @@ -22,7 +22,7 @@ export class CaptchaModule { } switch (config.type) { - case CaptchaDriverType.GoogleRecatpcha: + case CaptchaDriverType.GoogleRecaptcha: return new GoogleRecaptchaDriver(config.options); case CaptchaDriverType.Turnstile: return new TurnstileDriver(config.options); diff --git a/packages/twenty-server/src/engine/integrations/captcha/interfaces/captcha.interface.ts b/packages/twenty-server/src/engine/integrations/captcha/interfaces/captcha.interface.ts index e7f0ca5620bd..f19e15fa2985 100644 --- a/packages/twenty-server/src/engine/integrations/captcha/interfaces/captcha.interface.ts +++ b/packages/twenty-server/src/engine/integrations/captcha/interfaces/captcha.interface.ts @@ -2,7 +2,7 @@ import { FactoryProvider, ModuleMetadata } from '@nestjs/common'; import { registerEnumType } from '@nestjs/graphql'; export enum CaptchaDriverType { - GoogleRecatpcha = 'google-recaptcha', + GoogleRecaptcha = 'google-recaptcha', Turnstile = 'turnstile', } @@ -15,8 +15,8 @@ export type CaptchaDriverOptions = { secretKey: string; }; -export interface GoogleRecatpchaDriverFactoryOptions { - type: CaptchaDriverType.GoogleRecatpcha; +export interface GoogleRecaptchaDriverFactoryOptions { + type: CaptchaDriverType.GoogleRecaptcha; options: CaptchaDriverOptions; } @@ -26,7 +26,7 @@ export interface TurnstileDriverFactoryOptions { } export type CaptchaModuleOptions = - | GoogleRecatpchaDriverFactoryOptions + | GoogleRecaptchaDriverFactoryOptions | TurnstileDriverFactoryOptions; export type CaptchaModuleAsyncOptions = { diff --git a/packages/twenty-server/src/engine/integrations/email/email-sender.job.ts b/packages/twenty-server/src/engine/integrations/email/email-sender.job.ts index 7b284d4b26c3..d97117d5063c 100644 --- a/packages/twenty-server/src/engine/integrations/email/email-sender.job.ts +++ b/packages/twenty-server/src/engine/integrations/email/email-sender.job.ts @@ -1,15 +1,15 @@ -import { Injectable } from '@nestjs/common'; - import { SendMailOptions } from 'nodemailer'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { EmailSenderService } from 'src/engine/integrations/email/email-sender.service'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; -@Injectable() -export class EmailSenderJob implements MessageQueueJob<SendMailOptions> { +@Processor(MessageQueue.emailQueue) +export class EmailSenderJob { constructor(private readonly emailSenderService: EmailSenderService) {} + @Process(EmailSenderJob.name) async handle(data: SendMailOptions): Promise<void> { await this.emailSenderService.send(data); } diff --git a/packages/twenty-server/src/engine/integrations/email/email.service.ts b/packages/twenty-server/src/engine/integrations/email/email.service.ts index a149c78023b2..79217adb8d1e 100644 --- a/packages/twenty-server/src/engine/integrations/email/email.service.ts +++ b/packages/twenty-server/src/engine/integrations/email/email.service.ts @@ -1,15 +1,16 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { SendMailOptions } from 'nodemailer'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { EmailSenderJob } from 'src/engine/integrations/email/email-sender.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; @Injectable() export class EmailService { constructor( - @Inject(MessageQueue.emailQueue) + @InjectMessageQueue(MessageQueue.emailQueue) private readonly messageQueueService: MessageQueueService, ) {} diff --git a/packages/twenty-server/src/engine/integrations/environment/environment-variables.ts b/packages/twenty-server/src/engine/integrations/environment/environment-variables.ts index 8d739dcea957..91dfabab3312 100644 --- a/packages/twenty-server/src/engine/integrations/environment/environment-variables.ts +++ b/packages/twenty-server/src/engine/integrations/environment/environment-variables.ts @@ -17,6 +17,8 @@ import { import { EmailDriver } from 'src/engine/integrations/email/interfaces/email.interface'; import { NodeEnvironment } from 'src/engine/integrations/environment/interfaces/node-environment.interface'; +import { LLMChatModelDriver } from 'src/engine/integrations/llm-chat-model/interfaces/llm-chat-model.interface'; +import { LLMTracingDriver } from 'src/engine/integrations/llm-tracing/interfaces/llm-tracing.interface'; import { assert } from 'src/utils/assert'; import { CastToStringArray } from 'src/engine/integrations/environment/decorators/cast-to-string-array.decorator'; @@ -220,6 +222,16 @@ export class EnvironmentVariables { @IsOptional() STORAGE_S3_ENDPOINT: string; + @ValidateIf((env) => env.STORAGE_TYPE === StorageDriverType.S3) + @IsString() + @IsOptional() + STORAGE_S3_ACCESS_KEY_ID: string; + + @ValidateIf((env) => env.STORAGE_TYPE === StorageDriverType.S3) + @IsString() + @IsOptional() + STORAGE_S3_SECRET_ACCESS_KEY: string; + @IsString() @ValidateIf((env) => env.STORAGE_TYPE === StorageDriverType.Local) STORAGE_LOCAL_PATH = '.local-storage'; @@ -324,7 +336,7 @@ export class EnvironmentVariables { @CastToPositiveNumber() @IsOptional() @IsNumber() - MUTATION_MAXIMUM_RECORD_AFFECTED = 100; + MUTATION_MAXIMUM_AFFECTED_RECORDS = 100; REDIS_HOST = '127.0.0.1'; @@ -359,6 +371,16 @@ export class EnvironmentVariables { OPENROUTER_API_KEY: string; + LLM_CHAT_MODEL_DRIVER: LLMChatModelDriver; + + OPENAI_API_KEY: string; + + LANGFUSE_SECRET_KEY: string; + + LANGFUSE_PUBLIC_KEY: string; + + LLM_TRACING_DRIVER: LLMTracingDriver = LLMTracingDriver.Console; + @CastToPositiveNumber() API_RATE_LIMITING_TTL = 100; @@ -376,6 +398,15 @@ export class EnvironmentVariables { AUTH_GOOGLE_APIS_CALLBACK_URL: string; CHROME_EXTENSION_ID: string; + + // --------------------------------------- + // Funnelmink + // --------------------------------------- + + @CastToBoolean() + @IsBoolean() + @IsOptional() + FUNNELMINK_PREFILL_NEW_WORKSPACES_WITH_FSM_OBJECTS = true; } export const validate = ( diff --git a/packages/twenty-server/src/engine/integrations/event-emitter/utils/__tests__/object-record-changed-values.spec.ts b/packages/twenty-server/src/engine/integrations/event-emitter/utils/__tests__/object-record-changed-values.spec.ts index 9da50ae4c899..9e4e51791a20 100644 --- a/packages/twenty-server/src/engine/integrations/event-emitter/utils/__tests__/object-record-changed-values.spec.ts +++ b/packages/twenty-server/src/engine/integrations/event-emitter/utils/__tests__/object-record-changed-values.spec.ts @@ -22,8 +22,16 @@ const mockObjectMetadata: ObjectMetadataInterface = { describe('objectRecordChangedValues', () => { it('detects changes in scalar values correctly', () => { - const oldRecord = { id: 1, name: 'Original Name', updatedAt: new Date() }; - const newRecord = { id: 1, name: 'Updated Name', updatedAt: new Date() }; + const oldRecord = { + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516m', + name: 'Original Name', + updatedAt: new Date().toString(), + }; + const newRecord = { + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516m', + name: 'Updated Name', + updatedAt: new Date().toString(), + }; const result = objectRecordChangedValues( oldRecord, @@ -38,8 +46,14 @@ describe('objectRecordChangedValues', () => { }); it('ignores changes to the updatedAt field', () => { - const oldRecord = { id: 1, updatedAt: new Date('2020-01-01') }; - const newRecord = { id: 1, updatedAt: new Date('2024-01-01') }; + const oldRecord = { + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516d', + updatedAt: new Date('2020-01-01').toDateString(), + }; + const newRecord = { + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516d', + updatedAt: new Date('2024-01-01').toDateString(), + }; const result = objectRecordChangedValues( oldRecord, @@ -51,8 +65,16 @@ it('ignores changes to the updatedAt field', () => { }); it('returns an empty object when there are no changes', () => { - const oldRecord = { id: 1, name: 'Name', value: 100 }; - const newRecord = { id: 1, name: 'Name', value: 100 }; + const oldRecord = { + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516k', + name: 'Name', + value: 100, + }; + const newRecord = { + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516k', + name: 'Name', + value: 100, + }; const result = objectRecordChangedValues( oldRecord, @@ -65,17 +87,17 @@ it('returns an empty object when there are no changes', () => { it('correctly handles a mix of changed, unchanged, and special case values', () => { const oldRecord = { - id: 1, + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516l', name: 'Original', status: 'active', - updatedAt: new Date(2020, 1, 1), + updatedAt: new Date(2020, 1, 1).toDateString(), config: { theme: 'dark' }, }; const newRecord = { - id: 1, + id: '74316f58-29b0-4a6a-b8fa-d2b506d5516l', name: 'Updated', status: 'active', - updatedAt: new Date(2021, 1, 1), + updatedAt: new Date(2021, 1, 1).toDateString(), config: { theme: 'light' }, }; const expectedChanges = { diff --git a/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-properties.util.ts b/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-properties.util.ts index 53e2c7658be5..5d77c207623c 100644 --- a/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-properties.util.ts +++ b/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-properties.util.ts @@ -1,8 +1,14 @@ import deepEqual from 'deep-equal'; -export const objectRecordChangedProperties = ( - oldRecord: Record<string, any>, - newRecord: Record<string, any>, +import { Record } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; + +import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; + +export const objectRecordChangedProperties = < + PRecord extends Partial<Record | BaseWorkspaceEntity> = Partial<Record>, +>( + oldRecord: PRecord, + newRecord: PRecord, ) => { const changedProperties = Object.keys(newRecord).filter( (key) => !deepEqual(oldRecord[key], newRecord[key]), diff --git a/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-values.ts b/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-values.ts index ff300042d96a..062693cbd5d1 100644 --- a/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-values.ts +++ b/packages/twenty-server/src/engine/integrations/event-emitter/utils/object-record-changed-values.ts @@ -1,12 +1,13 @@ import deepEqual from 'deep-equal'; import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; +import { Record as IRecord } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; export const objectRecordChangedValues = ( - oldRecord: Record<string, any>, - newRecord: Record<string, any>, + oldRecord: Partial<IRecord>, + newRecord: Partial<IRecord>, objectMetadata: ObjectMetadataInterface, ) => { const changedValues = Object.keys(newRecord).reduce( diff --git a/packages/twenty-server/src/engine/integrations/exception-handler/hooks/use-exception-handler.hook.ts b/packages/twenty-server/src/engine/integrations/exception-handler/hooks/use-exception-handler.hook.ts index cdfae3bf7608..bb4522796c34 100644 --- a/packages/twenty-server/src/engine/integrations/exception-handler/hooks/use-exception-handler.hook.ts +++ b/packages/twenty-server/src/engine/integrations/exception-handler/hooks/use-exception-handler.hook.ts @@ -1,10 +1,10 @@ -import { GraphQLError, Kind, OperationDefinitionNode, print } from 'graphql'; import { - getDocumentString, - handleStreamOrSingleExecutionResult, OnExecuteDoneHookResultOnNextHook, Plugin, + getDocumentString, + handleStreamOrSingleExecutionResult, } from '@envelop/core'; +import { GraphQLError, Kind, OperationDefinitionNode, print } from 'graphql'; import { GraphQLContext } from 'src/engine/api/graphql/graphql-config/interfaces/graphql-context.interface'; @@ -26,6 +26,9 @@ export type ExceptionHandlerPluginOptions = { eventIdKey?: string | null; }; +// This hook is deprecated. +// We should either handle exception in the context of graphql, controller or command +// @deprecated export const useExceptionHandler = <PluginContext extends GraphQLContext>( options: ExceptionHandlerPluginOptions, ): Plugin<PluginContext> => { diff --git a/packages/twenty-server/src/engine/integrations/file-storage/file-storage.module-factory.ts b/packages/twenty-server/src/engine/integrations/file-storage/file-storage.module-factory.ts index f846dc96df89..b1ab03642ef3 100644 --- a/packages/twenty-server/src/engine/integrations/file-storage/file-storage.module-factory.ts +++ b/packages/twenty-server/src/engine/integrations/file-storage/file-storage.module-factory.ts @@ -32,15 +32,24 @@ export const fileStorageModuleFactory = async ( const bucketName = environmentService.get('STORAGE_S3_NAME'); const endpoint = environmentService.get('STORAGE_S3_ENDPOINT'); const region = environmentService.get('STORAGE_S3_REGION'); + const accessKeyId = environmentService.get('STORAGE_S3_ACCESS_KEY_ID'); + const secretAccessKey = environmentService.get( + 'STORAGE_S3_SECRET_ACCESS_KEY', + ); return { type: StorageDriverType.S3, options: { bucketName: bucketName ?? '', endpoint: endpoint, - credentials: fromNodeProviderChain({ - clientConfig: { region }, - }), + credentials: accessKeyId + ? { + accessKeyId, + secretAccessKey, + } + : fromNodeProviderChain({ + clientConfig: { region }, + }), forcePathStyle: true, region: region ?? '', }, diff --git a/packages/twenty-server/src/engine/integrations/integrations.module.ts b/packages/twenty-server/src/engine/integrations/integrations.module.ts index df855478af6e..fb2f40051ec7 100644 --- a/packages/twenty-server/src/engine/integrations/integrations.module.ts +++ b/packages/twenty-server/src/engine/integrations/integrations.module.ts @@ -12,6 +12,10 @@ import { emailModuleFactory } from 'src/engine/integrations/email/email.module-f import { CacheStorageModule } from 'src/engine/integrations/cache-storage/cache-storage.module'; import { CaptchaModule } from 'src/engine/integrations/captcha/captcha.module'; import { captchaModuleFactory } from 'src/engine/integrations/captcha/captcha.module-factory'; +import { LLMChatModelModule } from 'src/engine/integrations/llm-chat-model/llm-chat-model.module'; +import { llmChatModelModuleFactory } from 'src/engine/integrations/llm-chat-model/llm-chat-model.module-factory'; +import { LLMTracingModule } from 'src/engine/integrations/llm-tracing/llm-tracing.module'; +import { llmTracingModuleFactory } from 'src/engine/integrations/llm-tracing/llm-tracing.module-factory'; import { EnvironmentModule } from './environment/environment.module'; import { EnvironmentService } from './environment/environment.service'; @@ -30,7 +34,7 @@ import { MessageQueueModule } from './message-queue/message-queue.module'; useFactory: loggerModuleFactory, inject: [EnvironmentService], }), - MessageQueueModule.forRoot({ + MessageQueueModule.registerAsync({ useFactory: messageQueueModuleFactory, inject: [EnvironmentService], }), @@ -50,6 +54,14 @@ import { MessageQueueModule } from './message-queue/message-queue.module'; wildcard: true, }), CacheStorageModule, + LLMChatModelModule.forRoot({ + useFactory: llmChatModelModuleFactory, + inject: [EnvironmentService], + }), + LLMTracingModule.forRoot({ + useFactory: llmTracingModuleFactory, + inject: [EnvironmentService], + }), ], exports: [], providers: [], diff --git a/packages/twenty-server/src/engine/integrations/llm-chat-model/drivers/interfaces/llm-prompt-template-driver.interface.ts b/packages/twenty-server/src/engine/integrations/llm-chat-model/drivers/interfaces/llm-prompt-template-driver.interface.ts new file mode 100644 index 000000000000..cef61eccd353 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-chat-model/drivers/interfaces/llm-prompt-template-driver.interface.ts @@ -0,0 +1,5 @@ +import { BaseChatModel } from '@langchain/core/language_models/chat_models'; + +export interface LLMChatModelDriver { + getJSONChatModel(): BaseChatModel; +} diff --git a/packages/twenty-server/src/engine/integrations/llm-chat-model/drivers/openai.driver.ts b/packages/twenty-server/src/engine/integrations/llm-chat-model/drivers/openai.driver.ts new file mode 100644 index 000000000000..652a854ef831 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-chat-model/drivers/openai.driver.ts @@ -0,0 +1,22 @@ +import { BaseChatModel } from '@langchain/core/language_models/chat_models'; +import { ChatOpenAI } from '@langchain/openai'; + +import { LLMChatModelDriver } from 'src/engine/integrations/llm-chat-model/drivers/interfaces/llm-prompt-template-driver.interface'; + +export class OpenAIDriver implements LLMChatModelDriver { + private chatModel: BaseChatModel; + + constructor() { + this.chatModel = new ChatOpenAI({ + model: 'gpt-4o', + }).bind({ + response_format: { + type: 'json_object', + }, + }) as unknown as BaseChatModel; + } + + getJSONChatModel() { + return this.chatModel; + } +} diff --git a/packages/twenty-server/src/engine/integrations/llm-chat-model/interfaces/llm-chat-model.interface.ts b/packages/twenty-server/src/engine/integrations/llm-chat-model/interfaces/llm-chat-model.interface.ts new file mode 100644 index 000000000000..5c6edbcd0237 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-chat-model/interfaces/llm-chat-model.interface.ts @@ -0,0 +1,14 @@ +import { ModuleMetadata, FactoryProvider } from '@nestjs/common'; + +export enum LLMChatModelDriver { + OpenAI = 'openai', +} + +export interface LLMChatModelModuleOptions { + type: LLMChatModelDriver; +} + +export type LLMChatModelModuleAsyncOptions = { + useFactory: (...args: any[]) => LLMChatModelModuleOptions | undefined; +} & Pick<ModuleMetadata, 'imports'> & + Pick<FactoryProvider, 'inject'>; diff --git a/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.constants.ts b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.constants.ts new file mode 100644 index 000000000000..e6c3ea7b0e5c --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.constants.ts @@ -0,0 +1 @@ +export const LLM_CHAT_MODEL_DRIVER = Symbol('LLM_CHAT_MODEL_DRIVER'); diff --git a/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.module-factory.ts b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.module-factory.ts new file mode 100644 index 000000000000..2d91f280ce54 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.module-factory.ts @@ -0,0 +1,17 @@ +import { LLMChatModelDriver } from 'src/engine/integrations/llm-chat-model/interfaces/llm-chat-model.interface'; + +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; + +export const llmChatModelModuleFactory = ( + environmentService: EnvironmentService, +) => { + const driver = environmentService.get('LLM_CHAT_MODEL_DRIVER'); + + switch (driver) { + case LLMChatModelDriver.OpenAI: { + return { type: LLMChatModelDriver.OpenAI }; + } + default: + // `No LLM chat model driver (${driver})`); + } +}; diff --git a/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.module.ts b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.module.ts new file mode 100644 index 000000000000..279993f72868 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.module.ts @@ -0,0 +1,35 @@ +import { DynamicModule, Global } from '@nestjs/common'; + +import { + LLMChatModelDriver, + LLMChatModelModuleAsyncOptions, +} from 'src/engine/integrations/llm-chat-model/interfaces/llm-chat-model.interface'; + +import { LLM_CHAT_MODEL_DRIVER } from 'src/engine/integrations/llm-chat-model/llm-chat-model.constants'; +import { OpenAIDriver } from 'src/engine/integrations/llm-chat-model/drivers/openai.driver'; +import { LLMChatModelService } from 'src/engine/integrations/llm-chat-model/llm-chat-model.service'; + +@Global() +export class LLMChatModelModule { + static forRoot(options: LLMChatModelModuleAsyncOptions): DynamicModule { + const provider = { + provide: LLM_CHAT_MODEL_DRIVER, + useFactory: (...args: any[]) => { + const config = options.useFactory(...args); + + switch (config?.type) { + case LLMChatModelDriver.OpenAI: { + return new OpenAIDriver(); + } + } + }, + inject: options.inject || [], + }; + + return { + module: LLMChatModelModule, + providers: [LLMChatModelService, provider], + exports: [LLMChatModelService], + }; + } +} diff --git a/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.service.ts b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.service.ts new file mode 100644 index 000000000000..62beea8c6eca --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-chat-model/llm-chat-model.service.ts @@ -0,0 +1,16 @@ +import { Injectable, Inject } from '@nestjs/common'; + +import { LLMChatModelDriver } from 'src/engine/integrations/llm-chat-model/drivers/interfaces/llm-prompt-template-driver.interface'; + +import { LLM_CHAT_MODEL_DRIVER } from 'src/engine/integrations/llm-chat-model/llm-chat-model.constants'; + +@Injectable() +export class LLMChatModelService { + constructor( + @Inject(LLM_CHAT_MODEL_DRIVER) private driver: LLMChatModelDriver, + ) {} + + getJSONChatModel() { + return this.driver.getJSONChatModel(); + } +} diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/console.driver.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/console.driver.ts new file mode 100644 index 000000000000..47e126324d97 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/console.driver.ts @@ -0,0 +1,25 @@ +import { BaseCallbackHandler } from '@langchain/core/callbacks/base'; +import { ConsoleCallbackHandler } from '@langchain/core/tracers/console'; +import { Run } from '@langchain/core/tracers/base'; + +import { LLMTracingDriver } from 'src/engine/integrations/llm-tracing/drivers/interfaces/llm-tracing-driver.interface'; + +class WithMetadataConsoleCallbackHandler extends ConsoleCallbackHandler { + private metadata: Record<string, unknown>; + + constructor(metadata: Record<string, unknown>) { + super(); + this.metadata = metadata; + } + + onChainStart(run: Run) { + console.log(`Chain metadata: ${JSON.stringify(this.metadata)}`); + super.onChainStart(run); + } +} + +export class ConsoleDriver implements LLMTracingDriver { + getCallbackHandler(metadata: Record<string, unknown>): BaseCallbackHandler { + return new WithMetadataConsoleCallbackHandler(metadata); + } +} diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/interfaces/llm-tracing-driver.interface.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/interfaces/llm-tracing-driver.interface.ts new file mode 100644 index 000000000000..fe1944a60c08 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/interfaces/llm-tracing-driver.interface.ts @@ -0,0 +1,5 @@ +import { BaseCallbackHandler } from '@langchain/core/callbacks/base'; + +export interface LLMTracingDriver { + getCallbackHandler(metadata: Record<string, unknown>): BaseCallbackHandler; +} diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/langfuse.driver.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/langfuse.driver.ts new file mode 100644 index 000000000000..b9b84aad0860 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/drivers/langfuse.driver.ts @@ -0,0 +1,26 @@ +import { BaseCallbackHandler } from '@langchain/core/callbacks/base'; +import CallbackHandler from 'langfuse-langchain'; + +import { LLMTracingDriver } from 'src/engine/integrations/llm-tracing/drivers/interfaces/llm-tracing-driver.interface'; + +export interface LangfuseDriverOptions { + secretKey: string; + publicKey: string; +} + +export class LangfuseDriver implements LLMTracingDriver { + private options: LangfuseDriverOptions; + + constructor(options: LangfuseDriverOptions) { + this.options = options; + } + + getCallbackHandler(metadata: Record<string, unknown>): BaseCallbackHandler { + return new CallbackHandler({ + secretKey: this.options.secretKey, + publicKey: this.options.publicKey, + baseUrl: 'https://cloud.langfuse.com', + metadata: metadata, + }); + } +} diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/interfaces/llm-tracing.interface.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/interfaces/llm-tracing.interface.ts new file mode 100644 index 000000000000..a97031499b10 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/interfaces/llm-tracing.interface.ts @@ -0,0 +1,26 @@ +import { ModuleMetadata, FactoryProvider } from '@nestjs/common'; + +import { LangfuseDriverOptions } from 'src/engine/integrations/llm-tracing/drivers/langfuse.driver'; + +export enum LLMTracingDriver { + Langfuse = 'langfuse', + Console = 'console', +} + +export interface LangfuseDriverFactoryOptions { + type: LLMTracingDriver.Langfuse; + options: LangfuseDriverOptions; +} + +export interface ConsoleDriverFactoryOptions { + type: LLMTracingDriver.Console; +} + +export type LLMTracingModuleOptions = + | LangfuseDriverFactoryOptions + | ConsoleDriverFactoryOptions; + +export type LLMTracingModuleAsyncOptions = { + useFactory: (...args: any[]) => LLMTracingModuleOptions; +} & Pick<ModuleMetadata, 'imports'> & + Pick<FactoryProvider, 'inject'>; diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.constants.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.constants.ts new file mode 100644 index 000000000000..92371f0e4e88 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.constants.ts @@ -0,0 +1 @@ +export const LLM_TRACING_DRIVER = Symbol('LLM_TRACING_DRIVER'); diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.module-factory.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.module-factory.ts new file mode 100644 index 000000000000..754158e2a81e --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.module-factory.ts @@ -0,0 +1,34 @@ +import { LLMTracingDriver } from 'src/engine/integrations/llm-tracing/interfaces/llm-tracing.interface'; + +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; + +export const llmTracingModuleFactory = ( + environmentService: EnvironmentService, +) => { + const driver = environmentService.get('LLM_TRACING_DRIVER'); + + switch (driver) { + case LLMTracingDriver.Console: { + return { type: LLMTracingDriver.Console as const }; + } + case LLMTracingDriver.Langfuse: { + const secretKey = environmentService.get('LANGFUSE_SECRET_KEY'); + const publicKey = environmentService.get('LANGFUSE_PUBLIC_KEY'); + + if (!(secretKey && publicKey)) { + throw new Error( + `${driver} LLM tracing driver requires LANGFUSE_SECRET_KEY and LANGFUSE_PUBLIC_KEY to be defined, check your .env file`, + ); + } + + return { + type: LLMTracingDriver.Langfuse as const, + options: { secretKey, publicKey }, + }; + } + default: + throw new Error( + `Invalid LLM tracing driver (${driver}), check your .env file`, + ); + } +}; diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.module.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.module.ts new file mode 100644 index 000000000000..9e9c452e95ba --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.module.ts @@ -0,0 +1,39 @@ +import { Global, DynamicModule } from '@nestjs/common'; + +import { + LLMTracingModuleAsyncOptions, + LLMTracingDriver, +} from 'src/engine/integrations/llm-tracing/interfaces/llm-tracing.interface'; + +import { LangfuseDriver } from 'src/engine/integrations/llm-tracing/drivers/langfuse.driver'; +import { ConsoleDriver } from 'src/engine/integrations/llm-tracing/drivers/console.driver'; +import { LLMTracingService } from 'src/engine/integrations/llm-tracing/llm-tracing.service'; +import { LLM_TRACING_DRIVER } from 'src/engine/integrations/llm-tracing/llm-tracing.constants'; + +@Global() +export class LLMTracingModule { + static forRoot(options: LLMTracingModuleAsyncOptions): DynamicModule { + const provider = { + provide: LLM_TRACING_DRIVER, + useFactory: (...args: any[]) => { + const config = options.useFactory(...args); + + switch (config.type) { + case LLMTracingDriver.Langfuse: { + return new LangfuseDriver(config.options); + } + case LLMTracingDriver.Console: { + return new ConsoleDriver(); + } + } + }, + inject: options.inject || [], + }; + + return { + module: LLMTracingModule, + providers: [LLMTracingService, provider], + exports: [LLMTracingService], + }; + } +} diff --git a/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.service.ts b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.service.ts new file mode 100644 index 000000000000..6ff2023902d1 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/llm-tracing/llm-tracing.service.ts @@ -0,0 +1,16 @@ +import { Injectable, Inject } from '@nestjs/common'; + +import { BaseCallbackHandler } from '@langchain/core/callbacks/base'; + +import { LLMTracingDriver } from 'src/engine/integrations/llm-tracing/drivers/interfaces/llm-tracing-driver.interface'; + +import { LLM_TRACING_DRIVER } from 'src/engine/integrations/llm-tracing/llm-tracing.constants'; + +@Injectable() +export class LLMTracingService { + constructor(@Inject(LLM_TRACING_DRIVER) private driver: LLMTracingDriver) {} + + getCallbackHandler(metadata: Record<string, unknown>): BaseCallbackHandler { + return this.driver.getCallbackHandler(metadata); + } +} diff --git a/packages/twenty-server/src/engine/integrations/message-queue/decorators/message-queue.decorator.ts b/packages/twenty-server/src/engine/integrations/message-queue/decorators/message-queue.decorator.ts index 69fa50ad5ef7..5a275d3d5b82 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/decorators/message-queue.decorator.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/decorators/message-queue.decorator.ts @@ -1,7 +1,8 @@ import { Inject } from '@nestjs/common'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { getQueueToken } from 'src/engine/integrations/message-queue/utils/get-queue-token.util'; -export const InjectMessageQueue = (messageQueueName: MessageQueue) => { - return Inject(messageQueueName); +export const InjectMessageQueue = (queueName: MessageQueue) => { + return Inject(getQueueToken(queueName)); }; diff --git a/packages/twenty-server/src/engine/integrations/message-queue/decorators/process.decorator.ts b/packages/twenty-server/src/engine/integrations/message-queue/decorators/process.decorator.ts new file mode 100644 index 000000000000..214742cb0af0 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/decorators/process.decorator.ts @@ -0,0 +1,21 @@ +import { SetMetadata } from '@nestjs/common'; +import { isString } from '@nestjs/common/utils/shared.utils'; + +import { PROCESS_METADATA } from 'src/engine/integrations/message-queue/message-queue.constants'; + +export interface MessageQueueProcessOptions { + jobName: string; + concurrency?: number; +} + +export function Process(jobName: string): MethodDecorator; +export function Process(options: MessageQueueProcessOptions): MethodDecorator; +export function Process( + nameOrOptions: string | MessageQueueProcessOptions, +): MethodDecorator { + const options = isString(nameOrOptions) + ? { jobName: nameOrOptions } + : nameOrOptions; + + return SetMetadata(PROCESS_METADATA, options || {}); +} diff --git a/packages/twenty-server/src/engine/integrations/message-queue/decorators/processor.decorator.ts b/packages/twenty-server/src/engine/integrations/message-queue/decorators/processor.decorator.ts new file mode 100644 index 000000000000..a244dc4d14ce --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/decorators/processor.decorator.ts @@ -0,0 +1,69 @@ +import { Scope, SetMetadata } from '@nestjs/common'; +import { SCOPE_OPTIONS_METADATA } from '@nestjs/common/constants'; + +import { MessageQueueWorkerOptions } from 'src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface'; + +import { + MessageQueue, + PROCESSOR_METADATA, + WORKER_METADATA, +} from 'src/engine/integrations/message-queue/message-queue.constants'; + +export interface MessageQueueProcessorOptions { + /** + * Specifies the name of the queue to subscribe to. + */ + queueName: MessageQueue; + /** + * Specifies the lifetime of an injected Processor. + */ + scope?: Scope; +} + +/** + * Represents a worker that is able to process jobs from the queue. + * @param queueName name of the queue to process + */ +export function Processor(queueName: string): ClassDecorator; +/** + * Represents a worker that is able to process jobs from the queue. + * @param queueName name of the queue to process + * @param workerOptions additional worker options + */ +export function Processor( + queueName: string, + workerOptions: MessageQueueWorkerOptions, +): ClassDecorator; +/** + * Represents a worker that is able to process jobs from the queue. + * @param processorOptions processor options + */ +export function Processor( + processorOptions: MessageQueueProcessorOptions, +): ClassDecorator; +/** + * Represents a worker that is able to process jobs from the queue. + * @param processorOptions processor options (Nest-specific) + * @param workerOptions additional Bull worker options + */ +export function Processor( + processorOptions: MessageQueueProcessorOptions, + workerOptions: MessageQueueWorkerOptions, +): ClassDecorator; +export function Processor( + queueNameOrOptions?: string | MessageQueueProcessorOptions, + maybeWorkerOptions?: MessageQueueWorkerOptions, +): ClassDecorator { + const options = + queueNameOrOptions && typeof queueNameOrOptions === 'object' + ? queueNameOrOptions + : { queueName: queueNameOrOptions }; + + // eslint-disable-next-line @typescript-eslint/ban-types + return (target: Function) => { + SetMetadata(SCOPE_OPTIONS_METADATA, options)(target); + SetMetadata(PROCESSOR_METADATA, options)(target); + maybeWorkerOptions && + SetMetadata(WORKER_METADATA, maybeWorkerOptions)(target); + }; +} diff --git a/packages/twenty-server/src/engine/integrations/message-queue/drivers/bullmq.driver.ts b/packages/twenty-server/src/engine/integrations/message-queue/drivers/bullmq.driver.ts index fc78eaabb064..c3324a500604 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/drivers/bullmq.driver.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/drivers/bullmq.driver.ts @@ -1,9 +1,14 @@ +import { OnModuleDestroy } from '@nestjs/common'; + +import omitBy from 'lodash.omitby'; import { JobsOptions, Queue, QueueOptions, Worker } from 'bullmq'; import { QueueCronJobOptions, QueueJobOptions, } from 'src/engine/integrations/message-queue/drivers/interfaces/job-options.interface'; +import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { MessageQueueWorkerOptions } from 'src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; @@ -11,7 +16,7 @@ import { MessageQueueDriver } from './interfaces/message-queue-driver.interface' export type BullMQDriverOptions = QueueOptions; -export class BullMQDriver implements MessageQueueDriver { +export class BullMQDriver implements MessageQueueDriver, OnModuleDestroy { private queueMap: Record<MessageQueue, Queue> = {} as Record< MessageQueue, Queue @@ -27,7 +32,7 @@ export class BullMQDriver implements MessageQueueDriver { this.queueMap[queueName] = new Queue(queueName, this.options); } - async stop() { + async onModuleDestroy() { const workers = Object.values(this.workerMap); const queues = Object.values(this.queueMap); @@ -39,14 +44,22 @@ export class BullMQDriver implements MessageQueueDriver { async work<T>( queueName: MessageQueue, - handler: ({ data, id }: { data: T; id: string }) => Promise<void>, + handler: (job: MessageQueueJob<T>) => Promise<void>, + options?: MessageQueueWorkerOptions, ) { const worker = new Worker( queueName, async (job) => { - await handler(job as { data: T; id: string }); + // TODO: Correctly support for job.id + await handler({ data: job.data, id: job.id ?? '', name: job.name }); }, - this.options, + omitBy( + { + ...this.options, + concurrency: options?.concurrency, + }, + (value) => value === undefined, + ), ); this.workerMap[queueName] = worker; diff --git a/packages/twenty-server/src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface.ts b/packages/twenty-server/src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface.ts index cdd30913f439..1d53837d359a 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface.ts @@ -3,6 +3,7 @@ import { QueueJobOptions, } from 'src/engine/integrations/message-queue/drivers/interfaces/job-options.interface'; import { MessageQueueJobData } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { MessageQueueWorkerOptions } from 'src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; @@ -16,6 +17,7 @@ export interface MessageQueueDriver { work<T extends MessageQueueJobData>( queueName: MessageQueue, handler: ({ data, id }: { data: T; id: string }) => Promise<void> | void, + options?: MessageQueueWorkerOptions, ); addCron<T extends MessageQueueJobData | undefined>( queueName: MessageQueue, @@ -24,6 +26,5 @@ export interface MessageQueueDriver { options?: QueueCronJobOptions, ); removeCron(queueName: MessageQueue, jobName: string, pattern?: string); - stop?(): Promise<void>; register?(queueName: MessageQueue): void; } diff --git a/packages/twenty-server/src/engine/integrations/message-queue/drivers/pg-boss.driver.ts b/packages/twenty-server/src/engine/integrations/message-queue/drivers/pg-boss.driver.ts index 5210593c3ff0..98631bc88121 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/drivers/pg-boss.driver.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/drivers/pg-boss.driver.ts @@ -1,9 +1,13 @@ +import { OnModuleDestroy, OnModuleInit } from '@nestjs/common'; + import PgBoss from 'pg-boss'; import { QueueCronJobOptions, QueueJobOptions, } from 'src/engine/integrations/message-queue/drivers/interfaces/job-options.interface'; +import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { MessageQueueWorkerOptions } from 'src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; @@ -13,26 +17,50 @@ export type PgBossDriverOptions = PgBoss.ConstructorOptions; const DEFAULT_PG_BOSS_CRON_PATTERN_WHEN_NOT_PROVIDED = '*/1 * * * *'; -export class PgBossDriver implements MessageQueueDriver { +export class PgBossDriver + implements MessageQueueDriver, OnModuleInit, OnModuleDestroy +{ private pgBoss: PgBoss; constructor(options: PgBossDriverOptions) { this.pgBoss = new PgBoss(options); } - async stop() { - await this.pgBoss.stop(); + async onModuleInit() { + await this.pgBoss.start(); } - async init(): Promise<void> { - await this.pgBoss.start(); + async onModuleDestroy() { + await this.pgBoss.stop(); } async work<T>( queueName: string, - handler: ({ data, id }: { data: T; id: string }) => Promise<void>, + handler: (job: MessageQueueJob<T>) => Promise<void>, + options?: MessageQueueWorkerOptions, ) { - return this.pgBoss.work(`${queueName}.*`, handler); + return this.pgBoss.work<T>( + `${queueName}.*`, + options?.concurrency + ? { + teamConcurrency: options.concurrency, + } + : {}, + async (job) => { + // PGBoss work with wildcard job name + const jobName = job.name.split('.')?.[1]; + + if (!jobName) { + throw new Error('Job name could not be splited from the job.'); + } + + await handler({ + data: job.data, + id: job.id, + name: jobName, + }); + }, + ); } async addCron<T>( diff --git a/packages/twenty-server/src/engine/integrations/message-queue/drivers/sync.driver.ts b/packages/twenty-server/src/engine/integrations/message-queue/drivers/sync.driver.ts index ca48a0236510..7d9d5cca3faa 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/drivers/sync.driver.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/drivers/sync.driver.ts @@ -1,57 +1,66 @@ -import { ModuleRef } from '@nestjs/core'; import { Logger } from '@nestjs/common'; -import { MessageQueueDriver } from 'src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface'; import { - MessageQueueCronJobData, - MessageQueueJob, MessageQueueJobData, + MessageQueueJob, } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; -import { getJobClassName } from 'src/engine/integrations/message-queue/utils/get-job-class-name.util'; + +import { MessageQueueDriver } from './interfaces/message-queue-driver.interface'; export class SyncDriver implements MessageQueueDriver { private readonly logger = new Logger(SyncDriver.name); - constructor(private readonly jobsModuleRef: ModuleRef) {} + private workersMap: { + [queueName: string]: (job: MessageQueueJob<any>) => Promise<void> | void; + } = {}; + + constructor() {} async add<T extends MessageQueueJobData>( - _queueName: MessageQueue, + queueName: MessageQueue, jobName: string, data: T, ): Promise<void> { - const jobClassName = getJobClassName(jobName); - const job: MessageQueueJob<MessageQueueJobData> = this.jobsModuleRef.get( - jobClassName, - { strict: false }, - ); - - await job.handle(data); + await this.processJob(queueName, { id: '', name: jobName, data }); } async addCron<T extends MessageQueueJobData | undefined>( - _queueName: MessageQueue, + queueName: MessageQueue, jobName: string, data: T, ): Promise<void> { this.logger.log(`Running cron job with SyncDriver`); - - const jobClassName = getJobClassName(jobName); - const job: MessageQueueCronJobData<MessageQueueJobData | undefined> = - this.jobsModuleRef.get(jobClassName, { - strict: true, - }); - - await job.handle(data); + await this.processJob(queueName, { + id: '', + name: jobName, + // TODO: Fix this type issue + data: data as any, + }); } - async removeCron(_queueName: MessageQueue, jobName: string) { - this.logger.log(`Removing '${jobName}' cron job with SyncDriver`); + async removeCron(queueName: MessageQueue) { + this.logger.log(`Removing '${queueName}' cron job with SyncDriver`); + } - return; + work<T extends MessageQueueJobData>( + queueName: MessageQueue, + handler: (job: MessageQueueJob<T>) => Promise<void> | void, + ) { + this.logger.log(`Registering handler for queue: ${queueName}`); + this.workersMap[queueName] = handler; } - work() { - return; + async processJob<T extends MessageQueueJobData>( + queueName: string, + job: MessageQueueJob<T>, + ) { + const worker = this.workersMap[queueName]; + + if (worker) { + await worker(job); + } else { + this.logger.error(`No handler found for job: ${queueName}`); + } } } diff --git a/packages/twenty-server/src/engine/integrations/message-queue/interfaces/index.ts b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/index.ts index 300859d017d2..4fdb52388cd2 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/interfaces/index.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/index.ts @@ -1 +1 @@ -export * from './message-queue.interface'; +export * from './message-queue-module-options.interface'; diff --git a/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-job.interface.ts b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-job.interface.ts index 87423ffd2bee..8a1ced80ec50 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-job.interface.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-job.interface.ts @@ -1,5 +1,7 @@ -export interface MessageQueueJob<T extends MessageQueueJobData | undefined> { - handle(data: T): Promise<void> | void; +export interface MessageQueueJob<T = any> { + id: string; + name: string; + data: T; } export interface MessageQueueCronJobData< diff --git a/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue.interface.ts b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-module-options.interface.ts similarity index 72% rename from packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue.interface.ts rename to packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-module-options.interface.ts index 568e0819fc8e..b98990385204 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue.interface.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-module-options.interface.ts @@ -1,5 +1,3 @@ -import { FactoryProvider, ModuleMetadata } from '@nestjs/common'; - import { BullMQDriverOptions } from 'src/engine/integrations/message-queue/drivers/bullmq.driver'; import { PgBossDriverOptions } from 'src/engine/integrations/message-queue/drivers/pg-boss.driver'; @@ -28,10 +26,3 @@ export type MessageQueueModuleOptions = | PgBossDriverFactoryOptions | BullMQDriverFactoryOptions | SyncDriverFactoryOptions; - -export type MessageQueueModuleAsyncOptions = { - useFactory: ( - ...args: any[] - ) => MessageQueueModuleOptions | Promise<MessageQueueModuleOptions>; -} & Pick<ModuleMetadata, 'imports'> & - Pick<FactoryProvider, 'inject'>; diff --git a/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface.ts b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface.ts new file mode 100644 index 000000000000..c4def7121eb0 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface.ts @@ -0,0 +1,3 @@ +export interface MessageQueueWorkerOptions { + concurrency?: number; +} diff --git a/packages/twenty-server/src/engine/integrations/message-queue/jobs.module.ts b/packages/twenty-server/src/engine/integrations/message-queue/jobs.module.ts index 5447402d603b..b8eee0bbbb90 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/jobs.module.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/jobs.module.ts @@ -11,21 +11,18 @@ import { StripeModule } from 'src/engine/core-modules/billing/stripe/stripe.modu import { UserWorkspaceModule } from 'src/engine/core-modules/user-workspace/user-workspace.module'; import { UserModule } from 'src/engine/core-modules/user/user.module'; import { HandleWorkspaceMemberDeletedJob } from 'src/engine/core-modules/workspace/handle-workspace-member-deleted.job'; +import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; import { EmailSenderJob } from 'src/engine/integrations/email/email-sender.job'; import { EmailModule } from 'src/engine/integrations/email/email.module'; import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; import { CleanInactiveWorkspaceJob } from 'src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.job'; -import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module'; -import { TimelineActivityModule } from 'src/modules/timeline/timeline-activity.module'; -import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; -import { CalendarMessagingParticipantJobModule } from 'src/modules/calendar-messaging-participant/jobs/calendar-messaging-participant-job.module'; -import { CalendarCronJobModule } from 'src/modules/calendar/crons/jobs/calendar-cron-job.module'; -import { CalendarJobModule } from 'src/modules/calendar/jobs/calendar-job.module'; -import { AutoCompaniesAndContactsCreationJobModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/jobs/auto-companies-and-contacts-creation-job.module'; +import { CalendarEventParticipantManagerModule } from 'src/modules/calendar/calendar-event-participant-manager/calendar-event-participant-manager.module'; import { CalendarModule } from 'src/modules/calendar/calendar.module'; +import { AutoCompaniesAndContactsCreationJobModule } from 'src/modules/contact-creation-manager/jobs/auto-companies-and-contacts-creation-job.module'; import { MessagingModule } from 'src/modules/messaging/messaging.module'; import { TimelineJobModule } from 'src/modules/timeline/jobs/timeline-job.module'; +import { TimelineActivityModule } from 'src/modules/timeline/timeline-activity.module'; @Module({ imports: [ @@ -39,33 +36,20 @@ import { TimelineJobModule } from 'src/modules/timeline/jobs/timeline-job.module UserWorkspaceModule, WorkspaceModule, MessagingModule, - CalendarEventParticipantModule, + CalendarModule, + CalendarEventParticipantManagerModule, TimelineActivityModule, StripeModule, - CalendarModule, - // JobsModules WorkspaceQueryRunnerJobModule, - CalendarMessagingParticipantJobModule, - CalendarCronJobModule, - CalendarJobModule, AutoCompaniesAndContactsCreationJobModule, TimelineJobModule, ], providers: [ - { - provide: CleanInactiveWorkspaceJob.name, - useClass: CleanInactiveWorkspaceJob, - }, - { provide: EmailSenderJob.name, useClass: EmailSenderJob }, - { - provide: DataSeedDemoWorkspaceJob.name, - useClass: DataSeedDemoWorkspaceJob, - }, - { provide: UpdateSubscriptionJob.name, useClass: UpdateSubscriptionJob }, - { - provide: HandleWorkspaceMemberDeletedJob.name, - useClass: HandleWorkspaceMemberDeletedJob, - }, + CleanInactiveWorkspaceJob, + EmailSenderJob, + DataSeedDemoWorkspaceJob, + UpdateSubscriptionJob, + HandleWorkspaceMemberDeletedJob, ], }) export class JobsModule { diff --git a/packages/twenty-server/src/engine/integrations/message-queue/message-queue-core.module.ts b/packages/twenty-server/src/engine/integrations/message-queue/message-queue-core.module.ts new file mode 100644 index 000000000000..c22b0abce39c --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/message-queue-core.module.ts @@ -0,0 +1,121 @@ +import { + DynamicModule, + Global, + Logger, + Module, + Provider, +} from '@nestjs/common'; + +import { MessageQueueDriver } from 'src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface'; + +import { MessageQueueDriverType } from 'src/engine/integrations/message-queue/interfaces'; +import { + MessageQueue, + QUEUE_DRIVER, +} from 'src/engine/integrations/message-queue/message-queue.constants'; +import { PgBossDriver } from 'src/engine/integrations/message-queue/drivers/pg-boss.driver'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { BullMQDriver } from 'src/engine/integrations/message-queue/drivers/bullmq.driver'; +import { SyncDriver } from 'src/engine/integrations/message-queue/drivers/sync.driver'; +import { getQueueToken } from 'src/engine/integrations/message-queue/utils/get-queue-token.util'; +import { + ASYNC_OPTIONS_TYPE, + ConfigurableModuleClass, + OPTIONS_TYPE, +} from 'src/engine/integrations/message-queue/message-queue.module-definition'; + +@Global() +@Module({}) +export class MessageQueueCoreModule extends ConfigurableModuleClass { + private static readonly logger = new Logger(MessageQueueCoreModule.name); + + static register(options: typeof OPTIONS_TYPE): DynamicModule { + const dynamicModule = super.register(options); + + const driverProvider: Provider = { + provide: QUEUE_DRIVER, + useFactory: () => { + return this.createDriver(options); + }, + }; + + const queueProviders = this.createQueueProviders(); + + return { + ...dynamicModule, + providers: [ + ...(dynamicModule.providers ?? []), + driverProvider, + ...queueProviders, + ], + exports: [ + ...(dynamicModule.exports ?? []), + ...Object.values(MessageQueue).map((queueName) => + getQueueToken(queueName), + ), + ], + }; + } + + static registerAsync(options: typeof ASYNC_OPTIONS_TYPE): DynamicModule { + const dynamicModule = super.registerAsync(options); + + const driverProvider: Provider = { + provide: QUEUE_DRIVER, + useFactory: async (...args: any[]) => { + const config = await options.useFactory!(...args); + + return this.createDriver(config); + }, + inject: options.inject || [], + }; + + const queueProviders = MessageQueueCoreModule.createQueueProviders(); + + return { + ...dynamicModule, + providers: [ + ...(dynamicModule.providers ?? []), + driverProvider, + ...queueProviders, + ], + exports: [ + ...(dynamicModule.exports ?? []), + ...Object.values(MessageQueue).map((queueName) => + getQueueToken(queueName), + ), + ], + }; + } + + static async createDriver({ type, options }: typeof OPTIONS_TYPE) { + switch (type) { + case MessageQueueDriverType.PgBoss: { + return new PgBossDriver(options); + } + case MessageQueueDriverType.BullMQ: { + return new BullMQDriver(options); + } + case MessageQueueDriverType.Sync: { + return new SyncDriver(); + } + default: { + this.logger.warn( + `Unsupported message queue driver type: ${type}. Using SyncDriver by default.`, + ); + + return new SyncDriver(); + } + } + } + + static createQueueProviders(): Provider[] { + return Object.values(MessageQueue).map((queueName) => ({ + provide: getQueueToken(queueName), + useFactory: (driver: MessageQueueDriver) => { + return new MessageQueueService(driver, queueName); + }, + inject: [QUEUE_DRIVER], + })); + } +} diff --git a/packages/twenty-server/src/engine/integrations/message-queue/message-queue-metadata.accessor.ts b/packages/twenty-server/src/engine/integrations/message-queue/message-queue-metadata.accessor.ts new file mode 100644 index 000000000000..e509e1fd8718 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/message-queue-metadata.accessor.ts @@ -0,0 +1,54 @@ +/* eslint-disable @typescript-eslint/ban-types */ +import { Injectable, Type } from '@nestjs/common'; +import { Reflector } from '@nestjs/core'; + +import { MessageQueueWorkerOptions } from 'src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface'; + +import { MessageQueueProcessOptions } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { MessageQueueProcessorOptions } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { + PROCESSOR_METADATA, + PROCESS_METADATA, + WORKER_METADATA, +} from 'src/engine/integrations/message-queue/message-queue.constants'; + +@Injectable() +export class MessageQueueMetadataAccessor { + constructor(private readonly reflector: Reflector) {} + + isProcessor(target: Type<any> | Function): boolean { + if (!target) { + return false; + } + + return !!this.reflector.get(PROCESSOR_METADATA, target); + } + + isProcess(target: Type<any> | Function): boolean { + if (!target) { + return false; + } + + return !!this.reflector.get(PROCESS_METADATA, target); + } + + getProcessorMetadata( + target: Type<any> | Function, + ): MessageQueueProcessorOptions | undefined { + return this.reflector.get(PROCESSOR_METADATA, target); + } + + getProcessMetadata( + target: Type<any> | Function, + ): MessageQueueProcessOptions | undefined { + const metadata = this.reflector.get(PROCESS_METADATA, target); + + return metadata; + } + + getWorkerOptionsMetadata( + target: Type<any> | Function, + ): MessageQueueWorkerOptions { + return this.reflector.get(WORKER_METADATA, target) ?? {}; + } +} diff --git a/packages/twenty-server/src/engine/integrations/message-queue/message-queue.constants.ts b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.constants.ts index 7576b8693a01..e78cc50939bf 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/message-queue.constants.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.constants.ts @@ -1,4 +1,7 @@ -export const QUEUE_DRIVER = Symbol('QUEUE_DRIVER'); +export const PROCESSOR_METADATA = Symbol('message-queue:processor_metadata'); +export const PROCESS_METADATA = Symbol('message-queue:process_metadata'); +export const WORKER_METADATA = Symbol('bullmq:worker_metadata'); +export const QUEUE_DRIVER = Symbol('message-queue:queue_driver'); export enum MessageQueue { taskAssignedQueue = 'task-assigned-queue', @@ -12,4 +15,5 @@ export enum MessageQueue { workspaceQueue = 'workspace-queue', recordPositionBackfillQueue = 'record-position-backfill-queue', entityEventsToDbQueue = 'entity-events-to-db-queue', + testQueue = 'test-queue', } diff --git a/packages/twenty-server/src/engine/integrations/message-queue/message-queue.explorer.ts b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.explorer.ts new file mode 100644 index 000000000000..30e73f6bbc7b --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.explorer.ts @@ -0,0 +1,216 @@ +import { Injectable, Logger, OnModuleInit } from '@nestjs/common'; +import { + DiscoveryService, + MetadataScanner, + ModuleRef, + createContextId, +} from '@nestjs/core'; +import { Module } from '@nestjs/core/injector/module'; +import { Injector } from '@nestjs/core/injector/injector'; +import { InstanceWrapper } from '@nestjs/core/injector/instance-wrapper'; + +import { MessageQueueWorkerOptions } from 'src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface'; +import { + MessageQueueJob, + MessageQueueJobData, +} from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; + +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { getQueueToken } from 'src/engine/integrations/message-queue/utils/get-queue-token.util'; +import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service'; +import { shouldFilterException } from 'src/engine/utils/global-exception-handler.util'; + +import { MessageQueueMetadataAccessor } from './message-queue-metadata.accessor'; + +interface ProcessorGroup { + instance: object; + host: Module; + processMethodNames: string[]; + isRequestScoped: boolean; +} + +@Injectable() +export class MessageQueueExplorer implements OnModuleInit { + private readonly logger = new Logger('MessageQueueModule'); + private readonly injector = new Injector(); + + constructor( + private readonly moduleRef: ModuleRef, + private readonly discoveryService: DiscoveryService, + private readonly metadataAccessor: MessageQueueMetadataAccessor, + private readonly metadataScanner: MetadataScanner, + private readonly exceptionHandlerService: ExceptionHandlerService, + ) {} + + onModuleInit() { + this.explore(); + } + + explore() { + const processors = this.discoveryService + .getProviders() + .filter((wrapper) => + this.metadataAccessor.isProcessor( + !wrapper.metatype || wrapper.inject + ? wrapper.instance?.constructor + : wrapper.metatype, + ), + ); + + const groupedProcessors = this.groupProcessorsByQueueName(processors); + + for (const [queueName, processorGroupCollection] of Object.entries( + groupedProcessors, + )) { + const queueToken = getQueueToken(queueName); + const messageQueueService = this.getQueueService(queueToken); + + this.handleProcessorGroupCollection( + processorGroupCollection, + messageQueueService, + ); + } + } + + private groupProcessorsByQueueName(processors: InstanceWrapper[]) { + return processors.reduce( + (acc, wrapper) => { + const { instance, metatype } = wrapper; + const methodNames = this.metadataScanner.getAllMethodNames(instance); + const { queueName } = + this.metadataAccessor.getProcessorMetadata( + instance.constructor || metatype, + ) ?? {}; + + const processMethodNames = methodNames.filter((name) => + this.metadataAccessor.isProcess(instance[name]), + ); + + if (!queueName) { + this.logger.error( + `Processor ${wrapper.name} is missing queue name metadata`, + ); + + return acc; + } + + if (!wrapper.host) { + this.logger.error( + `Processor ${wrapper.name} is missing host metadata`, + ); + + return acc; + } + + if (!acc[queueName]) { + acc[queueName] = []; + } + + acc[queueName].push({ + instance, + host: wrapper.host, + processMethodNames, + isRequestScoped: !wrapper.isDependencyTreeStatic(), + }); + + return acc; + }, + {} as Record<string, ProcessorGroup[]>, + ); + } + + private getQueueService(queueToken: string): MessageQueueService { + try { + return this.moduleRef.get<MessageQueueService>(queueToken, { + strict: false, + }); + } catch (err) { + this.logger.error(`No queue found for token ${queueToken}`); + throw err; + } + } + + private async handleProcessorGroupCollection( + processorGroupCollection: ProcessorGroup[], + queue: MessageQueueService, + options?: MessageQueueWorkerOptions, + ) { + queue.work(async (job) => { + for (const processorGroup of processorGroupCollection) { + await this.handleProcessor(processorGroup, job); + } + }, options); + } + + private async handleProcessor( + { instance, host, processMethodNames, isRequestScoped }: ProcessorGroup, + job: MessageQueueJob<MessageQueueJobData>, + ) { + const filteredProcessMethodNames = processMethodNames.filter( + (processMethodName) => { + const metadata = this.metadataAccessor.getProcessMetadata( + instance[processMethodName], + ); + + return metadata && job.name === metadata.jobName; + }, + ); + + // Return early if no matching methods found + if (filteredProcessMethodNames.length === 0) { + return; + } + + if (isRequestScoped) { + const contextId = createContextId(); + + if (this.moduleRef.registerRequestByContextId) { + this.moduleRef.registerRequestByContextId( + { + // Add workspaceId to the request object + req: { + workspaceId: job.data?.workspaceId, + }, + }, + contextId, + ); + } + + const contextInstance = await this.injector.loadPerContext( + instance, + host, + host.providers, + contextId, + ); + + await this.invokeProcessMethods( + contextInstance, + filteredProcessMethodNames, + job, + ); + } else { + await this.invokeProcessMethods( + instance, + filteredProcessMethodNames, + job, + ); + } + } + + private async invokeProcessMethods( + instance: object, + processMethodNames: string[], + job: MessageQueueJob<MessageQueueJobData>, + ) { + for (const processMethodName of processMethodNames) { + try { + await instance[processMethodName].call(instance, job.data); + } catch (err) { + if (!shouldFilterException(err)) { + this.exceptionHandlerService.captureExceptions([err]); + } + throw err; + } + } + } +} diff --git a/packages/twenty-server/src/engine/integrations/message-queue/message-queue.module-definition.ts b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.module-definition.ts new file mode 100644 index 000000000000..b7a437032f3e --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.module-definition.ts @@ -0,0 +1,20 @@ +import { ConfigurableModuleBuilder } from '@nestjs/common'; + +import { MessageQueueModuleOptions } from 'src/engine/integrations/message-queue/interfaces'; + +export const { + ConfigurableModuleClass, + OPTIONS_TYPE, + ASYNC_OPTIONS_TYPE, + MODULE_OPTIONS_TOKEN, +} = new ConfigurableModuleBuilder<MessageQueueModuleOptions>() + .setExtras( + { + isGlobal: true, + }, + (definition, extras) => ({ + ...definition, + global: extras.isGlobal, + }), + ) + .build(); diff --git a/packages/twenty-server/src/engine/integrations/message-queue/message-queue.module.ts b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.module.ts index 1f7cebba3e3b..ee7fb385b1fb 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/message-queue.module.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/message-queue.module.ts @@ -1,62 +1,36 @@ -import { DynamicModule, Global } from '@nestjs/common'; +import { DynamicModule, Global, Module } from '@nestjs/common'; +import { DiscoveryModule } from '@nestjs/core'; -import { MessageQueueDriver } from 'src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface'; - -import { - MessageQueueDriverType, - MessageQueueModuleAsyncOptions, -} from 'src/engine/integrations/message-queue/interfaces'; +import { MessageQueueCoreModule } from 'src/engine/integrations/message-queue/message-queue-core.module'; +import { MessageQueueMetadataAccessor } from 'src/engine/integrations/message-queue/message-queue-metadata.accessor'; +import { MessageQueueExplorer } from 'src/engine/integrations/message-queue/message-queue.explorer'; import { - MessageQueue, - QUEUE_DRIVER, -} from 'src/engine/integrations/message-queue/message-queue.constants'; -import { PgBossDriver } from 'src/engine/integrations/message-queue/drivers/pg-boss.driver'; -import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { BullMQDriver } from 'src/engine/integrations/message-queue/drivers/bullmq.driver'; -import { SyncDriver } from 'src/engine/integrations/message-queue/drivers/sync.driver'; -import { JobsModule } from 'src/engine/integrations/message-queue/jobs.module'; + ASYNC_OPTIONS_TYPE, + OPTIONS_TYPE, +} from 'src/engine/integrations/message-queue/message-queue.module-definition'; @Global() +@Module({}) export class MessageQueueModule { - static forRoot(options: MessageQueueModuleAsyncOptions): DynamicModule { - const providers = [ - ...Object.values(MessageQueue).map((queue) => ({ - provide: queue, - useFactory: (driver: MessageQueueDriver) => { - return new MessageQueueService(driver, queue); - }, - inject: [QUEUE_DRIVER], - })), - { - provide: QUEUE_DRIVER, - useFactory: async (...args: any[]) => { - const config = await options.useFactory(...args); - - switch (config.type) { - case MessageQueueDriverType.PgBoss: { - const boss = new PgBossDriver(config.options); - - await boss.init(); + static register(options: typeof OPTIONS_TYPE): DynamicModule { + return { + module: MessageQueueModule, + imports: [MessageQueueCoreModule.register(options)], + }; + } - return boss; - } - case MessageQueueDriverType.BullMQ: { - return new BullMQDriver(config.options); - } - default: { - return new SyncDriver(JobsModule.moduleRef); - } - } - }, - inject: options.inject || [], - }, - ]; + static registerExplorer(): DynamicModule { + return { + module: MessageQueueModule, + imports: [DiscoveryModule], + providers: [MessageQueueExplorer, MessageQueueMetadataAccessor], + }; + } + static registerAsync(options: typeof ASYNC_OPTIONS_TYPE): DynamicModule { return { module: MessageQueueModule, - imports: [JobsModule, ...(options.imports || [])], - providers, - exports: Object.values(MessageQueue), + imports: [MessageQueueCoreModule.registerAsync(options)], }; } } diff --git a/packages/twenty-server/src/engine/integrations/message-queue/services/message-queue.service.ts b/packages/twenty-server/src/engine/integrations/message-queue/services/message-queue.service.ts index 87899696f907..6460e112d974 100644 --- a/packages/twenty-server/src/engine/integrations/message-queue/services/message-queue.service.ts +++ b/packages/twenty-server/src/engine/integrations/message-queue/services/message-queue.service.ts @@ -1,11 +1,15 @@ -import { Inject, Injectable, OnModuleDestroy } from '@nestjs/common'; +import { Inject, Injectable } from '@nestjs/common'; import { QueueCronJobOptions, QueueJobOptions, } from 'src/engine/integrations/message-queue/drivers/interfaces/job-options.interface'; import { MessageQueueDriver } from 'src/engine/integrations/message-queue/drivers/interfaces/message-queue-driver.interface'; -import { MessageQueueJobData } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { + MessageQueueJobData, + MessageQueueJob, +} from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { MessageQueueWorkerOptions } from 'src/engine/integrations/message-queue/interfaces/message-queue-worker-options.interface'; import { MessageQueue, @@ -13,7 +17,7 @@ import { } from 'src/engine/integrations/message-queue/message-queue.constants'; @Injectable() -export class MessageQueueService implements OnModuleDestroy { +export class MessageQueueService { constructor( @Inject(QUEUE_DRIVER) protected driver: MessageQueueDriver, protected queueName: MessageQueue, @@ -23,12 +27,6 @@ export class MessageQueueService implements OnModuleDestroy { } } - async onModuleDestroy() { - if (typeof this.driver.stop === 'function') { - await this.driver.stop(); - } - } - add<T extends MessageQueueJobData>( jobName: string, data: T, @@ -50,8 +48,9 @@ export class MessageQueueService implements OnModuleDestroy { } work<T extends MessageQueueJobData>( - handler: ({ data, id }: { data: T; id: string }) => Promise<void> | void, + handler: (job: MessageQueueJob<T>) => Promise<void> | void, + options?: MessageQueueWorkerOptions, ) { - return this.driver.work(this.queueName, handler); + return this.driver.work(this.queueName, handler, options); } } diff --git a/packages/twenty-server/src/engine/integrations/message-queue/utils/get-queue-token.util.ts b/packages/twenty-server/src/engine/integrations/message-queue/utils/get-queue-token.util.ts new file mode 100644 index 000000000000..e73faea901a5 --- /dev/null +++ b/packages/twenty-server/src/engine/integrations/message-queue/utils/get-queue-token.util.ts @@ -0,0 +1,2 @@ +export const getQueueToken = (queueName: string) => + `MESSAGE_QUEUE_${queueName}`; diff --git a/packages/twenty-server/src/engine/metadata-modules/data-source/data-source.exception.ts b/packages/twenty-server/src/engine/metadata-modules/data-source/data-source.exception.ts new file mode 100644 index 000000000000..89672fa26ac0 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/data-source/data-source.exception.ts @@ -0,0 +1,11 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class DataSourceException extends CustomException { + constructor(message: string, code: DataSourceExceptionCode) { + super(message, code); + } +} + +export enum DataSourceExceptionCode { + DATA_SOURCE_NOT_FOUND = 'DATA_SOURCE_NOT_FOUND', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/data-source/data-source.service.ts b/packages/twenty-server/src/engine/metadata-modules/data-source/data-source.service.ts index 226b5a9ca830..6ca6eb83b207 100644 --- a/packages/twenty-server/src/engine/metadata-modules/data-source/data-source.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/data-source/data-source.service.ts @@ -3,6 +3,11 @@ import { InjectRepository } from '@nestjs/typeorm'; import { FindManyOptions, Repository } from 'typeorm'; +import { + DataSourceException, + DataSourceExceptionCode, +} from 'src/engine/metadata-modules/data-source/data-source.exception'; + import { DataSourceEntity } from './data-source.entity'; @Injectable() @@ -46,15 +51,31 @@ export class DataSourceService { }); } - async getLastDataSourceMetadataFromWorkspaceIdOrFail( + async getLastDataSourceMetadataFromWorkspaceId( workspaceId: string, - ): Promise<DataSourceEntity> { - return this.dataSourceMetadataRepository.findOneOrFail({ + ): Promise<DataSourceEntity | null> { + return this.dataSourceMetadataRepository.findOne({ where: { workspaceId }, order: { createdAt: 'DESC' }, }); } + async getLastDataSourceMetadataFromWorkspaceIdOrFail( + workspaceId: string, + ): Promise<DataSourceEntity> { + try { + return this.dataSourceMetadataRepository.findOneOrFail({ + where: { workspaceId }, + order: { createdAt: 'DESC' }, + }); + } catch (error) { + throw new DataSourceException( + `Data source not found for workspace ${workspaceId}: ${error}`, + DataSourceExceptionCode.DATA_SOURCE_NOT_FOUND, + ); + } + } + async delete(workspaceId: string): Promise<void> { await this.dataSourceMetadataRepository.delete({ workspaceId }); } diff --git a/packages/twenty-server/src/engine/metadata-modules/errors/InvalidStringException.ts b/packages/twenty-server/src/engine/metadata-modules/errors/InvalidStringException.ts deleted file mode 100644 index eabfb0140b5c..000000000000 --- a/packages/twenty-server/src/engine/metadata-modules/errors/InvalidStringException.ts +++ /dev/null @@ -1,9 +0,0 @@ -import { BadRequestException } from '@nestjs/common'; - -export class InvalidStringException extends BadRequestException { - constructor(string: string) { - const message = `String "${string}" is not valid`; - - super(message); - } -} diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/composite-types/links.composite-type.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/composite-types/links.composite-type.ts index 4adb14f757b6..2238e2175847 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/composite-types/links.composite-type.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/composite-types/links.composite-type.ts @@ -20,7 +20,7 @@ export const linksCompositeType: CompositeType = { { name: 'secondaryLinks', type: FieldMetadataType.RAW_JSON, - hidden: 'input', + hidden: false, isRequired: false, }, ], diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.entity.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.entity.ts index dd24719ad690..5a80230f7865 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.entity.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.entity.ts @@ -9,6 +9,7 @@ import { CreateDateColumn, UpdateDateColumn, Relation, + OneToMany, } from 'typeorm'; import { FieldMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/field-metadata.interface'; @@ -18,6 +19,7 @@ import { FieldMetadataSettings } from 'src/engine/metadata-modules/field-metadat import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { RelationMetadataEntity } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { IndexFieldMetadataEntity } from 'src/engine/metadata-modules/index-field-metadata/index-field-metadata.entity'; export enum FieldMetadataType { UUID = 'UUID', @@ -29,7 +31,6 @@ export enum FieldMetadataType { BOOLEAN = 'BOOLEAN', NUMBER = 'NUMBER', NUMERIC = 'NUMERIC', - PROBABILITY = 'PROBABILITY', LINK = 'LINK', LINKS = 'LINKS', CURRENCY = 'CURRENCY', @@ -119,6 +120,16 @@ export class FieldMetadataEntity< ) toRelationMetadata: Relation<RelationMetadataEntity>; + @OneToMany( + () => IndexFieldMetadataEntity, + (indexFieldMetadata: IndexFieldMetadataEntity) => + indexFieldMetadata.fieldMetadata, + { + cascade: true, + }, + ) + indexFieldMetadatas: Relation<IndexFieldMetadataEntity>; + @CreateDateColumn({ type: 'timestamptz' }) createdAt: Date; diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.exception.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.exception.ts new file mode 100644 index 000000000000..e9390c099edd --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.exception.ts @@ -0,0 +1,17 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class FieldMetadataException extends CustomException { + code: FieldMetadataExceptionCode; + constructor(message: string, code: FieldMetadataExceptionCode) { + super(message, code); + } +} + +export enum FieldMetadataExceptionCode { + FIELD_METADATA_NOT_FOUND = 'FIELD_METADATA_NOT_FOUND', + INVALID_FIELD_INPUT = 'INVALID_FIELD_INPUT', + FIELD_MUTATION_NOT_ALLOWED = 'FIELD_MUTATION_NOT_ALLOWED', + FIELD_ALREADY_EXISTS = 'FIELD_ALREADY_EXISTS', + OBJECT_METADATA_NOT_FOUND = 'OBJECT_METADATA_NOT_FOUND', + INTERNAL_SERVER_ERROR = 'INTERNAL_SERVER_ERROR', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.module.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.module.ts index bcf9159b9938..321bbc9d9677 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.module.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.module.ts @@ -1,26 +1,27 @@ import { Module } from '@nestjs/common'; +import { SortDirection } from '@ptc-org/nestjs-query-core'; import { NestjsQueryGraphQLModule, PagingStrategies, } from '@ptc-org/nestjs-query-graphql'; import { NestjsQueryTypeOrmModule } from '@ptc-org/nestjs-query-typeorm'; -import { SortDirection } from '@ptc-org/nestjs-query-core'; -import { WorkspaceMigrationRunnerModule } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.module'; -import { WorkspaceMigrationModule } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.module'; -import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; +import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; import { JwtAuthGuard } from 'src/engine/guards/jwt.auth.guard'; import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; -import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; -import { IsFieldMetadataDefaultValue } from 'src/engine/metadata-modules/field-metadata/validators/is-field-metadata-default-value.validator'; -import { FieldMetadataResolver } from 'src/engine/metadata-modules/field-metadata/field-metadata.resolver'; import { FieldMetadataDTO } from 'src/engine/metadata-modules/field-metadata/dtos/field-metadata.dto'; +import { FieldMetadataResolver } from 'src/engine/metadata-modules/field-metadata/field-metadata.resolver'; +import { FieldMetadataGraphqlApiExceptionInterceptor } from 'src/engine/metadata-modules/field-metadata/interceptors/field-metadata-graphql-api-exception.interceptor'; +import { IsFieldMetadataDefaultValue } from 'src/engine/metadata-modules/field-metadata/validators/is-field-metadata-default-value.validator'; import { IsFieldMetadataOptions } from 'src/engine/metadata-modules/field-metadata/validators/is-field-metadata-options.validator'; +import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; +import { WorkspaceMigrationModule } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.module'; +import { WorkspaceMigrationRunnerModule } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.module'; -import { FieldMetadataService } from './field-metadata.service'; import { FieldMetadataEntity } from './field-metadata.entity'; +import { FieldMetadataService } from './field-metadata.service'; import { CreateFieldInput } from './dtos/create-field.input'; import { UpdateFieldInput } from './dtos/update-field.input'; @@ -61,6 +62,7 @@ import { UpdateFieldInput } from './dtos/update-field.input'; }, delete: { disabled: true }, guards: [JwtAuthGuard], + interceptors: [FieldMetadataGraphqlApiExceptionInterceptor], }, ], }), diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.resolver.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.resolver.ts index b4bc92e00fa8..f92b8144b53e 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.resolver.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.resolver.ts @@ -23,6 +23,7 @@ import { RelationDefinitionDTO } from 'src/engine/metadata-modules/field-metadat import { UpdateOneFieldMetadataInput } from 'src/engine/metadata-modules/field-metadata/dtos/update-field.input'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { FieldMetadataService } from 'src/engine/metadata-modules/field-metadata/field-metadata.service'; +import { fieldMetadataGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/field-metadata/utils/field-metadata-graphql-api-exception-handler.util'; @UseGuards(JwtAuthGuard) @Resolver(() => FieldMetadataDTO) @@ -30,25 +31,33 @@ export class FieldMetadataResolver { constructor(private readonly fieldMetadataService: FieldMetadataService) {} @Mutation(() => FieldMetadataDTO) - createOneField( + async createOneField( @Args('input') input: CreateOneFieldMetadataInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.fieldMetadataService.createOne({ - ...input.field, - workspaceId, - }); + try { + return await this.fieldMetadataService.createOne({ + ...input.field, + workspaceId, + }); + } catch (error) { + fieldMetadataGraphqlApiExceptionHandler(error); + } } @Mutation(() => FieldMetadataDTO) - updateOneField( + async updateOneField( @Args('input') input: UpdateOneFieldMetadataInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.fieldMetadataService.updateOne(input.id, { - ...input.update, - workspaceId, - }); + try { + return await this.fieldMetadataService.updateOne(input.id, { + ...input.update, + workspaceId, + }); + } catch (error) { + fieldMetadataGraphqlApiExceptionHandler(error); + } } @Mutation(() => FieldMetadataDTO) @@ -85,27 +94,32 @@ export class FieldMetadataResolver { ); } - return this.fieldMetadataService.deleteOneField(input, workspaceId); + try { + return await this.fieldMetadataService.deleteOneField(input, workspaceId); + } catch (error) { + fieldMetadataGraphqlApiExceptionHandler(error); + } } @ResolveField(() => RelationDefinitionDTO, { nullable: true }) async relationDefinition( @Parent() fieldMetadata: FieldMetadataDTO, @Context() context: { loaders: IDataloaders }, - ): Promise<RelationDefinitionDTO | null> { + ): Promise<RelationDefinitionDTO | null | undefined> { if (fieldMetadata.type !== FieldMetadataType.RELATION) { return null; } - const relationMetadataItem = - await context.loaders.relationMetadataLoader.load(fieldMetadata.id); + try { + const relationMetadataItem = + await context.loaders.relationMetadataLoader.load(fieldMetadata.id); - const relationDefinition = - await this.fieldMetadataService.getRelationDefinitionFromRelationMetadata( + return await this.fieldMetadataService.getRelationDefinitionFromRelationMetadata( fieldMetadata, relationMetadataItem, ); - - return relationDefinition; + } catch (error) { + fieldMetadataGraphqlApiExceptionHandler(error); + } } } diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.service.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.service.ts index 8098b4f26957..fe817577320f 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/field-metadata.service.ts @@ -1,56 +1,60 @@ -import { - BadRequestException, - ConflictException, - Injectable, - NotFoundException, -} from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { InjectDataSource, InjectRepository } from '@nestjs/typeorm'; -import { v4 as uuidV4 } from 'uuid'; -import { DataSource, FindOneOptions, Repository } from 'typeorm'; import { TypeOrmQueryService } from '@ptc-org/nestjs-query-typeorm'; +import { DataSource, FindOneOptions, Repository } from 'typeorm'; +import { v4 as uuidV4 } from 'uuid'; -import { WorkspaceMigrationRunnerService } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.service'; -import { WorkspaceMigrationService } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.service'; -import { ObjectMetadataService } from 'src/engine/metadata-modules/object-metadata/object-metadata.service'; -import { CreateFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/create-field.input'; -import { - WorkspaceMigrationColumnActionType, - WorkspaceMigrationColumnDrop, - WorkspaceMigrationTableAction, - WorkspaceMigrationTableActionType, -} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; import { TypeORMService } from 'src/database/typeorm/typeorm.service'; import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; -import { UpdateFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/update-field.input'; -import { WorkspaceMigrationFactory } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.factory'; -import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; -import { generateMigrationName } from 'src/engine/metadata-modules/workspace-migration/utils/generate-migration-name.util'; -import { generateNullable } from 'src/engine/metadata-modules/field-metadata/utils/generate-nullable'; +import { CreateFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/create-field.input'; +import { DeleteOneFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/delete-field.input'; import { FieldMetadataDTO } from 'src/engine/metadata-modules/field-metadata/dtos/field-metadata.dto'; import { RelationDefinitionDTO, RelationDefinitionType, } from 'src/engine/metadata-modules/field-metadata/dtos/relation-definition.dto'; +import { UpdateFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/update-field.input'; +import { + FieldMetadataException, + FieldMetadataExceptionCode, +} from 'src/engine/metadata-modules/field-metadata/field-metadata.exception'; +import { assertDoesNotNullifyDefaultValueForNonNullableField } from 'src/engine/metadata-modules/field-metadata/utils/assert-does-not-nullify-default-value-for-non-nullable-field.util'; +import { computeColumnName } from 'src/engine/metadata-modules/field-metadata/utils/compute-column-name.util'; +import { generateNullable } from 'src/engine/metadata-modules/field-metadata/utils/generate-nullable'; +import { ObjectMetadataService } from 'src/engine/metadata-modules/object-metadata/object-metadata.service'; +import { assertMutationNotOnRemoteObject } from 'src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util'; import { RelationMetadataEntity, RelationMetadataType, } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; -import { DeleteOneFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/delete-field.input'; -import { computeColumnName } from 'src/engine/metadata-modules/field-metadata/utils/compute-column-name.util'; -import { assertMutationNotOnRemoteObject } from 'src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util'; -import { InvalidStringException } from 'src/engine/metadata-modules/errors/InvalidStringException'; -import { validateMetadataName } from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; +import { exceedsDatabaseIdentifierMaximumLength } from 'src/engine/metadata-modules/utils/validate-database-identifier-length.utils'; +import { + InvalidStringException, + NameTooLongException, + validateMetadataNameOrThrow, +} from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; +import { generateMigrationName } from 'src/engine/metadata-modules/workspace-migration/utils/generate-migration-name.util'; +import { + WorkspaceMigrationColumnActionType, + WorkspaceMigrationColumnDrop, + WorkspaceMigrationTableAction, + WorkspaceMigrationTableActionType, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; +import { WorkspaceMigrationFactory } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.factory'; +import { WorkspaceMigrationService } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.service'; +import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; +import { WorkspaceMigrationRunnerService } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.service'; import { FieldMetadataEntity, FieldMetadataType, } from './field-metadata.entity'; -import { isEnumFieldMetadataType } from './utils/is-enum-field-metadata-type.util'; -import { generateRatingOptions } from './utils/generate-rating-optionts.util'; import { generateDefaultValue } from './utils/generate-default-value'; +import { generateRatingOptions } from './utils/generate-rating-optionts.util'; +import { isEnumFieldMetadataType } from './utils/is-enum-field-metadata-type.util'; @Injectable() export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntity> { @@ -94,7 +98,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit ); if (!objectMetadata) { - throw new NotFoundException('Object does not exist'); + throw new FieldMetadataException( + 'Object metadata does not exist', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); } if (!fieldMetadataInput.isRemoteCreation) { @@ -107,7 +114,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit !fieldMetadataInput.options && fieldMetadataInput.type !== FieldMetadataType.RATING ) { - throw new BadRequestException('Options are required for enum fields'); + throw new FieldMetadataException( + 'Options are required for enum fields', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); } } @@ -116,6 +126,13 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit fieldMetadataInput.options = generateRatingOptions(); } + if (fieldMetadataInput.type === FieldMetadataType.LINK) { + throw new FieldMetadataException( + '"Link" field types are being deprecated, please use Links type instead', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); + } + this.validateFieldMetadataInput<CreateFieldInput>(fieldMetadataInput); const fieldAlreadyExists = await fieldMetadataRepository.findOne({ @@ -127,7 +144,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit }); if (fieldAlreadyExists) { - throw new ConflictException('Field already exists'); + throw new FieldMetadataException( + 'Field already exists', + FieldMetadataExceptionCode.FIELD_ALREADY_EXISTS, + ); } const createdFieldMetadata = await fieldMetadataRepository.save({ @@ -183,7 +203,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit const workspaceQueryRunner = workspaceDataSource?.createQueryRunner(); if (!workspaceQueryRunner) { - throw new Error('Could not create workspace query runner'); + throw new FieldMetadataException( + 'Could not create workspace query runner', + FieldMetadataExceptionCode.INTERNAL_SERVER_ERROR, + ); } await workspaceQueryRunner.connect(); @@ -263,7 +286,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit }); if (!existingFieldMetadata) { - throw new NotFoundException('Field does not exist'); + throw new FieldMetadataException( + 'Field does not exist', + FieldMetadataExceptionCode.FIELD_METADATA_NOT_FOUND, + ); } const objectMetadata = @@ -277,25 +303,37 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit ); if (!objectMetadata) { - throw new NotFoundException('Object does not exist'); + throw new FieldMetadataException( + 'Object metadata does not exist', + FieldMetadataExceptionCode.OBJECT_METADATA_NOT_FOUND, + ); } assertMutationNotOnRemoteObject(objectMetadata); + assertDoesNotNullifyDefaultValueForNonNullableField({ + isNullable: existingFieldMetadata.isNullable, + defaultValueFromUpdate: fieldMetadataInput.defaultValue, + }); + if ( objectMetadata.labelIdentifierFieldMetadataId === existingFieldMetadata.id && fieldMetadataInput.isActive === false ) { - throw new BadRequestException( + throw new FieldMetadataException( 'Cannot deactivate label identifier field', + FieldMetadataExceptionCode.FIELD_MUTATION_NOT_ALLOWED, ); } if (fieldMetadataInput.options) { for (const option of fieldMetadataInput.options) { if (!option.id) { - throw new BadRequestException('Option id is required'); + throw new FieldMetadataException( + 'Option id is required', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); } } } @@ -325,10 +363,18 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit ? updatableFieldInput.defaultValue : null, }); - const updatedFieldMetadata = await fieldMetadataRepository.findOneOrFail({ + + const updatedFieldMetadata = await fieldMetadataRepository.findOne({ where: { id }, }); + if (!updatedFieldMetadata) { + throw new FieldMetadataException( + 'Field does not exist', + FieldMetadataExceptionCode.FIELD_METADATA_NOT_FOUND, + ); + } + if ( fieldMetadataInput.name || updatableFieldInput.options || @@ -392,7 +438,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit }); if (!fieldMetadata) { - throw new NotFoundException('Field does not exist'); + throw new FieldMetadataException( + 'Field does not exist', + FieldMetadataExceptionCode.FIELD_METADATA_NOT_FOUND, + ); } const objectMetadata = @@ -403,7 +452,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit }); if (!objectMetadata) { - throw new NotFoundException('Object does not exist'); + throw new FieldMetadataException( + 'Object metadata does not exist', + FieldMetadataExceptionCode.OBJECT_METADATA_NOT_FOUND, + ); } await fieldMetadataRepository.delete(fieldMetadata.id); @@ -454,7 +506,10 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit }); if (!fieldMetadata) { - throw new NotFoundException('Field does not exist'); + throw new FieldMetadataException( + 'Field does not exist', + FieldMetadataExceptionCode.FIELD_METADATA_NOT_FOUND, + ); } return fieldMetadata; @@ -517,9 +572,12 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit relationMetadata.relationType === RelationMetadataType.MANY_TO_MANY || relationMetadata.relationType === RelationMetadataType.MANY_TO_ONE ) { - throw new Error(` + throw new FieldMetadataException( + ` Relation type ${relationMetadata.relationType} not supported - `); + `, + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); } if (isRelationFromSource) { @@ -558,11 +616,17 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit >(fieldMetadataInput: T): T { if (fieldMetadataInput.name) { try { - validateMetadataName(fieldMetadataInput.name); + validateMetadataNameOrThrow(fieldMetadataInput.name); } catch (error) { if (error instanceof InvalidStringException) { - throw new BadRequestException( + throw new FieldMetadataException( `Characters used in name "${fieldMetadataInput.name}" are not supported`, + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); + } else if (error instanceof NameTooLongException) { + throw new FieldMetadataException( + `Name "${fieldMetadataInput.name}" exceeds 63 characters`, + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, ); } else { throw error; @@ -570,6 +634,17 @@ export class FieldMetadataService extends TypeOrmQueryService<FieldMetadataEntit } } + if (fieldMetadataInput.options) { + for (const option of fieldMetadataInput.options) { + if (exceedsDatabaseIdentifierMaximumLength(option.value)) { + throw new FieldMetadataException( + `Option value "${option.value}" exceeds 63 characters`, + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); + } + } + } + return fieldMetadataInput; } } diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/interceptors/field-metadata-graphql-api-exception.interceptor.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/interceptors/field-metadata-graphql-api-exception.interceptor.ts new file mode 100644 index 000000000000..4c2696e0d32f --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/interceptors/field-metadata-graphql-api-exception.interceptor.ts @@ -0,0 +1,15 @@ +import { CallHandler, ExecutionContext, NestInterceptor } from '@nestjs/common'; + +import { Observable, catchError } from 'rxjs'; + +import { fieldMetadataGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/field-metadata/utils/field-metadata-graphql-api-exception-handler.util'; + +export class FieldMetadataGraphqlApiExceptionInterceptor + implements NestInterceptor +{ + intercept(_: ExecutionContext, next: CallHandler): Observable<any> { + return next + .handle() + .pipe(catchError((err) => fieldMetadataGraphqlApiExceptionHandler(err))); + } +} diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/interfaces/field-metadata-default-value.interface.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/interfaces/field-metadata-default-value.interface.ts index c78af0480a45..f3e24005c7ef 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/interfaces/field-metadata-default-value.interface.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/interfaces/field-metadata-default-value.interface.ts @@ -35,7 +35,6 @@ type FieldMetadataDefaultValueMapping = { [FieldMetadataType.NUMBER]: FieldMetadataDefaultValueNumber; [FieldMetadataType.POSITION]: FieldMetadataDefaultValueNumber; [FieldMetadataType.NUMERIC]: FieldMetadataDefaultValueString; - [FieldMetadataType.PROBABILITY]: FieldMetadataDefaultValueNumber; [FieldMetadataType.LINK]: FieldMetadataDefaultValueLink; [FieldMetadataType.LINKS]: FieldMetadataDefaultValueLinks; [FieldMetadataType.CURRENCY]: FieldMetadataDefaultValueCurrency; diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/assert-does-not-nullify-default-value-for-non-nullable-field.spec.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/assert-does-not-nullify-default-value-for-non-nullable-field.spec.ts new file mode 100644 index 000000000000..04ecc103340d --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/assert-does-not-nullify-default-value-for-non-nullable-field.spec.ts @@ -0,0 +1,38 @@ +import { assertDoesNotNullifyDefaultValueForNonNullableField } from 'src/engine/metadata-modules/field-metadata/utils/assert-does-not-nullify-default-value-for-non-nullable-field.util'; + +describe('assertDoesNotNullifyDefaultValueForNonNullableField', () => { + it('should not throw if default value is set to null and field is nullable', () => { + expect(() => + assertDoesNotNullifyDefaultValueForNonNullableField({ + isNullable: true, + defaultValueFromUpdate: null, + }), + ).not.toThrow(); + }); + + it('should not throw if default value is undefined and field is non nullable', () => { + expect(() => + assertDoesNotNullifyDefaultValueForNonNullableField({ + isNullable: false, + }), + ).not.toThrow(); + }); + + it('should not throw if default value is not set to null and field is non nullable', () => { + expect(() => + assertDoesNotNullifyDefaultValueForNonNullableField({ + isNullable: false, + defaultValueFromUpdate: 'new default value', + }), + ).not.toThrow(); + }); + + it('should throw if default value is set to null and field is non nullable', () => { + expect(() => + assertDoesNotNullifyDefaultValueForNonNullableField({ + isNullable: false, + defaultValueFromUpdate: null, + }), + ).toThrow(); + }); +}); diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/serialize-default-value.spec.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/serialize-default-value.spec.ts index 61e46ec6a8c5..476104f1bba3 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/serialize-default-value.spec.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/serialize-default-value.spec.ts @@ -1,5 +1,4 @@ -import { BadRequestException } from '@nestjs/common'; - +import { FieldMetadataException } from 'src/engine/metadata-modules/field-metadata/field-metadata.exception'; import { serializeDefaultValue } from 'src/engine/metadata-modules/field-metadata/utils/serialize-default-value'; describe('serializeDefaultValue', () => { @@ -15,8 +14,10 @@ describe('serializeDefaultValue', () => { expect(serializeDefaultValue('now')).toBe('now()'); }); - it('should throw BadRequestException for invalid dynamic default value type', () => { - expect(() => serializeDefaultValue('invalid')).toThrow(BadRequestException); + it('should throw FieldMetadataException for invalid dynamic default value type', () => { + expect(() => serializeDefaultValue('invalid')).toThrow( + FieldMetadataException, + ); }); it('should handle string static default value', () => { diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/validate-default-value-based-on-type.spec.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/validate-default-value-based-on-type.spec.ts index 757303d4c1ea..3695a072ec9c 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/validate-default-value-based-on-type.spec.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/__tests__/validate-default-value-based-on-type.spec.ts @@ -78,18 +78,6 @@ describe('validateDefaultValueForType', () => { ).toBe(false); }); - it('should validate number default value for PROBABILITY type', () => { - expect( - validateDefaultValueForType(FieldMetadataType.PROBABILITY, 0.5).isValid, - ).toBe(true); - }); - - it('should return false for invalid number default value for PROBABILITY type', () => { - expect( - validateDefaultValueForType(FieldMetadataType.PROBABILITY, '50%').isValid, - ).toBe(false); - }); - it('should validate boolean default value for BOOLEAN type', () => { expect( validateDefaultValueForType(FieldMetadataType.BOOLEAN, true).isValid, diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/assert-does-not-nullify-default-value-for-non-nullable-field.util.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/assert-does-not-nullify-default-value-for-non-nullable-field.util.ts new file mode 100644 index 000000000000..7fcd625b82de --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/assert-does-not-nullify-default-value-for-non-nullable-field.util.ts @@ -0,0 +1,19 @@ +import { + FieldMetadataException, + FieldMetadataExceptionCode, +} from 'src/engine/metadata-modules/field-metadata/field-metadata.exception'; + +export const assertDoesNotNullifyDefaultValueForNonNullableField = ({ + isNullable, + defaultValueFromUpdate, +}: { + isNullable: boolean; + defaultValueFromUpdate?: any; +}) => { + if (!isNullable && defaultValueFromUpdate === null) { + throw new FieldMetadataException( + 'Default value cannot be nullified for non-nullable field', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); + } +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/compute-column-name.util.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/compute-column-name.util.ts index c1f358ad44b7..116eb94f165f 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/compute-column-name.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/compute-column-name.util.ts @@ -4,6 +4,10 @@ import { CompositeProperty } from 'src/engine/metadata-modules/field-metadata/in import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { isCompositeFieldMetadataType } from 'src/engine/metadata-modules/field-metadata/utils/is-composite-field-metadata-type.util'; import { pascalCase } from 'src/utils/pascal-case'; +import { + FieldMetadataException, + FieldMetadataExceptionCode, +} from 'src/engine/metadata-modules/field-metadata/field-metadata.exception'; type ComputeColumnNameOptions = { isForeignKey?: boolean }; @@ -29,8 +33,9 @@ export function computeColumnName<T extends FieldMetadataType | 'default'>( } if (isCompositeFieldMetadataType(fieldMetadataOrFieldName.type)) { - throw new Error( - `Cannot compute column name for composite field metadata type: ${fieldMetadataOrFieldName.type}`, + throw new FieldMetadataException( + `Cannot compute composite column name for non-composite field metadata type: ${fieldMetadataOrFieldName.type}`, + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, ); } @@ -61,8 +66,9 @@ export function computeCompositeColumnName< } if (!isCompositeFieldMetadataType(fieldMetadataOrFieldName.type)) { - throw new Error( + throw new FieldMetadataException( `Cannot compute composite column name for non-composite field metadata type: ${fieldMetadataOrFieldName.type}`, + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, ); } diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/field-metadata-graphql-api-exception-handler.util.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/field-metadata-graphql-api-exception-handler.util.ts new file mode 100644 index 000000000000..5ff6bf16169a --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/field-metadata-graphql-api-exception-handler.util.ts @@ -0,0 +1,32 @@ +import { + ConflictError, + ForbiddenError, + InternalServerError, + NotFoundError, + UserInputError, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; +import { + FieldMetadataException, + FieldMetadataExceptionCode, +} from 'src/engine/metadata-modules/field-metadata/field-metadata.exception'; + +export const fieldMetadataGraphqlApiExceptionHandler = (error: Error) => { + if (error instanceof FieldMetadataException) { + switch (error.code) { + case FieldMetadataExceptionCode.FIELD_METADATA_NOT_FOUND: + throw new NotFoundError(error.message); + case FieldMetadataExceptionCode.INVALID_FIELD_INPUT: + throw new UserInputError(error.message); + case FieldMetadataExceptionCode.FIELD_MUTATION_NOT_ALLOWED: + throw new ForbiddenError(error.message); + case FieldMetadataExceptionCode.FIELD_ALREADY_EXISTS: + throw new ConflictError(error.message); + case FieldMetadataExceptionCode.OBJECT_METADATA_NOT_FOUND: + case FieldMetadataExceptionCode.INTERNAL_SERVER_ERROR: + default: + throw new InternalServerError(error.message); + } + } + + throw error; +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/serialize-default-value.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/serialize-default-value.ts index 2f93da1b8cc9..7445e0fed2ec 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/serialize-default-value.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/serialize-default-value.ts @@ -1,7 +1,9 @@ -import { BadRequestException } from '@nestjs/common'; - import { FieldMetadataDefaultSerializableValue } from 'src/engine/metadata-modules/field-metadata/interfaces/field-metadata-default-value.interface'; +import { + FieldMetadataException, + FieldMetadataExceptionCode, +} from 'src/engine/metadata-modules/field-metadata/field-metadata.exception'; import { isFunctionDefaultValue } from 'src/engine/metadata-modules/field-metadata/utils/is-function-default-value.util'; import { serializeFunctionDefaultValue } from 'src/engine/metadata-modules/field-metadata/utils/serialize-function-default-value.util'; @@ -18,7 +20,10 @@ export const serializeDefaultValue = ( serializeFunctionDefaultValue(defaultValue); if (!serializedTypeDefaultValue) { - throw new BadRequestException('Invalid default value'); + throw new FieldMetadataException( + 'Invalid default value', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); } return serializedTypeDefaultValue; @@ -51,5 +56,8 @@ export const serializeDefaultValue = ( return `'${JSON.stringify(defaultValue)}'`; } - throw new BadRequestException(`Invalid default value "${defaultValue}"`); + throw new FieldMetadataException( + `Invalid default value "${defaultValue}"`, + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); }; diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-default-value-for-type.util.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-default-value-for-type.util.ts index bb7ec90f8658..776c49db65e3 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-default-value-for-type.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-default-value-for-type.util.ts @@ -41,7 +41,6 @@ export const defaultValueValidatorsMap = { [FieldMetadataType.BOOLEAN]: [FieldMetadataDefaultValueBoolean], [FieldMetadataType.NUMBER]: [FieldMetadataDefaultValueNumber], [FieldMetadataType.NUMERIC]: [FieldMetadataDefaultValueString], - [FieldMetadataType.PROBABILITY]: [FieldMetadataDefaultValueNumber], [FieldMetadataType.LINK]: [FieldMetadataDefaultValueLink], [FieldMetadataType.CURRENCY]: [FieldMetadataDefaultValueCurrency], [FieldMetadataType.FULL_NAME]: [FieldMetadataDefaultValueFullName], diff --git a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-options-for-type.util.ts b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-options-for-type.util.ts index 7e119d594638..aecb4eefe9fd 100644 --- a/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-options-for-type.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/field-metadata/utils/validate-options-for-type.util.ts @@ -8,6 +8,10 @@ import { FieldMetadataComplexOption, FieldMetadataDefaultOption, } from 'src/engine/metadata-modules/field-metadata/dtos/options.input'; +import { + FieldMetadataException, + FieldMetadataExceptionCode, +} from 'src/engine/metadata-modules/field-metadata/field-metadata.exception'; import { isEnumFieldMetadataType } from './is-enum-field-metadata-type.util'; @@ -24,7 +28,10 @@ export const validateOptionsForType = ( if (options === null) return true; if (!Array.isArray(options)) { - throw new Error('Options must be an array'); + throw new FieldMetadataException( + 'Options must be an array', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); } if (!isEnumFieldMetadataType(type)) { @@ -39,7 +46,10 @@ export const validateOptionsForType = ( // Check if all options are unique if (new Set(values).size !== options.length) { - throw new Error('Options must be unique'); + throw new FieldMetadataException( + 'Options must be unique', + FieldMetadataExceptionCode.INVALID_FIELD_INPUT, + ); } const validators = optionsValidatorsMap[type]; diff --git a/packages/twenty-server/src/engine/metadata-modules/index-field-metadata/index-field-metadata.entity.ts b/packages/twenty-server/src/engine/metadata-modules/index-field-metadata/index-field-metadata.entity.ts new file mode 100644 index 000000000000..036bee830ca0 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/index-field-metadata/index-field-metadata.entity.ts @@ -0,0 +1,54 @@ +import { + Column, + CreateDateColumn, + Entity, + JoinColumn, + ManyToOne, + PrimaryGeneratedColumn, + Relation, + UpdateDateColumn, +} from 'typeorm'; + +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; + +@Entity('indexFieldMetadata') +export class IndexFieldMetadataEntity { + @PrimaryGeneratedColumn('uuid') + id: string; + + @Column({ nullable: false }) + indexMetadataId: string; + + @ManyToOne( + () => IndexMetadataEntity, + (indexMetadata) => indexMetadata.indexFieldMetadatas, + { + onDelete: 'CASCADE', + }, + ) + @JoinColumn() + indexMetadata: Relation<IndexMetadataEntity>; + + @Column({ nullable: false }) + fieldMetadataId: string; + + @ManyToOne( + () => FieldMetadataEntity, + (fieldMetadata) => fieldMetadata.indexFieldMetadatas, + { + onDelete: 'CASCADE', + }, + ) + @JoinColumn() + fieldMetadata: Relation<FieldMetadataEntity>; + + @Column({ nullable: false }) + order: number; + + @CreateDateColumn({ type: 'timestamptz' }) + createdAt: Date; + + @UpdateDateColumn({ type: 'timestamptz' }) + updatedAt: Date; +} diff --git a/packages/twenty-server/src/engine/metadata-modules/index-field-metadata/index-field-metadata.module.ts b/packages/twenty-server/src/engine/metadata-modules/index-field-metadata/index-field-metadata.module.ts new file mode 100644 index 000000000000..937851df1ed9 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/index-field-metadata/index-field-metadata.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { IndexFieldMetadataEntity } from 'src/engine/metadata-modules/index-field-metadata/index-field-metadata.entity'; + +@Module({ + imports: [TypeOrmModule.forFeature([IndexFieldMetadataEntity], 'metadata')], + providers: [], + exports: [], +}) +export class IndexFieldMetadataModule {} diff --git a/packages/twenty-server/src/engine/metadata-modules/index-metadata/index-metadata.entity.ts b/packages/twenty-server/src/engine/metadata-modules/index-metadata/index-metadata.entity.ts new file mode 100644 index 000000000000..74b15a10d089 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/index-metadata/index-metadata.entity.ts @@ -0,0 +1,51 @@ +import { + Column, + CreateDateColumn, + Entity, + JoinColumn, + ManyToOne, + OneToMany, + PrimaryGeneratedColumn, + Relation, + UpdateDateColumn, +} from 'typeorm'; + +import { IndexFieldMetadataEntity } from 'src/engine/metadata-modules/index-field-metadata/index-field-metadata.entity'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; + +@Entity('indexMetadata') +export class IndexMetadataEntity { + @PrimaryGeneratedColumn('uuid') + id: string; + + @Column({ nullable: false }) + name: string; + + @Column({ nullable: true }) + workspaceId: string; + + @Column({ nullable: false, type: 'uuid' }) + objectMetadataId: string; + + @ManyToOne(() => ObjectMetadataEntity, (object) => object.indexes, { + onDelete: 'CASCADE', + }) + @JoinColumn() + objectMetadata: Relation<ObjectMetadataEntity>; + + @OneToMany( + () => IndexFieldMetadataEntity, + (indexFieldMetadata: IndexFieldMetadataEntity) => + indexFieldMetadata.indexMetadata, + { + cascade: true, + }, + ) + indexFieldMetadatas: Relation<IndexFieldMetadataEntity[]>; + + @CreateDateColumn({ type: 'timestamptz' }) + createdAt: Date; + + @UpdateDateColumn({ type: 'timestamptz' }) + updatedAt: Date; +} diff --git a/packages/twenty-server/src/engine/metadata-modules/index-metadata/index-metadata.module.ts b/packages/twenty-server/src/engine/metadata-modules/index-metadata/index-metadata.module.ts new file mode 100644 index 000000000000..01cb4a3e6a92 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/index-metadata/index-metadata.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; + +@Module({ + imports: [TypeOrmModule.forFeature([IndexMetadataEntity], 'metadata')], + providers: [], + exports: [], +}) +export class IndexMetadataModule {} diff --git a/packages/twenty-server/src/engine/metadata-modules/index-metadata/utils/generate-deterministic-index-name.ts b/packages/twenty-server/src/engine/metadata-modules/index-metadata/utils/generate-deterministic-index-name.ts new file mode 100644 index 000000000000..ebf84d17d2da --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/index-metadata/utils/generate-deterministic-index-name.ts @@ -0,0 +1,11 @@ +import { createHash } from 'crypto'; + +export const generateDeterministicIndexName = (columns: string[]): string => { + const hash = createHash('sha256'); + + columns.forEach((column) => { + hash.update(column); + }); + + return hash.digest('hex').slice(0, 27); +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/interceptors/object-metadata-graphql-api-exception.interceptor.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/interceptors/object-metadata-graphql-api-exception.interceptor.ts new file mode 100644 index 000000000000..117fb32f3790 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/interceptors/object-metadata-graphql-api-exception.interceptor.ts @@ -0,0 +1,15 @@ +import { CallHandler, ExecutionContext, NestInterceptor } from '@nestjs/common'; + +import { Observable, catchError } from 'rxjs'; + +import { objectMetadataGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/object-metadata/utils/object-metadata-graphql-api-exception-handler.util'; + +export class ObjectMetadataGraphqlApiExceptionInterceptor + implements NestInterceptor +{ + intercept(_: ExecutionContext, next: CallHandler): Observable<any> { + return next + .handle() + .pipe(catchError((err) => objectMetadataGraphqlApiExceptionHandler(err))); + } +} diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.constants.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.constants.ts new file mode 100644 index 000000000000..856d01f5db38 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.constants.ts @@ -0,0 +1 @@ +export const DEFAULT_LABEL_IDENTIFIER_FIELD_NAME = 'name'; diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.entity.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.entity.ts index d4cf45c14165..cc64c231b704 100644 --- a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.entity.ts +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.entity.ts @@ -15,6 +15,7 @@ import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metad import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { RelationMetadataEntity } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; @Entity('objectMetadata') @Unique('IndexOnNameSingularAndWorkspaceIdUnique', [ @@ -82,6 +83,11 @@ export class ObjectMetadataEntity implements ObjectMetadataInterface { }) fields: Relation<FieldMetadataEntity[]>; + @OneToMany(() => FieldMetadataEntity, (field) => field.object, { + cascade: true, + }) + indexes: Relation<IndexMetadataEntity[]>; + @OneToMany( () => RelationMetadataEntity, (relation: RelationMetadataEntity) => relation.fromObjectMetadata, diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.exception.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.exception.ts new file mode 100644 index 000000000000..2d489b1320ef --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.exception.ts @@ -0,0 +1,15 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class ObjectMetadataException extends CustomException { + code: ObjectMetadataExceptionCode; + constructor(message: string, code: ObjectMetadataExceptionCode) { + super(message, code); + } +} + +export enum ObjectMetadataExceptionCode { + OBJECT_METADATA_NOT_FOUND = 'OBJECT_METADATA_NOT_FOUND', + INVALID_OBJECT_INPUT = 'INVALID_OBJECT_INPUT', + OBJECT_MUTATION_NOT_ALLOWED = 'OBJECT_MUTATION_NOT_ALLOWED', + OBJECT_ALREADY_EXISTS = 'OBJECT_ALREADY_EXISTS', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.module.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.module.ts index 9974403628bd..efd5aeccb44b 100644 --- a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.module.ts +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.module.ts @@ -1,33 +1,34 @@ import { Module } from '@nestjs/common'; import { TypeOrmModule } from '@nestjs/typeorm'; +import { SortDirection } from '@ptc-org/nestjs-query-core'; import { NestjsQueryGraphQLModule, PagingStrategies, } from '@ptc-org/nestjs-query-graphql'; import { NestjsQueryTypeOrmModule } from '@ptc-org/nestjs-query-typeorm'; -import { SortDirection } from '@ptc-org/nestjs-query-core'; -import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; -import { WorkspaceMigrationRunnerModule } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.module'; -import { WorkspaceMigrationModule } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.module'; -import { JwtAuthGuard } from 'src/engine/guards/jwt.auth.guard'; import { TypeORMModule } from 'src/database/typeorm/typeorm.module'; -import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { RelationMetadataEntity } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; -import { ObjectMetadataResolver } from 'src/engine/metadata-modules/object-metadata/object-metadata.resolver'; -import { FeatureFlagModule } from 'src/engine/core-modules/feature-flag/feature-flag.module'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; +import { FeatureFlagModule } from 'src/engine/core-modules/feature-flag/feature-flag.module'; +import { JwtAuthGuard } from 'src/engine/guards/jwt.auth.guard'; +import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { BeforeUpdateOneObject } from 'src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook'; +import { ObjectMetadataGraphqlApiExceptionInterceptor } from 'src/engine/metadata-modules/object-metadata/interceptors/object-metadata-graphql-api-exception.interceptor'; +import { ObjectMetadataResolver } from 'src/engine/metadata-modules/object-metadata/object-metadata.resolver'; +import { RelationMetadataEntity } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { RemoteTableRelationsModule } from 'src/engine/metadata-modules/remote-server/remote-table/remote-table-relations/remote-table-relations.module'; +import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; +import { WorkspaceMigrationModule } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.module'; +import { WorkspaceMigrationRunnerModule } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.module'; -import { ObjectMetadataService } from './object-metadata.service'; import { ObjectMetadataEntity } from './object-metadata.entity'; +import { ObjectMetadataService } from './object-metadata.service'; import { CreateObjectInput } from './dtos/create-object.input'; -import { UpdateObjectPayload } from './dtos/update-object.input'; import { ObjectMetadataDTO } from './dtos/object-metadata.dto'; +import { UpdateObjectPayload } from './dtos/update-object.input'; @Module({ imports: [ @@ -64,6 +65,7 @@ import { ObjectMetadataDTO } from './dtos/object-metadata.dto'; update: { disabled: true }, delete: { disabled: true }, guards: [JwtAuthGuard], + interceptors: [ObjectMetadataGraphqlApiExceptionInterceptor], }, ], }), diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.resolver.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.resolver.ts index ccb1b2a60643..9ae44b1fe996 100644 --- a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.resolver.ts +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.resolver.ts @@ -12,6 +12,7 @@ import { UpdateOneObjectInput, } from 'src/engine/metadata-modules/object-metadata/dtos/update-object.input'; import { BeforeUpdateOneObject } from 'src/engine/metadata-modules/object-metadata/hooks/before-update-one-object.hook'; +import { objectMetadataGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/object-metadata/utils/object-metadata-graphql-api-exception-handler.util'; @UseGuards(JwtAuthGuard) @Resolver(() => ObjectMetadataDTO) @@ -22,11 +23,18 @@ export class ObjectMetadataResolver { ) {} @Mutation(() => ObjectMetadataDTO) - deleteOneObject( + async deleteOneObject( @Args('input') input: DeleteOneObjectInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.objectMetadataService.deleteOneObject(input, workspaceId); + try { + return await this.objectMetadataService.deleteOneObject( + input, + workspaceId, + ); + } catch (error) { + objectMetadataGraphqlApiExceptionHandler(error); + } } @Mutation(() => ObjectMetadataDTO) @@ -34,8 +42,15 @@ export class ObjectMetadataResolver { @Args('input') input: UpdateOneObjectInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - await this.beforeUpdateOneObject.run(input, workspaceId); + try { + await this.beforeUpdateOneObject.run(input, workspaceId); - return this.objectMetadataService.updateOneObject(input, workspaceId); + return await this.objectMetadataService.updateOneObject( + input, + workspaceId, + ); + } catch (error) { + objectMetadataGraphqlApiExceptionHandler(error); + } } } diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.service.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.service.ts index 878a12192a13..c1e1d8111009 100644 --- a/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/object-metadata.service.ts @@ -1,9 +1,4 @@ -import { - BadRequestException, - ConflictException, - Injectable, - NotFoundException, -} from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import console from 'console'; @@ -56,6 +51,10 @@ import { mapUdtNameToFieldType } from 'src/engine/metadata-modules/remote-server import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; import { UpdateOneObjectInput } from 'src/engine/metadata-modules/object-metadata/dtos/update-object.input'; import { RemoteTableRelationsService } from 'src/engine/metadata-modules/remote-server/remote-table/remote-table-relations/remote-table-relations.service'; +import { + ObjectMetadataException, + ObjectMetadataExceptionCode, +} from 'src/engine/metadata-modules/object-metadata/object-metadata.exception'; import { ObjectMetadataEntity } from './object-metadata.entity'; @@ -121,7 +120,10 @@ export class ObjectMetadataService extends TypeOrmQueryService<ObjectMetadataEnt }); if (!objectMetadata) { - throw new NotFoundException('Object does not exist'); + throw new ObjectMetadataException( + 'Object does not exist', + ObjectMetadataExceptionCode.OBJECT_METADATA_NOT_FOUND, + ); } // DELETE RELATIONS @@ -159,8 +161,9 @@ export class ObjectMetadataService extends TypeOrmQueryService<ObjectMetadataEnt objectMetadataInput.nameSingular.toLowerCase() === objectMetadataInput.namePlural.toLowerCase() ) { - throw new BadRequestException( + throw new ObjectMetadataException( 'The singular and plural name cannot be the same for an object', + ObjectMetadataExceptionCode.INVALID_OBJECT_INPUT, ); } @@ -186,7 +189,10 @@ export class ObjectMetadataService extends TypeOrmQueryService<ObjectMetadataEnt }); if (objectAlreadyExists) { - throw new ConflictException('Object already exists'); + throw new ObjectMetadataException( + 'Object already exists', + ObjectMetadataExceptionCode.OBJECT_ALREADY_EXISTS, + ); } const isCustom = !objectMetadataInput.isRemote; @@ -372,18 +378,25 @@ export class ObjectMetadataService extends TypeOrmQueryService<ObjectMetadataEnt workspaceId: string, options: FindOneOptions<ObjectMetadataEntity>, ): Promise<ObjectMetadataEntity> { - return this.objectMetadataRepository.findOneOrFail({ - relations: [ - 'fields', - 'fields.fromRelationMetadata', - 'fields.toRelationMetadata', - ], - ...options, - where: { - ...options.where, - workspaceId, - }, - }); + try { + return this.objectMetadataRepository.findOneOrFail({ + relations: [ + 'fields', + 'fields.fromRelationMetadata', + 'fields.toRelationMetadata', + ], + ...options, + where: { + ...options.where, + workspaceId, + }, + }); + } catch (error) { + throw new ObjectMetadataException( + 'Object does not exist', + ObjectMetadataExceptionCode.OBJECT_METADATA_NOT_FOUND, + ); + } } public async findManyWithinWorkspace( diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util.ts index 1bfcf6973ceb..0db389d95fef 100644 --- a/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/assert-mutation-not-on-remote-object.util.ts @@ -1,11 +1,17 @@ -import { BadRequestException } from '@nestjs/common'; - import { ObjectMetadataInterface } from 'src/engine/metadata-modules/field-metadata/interfaces/object-metadata.interface'; +import { + ObjectMetadataException, + ObjectMetadataExceptionCode, +} from 'src/engine/metadata-modules/object-metadata/object-metadata.exception'; + export const assertMutationNotOnRemoteObject = ( objectMetadataItem: ObjectMetadataInterface, ) => { if (objectMetadataItem.isRemote) { - throw new BadRequestException('Remote objects are read-only'); + throw new ObjectMetadataException( + 'Remote objects are read-only', + ObjectMetadataExceptionCode.OBJECT_MUTATION_NOT_ALLOWED, + ); } }; diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/object-metadata-graphql-api-exception-handler.util.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/object-metadata-graphql-api-exception-handler.util.ts new file mode 100644 index 000000000000..ef00ed60f9fa --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/object-metadata-graphql-api-exception-handler.util.ts @@ -0,0 +1,30 @@ +import { + ConflictError, + ForbiddenError, + InternalServerError, + NotFoundError, + UserInputError, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; +import { + ObjectMetadataException, + ObjectMetadataExceptionCode, +} from 'src/engine/metadata-modules/object-metadata/object-metadata.exception'; + +export const objectMetadataGraphqlApiExceptionHandler = (error: Error) => { + if (error instanceof ObjectMetadataException) { + switch (error.code) { + case ObjectMetadataExceptionCode.OBJECT_METADATA_NOT_FOUND: + throw new NotFoundError(error.message); + case ObjectMetadataExceptionCode.INVALID_OBJECT_INPUT: + throw new UserInputError(error.message); + case ObjectMetadataExceptionCode.OBJECT_MUTATION_NOT_ALLOWED: + throw new ForbiddenError(error.message); + case ObjectMetadataExceptionCode.OBJECT_ALREADY_EXISTS: + throw new ConflictError(error.message); + default: + throw new InternalServerError(error.message); + } + } + + throw error; +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/validate-object-metadata-input.util.ts b/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/validate-object-metadata-input.util.ts index 42d1c1b00545..a8a07ccc3353 100644 --- a/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/validate-object-metadata-input.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/object-metadata/utils/validate-object-metadata-input.util.ts @@ -1,9 +1,14 @@ -import { BadRequestException, ForbiddenException } from '@nestjs/common'; - -import { InvalidStringException } from 'src/engine/metadata-modules/errors/InvalidStringException'; import { CreateObjectInput } from 'src/engine/metadata-modules/object-metadata/dtos/create-object.input'; import { UpdateObjectPayload } from 'src/engine/metadata-modules/object-metadata/dtos/update-object.input'; -import { validateMetadataName } from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; +import { + ObjectMetadataException, + ObjectMetadataExceptionCode, +} from 'src/engine/metadata-modules/object-metadata/object-metadata.exception'; +import { exceedsDatabaseIdentifierMaximumLength } from 'src/engine/metadata-modules/utils/validate-database-identifier-length.utils'; +import { + validateMetadataNameOrThrow, + InvalidStringException, +} from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; import { camelCase } from 'src/utils/camel-case'; const coreObjectNames = [ @@ -50,12 +55,18 @@ export const validateObjectMetadataInputOrThrow = < validateNameIsNotReservedKeywordOrThrow(objectMetadataInput.nameSingular); validateNameIsNotReservedKeywordOrThrow(objectMetadataInput.namePlural); + + validateNameIsNotTooLongThrow(objectMetadataInput.nameSingular); + validateNameIsNotTooLongThrow(objectMetadataInput.namePlural); }; const validateNameIsNotReservedKeywordOrThrow = (name?: string) => { if (name) { if (reservedKeywords.includes(name)) { - throw new ForbiddenException(`The name "${name}" is not available`); + throw new ObjectMetadataException( + `The name "${name}" is not available`, + ObjectMetadataExceptionCode.INVALID_OBJECT_INPUT, + ); } } }; @@ -63,7 +74,21 @@ const validateNameIsNotReservedKeywordOrThrow = (name?: string) => { const validateNameCamelCasedOrThrow = (name?: string) => { if (name) { if (name !== camelCase(name)) { - throw new ForbiddenException(`Name should be in camelCase: ${name}`); + throw new ObjectMetadataException( + `Name should be in camelCase: ${name}`, + ObjectMetadataExceptionCode.INVALID_OBJECT_INPUT, + ); + } + } +}; + +const validateNameIsNotTooLongThrow = (name?: string) => { + if (name) { + if (exceedsDatabaseIdentifierMaximumLength(name)) { + throw new ObjectMetadataException( + `Name exceeds 63 characters: ${name}`, + ObjectMetadataExceptionCode.INVALID_OBJECT_INPUT, + ); } } }; @@ -71,12 +96,13 @@ const validateNameCamelCasedOrThrow = (name?: string) => { const validateNameCharactersOrThrow = (name?: string) => { try { if (name) { - validateMetadataName(name); + validateMetadataNameOrThrow(name); } } catch (error) { if (error instanceof InvalidStringException) { - throw new BadRequestException( + throw new ObjectMetadataException( `Characters used in name "${name}" are not supported`, + ObjectMetadataExceptionCode.INVALID_OBJECT_INPUT, ); } else { throw error; diff --git a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/interceptors/relation-metadata-graphql-api-exception.interceptor.ts b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/interceptors/relation-metadata-graphql-api-exception.interceptor.ts new file mode 100644 index 000000000000..04139a44a4e1 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/interceptors/relation-metadata-graphql-api-exception.interceptor.ts @@ -0,0 +1,17 @@ +import { CallHandler, ExecutionContext, NestInterceptor } from '@nestjs/common'; + +import { Observable, catchError } from 'rxjs'; + +import { relationMetadataGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/relation-metadata/utils/relation-metadata-graphql-api-exception-handler.util'; + +export class RelationMetadataGraphqlApiExceptionInterceptor + implements NestInterceptor +{ + intercept(_: ExecutionContext, next: CallHandler): Observable<any> { + return next + .handle() + .pipe( + catchError((err) => relationMetadataGraphqlApiExceptionHandler(err)), + ); + } +} diff --git a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.exception.ts b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.exception.ts new file mode 100644 index 000000000000..4ce48e8b8d65 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.exception.ts @@ -0,0 +1,15 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class RelationMetadataException extends CustomException { + code: RelationMetadataExceptionCode; + constructor(message: string, code: RelationMetadataExceptionCode) { + super(message, code); + } +} + +export enum RelationMetadataExceptionCode { + RELATION_METADATA_NOT_FOUND = 'RELATION_METADATA_NOT_FOUND', + INVALID_RELATION_INPUT = 'INVALID_RELATION_INPUT', + RELATION_ALREADY_EXISTS = 'RELATION_ALREADY_EXISTS', + FOREIGN_KEY_NOT_FOUND = 'FOREIGN_KEY_NOT_FOUND', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.module.ts b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.module.ts index a5fc287a5ed4..f706eb898d57 100644 --- a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.module.ts +++ b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.module.ts @@ -7,16 +7,17 @@ import { import { NestjsQueryTypeOrmModule } from '@ptc-org/nestjs-query-typeorm'; import { JwtAuthGuard } from 'src/engine/guards/jwt.auth.guard'; +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { FieldMetadataModule } from 'src/engine/metadata-modules/field-metadata/field-metadata.module'; import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; +import { RelationMetadataGraphqlApiExceptionInterceptor } from 'src/engine/metadata-modules/relation-metadata/interceptors/relation-metadata-graphql-api-exception.interceptor'; +import { RelationMetadataResolver } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.resolver'; +import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; import { WorkspaceMigrationModule } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.module'; import { WorkspaceMigrationRunnerModule } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.module'; -import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; -import { RelationMetadataResolver } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.resolver'; -import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { RelationMetadataService } from './relation-metadata.service'; import { RelationMetadataEntity } from './relation-metadata.entity'; +import { RelationMetadataService } from './relation-metadata.service'; import { CreateRelationInput } from './dtos/create-relation.input'; import { RelationMetadataDTO } from './dtos/relation-metadata.dto'; @@ -47,6 +48,7 @@ import { RelationMetadataDTO } from './dtos/relation-metadata.dto'; update: { disabled: true }, delete: { disabled: true }, guards: [JwtAuthGuard], + interceptors: [RelationMetadataGraphqlApiExceptionInterceptor], }, ], }), diff --git a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.resolver.ts b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.resolver.ts index c7eb414987b5..332f783e454d 100644 --- a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.resolver.ts +++ b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.resolver.ts @@ -7,6 +7,7 @@ import { RelationMetadataService } from 'src/engine/metadata-modules/relation-me import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { RelationMetadataDTO } from 'src/engine/metadata-modules/relation-metadata/dtos/relation-metadata.dto'; import { DeleteOneRelationInput } from 'src/engine/metadata-modules/relation-metadata/dtos/delete-relation.input'; +import { relationMetadataGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/relation-metadata/utils/relation-metadata-graphql-api-exception-handler.util'; @UseGuards(JwtAuthGuard) @Resolver() @@ -16,13 +17,17 @@ export class RelationMetadataResolver { ) {} @Mutation(() => RelationMetadataDTO) - deleteOneRelation( + async deleteOneRelation( @Args('input') input: DeleteOneRelationInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.relationMetadataService.deleteOneRelation( - input.id, - workspaceId, - ); + try { + return await this.relationMetadataService.deleteOneRelation( + input.id, + workspaceId, + ); + } catch (error) { + relationMetadataGraphqlApiExceptionHandler(error); + } } } diff --git a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.service.ts b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.service.ts index 8915b60ab4bf..b287f9e6fa90 100644 --- a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/relation-metadata.service.ts @@ -1,9 +1,4 @@ -import { - BadRequestException, - ConflictException, - Injectable, - NotFoundException, -} from '@nestjs/common'; +import { Injectable, NotFoundException } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { TypeOrmQueryService } from '@ptc-org/nestjs-query-typeorm'; @@ -28,9 +23,15 @@ import { import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; import { generateMigrationName } from 'src/engine/metadata-modules/workspace-migration/utils/generate-migration-name.util'; -import { InvalidStringException } from 'src/engine/metadata-modules/errors/InvalidStringException'; -import { validateMetadataName } from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; +import { + validateMetadataNameOrThrow, + InvalidStringException, +} from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; +import { + RelationMetadataException, + RelationMetadataExceptionCode, +} from 'src/engine/metadata-modules/relation-metadata/relation-metadata.exception'; import { RelationMetadataEntity, @@ -62,12 +63,13 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat ); try { - validateMetadataName(relationMetadataInput.fromName); - validateMetadataName(relationMetadataInput.toName); + validateMetadataNameOrThrow(relationMetadataInput.fromName); + validateMetadataNameOrThrow(relationMetadataInput.toName); } catch (error) { if (error instanceof InvalidStringException) { - throw new BadRequestException( + throw new RelationMetadataException( `Characters used in name "${relationMetadataInput.fromName}" or "${relationMetadataInput.toName}" are not supported`, + RelationMetadataExceptionCode.INVALID_RELATION_INPUT, ); } else { throw error; @@ -132,8 +134,9 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat if ( relationMetadataInput.relationType === RelationMetadataType.MANY_TO_MANY ) { - throw new BadRequestException( + throw new RelationMetadataException( 'Many to many relations are not supported yet', + RelationMetadataExceptionCode.INVALID_RELATION_INPUT, ); } @@ -142,8 +145,9 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat undefined || objectMetadataMap[relationMetadataInput.toObjectMetadataId] === undefined ) { - throw new NotFoundException( + throw new RelationMetadataException( 'Can\t find an existing object matching with fromObjectMetadataId or toObjectMetadataId', + RelationMetadataExceptionCode.RELATION_METADATA_NOT_FOUND, ); } @@ -177,12 +181,13 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat ); if (fieldAlreadyExists) { - throw new ConflictException( + throw new RelationMetadataException( `Field on ${ objectMetadataMap[ relationMetadataInput[`${relationDirection}ObjectMetadataId`] ].nameSingular } already exists`, + RelationMetadataExceptionCode.RELATION_ALREADY_EXISTS, ); } } @@ -335,7 +340,10 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat }); if (!relationMetadata) { - throw new NotFoundException('Relation does not exist'); + throw new RelationMetadataException( + 'Relation does not exist', + RelationMetadataExceptionCode.RELATION_METADATA_NOT_FOUND, + ); } const foreignKeyFieldMetadataName = `${camelCase( @@ -351,8 +359,9 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat }); if (!foreignKeyFieldMetadata) { - throw new NotFoundException( + throw new RelationMetadataException( `Foreign key fieldMetadata not found (${foreignKeyFieldMetadataName}) for relation ${relationMetadata.id}`, + RelationMetadataExceptionCode.FOREIGN_KEY_NOT_FOUND, ); } @@ -420,6 +429,7 @@ export class RelationMetadataService extends TypeOrmQueryService<RelationMetadat return ( foundRelationMetadataItem ?? + // TODO: return a relation metadata not found exception new NotFoundException( `RelationMetadata with fieldMetadataId ${fieldMetadataId} not found`, ) diff --git a/packages/twenty-server/src/engine/metadata-modules/relation-metadata/utils/relation-metadata-graphql-api-exception-handler.util.ts b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/utils/relation-metadata-graphql-api-exception-handler.util.ts new file mode 100644 index 000000000000..4367e4035127 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/relation-metadata/utils/relation-metadata-graphql-api-exception-handler.util.ts @@ -0,0 +1,28 @@ +import { + ConflictError, + InternalServerError, + NotFoundError, + UserInputError, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; +import { + RelationMetadataException, + RelationMetadataExceptionCode, +} from 'src/engine/metadata-modules/relation-metadata/relation-metadata.exception'; + +export const relationMetadataGraphqlApiExceptionHandler = (error: Error) => { + if (error instanceof RelationMetadataException) { + switch (error.code) { + case RelationMetadataExceptionCode.RELATION_METADATA_NOT_FOUND: + throw new NotFoundError(error.message); + case RelationMetadataExceptionCode.INVALID_RELATION_INPUT: + throw new UserInputError(error.message); + case RelationMetadataExceptionCode.RELATION_ALREADY_EXISTS: + throw new ConflictError(error.message); + case RelationMetadataExceptionCode.FOREIGN_KEY_NOT_FOUND: + default: + throw new InternalServerError(error.message); + } + } + + throw error; +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.exception.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.exception.ts new file mode 100644 index 000000000000..6aa035234bf9 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.exception.ts @@ -0,0 +1,16 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class RemoteServerException extends CustomException { + code: RemoteServerExceptionCode; + constructor(message: string, code: RemoteServerExceptionCode) { + super(message, code); + } +} + +export enum RemoteServerExceptionCode { + REMOTE_SERVER_NOT_FOUND = 'REMOTE_SERVER_NOT_FOUND', + REMOTE_SERVER_ALREADY_EXISTS = 'REMOTE_SERVER_ALREADY_EXISTS', + REMOTE_SERVER_MUTATION_NOT_ALLOWED = 'REMOTE_SERVER_MUTATION_NOT_ALLOWED', + REMOTE_SERVER_CONNECTION_ERROR = 'REMOTE_SERVER_CONNECTION_ERROR', + INVALID_REMOTE_SERVER_INPUT = 'INVALID_REMOTE_SERVER_INPUT', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.resolver.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.resolver.ts index 04d44a190d8c..262326669ea0 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.resolver.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.resolver.ts @@ -11,6 +11,7 @@ import { RemoteServerDTO } from 'src/engine/metadata-modules/remote-server/dtos/ import { UpdateRemoteServerInput } from 'src/engine/metadata-modules/remote-server/dtos/update-remote-server.input'; import { RemoteServerType } from 'src/engine/metadata-modules/remote-server/remote-server.entity'; import { RemoteServerService } from 'src/engine/metadata-modules/remote-server/remote-server.service'; +import { remoteServerGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/remote-server/utils/remote-server-graphql-api-exception-handler.util'; @UseGuards(JwtAuthGuard) @Resolver() @@ -24,7 +25,14 @@ export class RemoteServerResolver { @Args('input') input: CreateRemoteServerInput<RemoteServerType>, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteServerService.createOneRemoteServer(input, workspaceId); + try { + return await this.remoteServerService.createOneRemoteServer( + input, + workspaceId, + ); + } catch (error) { + remoteServerGraphqlApiExceptionHandler(error); + } } @Mutation(() => RemoteServerDTO) @@ -32,7 +40,14 @@ export class RemoteServerResolver { @Args('input') input: UpdateRemoteServerInput<RemoteServerType>, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteServerService.updateOneRemoteServer(input, workspaceId); + try { + return await this.remoteServerService.updateOneRemoteServer( + input, + workspaceId, + ); + } catch (error) { + remoteServerGraphqlApiExceptionHandler(error); + } } @Mutation(() => RemoteServerDTO) @@ -40,7 +55,14 @@ export class RemoteServerResolver { @Args('input') { id }: RemoteServerIdInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteServerService.deleteOneRemoteServer(id, workspaceId); + try { + return await this.remoteServerService.deleteOneRemoteServer( + id, + workspaceId, + ); + } catch (error) { + remoteServerGraphqlApiExceptionHandler(error); + } } @Query(() => RemoteServerDTO) @@ -48,7 +70,14 @@ export class RemoteServerResolver { @Args('input') { id }: RemoteServerIdInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteServerService.findOneByIdWithinWorkspace(id, workspaceId); + try { + return await this.remoteServerService.findOneByIdWithinWorkspace( + id, + workspaceId, + ); + } catch (error) { + remoteServerGraphqlApiExceptionHandler(error); + } } @Query(() => [RemoteServerDTO]) @@ -57,9 +86,13 @@ export class RemoteServerResolver { { foreignDataWrapperType }: RemoteServerTypeInput<RemoteServerType>, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteServerService.findManyByTypeWithinWorkspace( - foreignDataWrapperType, - workspaceId, - ); + try { + return await this.remoteServerService.findManyByTypeWithinWorkspace( + foreignDataWrapperType, + workspaceId, + ); + } catch (error) { + remoteServerGraphqlApiExceptionHandler(error); + } } } diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.service.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.service.ts index b7ad3a2e7fdc..d1c241827882 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-server.service.ts @@ -1,8 +1,4 @@ -import { - ForbiddenException, - Injectable, - NotFoundException, -} from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { InjectDataSource, InjectRepository } from '@nestjs/typeorm'; import isEmpty from 'lodash.isempty'; @@ -27,6 +23,10 @@ import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/work import { buildUpdateRemoteServerRawQuery } from 'src/engine/metadata-modules/remote-server/utils/build-update-remote-server-raw-query.utils'; import { validateRemoteServerType } from 'src/engine/metadata-modules/remote-server/utils/validate-remote-server-type.util'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { + RemoteServerException, + RemoteServerExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-server.exception'; @Injectable() export class RemoteServerService<T extends RemoteServerType> { @@ -122,7 +122,10 @@ export class RemoteServerService<T extends RemoteServerType> { ); if (!remoteServer) { - throw new NotFoundException('Remote server does not exist'); + throw new RemoteServerException( + 'Remote server does not exist', + RemoteServerExceptionCode.REMOTE_SERVER_NOT_FOUND, + ); } const currentRemoteTablesForServer = @@ -132,8 +135,9 @@ export class RemoteServerService<T extends RemoteServerType> { }); if (currentRemoteTablesForServer.length > 0) { - throw new ForbiddenException( + throw new RemoteServerException( 'Cannot update remote server with synchronized tables', + RemoteServerExceptionCode.REMOTE_SERVER_MUTATION_NOT_ALLOWED, ); } @@ -207,7 +211,10 @@ export class RemoteServerService<T extends RemoteServerType> { }); if (!remoteServer) { - throw new NotFoundException('Remote server does not exist'); + throw new RemoteServerException( + 'Remote server does not exist', + RemoteServerExceptionCode.REMOTE_SERVER_NOT_FOUND, + ); } await this.remoteTableService.unsyncAll(workspaceId, remoteServer); diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.exception.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.exception.ts new file mode 100644 index 000000000000..609da7a6fc65 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.exception.ts @@ -0,0 +1,13 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class DistantTableException extends CustomException { + code: DistantTableExceptionCode; + constructor(message: string, code: DistantTableExceptionCode) { + super(message, code); + } +} + +export enum DistantTableExceptionCode { + INTERNAL_SERVER_ERROR = 'INTERNAL_SERVER_ERROR', + TIMEOUT_ERROR = 'TIMEOUT_ERROR', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.service.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.service.ts index d0e5fdadeb3d..337ab25be7b4 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.service.ts @@ -1,8 +1,4 @@ -import { - BadRequestException, - Injectable, - RequestTimeoutException, -} from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { EntityManager, Repository } from 'typeorm'; @@ -17,6 +13,10 @@ import { DistantTables } from 'src/engine/metadata-modules/remote-server/remote- import { STRIPE_DISTANT_TABLES } from 'src/engine/metadata-modules/remote-server/remote-table/distant-table/utils/stripe-distant-tables.util'; import { PostgresTableSchemaColumn } from 'src/engine/metadata-modules/remote-server/types/postgres-table-schema-column'; import { isQueryTimeoutError } from 'src/engine/utils/query-timeout.util'; +import { + DistantTableException, + DistantTableExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-table/distant-table/distant-table.exception'; @Injectable() export class DistantTableService { @@ -64,7 +64,10 @@ export class DistantTableService { tableName?: string, ): Promise<DistantTables> { if (!remoteServer.schema) { - throw new BadRequestException('Remote server schema is not defined'); + throw new DistantTableException( + 'Remote server schema is not defined', + DistantTableExceptionCode.INTERNAL_SERVER_ERROR, + ); } const tmpSchemaId = v4(); @@ -116,8 +119,9 @@ export class DistantTableService { return distantTables; } catch (error) { if (isQueryTimeoutError(error)) { - throw new RequestTimeoutException( + throw new DistantTableException( `Could not find distant tables: ${error.message}`, + DistantTableExceptionCode.TIMEOUT_ERROR, ); } @@ -132,8 +136,9 @@ export class DistantTableService { case RemoteServerType.STRIPE_FDW: return STRIPE_DISTANT_TABLES; default: - throw new BadRequestException( + throw new DistantTableException( `Type ${remoteServer.foreignDataWrapperType} does not have a static schema.`, + DistantTableExceptionCode.INTERNAL_SERVER_ERROR, ); } } diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.exception.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.exception.ts new file mode 100644 index 000000000000..3f3305823784 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.exception.ts @@ -0,0 +1,13 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class ForeignTableException extends CustomException { + code: ForeignTableExceptionCode; + constructor(message: string, code: ForeignTableExceptionCode) { + super(message, code); + } +} + +export enum ForeignTableExceptionCode { + FOREIGN_TABLE_MUTATION_NOT_ALLOWED = 'FOREIGN_TABLE_MUTATION_NOT_ALLOWED', + INVALID_FOREIGN_TABLE_INPUT = 'INVALID_FOREIGN_TABLE_INPUT', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.service.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.service.ts index c5951b69b173..4f1f2d1cdc64 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.service.ts @@ -1,10 +1,14 @@ -import { BadRequestException, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { RemoteServerEntity, RemoteServerType, } from 'src/engine/metadata-modules/remote-server/remote-server.entity'; import { RemoteTableStatus } from 'src/engine/metadata-modules/remote-server/remote-table/dtos/remote-table.dto'; +import { + ForeignTableException, + ForeignTableExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-table/foreign-table/foreign-table.exception'; import { getForeignTableColumnName } from 'src/engine/metadata-modules/remote-server/remote-table/foreign-table/utils/get-foreign-table-column-name.util'; import { PostgresTableSchemaColumn } from 'src/engine/metadata-modules/remote-server/types/postgres-table-schema-column'; import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; @@ -90,8 +94,9 @@ export class ForeignTableService { } catch (exception) { this.workspaceMigrationService.deleteById(workspaceMigration.id); - throw new BadRequestException( + throw new ForeignTableException( 'Could not create foreign table. The table may already exists or a column type may not be supported.', + ForeignTableExceptionCode.INVALID_FOREIGN_TABLE_INPUT, ); } } @@ -130,7 +135,10 @@ export class ForeignTableService { } catch (exception) { this.workspaceMigrationService.deleteById(workspaceMigration.id); - throw new BadRequestException('Could not alter foreign table.'); + throw new ForeignTableException( + 'Could not alter foreign table.', + ForeignTableExceptionCode.FOREIGN_TABLE_MUTATION_NOT_ALLOWED, + ); } } @@ -167,7 +175,10 @@ export class ForeignTableService { case RemoteServerType.STRIPE_FDW: return { object: distantTableName }; default: - throw new BadRequestException('Foreign data wrapper not supported'); + throw new ForeignTableException( + 'Foreign data wrapper not supported', + ForeignTableExceptionCode.INVALID_FOREIGN_TABLE_INPUT, + ); } } } diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.exception.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.exception.ts new file mode 100644 index 000000000000..32e99227c5a3 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.exception.ts @@ -0,0 +1,17 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class RemoteTableException extends CustomException { + code: RemoteTableExceptionCode; + constructor(message: string, code: RemoteTableExceptionCode) { + super(message, code); + } +} + +export enum RemoteTableExceptionCode { + REMOTE_TABLE_NOT_FOUND = 'REMOTE_TABLE_NOT_FOUND', + INVALID_REMOTE_TABLE_INPUT = 'INVALID_REMOTE_TABLE_INPUT', + REMOTE_TABLE_ALREADY_EXISTS = 'REMOTE_TABLE_ALREADY_EXISTS', + NO_FOREIGN_TABLES_FOUND = 'NO_FOREIGN_TABLES_FOUND', + NO_OBJECT_METADATA_FOUND = 'NO_OBJECT_METADATA_FOUND', + NO_FIELD_METADATA_FOUND = 'NO_FIELD_METADATA_FOUND', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.resolver.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.resolver.ts index 37abfc748688..7698f25d094f 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.resolver.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.resolver.ts @@ -8,6 +8,7 @@ import { FindManyRemoteTablesInput } from 'src/engine/metadata-modules/remote-se import { RemoteTableInput } from 'src/engine/metadata-modules/remote-server/remote-table/dtos/remote-table-input'; import { RemoteTableDTO } from 'src/engine/metadata-modules/remote-server/remote-table/dtos/remote-table.dto'; import { RemoteTableService } from 'src/engine/metadata-modules/remote-server/remote-table/remote-table.service'; +import { remoteTableGraphqlApiExceptionHandler } from 'src/engine/metadata-modules/remote-server/remote-table/utils/remote-table-graphql-api-exception-handler.util'; @UseGuards(JwtAuthGuard) @Resolver() @@ -19,11 +20,15 @@ export class RemoteTableResolver { @Args('input') input: FindManyRemoteTablesInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteTableService.findDistantTablesWithStatus( - input.id, - workspaceId, - input.shouldFetchPendingSchemaUpdates, - ); + try { + return await this.remoteTableService.findDistantTablesWithStatus( + input.id, + workspaceId, + input.shouldFetchPendingSchemaUpdates, + ); + } catch (error) { + remoteTableGraphqlApiExceptionHandler(error); + } } @Mutation(() => RemoteTableDTO) @@ -31,7 +36,11 @@ export class RemoteTableResolver { @Args('input') input: RemoteTableInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteTableService.syncRemoteTable(input, workspaceId); + try { + return await this.remoteTableService.syncRemoteTable(input, workspaceId); + } catch (error) { + remoteTableGraphqlApiExceptionHandler(error); + } } @Mutation(() => RemoteTableDTO) @@ -39,7 +48,14 @@ export class RemoteTableResolver { @Args('input') input: RemoteTableInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteTableService.unsyncRemoteTable(input, workspaceId); + try { + return await this.remoteTableService.unsyncRemoteTable( + input, + workspaceId, + ); + } catch (error) { + remoteTableGraphqlApiExceptionHandler(error); + } } @Mutation(() => RemoteTableDTO) @@ -47,9 +63,13 @@ export class RemoteTableResolver { @Args('input') input: RemoteTableInput, @AuthWorkspace() { id: workspaceId }: Workspace, ) { - return this.remoteTableService.syncRemoteTableSchemaChanges( - input, - workspaceId, - ); + try { + return await this.remoteTableService.syncRemoteTableSchemaChanges( + input, + workspaceId, + ); + } catch (error) { + remoteTableGraphqlApiExceptionHandler(error); + } } } diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.service.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.service.ts index 732108af8330..9102839eddac 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.service.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/remote-table.service.ts @@ -1,4 +1,4 @@ -import { BadRequestException, Logger, NotFoundException } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; @@ -40,6 +40,10 @@ import { } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; import { CreateFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/create-field.input'; import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { + RemoteTableException, + RemoteTableExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-table/remote-table.exception'; export class RemoteTableService { private readonly logger = new Logger(RemoteTableService.name); @@ -74,7 +78,10 @@ export class RemoteTableService { }); if (!remoteServer) { - throw new NotFoundException('Remote server does not exist'); + throw new RemoteTableException( + 'Remote server does not exist', + RemoteTableExceptionCode.INVALID_REMOTE_TABLE_INPUT, + ); } const currentRemoteTables = await this.findRemoteTablesByServerId({ @@ -148,7 +155,10 @@ export class RemoteTableService { }); if (!remoteServer) { - throw new NotFoundException('Remote server does not exist'); + throw new RemoteTableException( + 'Remote server does not exist', + RemoteTableExceptionCode.INVALID_REMOTE_TABLE_INPUT, + ); } const currentRemoteTableWithSameDistantName = @@ -161,7 +171,10 @@ export class RemoteTableService { }); if (currentRemoteTableWithSameDistantName) { - throw new BadRequestException('Remote table already exists'); + throw new RemoteTableException( + 'Remote server does not exist', + RemoteTableExceptionCode.REMOTE_TABLE_ALREADY_EXISTS, + ); } const dataSourceMetatada = @@ -200,7 +213,10 @@ export class RemoteTableService { ); if (!distantTableColumns) { - throw new BadRequestException('Table not found'); + throw new RemoteTableException( + 'Remote server does not exist', + RemoteTableExceptionCode.REMOTE_TABLE_NOT_FOUND, + ); } // We only support remote tables with an id column for now. @@ -209,7 +225,10 @@ export class RemoteTableService { ); if (!distantTableIdColumn) { - throw new BadRequestException('Remote table must have an id column'); + throw new RemoteTableException( + 'Remote server does not exist', + RemoteTableExceptionCode.INVALID_REMOTE_TABLE_INPUT, + ); } await this.foreignTableService.createForeignTable( @@ -250,7 +269,10 @@ export class RemoteTableService { }); if (!remoteServer) { - throw new NotFoundException('Remote server does not exist'); + throw new RemoteTableException( + 'Remote server does not exist', + RemoteTableExceptionCode.INVALID_REMOTE_TABLE_INPUT, + ); } const remoteTable = await this.remoteTableRepository.findOne({ @@ -262,7 +284,10 @@ export class RemoteTableService { }); if (!remoteTable) { - throw new NotFoundException('Remote table does not exist'); + throw new RemoteTableException( + 'Remote table does not exist', + RemoteTableExceptionCode.REMOTE_TABLE_NOT_FOUND, + ); } await this.unsyncOne(workspaceId, remoteTable, remoteServer); @@ -302,7 +327,10 @@ export class RemoteTableService { }); if (!remoteServer) { - throw new NotFoundException('Remote server does not exist'); + throw new RemoteTableException( + 'Remote server does not exist', + RemoteTableExceptionCode.INVALID_REMOTE_TABLE_INPUT, + ); } const remoteTable = await this.remoteTableRepository.findOne({ @@ -314,7 +342,10 @@ export class RemoteTableService { }); if (!remoteTable) { - throw new NotFoundException('Remote table does not exist'); + throw new RemoteTableException( + 'Remote table does not exist', + RemoteTableExceptionCode.REMOTE_TABLE_NOT_FOUND, + ); } const distantTableColumns = @@ -379,7 +410,10 @@ export class RemoteTableService { ); if (!currentForeignTableNames.includes(remoteTable.localTableName)) { - throw new NotFoundException('Foreign table does not exist'); + throw new RemoteTableException( + 'Foreign table does not exist', + RemoteTableExceptionCode.NO_FOREIGN_TABLES_FOUND, + ); } const objectMetadata = @@ -507,8 +541,9 @@ export class RemoteTableService { }); if (!objectMetadata) { - throw new NotFoundException( + throw new RemoteTableException( `Cannot find associated object for table ${foreignTableName}`, + RemoteTableExceptionCode.NO_OBJECT_METADATA_FOUND, ); } for (const columnUpdate of columnsUpdates) { @@ -547,8 +582,9 @@ export class RemoteTableService { }); if (!fieldMetadataToDelete) { - throw new NotFoundException( + throw new RemoteTableException( `Cannot find associated field metadata for column ${columnName}`, + RemoteTableExceptionCode.NO_FIELD_METADATA_FOUND, ); } diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/utils/get-remote-table-local-name.util.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/utils/get-remote-table-local-name.util.ts index 92008e169d00..1df79156fafc 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/utils/get-remote-table-local-name.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/utils/get-remote-table-local-name.util.ts @@ -1,9 +1,11 @@ -import { BadRequestException } from '@nestjs/common/exceptions'; - import { singular } from 'pluralize'; import { DataSource } from 'typeorm'; import { camelCase } from 'src/utils/camel-case'; +import { + RemoteTableException, + RemoteTableExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-table/remote-table.exception'; const MAX_SUFFIX = 10; @@ -55,5 +57,8 @@ export const getRemoteTableLocalName = async ( } } - throw new BadRequestException('Table name is already taken.'); + throw new RemoteTableException( + 'Table name is already taken', + RemoteTableExceptionCode.INVALID_REMOTE_TABLE_INPUT, + ); }; diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/utils/remote-table-graphql-api-exception-handler.util.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/utils/remote-table-graphql-api-exception-handler.util.ts new file mode 100644 index 000000000000..13e2c976a623 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/remote-table/utils/remote-table-graphql-api-exception-handler.util.ts @@ -0,0 +1,30 @@ +import { + ConflictError, + InternalServerError, + NotFoundError, + UserInputError, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; +import { + RemoteTableException, + RemoteTableExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-table/remote-table.exception'; + +export const remoteTableGraphqlApiExceptionHandler = (error: Error) => { + if (error instanceof RemoteTableException) { + switch (error.code) { + case RemoteTableExceptionCode.REMOTE_TABLE_NOT_FOUND: + case RemoteTableExceptionCode.NO_OBJECT_METADATA_FOUND: + case RemoteTableExceptionCode.NO_FOREIGN_TABLES_FOUND: + case RemoteTableExceptionCode.NO_FIELD_METADATA_FOUND: + throw new NotFoundError(error.message); + case RemoteTableExceptionCode.INVALID_REMOTE_TABLE_INPUT: + throw new UserInputError(error.message); + case RemoteTableExceptionCode.REMOTE_TABLE_ALREADY_EXISTS: + throw new ConflictError(error.message); + default: + throw new InternalServerError(error.message); + } + } + + throw error; +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/build-update-remote-server-raw-query.utils.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/build-update-remote-server-raw-query.utils.ts index fff4778db0fa..3e23e9b55cb5 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/build-update-remote-server-raw-query.utils.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/build-update-remote-server-raw-query.utils.ts @@ -1,10 +1,12 @@ -import { BadRequestException } from '@nestjs/common'; - import { ForeignDataWrapperOptions, RemoteServerEntity, RemoteServerType, } from 'src/engine/metadata-modules/remote-server/remote-server.entity'; +import { + RemoteServerException, + RemoteServerExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-server.exception'; import { UserMappingOptions } from 'src/engine/metadata-modules/remote-server/types/user-mapping-options'; export type DeepPartial<T> = { @@ -49,7 +51,10 @@ export const buildUpdateRemoteServerRawQuery = ( } if (options.length < 1) { - throw new BadRequestException('No fields to update'); + throw new RemoteServerException( + 'No fields to update', + RemoteServerExceptionCode.INVALID_REMOTE_SERVER_INPUT, + ); } const rawQuery = `UPDATE metadata."remoteServer" SET ${options.join( diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/remote-server-graphql-api-exception-handler.util.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/remote-server-graphql-api-exception-handler.util.ts new file mode 100644 index 000000000000..e289d02af499 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/remote-server-graphql-api-exception-handler.util.ts @@ -0,0 +1,30 @@ +import { + ConflictError, + ForbiddenError, + InternalServerError, + NotFoundError, + UserInputError, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; +import { + RemoteServerException, + RemoteServerExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-server.exception'; + +export const remoteServerGraphqlApiExceptionHandler = (error: any) => { + if (error instanceof RemoteServerException) { + switch (error.code) { + case RemoteServerExceptionCode.REMOTE_SERVER_NOT_FOUND: + throw new NotFoundError(error.message); + case RemoteServerExceptionCode.INVALID_REMOTE_SERVER_INPUT: + throw new UserInputError(error.message); + case RemoteServerExceptionCode.REMOTE_SERVER_MUTATION_NOT_ALLOWED: + throw new ForbiddenError(error.message); + case RemoteServerExceptionCode.REMOTE_SERVER_ALREADY_EXISTS: + throw new ConflictError(error.message); + default: + throw new InternalServerError(error.message); + } + } + + throw error; +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-input.utils.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-input.utils.ts index 99de7fa95d52..38cfecdcf4ac 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-input.utils.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-input.utils.ts @@ -1,7 +1,10 @@ -import { BadRequestException } from '@nestjs/common'; - import { isDefined } from 'class-validator'; +import { + RemoteServerException, + RemoteServerExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-server.exception'; + const INPUT_REGEX = /^([A-Za-z0-9\-_.@]+)$/; export const validateObjectAgainstInjections = (input: object) => { @@ -21,6 +24,9 @@ export const validateObjectAgainstInjections = (input: object) => { export const validateStringAgainstInjections = (input: string) => { if (!INPUT_REGEX.test(input)) { - throw new BadRequestException('Invalid remote server input'); + throw new RemoteServerException( + 'Invalid remote server input', + RemoteServerExceptionCode.INVALID_REMOTE_SERVER_INPUT, + ); } }; diff --git a/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-type.util.ts b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-type.util.ts index 7a158e6a7a36..333353970c8d 100644 --- a/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-type.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/remote-server/utils/validate-remote-server-type.util.ts @@ -1,5 +1,3 @@ -import { BadRequestException } from '@nestjs/common'; - import { Repository } from 'typeorm'; import { @@ -7,6 +5,10 @@ import { FeatureFlagKeys, } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; import { RemoteServerType } from 'src/engine/metadata-modules/remote-server/remote-server.entity'; +import { + RemoteServerException, + RemoteServerExceptionCode, +} from 'src/engine/metadata-modules/remote-server/remote-server.exception'; export const validateRemoteServerType = async ( remoteServerType: RemoteServerType, @@ -24,7 +26,10 @@ export const validateRemoteServerType = async ( const featureFlagEnabled = featureFlag && featureFlag.value; if (!featureFlagEnabled) { - throw new BadRequestException(`Type ${remoteServerType} is not supported.`); + throw new RemoteServerException( + `Type ${remoteServerType} is not supported.`, + RemoteServerExceptionCode.INVALID_REMOTE_SERVER_INPUT, + ); } }; @@ -35,8 +40,9 @@ const getFeatureFlagKey = (remoteServerType: RemoteServerType) => { case RemoteServerType.STRIPE_FDW: return FeatureFlagKeys.IsStripeIntegrationEnabled; default: - throw new BadRequestException( + throw new RemoteServerException( `Type ${remoteServerType} is not supported.`, + RemoteServerExceptionCode.INVALID_REMOTE_SERVER_INPUT, ); } }; diff --git a/packages/twenty-server/src/engine/metadata-modules/utils/__tests__/validate-metadata-name.spec.ts b/packages/twenty-server/src/engine/metadata-modules/utils/__tests__/validate-metadata-name.spec.ts index d2e03eac6fef..445620dd4236 100644 --- a/packages/twenty-server/src/engine/metadata-modules/utils/__tests__/validate-metadata-name.spec.ts +++ b/packages/twenty-server/src/engine/metadata-modules/utils/__tests__/validate-metadata-name.spec.ts @@ -1,32 +1,57 @@ -import { InvalidStringException } from 'src/engine/metadata-modules/errors/InvalidStringException'; -import { validateMetadataName } from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; +import { + validateMetadataNameOrThrow, + InvalidStringException, + NameTooLongException, +} from 'src/engine/metadata-modules/utils/validate-metadata-name.utils'; -describe('validateMetadataName', () => { +describe('validateMetadataNameOrThrow', () => { it('does not throw if string is valid', () => { const input = 'testName'; - expect(validateMetadataName(input)).not.toThrow; + expect(validateMetadataNameOrThrow(input)).not.toThrow; }); it('throws error if string has spaces', () => { const input = 'name with spaces'; - expect(() => validateMetadataName(input)).toThrow(InvalidStringException); + expect(() => validateMetadataNameOrThrow(input)).toThrow( + InvalidStringException, + ); }); it('throws error if string starts with capital letter', () => { const input = 'StringStartingWithCapitalLetter'; - expect(() => validateMetadataName(input)).toThrow(InvalidStringException); + expect(() => validateMetadataNameOrThrow(input)).toThrow( + InvalidStringException, + ); }); it('throws error if string has non latin characters', () => { const input = 'בְרִבְרִ'; - expect(() => validateMetadataName(input)).toThrow(InvalidStringException); + expect(() => validateMetadataNameOrThrow(input)).toThrow( + InvalidStringException, + ); }); it('throws error if starts with digits', () => { const input = '123string'; - expect(() => validateMetadataName(input)).toThrow(InvalidStringException); + expect(() => validateMetadataNameOrThrow(input)).toThrow( + InvalidStringException, + ); + }); + it('does not throw if string is less than 63 characters', () => { + const inputWith63Characters = + 'thisIsAstringWithSixtyThreeCharacters11111111111111111111111111'; + + expect(validateMetadataNameOrThrow(inputWith63Characters)).not.toThrow; + }); + it('throws error if string is above 63 characters', () => { + const inputWith64Characters = + 'thisIsAstringWithSixtyFourCharacters1111111111111111111111111111'; + + expect(() => validateMetadataNameOrThrow(inputWith64Characters)).toThrow( + NameTooLongException, + ); }); }); diff --git a/packages/twenty-server/src/engine/metadata-modules/utils/metadata.constants.ts b/packages/twenty-server/src/engine/metadata-modules/utils/metadata.constants.ts new file mode 100644 index 000000000000..ce2dabd729b6 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/utils/metadata.constants.ts @@ -0,0 +1 @@ +export const IDENTIFIER_MAX_CHAR_LENGTH = 63; diff --git a/packages/twenty-server/src/engine/metadata-modules/utils/validate-database-identifier-length.utils.ts b/packages/twenty-server/src/engine/metadata-modules/utils/validate-database-identifier-length.utils.ts new file mode 100644 index 000000000000..dff01b4f33b3 --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/utils/validate-database-identifier-length.utils.ts @@ -0,0 +1,5 @@ +import { IDENTIFIER_MAX_CHAR_LENGTH } from 'src/engine/metadata-modules/utils/metadata.constants'; + +export const exceedsDatabaseIdentifierMaximumLength = (string: string) => { + return string.length > IDENTIFIER_MAX_CHAR_LENGTH; +}; diff --git a/packages/twenty-server/src/engine/metadata-modules/utils/validate-metadata-name.utils.ts b/packages/twenty-server/src/engine/metadata-modules/utils/validate-metadata-name.utils.ts index 16e8bfd56c9c..72835eead45e 100644 --- a/packages/twenty-server/src/engine/metadata-modules/utils/validate-metadata-name.utils.ts +++ b/packages/twenty-server/src/engine/metadata-modules/utils/validate-metadata-name.utils.ts @@ -1,9 +1,28 @@ -import { InvalidStringException } from 'src/engine/metadata-modules/errors/InvalidStringException'; +import { exceedsDatabaseIdentifierMaximumLength } from 'src/engine/metadata-modules/utils/validate-database-identifier-length.utils'; const VALID_STRING_PATTERN = /^[a-z][a-zA-Z0-9]*$/; -export const validateMetadataName = (string: string) => { - if (!string.match(VALID_STRING_PATTERN)) { - throw new InvalidStringException(string); +export const validateMetadataNameOrThrow = (name: string) => { + if (!name.match(VALID_STRING_PATTERN)) { + throw new InvalidStringException(name); + } + if (exceedsDatabaseIdentifierMaximumLength(name)) { + throw new NameTooLongException(name); } }; + +export class InvalidStringException extends Error { + constructor(string: string) { + const message = `String "${string}" is not valid`; + + super(message); + } +} + +export class NameTooLongException extends Error { + constructor(string: string) { + const message = `String "${string}" exceeds 63 characters limit`; + + super(message); + } +} diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/basic-column-action.factory.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/basic-column-action.factory.ts index 093b3cba9cd8..93e38a5966f6 100644 --- a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/basic-column-action.factory.ts +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/basic-column-action.factory.ts @@ -13,6 +13,10 @@ import { serializeDefaultValue } from 'src/engine/metadata-modules/field-metadat import { fieldMetadataTypeToColumnType } from 'src/engine/metadata-modules/workspace-migration/utils/field-metadata-type-to-column-type.util'; import { ColumnActionAbstractFactory } from 'src/engine/metadata-modules/workspace-migration/factories/column-action-abstract.factory'; import { computeColumnName } from 'src/engine/metadata-modules/field-metadata/utils/compute-column-name.util'; +import { + WorkspaceMigrationException, + WorkspaceMigrationExceptionCode, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.exception'; export type BasicFieldMetadataType = | FieldMetadataType.UUID @@ -21,7 +25,6 @@ export type BasicFieldMetadataType = | FieldMetadataType.EMAIL | FieldMetadataType.NUMERIC | FieldMetadataType.NUMBER - | FieldMetadataType.PROBABILITY | FieldMetadataType.BOOLEAN | FieldMetadataType.POSITION | FieldMetadataType.DATE_TIME @@ -66,8 +69,9 @@ export class BasicColumnActionFactory extends ColumnActionAbstractFactory<BasicF this.logger.error( `Column name not found for current or altered field metadata, can be due to a missing or an invalid target column map. Current column name: ${currentColumnName}, Altered column name: ${alteredColumnName}.`, ); - throw new Error( + throw new WorkspaceMigrationException( `Column name not found for current or altered field metadata`, + WorkspaceMigrationExceptionCode.INVALID_FIELD_METADATA, ); } diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/column-action-abstract.factory.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/column-action-abstract.factory.ts index a56d3a8fdb28..0e600aa6f618 100644 --- a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/column-action-abstract.factory.ts +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/column-action-abstract.factory.ts @@ -12,6 +12,10 @@ import { WorkspaceMigrationColumnAlter, } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { + WorkspaceMigrationException, + WorkspaceMigrationExceptionCode, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.exception'; export class ColumnActionAbstractFactory< T extends FieldMetadataType | 'default', @@ -32,7 +36,10 @@ export class ColumnActionAbstractFactory< return this.handleCreateAction(alteredFieldMetadata, options); case WorkspaceMigrationColumnActionType.ALTER: { if (!currentFieldMetadata) { - throw new Error('current field metadata is required for alter'); + throw new WorkspaceMigrationException( + 'current field metadata is required for alter', + WorkspaceMigrationExceptionCode.INVALID_FIELD_METADATA, + ); } return this.handleAlterAction( @@ -43,8 +50,10 @@ export class ColumnActionAbstractFactory< } default: { this.logger.error(`Invalid action: ${action}`); - - throw new Error('[AbstractFactory]: invalid action'); + throw new WorkspaceMigrationException( + '[AbstractFactory]: invalid action', + WorkspaceMigrationExceptionCode.INVALID_ACTION, + ); } } } @@ -53,7 +62,10 @@ export class ColumnActionAbstractFactory< _fieldMetadata: FieldMetadataInterface<T>, _options?: WorkspaceColumnActionOptions, ): WorkspaceMigrationColumnCreate[] { - throw new Error('handleCreateAction method not implemented.'); + throw new WorkspaceMigrationException( + 'handleCreateAction method not implemented.', + WorkspaceMigrationExceptionCode.INVALID_ACTION, + ); } protected handleAlterAction( @@ -61,6 +73,9 @@ export class ColumnActionAbstractFactory< _alteredFieldMetadata: FieldMetadataInterface<T>, _options?: WorkspaceColumnActionOptions, ): WorkspaceMigrationColumnAlter[] { - throw new Error('handleAlterAction method not implemented.'); + throw new WorkspaceMigrationException( + 'handleAlterAction method not implemented.', + WorkspaceMigrationExceptionCode.INVALID_ACTION, + ); } } diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/composite-column-action.factory.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/composite-column-action.factory.ts index 5947ba2c9723..a74bdd4e17d3 100644 --- a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/composite-column-action.factory.ts +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/composite-column-action.factory.ts @@ -13,6 +13,10 @@ import { fieldMetadataTypeToColumnType } from 'src/engine/metadata-modules/works import { ColumnActionAbstractFactory } from 'src/engine/metadata-modules/workspace-migration/factories/column-action-abstract.factory'; import { computeCompositeColumnName } from 'src/engine/metadata-modules/field-metadata/utils/compute-column-name.util'; import { compositeTypeDefintions } from 'src/engine/metadata-modules/field-metadata/composite-types'; +import { + WorkspaceMigrationException, + WorkspaceMigrationExceptionCode, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.exception'; export type CompositeFieldMetadataType = | FieldMetadataType.ADDRESS @@ -34,8 +38,9 @@ export class CompositeColumnActionFactory extends ColumnActionAbstractFactory<Co this.logger.error( `Composite type not found for field metadata type: ${fieldMetadata.type}`, ); - throw new Error( + throw new WorkspaceMigrationException( `Composite type not found for field metadata type: ${fieldMetadata.type}`, + WorkspaceMigrationExceptionCode.INVALID_FIELD_METADATA, ); } @@ -74,8 +79,9 @@ export class CompositeColumnActionFactory extends ColumnActionAbstractFactory<Co this.logger.error( `Composite type not found for field metadata type: ${currentFieldMetadata.type} or ${alteredFieldMetadata.type}`, ); - throw new Error( + throw new WorkspaceMigrationException( `Composite type not found for field metadata type: ${currentFieldMetadata.type} or ${alteredFieldMetadata.type}`, + WorkspaceMigrationExceptionCode.INVALID_FIELD_METADATA, ); } @@ -91,8 +97,9 @@ export class CompositeColumnActionFactory extends ColumnActionAbstractFactory<Co this.logger.error( `Current property not found for altered property: ${alteredProperty.name}`, ); - throw new Error( + throw new WorkspaceMigrationException( `Current property not found for altered property: ${alteredProperty.name}`, + WorkspaceMigrationExceptionCode.INVALID_FIELD_METADATA, ); } diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/enum-column-action.factory.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/enum-column-action.factory.ts index 1007f41e6297..939cc95b8901 100644 --- a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/enum-column-action.factory.ts +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/factories/enum-column-action.factory.ts @@ -13,6 +13,10 @@ import { serializeDefaultValue } from 'src/engine/metadata-modules/field-metadat import { fieldMetadataTypeToColumnType } from 'src/engine/metadata-modules/workspace-migration/utils/field-metadata-type-to-column-type.util'; import { ColumnActionAbstractFactory } from 'src/engine/metadata-modules/workspace-migration/factories/column-action-abstract.factory'; import { computeColumnName } from 'src/engine/metadata-modules/field-metadata/utils/compute-column-name.util'; +import { + WorkspaceMigrationException, + WorkspaceMigrationExceptionCode, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.exception'; export type EnumFieldMetadataType = | FieldMetadataType.RATING @@ -82,8 +86,9 @@ export class EnumColumnActionFactory extends ColumnActionAbstractFactory<EnumFie this.logger.error( `Column name not found for current or altered field metadata, can be due to a missing or an invalid target column map. Current column name: ${currentColumnName}, Altered column name: ${alteredColumnName}.`, ); - throw new Error( + throw new WorkspaceMigrationException( `Column name not found for current or altered field metadata`, + WorkspaceMigrationExceptionCode.INVALID_FIELD_METADATA, ); } diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/utils/field-metadata-type-to-column-type.util.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/utils/field-metadata-type-to-column-type.util.ts index f3999fd7f9f1..5c4f64213b27 100644 --- a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/utils/field-metadata-type-to-column-type.util.ts +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/utils/field-metadata-type-to-column-type.util.ts @@ -1,4 +1,8 @@ import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { + WorkspaceMigrationException, + WorkspaceMigrationExceptionCode, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.exception'; export const fieldMetadataTypeToColumnType = <Type extends FieldMetadataType>( fieldMetadataType: Type, @@ -18,7 +22,6 @@ export const fieldMetadataTypeToColumnType = <Type extends FieldMetadataType>( case FieldMetadataType.NUMERIC: return 'numeric'; case FieldMetadataType.NUMBER: - case FieldMetadataType.PROBABILITY: case FieldMetadataType.POSITION: return 'float'; case FieldMetadataType.BOOLEAN: @@ -34,6 +37,9 @@ export const fieldMetadataTypeToColumnType = <Type extends FieldMetadataType>( case FieldMetadataType.RAW_JSON: return 'jsonb'; default: - throw new Error(`Cannot convert ${fieldMetadataType} to column type.`); + throw new WorkspaceMigrationException( + `Cannot convert ${fieldMetadataType} to column type.`, + WorkspaceMigrationExceptionCode.INVALID_FIELD_METADATA, + ); } }; diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.entity.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.entity.ts index 8116e0b3dc9e..12084f80f1aa 100644 --- a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.entity.ts +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.entity.ts @@ -18,6 +18,11 @@ export enum WorkspaceMigrationColumnActionType { export type WorkspaceMigrationRenamedEnum = { from: string; to: string }; export type WorkspaceMigrationEnum = string | WorkspaceMigrationRenamedEnum; +export enum WorkspaceMigrationIndexActionType { + CREATE = 'CREATE', + DROP = 'DROP', +} + export interface WorkspaceMigrationColumnDefinition { columnName: string; columnType: string; @@ -27,6 +32,12 @@ export interface WorkspaceMigrationColumnDefinition { defaultValue?: any; } +export interface WorkspaceMigrationIndexAction { + action: WorkspaceMigrationIndexActionType; + name: string; + columns: string[]; +} + export interface WorkspaceMigrationColumnCreate extends WorkspaceMigrationColumnDefinition { action: WorkspaceMigrationColumnActionType.CREATE; @@ -105,6 +116,7 @@ export enum WorkspaceMigrationTableActionType { CREATE_FOREIGN_TABLE = 'create_foreign_table', DROP_FOREIGN_TABLE = 'drop_foreign_table', ALTER_FOREIGN_TABLE = 'alter_foreign_table', + ALTER_INDEXES = 'alter_indexes', } export type WorkspaceMigrationTableAction = { @@ -113,6 +125,7 @@ export type WorkspaceMigrationTableAction = { action: WorkspaceMigrationTableActionType; columns?: WorkspaceMigrationColumnAction[]; foreignTable?: WorkspaceMigrationForeignTable; + indexes?: WorkspaceMigrationIndexAction[]; }; @Entity('workspaceMigration') diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.exception.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.exception.ts new file mode 100644 index 000000000000..6340aeaa20af --- /dev/null +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.exception.ts @@ -0,0 +1,14 @@ +import { CustomException } from 'src/utils/custom-exception'; + +export class WorkspaceMigrationException extends CustomException { + code: WorkspaceMigrationExceptionCode; + constructor(message: string, code: WorkspaceMigrationExceptionCode) { + super(message, code); + } +} + +export enum WorkspaceMigrationExceptionCode { + NO_FACTORY_FOUND = 'NO_FACTORY_FOUND', + INVALID_ACTION = 'INVALID_ACTION', + INVALID_FIELD_METADATA = 'INVALID_FIELD_METADATA', +} diff --git a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.factory.ts b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.factory.ts index 6b3697294f86..02dcfe59ac27 100644 --- a/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.factory.ts +++ b/packages/twenty-server/src/engine/metadata-modules/workspace-migration/workspace-migration.factory.ts @@ -12,6 +12,10 @@ import { import { BasicColumnActionFactory } from 'src/engine/metadata-modules/workspace-migration/factories/basic-column-action.factory'; import { EnumColumnActionFactory } from 'src/engine/metadata-modules/workspace-migration/factories/enum-column-action.factory'; import { CompositeColumnActionFactory } from 'src/engine/metadata-modules/workspace-migration/factories/composite-column-action.factory'; +import { + WorkspaceMigrationException, + WorkspaceMigrationExceptionCode, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.exception'; @Injectable() export class WorkspaceMigrationFactory { @@ -68,10 +72,6 @@ export class WorkspaceMigrationFactory { [FieldMetadataType.NUMBER, { factory: this.basicColumnActionFactory }], [FieldMetadataType.POSITION, { factory: this.basicColumnActionFactory }], [FieldMetadataType.RAW_JSON, { factory: this.basicColumnActionFactory }], - [ - FieldMetadataType.PROBABILITY, - { factory: this.basicColumnActionFactory }, - ], [FieldMetadataType.BOOLEAN, { factory: this.basicColumnActionFactory }], [FieldMetadataType.DATE_TIME, { factory: this.basicColumnActionFactory }], [FieldMetadataType.DATE, { factory: this.basicColumnActionFactory }], @@ -131,7 +131,10 @@ export class WorkspaceMigrationFactory { undefinedOrAlteredFieldMetadata, ); - throw new Error(`No field metadata provided for action ${action}`); + throw new WorkspaceMigrationException( + `No field metadata provided for action ${action}`, + WorkspaceMigrationExceptionCode.INVALID_ACTION, + ); } const columnActions = this.createColumnAction( @@ -161,7 +164,10 @@ export class WorkspaceMigrationFactory { }, ); - throw new Error(`No factory found for type ${alteredFieldMetadata.type}`); + throw new WorkspaceMigrationException( + `No factory found for type ${alteredFieldMetadata.type}`, + WorkspaceMigrationExceptionCode.NO_FACTORY_FOUND, + ); } return factory.create( diff --git a/packages/twenty-server/src/engine/middlewares/graphql-hydrate-request-from-token.middleware.ts b/packages/twenty-server/src/engine/middlewares/graphql-hydrate-request-from-token.middleware.ts index b9d3d2cd5078..cbe7f3092963 100644 --- a/packages/twenty-server/src/engine/middlewares/graphql-hydrate-request-from-token.middleware.ts +++ b/packages/twenty-server/src/engine/middlewares/graphql-hydrate-request-from-token.middleware.ts @@ -55,6 +55,7 @@ export class GraphQLHydrateRequestFromTokenMiddleware req.user = data.user; req.workspace = data.workspace; + req.workspaceId = data.workspace.id; req.cacheVersion = cacheVersion; } catch (error) { res.writeHead(200, { 'Content-Type': 'application/json' }); diff --git a/packages/twenty-server/src/engine/object-metadata-repository/metadata-to-repository.mapping.ts b/packages/twenty-server/src/engine/object-metadata-repository/metadata-to-repository.mapping.ts index 17b4d0e8028c..1ead430c1a88 100644 --- a/packages/twenty-server/src/engine/object-metadata-repository/metadata-to-repository.mapping.ts +++ b/packages/twenty-server/src/engine/object-metadata-repository/metadata-to-repository.mapping.ts @@ -1,30 +1,19 @@ -import { CalendarChannelEventAssociationRepository } from 'src/modules/calendar/repositories/calendar-channel-event-association.repository'; -import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; -import { CalendarEventParticipantRepository } from 'src/modules/calendar/repositories/calendar-event-participant.repository'; -import { CalendarEventRepository } from 'src/modules/calendar/repositories/calendar-event.repository'; +import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; import { CompanyRepository } from 'src/modules/company/repositories/company.repository'; -import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; -import { AuditLogRepository } from 'src/modules/timeline/repositiories/audit-log.repository'; -import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository'; -import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; -import { AttachmentRepository } from 'src/modules/attachment/repositories/attachment.repository'; -import { CommentRepository } from 'src/modules/activity/repositories/comment.repository'; import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessageParticipantRepository } from 'src/modules/messaging/common/repositories/message-participant.repository'; import { MessageThreadRepository } from 'src/modules/messaging/common/repositories/message-thread.repository'; import { MessageRepository } from 'src/modules/messaging/common/repositories/message.repository'; import { PersonRepository } from 'src/modules/person/repositories/person.repository'; +import { AuditLogRepository } from 'src/modules/timeline/repositiories/audit-log.repository'; +import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository'; +import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; export const metadataToRepositoryMapping = { AuditLogWorkspaceEntity: AuditLogRepository, BlocklistWorkspaceEntity: BlocklistRepository, - CalendarChannelEventAssociationWorkspaceEntity: - CalendarChannelEventAssociationRepository, - CalendarChannelWorkspaceEntity: CalendarChannelRepository, - CalendarEventParticipantWorkspaceEntity: CalendarEventParticipantRepository, - CalendarEventWorkspaceEntity: CalendarEventRepository, CompanyWorkspaceEntity: CompanyRepository, ConnectedAccountWorkspaceEntity: ConnectedAccountRepository, MessageChannelMessageAssociationWorkspaceEntity: @@ -36,6 +25,4 @@ export const metadataToRepositoryMapping = { PersonWorkspaceEntity: PersonRepository, TimelineActivityWorkspaceEntity: TimelineActivityRepository, WorkspaceMemberWorkspaceEntity: WorkspaceMemberRepository, - AttachmentWorkspaceEntity: AttachmentRepository, - CommentWorkspaceEntity: CommentRepository, }; diff --git a/packages/twenty-server/src/engine/twenty-orm/context/load-service-with-workspace.context.ts b/packages/twenty-server/src/engine/twenty-orm/context/load-service-with-workspace.context.ts new file mode 100644 index 000000000000..02af97805044 --- /dev/null +++ b/packages/twenty-server/src/engine/twenty-orm/context/load-service-with-workspace.context.ts @@ -0,0 +1,39 @@ +import { Inject, Type } from '@nestjs/common'; +import { ModuleRef, createContextId } from '@nestjs/core'; +import { Injector } from '@nestjs/core/injector/injector'; + +export class LoadServiceWithWorkspaceContext { + private readonly injector = new Injector(); + + constructor( + @Inject(ModuleRef) + private readonly moduleRef: ModuleRef, + ) {} + + async load<T>(service: T, workspaceId: string): Promise<T> { + const modules = this.moduleRef['container'].getModules(); + const host = [...modules.values()].find((module) => + module.providers.has((service as Type<T>).constructor), + ); + + if (!host) { + throw new Error('Host module not found for the service'); + } + + const contextId = createContextId(); + + if (this.moduleRef.registerRequestByContextId) { + this.moduleRef.registerRequestByContextId( + { req: { workspaceId } }, + contextId, + ); + } + + return this.injector.loadPerContext( + service, + host, + new Map(host.providers), + contextId, + ); + } +} diff --git a/packages/twenty-server/src/engine/twenty-orm/custom.workspace-entity.ts b/packages/twenty-server/src/engine/twenty-orm/custom.workspace-entity.ts index ee7740f065ee..56f9238d249f 100644 --- a/packages/twenty-server/src/engine/twenty-orm/custom.workspace-entity.ts +++ b/packages/twenty-server/src/engine/twenty-orm/custom.workspace-entity.ts @@ -36,7 +36,7 @@ export class CustomWorkspaceEntity extends BaseWorkspaceEntity { }) @WorkspaceIsNullable() @WorkspaceIsSystem() - position: number; + position: number | null; @WorkspaceRelation({ standardId: CUSTOM_OBJECT_STANDARD_FIELD_IDS.activityTargets, diff --git a/packages/twenty-server/src/engine/twenty-orm/datasource/workspace.datasource.ts b/packages/twenty-server/src/engine/twenty-orm/datasource/workspace.datasource.ts index ab51a2492706..71ca2190903c 100644 --- a/packages/twenty-server/src/engine/twenty-orm/datasource/workspace.datasource.ts +++ b/packages/twenty-server/src/engine/twenty-orm/datasource/workspace.datasource.ts @@ -6,8 +6,8 @@ import { QueryRunner, } from 'typeorm'; -import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; import { WorkspaceEntityManager } from 'src/engine/twenty-orm/entity-manager/entity.manager'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; export class WorkspaceDataSource extends DataSource { readonly manager: WorkspaceEntityManager; diff --git a/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-field.decorator.ts b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-field.decorator.ts index 60a228d90d5d..cfbb7a96407d 100644 --- a/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-field.decorator.ts +++ b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-field.decorator.ts @@ -46,6 +46,13 @@ export function WorkspaceField<T extends FieldMetadataType>( object, propertyKey.toString(), ); + const isDeprecated = + TypedReflect.getMetadata( + 'workspace:is-deprecated-field-metadata-args', + object, + propertyKey.toString(), + ) ?? false; + const defaultValue = (options.defaultValue ?? generateDefaultValue( options.type, @@ -65,6 +72,7 @@ export function WorkspaceField<T extends FieldMetadataType>( isNullable, isSystem, gate, + isDeprecated, }); }; } diff --git a/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-index.decorator.ts b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-index.decorator.ts new file mode 100644 index 000000000000..0ba01f8e037a --- /dev/null +++ b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-index.decorator.ts @@ -0,0 +1,43 @@ +import { generateDeterministicIndexName } from 'src/engine/metadata-modules/index-metadata/utils/generate-deterministic-index-name'; +import { metadataArgsStorage } from 'src/engine/twenty-orm/storage/metadata-args.storage'; +import { convertClassNameToObjectMetadataName } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/convert-class-to-object-metadata-name.util'; + +export interface WorkspaceIndexOptions { + columns?: string[]; +} + +export function WorkspaceIndex(): PropertyDecorator; +export function WorkspaceIndex(columns: string[]): ClassDecorator; +export function WorkspaceIndex( + columns?: string[], +): PropertyDecorator | ClassDecorator { + return (target: any, propertyKey: string | symbol) => { + if (propertyKey === undefined && columns === undefined) { + throw new Error('Class level WorkspaceIndex should be used with columns'); + } + + // TODO: handle composite field metadata types + + if (Array.isArray(columns) && columns.length > 0) { + metadataArgsStorage.addIndexes({ + name: `IDX_${generateDeterministicIndexName([ + convertClassNameToObjectMetadataName(target.name), + ...columns, + ])}`, + columns, + target: target, + }); + + return; + } + + metadataArgsStorage.addIndexes({ + name: `IDX_${generateDeterministicIndexName([ + convertClassNameToObjectMetadataName(target.constructor.name), + propertyKey.toString(), + ])}`, + columns: [propertyKey.toString()], + target: target.constructor, + }); + }; +} diff --git a/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-is-deprecated.decorator.ts b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-is-deprecated.decorator.ts new file mode 100644 index 000000000000..a9bbb192e6e2 --- /dev/null +++ b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-is-deprecated.decorator.ts @@ -0,0 +1,12 @@ +import { TypedReflect } from 'src/utils/typed-reflect'; + +export function WorkspaceIsDeprecated(): PropertyDecorator { + return (object, propertyKey) => { + TypedReflect.defineMetadata( + 'workspace:is-deprecated-field-metadata-args', + true, + object, + propertyKey.toString(), + ); + }; +} diff --git a/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-join-column.decorator.ts b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-join-column.decorator.ts new file mode 100644 index 000000000000..98dc6eedb10c --- /dev/null +++ b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-join-column.decorator.ts @@ -0,0 +1,17 @@ +import { WorkspaceIndex } from 'src/engine/twenty-orm/decorators/workspace-index.decorator'; +import { metadataArgsStorage } from 'src/engine/twenty-orm/storage/metadata-args.storage'; + +export function WorkspaceJoinColumn( + relationPropertyKey: string, +): PropertyDecorator { + return (object, propertyKey) => { + metadataArgsStorage.addJoinColumns({ + target: object.constructor, + relationName: relationPropertyKey, + joinColumn: propertyKey.toString(), + }); + + // Register index for join column + WorkspaceIndex()(object, propertyKey); + }; +} diff --git a/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-relation.decorator.ts b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-relation.decorator.ts index e20877f00eab..2ead553d039b 100644 --- a/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-relation.decorator.ts +++ b/packages/twenty-server/src/engine/twenty-orm/decorators/workspace-relation.decorator.ts @@ -8,35 +8,19 @@ import { metadataArgsStorage } from 'src/engine/twenty-orm/storage/metadata-args import { TypedReflect } from 'src/utils/typed-reflect'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; -interface WorkspaceBaseRelationOptions<TType, TClass> { +interface WorkspaceRelationOptions<TClass> { standardId: string; label: string | ((objectMetadata: ObjectMetadataEntity) => string); description?: string | ((objectMetadata: ObjectMetadataEntity) => string); icon?: string; - type: TType; + type: RelationMetadataType; inverseSideTarget: () => ObjectType<TClass>; inverseSideFieldKey?: keyof TClass; onDelete?: RelationOnDeleteAction; } -export interface WorkspaceManyToOneRelationOptions<TClass> - extends WorkspaceBaseRelationOptions< - RelationMetadataType.MANY_TO_ONE | RelationMetadataType.ONE_TO_ONE, - TClass - > { - joinColumn?: string; -} - -export interface WorkspaceOtherRelationOptions<TClass> - extends WorkspaceBaseRelationOptions< - RelationMetadataType.ONE_TO_MANY | RelationMetadataType.MANY_TO_MANY, - TClass - > {} - export function WorkspaceRelation<TClass extends object>( - options: - | WorkspaceManyToOneRelationOptions<TClass> - | WorkspaceOtherRelationOptions<TClass>, + options: WorkspaceRelationOptions<TClass>, ): PropertyDecorator { return (object, propertyKey) => { const isPrimary = @@ -63,14 +47,6 @@ export function WorkspaceRelation<TClass extends object>( propertyKey.toString(), ); - let joinColumn: string | undefined; - - if ('joinColumn' in options) { - joinColumn = options.joinColumn - ? options.joinColumn - : `${propertyKey.toString()}Id`; - } - metadataArgsStorage.addRelations({ target: object.constructor, standardId: options.standardId, @@ -82,7 +58,6 @@ export function WorkspaceRelation<TClass extends object>( inverseSideTarget: options.inverseSideTarget, inverseSideFieldKey: options.inverseSideFieldKey as string | undefined, onDelete: options.onDelete, - joinColumn, isPrimary, isNullable, isSystem, diff --git a/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-column.factory.ts b/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-column.factory.ts index 68b537d8634b..83106f92fa68 100644 --- a/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-column.factory.ts +++ b/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-column.factory.ts @@ -4,6 +4,7 @@ import { ColumnType, EntitySchemaColumnOptions } from 'typeorm'; import { WorkspaceFieldMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-field-metadata-args.interface'; import { WorkspaceRelationMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface'; +import { WorkspaceJoinColumnsMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-join-columns-metadata-args.interface'; import { fieldMetadataTypeToColumnType } from 'src/engine/metadata-modules/workspace-migration/utils/field-metadata-type-to-column-type.util'; import { isEnumFieldMetadataType } from 'src/engine/metadata-modules/field-metadata/utils/is-enum-field-metadata-type.util'; @@ -12,6 +13,7 @@ import { computeCompositeColumnName } from 'src/engine/metadata-modules/field-me import { isCompositeFieldMetadataType } from 'src/engine/metadata-modules/field-metadata/utils/is-composite-field-metadata-type.util'; import { compositeTypeDefintions } from 'src/engine/metadata-modules/field-metadata/composite-types'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { getJoinColumn } from 'src/engine/twenty-orm/utils/get-join-column.util'; type EntitySchemaColumnMap = { [key: string]: EntitySchemaColumnOptions; @@ -22,6 +24,7 @@ export class EntitySchemaColumnFactory { create( fieldMetadataArgsCollection: WorkspaceFieldMetadataArgs[], relationMetadataArgsCollection: WorkspaceRelationMetadataArgs[], + joinColumnsMetadataArgsCollection: WorkspaceJoinColumnsMetadataArgs[], ): EntitySchemaColumnMap { let entitySchemaColumnMap: EntitySchemaColumnMap = {}; @@ -56,9 +59,14 @@ export class EntitySchemaColumnFactory { }; for (const relationMetadataArgs of relationMetadataArgsCollection) { - if (relationMetadataArgs.joinColumn) { - entitySchemaColumnMap[relationMetadataArgs.joinColumn] = { - name: relationMetadataArgs.joinColumn, + const joinColumn = getJoinColumn( + joinColumnsMetadataArgsCollection, + relationMetadataArgs, + ); + + if (joinColumn) { + entitySchemaColumnMap[joinColumn] = { + name: joinColumn, type: 'uuid', nullable: relationMetadataArgs.isNullable, }; diff --git a/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-relation.factory.ts b/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-relation.factory.ts index d194879214f3..bc83b460b26a 100644 --- a/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-relation.factory.ts +++ b/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema-relation.factory.ts @@ -4,9 +4,11 @@ import { EntitySchemaRelationOptions } from 'typeorm'; import { RelationType } from 'typeorm/metadata/types/RelationTypes'; import { WorkspaceRelationMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface'; +import { WorkspaceJoinColumnsMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-join-columns-metadata-args.interface'; import { convertClassNameToObjectMetadataName } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/convert-class-to-object-metadata-name.util'; import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { getJoinColumn } from 'src/engine/twenty-orm/utils/get-join-column.util'; type EntitySchemaRelationMap = { [key: string]: EntitySchemaRelationOptions; @@ -18,6 +20,7 @@ export class EntitySchemaRelationFactory { // eslint-disable-next-line @typescript-eslint/ban-types target: Function, relationMetadataArgsCollection: WorkspaceRelationMetadataArgs[], + joinColumnsMetadataArgsCollection: WorkspaceJoinColumnsMetadataArgs[], ): EntitySchemaRelationMap { const entitySchemaRelationMap: EntitySchemaRelationMap = {}; @@ -27,16 +30,19 @@ export class EntitySchemaRelationFactory { const oppositeObjectName = convertClassNameToObjectMetadataName( oppositeTarget.name, ); - const relationType = this.getRelationType(relationMetadataArgs); + const joinColumn = getJoinColumn( + joinColumnsMetadataArgsCollection, + relationMetadataArgs, + ); entitySchemaRelationMap[relationMetadataArgs.name] = { type: relationType, target: oppositeObjectName, inverseSide: relationMetadataArgs.inverseSideFieldKey ?? objectName, - joinColumn: relationMetadataArgs.joinColumn + joinColumn: joinColumn ? { - name: relationMetadataArgs.joinColumn, + name: joinColumn, } : undefined, }; diff --git a/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema.factory.ts b/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema.factory.ts index 326f5558a01a..75f63dcb221d 100644 --- a/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema.factory.ts +++ b/packages/twenty-server/src/engine/twenty-orm/factories/entity-schema.factory.ts @@ -23,17 +23,21 @@ export class EntitySchemaFactory { const fieldMetadataArgsCollection = metadataArgsStorage.filterFields(target); + const joinColumnsMetadataArgsCollection = + metadataArgsStorage.filterJoinColumns(target); const relationMetadataArgsCollection = metadataArgsStorage.filterRelations(target); const columns = this.entitySchemaColumnFactory.create( fieldMetadataArgsCollection, relationMetadataArgsCollection, + joinColumnsMetadataArgsCollection, ); const relations = this.entitySchemaRelationFactory.create( target, relationMetadataArgsCollection, + joinColumnsMetadataArgsCollection, ); const entitySchema = new EntitySchema({ diff --git a/packages/twenty-server/src/engine/twenty-orm/factories/scoped-workspace-datasource.factory.ts b/packages/twenty-server/src/engine/twenty-orm/factories/scoped-workspace-datasource.factory.ts index 0c21b6e74b6d..096be2194307 100644 --- a/packages/twenty-server/src/engine/twenty-orm/factories/scoped-workspace-datasource.factory.ts +++ b/packages/twenty-server/src/engine/twenty-orm/factories/scoped-workspace-datasource.factory.ts @@ -1,25 +1,27 @@ -import { Inject, Injectable, Scope } from '@nestjs/common'; +import { Inject, Injectable, Optional, Scope } from '@nestjs/common'; import { REQUEST } from '@nestjs/core'; import { EntitySchema } from 'typeorm'; -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { WorkspaceDatasourceFactory } from 'src/engine/twenty-orm/factories/workspace-datasource.factory'; @Injectable({ scope: Scope.REQUEST }) export class ScopedWorkspaceDatasourceFactory { constructor( - @Inject(REQUEST) private readonly request: Request, + @Optional() + @Inject(REQUEST) + private readonly request: Request | null, private readonly workspaceDataSourceFactory: WorkspaceDatasourceFactory, ) {} public async create(entities: EntitySchema[]) { - const workspace: Workspace | undefined = this.request['req']?.['workspace']; + const workspaceId: string | undefined = + this.request?.['req']?.['workspaceId']; - if (!workspace) { + if (!workspaceId) { return null; } - return this.workspaceDataSourceFactory.create(entities, workspace.id); + return this.workspaceDataSourceFactory.create(entities, workspaceId); } } diff --git a/packages/twenty-server/src/engine/twenty-orm/factories/workspace-datasource.factory.ts b/packages/twenty-server/src/engine/twenty-orm/factories/workspace-datasource.factory.ts index 27aead734ab4..2c733961a265 100644 --- a/packages/twenty-server/src/engine/twenty-orm/factories/workspace-datasource.factory.ts +++ b/packages/twenty-server/src/engine/twenty-orm/factories/workspace-datasource.factory.ts @@ -14,7 +14,10 @@ export class WorkspaceDatasourceFactory { private readonly environmentService: EnvironmentService, ) {} - public async create(entities: EntitySchema[], workspaceId: string) { + public async create( + entities: EntitySchema[], + workspaceId: string, + ): Promise<WorkspaceDataSource | null> { const storedWorkspaceDataSource = DataSourceStorage.getDataSource(workspaceId); @@ -23,19 +26,22 @@ export class WorkspaceDatasourceFactory { } const dataSourceMetadata = - await this.dataSourceService.getLastDataSourceMetadataFromWorkspaceIdOrFail( + await this.dataSourceService.getLastDataSourceMetadataFromWorkspaceId( workspaceId, ); + if (!dataSourceMetadata) { + return null; + } + const workspaceDataSource = new WorkspaceDataSource({ url: dataSourceMetadata.url ?? this.environmentService.get('PG_DATABASE_URL'), type: 'postgres', - // logging: this.environmentService.get('DEBUG_MODE') - // ? ['query', 'error'] - // : ['error'], - logging: 'all', + logging: this.environmentService.get('DEBUG_MODE') + ? ['query', 'error'] + : ['error'], schema: dataSourceMetadata.schema, entities, ssl: this.environmentService.get('PG_SSL_ALLOW_SELF_SIGNED') diff --git a/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-field-metadata-args.interface.ts b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-field-metadata-args.interface.ts index cad77a674e52..0574fb7922cf 100644 --- a/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-field-metadata-args.interface.ts +++ b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-field-metadata-args.interface.ts @@ -73,4 +73,9 @@ export interface WorkspaceFieldMetadataArgs { * Field gate. */ readonly gate?: Gate; + + /** + * Is deprecated field. + */ + readonly isDeprecated?: boolean; } diff --git a/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-index-metadata-args.interface.ts b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-index-metadata-args.interface.ts new file mode 100644 index 000000000000..add4c89e7476 --- /dev/null +++ b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-index-metadata-args.interface.ts @@ -0,0 +1,17 @@ +export interface WorkspaceIndexMetadataArgs { + /** + * Class to which index is applied. + */ + // eslint-disable-next-line @typescript-eslint/ban-types + readonly target: Function; + + /* + * Index name. + */ + name: string; + + /* + * Index columns. + */ + columns: string[]; +} diff --git a/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-join-columns-metadata-args.interface.ts b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-join-columns-metadata-args.interface.ts new file mode 100644 index 000000000000..481f4a77ea2d --- /dev/null +++ b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-join-columns-metadata-args.interface.ts @@ -0,0 +1,17 @@ +export interface WorkspaceJoinColumnsMetadataArgs { + /** + * Class to which relation is applied. + */ + // eslint-disable-next-line @typescript-eslint/ban-types + readonly target: Function; + + /** + * Relation name. + */ + readonly relationName: string; + + /** + * Relation label. + */ + readonly joinColumn: string; +} diff --git a/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface.ts b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface.ts index 911e9d78e8ed..861b1e7d44d4 100644 --- a/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface.ts +++ b/packages/twenty-server/src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface.ts @@ -62,11 +62,6 @@ export interface WorkspaceRelationMetadataArgs { */ readonly onDelete?: RelationOnDeleteAction; - /** - * Relation join column. - */ - readonly joinColumn?: string; - /** * Is primary field. */ diff --git a/packages/twenty-server/src/engine/twenty-orm/repository/workspace.repository.ts b/packages/twenty-server/src/engine/twenty-orm/repository/workspace.repository.ts index f03123e0734f..0f336768589d 100644 --- a/packages/twenty-server/src/engine/twenty-orm/repository/workspace.repository.ts +++ b/packages/twenty-server/src/engine/twenty-orm/repository/workspace.repository.ts @@ -1,6 +1,7 @@ import { DeepPartial, DeleteResult, + EntityManager, FindManyOptions, FindOneOptions, FindOptionsWhere, @@ -29,9 +30,13 @@ export class WorkspaceRepository< /** * FIND METHODS */ - override async find(options?: FindManyOptions<Entity>): Promise<Entity[]> { + override async find( + options?: FindManyOptions<Entity>, + entityManager?: EntityManager, + ): Promise<Entity[]> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions(options); - const result = await super.find(computedOptions); + const result = await manager.find(this.target, computedOptions); const formattedResult = this.formatResult(result); return formattedResult; @@ -39,9 +44,11 @@ export class WorkspaceRepository< override async findBy( where: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<Entity[]> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - const result = await super.findBy(computedOptions.where); + const result = await manager.findBy(this.target, computedOptions.where); const formattedResult = this.formatResult(result); return formattedResult; @@ -49,9 +56,11 @@ export class WorkspaceRepository< override async findAndCount( options?: FindManyOptions<Entity>, + entityManager?: EntityManager, ): Promise<[Entity[], number]> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions(options); - const result = await super.findAndCount(computedOptions); + const result = await manager.findAndCount(this.target, computedOptions); const formattedResult = this.formatResult(result); return formattedResult; @@ -59,9 +68,14 @@ export class WorkspaceRepository< override async findAndCountBy( where: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<[Entity[], number]> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - const result = await super.findAndCountBy(computedOptions.where); + const result = await manager.findAndCountBy( + this.target, + computedOptions.where, + ); const formattedResult = this.formatResult(result); return formattedResult; @@ -69,9 +83,11 @@ export class WorkspaceRepository< override async findOne( options: FindOneOptions<Entity>, + entityManager?: EntityManager, ): Promise<Entity | null> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions(options); - const result = await super.findOne(computedOptions); + const result = await manager.findOne(this.target, computedOptions); const formattedResult = this.formatResult(result); return formattedResult; @@ -79,9 +95,11 @@ export class WorkspaceRepository< override async findOneBy( where: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<Entity | null> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - const result = await super.findOneBy(computedOptions.where); + const result = await manager.findOneBy(this.target, computedOptions.where); const formattedResult = this.formatResult(result); return formattedResult; @@ -89,9 +107,11 @@ export class WorkspaceRepository< override async findOneOrFail( options: FindOneOptions<Entity>, + entityManager?: EntityManager, ): Promise<Entity> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions(options); - const result = await super.findOneOrFail(computedOptions); + const result = await manager.findOneOrFail(this.target, computedOptions); const formattedResult = this.formatResult(result); return formattedResult; @@ -99,9 +119,14 @@ export class WorkspaceRepository< override async findOneByOrFail( where: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<Entity> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - const result = await super.findOneByOrFail(computedOptions.where); + const result = await manager.findOneByOrFail( + this.target, + computedOptions.where, + ); const formattedResult = this.formatResult(result); return formattedResult; @@ -113,29 +138,40 @@ export class WorkspaceRepository< override save<T extends DeepPartial<Entity>>( entities: T[], options: SaveOptions & { reload: false }, + entityManager?: EntityManager, ): Promise<T[]>; override save<T extends DeepPartial<Entity>>( entities: T[], options?: SaveOptions, + entityManager?: EntityManager, ): Promise<(T & Entity)[]>; override save<T extends DeepPartial<Entity>>( entity: T, options: SaveOptions & { reload: false }, + entityManager?: EntityManager, ): Promise<T>; override save<T extends DeepPartial<Entity>>( entity: T, options?: SaveOptions, + entityManager?: EntityManager, ): Promise<T & Entity>; override async save<T extends DeepPartial<Entity>>( entityOrEntities: T | T[], options?: SaveOptions, + entityManager?: EntityManager, ): Promise<T | T[]> { + const manager = entityManager || this.manager; const formattedEntityOrEntities = this.formatData(entityOrEntities); - const result = await super.save(formattedEntityOrEntities as any, options); + const result = await manager.save( + this.target, + formattedEntityOrEntities as any, + options, + ); + const formattedResult = this.formatResult(result); return formattedResult; @@ -147,15 +183,27 @@ export class WorkspaceRepository< override remove( entities: Entity[], options?: RemoveOptions, + entityManager?: EntityManager, ): Promise<Entity[]>; - override remove(entity: Entity, options?: RemoveOptions): Promise<Entity>; + override remove( + entity: Entity, + options?: RemoveOptions, + entityManager?: EntityManager, + ): Promise<Entity>; override async remove( entityOrEntities: Entity | Entity[], + options?: RemoveOptions, + entityManager?: EntityManager, ): Promise<Entity | Entity[]> { + const manager = entityManager || this.manager; const formattedEntityOrEntities = this.formatData(entityOrEntities); - const result = await super.remove(formattedEntityOrEntities as any); + const result = await manager.remove( + this.target, + formattedEntityOrEntities, + options, + ); const formattedResult = this.formatResult(result); return formattedResult; @@ -172,40 +220,50 @@ export class WorkspaceRepository< | ObjectId | ObjectId[] | FindOptionsWhere<Entity>, + entityManager?: EntityManager, ): Promise<DeleteResult> { + const manager = entityManager || this.manager; + if (typeof criteria === 'object' && 'where' in criteria) { criteria = this.transformOptions(criteria); } - return this.delete(criteria); + return manager.delete(this.target, criteria); } override softRemove<T extends DeepPartial<Entity>>( entities: T[], options: SaveOptions & { reload: false }, + entityManager?: EntityManager, ): Promise<T[]>; override softRemove<T extends DeepPartial<Entity>>( entities: T[], options?: SaveOptions, + entityManager?: EntityManager, ): Promise<(T & Entity)[]>; override softRemove<T extends DeepPartial<Entity>>( entity: T, options: SaveOptions & { reload: false }, + entityManager?: EntityManager, ): Promise<T>; override softRemove<T extends DeepPartial<Entity>>( entity: T, options?: SaveOptions, + entityManager?: EntityManager, ): Promise<T & Entity>; override async softRemove<T extends DeepPartial<Entity>>( entityOrEntities: T | T[], options?: SaveOptions, + entityManager?: EntityManager, ): Promise<T | T[]> { + const manager = entityManager || this.manager; const formattedEntityOrEntities = this.formatData(entityOrEntities); - const result = await super.softRemove( + const result = await manager.softRemove( + this.target, formattedEntityOrEntities as any, options, ); @@ -225,12 +283,15 @@ export class WorkspaceRepository< | ObjectId | ObjectId[] | FindOptionsWhere<Entity>, + entityManager?: EntityManager, ): Promise<UpdateResult> { + const manager = entityManager || this.manager; + if (typeof criteria === 'object' && 'where' in criteria) { criteria = this.transformOptions(criteria); } - return this.softDelete(criteria); + return manager.softDelete(this.target, criteria); } /** @@ -239,29 +300,36 @@ export class WorkspaceRepository< override recover<T extends DeepPartial<Entity>>( entities: T[], options: SaveOptions & { reload: false }, + entityManager?: EntityManager, ): Promise<T[]>; override recover<T extends DeepPartial<Entity>>( entities: T[], options?: SaveOptions, + entityManager?: EntityManager, ): Promise<(T & Entity)[]>; override recover<T extends DeepPartial<Entity>>( entity: T, options: SaveOptions & { reload: false }, + entityManager?: EntityManager, ): Promise<T>; override recover<T extends DeepPartial<Entity>>( entity: T, options?: SaveOptions, + entityManager?: EntityManager, ): Promise<T & Entity>; override async recover<T extends DeepPartial<Entity>>( entityOrEntities: T | T[], options?: SaveOptions, + entityManager?: EntityManager, ): Promise<T | T[]> { + const manager = entityManager || this.manager; const formattedEntityOrEntities = this.formatData(entityOrEntities); - const result = await super.recover( + const result = await manager.recover( + this.target, formattedEntityOrEntities as any, options, ); @@ -281,12 +349,15 @@ export class WorkspaceRepository< | ObjectId | ObjectId[] | FindOptionsWhere<Entity>, + entityManager?: EntityManager, ): Promise<UpdateResult> { + const manager = entityManager || this.manager; + if (typeof criteria === 'object' && 'where' in criteria) { criteria = this.transformOptions(criteria); } - return this.restore(criteria); + return manager.restore(this.target, criteria); } /** @@ -294,9 +365,11 @@ export class WorkspaceRepository< */ override async insert( entity: QueryDeepPartialEntity<Entity> | QueryDeepPartialEntity<Entity>[], + entityManager?: EntityManager, ): Promise<InsertResult> { + const manager = entityManager || this.manager; const formatedEntity = this.formatData(entity); - const result = await super.insert(formatedEntity); + const result = await manager.insert(this.target, formatedEntity); const formattedResult = this.formatResult(result); return formattedResult; @@ -317,12 +390,15 @@ export class WorkspaceRepository< | ObjectId[] | FindOptionsWhere<Entity>, partialEntity: QueryDeepPartialEntity<Entity>, + entityManager?: EntityManager, ): Promise<UpdateResult> { + const manager = entityManager || this.manager; + if (typeof criteria === 'object' && 'where' in criteria) { criteria = this.transformOptions(criteria); } - return this.update(criteria, partialEntity); + return manager.update(this.target, criteria, partialEntity); } override upsert( @@ -330,50 +406,63 @@ export class WorkspaceRepository< | QueryDeepPartialEntity<Entity> | QueryDeepPartialEntity<Entity>[], conflictPathsOrOptions: string[] | UpsertOptions<Entity>, + entityManager?: EntityManager, ): Promise<InsertResult> { + const manager = entityManager || this.manager; + const formattedEntityOrEntities = this.formatData(entityOrEntities); - return this.upsert(formattedEntityOrEntities, conflictPathsOrOptions); + return manager.upsert( + this.target, + formattedEntityOrEntities, + conflictPathsOrOptions, + ); } /** * EXIST METHODS */ - override exist(options?: FindManyOptions<Entity>): Promise<boolean> { - const computedOptions = this.transformOptions(options); - - return super.exist(computedOptions); - } - - override exists(options?: FindManyOptions<Entity>): Promise<boolean> { + override exists( + options?: FindManyOptions<Entity>, + entityManager?: EntityManager, + ): Promise<boolean> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions(options); - return super.exists(computedOptions); + return manager.exists(this.target, computedOptions); } override existsBy( where: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<boolean> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - return super.existsBy(computedOptions.where); + return manager.existsBy(this.target, computedOptions.where); } /** * COUNT METHODS */ - override count(options?: FindManyOptions<Entity>): Promise<number> { + override count( + options?: FindManyOptions<Entity>, + entityManager?: EntityManager, + ): Promise<number> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions(options); - return super.count(computedOptions); + return manager.count(this.target, computedOptions); } override countBy( where: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<number> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - return super.countBy(computedOptions.where); + return manager.countBy(this.target, computedOptions.where); } /** @@ -382,57 +471,79 @@ export class WorkspaceRepository< override sum( columnName: PickKeysByType<Entity, number>, where?: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<number | null> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - return super.sum(columnName, computedOptions.where); + return manager.sum(this.target, columnName, computedOptions.where); } override average( columnName: PickKeysByType<Entity, number>, where?: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<number | null> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - return super.average(columnName, computedOptions.where); + return manager.average(this.target, columnName, computedOptions.where); } override minimum( columnName: PickKeysByType<Entity, number>, where?: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<number | null> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - return super.minimum(columnName, computedOptions.where); + return manager.minimum(this.target, columnName, computedOptions.where); } override maximum( columnName: PickKeysByType<Entity, number>, where?: FindOptionsWhere<Entity> | FindOptionsWhere<Entity>[], + entityManager?: EntityManager, ): Promise<number | null> { + const manager = entityManager || this.manager; const computedOptions = this.transformOptions({ where }); - return super.maximum(columnName, computedOptions.where); + return manager.maximum(this.target, columnName, computedOptions.where); } override increment( conditions: FindOptionsWhere<Entity>, propertyPath: string, value: number | string, + entityManager?: EntityManager, ): Promise<UpdateResult> { + const manager = entityManager || this.manager; const computedConditions = this.transformOptions({ where: conditions }); - return this.increment(computedConditions.where, propertyPath, value); + return manager.increment( + this.target, + computedConditions.where, + propertyPath, + value, + ); } override decrement( conditions: FindOptionsWhere<Entity>, propertyPath: string, value: number | string, + entityManager?: EntityManager, ): Promise<UpdateResult> { + const manager = entityManager || this.manager; const computedConditions = this.transformOptions({ where: conditions }); - return this.decrement(computedConditions.where, propertyPath, value); + return manager.decrement( + this.target, + computedConditions.where, + propertyPath, + value, + ); } /** diff --git a/packages/twenty-server/src/engine/twenty-orm/storage/metadata-args.storage.ts b/packages/twenty-server/src/engine/twenty-orm/storage/metadata-args.storage.ts index 9fb114be94f2..77b038b335c7 100644 --- a/packages/twenty-server/src/engine/twenty-orm/storage/metadata-args.storage.ts +++ b/packages/twenty-server/src/engine/twenty-orm/storage/metadata-args.storage.ts @@ -5,6 +5,8 @@ import { WorkspaceFieldMetadataArgs } from 'src/engine/twenty-orm/interfaces/wor import { WorkspaceEntityMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-entity-metadata-args.interface'; import { WorkspaceRelationMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface'; import { WorkspaceExtendedEntityMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-extended-entity-metadata-args.interface'; +import { WorkspaceIndexMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-index-metadata-args.interface'; +import { WorkspaceJoinColumnsMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-join-columns-metadata-args.interface'; export class MetadataArgsStorage { private readonly entities: WorkspaceEntityMetadataArgs[] = []; @@ -13,6 +15,8 @@ export class MetadataArgsStorage { private readonly relations: WorkspaceRelationMetadataArgs[] = []; private readonly dynamicRelations: WorkspaceDynamicRelationMetadataArgs[] = []; + private readonly indexes: WorkspaceIndexMetadataArgs[] = []; + private readonly joinColumns: WorkspaceJoinColumnsMetadataArgs[] = []; addEntities(...entities: WorkspaceEntityMetadataArgs[]): void { this.entities.push(...entities); @@ -32,12 +36,20 @@ export class MetadataArgsStorage { this.relations.push(...relations); } + addIndexes(...indexes: WorkspaceIndexMetadataArgs[]): void { + this.indexes.push(...indexes); + } + addDynamicRelations( ...dynamicRelations: WorkspaceDynamicRelationMetadataArgs[] ): void { this.dynamicRelations.push(...dynamicRelations); } + addJoinColumns(...joinColumns: WorkspaceJoinColumnsMetadataArgs[]): void { + this.joinColumns.push(...joinColumns); + } + filterEntities( target: Function | string, ): WorkspaceEntityMetadataArgs | undefined; @@ -93,6 +105,16 @@ export class MetadataArgsStorage { return this.filterByTarget(this.relations, target); } + filterIndexes(target: Function | string): WorkspaceIndexMetadataArgs[]; + + filterIndexes(target: (Function | string)[]): WorkspaceIndexMetadataArgs[]; + + filterIndexes( + target: (Function | string) | (Function | string)[], + ): WorkspaceIndexMetadataArgs[] { + return this.filterByTarget(this.indexes, target); + } + filterDynamicRelations( target: Function | string, ): WorkspaceDynamicRelationMetadataArgs[]; @@ -107,6 +129,20 @@ export class MetadataArgsStorage { return this.filterByTarget(this.dynamicRelations, target); } + filterJoinColumns( + target: Function | string, + ): WorkspaceJoinColumnsMetadataArgs[]; + + filterJoinColumns( + target: (Function | string)[], + ): WorkspaceJoinColumnsMetadataArgs[]; + + filterJoinColumns( + target: (Function | string) | (Function | string)[], + ): WorkspaceJoinColumnsMetadataArgs[] { + return this.filterByTarget(this.joinColumns, target); + } + protected filterByTarget<T extends { target: Function | string }>( array: T[], target: (Function | string) | (Function | string)[], diff --git a/packages/twenty-server/src/engine/twenty-orm/twenty-orm-core.module.ts b/packages/twenty-server/src/engine/twenty-orm/twenty-orm-core.module.ts index befe7c28a717..efed8ca7bcae 100644 --- a/packages/twenty-server/src/engine/twenty-orm/twenty-orm-core.module.ts +++ b/packages/twenty-server/src/engine/twenty-orm/twenty-orm-core.module.ts @@ -7,10 +7,6 @@ import { Provider, Type, } from '@nestjs/common'; -import { - ConfigurableModuleClass, - MODULE_OPTIONS_TOKEN, -} from '@nestjs/common/cache/cache.module-definition'; import { importClassesFromDirectories } from 'typeorm/util/DirectoryExportedClassesLoader'; import { Logger as TypeORMLogger } from 'typeorm/logger/Logger'; @@ -30,12 +26,25 @@ import { ScopedWorkspaceDatasourceFactory } from 'src/engine/twenty-orm/factorie import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; import { splitClassesAndStrings } from 'src/engine/twenty-orm/utils/split-classes-and-strings.util'; import { CustomWorkspaceEntity } from 'src/engine/twenty-orm/custom.workspace-entity'; +import { + ConfigurableModuleClass, + MODULE_OPTIONS_TOKEN, +} from 'src/engine/twenty-orm/twenty-orm.module-definition'; +import { LoadServiceWithWorkspaceContext } from 'src/engine/twenty-orm/context/load-service-with-workspace.context'; @Global() @Module({ imports: [DataSourceModule], - providers: [...entitySchemaFactories, TwentyORMManager], - exports: [EntitySchemaFactory, TwentyORMManager], + providers: [ + ...entitySchemaFactories, + TwentyORMManager, + LoadServiceWithWorkspaceContext, + ], + exports: [ + EntitySchemaFactory, + TwentyORMManager, + LoadServiceWithWorkspaceContext, + ], }) export class TwentyORMCoreModule extends ConfigurableModuleClass @@ -46,7 +55,6 @@ export class TwentyORMCoreModule static register(options: TwentyORMOptions): DynamicModule { const dynamicModule = super.register(options); - console.log('register', options); const providers: Provider[] = [ { provide: TWENTY_ORM_WORKSPACE_DATASOURCE, diff --git a/packages/twenty-server/src/engine/twenty-orm/twenty-orm.manager.ts b/packages/twenty-server/src/engine/twenty-orm/twenty-orm.manager.ts index caaa4c21646d..668379a0aa51 100644 --- a/packages/twenty-server/src/engine/twenty-orm/twenty-orm.manager.ts +++ b/packages/twenty-server/src/engine/twenty-orm/twenty-orm.manager.ts @@ -1,19 +1,46 @@ -import { Injectable, Type } from '@nestjs/common'; +import { Injectable, Optional, Type } from '@nestjs/common'; import { ObjectLiteral } from 'typeorm'; -import { EntitySchemaFactory } from 'src/engine/twenty-orm/factories/entity-schema.factory'; -import { InjectWorkspaceDatasource } from 'src/engine/twenty-orm/decorators/inject-workspace-datasource.decorator'; import { WorkspaceDataSource } from 'src/engine/twenty-orm/datasource/workspace.datasource'; -import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { EntitySchemaFactory } from 'src/engine/twenty-orm/factories/entity-schema.factory'; import { WorkspaceDatasourceFactory } from 'src/engine/twenty-orm/factories/workspace-datasource.factory'; -import { ObjectLiteralStorage } from 'src/engine/twenty-orm/storage/object-literal.storage'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { ActivityTargetWorkspaceEntity } from 'src/modules/activity/standard-objects/activity-target.workspace-entity'; +import { ActivityWorkspaceEntity } from 'src/modules/activity/standard-objects/activity.workspace-entity'; +import { CommentWorkspaceEntity } from 'src/modules/activity/standard-objects/comment.workspace-entity'; +import { ApiKeyWorkspaceEntity } from 'src/modules/api-key/standard-objects/api-key.workspace-entity'; +import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objects/attachment.workspace-entity'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; +import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/company.workspace-entity'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { FavoriteWorkspaceEntity } from 'src/modules/favorite/standard-objects/favorite.workspace-entity'; +import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; +import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; +import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; +import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; +import { OpportunityWorkspaceEntity } from 'src/modules/opportunity/standard-objects/opportunity.workspace-entity'; +import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; +import { AuditLogWorkspaceEntity } from 'src/modules/timeline/standard-objects/audit-log.workspace-entity'; +import { BehavioralEventWorkspaceEntity } from 'src/modules/timeline/standard-objects/behavioral-event.workspace-entity'; +import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; +import { ViewFieldWorkspaceEntity } from 'src/modules/view/standard-objects/view-field.workspace-entity'; +import { ViewFilterWorkspaceEntity } from 'src/modules/view/standard-objects/view-filter.workspace-entity'; +import { ViewSortWorkspaceEntity } from 'src/modules/view/standard-objects/view-sort.workspace-entity'; +import { ViewWorkspaceEntity } from 'src/modules/view/standard-objects/view.workspace-entity'; +import { WebhookWorkspaceEntity } from 'src/modules/webhook/standard-objects/webhook.workspace-entity'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; @Injectable() export class TwentyORMManager { constructor( - @InjectWorkspaceDatasource() - private readonly workspaceDataSource: WorkspaceDataSource, + @Optional() + private readonly workspaceDataSource: WorkspaceDataSource | null, private readonly entitySchemaFactory: EntitySchemaFactory, private readonly workspaceDataSourceFactory: WorkspaceDatasourceFactory, ) {} @@ -23,6 +50,10 @@ export class TwentyORMManager { ): WorkspaceRepository<T> { const entitySchema = this.entitySchemaFactory.create(entityClass); + if (!this.workspaceDataSource) { + throw new Error('Workspace data source not found'); + } + return this.workspaceDataSource.getRepository<T>(entitySchema); } @@ -30,11 +61,52 @@ export class TwentyORMManager { workspaceId: string, entityClass: Type<T>, ): Promise<WorkspaceRepository<T>> { - const entities = ObjectLiteralStorage.getAllEntitySchemas(); + // TODO: This is a temporary solution to get all workspace entities + const workspaceEntities = [ + ActivityTargetWorkspaceEntity, + ActivityWorkspaceEntity, + ApiKeyWorkspaceEntity, + AttachmentWorkspaceEntity, + BlocklistWorkspaceEntity, + BehavioralEventWorkspaceEntity, + CalendarChannelEventAssociationWorkspaceEntity, + CalendarChannelWorkspaceEntity, + CalendarEventParticipantWorkspaceEntity, + CalendarEventWorkspaceEntity, + CommentWorkspaceEntity, + CompanyWorkspaceEntity, + ConnectedAccountWorkspaceEntity, + FavoriteWorkspaceEntity, + AuditLogWorkspaceEntity, + MessageChannelMessageAssociationWorkspaceEntity, + MessageChannelWorkspaceEntity, + MessageParticipantWorkspaceEntity, + MessageThreadWorkspaceEntity, + MessageWorkspaceEntity, + OpportunityWorkspaceEntity, + PersonWorkspaceEntity, + TimelineActivityWorkspaceEntity, + ViewFieldWorkspaceEntity, + ViewFilterWorkspaceEntity, + ViewSortWorkspaceEntity, + ViewWorkspaceEntity, + WebhookWorkspaceEntity, + WorkspaceMemberWorkspaceEntity, + ]; + + const entities = workspaceEntities.map((workspaceEntity) => + this.entitySchemaFactory.create(workspaceEntity as any), + ); + const workspaceDataSource = await this.workspaceDataSourceFactory.create( entities, workspaceId, ); + + if (!workspaceDataSource) { + throw new Error('Workspace data source not found'); + } + const entitySchema = this.entitySchemaFactory.create(entityClass); return workspaceDataSource.getRepository<T>(entitySchema); diff --git a/packages/twenty-server/src/engine/twenty-orm/twenty-orm.module.ts b/packages/twenty-server/src/engine/twenty-orm/twenty-orm.module.ts index 3cf38e38c3f0..cd94d01298c4 100644 --- a/packages/twenty-server/src/engine/twenty-orm/twenty-orm.module.ts +++ b/packages/twenty-server/src/engine/twenty-orm/twenty-orm.module.ts @@ -1,5 +1,4 @@ import { DynamicModule, Global, Module } from '@nestjs/common'; -import { ConfigurableModuleClass } from '@nestjs/common/cache/cache.module-definition'; import { EntityClassOrSchema } from '@nestjs/typeorm/dist/interfaces/entity-class-or-schema.type'; import { @@ -12,7 +11,7 @@ import { TwentyORMCoreModule } from 'src/engine/twenty-orm/twenty-orm-core.modul @Global() @Module({}) -export class TwentyORMModule extends ConfigurableModuleClass { +export class TwentyORMModule { static register(options: TwentyORMOptions): DynamicModule { return { module: TwentyORMModule, diff --git a/packages/twenty-server/src/engine/twenty-orm/utils/get-join-column.util.ts b/packages/twenty-server/src/engine/twenty-orm/utils/get-join-column.util.ts new file mode 100644 index 000000000000..a328b5a3f6b4 --- /dev/null +++ b/packages/twenty-server/src/engine/twenty-orm/utils/get-join-column.util.ts @@ -0,0 +1,87 @@ +import { WorkspaceJoinColumnsMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-join-columns-metadata-args.interface'; +import { WorkspaceRelationMetadataArgs } from 'src/engine/twenty-orm/interfaces/workspace-relation-metadata-args.interface'; + +import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { metadataArgsStorage } from 'src/engine/twenty-orm/storage/metadata-args.storage'; + +export const getJoinColumn = ( + joinColumnsMetadataArgsCollection: WorkspaceJoinColumnsMetadataArgs[], + relationMetadataArgs: WorkspaceRelationMetadataArgs, + opposite = false, +): string | null => { + if ( + relationMetadataArgs.type === RelationMetadataType.ONE_TO_MANY || + relationMetadataArgs.type === RelationMetadataType.MANY_TO_MANY + ) { + return null; + } + + const inverseSideTarget = relationMetadataArgs.inverseSideTarget(); + const inverseSideJoinColumnsMetadataArgsCollection = + metadataArgsStorage.filterJoinColumns(inverseSideTarget); + const filteredJoinColumnsMetadataArgsCollection = + joinColumnsMetadataArgsCollection.filter( + (joinColumnsMetadataArgs) => + joinColumnsMetadataArgs.relationName === relationMetadataArgs.name, + ); + const oppositeFilteredJoinColumnsMetadataArgsCollection = + inverseSideJoinColumnsMetadataArgsCollection.filter( + (joinColumnsMetadataArgs) => + joinColumnsMetadataArgs.relationName === relationMetadataArgs.name, + ); + + if ( + filteredJoinColumnsMetadataArgsCollection.length > 0 && + oppositeFilteredJoinColumnsMetadataArgsCollection.length > 0 + ) { + throw new Error( + `Join column for ${relationMetadataArgs.name} relation is present on both sides`, + ); + } + + // If we're in a ONE_TO_ONE relation and there are no join columns, we need to find the join column on the inverse side + if ( + relationMetadataArgs.type === RelationMetadataType.ONE_TO_ONE && + filteredJoinColumnsMetadataArgsCollection.length === 0 && + !opposite + ) { + const inverseSideRelationMetadataArgsCollection = + metadataArgsStorage.filterRelations(inverseSideTarget); + const inverseSideRelationMetadataArgs = + inverseSideRelationMetadataArgsCollection.find( + (inverseSideRelationMetadataArgs) => + inverseSideRelationMetadataArgs.inverseSideFieldKey === + relationMetadataArgs.name, + ); + + if (!inverseSideRelationMetadataArgs) { + throw new Error( + `Inverse side join column of relation ${relationMetadataArgs.name} is missing`, + ); + } + + return getJoinColumn( + inverseSideJoinColumnsMetadataArgsCollection, + inverseSideRelationMetadataArgs, + // Avoid infinite recursion + true, + ); + } + + // Check if there are multiple join columns for the relation + if (filteredJoinColumnsMetadataArgsCollection.length > 1) { + throw new Error( + `Multiple join columns found for relation ${relationMetadataArgs.name}`, + ); + } + + const joinColumnsMetadataArgs = filteredJoinColumnsMetadataArgsCollection[0]; + + if (!joinColumnsMetadataArgs) { + throw new Error( + `Join column is missing for relation ${relationMetadataArgs.name}`, + ); + } + + return joinColumnsMetadataArgs.joinColumn; +}; diff --git a/packages/twenty-server/src/engine/utils/global-exception-handler.util.ts b/packages/twenty-server/src/engine/utils/global-exception-handler.util.ts index 326f991bb401..7bb281278a9b 100644 --- a/packages/twenty-server/src/engine/utils/global-exception-handler.util.ts +++ b/packages/twenty-server/src/engine/utils/global-exception-handler.util.ts @@ -7,13 +7,14 @@ import { ExceptionHandlerUser } from 'src/engine/integrations/exception-handler/ import { AuthenticationError, BaseGraphQLError, - ForbiddenError, - ValidationError, - NotFoundError, ConflictError, + ErrorCode, + ForbiddenError, MethodNotAllowedError, + NotFoundError, TimeoutError, -} from 'src/engine/utils/graphql-errors.util'; + ValidationError, +} from 'src/engine/core-modules/graphql/utils/graphql-errors.util'; import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service'; const graphQLPredefinedExceptions = { @@ -26,6 +27,17 @@ const graphQLPredefinedExceptions = { 409: ConflictError, }; +export const graphQLErrorCodesToFilter = [ + ErrorCode.GRAPHQL_VALIDATION_FAILED, + ErrorCode.UNAUTHENTICATED, + ErrorCode.FORBIDDEN, + ErrorCode.NOT_FOUND, + ErrorCode.METHOD_NOT_ALLOWED, + ErrorCode.TIMEOUT, + ErrorCode.CONFLICT, + ErrorCode.BAD_USER_INPUT, +]; + export const handleExceptionAndConvertToGraphQLError = ( exception: Error, exceptionHandlerService: ExceptionHandlerService, @@ -43,6 +55,14 @@ export const shouldFilterException = (exception: Error): boolean => { ) { return true; } + + if ( + exception instanceof BaseGraphQLError && + graphQLErrorCodesToFilter.includes(exception?.extensions?.code) + ) { + return true; + } + if (exception instanceof HttpException && exception.getStatus() < 500) { return true; } @@ -50,7 +70,7 @@ export const shouldFilterException = (exception: Error): boolean => { return false; }; -export const handleException = ( +const handleException = ( exception: Error, exceptionHandlerService: ExceptionHandlerService, user?: ExceptionHandlerUser, @@ -68,11 +88,14 @@ export const convertExceptionToGraphQLError = ( if (exception instanceof HttpException) { return convertHttpExceptionToGraphql(exception); } + if (exception instanceof BaseGraphQLError) { + return exception; + } return convertExceptionToGraphql(exception); }; -export const convertHttpExceptionToGraphql = (exception: HttpException) => { +const convertHttpExceptionToGraphql = (exception: HttpException) => { const status = exception.getStatus(); let error: BaseGraphQLError; @@ -97,7 +120,10 @@ export const convertHttpExceptionToGraphql = (exception: HttpException) => { }; export const convertExceptionToGraphql = (exception: Error) => { - const error = new BaseGraphQLError(exception.name, 'INTERNAL_SERVER_ERROR'); + const error = new BaseGraphQLError( + exception.name, + ErrorCode.INTERNAL_SERVER_ERROR, + ); error.stack = exception.stack; error.extensions['response'] = exception.message; diff --git a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/companies-demo.json.ts b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/companies-demo.json.ts index 32a29ab1609a..258d4d6730f0 100644 --- a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/companies-demo.json.ts +++ b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/companies-demo.json.ts @@ -2,532 +2,532 @@ export const companiesDemo = [ { name: 'Google', domainName: 'goo.gle', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 284571, linkedinLinkUrl: 'https://linkedin.com/company/google', }, { name: 'Microsoft', domainName: 'microsoft.com', - address: 'Redmond', + addressAddressCity: 'Redmond', employees: 226067, linkedinLinkUrl: 'https://linkedin.com/company/microsoft', }, { name: 'Meta', domainName: 'metacareers.com', - address: 'Menlo Park', + addressAddressCity: 'Menlo Park', employees: 119511, linkedinLinkUrl: 'https://linkedin.com/company/meta', }, { name: 'SLB', domainName: 'slb.com', - address: 'Houston', + addressAddressCity: 'Houston', employees: 113151, linkedinLinkUrl: 'https://linkedin.com/company/slbglobal', }, { name: 'Cisco', domainName: 'cisco.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 99625, linkedinLinkUrl: 'https://linkedin.com/company/cisco', }, { name: 'Uber', domainName: 'uber.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 90545, linkedinLinkUrl: 'https://linkedin.com/company/uber-com', }, { name: 'Salesforce', domainName: 'salesforce.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 71322, linkedinLinkUrl: 'https://linkedin.com/company/salesforce', }, { name: 'Amdocs', domainName: 'amdocs.com', - address: 'Chesterfield', + addressAddressCity: 'Chesterfield', employees: 35731, linkedinLinkUrl: 'https://linkedin.com/company/amdocs', }, { name: 'VMware', domainName: 'vmware.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 34759, linkedinLinkUrl: 'https://linkedin.com/company/vmware', }, { name: 'GlobalLogic', domainName: 'globallogic.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 24461, linkedinLinkUrl: 'https://linkedin.com/company/globallogic', }, { name: 'ServiceNow', domainName: 'servicenow.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 24104, linkedinLinkUrl: 'https://linkedin.com/company/servicenow', }, { name: 'SS&C Technologies', domainName: 'ssctech.com', - address: 'Windsor', + addressAddressCity: 'Windsor', employees: 20311, linkedinLinkUrl: 'https://linkedin.com/company/ss-c-technologies', }, { name: 'Workday', domainName: 'workday.com', - address: 'Pleasanton', + addressAddressCity: 'Pleasanton', employees: 20036, linkedinLinkUrl: 'https://linkedin.com/company/workday', }, { name: 'Red Hat', domainName: 'redhat.com', - address: 'Raleigh', + addressAddressCity: 'Raleigh', employees: 19945, linkedinLinkUrl: 'https://linkedin.com/company/red-hat', }, { name: 'NetSuite', domainName: 'netsuite.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 19269, linkedinLinkUrl: 'https://linkedin.com/company/netsuite', }, { name: 'Synopsys Inc', domainName: 'synopsys.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 18061, linkedinLinkUrl: 'https://linkedin.com/company/synopsys', }, { name: 'Siemens Digital Industries Software', domainName: 'sw.siemens.com', - address: 'Plano', + addressAddressCity: 'Plano', employees: 17262, linkedinLinkUrl: 'https://linkedin.com/company/siemenssoftware', }, { name: 'SAS', domainName: 'sas.com', - address: 'Cary', + addressAddressCity: 'Cary', employees: 16287, linkedinLinkUrl: 'https://linkedin.com/company/sas', }, { name: 'Intuit', domainName: 'intuit.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 15851, linkedinLinkUrl: 'https://linkedin.com/company/intuit', }, { name: 'Broadcom Software', domainName: 'broadcom.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 15127, linkedinLinkUrl: 'https://linkedin.com/company/broadcomsoftware', }, { name: 'Autodesk', domainName: 'autodesk.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 14593, linkedinLinkUrl: 'https://linkedin.com/company/autodesk', }, { name: 'Epic', domainName: 'epic.com', - address: 'Verona', + addressAddressCity: 'Verona', employees: 13765, linkedinLinkUrl: 'https://linkedin.com/company/epic1979', }, { name: 'Bosch USA', domainName: 'bosch.us', - address: 'Farmington', + addressAddressCity: 'Farmington', employees: 13754, linkedinLinkUrl: 'https://linkedin.com/company/boschusa', }, { name: 'Cloud Software Group', domainName: 'cloudsoftwaregroup.com', - address: 'Fort Lauderdale', + addressAddressCity: 'Fort Lauderdale', employees: 13111, linkedinLinkUrl: 'https://linkedin.com/company/cloudsoftwaregroup', }, { name: 'Pitney Bowes', domainName: 'pitneybowes.com', - address: 'Stamford', + addressAddressCity: 'Stamford', employees: 12306, linkedinLinkUrl: 'https://linkedin.com/company/pitney-bowes', }, { name: 'Juniper Networks', domainName: 'juniper.net', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 11928, linkedinLinkUrl: 'https://linkedin.com/company/juniper-networks', }, { name: 'Chegg Inc.', domainName: 'chegg.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 10790, linkedinLinkUrl: 'https://linkedin.com/company/chegg-inc-', }, { name: 'Teradata', domainName: 'teradata.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 10748, linkedinLinkUrl: 'https://linkedin.com/company/teradata', }, { name: 'NICE', domainName: 'nice.com', - address: 'Hoboken', + addressAddressCity: 'Hoboken', employees: 10258, linkedinLinkUrl: 'https://linkedin.com/company/nice-systems', }, { name: 'Cadence Design Systems', domainName: 'cadence.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 9377, linkedinLinkUrl: 'https://linkedin.com/company/cadence-design-systems', }, { name: 'Cox Automotive Inc.', domainName: 'coxautoinc.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 9331, linkedinLinkUrl: 'https://linkedin.com/company/cox-automotive-inc-', }, { name: 'Trimble Inc.', domainName: 'trimble.com', - address: 'Broomfield', + addressAddressCity: 'Broomfield', employees: 9311, linkedinLinkUrl: 'https://linkedin.com/company/trimble', }, { name: '[24]7.ai', domainName: '247.ai', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 9170, linkedinLinkUrl: 'https://linkedin.com/company/24-7-ai', }, { name: 'Akamai Technologies', domainName: 'akamai.com', - address: 'Cambridge', + addressAddressCity: 'Cambridge', employees: 9168, linkedinLinkUrl: 'https://linkedin.com/company/akamai-technologies', }, { name: 'Splunk', domainName: 'splunk.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 8891, linkedinLinkUrl: 'https://linkedin.com/company/splunk', }, { name: 'Okta', domainName: 'okta.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 8860, linkedinLinkUrl: 'https://linkedin.com/company/okta-inc-', }, { name: 'Ceridian', domainName: 'ceridian.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 8813, linkedinLinkUrl: 'https://linkedin.com/company/ceridian', }, { name: 'RealPage, Inc.', domainName: 'realpage.com', - address: 'Richardson', + addressAddressCity: 'Richardson', employees: 8227, linkedinLinkUrl: 'https://linkedin.com/company/realpage', }, { name: 'Freelance', domainName: 'jobicy.com', - address: 'Ny', + addressAddressCity: 'Ny', employees: 8180, linkedinLinkUrl: 'https://linkedin.com/company/pro-freelance', }, { name: 'Stripe', domainName: 'stripe.com', - address: 'South San Francisco', + addressAddressCity: 'South San Francisco', employees: 8145, linkedinLinkUrl: 'https://linkedin.com/company/stripe', }, { name: 'Shutterfly', domainName: 'shutterflyinc.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 8070, linkedinLinkUrl: 'https://linkedin.com/company/shutterfly', }, { name: 'Unity', domainName: 'unity.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 8063, linkedinLinkUrl: 'https://linkedin.com/company/unity', }, { name: 'Veeva Systems', domainName: 'veeva.com', - address: 'Pleasanton', + addressAddressCity: 'Pleasanton', employees: 7831, linkedinLinkUrl: 'https://linkedin.com/company/veeva-systems', }, { name: 'Nuance Communications', domainName: 'nuance.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 7761, linkedinLinkUrl: 'https://linkedin.com/company/nuance-communications', }, { name: 'Freshworks', domainName: 'freshworks.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 7687, linkedinLinkUrl: 'https://linkedin.com/company/freshworks-inc', }, { name: 'Seal Software, a DocuSign Company', domainName: 'seal-software.com', - address: 'Walnut Creek', + addressAddressCity: 'Walnut Creek', employees: 7586, linkedinLinkUrl: 'https://linkedin.com/company/seal-software-group', }, { name: 'DocuSign', domainName: 'docusign.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 7557, linkedinLinkUrl: 'https://linkedin.com/company/docusign', }, { name: 'Nutanix', domainName: 'nutanix.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 7454, linkedinLinkUrl: 'https://linkedin.com/company/nutanix', }, { name: 'Genesys', domainName: 'genesys.com', - address: 'Menlo Park', + addressAddressCity: 'Menlo Park', employees: 7371, linkedinLinkUrl: 'https://linkedin.com/company/genesys', }, { name: 'SAP Concur', domainName: 'concur.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 7305, linkedinLinkUrl: 'https://linkedin.com/company/sapconcur', }, { name: 'Square', domainName: 'squareup.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 7233, linkedinLinkUrl: 'https://linkedin.com/company/joinsquare', }, { name: 'Snap Inc.', domainName: 'snap.com', - address: 'Santa Monica', + addressAddressCity: 'Santa Monica', employees: 7219, linkedinLinkUrl: 'https://linkedin.com/company/snap-inc-co', }, { name: 'MathWorks', domainName: 'mathworks.com', - address: 'Natick', + addressAddressCity: 'Natick', employees: 7188, linkedinLinkUrl: 'https://linkedin.com/company/the-mathworks_2', }, { name: 'PTC', domainName: 'ptc.co', - address: 'Boston', + addressAddressCity: 'Boston', employees: 7119, linkedinLinkUrl: 'https://linkedin.com/company/ptcinc', }, { name: 'Ansys', domainName: 'ansys.com', - address: 'Canonsburg', + addressAddressCity: 'Canonsburg', employees: 7112, linkedinLinkUrl: 'https://linkedin.com/company/ansys-inc', }, { name: 'Aricent', domainName: 'altran.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 7016, linkedinLinkUrl: 'https://linkedin.com/company/aricent', }, { name: 'Databricks', domainName: 'databricks.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 6927, linkedinLinkUrl: 'https://linkedin.com/company/databricks', }, { name: 'Shipt', domainName: 'shipt.com', - address: 'Birmingham', + addressAddressCity: 'Birmingham', employees: 6902, linkedinLinkUrl: 'https://linkedin.com/company/shipt', }, { name: 'CSG', domainName: 'csgi.com', - address: 'Englewood', + addressAddressCity: 'Englewood', employees: 6849, linkedinLinkUrl: 'https://linkedin.com/company/csg-', }, { name: 'Twilio', domainName: 'twilio.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 6721, linkedinLinkUrl: 'https://linkedin.com/company/twilio-inc-', }, { name: 'Veritas Technologies LLC', domainName: 'veritas.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 6718, linkedinLinkUrl: 'https://linkedin.com/company/veritas-technologies-llc', }, { name: 'Citrix', domainName: 'citrix.com', - address: 'Fort Lauderdale', + addressAddressCity: 'Fort Lauderdale', employees: 6528, linkedinLinkUrl: 'https://linkedin.com/company/citrix', }, { name: 'Tyler Technologies', domainName: 'tylertech.com', - address: 'Plano', + addressAddressCity: 'Plano', employees: 6496, linkedinLinkUrl: 'https://linkedin.com/company/tyler-technologies', }, { name: 'Esri', domainName: 'esri.com', - address: 'Redlands', + addressAddressCity: 'Redlands', employees: 6463, linkedinLinkUrl: 'https://linkedin.com/company/esri', }, { name: 'Paycom', domainName: 'paycom.com', - address: 'Oklahoma City', + addressAddressCity: 'Oklahoma City', employees: 6378, linkedinLinkUrl: 'https://linkedin.com/company/paycom', }, { name: 'Roblox', domainName: 'roblox.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 6297, linkedinLinkUrl: 'https://linkedin.com/company/roblox', }, { name: 'Zendesk', domainName: 'zendesk.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 6255, linkedinLinkUrl: 'https://linkedin.com/company/zendesk', }, { name: 'Newfold Digital', domainName: 'newfold.com', - address: 'Jacksonville', + addressAddressCity: 'Jacksonville', employees: 6213, linkedinLinkUrl: 'https://linkedin.com/company/newfold', }, { name: 'Informatica', domainName: 'informatica.com', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 5850, linkedinLinkUrl: 'https://linkedin.com/company/informatica', }, { name: 'Caf\u00e9', domainName: 'at.cafe', - address: 'New York', + addressAddressCity: 'New York', employees: 5795, linkedinLinkUrl: 'https://linkedin.com/company/get-cafe', }, { name: 'Mavenir', domainName: 'mavenir.com', - address: 'Richardson', + addressAddressCity: 'Richardson', employees: 5763, linkedinLinkUrl: 'https://linkedin.com/company/mavenir', }, { name: 'Allscripts', domainName: 'allscripts.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 5719, linkedinLinkUrl: 'https://linkedin.com/company/allscripts', }, { name: 'Yardi', domainName: 'yardi.com', - address: 'Goleta', + addressAddressCity: 'Goleta', employees: 5583, linkedinLinkUrl: 'https://linkedin.com/company/yardi', }, { name: 'Datadog', domainName: 'datadoghq.com', - address: 'New York', + addressAddressCity: 'New York', employees: 5470, linkedinLinkUrl: 'https://linkedin.com/company/datadog', }, { name: 'Epicor', domainName: 'epicor.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 5310, linkedinLinkUrl: 'https://linkedin.com/company/epicor-software-corp', }, { name: 'Hexagon Asset Lifecycle Intelligence', domainName: 'hexagonppm.com', - address: 'Madison', + addressAddressCity: 'Madison', employees: 5262, linkedinLinkUrl: 'https://linkedin.com/company/hexagonassetlifecycleintelligence', @@ -535,105 +535,105 @@ export const companiesDemo = [ { name: 'Blue Yonder', domainName: 'blueyonder.com', - address: 'Scottsdale', + addressAddressCity: 'Scottsdale', employees: 5205, linkedinLinkUrl: 'https://linkedin.com/company/blueyonder', }, { name: 'MongoDB', domainName: 'mongodb.com', - address: 'New York', + addressAddressCity: 'New York', employees: 5182, linkedinLinkUrl: 'https://linkedin.com/company/mongodbinc', }, { name: 'uTest', domainName: 'utest.com', - address: 'Framingham', + addressAddressCity: 'Framingham', employees: 5125, linkedinLinkUrl: 'https://linkedin.com/company/utest', }, { name: 'Paylocity', domainName: 'paylocity.com', - address: 'Schaumburg', + addressAddressCity: 'Schaumburg', employees: 5095, linkedinLinkUrl: 'https://linkedin.com/company/paylocity', }, { name: 'IAC', domainName: 'iac.com', - address: 'New York', + addressAddressCity: 'New York', employees: 5040, linkedinLinkUrl: 'https://linkedin.com/company/iac', }, { name: 'Toast', domainName: 'toasttab.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 5008, linkedinLinkUrl: 'https://linkedin.com/company/toast-inc', }, { name: 'Bentley Systems', domainName: 'bentley.com', - address: 'Exton', + addressAddressCity: 'Exton', employees: 4862, linkedinLinkUrl: 'https://linkedin.com/company/bentley-systems', }, { name: 'Owner.com', domainName: 'owner.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 4677, linkedinLinkUrl: 'https://linkedin.com/company/profitboss', }, { name: 'eClinicalWorks', domainName: 'eclinicalworks.com', - address: 'Westborough', + addressAddressCity: 'Westborough', employees: 4661, linkedinLinkUrl: 'https://linkedin.com/company/eclinicalworks', }, { name: 'Altimetrik', domainName: 'altimetrik.com', - address: 'Southfield', + addressAddressCity: 'Southfield', employees: 4629, linkedinLinkUrl: 'https://linkedin.com/company/altimetrik', }, { name: 'CA Technologies', domainName: 'ca.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 4616, linkedinLinkUrl: 'https://linkedin.com/company/ca-technologies', }, { name: 'Dynatrace', domainName: 'dynatrace.com', - address: 'Waltham', + addressAddressCity: 'Waltham', employees: 4502, linkedinLinkUrl: 'https://linkedin.com/company/dynatrace', }, { name: 'Sprinklr', domainName: 'sprinklr.com', - address: 'New York', + addressAddressCity: 'New York', employees: 4495, linkedinLinkUrl: 'https://linkedin.com/company/sprinklr', }, { name: 'UiPath', domainName: 'uipath.com', - address: 'New York', + addressAddressCity: 'New York', employees: 4484, linkedinLinkUrl: 'https://linkedin.com/company/uipath', }, { name: 'The Reynolds and Reynolds Company', domainName: 'reyrey.com', - address: 'Dayton', + addressAddressCity: 'Dayton', employees: 4473, linkedinLinkUrl: 'https://linkedin.com/company/the-reynolds-and-reynolds-company', @@ -641,679 +641,679 @@ export const companiesDemo = [ { name: 'Stealth', domainName: 'stealthstartup.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 4472, linkedinLinkUrl: 'https://linkedin.com/company/stealthstartup', }, { name: 'WEX', domainName: 'wexinc.com', - address: 'Portland', + addressAddressCity: 'Portland', employees: 4377, linkedinLinkUrl: 'https://linkedin.com/company/wexinc', }, { name: 'HighRadius', domainName: 'highradius.com', - address: 'Houston', + addressAddressCity: 'Houston', employees: 4316, linkedinLinkUrl: 'https://linkedin.com/company/highradius', }, { name: 'Avalara', domainName: 'avalara.com', - address: 'Seattle', + addressAddressCity: 'Seattle', employees: 4311, linkedinLinkUrl: 'https://linkedin.com/company/avalara', }, { name: 'Manhattan Associates', domainName: 'manh.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 4236, linkedinLinkUrl: 'https://linkedin.com/company/manhattan-associates', }, { name: 'Aspen Technology', domainName: 'aspentech.com', - address: 'Bedford', + addressAddressCity: 'Bedford', employees: 4194, linkedinLinkUrl: 'https://linkedin.com/company/aspen-technology', }, { name: 'Hyland', domainName: 'hyland.com', - address: 'Westlake', + addressAddressCity: 'Westlake', employees: 4166, linkedinLinkUrl: 'https://linkedin.com/company/hyland-software', }, { name: 'Palantir Technologies', domainName: 'palantir.com', - address: 'Denver', + addressAddressCity: 'Denver', employees: 4104, linkedinLinkUrl: 'https://linkedin.com/company/palantir-technologies', }, { name: 'Market America, Inc.', domainName: 'marketamerica.com', - address: 'Greensboro', + addressAddressCity: 'Greensboro', employees: 4091, linkedinLinkUrl: 'https://linkedin.com/company/market-america-inc-', }, { name: 'Procore Technologies', domainName: 'procore.com', - address: 'Carpinteria', + addressAddressCity: 'Carpinteria', employees: 4010, linkedinLinkUrl: 'https://linkedin.com/company/procore-technologies', }, { name: 'ZoomInfo', domainName: 'zoominfo.com', - address: 'Vancouver', + addressAddressCity: 'Vancouver', employees: 3875, linkedinLinkUrl: 'https://linkedin.com/company/zoominfo', }, { name: 'TIBCO', domainName: 'tibco.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 3871, linkedinLinkUrl: 'https://linkedin.com/company/tibco', }, { name: 'GE Digital', domainName: 'ge.com', - address: 'San Ramon', + addressAddressCity: 'San Ramon', employees: 3849, linkedinLinkUrl: 'https://linkedin.com/company/ge-digital', }, { name: 'RMS', domainName: 'rms.com', - address: 'Newark', + addressAddressCity: 'Newark', employees: 3844, linkedinLinkUrl: 'https://linkedin.com/company/rms', }, { name: 'Tableau', domainName: 'tableau.com', - address: 'Seattle', + addressAddressCity: 'Seattle', employees: 3838, linkedinLinkUrl: 'https://linkedin.com/company/tableau-software', }, { name: 'Extreme Networks', domainName: 'extremenetworks.com', - address: 'Morrisville', + addressAddressCity: 'Morrisville', employees: 3799, linkedinLinkUrl: 'https://linkedin.com/company/extreme-networks', }, { name: 'Smartsheet', domainName: 'smartsheet.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 3798, linkedinLinkUrl: 'https://linkedin.com/company/smartsheet-com', }, { name: 'Quest Software', domainName: 'quest.com', - address: 'Aliso Viejo', + addressAddressCity: 'Aliso Viejo', employees: 3795, linkedinLinkUrl: 'https://linkedin.com/company/quest-software', }, { name: 'Motive', domainName: 'gomotive.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 3788, linkedinLinkUrl: 'https://linkedin.com/company/motive-inc', }, { name: 'Retired Life', domainName: 'swde.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 3774, linkedinLinkUrl: 'https://linkedin.com/company/retired-life', }, { name: 'Dropbox', domainName: 'dropbox.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 3751, linkedinLinkUrl: 'https://linkedin.com/company/dropbox', }, { name: 'Deltek', domainName: 'deltek.com', - address: 'Herndon', + addressAddressCity: 'Herndon', employees: 3727, linkedinLinkUrl: 'https://linkedin.com/company/deltek', }, { name: 'e2open', domainName: 'e2open.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 3694, linkedinLinkUrl: 'https://linkedin.com/company/e2open', }, { name: 'Altair', domainName: 'altair.com', - address: 'Troy', + addressAddressCity: 'Troy', employees: 3596, linkedinLinkUrl: 'https://linkedin.com/company/altair-engineering', }, { name: 'Gopuff', domainName: 'gopuff.com', - address: 'Philadelphia', + addressAddressCity: 'Philadelphia', employees: 3574, linkedinLinkUrl: 'https://linkedin.com/company/gopuff', }, { name: 'FICO', domainName: 'fico.com', - address: 'Bozeman', + addressAddressCity: 'Bozeman', employees: 3511, linkedinLinkUrl: 'https://linkedin.com/company/fico', }, { name: 'Elastic', domainName: 'elastic.co', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 3489, linkedinLinkUrl: 'https://linkedin.com/company/elastic-co', }, { name: 'Blackbaud', domainName: 'blackbaud.com', - address: 'Charleston', + addressAddressCity: 'Charleston', employees: 3478, linkedinLinkUrl: 'https://linkedin.com/company/blackbaud', }, { name: 'MicroStrategy', domainName: 'microstrategy.com', - address: 'Vienna', + addressAddressCity: 'Vienna', employees: 3469, linkedinLinkUrl: 'https://linkedin.com/company/microstrategy', }, { name: 'Discord', domainName: 'discord.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 3467, linkedinLinkUrl: 'https://linkedin.com/company/discord', }, { name: 'Inovalon', domainName: 'inovalon.com', - address: 'Bowie', + addressAddressCity: 'Bowie', employees: 3459, linkedinLinkUrl: 'https://linkedin.com/company/inovalon', }, { name: 'Progress', domainName: 'progress.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 3428, linkedinLinkUrl: 'https://linkedin.com/company/progress-software', }, { name: 'Rubrik', domainName: 'rbrk.co', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 3370, linkedinLinkUrl: 'https://linkedin.com/company/rubrik-inc', }, { name: 'Axtria - Ingenious Insights', domainName: 'axtria.com', - address: 'Berkeley Heights', + addressAddressCity: 'Berkeley Heights', employees: 3367, linkedinLinkUrl: 'https://linkedin.com/company/axtria', }, { name: 'Audible', domainName: 'audible.com', - address: 'Newark', + addressAddressCity: 'Newark', employees: 3192, linkedinLinkUrl: 'https://linkedin.com/company/audible', }, { name: 'Kaseya', domainName: 'kaseya.com', - address: 'Miami', + addressAddressCity: 'Miami', employees: 3191, linkedinLinkUrl: 'https://linkedin.com/company/kaseya', }, { name: 'MRI Software', domainName: 'mrisoftware.com', - address: 'Solon', + addressAddressCity: 'Solon', employees: 3107, linkedinLinkUrl: 'https://linkedin.com/company/mri-software-llc', }, { name: 'CyberArk', domainName: 'cyberark.com', - address: 'Newton Center', + addressAddressCity: 'Newton Center', employees: 3099, linkedinLinkUrl: 'https://linkedin.com/company/cyber-ark-software', }, { name: 'Cornerstone OnDemand', domainName: 'cornerstoneondemand.com', - address: 'Santa Monica', + addressAddressCity: 'Santa Monica', employees: 3089, linkedinLinkUrl: 'https://linkedin.com/company/cornerstone-ondemand', }, { name: 'Reddit, Inc.', domainName: 'redditinc.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 3061, linkedinLinkUrl: 'https://linkedin.com/company/reddit-com', }, { name: 'Ivanti', domainName: 'ivanti.com', - address: 'South Jordan', + addressAddressCity: 'South Jordan', employees: 3056, linkedinLinkUrl: 'https://linkedin.com/company/ivanti', }, { name: 'Cloudera', domainName: 'cloudera.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 3007, linkedinLinkUrl: 'https://linkedin.com/company/cloudera', }, { name: 'Medidata Solutions', domainName: 'medidata.com', - address: 'New York', + addressAddressCity: 'New York', employees: 3001, linkedinLinkUrl: 'https://linkedin.com/company/medidata-solutions', }, { name: 'Commvault', domainName: 'commvault.com', - address: 'Eatontown', + addressAddressCity: 'Eatontown', employees: 2974, linkedinLinkUrl: 'https://linkedin.com/company/commvault', }, { name: 'ConnectWise', domainName: 'connectwise.com', - address: 'Tampa', + addressAddressCity: 'Tampa', employees: 2937, linkedinLinkUrl: 'https://linkedin.com/company/connectwise', }, { name: 'BILL', domainName: 'bill.com', - address: 'Alviso', + addressAddressCity: 'Alviso', employees: 2932, linkedinLinkUrl: 'https://linkedin.com/company/bill', }, { name: 'Alteryx', domainName: 'alteryx.com', - address: 'Irvine', + addressAddressCity: 'Irvine', employees: 2916, linkedinLinkUrl: 'https://linkedin.com/company/alteryx', }, { name: 'MNC Software', domainName: 'mncsoftware.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 2912, linkedinLinkUrl: 'https://linkedin.com/company/mnc-software', }, { name: 'Celonis', domainName: 'celonis.com', - address: 'New York', + addressAddressCity: 'New York', employees: 2906, linkedinLinkUrl: 'https://linkedin.com/company/celonis', }, { name: 'Attachmate', domainName: 'microfocus.com', - address: 'Seattle', + addressAddressCity: 'Seattle', employees: 2889, linkedinLinkUrl: 'https://linkedin.com/company/attachmate', }, { name: 'NETSCOUT', domainName: 'netscout.com', - address: 'Westford', + addressAddressCity: 'Westford', employees: 2853, linkedinLinkUrl: 'https://linkedin.com/company/netscout', }, { name: 'Confluent', domainName: 'confluent.io', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 2844, linkedinLinkUrl: 'https://linkedin.com/company/confluent', }, { name: 'Samsara', domainName: 'samsara.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2824, linkedinLinkUrl: 'https://linkedin.com/company/samsara', }, { name: 'Chetu, Inc.', domainName: 'chetu.com', - address: 'Fort Lauderdale', + addressAddressCity: 'Fort Lauderdale', employees: 2809, linkedinLinkUrl: 'https://linkedin.com/company/chetu-inc-', }, { name: 'Kronos Incorporated', domainName: 'ukg.com', - address: 'Lowell', + addressAddressCity: 'Lowell', employees: 2808, linkedinLinkUrl: 'https://linkedin.com/company/kronos', }, { name: 'Qlik', domainName: 'qlik.com', - address: 'King Of Prussia', + addressAddressCity: 'King Of Prussia', employees: 2779, linkedinLinkUrl: 'https://linkedin.com/company/qlik', }, { name: 'Vertafore', domainName: 'vertafore.com', - address: 'Denver', + addressAddressCity: 'Denver', employees: 2768, linkedinLinkUrl: 'https://linkedin.com/company/vertafore', }, { name: 'Asana', domainName: 'asana.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2753, linkedinLinkUrl: 'https://linkedin.com/company/asana', }, { name: 'Jamf', domainName: 'jamf.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 2721, linkedinLinkUrl: 'https://linkedin.com/company/jamf-software', }, { name: 'Paycor', domainName: 'paycor.com', - address: 'Cincinnati', + addressAddressCity: 'Cincinnati', employees: 2719, linkedinLinkUrl: 'https://linkedin.com/company/paycor', }, { name: 'Hudl', domainName: 'hudl.com', - address: 'Lincoln', + addressAddressCity: 'Lincoln', employees: 2709, linkedinLinkUrl: 'https://linkedin.com/company/hudl', }, { name: 'Precisely', domainName: 'precisely.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 2662, linkedinLinkUrl: 'https://linkedin.com/company/preciselydata', }, { name: 'New Relic', domainName: 'newrelic.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2636, linkedinLinkUrl: 'https://linkedin.com/company/new-relic-inc-', }, { name: 'Aptean', domainName: 'aptean.com', - address: 'Alpharetta', + addressAddressCity: 'Alpharetta', employees: 2617, linkedinLinkUrl: 'https://linkedin.com/company/aptean', }, { name: 'o9 Solutions, Inc.', domainName: 'o9solutions.com', - address: 'Dallas', + addressAddressCity: 'Dallas', employees: 2612, linkedinLinkUrl: 'https://linkedin.com/company/o9solutions', }, { name: 'SpotOn', domainName: 'spoton.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2608, linkedinLinkUrl: 'https://linkedin.com/company/spoton', }, { name: 'Automation Anywhere', domainName: 'automationanywhere.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 2588, linkedinLinkUrl: 'https://linkedin.com/company/automation-anywhere', }, { name: 'Tekion Corp', domainName: 'tekion.com', - address: 'Pleasanton', + addressAddressCity: 'Pleasanton', employees: 2579, linkedinLinkUrl: 'https://linkedin.com/company/tekion', }, { name: 'Aurora', domainName: 'aurora.tech', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 2557, linkedinLinkUrl: 'https://linkedin.com/company/aurora-inc.', }, { name: 'SolarWinds', domainName: 'solarwinds.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 2529, linkedinLinkUrl: 'https://linkedin.com/company/solarwinds', }, { name: 'GoTo', domainName: 'goto.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 2505, linkedinLinkUrl: 'https://linkedin.com/company/goto', }, { name: 'PROS', domainName: 'pros.com', - address: 'Houston', + addressAddressCity: 'Houston', employees: 2479, linkedinLinkUrl: 'https://linkedin.com/company/pros', }, { name: 'Miro', domainName: 'miro.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2445, linkedinLinkUrl: 'https://linkedin.com/company/mirohq', }, { name: 'Kofax', domainName: 'kofax.com', - address: 'Irvine', + addressAddressCity: 'Irvine', employees: 2442, linkedinLinkUrl: 'https://linkedin.com/company/kofax', }, { name: 'Accolite Digital', domainName: 'accolite.com', - address: 'Addison', + addressAddressCity: 'Addison', employees: 2438, linkedinLinkUrl: 'https://linkedin.com/company/accolitedigital', }, { name: 'HashiCorp', domainName: 'hashicorp.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2436, linkedinLinkUrl: 'https://linkedin.com/company/hashicorp', }, { name: 'Pluralsight', domainName: 'pluralsight.com', - address: 'Draper', + addressAddressCity: 'Draper', employees: 2433, linkedinLinkUrl: 'https://linkedin.com/company/pluralsight', }, { name: 'Bottomline Technologies', domainName: 'bottomline.com', - address: 'Portsmouth', + addressAddressCity: 'Portsmouth', employees: 2407, linkedinLinkUrl: 'https://linkedin.com/company/bottomline-technologies', }, { name: 'Anaplan', domainName: 'anaplan.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2401, linkedinLinkUrl: 'https://linkedin.com/company/anaplan', }, { name: 'OneTrust', domainName: 'onetrust.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 2383, linkedinLinkUrl: 'https://linkedin.com/company/onetrust', }, { name: 'Medallia', domainName: 'medallia.com', - address: 'Pleasanton', + addressAddressCity: 'Pleasanton', employees: 2381, linkedinLinkUrl: 'https://linkedin.com/company/medallia-inc.', }, { name: 'SailPoint', domainName: 'sailpoint.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 2366, linkedinLinkUrl: 'https://linkedin.com/company/sailpoint-technologies', }, { name: 'Appian Corporation', domainName: 'appian.com', - address: 'Mc Lean', + addressAddressCity: 'Mc Lean', employees: 2345, linkedinLinkUrl: 'https://linkedin.com/company/appian-corporation', }, { name: 'Dealertrack', domainName: 'dealertrack.com', - address: 'New Hyde Park', + addressAddressCity: 'New Hyde Park', employees: 2335, linkedinLinkUrl: 'https://linkedin.com/company/dealertrack', }, { name: 'impact.com', domainName: 'impact.com', - address: 'Santa Barbara', + addressAddressCity: 'Santa Barbara', employees: 2327, linkedinLinkUrl: 'https://linkedin.com/company/impactdotcom', }, { name: 'Inhabit\u00ae', domainName: 'inhabitiq.com', - address: 'Knoxville', + addressAddressCity: 'Knoxville', employees: 2286, linkedinLinkUrl: 'https://linkedin.com/company/inhabit-iq', }, { name: 'SymphonyAI', domainName: 'symphonyai.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 2282, linkedinLinkUrl: 'https://linkedin.com/company/symphonyai', }, { name: 'CCC Intelligent Solutions', domainName: 'cccis.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 2282, linkedinLinkUrl: 'https://linkedin.com/company/ccc-intelligent-solutions', }, { name: 'Toshiba Global Commerce Solutions', domainName: 'toshiba.com', - address: 'Durham', + addressAddressCity: 'Durham', employees: 2281, linkedinLinkUrl: 'https://linkedin.com/company/toshibacommerce', }, { name: 'Vertex Inc.', domainName: 'vertexinc.com', - address: 'King Of Prussia', + addressAddressCity: 'King Of Prussia', employees: 2265, linkedinLinkUrl: 'https://linkedin.com/company/vertex-inc.', }, { name: 'PRO Unlimited', domainName: 'magnitglobal.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 2264, linkedinLinkUrl: 'https://linkedin.com/company/prounlimited', }, { name: 'Five9', domainName: 'five9.com', - address: 'San Ramon', + addressAddressCity: 'San Ramon', employees: 2253, linkedinLinkUrl: 'https://linkedin.com/company/five9', }, { name: 'Cohesity', domainName: 'cohesity.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 2252, linkedinLinkUrl: 'https://linkedin.com/company/cohesity', }, { name: 'Wind River', domainName: 'windriver.com', - address: 'Alameda', + addressAddressCity: 'Alameda', employees: 2244, linkedinLinkUrl: 'https://linkedin.com/company/wind-river', }, { name: 'Icertis', domainName: 'icertis.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 2233, linkedinLinkUrl: 'https://linkedin.com/company/icertis', }, { name: 'Navan', domainName: 'navan.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 2221, linkedinLinkUrl: 'https://linkedin.com/company/navan', }, { name: 'Diligent', domainName: 'diligent.com', - address: 'New York', + addressAddressCity: 'New York', employees: 2215, linkedinLinkUrl: 'https://linkedin.com/company/diligent-board-member-services', @@ -1321,35 +1321,35 @@ export const companiesDemo = [ { name: 'Applied Systems', domainName: 'appliedsystems.com', - address: 'University Park', + addressAddressCity: 'University Park', employees: 2198, linkedinLinkUrl: 'https://linkedin.com/company/applied-systems', }, { name: 'Forcepoint', domainName: 'forcepoint.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 2196, linkedinLinkUrl: 'https://linkedin.com/company/forcepoint', }, { name: 'Compuware', domainName: 'bmc.com', - address: 'Detroit', + addressAddressCity: 'Detroit', employees: 2183, linkedinLinkUrl: 'https://linkedin.com/company/compuware', }, { name: 'Netsmart', domainName: 'ntst.com', - address: 'Leawood', + addressAddressCity: 'Leawood', employees: 2177, linkedinLinkUrl: 'https://linkedin.com/company/netsmart', }, { name: 'The Apache Software Foundation', domainName: 'apache.org', - address: 'Wilmington', + addressAddressCity: 'Wilmington', employees: 2177, linkedinLinkUrl: 'https://linkedin.com/company/the-apache-software-foundation', @@ -1357,938 +1357,938 @@ export const companiesDemo = [ { name: 'ArisGlobal', domainName: 'arisglobal.com', - address: 'Miami', + addressAddressCity: 'Miami', employees: 2168, linkedinLinkUrl: 'https://linkedin.com/company/aris-global', }, { name: 'WORKING BY MY SELF', domainName: 'fcutechnologies.com', - address: 'Fort Lauderdale', + addressAddressCity: 'Fort Lauderdale', employees: 2148, linkedinLinkUrl: 'https://linkedin.com/company/working-by-my-self', }, { name: 'Varonis', domainName: 'varonis.com', - address: 'New York', + addressAddressCity: 'New York', employees: 2140, linkedinLinkUrl: 'https://linkedin.com/company/varonis', }, { name: 'Fever', domainName: 'feverup.com', - address: 'New York', + addressAddressCity: 'New York', employees: 2125, linkedinLinkUrl: 'https://linkedin.com/company/fever-up', }, { name: 'Agilysys', domainName: 'agilysys.com', - address: 'Alpharetta', + addressAddressCity: 'Alpharetta', employees: 2081, linkedinLinkUrl: 'https://linkedin.com/company/agilysys', }, { name: 'OutSystems', domainName: 'outsystems.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 2057, linkedinLinkUrl: 'https://linkedin.com/company/outsystems', }, { name: 'Entrata', domainName: 'entrata.com', - address: 'Lehi', + addressAddressCity: 'Lehi', employees: 2045, linkedinLinkUrl: 'https://linkedin.com/company/entratasoftware', }, { name: 'Verkada', domainName: 'verkada.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 2044, linkedinLinkUrl: 'https://linkedin.com/company/verkada', }, { name: 'Majesco', domainName: 'majesco.com', - address: 'Morristown', + addressAddressCity: 'Morristown', employees: 2021, linkedinLinkUrl: 'https://linkedin.com/company/majesco', }, { name: 'Boomi', domainName: 'boomi.com', - address: 'Wayne', + addressAddressCity: 'Wayne', employees: 2009, linkedinLinkUrl: 'https://linkedin.com/company/boomi-inc', }, { name: 'PDI Technologies', domainName: 'pditechnologies.com', - address: 'Alpharetta', + addressAddressCity: 'Alpharetta', employees: 2005, linkedinLinkUrl: 'https://linkedin.com/company/pdi-technologies', }, { name: 'ServiceTitan', domainName: 'servicetitan.com', - address: 'Glendale', + addressAddressCity: 'Glendale', employees: 1997, linkedinLinkUrl: 'https://linkedin.com/company/servicetitan', }, { name: 'Sitecore', domainName: 'sitecore.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1943, linkedinLinkUrl: 'https://linkedin.com/company/sitecore', }, { name: 'SAP SuccessFactors', domainName: 'sap.com', - address: 'South San Francisco', + addressAddressCity: 'South San Francisco', employees: 1941, linkedinLinkUrl: 'https://linkedin.com/company/successfactors', }, { name: 'Postman', domainName: 'postman.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1928, linkedinLinkUrl: 'https://linkedin.com/company/postman-platform', }, { name: 'Scale AI', domainName: 'scale.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1906, linkedinLinkUrl: 'https://linkedin.com/company/scaleai', }, { name: 'Duck Creek Technologies', domainName: 'duckcreek.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 1894, linkedinLinkUrl: 'https://linkedin.com/company/duck-creek-technologies', }, { name: 'MICROS Systems Inc', domainName: 'oracle.com', - address: 'Columbia', + addressAddressCity: 'Columbia', employees: 1882, linkedinLinkUrl: 'https://linkedin.com/company/micros-systems-inc', }, { name: 'Riverbed Technology', domainName: 'riverbed.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1874, linkedinLinkUrl: 'https://linkedin.com/company/riverbed-technology', }, { name: 'Fast Enterprises, LLC', domainName: 'fastenterprises.com', - address: 'Englewood', + addressAddressCity: 'Englewood', employees: 1833, linkedinLinkUrl: 'https://linkedin.com/company/fast-enterprises', }, { name: 'Alvaria, Inc.', domainName: 'alvaria.com', - address: 'Westford', + addressAddressCity: 'Westford', employees: 1830, linkedinLinkUrl: 'https://linkedin.com/company/alvaria-inc', }, { name: 'BlackLine', domainName: 'blackline.com', - address: 'Woodland Hills', + addressAddressCity: 'Woodland Hills', employees: 1826, linkedinLinkUrl: 'https://linkedin.com/company/blackline', }, { name: '3Pillar Global', domainName: '3pillarglobal.com', - address: 'Fairfax', + addressAddressCity: 'Fairfax', employees: 1824, linkedinLinkUrl: 'https://linkedin.com/company/3pillar-global', }, { name: 'Saama', domainName: 'saama.com', - address: 'Campbell', + addressAddressCity: 'Campbell', employees: 1809, linkedinLinkUrl: 'https://linkedin.com/company/saama-technologies', }, { name: 'Ancestry', domainName: 'ancestry.com', - address: 'Lehi', + addressAddressCity: 'Lehi', employees: 1794, linkedinLinkUrl: 'https://linkedin.com/company/ancestry.com', }, { name: 'insightsoftware', domainName: 'insightsoftware.com', - address: 'Raleigh', + addressAddressCity: 'Raleigh', employees: 1788, linkedinLinkUrl: 'https://linkedin.com/company/outcomes-by-insightsoftware', }, { name: 'Ebix', domainName: 'ebix.com', - address: 'Duluth', + addressAddressCity: 'Duluth', employees: 1757, linkedinLinkUrl: 'https://linkedin.com/company/ebix', }, { name: 'Zuora', domainName: 'zuora.com', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 1746, linkedinLinkUrl: 'https://linkedin.com/company/zuora', }, { name: 'IntelyCare', domainName: 'intelycare.com', - address: 'Quincy', + addressAddressCity: 'Quincy', employees: 1731, linkedinLinkUrl: 'https://linkedin.com/company/intelycare', }, { name: 'Axway', domainName: 'axway.com', - address: 'Scottsdale', + addressAddressCity: 'Scottsdale', employees: 1731, linkedinLinkUrl: 'https://linkedin.com/company/axway', }, { name: 'Community Brands', domainName: 'communitybrands.com', - address: 'Saint Petersburg', + addressAddressCity: 'Saint Petersburg', employees: 1731, linkedinLinkUrl: 'https://linkedin.com/company/communitybrands', }, { name: 'InterSystems', domainName: 'intersystems.com', - address: 'Cambridge', + addressAddressCity: 'Cambridge', employees: 1730, linkedinLinkUrl: 'https://linkedin.com/company/intersystems', }, { name: 'Mozilla', domainName: 'mozilla.org', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1721, linkedinLinkUrl: 'https://linkedin.com/company/mozilla-corporation', }, { name: 'Semrush', domainName: 'semrush.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 1706, linkedinLinkUrl: 'https://linkedin.com/company/semrush', }, { name: 'Avid', domainName: 'avid.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 1705, linkedinLinkUrl: 'https://linkedin.com/company/avid-technology', }, { name: 'Conga', domainName: 'conga.com', - address: 'Broomfield', + addressAddressCity: 'Broomfield', employees: 1695, linkedinLinkUrl: 'https://linkedin.com/company/conga', }, { name: 'InfoBeans', domainName: 'infobeans.com', - address: 'Danville', + addressAddressCity: 'Danville', employees: 1691, linkedinLinkUrl: 'https://linkedin.com/company/infobeans', }, { name: 'AppFolio, Inc.', domainName: 'appfolioinc.com', - address: 'Goleta', + addressAddressCity: 'Goleta', employees: 1688, linkedinLinkUrl: 'https://linkedin.com/company/appfolio-inc', }, { name: 'Sovos', domainName: 'sovos.com', - address: 'Wilmington', + addressAddressCity: 'Wilmington', employees: 1684, linkedinLinkUrl: 'https://linkedin.com/company/sovos', }, { name: 'nCino, Inc.', domainName: 'ncino.com', - address: 'Wilmington', + addressAddressCity: 'Wilmington', employees: 1680, linkedinLinkUrl: 'https://linkedin.com/company/ncino-inc-', }, { name: 'Vistex', domainName: 'vistex.com', - address: 'Hoffman Estates', + addressAddressCity: 'Hoffman Estates', employees: 1677, linkedinLinkUrl: 'https://linkedin.com/company/vistex', }, { name: 'Taboola', domainName: 'taboola.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1677, linkedinLinkUrl: 'https://linkedin.com/company/taboola', }, { name: 'EverCommerce', domainName: 'evercommerce.com', - address: 'Denver', + addressAddressCity: 'Denver', employees: 1673, linkedinLinkUrl: 'https://linkedin.com/company/evercommerce', }, { name: 'Virgin Pulse', domainName: 'virginpulse.com', - address: 'Providence', + addressAddressCity: 'Providence', employees: 1666, linkedinLinkUrl: 'https://linkedin.com/company/virgin-pulse', }, { name: 'Houzz', domainName: 'houzz.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 1641, linkedinLinkUrl: 'https://linkedin.com/company/houzz', }, { name: 'AvidXchange, Inc.', domainName: 'avidxchange.com', - address: 'Charlotte', + addressAddressCity: 'Charlotte', employees: 1639, linkedinLinkUrl: 'https://linkedin.com/company/avidxchange-inc-', }, { name: 'Planview, Inc.', domainName: 'planview.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1634, linkedinLinkUrl: 'https://linkedin.com/company/planview', }, { name: 'HackerRank', domainName: 'hackerrank.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 1632, linkedinLinkUrl: 'https://linkedin.com/company/hackerrank', }, { name: 'Clearwater Analytics', domainName: 'clearwateranalytics.com', - address: 'Boise', + addressAddressCity: 'Boise', employees: 1615, linkedinLinkUrl: 'https://linkedin.com/company/clearwateranalytics', }, { name: 'Outreach', domainName: 'outreach.io', - address: 'Seattle', + addressAddressCity: 'Seattle', employees: 1612, linkedinLinkUrl: 'https://linkedin.com/company/outreach-saas', }, { name: 'Everbridge', domainName: 'everbridge.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 1607, linkedinLinkUrl: 'https://linkedin.com/company/everbridge', }, { name: 'Zycus', domainName: 'zycus.com', - address: 'Princeton', + addressAddressCity: 'Princeton', employees: 1604, linkedinLinkUrl: 'https://linkedin.com/company/zycus', }, { name: 'Bullhorn', domainName: 'bullhorn.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 1604, linkedinLinkUrl: 'https://linkedin.com/company/bullhorn', }, { name: 'LivePerson', domainName: 'liveperson.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1603, linkedinLinkUrl: 'https://linkedin.com/company/liveperson', }, { name: 'Relativity', domainName: 'relativity.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1601, linkedinLinkUrl: 'https://linkedin.com/company/relativityhq', }, { name: 'HealthEdge', domainName: 'healthedge.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 1600, linkedinLinkUrl: 'https://linkedin.com/company/healthedge', }, { name: 'QAD', domainName: 'qad.com', - address: 'Santa Barbara', + addressAddressCity: 'Santa Barbara', employees: 1598, linkedinLinkUrl: 'https://linkedin.com/company/qad', }, { name: 'Braze', domainName: 'braze.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1598, linkedinLinkUrl: 'https://linkedin.com/company/braze-', }, { name: 'Exadel', domainName: 'exadel.com', - address: 'Walnut Creek', + addressAddressCity: 'Walnut Creek', employees: 1592, linkedinLinkUrl: 'https://linkedin.com/company/exadel', }, { name: 'Phenom', domainName: 'phenom.com', - address: 'Ambler', + addressAddressCity: 'Ambler', employees: 1592, linkedinLinkUrl: 'https://linkedin.com/company/phenomtxm', }, { name: 'Bazaarvoice', domainName: 'bazaarvoice.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1587, linkedinLinkUrl: 'https://linkedin.com/company/bazaarvoice', }, { name: 'AppDynamics', domainName: 'appdynamics.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1553, linkedinLinkUrl: 'https://linkedin.com/company/appdynamics', }, { name: 'Mitchell International', domainName: 'mitchell.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 1548, linkedinLinkUrl: 'https://linkedin.com/company/mitchell-international', }, { name: 'Talkdesk', domainName: 'talkdesk.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1491, linkedinLinkUrl: 'https://linkedin.com/company/talkdesk', }, { name: 'Hughes Systique Corporation (HSC)', domainName: 'hsc.com', - address: 'Rockville', + addressAddressCity: 'Rockville', employees: 1481, linkedinLinkUrl: 'https://linkedin.com/company/hsc', }, { name: 'Avature', domainName: 'avature.net', - address: 'New York', + addressAddressCity: 'New York', employees: 1478, linkedinLinkUrl: 'https://linkedin.com/company/avature', }, { name: 'Anyone Home Inc', domainName: 'anyonehome.com', - address: 'Lake Forest', + addressAddressCity: 'Lake Forest', employees: 1476, linkedinLinkUrl: 'https://linkedin.com/company/anyone-home-inc', }, { name: 'Engineer.ai', domainName: 'builder.ai', - address: 'Venice', + addressAddressCity: 'Venice', employees: 1474, linkedinLinkUrl: 'https://linkedin.com/company/engineer.ai', }, { name: 'Apptio', domainName: 'apptio.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 1467, linkedinLinkUrl: 'https://linkedin.com/company/apptio', }, { name: 'KMS Technology, Inc.', domainName: 'kms-technology.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 1464, linkedinLinkUrl: 'https://linkedin.com/company/kms-technology', }, { name: 'JFrog', domainName: 'jfrog.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 1459, linkedinLinkUrl: 'https://linkedin.com/company/jfrog-ltd', }, { name: 'ASG Technologies', domainName: 'asg.com', - address: 'Naples', + addressAddressCity: 'Naples', employees: 1459, linkedinLinkUrl: 'https://linkedin.com/company/asg', }, { name: 'Seismic', domainName: 'seismic.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 1457, linkedinLinkUrl: 'https://linkedin.com/company/seismic', }, { name: 'ModMed', domainName: 'modmed.com', - address: 'Boca Raton', + addressAddressCity: 'Boca Raton', employees: 1452, linkedinLinkUrl: 'https://linkedin.com/company/modernizing-medicine', }, { name: 'ACV Auctions', domainName: 'acvauctions.com', - address: 'Buffalo', + addressAddressCity: 'Buffalo', employees: 1450, linkedinLinkUrl: 'https://linkedin.com/company/acv-auctions', }, { name: 'Cerence Inc.', domainName: 'cerence.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 1448, linkedinLinkUrl: 'https://linkedin.com/company/cerence', }, { name: 'Via', domainName: 'ridewithvia.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1446, linkedinLinkUrl: 'https://linkedin.com/company/ridewithvia', }, { name: 'Kingsoft', domainName: 'ksosoft.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 1445, linkedinLinkUrl: 'https://linkedin.com/company/kingsoft', }, { name: 'Model N', domainName: 'modeln.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 1445, linkedinLinkUrl: 'https://linkedin.com/company/modeln', }, { name: 'ThoughtSpot', domainName: 'thoughtspot.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 1436, linkedinLinkUrl: 'https://linkedin.com/company/thoughtspot', }, { name: 'SSS', domainName: 'getebs.com', - address: 'Littleton', + addressAddressCity: 'Littleton', employees: 1431, linkedinLinkUrl: 'https://linkedin.com/company/employee-based-software', }, { name: 'BeyondTrust', domainName: 'beyondtrust.com', - address: 'Duluth', + addressAddressCity: 'Duluth', employees: 1428, linkedinLinkUrl: 'https://linkedin.com/company/beyondtrust', }, { name: 'MetricStream', domainName: 'metricstream.com', - address: 'Alviso', + addressAddressCity: 'Alviso', employees: 1426, linkedinLinkUrl: 'https://linkedin.com/company/metricstream', }, { name: 'LogMeIn', domainName: 'logmeininc.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 1425, linkedinLinkUrl: 'https://linkedin.com/company/logmein', }, { name: 'Khoros', domainName: 'khoros.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1424, linkedinLinkUrl: 'https://linkedin.com/company/khoros', }, { name: 'Sprout Social, Inc.', domainName: 'sproutsocial.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1416, linkedinLinkUrl: 'https://linkedin.com/company/sprout-social-inc-', }, { name: 'Odessa', domainName: 'odessainc.com', - address: 'Philadelphia', + addressAddressCity: 'Philadelphia', employees: 1415, linkedinLinkUrl: 'https://linkedin.com/company/odessa-inc-', }, { name: 'Enverus', domainName: 'enverus.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1404, linkedinLinkUrl: 'https://linkedin.com/company/enverus-energy', }, { name: 'AvePoint', domainName: 'avepoint.com', - address: 'Jersey City', + addressAddressCity: 'Jersey City', employees: 1404, linkedinLinkUrl: 'https://linkedin.com/company/avepoint', }, { name: 'Gong', domainName: 'gong.io', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1398, linkedinLinkUrl: 'https://linkedin.com/company/gong-io', }, { name: 'Syncfusion', domainName: 'syncfusion.com', - address: 'Morrisville', + addressAddressCity: 'Morrisville', employees: 1397, linkedinLinkUrl: 'https://linkedin.com/company/syncfusion', }, { name: 'Ping Identity', domainName: 'pingidentity.com', - address: 'Denver', + addressAddressCity: 'Denver', employees: 1388, linkedinLinkUrl: 'https://linkedin.com/company/ping-identity', }, { name: 'WellSky', domainName: 'wellsky.com', - address: 'Overland Park', + addressAddressCity: 'Overland Park', employees: 1387, linkedinLinkUrl: 'https://linkedin.com/company/wellsky', }, { name: 'Tricentis', domainName: 'tricentis.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1383, linkedinLinkUrl: 'https://linkedin.com/company/tricentis', }, { name: 'Taskrabbit', domainName: 'taskrabbit.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1383, linkedinLinkUrl: 'https://linkedin.com/company/taskrabbit', }, { name: 'Syniti', domainName: 'syniti.com', - address: 'Needham Heights', + addressAddressCity: 'Needham Heights', employees: 1372, linkedinLinkUrl: 'https://linkedin.com/company/synitidata', }, { name: 'BigCommerce', domainName: 'bigcommerce.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1360, linkedinLinkUrl: 'https://linkedin.com/company/bigcommerce', }, { name: 'OEC', domainName: 'oeconnection.com', - address: 'Richfield', + addressAddressCity: 'Richfield', employees: 1357, linkedinLinkUrl: 'https://linkedin.com/company/oeconnection', }, { name: 'Calsoft', domainName: 'calsoftinc.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 1357, linkedinLinkUrl: 'https://linkedin.com/company/calsoft', }, { name: 'Taller', domainName: 'tallertechnologies.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1351, linkedinLinkUrl: 'https://linkedin.com/company/taller-technologies', }, { name: 'Planet', domainName: 'planet.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1348, linkedinLinkUrl: 'https://linkedin.com/company/planet-labs', }, { name: '6sense', domainName: '6sense.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1346, linkedinLinkUrl: 'https://linkedin.com/company/6sense', }, { name: 'Vitech Systems Group', domainName: 'vitechinc.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1345, linkedinLinkUrl: 'https://linkedin.com/company/vitech-systems-group', }, { name: 'Smarsh', domainName: 'smarsh.com', - address: 'Portland', + addressAddressCity: 'Portland', employees: 1344, linkedinLinkUrl: 'https://linkedin.com/company/smarsh', }, { name: 'NICE Actimize', domainName: 'niceactimize.com', - address: 'Hoboken', + addressAddressCity: 'Hoboken', employees: 1343, linkedinLinkUrl: 'https://linkedin.com/company/actimize', }, { name: 'Dataiku', domainName: 'dataiku.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1340, linkedinLinkUrl: 'https://linkedin.com/company/dataiku', }, { name: 'Liferay', domainName: 'liferay.com', - address: 'Diamond Bar', + addressAddressCity: 'Diamond Bar', employees: 1329, linkedinLinkUrl: 'https://linkedin.com/company/liferay-inc-', }, { name: 'Gainsight', domainName: 'gainsight.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1328, linkedinLinkUrl: 'https://linkedin.com/company/gainsight', }, { name: 'Infotech', domainName: 'infotechinc.com', - address: 'Gainesville', + addressAddressCity: 'Gainesville', employees: 1322, linkedinLinkUrl: 'https://linkedin.com/company/infotech-inc', }, { name: 'JAGGAER', domainName: 'jaggaer.com', - address: 'Morrisville', + addressAddressCity: 'Morrisville', employees: 1317, linkedinLinkUrl: 'https://linkedin.com/company/jaggaer', }, { name: 'Checkr, Inc.', domainName: 'checkr.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1304, linkedinLinkUrl: 'https://linkedin.com/company/checkr-com', }, { name: 'CARFAX', domainName: 'carfax.com', - address: 'Centreville', + addressAddressCity: 'Centreville', employees: 1296, linkedinLinkUrl: 'https://linkedin.com/company/carfax', }, { name: 'Lucid Software', domainName: 'lucid.co', - address: 'South Jordan', + addressAddressCity: 'South Jordan', employees: 1295, linkedinLinkUrl: 'https://linkedin.com/company/lucidsoftware', }, { name: 'Domo', domainName: 'domo.com', - address: 'American Fork', + addressAddressCity: 'American Fork', employees: 1293, linkedinLinkUrl: 'https://linkedin.com/company/domotalk', }, { name: 'Podium', domainName: 'podium.com', - address: 'Lehi', + addressAddressCity: 'Lehi', employees: 1292, linkedinLinkUrl: 'https://linkedin.com/company/podium', }, { name: 'Mendix', domainName: 'mendix.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 1290, linkedinLinkUrl: 'https://linkedin.com/company/mendix', }, { name: 'EDB', domainName: 'edbpostgres.com', - address: 'Bedford', + addressAddressCity: 'Bedford', employees: 1289, linkedinLinkUrl: 'https://linkedin.com/company/edbpostgres', }, { name: 'OneStream Software', domainName: 'onestreamsoftware.com', - address: 'Birmingham', + addressAddressCity: 'Birmingham', employees: 1288, linkedinLinkUrl: 'https://linkedin.com/company/onestream-software', }, { name: 'Rent.', domainName: 'rent.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 1285, linkedinLinkUrl: 'https://linkedin.com/company/rentsolutions', }, { name: 'Waystar', domainName: 'waystar.com', - address: 'Louisville', + addressAddressCity: 'Louisville', employees: 1273, linkedinLinkUrl: 'https://linkedin.com/company/waystar', }, { name: '2020', domainName: '2020spaces.com', - address: 'Westford', + addressAddressCity: 'Westford', employees: 1267, linkedinLinkUrl: 'https://linkedin.com/company/2020spaces', }, { name: 'isolved', domainName: 'isolvedhcm.com', - address: 'Charlotte', + addressAddressCity: 'Charlotte', employees: 1261, linkedinLinkUrl: 'https://linkedin.com/company/isolved', }, { name: 'Art Technology Group', domainName: 'atg.com', - address: 'Cambridge', + addressAddressCity: 'Cambridge', employees: 1259, linkedinLinkUrl: 'https://linkedin.com/company/atg', }, { name: 'CAST', domainName: 'castsoftware.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1259, linkedinLinkUrl: 'https://linkedin.com/company/cast', }, { name: 'OCLC', domainName: 'oc.lc', - address: 'Dublin', + addressAddressCity: 'Dublin', employees: 1258, linkedinLinkUrl: 'https://linkedin.com/company/oclc', }, { name: 'Mediaocean', domainName: 'mediaocean.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1255, linkedinLinkUrl: 'https://linkedin.com/company/mediaocean', }, { name: 'Bandwidth Inc.', domainName: 'bandwidth.com', - address: 'Raleigh', + addressAddressCity: 'Raleigh', employees: 1252, linkedinLinkUrl: 'https://linkedin.com/company/bandwidth-inc', }, { name: 'Hexagon Safety, Infrastructure & Geospatial', domainName: 'hexagonsafetyinfrastructure.com', - address: 'Madison', + addressAddressCity: 'Madison', employees: 1252, linkedinLinkUrl: 'https://linkedin.com/company/hexagon-geospatial', }, { name: 'Wish', domainName: 'wish.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1248, linkedinLinkUrl: 'https://linkedin.com/company/wishshopping', }, { name: 'Sagitec Solutions', domainName: 'sagitec.com', - address: 'Saint Paul', + addressAddressCity: 'Saint Paul', employees: 1244, linkedinLinkUrl: 'https://linkedin.com/company/sagitec-solutions', }, { name: 'Zinnia ', domainName: 'zinnia.com', - address: 'Greenwich', + addressAddressCity: 'Greenwich', employees: 1243, linkedinLinkUrl: 'https://linkedin.com/company/zinniatm', }, { name: 'CureMD', domainName: 'curemd.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1243, linkedinLinkUrl: 'https://linkedin.com/company/curemd', }, { name: 'Druva', domainName: 'druva.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 1238, linkedinLinkUrl: 'https://linkedin.com/company/druva', }, { name: 'Restaurant365', domainName: 'restaurant365.com', - address: 'Irvine', + addressAddressCity: 'Irvine', employees: 1234, linkedinLinkUrl: 'https://linkedin.com/company/restaurant365-cloud-erp-for-restaurants', @@ -2296,364 +2296,364 @@ export const companiesDemo = [ { name: 'Lawson Software', domainName: 'lawson.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1231, linkedinLinkUrl: 'https://linkedin.com/company/lawson-software', }, { name: 'AlphaSense', domainName: 'alpha-sense.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1223, linkedinLinkUrl: 'https://linkedin.com/company/alphasense', }, { name: 'ECI Software Solutions', domainName: 'ecisolutions.com', - address: 'Fort Worth', + addressAddressCity: 'Fort Worth', employees: 1223, linkedinLinkUrl: 'https://linkedin.com/company/eci-software--solutions', }, { name: 'Wrike', domainName: 'wrike.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 1210, linkedinLinkUrl: 'https://linkedin.com/company/wrike', }, { name: 'Syndigo', domainName: 'syndigo.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1208, linkedinLinkUrl: 'https://linkedin.com/company/syndigo', }, { name: 'Gigamon', domainName: 'gigamon.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 1196, linkedinLinkUrl: 'https://linkedin.com/company/gigamon', }, { name: 'Fastly', domainName: 'fastly.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1188, linkedinLinkUrl: 'https://linkedin.com/company/fastly', }, { name: 'Cantaloupe Inc', domainName: 'cantaloupe.com', - address: 'Malvern', + addressAddressCity: 'Malvern', employees: 1187, linkedinLinkUrl: 'https://linkedin.com/company/cantaloupeinc', }, { name: 'EagleView', domainName: 'eagleview.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 1184, linkedinLinkUrl: 'https://linkedin.com/company/eagleview-technologies-inc', }, { name: 'Litera', domainName: 'litera.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1183, linkedinLinkUrl: 'https://linkedin.com/company/literamicrosystems', }, { name: 'Collibra', domainName: 'collibra.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1183, linkedinLinkUrl: 'https://linkedin.com/company/collibra', }, { name: 'Picsart', domainName: 'picsart.com', - address: 'Miami Beach', + addressAddressCity: 'Miami Beach', employees: 1180, linkedinLinkUrl: 'https://linkedin.com/company/picsart-photo-studio', }, { name: 'CalAmp', domainName: 'calamp.com', - address: 'Irvine', + addressAddressCity: 'Irvine', employees: 1180, linkedinLinkUrl: 'https://linkedin.com/company/calamp-corp', }, { name: 'ESS', domainName: 'ess-home.com', - address: 'Tempe', + addressAddressCity: 'Tempe', employees: 1178, linkedinLinkUrl: 'https://linkedin.com/company/ess', }, { name: 'Grafana Labs', domainName: 'grafana.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1178, linkedinLinkUrl: 'https://linkedin.com/company/grafana-labs', }, { name: 'Fivetran', domainName: '5tran.co', - address: 'Oakland', + addressAddressCity: 'Oakland', employees: 1176, linkedinLinkUrl: 'https://linkedin.com/company/fivetran', }, { name: 'CentralSquare Technologies', domainName: 'centralsquare.com', - address: 'Lake Mary', + addressAddressCity: 'Lake Mary', employees: 1175, linkedinLinkUrl: 'https://linkedin.com/company/centralsqtech', }, { name: 'StubHub', domainName: 'stubhub.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1164, linkedinLinkUrl: 'https://linkedin.com/company/stubhub', }, { name: 'EIS Ltd', domainName: 'eisgroup.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1154, linkedinLinkUrl: 'https://linkedin.com/company/eisgroupltd', }, { name: 'Tebra', domainName: 'tebra.com', - address: 'Corona Del Mar', + addressAddressCity: 'Corona Del Mar', employees: 1151, linkedinLinkUrl: 'https://linkedin.com/company/tebra', }, { name: 'Benefitfocus', domainName: 'benefitfocus.com', - address: 'Charleston', + addressAddressCity: 'Charleston', employees: 1148, linkedinLinkUrl: 'https://linkedin.com/company/benefitfocus', }, { name: 'NISC', domainName: 'nisc.coop', - address: 'Lake Saint Louis', + addressAddressCity: 'Lake Saint Louis', employees: 1140, linkedinLinkUrl: 'https://linkedin.com/company/nisc', }, { name: 'Dell Compellent', domainName: 'dell.com', - address: 'Eden Prairie', + addressAddressCity: 'Eden Prairie', employees: 1138, linkedinLinkUrl: 'https://linkedin.com/company/dell-compellent', }, { name: 'Radancy', domainName: 'radancy.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1137, linkedinLinkUrl: 'https://linkedin.com/company/radancy', }, { name: 'Granicus', domainName: 'granicus.com', - address: 'Denver', + addressAddressCity: 'Denver', employees: 1134, linkedinLinkUrl: 'https://linkedin.com/company/granicusinc', }, { name: 'ACTIVE Network', domainName: 'activenetwork.com', - address: 'Plano', + addressAddressCity: 'Plano', employees: 1134, linkedinLinkUrl: 'https://linkedin.com/company/the-active-network', }, { name: 'Acquia', domainName: 'acquia.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 1134, linkedinLinkUrl: 'https://linkedin.com/company/acquia', }, { name: 'WalkMe\u2122', domainName: 'walkme.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1127, linkedinLinkUrl: 'https://linkedin.com/company/walkme', }, { name: 'Outbrain', domainName: 'outbrain.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1123, linkedinLinkUrl: 'https://linkedin.com/company/outbrain', }, { name: 'WillowTree', domainName: 'willowtreeapps.com', - address: 'Charlottesville', + addressAddressCity: 'Charlottesville', employees: 1117, linkedinLinkUrl: 'https://linkedin.com/company/willowtreeapps', }, { name: 'LogicMonitor', domainName: 'logicmonitor.com', - address: 'Santa Barbara', + addressAddressCity: 'Santa Barbara', employees: 1113, linkedinLinkUrl: 'https://linkedin.com/company/logicmonitor', }, { name: 'Jellysmack', domainName: 'jellysmack.com', - address: 'New York', + addressAddressCity: 'New York', employees: 1109, linkedinLinkUrl: 'https://linkedin.com/company/jellysmack', }, { name: 'Henry Schein One', domainName: 'henryscheinone.com', - address: 'American Fork', + addressAddressCity: 'American Fork', employees: 1108, linkedinLinkUrl: 'https://linkedin.com/company/henry-schein-one', }, { name: 'Prometheus Group', domainName: 'prometheusgroup.com', - address: 'Raleigh', + addressAddressCity: 'Raleigh', employees: 1102, linkedinLinkUrl: 'https://linkedin.com/company/prometheusgroup', }, { name: 'Atlas', domainName: 'atlashxm.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1101, linkedinLinkUrl: 'https://linkedin.com/company/atlashxm', }, { name: 'Dialpad', domainName: 'dialpad.com', - address: 'San Ramon', + addressAddressCity: 'San Ramon', employees: 1101, linkedinLinkUrl: 'https://linkedin.com/company/dialpad', }, { name: 'Accruent', domainName: 'accruent.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1098, linkedinLinkUrl: 'https://linkedin.com/company/accruent', }, { name: 'Charles River Development', domainName: 'crd.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 1090, linkedinLinkUrl: 'https://linkedin.com/company/charles-river-development', }, { name: 'Flexera', domainName: 'flexera.com', - address: 'Itasca', + addressAddressCity: 'Itasca', employees: 1089, linkedinLinkUrl: 'https://linkedin.com/company/flexera', }, { name: 'Quotient Technology Inc.', domainName: 'quotient.com', - address: 'Salt Lake City', + addressAddressCity: 'Salt Lake City', employees: 1087, linkedinLinkUrl: 'https://linkedin.com/company/quotient-technology', }, { name: 'Sage Intacct, Inc.', domainName: 'sageintacct.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 1087, linkedinLinkUrl: 'https://linkedin.com/company/sageintacct', }, { name: 'Plaid', domainName: 'plaid.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1081, linkedinLinkUrl: 'https://linkedin.com/company/plaid-', }, { name: 'C3 AI', domainName: 'c3.ai', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 1077, linkedinLinkUrl: 'https://linkedin.com/company/c3-ai', }, { name: 'Upland Software', domainName: 'uplandsoftware.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1072, linkedinLinkUrl: 'https://linkedin.com/company/upland-software', }, { name: 'Zapier', domainName: 'zapier.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1066, linkedinLinkUrl: 'https://linkedin.com/company/zapier', }, { name: 'WSO2', domainName: 'wso2.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 1065, linkedinLinkUrl: 'https://linkedin.com/company/wso2', }, { name: 'Auctane', domainName: 'auctane.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 1055, linkedinLinkUrl: 'https://linkedin.com/company/auctane', }, { name: 'Salesloft', domainName: 'salesloft.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 1055, linkedinLinkUrl: 'https://linkedin.com/company/salesloft', }, { name: 'RLDatix', domainName: 'rldatix.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1048, linkedinLinkUrl: 'https://linkedin.com/company/rldatix', }, { name: 'SS&C Blue Prism', domainName: 'blueprism.com', - address: 'Windsor', + addressAddressCity: 'Windsor', employees: 1048, linkedinLinkUrl: 'https://linkedin.com/company/blue-prism-limited', }, { name: 'Waitr', domainName: 'waitrapp.com', - address: 'Lafayette', + addressAddressCity: 'Lafayette', employees: 1043, linkedinLinkUrl: 'https://linkedin.com/company/waitr-inc-', }, { name: 'Software Engineering Institute | Carnegie Mellon University', domainName: 'sei.cmu.edu', - address: 'Pittsburgh', + addressAddressCity: 'Pittsburgh', employees: 1043, linkedinLinkUrl: 'https://linkedin.com/company/software-engineering-institute', @@ -2661,7 +2661,7 @@ export const companiesDemo = [ { name: 'Downey Unified School District', domainName: 'dusd.net', - address: 'Downey', + addressAddressCity: 'Downey', employees: 1038, linkedinLinkUrl: 'https://linkedin.com/company/downey-unified-school-district', @@ -2669,1225 +2669,1225 @@ export const companiesDemo = [ { name: 'Private Access, Inc.', domainName: 'privateaccess.com', - address: 'Irvine', + addressAddressCity: 'Irvine', employees: 1037, linkedinLinkUrl: 'https://linkedin.com/company/private-access-inc.', }, { name: 'iManage', domainName: 'imanage.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1036, linkedinLinkUrl: 'https://linkedin.com/company/imanage', }, { name: 'QASource', domainName: 'qasource.com', - address: 'Pleasanton', + addressAddressCity: 'Pleasanton', employees: 1032, linkedinLinkUrl: 'https://linkedin.com/company/qasource', }, { name: 'Azuga, Inc.', domainName: 'azuga.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 1026, linkedinLinkUrl: 'https://linkedin.com/company/azuga-inc-', }, { name: 'Talent Systems, LLC', domainName: 'talentsystems.com', - address: 'Los Angeles', + addressAddressCity: 'Los Angeles', employees: 1022, linkedinLinkUrl: 'https://linkedin.com/company/talent-systems-llc', }, { name: 'Datasite', domainName: 'datasite.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 1021, linkedinLinkUrl: 'https://linkedin.com/company/datasiteglobal', }, { name: 'AVASOFT', domainName: 'avasoft.com', - address: 'Blue Bell', + addressAddressCity: 'Blue Bell', employees: 1017, linkedinLinkUrl: 'https://linkedin.com/company/avasoft', }, { name: 'DataRobot', domainName: 'datarobot.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 1015, linkedinLinkUrl: 'https://linkedin.com/company/datarobot', }, { name: 'Technisys', domainName: 'technisys.com', - address: 'Miami', + addressAddressCity: 'Miami', employees: 1014, linkedinLinkUrl: 'https://linkedin.com/company/technisys', }, { name: 'project44', domainName: 'project44.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 1013, linkedinLinkUrl: 'https://linkedin.com/company/project-44', }, { name: 'Imprivata', domainName: 'imprivata.com', - address: 'Waltham', + addressAddressCity: 'Waltham', employees: 1013, linkedinLinkUrl: 'https://linkedin.com/company/imprivata', }, { name: 'Webflow', domainName: 'webflow.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1011, linkedinLinkUrl: 'https://linkedin.com/company/webflow-inc-', }, { name: 'Blend', domainName: 'blend.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 1011, linkedinLinkUrl: 'https://linkedin.com/company/blend-', }, { name: 'Egnyte', domainName: 'egnyte.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 1009, linkedinLinkUrl: 'https://linkedin.com/company/egnyte', }, { name: 'SS&C Eze', domainName: 'ezesoft.com', - address: 'Windsor', + addressAddressCity: 'Windsor', employees: 1008, linkedinLinkUrl: 'https://linkedin.com/company/ezesoftware', }, { name: 'Tipalti', domainName: 'tipalti.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 1007, linkedinLinkUrl: 'https://linkedin.com/company/tipalti', }, { name: 'Altium\u00ae', domainName: 'altium.com', - address: 'La Jolla', + addressAddressCity: 'La Jolla', employees: 1005, linkedinLinkUrl: 'https://linkedin.com/company/altium', }, { name: 'airSlate', domainName: 'airslate.com', - address: 'Brookline', + addressAddressCity: 'Brookline', employees: 1001, linkedinLinkUrl: 'https://linkedin.com/company/airslate', }, { name: 'Arbisoft', domainName: 'arbisoft.com', - address: 'Mckinney', + addressAddressCity: 'Mckinney', employees: 996, linkedinLinkUrl: 'https://linkedin.com/company/arbisoft', }, { name: 'Airtable', domainName: 'airtable.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 989, linkedinLinkUrl: 'https://linkedin.com/company/airtable', }, { name: 'Birdeye', domainName: 'birdeye.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 988, linkedinLinkUrl: 'https://linkedin.com/company/birdeye', }, { name: 'Ultimate Software', domainName: 'ultimatesoftware.com', - address: 'Fort Lauderdale', + addressAddressCity: 'Fort Lauderdale', employees: 988, linkedinLinkUrl: 'https://linkedin.com/company/ultimate-software', }, { name: 'Homecare Homebase', domainName: 'hchb.com', - address: 'Dallas', + addressAddressCity: 'Dallas', employees: 987, linkedinLinkUrl: 'https://linkedin.com/company/homecare-homebase', }, { name: 'DISCO', domainName: 'csdisco.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 984, linkedinLinkUrl: 'https://linkedin.com/company/cs-disco-llc', }, { name: 'Highspot', domainName: 'highspot.com', - address: 'Seattle', + addressAddressCity: 'Seattle', employees: 982, linkedinLinkUrl: 'https://linkedin.com/company/highspot', }, { name: 'Sagent', domainName: 'sagent.com', - address: 'King Of Prussia', + addressAddressCity: 'King Of Prussia', employees: 981, linkedinLinkUrl: 'https://linkedin.com/company/sagent-lending-technologies', }, { name: 'Apollo.io', domainName: 'apollo.io', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 981, linkedinLinkUrl: 'https://linkedin.com/company/apolloio', }, { name: 'PAS', domainName: 'pas.com', - address: 'Houston', + addressAddressCity: 'Houston', employees: 981, linkedinLinkUrl: 'https://linkedin.com/company/pas', }, { name: 'Wikimedia Foundation', domainName: 'wikimediafoundation.org', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 981, linkedinLinkUrl: 'https://linkedin.com/company/wikimedia-foundation', }, { name: 'Nintex', domainName: 'nintex.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 978, linkedinLinkUrl: 'https://linkedin.com/company/nintex', }, { name: 'RUCKUS Networks', domainName: 'commscope.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 978, linkedinLinkUrl: 'https://linkedin.com/company/ruckus-networks', }, { name: 'ForgeRock', domainName: 'forgerock.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 977, linkedinLinkUrl: 'https://linkedin.com/company/forgerock', }, { name: 'Trading Technologies', domainName: 'tradingtechnologies.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 975, linkedinLinkUrl: 'https://linkedin.com/company/trading-technologies', }, { name: 'KANINI', domainName: 'kanini.com', - address: 'Nashville', + addressAddressCity: 'Nashville', employees: 972, linkedinLinkUrl: 'https://linkedin.com/company/kanini', }, { name: 'Dealer.com', domainName: 'dealer.com', - address: 'Burlington', + addressAddressCity: 'Burlington', employees: 962, linkedinLinkUrl: 'https://linkedin.com/company/dealer-com', }, { name: 'WS', domainName: 'ws-inc.com', - address: 'Pinehurst', + addressAddressCity: 'Pinehurst', employees: 960, linkedinLinkUrl: 'https://linkedin.com/company/wbem-solutions', }, { name: 'Kyriba', domainName: 'kyriba.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 960, linkedinLinkUrl: 'https://linkedin.com/company/kyriba', }, { name: 'Demandbase', domainName: 'demandbase.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 958, linkedinLinkUrl: 'https://linkedin.com/company/demandbase', }, { name: 'Sumo Logic', domainName: 'sumologic.com', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 954, linkedinLinkUrl: 'https://linkedin.com/company/sumo-logic', }, { name: 'Edifecs', domainName: 'edifecs.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 949, linkedinLinkUrl: 'https://linkedin.com/company/edifecs', }, { name: 'ibi | Information Builders', domainName: 'ibi.com', - address: 'Fort Lauderdale', + addressAddressCity: 'Fort Lauderdale', employees: 948, linkedinLinkUrl: 'https://linkedin.com/company/information-builders', }, { name: 'Emburse', domainName: 'emburse.com', - address: 'Los Angeles', + addressAddressCity: 'Los Angeles', employees: 941, linkedinLinkUrl: 'https://linkedin.com/company/emburse', }, { name: 'ConstructConnect', domainName: 'constructconnect.com', - address: 'Cincinnati', + addressAddressCity: 'Cincinnati', employees: 940, linkedinLinkUrl: 'https://linkedin.com/company/constructconnect', }, { name: 'Perforce Software', domainName: 'perforce.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 939, linkedinLinkUrl: 'https://linkedin.com/company/perforce', }, { name: 'Insurity', domainName: 'insurity.com', - address: 'Hartford', + addressAddressCity: 'Hartford', employees: 938, linkedinLinkUrl: 'https://linkedin.com/company/insurity', }, { name: 'webOS', domainName: 'developer.lge.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 936, linkedinLinkUrl: 'https://linkedin.com/company/webos', }, { name: 'Zenoti', domainName: 'zenoti.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 934, linkedinLinkUrl: 'https://linkedin.com/company/zenoti', }, { name: 'Intapp', domainName: 'intapp.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 930, linkedinLinkUrl: 'https://linkedin.com/company/intapp', }, { name: 'OATI', domainName: 'oati.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 930, linkedinLinkUrl: 'https://linkedin.com/company/oati', }, { name: 'Frontline Education', domainName: 'frontlineeducation.com', - address: 'Malvern', + addressAddressCity: 'Malvern', employees: 926, linkedinLinkUrl: 'https://linkedin.com/company/frontline-education', }, { name: 'Aspect Software', domainName: 'aspect.com', - address: 'Westford', + addressAddressCity: 'Westford', employees: 920, linkedinLinkUrl: 'https://linkedin.com/company/aspect-software', }, { name: 'GreyOrange', domainName: 'greyorange.com', - address: 'Roswell', + addressAddressCity: 'Roswell', employees: 919, linkedinLinkUrl: 'https://linkedin.com/company/gogreyorange', }, { name: 'Sirion', domainName: 'sirionlabs.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 918, linkedinLinkUrl: 'https://linkedin.com/company/sirionlabs', }, { name: 'In Time Tec', domainName: 'intimetec.com', - address: 'Meridian', + addressAddressCity: 'Meridian', employees: 917, linkedinLinkUrl: 'https://linkedin.com/company/in-time-tec', }, { name: 'Operative', domainName: 'operative.com', - address: 'New York', + addressAddressCity: 'New York', employees: 910, linkedinLinkUrl: 'https://linkedin.com/company/operative', }, { name: 'Kore.ai', domainName: 'kore.ai', - address: 'Orlando', + addressAddressCity: 'Orlando', employees: 908, linkedinLinkUrl: 'https://linkedin.com/company/kore-inc', }, { name: 'Redis', domainName: 'redis.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 908, linkedinLinkUrl: 'https://linkedin.com/company/redisinc', }, { name: 'Addepar', domainName: 'addepar.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 907, linkedinLinkUrl: 'https://linkedin.com/company/addepar', }, { name: 'TCP Software', domainName: 'tcpsoftware.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 902, linkedinLinkUrl: 'https://linkedin.com/company/tcpsoftware', }, { name: 'TraceLink', domainName: 'tracelink.com', - address: 'Wilmington', + addressAddressCity: 'Wilmington', employees: 897, linkedinLinkUrl: 'https://linkedin.com/company/tracelink', }, { name: 'Benchling', domainName: 'benchling.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 895, linkedinLinkUrl: 'https://linkedin.com/company/benchling', }, { name: 'Housecall Pro', domainName: 'housecallpro.com', - address: 'Denver', + addressAddressCity: 'Denver', employees: 894, linkedinLinkUrl: 'https://linkedin.com/company/housecallpro', }, { name: 'Turnitin', domainName: 'turnitin.com', - address: 'Oakland', + addressAddressCity: 'Oakland', employees: 885, linkedinLinkUrl: 'https://linkedin.com/company/turnitin', }, { name: 'Schr\u00f6dinger', domainName: 'schrodinger.com', - address: 'New York', + addressAddressCity: 'New York', employees: 885, linkedinLinkUrl: 'https://linkedin.com/company/schr-dinger', }, { name: 'eGain Corporation', domainName: 'egain.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 879, linkedinLinkUrl: 'https://linkedin.com/company/egain-corporation', }, { name: 'Brightly', domainName: 'brightlysoftware.com', - address: 'Cary', + addressAddressCity: 'Cary', employees: 878, linkedinLinkUrl: 'https://linkedin.com/company/brightlysoftware', }, { name: 'Snap-on Business Solutions', domainName: 'snapon.com', - address: 'Richfield', + addressAddressCity: 'Richfield', employees: 876, linkedinLinkUrl: 'https://linkedin.com/company/snap-on-business-solutions', }, { name: 'ACS Technologies', domainName: 'acstechnologies.com', - address: 'Florence', + addressAddressCity: 'Florence', employees: 874, linkedinLinkUrl: 'https://linkedin.com/company/acs-technologies', }, { name: 'Uniphore', domainName: 'uniphore.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 872, linkedinLinkUrl: 'https://linkedin.com/company/uniphore', }, { name: 'Folio3 Software', domainName: 'folio3.com', - address: 'Belmont', + addressAddressCity: 'Belmont', employees: 872, linkedinLinkUrl: 'https://linkedin.com/company/folio3', }, { name: 'MHC', domainName: 'mhcautomation.com', - address: 'Burnsville', + addressAddressCity: 'Burnsville', employees: 871, linkedinLinkUrl: 'https://linkedin.com/company/mhcautomation', }, { name: 'Xactly Corp', domainName: 'xactlycorp.com', - address: 'Los Gatos', + addressAddressCity: 'Los Gatos', employees: 865, linkedinLinkUrl: 'https://linkedin.com/company/xactly-corporation', }, { name: 'Weave', domainName: 'getweave.com', - address: 'Lehi', + addressAddressCity: 'Lehi', employees: 864, linkedinLinkUrl: 'https://linkedin.com/company/getweave', }, { name: 'Microworkers', domainName: 'microworkers.com', - address: 'Frisco', + addressAddressCity: 'Frisco', employees: 862, linkedinLinkUrl: 'https://linkedin.com/company/microworkers.com', }, { name: 'Trilogy', domainName: 'trilogy.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 861, linkedinLinkUrl: 'https://linkedin.com/company/trilogy', }, { name: 'Akvelon, Inc.', domainName: 'akvelon.com', - address: 'Bellevue', + addressAddressCity: 'Bellevue', employees: 860, linkedinLinkUrl: 'https://linkedin.com/company/akvelon', }, { name: 'iPipeline', domainName: 'ipipeline.com', - address: 'Exton', + addressAddressCity: 'Exton', employees: 856, linkedinLinkUrl: 'https://linkedin.com/company/ipipeline', }, { name: 'Salary.com', domainName: 'salary.com', - address: 'Wellesley Hills', + addressAddressCity: 'Wellesley Hills', employees: 854, linkedinLinkUrl: 'https://linkedin.com/company/salarydotcom', }, { name: 'PandaDoc', domainName: 'pandadoc.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 851, linkedinLinkUrl: 'https://linkedin.com/company/pandadoc', }, { name: 'MSC Software', domainName: 'mscsoftware.com', - address: 'Newport Beach', + addressAddressCity: 'Newport Beach', employees: 849, linkedinLinkUrl: 'https://linkedin.com/company/msc-software', }, { name: 'Harness', domainName: 'harness.io', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 848, linkedinLinkUrl: 'https://linkedin.com/company/harnessinc', }, { name: 'ActiveCampaign', domainName: 'activecampaign.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 848, linkedinLinkUrl: 'https://linkedin.com/company/activecampaign-inc-', }, { name: 'Doximity', domainName: 'doximity.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 848, linkedinLinkUrl: 'https://linkedin.com/company/doximity', }, { name: 'Couchbase', domainName: 'couchbase.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 847, linkedinLinkUrl: 'https://linkedin.com/company/couchbase', }, { name: 'Lytx, Inc.', domainName: 'lytx.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 845, linkedinLinkUrl: 'https://linkedin.com/company/lytxinc', }, { name: 'Pendo.io', domainName: 'pendo.io', - address: 'Raleigh', + addressAddressCity: 'Raleigh', employees: 844, linkedinLinkUrl: 'https://linkedin.com/company/pendo-io', }, { name: 'Workato', domainName: 'workato.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 842, linkedinLinkUrl: 'https://linkedin.com/company/workato', }, { name: 'Saviynt', domainName: 'saviynt.com', - address: 'El Segundo', + addressAddressCity: 'El Segundo', employees: 842, linkedinLinkUrl: 'https://linkedin.com/company/saviynt', }, { name: 'SmartBear', domainName: 'smartbear.com', - address: 'Somerville', + addressAddressCity: 'Somerville', employees: 838, linkedinLinkUrl: 'https://linkedin.com/company/smartbear', }, { name: 'Rovi Corporation (now TiVo)', domainName: 'tivo.com', - address: 'San Carlos', + addressAddressCity: 'San Carlos', employees: 837, linkedinLinkUrl: 'https://linkedin.com/company/rovi', }, { name: 'Handshake', domainName: 'joinhandshake.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 833, linkedinLinkUrl: 'https://linkedin.com/company/team-handshake', }, { name: 'Navitaire, an Amadeus company', domainName: 'navitaire.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 829, linkedinLinkUrl: 'https://linkedin.com/company/navitaire', }, { name: 'OneSpan', domainName: 'onespan.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 826, linkedinLinkUrl: 'https://linkedin.com/company/onespan', }, { name: 'Bitsight', domainName: 'bitsight.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 824, linkedinLinkUrl: 'https://linkedin.com/company/bitsight', }, { name: 'ID.me', domainName: 'id.me', - address: 'Mc Lean', + addressAddressCity: 'Mc Lean', employees: 823, linkedinLinkUrl: 'https://linkedin.com/company/id.me', }, { name: 'SymphonyAI Retail CPG', domainName: 'symphonyretailai.com', - address: 'Frisco', + addressAddressCity: 'Frisco', employees: 823, linkedinLinkUrl: 'https://linkedin.com/company/symphonyretailcpg', }, { name: 'Unilog', domainName: 'unilogcorp.com', - address: 'Wayne', + addressAddressCity: 'Wayne', employees: 823, linkedinLinkUrl: 'https://linkedin.com/company/unilog-inc', }, { name: 'Teletrac Navman', domainName: 'teletracnavman.com', - address: 'Irvine', + addressAddressCity: 'Irvine', employees: 821, linkedinLinkUrl: 'https://linkedin.com/company/teletrac', }, { name: 'Buildertrend', domainName: 'buildertrend.com', - address: 'Omaha', + addressAddressCity: 'Omaha', employees: 819, linkedinLinkUrl: 'https://linkedin.com/company/buildertrend', }, { name: 'Tecsys Inc.', domainName: 'tecsys.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 816, linkedinLinkUrl: 'https://linkedin.com/company/tecsys-inc', }, { name: 'ThousandEyes (part of Cisco)', domainName: 'thousandeyes.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 816, linkedinLinkUrl: 'https://linkedin.com/company/thousandeyes', }, { name: 'Greenhouse Software', domainName: 'greenhouse.io', - address: 'New York', + addressAddressCity: 'New York', employees: 814, linkedinLinkUrl: 'https://linkedin.com/company/greenhouse-inc-', }, { name: 'Exiger', domainName: 'exiger.com', - address: 'New York', + addressAddressCity: 'New York', employees: 811, linkedinLinkUrl: 'https://linkedin.com/company/exiger', }, { name: 'MBO Partners', domainName: 'mbopartners.com', - address: 'Ashburn', + addressAddressCity: 'Ashburn', employees: 808, linkedinLinkUrl: 'https://linkedin.com/company/mbo-partners', }, { name: 'Neo4j', domainName: 'neo4j.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 808, linkedinLinkUrl: 'https://linkedin.com/company/neo4j', }, { name: 'VTS', domainName: 'vts.com', - address: 'New York', + addressAddressCity: 'New York', employees: 805, linkedinLinkUrl: 'https://linkedin.com/company/we-are-vts', }, { name: 'Slice', domainName: 'slicelife.com', - address: 'New York', + addressAddressCity: 'New York', employees: 805, linkedinLinkUrl: 'https://linkedin.com/company/slice', }, { name: 'Amplitude', domainName: 'amplitude.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 803, linkedinLinkUrl: 'https://linkedin.com/company/amplitude-analytics', }, { name: 'Daxko', domainName: 'daxko.com', - address: 'Birmingham', + addressAddressCity: 'Birmingham', employees: 802, linkedinLinkUrl: 'https://linkedin.com/company/daxko', }, { name: 'AppLovin', domainName: 'applovin.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 802, linkedinLinkUrl: 'https://linkedin.com/company/applovin', }, { name: 'Xometry', domainName: 'xometry.com', - address: 'Rockville', + addressAddressCity: 'Rockville', employees: 801, linkedinLinkUrl: 'https://linkedin.com/company/xometry', }, { name: 'Quickbase', domainName: 'quickbase.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 796, linkedinLinkUrl: 'https://linkedin.com/company/quickbase', }, { name: 'Agora', domainName: 'agora.io', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 793, linkedinLinkUrl: 'https://linkedin.com/company/agora-lab-inc', }, { name: 'InMoment', domainName: 'inmoment.com', - address: 'South Jordan', + addressAddressCity: 'South Jordan', employees: 793, linkedinLinkUrl: 'https://linkedin.com/company/weareinmoment', }, { name: 'PatientPoint\u00ae', domainName: 'patientpoint.com', - address: 'Cincinnati', + addressAddressCity: 'Cincinnati', employees: 789, linkedinLinkUrl: 'https://linkedin.com/company/patientpoint', }, { name: 'HHAeXchange', domainName: 'hhaexchange.com', - address: 'New York', + addressAddressCity: 'New York', employees: 788, linkedinLinkUrl: 'https://linkedin.com/company/hhaexchange', }, { name: 'NinjaOne', domainName: 'ninjaone.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 787, linkedinLinkUrl: 'https://linkedin.com/company/ninjaone', }, { name: 'Zywave', domainName: 'zywave.com', - address: 'Milwaukee', + addressAddressCity: 'Milwaukee', employees: 785, linkedinLinkUrl: 'https://linkedin.com/company/zywave', }, { name: 'Adobe Marketo', domainName: 'marketo.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 784, linkedinLinkUrl: 'https://linkedin.com/company/adobemarketoengage', }, { name: 'MasterControl', domainName: 'mastercontrol.com', - address: 'Salt Lake City', + addressAddressCity: 'Salt Lake City', employees: 783, linkedinLinkUrl: 'https://linkedin.com/company/mastercontrol', }, { name: 'Jumio Corporation', domainName: 'jumio.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 779, linkedinLinkUrl: 'https://linkedin.com/company/jumio-corporation', }, { name: 'CRMNEXT', domainName: 'crmnext.com', - address: 'Raleigh', + addressAddressCity: 'Raleigh', employees: 778, linkedinLinkUrl: 'https://linkedin.com/company/crmnext', }, { name: 'ChannelAdvisor', domainName: 'channeladvisor.com', - address: 'Morrisville', + addressAddressCity: 'Morrisville', employees: 777, linkedinLinkUrl: 'https://linkedin.com/company/channeladvisor', }, { name: 'SumTotal Systems, LLC', domainName: 'sumtotalsystems.com', - address: 'Gainesville', + addressAddressCity: 'Gainesville', employees: 776, linkedinLinkUrl: 'https://linkedin.com/company/sumtotal-systems', }, { name: 'Payscale', domainName: 'payscale.com', - address: 'Seattle', + addressAddressCity: 'Seattle', employees: 775, linkedinLinkUrl: 'https://linkedin.com/company/payscale', }, { name: 'Riskonnect, Inc.', domainName: 'riskonnect.com', - address: 'Kennesaw', + addressAddressCity: 'Kennesaw', employees: 775, linkedinLinkUrl: 'https://linkedin.com/company/riskonnect-inc', }, { name: 'Riskified', domainName: 'riskified.com', - address: 'New York', + addressAddressCity: 'New York', employees: 770, linkedinLinkUrl: 'https://linkedin.com/company/riskified', }, { name: 'Shopkeeper', domainName: 'shopkeeper.com', - address: 'Pompano Beach', + addressAddressCity: 'Pompano Beach', employees: 770, linkedinLinkUrl: 'https://linkedin.com/company/shopkeeperapp', }, { name: 'Stack Overflow', domainName: 'stackoverflow.com', - address: 'New York', + addressAddressCity: 'New York', employees: 768, linkedinLinkUrl: 'https://linkedin.com/company/stack-overflow', }, { name: 'Netwrix Corporation', domainName: 'netwrix.com', - address: 'Frisco', + addressAddressCity: 'Frisco', employees: 768, linkedinLinkUrl: 'https://linkedin.com/company/netwrix-corporation', }, { name: 'Securonix', domainName: 'securonix.com', - address: 'Addison', + addressAddressCity: 'Addison', employees: 767, linkedinLinkUrl: 'https://linkedin.com/company/securonix', }, { name: 'Draup', domainName: 'draup.com', - address: 'Spring', + addressAddressCity: 'Spring', employees: 766, linkedinLinkUrl: 'https://linkedin.com/company/draupplatform', }, { name: 'eQ Technologic', domainName: '1eq.com', - address: 'Costa Mesa', + addressAddressCity: 'Costa Mesa', employees: 766, linkedinLinkUrl: 'https://linkedin.com/company/eq-technologic', }, { name: 'Mindtickle', domainName: 'mindtickle.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 765, linkedinLinkUrl: 'https://linkedin.com/company/mindtickle', }, { name: 'Omnitracs', domainName: 'omnitracs.com', - address: 'Roanoke', + addressAddressCity: 'Roanoke', employees: 764, linkedinLinkUrl: 'https://linkedin.com/company/omnitracs', }, { name: 'Programmer', domainName: 'gregoryleroy.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 762, linkedinLinkUrl: 'https://linkedin.com/company/programmer', }, { name: 'Navis', domainName: 'navis.com', - address: 'Alpharetta', + addressAddressCity: 'Alpharetta', employees: 761, linkedinLinkUrl: 'https://linkedin.com/company/navis', }, { name: 'AuditBoard', domainName: 'auditboard.com', - address: 'Cerritos', + addressAddressCity: 'Cerritos', employees: 759, linkedinLinkUrl: 'https://linkedin.com/company/auditboard', }, { name: 'Algolia', domainName: 'algolia.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 759, linkedinLinkUrl: 'https://linkedin.com/company/algolia', }, { name: 'YML', domainName: 'yml.co', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 754, linkedinLinkUrl: 'https://linkedin.com/company/ymlco', }, { name: 'Bolt', domainName: 'bolt.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 750, linkedinLinkUrl: 'https://linkedin.com/company/bolt-com', }, { name: 'Dandy', domainName: 'meetdandy.com', - address: 'New York', + addressAddressCity: 'New York', employees: 745, linkedinLinkUrl: 'https://linkedin.com/company/dandyofficial', }, { name: 'Diverse Lynx', domainName: 'diverselynx.com', - address: 'Princeton', + addressAddressCity: 'Princeton', employees: 743, linkedinLinkUrl: 'https://linkedin.com/company/diverselynx', }, { name: 'JMP', domainName: 'jmp.com', - address: 'Cary', + addressAddressCity: 'Cary', employees: 741, linkedinLinkUrl: 'https://linkedin.com/company/jmp', }, { name: 'ON24', domainName: 'on24.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 741, linkedinLinkUrl: 'https://linkedin.com/company/on24', }, { name: 'LabVantage Solutions, Inc', domainName: 'labvantage.com', - address: 'Somerset', + addressAddressCity: 'Somerset', employees: 740, linkedinLinkUrl: 'https://linkedin.com/company/labvantage', }, { name: 'Exabeam', domainName: 'exabeam.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 739, linkedinLinkUrl: 'https://linkedin.com/company/exabeam', }, { name: 'Iterable', domainName: 'iterable.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 739, linkedinLinkUrl: 'https://linkedin.com/company/iterable', }, { name: 'Clari', domainName: 'clari.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 737, linkedinLinkUrl: 'https://linkedin.com/company/clari', }, { name: 'Komodo Health', domainName: 'komodohealth.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 737, linkedinLinkUrl: 'https://linkedin.com/company/komodo-health', }, { name: 'Alation', domainName: 'alation.com', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 736, linkedinLinkUrl: 'https://linkedin.com/company/alation', }, { name: 'Celigo', domainName: 'celigo.com', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 733, linkedinLinkUrl: 'https://linkedin.com/company/celigo-inc', }, { name: 'Aptos Retail', domainName: 'aptos.com', - address: 'Alpharetta', + addressAddressCity: 'Alpharetta', employees: 733, linkedinLinkUrl: 'https://linkedin.com/company/aptos-retail', }, { name: 'WorkForce Software', domainName: 'workforcesoftware.com', - address: 'Livonia', + addressAddressCity: 'Livonia', employees: 721, linkedinLinkUrl: 'https://linkedin.com/company/workforce-software', }, { name: 'HPE Security - Data Security', domainName: 'voltage.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 713, linkedinLinkUrl: 'https://linkedin.com/company/hpe-security-data-security', }, { name: 'DDN Storage', domainName: 'ddn.com', - address: 'Chatsworth', + addressAddressCity: 'Chatsworth', employees: 712, linkedinLinkUrl: 'https://linkedin.com/company/ddn-storage', }, { name: 'KPA', domainName: 'kpa.io', - address: 'Broomfield', + addressAddressCity: 'Broomfield', employees: 709, linkedinLinkUrl: 'https://linkedin.com/company/kpa-llc', }, { name: 'Lohika', domainName: 'lohika.com', - address: 'San Mateo', + addressAddressCity: 'San Mateo', employees: 705, linkedinLinkUrl: 'https://linkedin.com/company/lohika', }, { name: 'Qualifacts', domainName: 'qualifacts.com', - address: 'Nashville', + addressAddressCity: 'Nashville', employees: 705, linkedinLinkUrl: 'https://linkedin.com/company/qualifacts', }, { name: 'Centric Software', domainName: 'centricsoftware.com', - address: 'Campbell', + addressAddressCity: 'Campbell', employees: 705, linkedinLinkUrl: 'https://linkedin.com/company/centric-software', }, { name: 'Omdena', domainName: 'omdena.com', - address: 'New York', + addressAddressCity: 'New York', employees: 704, linkedinLinkUrl: 'https://linkedin.com/company/omdena', }, { name: 'AccountantsWorld', domainName: 'accountantsworld.com', - address: 'Hauppauge', + addressAddressCity: 'Hauppauge', employees: 704, linkedinLinkUrl: 'https://linkedin.com/company/accountantsworld', }, { name: 'Aderant', domainName: 'aderant.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 704, linkedinLinkUrl: 'https://linkedin.com/company/aderant', }, { name: 'Python Software Foundation', domainName: 'python.org', - address: 'Beaverton', + addressAddressCity: 'Beaverton', employees: 704, linkedinLinkUrl: 'https://linkedin.com/company/python-software-foundation', }, { name: 'OpenGov Inc.', domainName: 'opengov.com', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 703, linkedinLinkUrl: 'https://linkedin.com/company/opengov-inc', }, { name: 'Denodo', domainName: 'denodo.com', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 702, linkedinLinkUrl: 'https://linkedin.com/company/denodo-technologies', }, { name: 'NEOGOV', domainName: 'neogov.com', - address: 'El Segundo', + addressAddressCity: 'El Segundo', employees: 698, linkedinLinkUrl: 'https://linkedin.com/company/neogov', }, { name: 'VertexOne', domainName: 'vertexone.net', - address: 'Dallas', + addressAddressCity: 'Dallas', employees: 696, linkedinLinkUrl: 'https://linkedin.com/company/vertex-one', }, { name: 'The Linux Foundation', domainName: 'linuxfoundation.org', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 694, linkedinLinkUrl: 'https://linkedin.com/company/the-linux-foundation', }, { name: 'Reputation', domainName: 'reputation.com', - address: 'San Ramon', + addressAddressCity: 'San Ramon', employees: 694, linkedinLinkUrl: 'https://linkedin.com/company/reputation-com', }, { name: 'Relevantz ', domainName: 'relevantz.com', - address: 'Alpharetta', + addressAddressCity: 'Alpharetta', employees: 691, linkedinLinkUrl: 'https://linkedin.com/company/relevantz', }, { name: 'M-Files', domainName: 'm-files.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 691, linkedinLinkUrl: 'https://linkedin.com/company/m-files-corporation', }, { name: 'Homebase', domainName: 'joinhomebase.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 688, linkedinLinkUrl: 'https://linkedin.com/company/homebase-app', }, { name: 'Calypso Technology', domainName: 'calypso.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 688, linkedinLinkUrl: 'https://linkedin.com/company/calypso-technology', }, { name: 'Viewpoint', domainName: 'viewpoint.com', - address: 'Broomfield', + addressAddressCity: 'Broomfield', employees: 686, linkedinLinkUrl: 'https://linkedin.com/company/viewpoint-construction-software', @@ -3895,315 +3895,315 @@ export const companiesDemo = [ { name: 'Devo', domainName: 'devo.com', - address: 'Cambridge', + addressAddressCity: 'Cambridge', employees: 685, linkedinLinkUrl: 'https://linkedin.com/company/devoinc', }, { name: 'WebPT', domainName: 'webpt.com', - address: 'Phoenix', + addressAddressCity: 'Phoenix', employees: 685, linkedinLinkUrl: 'https://linkedin.com/company/webpt', }, { name: 'MatrixCare', domainName: 'matrixcare.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 683, linkedinLinkUrl: 'https://linkedin.com/company/matrixcare', }, { name: 'Sisense', domainName: 'sisense.com', - address: 'New York', + addressAddressCity: 'New York', employees: 683, linkedinLinkUrl: 'https://linkedin.com/company/sisense', }, { name: 'Calendly', domainName: 'calendly.com', - address: 'Atlanta', + addressAddressCity: 'Atlanta', employees: 681, linkedinLinkUrl: 'https://linkedin.com/company/calendly', }, { name: 'Placer.ai', domainName: 'placer.io', - address: 'Los Altos', + addressAddressCity: 'Los Altos', employees: 677, linkedinLinkUrl: 'https://linkedin.com/company/placer', }, { name: 'MResult', domainName: 'mresult.com', - address: 'Mystic', + addressAddressCity: 'Mystic', employees: 674, linkedinLinkUrl: 'https://linkedin.com/company/mresult', }, { name: 'Coherent Solutions', domainName: 'coherentsolutions.com', - address: 'Minneapolis', + addressAddressCity: 'Minneapolis', employees: 672, linkedinLinkUrl: 'https://linkedin.com/company/coherent-solutions', }, { name: 'Mirantis', domainName: 'mirantis.com', - address: 'Campbell', + addressAddressCity: 'Campbell', employees: 671, linkedinLinkUrl: 'https://linkedin.com/company/mirantis', }, { name: 'Simplify Healthcare', domainName: 'simplifyhealthcare.com', - address: 'Aurora', + addressAddressCity: 'Aurora', employees: 671, linkedinLinkUrl: 'https://linkedin.com/company/simplifyhealthcare', }, { name: 'JumpCloud', domainName: 'jumpcloud.com', - address: 'Louisville', + addressAddressCity: 'Louisville', employees: 671, linkedinLinkUrl: 'https://linkedin.com/company/jumpcloud', }, { name: 'ASAP', domainName: 'asap.com', - address: 'Lafayette', + addressAddressCity: 'Lafayette', employees: 667, linkedinLinkUrl: 'https://linkedin.com/company/asap', }, { name: 'Xoxoday', domainName: 'xoxoday.com', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 666, linkedinLinkUrl: 'https://linkedin.com/company/xoxoday', }, { name: 'DataStax', domainName: 'datastax.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 666, linkedinLinkUrl: 'https://linkedin.com/company/datastax', }, { name: 'Foursquare', domainName: 'foursquare.com', - address: 'New York', + addressAddressCity: 'New York', employees: 665, linkedinLinkUrl: 'https://linkedin.com/company/foursquare', }, { name: 'LastPass', domainName: 'lastpass.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 664, linkedinLinkUrl: 'https://linkedin.com/company/lastpass', }, { name: 'SOCi, Inc.', domainName: 'meetsoci.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 664, linkedinLinkUrl: 'https://linkedin.com/company/soci-inc-', }, { name: 'Stratus Technologies', domainName: 'stratus.com', - address: 'Maynard', + addressAddressCity: 'Maynard', employees: 662, linkedinLinkUrl: 'https://linkedin.com/company/stratus-technologies', }, { name: 'AdvancedMD', domainName: 'advancedmd.com', - address: 'South Jordan', + addressAddressCity: 'South Jordan', employees: 661, linkedinLinkUrl: 'https://linkedin.com/company/advancedmd', }, { name: 'Matterport', domainName: 'matterport.com', - address: 'Sunnyvale', + addressAddressCity: 'Sunnyvale', employees: 660, linkedinLinkUrl: 'https://linkedin.com/company/matterport', }, { name: 'Samsung Research America (SRA)', domainName: 'sra.samsung.com', - address: 'Mountain View', + addressAddressCity: 'Mountain View', employees: 658, linkedinLinkUrl: 'https://linkedin.com/company/sra-samsungreasearchamerica', }, { name: 'Creatio', domainName: 'creatio.com', - address: 'Boston', + addressAddressCity: 'Boston', employees: 657, linkedinLinkUrl: 'https://linkedin.com/company/creatioglobal', }, { name: 'Branch', domainName: 'branch.io', - address: 'Palo Alto', + addressAddressCity: 'Palo Alto', employees: 657, linkedinLinkUrl: 'https://linkedin.com/company/branch-metrics', }, { name: 'Versa Networks', domainName: 'versa-networks.com', - address: 'Alviso', + addressAddressCity: 'Alviso', employees: 655, linkedinLinkUrl: 'https://linkedin.com/company/versa-networks', }, { name: 'Mitek Systems', domainName: 'miteksystems.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 653, linkedinLinkUrl: 'https://linkedin.com/company/miteksystems', }, { name: 'PDF Solutions', domainName: 'pdf.com', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 653, linkedinLinkUrl: 'https://linkedin.com/company/pdf-solutions', }, { name: 'ESO', domainName: 'eso.com', - address: 'Austin', + addressAddressCity: 'Austin', employees: 652, linkedinLinkUrl: 'https://linkedin.com/company/eso-solutions', }, { name: 'Mural', domainName: 'mural.co', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 650, linkedinLinkUrl: 'https://linkedin.com/company/mural.co', }, { name: 'FourKites, Inc.', domainName: 'fourkites.com', - address: 'Chicago', + addressAddressCity: 'Chicago', employees: 650, linkedinLinkUrl: 'https://linkedin.com/company/fourkites-inc', }, { name: 'Aras Corporation', domainName: 'aras.com', - address: 'Andover', + addressAddressCity: 'Andover', employees: 648, linkedinLinkUrl: 'https://linkedin.com/company/aras-corporation', }, { name: 'Delphix', domainName: 'delphix.com', - address: 'Redwood City', + addressAddressCity: 'Redwood City', employees: 648, linkedinLinkUrl: 'https://linkedin.com/company/delphix', }, { name: 'Wolfram', domainName: 'wolfram.com', - address: 'Champaign', + addressAddressCity: 'Champaign', employees: 644, linkedinLinkUrl: 'https://linkedin.com/company/wolfram-research', }, { name: 'Eightfold', domainName: 'eightfold.ai', - address: 'Santa Clara', + addressAddressCity: 'Santa Clara', employees: 643, linkedinLinkUrl: 'https://linkedin.com/company/eightfoldai', }, { name: 'Quark Software Inc.', domainName: 'quark.com', - address: 'Grand Rapids', + addressAddressCity: 'Grand Rapids', employees: 641, linkedinLinkUrl: 'https://linkedin.com/company/quark', }, { name: 'connectRN', domainName: 'connectrn.com', - address: 'Waltham', + addressAddressCity: 'Waltham', employees: 640, linkedinLinkUrl: 'https://linkedin.com/company/connectrn', }, { name: 'RSI', domainName: 'rsidelivers.com', - address: 'Pembroke', + addressAddressCity: 'Pembroke', employees: 638, linkedinLinkUrl: 'https://linkedin.com/company/revenue-solutions-inc-', }, { name: 'Macrosoft', domainName: 'macrosoftinc.com', - address: 'Parsippany', + addressAddressCity: 'Parsippany', employees: 638, linkedinLinkUrl: 'https://linkedin.com/company/macrosoft', }, { name: 'Paradox', domainName: 'paradox.ai', - address: 'Scottsdale', + addressAddressCity: 'Scottsdale', employees: 637, linkedinLinkUrl: 'https://linkedin.com/company/paradoxolivia', }, { name: 'SmartRecruiters', domainName: 'smartrecruiters.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 637, linkedinLinkUrl: 'https://linkedin.com/company/smartrecruiters', }, { name: 'Tealium', domainName: 'tealium.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 635, linkedinLinkUrl: 'https://linkedin.com/company/tealium', }, { name: 'Securiti', domainName: 'securiti.ai', - address: 'San Jose', + addressAddressCity: 'San Jose', employees: 634, linkedinLinkUrl: 'https://linkedin.com/company/securitiai', }, { name: 'Lattice', domainName: 'lattice.com', - address: 'San Francisco', + addressAddressCity: 'San Francisco', employees: 634, linkedinLinkUrl: 'https://linkedin.com/company/lattice-hq', }, { name: 'TuSimple', domainName: 'tusimple.com', - address: 'San Diego', + addressAddressCity: 'San Diego', employees: 633, linkedinLinkUrl: 'https://linkedin.com/company/tusimple', }, { name: 'Ceipal', domainName: 'ceipal.com', - address: 'Rochester', + addressAddressCity: 'Rochester', employees: 633, linkedinLinkUrl: 'https://linkedin.com/company/ceipal', }, { name: 'RSD', domainName: 'rocketsoftware.com', - address: 'Waltham', + addressAddressCity: 'Waltham', employees: 633, linkedinLinkUrl: 'https://linkedin.com/company/rsd', }, diff --git a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/company.ts b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/company.ts index 49834df3933a..7476368f9ae3 100644 --- a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/company.ts +++ b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/company.ts @@ -12,7 +12,7 @@ export const companyPrefillDemoData = async ( .into(`${schemaName}.company`, [ 'name', 'domainName', - 'address', + 'addressAddressCity', 'employees', 'linkedinLinkUrl', 'position', diff --git a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/opportunity.ts b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/opportunity.ts index e2d53abfb311..23c51142980f 100644 --- a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/opportunity.ts +++ b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/opportunity.ts @@ -3,12 +3,6 @@ import { v4 } from 'uuid'; const tableName = 'opportunity'; -const getRandomProbability = () => { - const firstDigit = Math.floor(Math.random() * 9) + 1; - - return firstDigit / 10; -}; - const getRandomStage = () => { const stages = ['NEW', 'SCREENING', 'MEETING', 'PROPOSAL', 'CUSTOMER']; @@ -28,7 +22,6 @@ const generateOpportunities = (companies) => { amountCurrencyCode: 'USD', closeDate: new Date(), stage: getRandomStage(), - probability: getRandomProbability(), pointOfContactId: company.personId, companyId: company.id, })); @@ -56,7 +49,6 @@ export const opportunityPrefillDemoData = async ( 'amountCurrencyCode', 'closeDate', 'stage', - 'probability', 'pointOfContactId', 'companyId', 'position', diff --git a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/people-demo.json.ts b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/people-demo.json.ts index 2bd808d0bc65..0af53c10f23f 100644 --- a/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/people-demo.json.ts +++ b/packages/twenty-server/src/engine/workspace-manager/demo-objects-prefill-data/people-demo.json.ts @@ -5,7 +5,7 @@ export const peopleDemo = [ city: 'West Justin', email: 'mark.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/mark-young-7edccf6aca', jobTitle: 'Surveyor, minerals', }, @@ -15,7 +15,7 @@ export const peopleDemo = [ city: 'Larryview', email: 'gabriel.robinson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/gabriel-robinson-3157ccba23', jobTitle: 'Armed forces logistics/support/administrative officer', }, @@ -25,7 +25,7 @@ export const peopleDemo = [ city: 'Victoriamouth', email: 'kimberly.gordon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/kimberly-gordon-4a10fde4c9', jobTitle: 'Engineer, manufacturing systems', }, @@ -35,7 +35,7 @@ export const peopleDemo = [ city: 'Franciscoland', email: 'cindy.baker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/cindy-baker-788ab17f8b', jobTitle: 'Learning disability nurse', }, @@ -45,7 +45,7 @@ export const peopleDemo = [ city: 'South Kaitlin', email: 'anthony.may@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/anthony-may-2020930433', jobTitle: 'Optometrist', }, @@ -55,7 +55,7 @@ export const peopleDemo = [ city: 'New Margaretshire', email: 'vicki.meyer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/vicki-meyer-f2e0fdfbd9', jobTitle: 'Farm manager', }, @@ -65,7 +65,7 @@ export const peopleDemo = [ city: 'Clayton', email: 'billy.mckinney@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/billy-mckinney-709e41f9ba', jobTitle: 'Therapist, nutritional', }, @@ -75,7 +75,7 @@ export const peopleDemo = [ city: 'New Markborough', email: 'andrew.king@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/andrew-king-9eee067c59', jobTitle: 'Paramedic', }, @@ -85,7 +85,7 @@ export const peopleDemo = [ city: 'West Aaronchester', email: 'todd.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/todd-jones-d1cae42f61', jobTitle: 'Media planner', }, @@ -95,7 +95,7 @@ export const peopleDemo = [ city: 'New Cassiechester', email: 'gregory.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/gregory-perez-5ca5d506c0', jobTitle: 'Special effects artist', }, @@ -105,7 +105,7 @@ export const peopleDemo = [ city: 'Gordonhaven', email: 'vanessa.farmer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/vanessa-farmer-c79ab76e62', jobTitle: 'Engineer, land', }, @@ -115,7 +115,7 @@ export const peopleDemo = [ city: 'North Amy', email: 'elizabeth.chung@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/elizabeth-chung-72c8e6d73e', jobTitle: 'Race relations officer', }, @@ -125,7 +125,7 @@ export const peopleDemo = [ city: 'North Kristopher', email: 'melissa.huerta@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/melissa-huerta-65292000ee', jobTitle: 'Museum/gallery exhibitions officer', }, @@ -135,7 +135,7 @@ export const peopleDemo = [ city: 'West Oliviaburgh', email: 'debbie.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/debbie-johnson-6108ee5a49', jobTitle: 'Wellsite geologist', }, @@ -145,7 +145,7 @@ export const peopleDemo = [ city: 'Barretttown', email: 'kathy.mcclain@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/kathy-mcclain-cf5890c5b5', jobTitle: 'Surveyor, building control', }, @@ -155,7 +155,7 @@ export const peopleDemo = [ city: 'Cassidyburgh', email: 'michael.elliott@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/michael-elliott-d5e13ac5c8', jobTitle: 'Ergonomist', }, @@ -165,7 +165,7 @@ export const peopleDemo = [ city: 'Wareport', email: 'kimberly.edwards@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/kimberly-edwards-00862e876a', jobTitle: 'Exercise physiologist', }, @@ -175,7 +175,7 @@ export const peopleDemo = [ city: 'Jefferyport', email: 'regina.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/regina-williams-9d91d1682f', jobTitle: 'Clinical scientist, histocompatibility and immunogenetics', }, @@ -185,7 +185,7 @@ export const peopleDemo = [ city: 'Ericaland', email: 'john.guerrero@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/john-guerrero-f5c763a584', jobTitle: 'Wellsite geologist', }, @@ -195,7 +195,7 @@ export const peopleDemo = [ city: 'Jamesborough', email: 'david.bailey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/david-bailey-a321ec2517', jobTitle: 'Radiographer, therapeutic', }, @@ -205,7 +205,7 @@ export const peopleDemo = [ city: 'Calvinton', email: 'emily.davidson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/emily-davidson-4cfca34af8', jobTitle: 'Health visitor', }, @@ -215,7 +215,7 @@ export const peopleDemo = [ city: 'South Veronica', email: 'michelle.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/michelle-jackson-bcb3423e3e', jobTitle: 'Social research officer, government', }, @@ -225,7 +225,7 @@ export const peopleDemo = [ city: 'North Nicole', email: 'ryan.romero@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/ryan-romero-36790b1367', jobTitle: 'Forest/woodland manager', }, @@ -235,7 +235,7 @@ export const peopleDemo = [ city: 'Spencemouth', email: 'victor.lewis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/victor-lewis-71ac9f14ee', jobTitle: 'Surgeon', }, @@ -245,7 +245,7 @@ export const peopleDemo = [ city: 'Jacksonhaven', email: 'christopher.powell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/christopher-powell-2e521c68f3', jobTitle: 'Hydrogeologist', }, @@ -255,7 +255,7 @@ export const peopleDemo = [ city: 'South Jacqueline', email: 'jack.george@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/jack-george-1b6352407c', jobTitle: 'Engineer, site', }, @@ -265,7 +265,7 @@ export const peopleDemo = [ city: 'Markchester', email: 'manuel.lara@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/manuel-lara-f896ffc5d1', jobTitle: 'Government social research officer', }, @@ -275,7 +275,7 @@ export const peopleDemo = [ city: 'Brianfurt', email: 'john.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/john-gonzalez-1077ebc9e6', jobTitle: 'Horticultural therapist', }, @@ -285,7 +285,7 @@ export const peopleDemo = [ city: 'North Christian', email: 'theodore.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/theodore-gonzalez-b0caf15fb1', jobTitle: 'Administrator', }, @@ -295,7 +295,7 @@ export const peopleDemo = [ city: 'North Jacob', email: 'christine.bishop@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/christine-bishop-c6ec520c6c', jobTitle: 'Geneticist, molecular', }, @@ -305,7 +305,7 @@ export const peopleDemo = [ city: 'Cooperport', email: 'alejandro.moran@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/alejandro-moran-03db39c63a', jobTitle: 'Applications developer', }, @@ -315,7 +315,7 @@ export const peopleDemo = [ city: 'Carmenchester', email: 'john.cook@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/john-cook-ced58e0bb5', jobTitle: 'Chief Marketing Officer', }, @@ -325,7 +325,7 @@ export const peopleDemo = [ city: 'Claudiaborough', email: 'leslie.calderon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/leslie-calderon-79bad778f2', jobTitle: 'Teacher, music', }, @@ -335,7 +335,7 @@ export const peopleDemo = [ city: 'Gibsontown', email: 'barbara.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/barbara-young-d2423e1be3', jobTitle: 'Television floor manager', }, @@ -345,7 +345,7 @@ export const peopleDemo = [ city: 'Alyssastad', email: 'maria.thomas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/maria-thomas-833d46722e', jobTitle: 'Investment banker, operational', }, @@ -355,7 +355,7 @@ export const peopleDemo = [ city: 'North Christopher', email: 'paul.villegas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/paul-villegas-dda25fb766', jobTitle: 'Veterinary surgeon', }, @@ -365,7 +365,7 @@ export const peopleDemo = [ city: 'Williamsland', email: 'bradley.turner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/bradley-turner-eb4dd1bbce', jobTitle: 'Financial controller', }, @@ -375,7 +375,7 @@ export const peopleDemo = [ city: 'Whitemouth', email: 'matthew.alexander@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/matthew-alexander-57352bb034', jobTitle: 'Engineer, electrical', }, @@ -385,7 +385,7 @@ export const peopleDemo = [ city: 'Josephberg', email: 'nancy.green@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/nancy-green-08c0a785dc', jobTitle: 'Horticulturist, amenity', }, @@ -395,7 +395,7 @@ export const peopleDemo = [ city: 'Lake Jamesside', email: 'cindy.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/cindy-martin-cef98190f9', jobTitle: 'Geographical information systems officer', }, @@ -405,7 +405,7 @@ export const peopleDemo = [ city: 'New Tracy', email: 'lori.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/lori-martin-119def9345', jobTitle: 'Logistics and distribution manager', }, @@ -415,7 +415,7 @@ export const peopleDemo = [ city: 'North Joeborough', email: 'kathryn.cruz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/kathryn-cruz-602e728f69', jobTitle: 'Designer, graphic', }, @@ -425,7 +425,7 @@ export const peopleDemo = [ city: 'Lake Jeffrey', email: 'robert.terry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/robert-terry-8967a9c9ba', jobTitle: 'Sub', }, @@ -435,7 +435,7 @@ export const peopleDemo = [ city: 'New Hannahland', email: 'andrea.walker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/andrea-walker-1047d1ba76', jobTitle: 'Financial adviser', }, @@ -445,7 +445,7 @@ export const peopleDemo = [ city: 'Tylermouth', email: 'steve.campos@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/steve-campos-1225440c77', jobTitle: 'Osteopath', }, @@ -455,7 +455,7 @@ export const peopleDemo = [ city: 'New Annafort', email: 'allison.morgan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/allison-morgan-6486ee71fe', jobTitle: 'Microbiologist', }, @@ -465,7 +465,7 @@ export const peopleDemo = [ city: 'Samanthabury', email: 'tamara.melendez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/tamara-melendez-33bb698d07', jobTitle: 'Accounting technician', }, @@ -475,7 +475,7 @@ export const peopleDemo = [ city: 'Port Howard', email: 'larry.robertson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/larry-robertson-6829bf55c6', jobTitle: 'Production assistant, television', }, @@ -485,7 +485,7 @@ export const peopleDemo = [ city: 'Rachelmouth', email: 'lisa.cook@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/lisa-cook-87669acc77', jobTitle: 'Clinical cytogeneticist', }, @@ -495,7 +495,7 @@ export const peopleDemo = [ city: 'Marvinborough', email: 'kirsten.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/kirsten-moore-91e3033de9', jobTitle: 'Media buyer', }, @@ -505,7 +505,7 @@ export const peopleDemo = [ city: 'Mcbridemouth', email: 'amanda.frye@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/amanda-frye-1f5377943e', jobTitle: 'Metallurgist', }, @@ -515,7 +515,7 @@ export const peopleDemo = [ city: 'New Troy', email: 'jennifer.chambers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/jennifer-chambers-ca724f5258', jobTitle: 'Agricultural engineer', }, @@ -525,7 +525,7 @@ export const peopleDemo = [ city: 'South Alexandra', email: 'rodney.roberts@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/rodney-roberts-4b08e437c4', jobTitle: 'Recycling officer', }, @@ -535,7 +535,7 @@ export const peopleDemo = [ city: 'North Erinton', email: 'lindsay.wagner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/lindsay-wagner-0562c96aa1', jobTitle: 'Psychotherapist, child', }, @@ -545,7 +545,7 @@ export const peopleDemo = [ city: 'East Patricia', email: 'mary.haynes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/mary-haynes-a41c2d1798', jobTitle: 'Arts development officer', }, @@ -555,7 +555,7 @@ export const peopleDemo = [ city: 'New Kaylee', email: 'david.phelps@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/david-phelps-0dbf8cb8d9', jobTitle: 'IT technical support officer', }, @@ -565,7 +565,7 @@ export const peopleDemo = [ city: 'East Bruce', email: 'patricia.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/patricia-smith-5277401ff1', jobTitle: 'Scientist, biomedical', }, @@ -575,7 +575,7 @@ export const peopleDemo = [ city: 'Port Crystalbury', email: 'rachel.morse@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/rachel-morse-86b5f8b59c', jobTitle: 'Patent attorney', }, @@ -585,7 +585,7 @@ export const peopleDemo = [ city: 'Mcdowellside', email: 'rhonda.nelson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/rhonda-nelson-0f7ce0e497', jobTitle: 'Environmental manager', }, @@ -595,7 +595,7 @@ export const peopleDemo = [ city: 'North Vanessaport', email: 'lauren.carroll@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/lauren-carroll-319b96609a', jobTitle: 'Statistician', }, @@ -605,7 +605,7 @@ export const peopleDemo = [ city: 'South Rachelmouth', email: 'shannon.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/shannon-martinez-5f73530423', jobTitle: 'Designer, blown glass/stained glass', }, @@ -615,7 +615,7 @@ export const peopleDemo = [ city: 'New John', email: 'daniel.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/daniel-williams-f8d496db83', jobTitle: 'Nature conservation officer', }, @@ -625,7 +625,7 @@ export const peopleDemo = [ city: 'Arthurfurt', email: 'willie.cannon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/willie-cannon-44b80be6a1', jobTitle: 'Engineer, electronics', }, @@ -635,7 +635,7 @@ export const peopleDemo = [ city: 'South Kellietown', email: 'donna.cole@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/donna-cole-2daf2f491d', jobTitle: 'Land/geomatics surveyor', }, @@ -645,7 +645,7 @@ export const peopleDemo = [ city: 'Estradaborough', email: 'morgan.cook@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/morgan-cook-82f95695fe', jobTitle: 'Research officer, political party', }, @@ -655,7 +655,7 @@ export const peopleDemo = [ city: 'Colemanville', email: 'elizabeth.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/elizabeth-smith-205530e011', jobTitle: 'Prison officer', }, @@ -665,7 +665,7 @@ export const peopleDemo = [ city: 'East Jared', email: 'nathaniel.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/nathaniel-johnson-847d2defe7', jobTitle: 'Geochemist', }, @@ -675,7 +675,7 @@ export const peopleDemo = [ city: 'Barkerchester', email: 'rebecca.elliott@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/rebecca-elliott-033486b7fa', jobTitle: 'Fine artist', }, @@ -685,7 +685,7 @@ export const peopleDemo = [ city: 'Vargasshire', email: 'kristina.olson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/kristina-olson-be254bb623', jobTitle: 'Warden/ranger', }, @@ -695,7 +695,7 @@ export const peopleDemo = [ city: 'Palmerfurt', email: 'robert.henderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/robert-henderson-39d03b4d6f', jobTitle: 'Video editor', }, @@ -705,7 +705,7 @@ export const peopleDemo = [ city: 'East Travisberg', email: 'kendra.knox@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/kendra-knox-8b8db240fa', jobTitle: 'Conservation officer, nature', }, @@ -715,7 +715,7 @@ export const peopleDemo = [ city: 'Woodburgh', email: 'donna.jacobs@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/donna-jacobs-75f4cb3e8a', jobTitle: 'Wellsite geologist', }, @@ -725,7 +725,7 @@ export const peopleDemo = [ city: 'North Andrea', email: 'michael.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/michael-martinez-37a2df073b', jobTitle: 'Scientist, water quality', }, @@ -735,7 +735,7 @@ export const peopleDemo = [ city: 'Wendyfurt', email: 'natalie.hansen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/natalie-hansen-eba76059eb', jobTitle: 'Designer, furniture', }, @@ -745,7 +745,7 @@ export const peopleDemo = [ city: 'New Stephanie', email: 'katie.russo@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/katie-russo-2a475932df', jobTitle: 'Tourism officer', }, @@ -755,7 +755,7 @@ export const peopleDemo = [ city: 'Jefferyton', email: 'danielle.park@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/danielle-park-fadefa41f3', jobTitle: 'Transport planner', }, @@ -765,7 +765,7 @@ export const peopleDemo = [ city: 'East Gabrielborough', email: 'nicholas.guzman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/nicholas-guzman-41594d6dc9', jobTitle: 'Scientist, research (medical)', }, @@ -775,7 +775,7 @@ export const peopleDemo = [ city: 'Port Jenniferstad', email: 'brandi.dodson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/brandi-dodson-428d92f283', jobTitle: 'Therapist, drama', }, @@ -785,7 +785,7 @@ export const peopleDemo = [ city: 'New Charlesfurt', email: 'sara.kane@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/sara-kane-778b92a3ff', jobTitle: 'Associate Professor', }, @@ -795,7 +795,7 @@ export const peopleDemo = [ city: 'Port Kurt', email: 'allison.howard@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/allison-howard-5cc9c06425', jobTitle: 'Hydrographic surveyor', }, @@ -805,7 +805,7 @@ export const peopleDemo = [ city: 'Lake William', email: 'jonathan.drake@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/jonathan-drake-5ef0430b02', jobTitle: 'Learning disability nurse', }, @@ -815,7 +815,7 @@ export const peopleDemo = [ city: 'South Shelly', email: 'samantha.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/samantha-williams-8b316b4a9d', jobTitle: 'Technical brewer', }, @@ -825,7 +825,7 @@ export const peopleDemo = [ city: 'North Lori', email: 'katherine.mooney@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/katherine-mooney-ef2c2c12dd', jobTitle: 'Accountant, chartered', }, @@ -835,7 +835,7 @@ export const peopleDemo = [ city: 'Roberttown', email: 'luis.lloyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/luis-lloyd-ee56c3462c', jobTitle: 'Producer, television/film/video', }, @@ -845,7 +845,7 @@ export const peopleDemo = [ city: 'Matthewchester', email: 'travis.serrano@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/travis-serrano-789399815b', jobTitle: 'Therapist, art', }, @@ -855,7 +855,7 @@ export const peopleDemo = [ city: 'Hayesstad', email: 'amy.newton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/amy-newton-ff6b79dfce', jobTitle: 'Exhibition designer', }, @@ -865,7 +865,7 @@ export const peopleDemo = [ city: 'Parkerland', email: 'jonathan.hawkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/jonathan-hawkins-c68fe4ecec', jobTitle: 'Surveyor, land/geomatics', }, @@ -875,7 +875,7 @@ export const peopleDemo = [ city: 'East Micheal', email: 'patricia.anthony@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/patricia-anthony-64ce02febc', jobTitle: 'Paediatric nurse', }, @@ -885,7 +885,7 @@ export const peopleDemo = [ city: 'Knightport', email: 'matthew.gomez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/matthew-gomez-68cc301b54', jobTitle: 'Site engineer', }, @@ -895,7 +895,7 @@ export const peopleDemo = [ city: 'West Patrickshire', email: 'jonathan.schultz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/jonathan-schultz-3c0c1ecc59', jobTitle: 'Sports therapist', }, @@ -905,7 +905,7 @@ export const peopleDemo = [ city: 'Longchester', email: 'matthew.cummings@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/matthew-cummings-27d772de78', jobTitle: 'Designer, interior/spatial', }, @@ -915,7 +915,7 @@ export const peopleDemo = [ city: 'New Ashley', email: 'joshua.richards@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/joshua-richards-ec2044ed2a', jobTitle: 'Civil engineer, consulting', }, @@ -925,7 +925,7 @@ export const peopleDemo = [ city: 'Reginaville', email: 'ryan.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/ryan-johnson-7dda846caa', jobTitle: 'Office manager', }, @@ -935,7 +935,7 @@ export const peopleDemo = [ city: 'Lake Mandy', email: 'teresa.terrell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/teresa-terrell-fe92ba9d84', jobTitle: "Barrister's clerk", }, @@ -945,7 +945,7 @@ export const peopleDemo = [ city: 'North Haley', email: 'jacob.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/jacob-jenkins-b378edc103', jobTitle: 'Lecturer, higher education', }, @@ -955,7 +955,7 @@ export const peopleDemo = [ city: 'Lake Deborah', email: 'michael.myers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/michael-myers-5d468b5c78', jobTitle: 'Psychologist, clinical', }, @@ -965,7 +965,7 @@ export const peopleDemo = [ city: 'Daltonstad', email: 'jennifer.phillips@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/jennifer-phillips-83c06e63a9', jobTitle: 'Immunologist', }, @@ -975,7 +975,7 @@ export const peopleDemo = [ city: 'Batesside', email: 'alison.ortega@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/alison-ortega-0b7bad2804', jobTitle: 'Professor Emeritus', }, @@ -985,7 +985,7 @@ export const peopleDemo = [ city: 'Jamesstad', email: 'gregory.little@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/gregory-little-bf88b7a274', jobTitle: 'Ship broker', }, @@ -995,7 +995,7 @@ export const peopleDemo = [ city: 'New Melanieberg', email: 'barry.finley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/barry-finley-72ad288ea6', jobTitle: 'Network engineer', }, @@ -1005,7 +1005,7 @@ export const peopleDemo = [ city: 'Port Jeremy', email: 'brian.evans@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/brian-evans-7848e0a7de', jobTitle: 'Information officer', }, @@ -1015,7 +1015,7 @@ export const peopleDemo = [ city: 'Parsonsberg', email: 'troy.davidson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/troy-davidson-20a1718421', jobTitle: 'Civil Service fast streamer', }, @@ -1025,7 +1025,7 @@ export const peopleDemo = [ city: 'South Stevenbury', email: 'aaron.schroeder@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/aaron-schroeder-c0b91178c9', jobTitle: 'Editorial assistant', }, @@ -1035,7 +1035,7 @@ export const peopleDemo = [ city: 'South Jenniferhaven', email: 'mary.anderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/mary-anderson-17e7aeb95e', jobTitle: 'Quality manager', }, @@ -1045,7 +1045,7 @@ export const peopleDemo = [ city: 'West Jeremystad', email: 'david.obrien@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/david-obrien-5393bdf2d9', jobTitle: 'Air cabin crew', }, @@ -1055,7 +1055,7 @@ export const peopleDemo = [ city: 'South Daniel', email: 'colin.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/colin-miller-fdfec5eb0f', jobTitle: 'Museum/gallery conservator', }, @@ -1065,7 +1065,7 @@ export const peopleDemo = [ city: 'West Kathy', email: 'jesus.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/jesus-johnson-8a2b7bb431', jobTitle: 'Radio broadcast assistant', }, @@ -1075,7 +1075,7 @@ export const peopleDemo = [ city: 'Desireemouth', email: 'brooke.henderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/brooke-henderson-98aee4e9a6', jobTitle: 'Market researcher', }, @@ -1085,7 +1085,7 @@ export const peopleDemo = [ city: 'Lynchstad', email: 'meredith.gregory@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/meredith-gregory-a3f977f6ef', jobTitle: 'Environmental manager', }, @@ -1095,7 +1095,7 @@ export const peopleDemo = [ city: 'Lewisfurt', email: 'crystal.vaughn@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/crystal-vaughn-2886394a50', jobTitle: 'Operations geologist', }, @@ -1105,7 +1105,7 @@ export const peopleDemo = [ city: 'South Hollyfurt', email: 'william.greene@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/william-greene-da956010f3', jobTitle: 'Surveyor, commercial/residential', }, @@ -1115,7 +1115,7 @@ export const peopleDemo = [ city: 'West Darrellshire', email: 'aaron.griffin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/aaron-griffin-361a228e01', jobTitle: 'Engineer, maintenance (IT)', }, @@ -1125,7 +1125,7 @@ export const peopleDemo = [ city: 'East Shane', email: 'steven.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/steven-smith-6e762c7e90', jobTitle: 'Psychologist, forensic', }, @@ -1135,7 +1135,7 @@ export const peopleDemo = [ city: 'East James', email: 'mark.faulkner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/mark-faulkner-5b751ca394', jobTitle: 'Nurse, adult', }, @@ -1145,7 +1145,7 @@ export const peopleDemo = [ city: 'South Victoria', email: 'jeffrey.hunt@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/jeffrey-hunt-c364e07096', jobTitle: 'Dramatherapist', }, @@ -1155,7 +1155,7 @@ export const peopleDemo = [ city: 'Popehaven', email: 'tara.mathis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/tara-mathis-1d1d3c04ed', jobTitle: 'Surveyor, minerals', }, @@ -1165,7 +1165,7 @@ export const peopleDemo = [ city: 'Brandonburgh', email: 'anna.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/anna-davis-ec88f7642f', jobTitle: 'Programme researcher, broadcasting/film/video', }, @@ -1175,7 +1175,7 @@ export const peopleDemo = [ city: 'Port Alyssaland', email: 'kevin.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/kevin-johnson-c2a28b524a', jobTitle: 'Fast food restaurant manager', }, @@ -1185,7 +1185,7 @@ export const peopleDemo = [ city: 'New Sarahberg', email: 'sergio.glenn@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/sergio-glenn-5cdb7d803d', jobTitle: 'Surgeon', }, @@ -1195,7 +1195,7 @@ export const peopleDemo = [ city: 'Logantown', email: 'nicole.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/nicole-allen-d6ccf222ff', jobTitle: 'Fine artist', }, @@ -1205,7 +1205,7 @@ export const peopleDemo = [ city: 'Gonzaleztown', email: 'christopher.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/christopher-jones-7b60309d81', jobTitle: 'Visual merchandiser', }, @@ -1215,7 +1215,7 @@ export const peopleDemo = [ city: 'West Ericville', email: 'brandon.sanchez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/brandon-sanchez-f78c3a4268', jobTitle: 'Radio producer', }, @@ -1225,7 +1225,7 @@ export const peopleDemo = [ city: 'Kathleenhaven', email: 'cindy.schmidt@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/cindy-schmidt-19ce5324d9', jobTitle: 'Production assistant, television', }, @@ -1235,7 +1235,7 @@ export const peopleDemo = [ city: 'Stephensonville', email: 'john.gillespie@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/john-gillespie-fa4d7d67f3', jobTitle: 'Television/film/video producer', }, @@ -1245,7 +1245,7 @@ export const peopleDemo = [ city: 'East Mercedesbury', email: 'andrew.lyons@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/andrew-lyons-02809c0ff1', jobTitle: 'Accountant, chartered public finance', }, @@ -1255,7 +1255,7 @@ export const peopleDemo = [ city: 'East Amandamouth', email: 'joseph.willis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/joseph-willis-242605ba96', jobTitle: 'Accommodation manager', }, @@ -1265,7 +1265,7 @@ export const peopleDemo = [ city: 'Sabrinahaven', email: 'charles.stanton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/charles-stanton-89aca5d45a', jobTitle: 'Music tutor', }, @@ -1275,7 +1275,7 @@ export const peopleDemo = [ city: 'Lake Barbarachester', email: 'gary.hall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/gary-hall-1ec55a0077', jobTitle: 'Theme park manager', }, @@ -1285,7 +1285,7 @@ export const peopleDemo = [ city: 'New Seanstad', email: 'steven.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/steven-martin-82c1e4ee0a', jobTitle: 'Artist', }, @@ -1295,7 +1295,7 @@ export const peopleDemo = [ city: 'Colemanton', email: 'jennifer.mcgee@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/jennifer-mcgee-f4f8b04f40', jobTitle: 'Special effects artist', }, @@ -1305,7 +1305,7 @@ export const peopleDemo = [ city: 'Thompsonhaven', email: 'bonnie.warren@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/bonnie-warren-c433803c71', jobTitle: 'Lobbyist', }, @@ -1315,7 +1315,7 @@ export const peopleDemo = [ city: 'Gloriahaven', email: 'gregory.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/gregory-martinez-81489f760c', jobTitle: 'Secretary, company', }, @@ -1325,7 +1325,7 @@ export const peopleDemo = [ city: 'Gregorymouth', email: 'bradley.randall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/bradley-randall-00c2f788ab', jobTitle: 'Product manager', }, @@ -1335,7 +1335,7 @@ export const peopleDemo = [ city: 'South Eileen', email: 'brian.steele@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/brian-steele-fe89252d69', jobTitle: 'Sound technician, broadcasting/film/video', }, @@ -1345,7 +1345,7 @@ export const peopleDemo = [ city: 'New Jenny', email: 'ann.mercer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/ann-mercer-e7eedca5ab', jobTitle: 'Editorial assistant', }, @@ -1355,7 +1355,7 @@ export const peopleDemo = [ city: 'New Benjaminmouth', email: 'billy.fuentes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/billy-fuentes-2b9e1e559f', jobTitle: 'Energy manager', }, @@ -1365,7 +1365,7 @@ export const peopleDemo = [ city: 'Bennettberg', email: 'kelsey.palmer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/kelsey-palmer-25e5d90ac8', jobTitle: 'Medical secretary', }, @@ -1375,7 +1375,7 @@ export const peopleDemo = [ city: 'Castroshire', email: 'ryan.holmes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/ryan-holmes-0c335917d6', jobTitle: 'Armed forces operational officer', }, @@ -1385,7 +1385,7 @@ export const peopleDemo = [ city: 'New Sara', email: 'larry.castro@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/larry-castro-631a1061ce', jobTitle: 'Call centre manager', }, @@ -1395,7 +1395,7 @@ export const peopleDemo = [ city: 'Spencerstad', email: 'elizabeth.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/elizabeth-gonzalez-7717b4d0bd', jobTitle: 'Automotive engineer', }, @@ -1405,7 +1405,7 @@ export const peopleDemo = [ city: 'South Erintown', email: 'christopher.matthews@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/christopher-matthews-28232e7783', jobTitle: 'Surveyor, building', }, @@ -1415,7 +1415,7 @@ export const peopleDemo = [ city: 'Parkerview', email: 'rodney.briggs@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/rodney-briggs-634d20e650', jobTitle: 'Clinical cytogeneticist', }, @@ -1425,7 +1425,7 @@ export const peopleDemo = [ city: 'Haynesborough', email: 'donald.khan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/donald-khan-5aae10186f', jobTitle: 'Surveyor, rural practice', }, @@ -1435,7 +1435,7 @@ export const peopleDemo = [ city: 'South Katherine', email: 'spencer.lee@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/spencer-lee-030eb9cd9d', jobTitle: 'Multimedia programmer', }, @@ -1445,7 +1445,7 @@ export const peopleDemo = [ city: 'Powellshire', email: 'katherine.parker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/katherine-parker-87afc741ec', jobTitle: 'Arts development officer', }, @@ -1455,7 +1455,7 @@ export const peopleDemo = [ city: 'South Jasonberg', email: 'amanda.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/amanda-jackson-31a0288c40', jobTitle: 'Engineer, energy', }, @@ -1465,7 +1465,7 @@ export const peopleDemo = [ city: 'South Paul', email: 'kimberly.lloyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/kimberly-lloyd-851cd20ebf', jobTitle: 'Armed forces operational officer', }, @@ -1475,7 +1475,7 @@ export const peopleDemo = [ city: 'Port David', email: 'eric.hunter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/eric-hunter-a131354d18', jobTitle: 'Call centre manager', }, @@ -1485,7 +1485,7 @@ export const peopleDemo = [ city: 'Port Davidberg', email: 'ashley.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/ashley-taylor-4aa9cb790c', jobTitle: 'Arts administrator', }, @@ -1495,7 +1495,7 @@ export const peopleDemo = [ city: 'South Anna', email: 'michael.ayers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/michael-ayers-a7a9c15b39', jobTitle: 'Community pharmacist', }, @@ -1505,7 +1505,7 @@ export const peopleDemo = [ city: 'Port Rhondaton', email: 'stephen.fisher@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/stephen-fisher-696b8604f2', jobTitle: 'Equities trader', }, @@ -1515,7 +1515,7 @@ export const peopleDemo = [ city: 'Calderonshire', email: 'kara.james@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/kara-james-5cf607e6c1', jobTitle: 'Maintenance engineer', }, @@ -1525,7 +1525,7 @@ export const peopleDemo = [ city: 'South Toddview', email: 'gary.lamb@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/gary-lamb-c0f66f8f75', jobTitle: 'General practice doctor', }, @@ -1535,7 +1535,7 @@ export const peopleDemo = [ city: 'Laurahaven', email: 'james.griffin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/james-griffin-8a5e63d24f', jobTitle: 'Public relations officer', }, @@ -1545,7 +1545,7 @@ export const peopleDemo = [ city: 'New Brandonton', email: 'wanda.chambers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/wanda-chambers-8ea5f97b07', jobTitle: 'Advertising account planner', }, @@ -1555,7 +1555,7 @@ export const peopleDemo = [ city: 'New Meganberg', email: 'lisa.kline@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/lisa-kline-513a938279', jobTitle: 'Actuary', }, @@ -1565,7 +1565,7 @@ export const peopleDemo = [ city: 'South Ronald', email: 'jason.roberts@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/jason-roberts-a33d0ea932', jobTitle: 'Tourism officer', }, @@ -1575,7 +1575,7 @@ export const peopleDemo = [ city: 'Port Kevinbury', email: 'john.mcpherson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/john-mcpherson-0d3f263d26', jobTitle: 'Editor, magazine features', }, @@ -1585,7 +1585,7 @@ export const peopleDemo = [ city: 'Brandonchester', email: 'karen.rhodes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/karen-rhodes-239af63ac9', jobTitle: 'Commercial art gallery manager', }, @@ -1595,7 +1595,7 @@ export const peopleDemo = [ city: 'Meganhaven', email: 'kathy.sparks@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/kathy-sparks-3dcdea4c5a', jobTitle: 'Engineer, production', }, @@ -1605,7 +1605,7 @@ export const peopleDemo = [ city: 'East Bryanshire', email: 'richard.murphy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/richard-murphy-66d2f794a5', jobTitle: 'Osteopath', }, @@ -1615,7 +1615,7 @@ export const peopleDemo = [ city: 'Gardnerhaven', email: 'nicole.peterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/nicole-peterson-2d8cc71386', jobTitle: 'Production engineer', }, @@ -1625,7 +1625,7 @@ export const peopleDemo = [ city: 'Port Peter', email: 'bryan.ward@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/bryan-ward-9146d84428', jobTitle: 'Interior and spatial designer', }, @@ -1635,7 +1635,7 @@ export const peopleDemo = [ city: 'North Ryanport', email: 'rebecca.howell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/rebecca-howell-b51b44e75e', jobTitle: 'Engineer, electronics', }, @@ -1645,7 +1645,7 @@ export const peopleDemo = [ city: 'Port Nicoleshire', email: 'lori.bean@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/lori-bean-adcfd993d8', jobTitle: 'Designer, blown glass/stained glass', }, @@ -1655,7 +1655,7 @@ export const peopleDemo = [ city: 'South Mary', email: 'kevin.eaton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/kevin-eaton-16b640da6b', jobTitle: 'Higher education lecturer', }, @@ -1665,7 +1665,7 @@ export const peopleDemo = [ city: 'West Spencerville', email: 'nicholas.wright@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/nicholas-wright-6c27afa8b3', jobTitle: 'Environmental manager', }, @@ -1675,7 +1675,7 @@ export const peopleDemo = [ city: 'West Sarahshire', email: 'kevin.paul@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/kevin-paul-12c8e885e9', jobTitle: 'Health and safety inspector', }, @@ -1685,7 +1685,7 @@ export const peopleDemo = [ city: 'Stevensshire', email: 'joshua.black@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/joshua-black-08413d2634', jobTitle: 'Banker', }, @@ -1695,7 +1695,7 @@ export const peopleDemo = [ city: 'Kington', email: 'scott.bruce@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/scott-bruce-f345ae71e4', jobTitle: 'Educational psychologist', }, @@ -1705,7 +1705,7 @@ export const peopleDemo = [ city: 'Brownfort', email: 'ebony.nixon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/ebony-nixon-6ce991d391', jobTitle: 'Arboriculturist', }, @@ -1715,7 +1715,7 @@ export const peopleDemo = [ city: 'Kathrynton', email: 'jesse.hartman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/jesse-hartman-f27f702502', jobTitle: 'Chiropractor', }, @@ -1725,7 +1725,7 @@ export const peopleDemo = [ city: 'South Matthewton', email: 'julie.whitney@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/julie-whitney-527c01e206', jobTitle: 'Leisure centre manager', }, @@ -1735,7 +1735,7 @@ export const peopleDemo = [ city: 'North Andrewbury', email: 'barbara.diaz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/barbara-diaz-5f4a19157d', jobTitle: 'Press photographer', }, @@ -1745,7 +1745,7 @@ export const peopleDemo = [ city: 'East Wandaport', email: 'jordan.montoya@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/jordan-montoya-f49fb6a720', jobTitle: 'Broadcast engineer', }, @@ -1755,7 +1755,7 @@ export const peopleDemo = [ city: 'Ashleyburgh', email: 'jorge.hanson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/jorge-hanson-c899de111f', jobTitle: 'General practice doctor', }, @@ -1765,7 +1765,7 @@ export const peopleDemo = [ city: 'Port Shawn', email: 'anna.robbins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/anna-robbins-8ddf20e83c', jobTitle: 'Nurse, mental health', }, @@ -1775,7 +1775,7 @@ export const peopleDemo = [ city: 'East Eric', email: 'steve.keller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/steve-keller-377272ac44', jobTitle: 'Industrial buyer', }, @@ -1785,7 +1785,7 @@ export const peopleDemo = [ city: 'New Georgeport', email: 'brianna.moreno@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/brianna-moreno-dae63a4f83', jobTitle: 'General practice doctor', }, @@ -1795,7 +1795,7 @@ export const peopleDemo = [ city: 'South Stephanie', email: 'shawn.krause@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/shawn-krause-f42275b298', jobTitle: 'Insurance broker', }, @@ -1805,7 +1805,7 @@ export const peopleDemo = [ city: 'New Jorgeland', email: 'michael.pierce@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/michael-pierce-b01914124c', jobTitle: 'Engineer, broadcasting (operations)', }, @@ -1815,7 +1815,7 @@ export const peopleDemo = [ city: 'Butlerchester', email: 'elizabeth.leon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/elizabeth-leon-7618636547', jobTitle: 'Clinical embryologist', }, @@ -1825,7 +1825,7 @@ export const peopleDemo = [ city: 'North Jamesburgh', email: 'carl.wagner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/carl-wagner-c6d92ca2ac', jobTitle: 'Programmer, multimedia', }, @@ -1835,7 +1835,7 @@ export const peopleDemo = [ city: 'Montgomeryborough', email: 'erica.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/erica-taylor-99a8f528f9', jobTitle: 'Catering manager', }, @@ -1845,7 +1845,7 @@ export const peopleDemo = [ city: 'Kramerville', email: 'sandy.gomez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/sandy-gomez-fe6cfbd6f8', jobTitle: 'Event organiser', }, @@ -1855,7 +1855,7 @@ export const peopleDemo = [ city: 'Cheyenneton', email: 'tracy.gray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/tracy-gray-e7455e5048', jobTitle: 'Charity officer', }, @@ -1865,7 +1865,7 @@ export const peopleDemo = [ city: 'South Martha', email: 'amy.davies@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/amy-davies-6d15292b3d', jobTitle: 'Trading standards officer', }, @@ -1875,7 +1875,7 @@ export const peopleDemo = [ city: 'Vaughnmouth', email: 'mary.wood@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/mary-wood-ac63c86744', jobTitle: 'Nutritional therapist', }, @@ -1885,7 +1885,7 @@ export const peopleDemo = [ city: 'Onealshire', email: 'james.green@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/james-green-0c06ff4286', jobTitle: 'Journalist, broadcasting', }, @@ -1895,7 +1895,7 @@ export const peopleDemo = [ city: 'East Jesseburgh', email: 'rebecca.petersen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/rebecca-petersen-3213856af4', jobTitle: 'Licensed conveyancer', }, @@ -1905,7 +1905,7 @@ export const peopleDemo = [ city: 'Sherylport', email: 'hunter.pierce@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/hunter-pierce-6873b9e186', jobTitle: 'Product manager', }, @@ -1915,7 +1915,7 @@ export const peopleDemo = [ city: 'West Zachary', email: 'christian.bailey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/christian-bailey-b3f28cc8db', jobTitle: 'Community education officer', }, @@ -1925,7 +1925,7 @@ export const peopleDemo = [ city: 'Port Scott', email: 'william.mitchell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/william-mitchell-90c3f0b311', jobTitle: 'Insurance account manager', }, @@ -1935,7 +1935,7 @@ export const peopleDemo = [ city: 'New Jonathon', email: 'brent.gray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/brent-gray-19b7fd459b', jobTitle: 'Careers information officer', }, @@ -1945,7 +1945,7 @@ export const peopleDemo = [ city: 'Port Tara', email: 'melissa.myers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/melissa-myers-0802db2cf0', jobTitle: 'Arboriculturist', }, @@ -1955,7 +1955,7 @@ export const peopleDemo = [ city: 'East Natasha', email: 'brittney.nguyen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/brittney-nguyen-c1eb8b312b', jobTitle: 'Logistics and distribution manager', }, @@ -1965,7 +1965,7 @@ export const peopleDemo = [ city: 'North Kariside', email: 'jacob.franklin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/jacob-franklin-fe55bfc993', jobTitle: 'Freight forwarder', }, @@ -1975,7 +1975,7 @@ export const peopleDemo = [ city: 'West Jamesburgh', email: 'kathy.burgess@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/kathy-burgess-99f2ff488c', jobTitle: 'Learning disability nurse', }, @@ -1985,7 +1985,7 @@ export const peopleDemo = [ city: 'South Jenniferstad', email: 'nicole.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/nicole-smith-8aec41062c', jobTitle: 'Legal secretary', }, @@ -1995,7 +1995,7 @@ export const peopleDemo = [ city: 'Barrstad', email: 'troy.decker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/troy-decker-bd98e78a69', jobTitle: 'Surveyor, minerals', }, @@ -2005,7 +2005,7 @@ export const peopleDemo = [ city: 'Myershaven', email: 'corey.thompson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/corey-thompson-f8d368eaf4', jobTitle: 'Structural engineer', }, @@ -2015,7 +2015,7 @@ export const peopleDemo = [ city: 'Cooperview', email: 'angela.webster@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/angela-webster-0702bee349', jobTitle: 'Chartered public finance accountant', }, @@ -2025,7 +2025,7 @@ export const peopleDemo = [ city: 'Reneebury', email: 'jenna.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/jenna-smith-74d6913e06', jobTitle: 'Colour technologist', }, @@ -2035,7 +2035,7 @@ export const peopleDemo = [ city: 'Eileenmouth', email: 'johnny.lee@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/johnny-lee-37995fae9a', jobTitle: 'Sports development officer', }, @@ -2045,7 +2045,7 @@ export const peopleDemo = [ city: 'Port Mackenzieshire', email: 'curtis.cross@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/curtis-cross-8df12d4c6d', jobTitle: 'Broadcast journalist', }, @@ -2055,7 +2055,7 @@ export const peopleDemo = [ city: 'Hansonfurt', email: 'paula.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/paula-perez-a4abb3240e', jobTitle: 'Bonds trader', }, @@ -2065,7 +2065,7 @@ export const peopleDemo = [ city: 'East Marybury', email: 'stephen.berger@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/stephen-berger-fcf9446666', jobTitle: 'Production engineer', }, @@ -2075,7 +2075,7 @@ export const peopleDemo = [ city: 'South Michael', email: 'candace.michael@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/candace-michael-a468f60b3c', jobTitle: 'Surveyor, building control', }, @@ -2085,7 +2085,7 @@ export const peopleDemo = [ city: 'Diazshire', email: 'jessica.lawrence@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/jessica-lawrence-61fd28400c', jobTitle: 'Regulatory affairs officer', }, @@ -2095,7 +2095,7 @@ export const peopleDemo = [ city: 'Garrettfurt', email: 'victoria.west@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/victoria-west-4514ebe192', jobTitle: 'Therapist, sports', }, @@ -2105,7 +2105,7 @@ export const peopleDemo = [ city: 'New Brian', email: 'matthew.matthews@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/matthew-matthews-42bbc7a5f8', jobTitle: 'Pathologist', }, @@ -2115,7 +2115,7 @@ export const peopleDemo = [ city: 'Powellfort', email: 'megan.lopez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/megan-lopez-936955bbda', jobTitle: 'Ophthalmologist', }, @@ -2125,7 +2125,7 @@ export const peopleDemo = [ city: 'Michelleside', email: 'kim.campbell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/kim-campbell-7d447d5ec6', jobTitle: 'Recruitment consultant', }, @@ -2135,7 +2135,7 @@ export const peopleDemo = [ city: 'Jamesberg', email: 'william.ryan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/william-ryan-a5c6cbc922', jobTitle: 'Ergonomist', }, @@ -2145,7 +2145,7 @@ export const peopleDemo = [ city: 'Castillostad', email: 'lauren.walker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/lauren-walker-74bbb465a8', jobTitle: 'Nature conservation officer', }, @@ -2155,7 +2155,7 @@ export const peopleDemo = [ city: 'Garciaport', email: 'jordan.castro@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/jordan-castro-5db609defa', jobTitle: 'Research officer, trade union', }, @@ -2165,7 +2165,7 @@ export const peopleDemo = [ city: 'Joshuaport', email: 'scott.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/scott-williams-0c2a344935', jobTitle: 'Bonds trader', }, @@ -2175,7 +2175,7 @@ export const peopleDemo = [ city: 'Webbfurt', email: 'peter.thompson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/peter-thompson-341acf2282', jobTitle: 'Waste management officer', }, @@ -2185,7 +2185,7 @@ export const peopleDemo = [ city: 'Seanside', email: 'jill.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/jill-williams-15b1d79aa6', jobTitle: 'Pathologist', }, @@ -2195,7 +2195,7 @@ export const peopleDemo = [ city: 'North Kimberly', email: 'joyce.diaz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/joyce-diaz-c25c3d067c', jobTitle: 'Loss adjuster, chartered', }, @@ -2205,7 +2205,7 @@ export const peopleDemo = [ city: 'South Andrewchester', email: 'robert.owens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/robert-owens-39e8664408', jobTitle: 'Land/geomatics surveyor', }, @@ -2215,7 +2215,7 @@ export const peopleDemo = [ city: 'Chadport', email: 'christine.fernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/christine-fernandez-4bde5f26e0', jobTitle: 'Theme park manager', }, @@ -2225,7 +2225,7 @@ export const peopleDemo = [ city: 'East Robert', email: 'gary.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/gary-jones-a993bf28c1', jobTitle: 'Community education officer', }, @@ -2235,7 +2235,7 @@ export const peopleDemo = [ city: 'South Stevenfort', email: 'ronald.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/ronald-brown-921558ef53', jobTitle: 'Arts administrator', }, @@ -2245,7 +2245,7 @@ export const peopleDemo = [ city: 'North Noah', email: 'curtis.oliver@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/curtis-oliver-1f7c8563a6', jobTitle: 'Oncologist', }, @@ -2255,7 +2255,7 @@ export const peopleDemo = [ city: 'Kylebury', email: 'stacy.vasquez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/stacy-vasquez-72e5c629d2', jobTitle: 'Engineer, biomedical', }, @@ -2265,7 +2265,7 @@ export const peopleDemo = [ city: 'West Jennifer', email: 'susan.hancock@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/susan-hancock-00b3694621', jobTitle: 'Charity fundraiser', }, @@ -2275,7 +2275,7 @@ export const peopleDemo = [ city: 'West Heather', email: 'amy.conner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/amy-conner-65d26933e9', jobTitle: 'Web designer', }, @@ -2285,7 +2285,7 @@ export const peopleDemo = [ city: 'New Kelli', email: 'keith.stein@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/keith-stein-dae7ba0be7', jobTitle: 'Airline pilot', }, @@ -2295,7 +2295,7 @@ export const peopleDemo = [ city: 'Kristyshire', email: 'kristen.lane@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/kristen-lane-7988cdb7fc', jobTitle: 'Immigration officer', }, @@ -2305,7 +2305,7 @@ export const peopleDemo = [ city: 'Lake Eric', email: 'leroy.bright@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/leroy-bright-5ce647996d', jobTitle: 'Chief Marketing Officer', }, @@ -2315,7 +2315,7 @@ export const peopleDemo = [ city: 'South Jason', email: 'bradley.patterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/bradley-patterson-590ba76c7d', jobTitle: 'Public house manager', }, @@ -2325,7 +2325,7 @@ export const peopleDemo = [ city: 'Dawnport', email: 'sarah.mcdaniel@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/sarah-mcdaniel-a2c7767c1e', jobTitle: 'Media planner', }, @@ -2335,7 +2335,7 @@ export const peopleDemo = [ city: 'Port Pamelafurt', email: 'brandon.boyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/brandon-boyd-4e2e998802', jobTitle: 'Surveyor, insurance', }, @@ -2345,7 +2345,7 @@ export const peopleDemo = [ city: 'Matthewport', email: 'carolyn.villarreal@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/carolyn-villarreal-3f2ca6cc5f', jobTitle: 'Engineer, automotive', }, @@ -2355,7 +2355,7 @@ export const peopleDemo = [ city: 'South Carriechester', email: 'harry.garrett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/harry-garrett-5faf263278', jobTitle: 'Pharmacist, hospital', }, @@ -2365,7 +2365,7 @@ export const peopleDemo = [ city: 'East Jason', email: 'richard.lee@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/richard-lee-8f9dff7c18', jobTitle: 'Financial trader', }, @@ -2375,7 +2375,7 @@ export const peopleDemo = [ city: 'New Karen', email: 'kristen.landry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/kristen-landry-106c6fc0b5', jobTitle: 'Facilities manager', }, @@ -2385,7 +2385,7 @@ export const peopleDemo = [ city: 'East Ronaldmouth', email: 'joshua.burgess@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/joshua-burgess-3596563692', jobTitle: 'IT technical support officer', }, @@ -2395,7 +2395,7 @@ export const peopleDemo = [ city: 'North Carmen', email: 'alicia.stevens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/alicia-stevens-428bceb969', jobTitle: 'Chiropodist', }, @@ -2405,7 +2405,7 @@ export const peopleDemo = [ city: 'Codyville', email: 'jason.torres@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/jason-torres-8250efe63e', jobTitle: 'QuickActions analyst', }, @@ -2415,7 +2415,7 @@ export const peopleDemo = [ city: 'Thomasview', email: 'michael.ortiz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/michael-ortiz-4c03016b05', jobTitle: 'Sound technician, broadcasting/film/video', }, @@ -2425,7 +2425,7 @@ export const peopleDemo = [ city: 'Lake Justin', email: 'deanna.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/deanna-williams-08b67a9a7c', jobTitle: 'Research officer, political party', }, @@ -2435,7 +2435,7 @@ export const peopleDemo = [ city: 'North Brandy', email: 'kevin.ray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/kevin-ray-dbea2dd52d', jobTitle: 'Naval architect', }, @@ -2445,7 +2445,7 @@ export const peopleDemo = [ city: 'Rebeccaberg', email: 'tony.walters@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/tony-walters-2bfeca8dd4', jobTitle: 'Purchasing manager', }, @@ -2455,7 +2455,7 @@ export const peopleDemo = [ city: 'Batesville', email: 'andrew.roberts@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/andrew-roberts-41c13e9f43', jobTitle: 'Physiotherapist', }, @@ -2465,7 +2465,7 @@ export const peopleDemo = [ city: 'Stephanieport', email: 'lucas.fisher@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/lucas-fisher-7a563c63b5', jobTitle: 'Materials engineer', }, @@ -2475,7 +2475,7 @@ export const peopleDemo = [ city: 'South Ashleystad', email: 'sarah.bates@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/sarah-bates-477cbbc18f', jobTitle: 'Educational psychologist', }, @@ -2485,7 +2485,7 @@ export const peopleDemo = [ city: 'West Dana', email: 'jon.osborne@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/jon-osborne-e72d657289', jobTitle: 'Geologist, engineering', }, @@ -2495,7 +2495,7 @@ export const peopleDemo = [ city: 'West Michealland', email: 'lisa.green@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/lisa-green-e3c92b78e0', jobTitle: 'Art therapist', }, @@ -2505,7 +2505,7 @@ export const peopleDemo = [ city: 'Michelleville', email: 'george.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/george-allen-9de1c6f0d2', jobTitle: 'Retail merchandiser', }, @@ -2515,7 +2515,7 @@ export const peopleDemo = [ city: 'Hoffmanberg', email: 'rhonda.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/rhonda-smith-7c802edf30', jobTitle: 'Therapist, nutritional', }, @@ -2525,7 +2525,7 @@ export const peopleDemo = [ city: 'East Gabrielberg', email: 'kyle.day@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/kyle-day-bedc45ac4c', jobTitle: 'Dispensing optician', }, @@ -2535,7 +2535,7 @@ export const peopleDemo = [ city: 'Pamelabury', email: 'valerie.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/valerie-smith-71a58f5833', jobTitle: 'Diagnostic radiographer', }, @@ -2545,7 +2545,7 @@ export const peopleDemo = [ city: 'South Brycefort', email: 'matthew.nelson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/matthew-nelson-a288db243c', jobTitle: 'Surveyor, insurance', }, @@ -2555,7 +2555,7 @@ export const peopleDemo = [ city: 'Jessicaberg', email: 'terri.ramos@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/terri-ramos-eb86c1f353', jobTitle: 'Tour manager', }, @@ -2565,7 +2565,7 @@ export const peopleDemo = [ city: 'Mathewsburgh', email: 'brian.bell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/brian-bell-50b9b8cc1c', jobTitle: 'Engineer, electrical', }, @@ -2575,7 +2575,7 @@ export const peopleDemo = [ city: 'West Jeffrey', email: 'troy.stuart@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/troy-stuart-f056c7bfcb', jobTitle: 'Corporate investment banker', }, @@ -2585,7 +2585,7 @@ export const peopleDemo = [ city: 'Valenciaside', email: 'beth.shea@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/beth-shea-2f51929bb5', jobTitle: 'Forensic psychologist', }, @@ -2595,7 +2595,7 @@ export const peopleDemo = [ city: 'Baileystad', email: 'erin.barrera@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/erin-barrera-74010827a9', jobTitle: 'Youth worker', }, @@ -2605,7 +2605,7 @@ export const peopleDemo = [ city: 'West Darrylmouth', email: 'danielle.maynard@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/danielle-maynard-ba9f453d8e', jobTitle: 'Location manager', }, @@ -2615,7 +2615,7 @@ export const peopleDemo = [ city: 'Colinville', email: 'sarah.oneal@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/sarah-oneal-814e3f10b5', jobTitle: 'Programmer, multimedia', }, @@ -2625,7 +2625,7 @@ export const peopleDemo = [ city: 'North Paul', email: 'carrie.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/carrie-taylor-33e1df2a16', jobTitle: 'Archaeologist', }, @@ -2635,7 +2635,7 @@ export const peopleDemo = [ city: 'Annashire', email: 'andrea.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/andrea-smith-63bc9b52c2', jobTitle: 'Mechanical engineer', }, @@ -2645,7 +2645,7 @@ export const peopleDemo = [ city: 'East Barbarafort', email: 'adam.cowan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/adam-cowan-a8cb02d136', jobTitle: 'Sports development officer', }, @@ -2655,7 +2655,7 @@ export const peopleDemo = [ city: 'Annaton', email: 'bryan.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/bryan-johnson-ec29fc1f9a', jobTitle: 'Clinical molecular geneticist', }, @@ -2665,7 +2665,7 @@ export const peopleDemo = [ city: 'Dennischester', email: 'richard.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/richard-jackson-a17e46e0d7', jobTitle: 'Engineer, water', }, @@ -2675,7 +2675,7 @@ export const peopleDemo = [ city: 'Williamtown', email: 'kelly.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/kelly-jackson-8b32bfb505', jobTitle: 'Pension scheme manager', }, @@ -2685,7 +2685,7 @@ export const peopleDemo = [ city: 'Samanthafort', email: 'jack.ingram@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/jack-ingram-e68114f5c7', jobTitle: 'Pharmacist, community', }, @@ -2695,7 +2695,7 @@ export const peopleDemo = [ city: 'Port Michelle', email: 'rhonda.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/rhonda-jenkins-f071128f22', jobTitle: 'Aeronautical engineer', }, @@ -2705,7 +2705,7 @@ export const peopleDemo = [ city: 'Youngborough', email: 'jacqueline.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/jacqueline-johnson-dda681b600', jobTitle: 'Water quality scientist', }, @@ -2715,7 +2715,7 @@ export const peopleDemo = [ city: 'South Matthew', email: 'casey.oneill@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/casey-oneill-ab8f249833', jobTitle: 'Local government officer', }, @@ -2725,7 +2725,7 @@ export const peopleDemo = [ city: 'Lake Christopher', email: 'kathleen.francis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/kathleen-francis-097093196c', jobTitle: 'Paediatric nurse', }, @@ -2735,7 +2735,7 @@ export const peopleDemo = [ city: 'East Diane', email: 'gary.woods@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/gary-woods-4b0cba46d6', jobTitle: 'Multimedia specialist', }, @@ -2745,7 +2745,7 @@ export const peopleDemo = [ city: 'West Michael', email: 'rachel.harris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/rachel-harris-93dfc3611e', jobTitle: 'Medical laboratory scientific officer', }, @@ -2755,7 +2755,7 @@ export const peopleDemo = [ city: 'North Douglas', email: 'charlene.rose@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/charlene-rose-98947b0547', jobTitle: 'Solicitor', }, @@ -2765,7 +2765,7 @@ export const peopleDemo = [ city: 'Hudsonmouth', email: 'donna.saunders@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/donna-saunders-4f712712f8', jobTitle: 'Actor', }, @@ -2775,7 +2775,7 @@ export const peopleDemo = [ city: 'Lake Jennifer', email: 'thomas.singh@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/thomas-singh-640bdc36f3', jobTitle: 'Press sub', }, @@ -2785,7 +2785,7 @@ export const peopleDemo = [ city: 'Jordanton', email: 'ryan.morales@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/ryan-morales-081c6966a9', jobTitle: 'Food technologist', }, @@ -2795,7 +2795,7 @@ export const peopleDemo = [ city: 'South Philipbury', email: 'christy.hall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/christy-hall-916136bbc9', jobTitle: 'Pension scheme manager', }, @@ -2805,7 +2805,7 @@ export const peopleDemo = [ city: 'Dustintown', email: 'joshua.hawkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/joshua-hawkins-9a0340bb4e', jobTitle: 'Occupational psychologist', }, @@ -2815,7 +2815,7 @@ export const peopleDemo = [ city: 'West Timside', email: 'jasmine.stanley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/jasmine-stanley-94eeb75ca5', jobTitle: 'Nutritional therapist', }, @@ -2825,7 +2825,7 @@ export const peopleDemo = [ city: 'Jessicafurt', email: 'morgan.thomas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/morgan-thomas-a18e91a104', jobTitle: 'Clinical research associate', }, @@ -2835,7 +2835,7 @@ export const peopleDemo = [ city: 'Carlosview', email: 'laura.gomez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/laura-gomez-7ee145bfc4', jobTitle: 'Bookseller', }, @@ -2845,7 +2845,7 @@ export const peopleDemo = [ city: 'Rileychester', email: 'anne.montgomery@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/anne-montgomery-10245f347a', jobTitle: 'Arts development officer', }, @@ -2855,7 +2855,7 @@ export const peopleDemo = [ city: 'Nicolefurt', email: 'tiffany.peterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/tiffany-peterson-fed8500c17', jobTitle: 'Multimedia specialist', }, @@ -2865,7 +2865,7 @@ export const peopleDemo = [ city: 'North Kelsey', email: 'ian.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/ian-martinez-03c2967428', jobTitle: 'Phytotherapist', }, @@ -2875,7 +2875,7 @@ export const peopleDemo = [ city: 'Lake Daniel', email: 'shelly.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/shelly-rodriguez-42e2b8178f', jobTitle: 'Conservation officer, nature', }, @@ -2885,7 +2885,7 @@ export const peopleDemo = [ city: 'East Michael', email: 'philip.santos@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/philip-santos-0c763b15db', jobTitle: 'Field trials officer', }, @@ -2895,7 +2895,7 @@ export const peopleDemo = [ city: 'Jeremyburgh', email: 'michael.foley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/michael-foley-a0535b3102', jobTitle: 'Exhibition designer', }, @@ -2905,7 +2905,7 @@ export const peopleDemo = [ city: 'Port Caseymouth', email: 'david.raymond@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/david-raymond-1e1178af61', jobTitle: 'Psychologist, occupational', }, @@ -2915,7 +2915,7 @@ export const peopleDemo = [ city: 'Spencerberg', email: 'amanda.booker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/amanda-booker-6781b9995d', jobTitle: 'Network engineer', }, @@ -2925,7 +2925,7 @@ export const peopleDemo = [ city: 'Lake Alexandra', email: 'emily.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/emily-jenkins-54d2700826', jobTitle: 'Technical brewer', }, @@ -2935,7 +2935,7 @@ export const peopleDemo = [ city: 'Donnaside', email: 'timothy.larsen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/timothy-larsen-fb5366ba23', jobTitle: 'Geneticist, molecular', }, @@ -2945,7 +2945,7 @@ export const peopleDemo = [ city: 'Campbellville', email: 'ashley.barrett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/ashley-barrett-014f05cf8c', jobTitle: 'Writer', }, @@ -2955,7 +2955,7 @@ export const peopleDemo = [ city: 'Patriciafort', email: 'sandra.adkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/sandra-adkins-8cbb8957df', jobTitle: 'Chief Marketing Officer', }, @@ -2965,7 +2965,7 @@ export const peopleDemo = [ city: 'North Rebecca', email: 'kelly.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/kelly-johnson-c74ca2d4f3', jobTitle: 'Farm manager', }, @@ -2975,7 +2975,7 @@ export const peopleDemo = [ city: 'Anthonymouth', email: 'patricia.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/patricia-rodriguez-e83155aa16', jobTitle: 'QuickActions analyst', }, @@ -2985,7 +2985,7 @@ export const peopleDemo = [ city: 'West Keith', email: 'dawn.scott@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/dawn-scott-c95e0080c3', jobTitle: 'Sports therapist', }, @@ -2995,7 +2995,7 @@ export const peopleDemo = [ city: 'Thomasfurt', email: 'timothy.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/timothy-jones-eb531fbe5d', jobTitle: 'Manufacturing engineer', }, @@ -3005,7 +3005,7 @@ export const peopleDemo = [ city: 'Port Christineshire', email: 'william.walker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/william-walker-492bb70711', jobTitle: 'Water engineer', }, @@ -3015,7 +3015,7 @@ export const peopleDemo = [ city: 'Port Tyler', email: 'jesus.santana@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/jesus-santana-fc2b873356', jobTitle: 'Designer, industrial/product', }, @@ -3025,7 +3025,7 @@ export const peopleDemo = [ city: 'West Christopher', email: 'maurice.carpenter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/maurice-carpenter-2f52cbbda1', jobTitle: 'Surveyor, insurance', }, @@ -3035,7 +3035,7 @@ export const peopleDemo = [ city: 'East Sarahtown', email: 'robert.barnes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/robert-barnes-b7ef473c08', jobTitle: 'Ecologist', }, @@ -3045,7 +3045,7 @@ export const peopleDemo = [ city: 'South Tyler', email: 'matthew.aguilar@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/matthew-aguilar-c9fde35926', jobTitle: 'Biomedical scientist', }, @@ -3055,7 +3055,7 @@ export const peopleDemo = [ city: 'West Dannyport', email: 'anthony.stanley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/anthony-stanley-3854ad13be', jobTitle: 'Designer, interior/spatial', }, @@ -3065,7 +3065,7 @@ export const peopleDemo = [ city: 'Jeffreyport', email: 'brian.garza@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/brian-garza-66afe09a9a', jobTitle: 'Medical illustrator', }, @@ -3075,7 +3075,7 @@ export const peopleDemo = [ city: 'East Benjamin', email: 'lisa.oconnor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/lisa-oconnor-ec7fe69b0a', jobTitle: 'Systems analyst', }, @@ -3085,7 +3085,7 @@ export const peopleDemo = [ city: 'Frazierchester', email: 'william.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/william-wilson-fdb80928cc', jobTitle: 'Radio broadcast assistant', }, @@ -3095,7 +3095,7 @@ export const peopleDemo = [ city: 'Allenport', email: 'joanna.alvarez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/joanna-alvarez-9f444908cb', jobTitle: 'Data processing manager', }, @@ -3105,7 +3105,7 @@ export const peopleDemo = [ city: 'New Charles', email: 'denise.hill@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/denise-hill-3e8506c956', jobTitle: 'Trade union research officer', }, @@ -3115,7 +3115,7 @@ export const peopleDemo = [ city: 'Wilsonshire', email: 'marie.frey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/marie-frey-6f12dc710b', jobTitle: 'Lecturer, higher education', }, @@ -3125,7 +3125,7 @@ export const peopleDemo = [ city: 'South Alejandra', email: 'sarah.anderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/sarah-anderson-fe0d0f83d5', jobTitle: 'Pensions consultant', }, @@ -3135,7 +3135,7 @@ export const peopleDemo = [ city: 'East Cliffordmouth', email: 'mary.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/mary-garcia-ed7168a5f9', jobTitle: 'Optometrist', }, @@ -3145,7 +3145,7 @@ export const peopleDemo = [ city: 'Kristyton', email: 'richard.massey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/richard-massey-af5c02d5ad', jobTitle: 'Armed forces operational officer', }, @@ -3155,7 +3155,7 @@ export const peopleDemo = [ city: 'Lindahaven', email: 'megan.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/megan-rodriguez-377a7d26f7', jobTitle: 'Editor, magazine features', }, @@ -3165,7 +3165,7 @@ export const peopleDemo = [ city: 'Jeffreyfurt', email: 'sandra.conway@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/sandra-conway-49f0ce34eb', jobTitle: 'Press sub', }, @@ -3175,7 +3175,7 @@ export const peopleDemo = [ city: 'Rebeccafort', email: 'rachael.dalton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/rachael-dalton-6bab20a6a3', jobTitle: 'Catering manager', }, @@ -3185,7 +3185,7 @@ export const peopleDemo = [ city: 'East Gabrielashire', email: 'katherine.little@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/katherine-little-d826db995d', jobTitle: 'Production assistant, radio', }, @@ -3195,7 +3195,7 @@ export const peopleDemo = [ city: 'Aprilfort', email: 'faith.cross@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/faith-cross-029912b4f5', jobTitle: 'Pensions consultant', }, @@ -3205,7 +3205,7 @@ export const peopleDemo = [ city: 'Amandastad', email: 'amy.farmer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/amy-farmer-e9024737cf', jobTitle: 'Printmaker', }, @@ -3215,7 +3215,7 @@ export const peopleDemo = [ city: 'South Andrew', email: 'stanley.todd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/stanley-todd-d92752ad42', jobTitle: 'Psychologist, forensic', }, @@ -3225,7 +3225,7 @@ export const peopleDemo = [ city: 'Tonyastad', email: 'bradley.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/bradley-miller-4b0d674c5d', jobTitle: 'Hospital doctor', }, @@ -3235,7 +3235,7 @@ export const peopleDemo = [ city: 'North Dawnport', email: 'sharon.rhodes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/sharon-rhodes-708695e723', jobTitle: 'Lighting technician, broadcasting/film/video', }, @@ -3245,7 +3245,7 @@ export const peopleDemo = [ city: 'Marquezhaven', email: 'emily.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/emily-young-721fb2a8e0', jobTitle: 'Bonds trader', }, @@ -3255,7 +3255,7 @@ export const peopleDemo = [ city: 'Benjaminborough', email: 'victoria.harris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/victoria-harris-6f291283ab', jobTitle: 'Hospital pharmacist', }, @@ -3265,7 +3265,7 @@ export const peopleDemo = [ city: 'Lake Marilyn', email: 'andrew.massey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/andrew-massey-0f7983ac40', jobTitle: 'Podiatrist', }, @@ -3275,7 +3275,7 @@ export const peopleDemo = [ city: 'Lake Jennifer', email: 'heather.mack@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/heather-mack-fc2f7ee6fd', jobTitle: "Politician's assistant", }, @@ -3285,7 +3285,7 @@ export const peopleDemo = [ city: 'South Jacquelinefort', email: 'michelle.richards@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/michelle-richards-5d0d907e8c', jobTitle: 'Geochemist', }, @@ -3295,7 +3295,7 @@ export const peopleDemo = [ city: 'Christineburgh', email: 'billy.jacobs@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/billy-jacobs-e9d18903f6', jobTitle: 'Pensions consultant', }, @@ -3305,7 +3305,7 @@ export const peopleDemo = [ city: 'New Micheleside', email: 'michael.white@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/michael-white-6548d80612', jobTitle: 'Metallurgist', }, @@ -3315,7 +3315,7 @@ export const peopleDemo = [ city: 'North Brookeville', email: 'jose.frazier@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/jose-frazier-c989680d85', jobTitle: 'Conservation officer, historic buildings', }, @@ -3325,7 +3325,7 @@ export const peopleDemo = [ city: 'South Hannahchester', email: 'michael.barrett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/michael-barrett-9a6fdee03b', jobTitle: 'Sound technician, broadcasting/film/video', }, @@ -3335,7 +3335,7 @@ export const peopleDemo = [ city: 'Taylorfurt', email: 'lisa.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/lisa-allen-65f7418d40', jobTitle: 'Contracting civil engineer', }, @@ -3345,7 +3345,7 @@ export const peopleDemo = [ city: 'Coletown', email: 'kristopher.berg@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/kristopher-berg-66eb2e31cf', jobTitle: 'Community pharmacist', }, @@ -3355,7 +3355,7 @@ export const peopleDemo = [ city: 'South Aaron', email: 'regina.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/regina-allen-f1f56a375b', jobTitle: 'Civil Service administrator', }, @@ -3365,7 +3365,7 @@ export const peopleDemo = [ city: 'Bakerfort', email: 'angela.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/angela-williams-00ba2b783f', jobTitle: 'Designer, furniture', }, @@ -3375,7 +3375,7 @@ export const peopleDemo = [ city: 'South Sherrychester', email: 'aaron.watts@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/aaron-watts-423d6e63d0', jobTitle: 'Commissioning editor', }, @@ -3385,7 +3385,7 @@ export const peopleDemo = [ city: 'Lake Jennifer', email: 'angela.callahan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/angela-callahan-c21517a078', jobTitle: 'Merchant navy officer', }, @@ -3395,7 +3395,7 @@ export const peopleDemo = [ city: 'Lake Samantha', email: 'walter.mclaughlin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/walter-mclaughlin-7ec17dd691', jobTitle: 'Lecturer, further education', }, @@ -3405,7 +3405,7 @@ export const peopleDemo = [ city: 'East Debbiefurt', email: 'brian.ellis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/brian-ellis-bc4c380ff5', jobTitle: 'Corporate investment banker', }, @@ -3415,7 +3415,7 @@ export const peopleDemo = [ city: 'East Robert', email: 'joshua.watson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/joshua-watson-5613c41859', jobTitle: 'Scientist, clinical (histocompatibility and immunogenetics)', }, @@ -3425,7 +3425,7 @@ export const peopleDemo = [ city: 'North Donna', email: 'monica.molina@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/monica-molina-2213f0ccce', jobTitle: 'Press sub', }, @@ -3435,7 +3435,7 @@ export const peopleDemo = [ city: 'Cunninghamstad', email: 'justin.castro@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/justin-castro-43678bde89', jobTitle: 'Plant breeder/geneticist', }, @@ -3445,7 +3445,7 @@ export const peopleDemo = [ city: 'Port Angela', email: 'austin.dixon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/austin-dixon-791d6f8fa6', jobTitle: 'Dealer', }, @@ -3455,7 +3455,7 @@ export const peopleDemo = [ city: 'Christopherchester', email: 'mitchell.massey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/mitchell-massey-22fdbe6b77', jobTitle: 'Phytotherapist', }, @@ -3465,7 +3465,7 @@ export const peopleDemo = [ city: 'Glennview', email: 'eric.reid@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/eric-reid-4a1a08817b', jobTitle: 'TEFL teacher', }, @@ -3475,7 +3475,7 @@ export const peopleDemo = [ city: 'Hernandezview', email: 'andrea.park@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/andrea-park-80bc0bf8a0', jobTitle: 'Brewing technologist', }, @@ -3485,7 +3485,7 @@ export const peopleDemo = [ city: 'Port Emilyport', email: 'wendy.page@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/wendy-page-7d996e5a2f', jobTitle: 'Chief Executive Officer', }, @@ -3495,7 +3495,7 @@ export const peopleDemo = [ city: 'East Brett', email: 'vanessa.carpenter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/vanessa-carpenter-a0b8dc7720', jobTitle: 'Chiropractor', }, @@ -3505,7 +3505,7 @@ export const peopleDemo = [ city: 'Kennethfort', email: 'lisa.bailey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/lisa-bailey-f737183c93', jobTitle: 'Agricultural engineer', }, @@ -3515,7 +3515,7 @@ export const peopleDemo = [ city: 'Port Kimberly', email: 'jason.wagner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/jason-wagner-6eafd392ce', jobTitle: 'Surveyor, quantity', }, @@ -3525,7 +3525,7 @@ export const peopleDemo = [ city: 'West Tammy', email: 'judith.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/judith-moore-e8bdfc83e3', jobTitle: 'Translator', }, @@ -3535,7 +3535,7 @@ export const peopleDemo = [ city: 'Andrewfort', email: 'steven.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/steven-moore-a4d5be1e3c', jobTitle: 'Accountant, chartered certified', }, @@ -3545,7 +3545,7 @@ export const peopleDemo = [ city: 'Robinsonberg', email: 'darren.castillo@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/darren-castillo-8ed7a1702b', jobTitle: 'Advertising copywriter', }, @@ -3555,7 +3555,7 @@ export const peopleDemo = [ city: 'Andrewstad', email: 'regina.quinn@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/regina-quinn-3983eab37e', jobTitle: 'Materials engineer', }, @@ -3565,7 +3565,7 @@ export const peopleDemo = [ city: 'Lake Wendyland', email: 'michelle.delgado@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/michelle-delgado-53f8093373', jobTitle: 'Soil scientist', }, @@ -3575,7 +3575,7 @@ export const peopleDemo = [ city: 'North Briannastad', email: 'miguel.rose@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/miguel-rose-31034cf613', jobTitle: 'Therapeutic radiographer', }, @@ -3585,7 +3585,7 @@ export const peopleDemo = [ city: 'Carolynville', email: 'gary.mason@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/gary-mason-549d3828f5', jobTitle: 'Theatre director', }, @@ -3595,7 +3595,7 @@ export const peopleDemo = [ city: 'New Craigburgh', email: 'david.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/david-young-ad2e589cdb', jobTitle: 'Medical illustrator', }, @@ -3605,7 +3605,7 @@ export const peopleDemo = [ city: 'Alexshire', email: 'jennifer.lewis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/jennifer-lewis-008f8dc7e6', jobTitle: 'Teaching laboratory technician', }, @@ -3615,7 +3615,7 @@ export const peopleDemo = [ city: 'East Taylor', email: 'amanda.carson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/amanda-carson-8e728fa39a', jobTitle: 'Product/process development scientist', }, @@ -3625,7 +3625,7 @@ export const peopleDemo = [ city: 'Port Matthewtown', email: 'jeffrey.valentine@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/jeffrey-valentine-ccb2756410', jobTitle: 'Surveyor, hydrographic', }, @@ -3635,7 +3635,7 @@ export const peopleDemo = [ city: 'North Rodney', email: 'helen.gordon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/helen-gordon-71ec22bf50', jobTitle: 'Engineer, communications', }, @@ -3645,7 +3645,7 @@ export const peopleDemo = [ city: 'Danielleport', email: 'cameron.lopez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/cameron-lopez-e66f1d22d8', jobTitle: 'Engineer, production', }, @@ -3655,7 +3655,7 @@ export const peopleDemo = [ city: 'Coleview', email: 'troy.gray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/troy-gray-756cd40db4', jobTitle: 'Health and safety inspector', }, @@ -3665,7 +3665,7 @@ export const peopleDemo = [ city: 'Lake Brooke', email: 'tonya.payne@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/tonya-payne-5e0b7eec25', jobTitle: 'Training and development officer', }, @@ -3675,7 +3675,7 @@ export const peopleDemo = [ city: 'Jacobberg', email: 'april.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/april-williams-f4fa095134', jobTitle: 'Tourist information centre manager', }, @@ -3685,7 +3685,7 @@ export const peopleDemo = [ city: 'New Jonathanview', email: 'gregory.baker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/gregory-baker-58cbfbde95', jobTitle: 'Pharmacist, community', }, @@ -3695,7 +3695,7 @@ export const peopleDemo = [ city: 'Loganborough', email: 'bobby.cummings@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/bobby-cummings-18e64e1c24', jobTitle: 'Mechanical engineer', }, @@ -3705,7 +3705,7 @@ export const peopleDemo = [ city: 'Villegasberg', email: 'melissa.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/melissa-jackson-36c2e59b76', jobTitle: 'Pharmacologist', }, @@ -3715,7 +3715,7 @@ export const peopleDemo = [ city: 'Bestfort', email: 'lance.norman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/lance-norman-7b07c0a723', jobTitle: 'Furniture conservator/restorer', }, @@ -3725,7 +3725,7 @@ export const peopleDemo = [ city: 'Tinaview', email: 'amy.roberts@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/amy-roberts-b98996c9ef', jobTitle: 'Cytogeneticist', }, @@ -3735,7 +3735,7 @@ export const peopleDemo = [ city: 'Rossmouth', email: 'kaitlyn.kelly@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/kaitlyn-kelly-e6d2be7a00', jobTitle: 'Commercial/residential surveyor', }, @@ -3745,7 +3745,7 @@ export const peopleDemo = [ city: 'Mitchellland', email: 'alan.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/alan-perez-6a584f8e83', jobTitle: 'Advertising copywriter', }, @@ -3755,7 +3755,7 @@ export const peopleDemo = [ city: 'Evanmouth', email: 'kaylee.garrett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/kaylee-garrett-5c56311f4e', jobTitle: 'Chief Operating Officer', }, @@ -3765,7 +3765,7 @@ export const peopleDemo = [ city: 'Lake Melissamouth', email: 'deanna.ball@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/deanna-ball-cac5d9cf3c', jobTitle: 'Nurse, learning disability', }, @@ -3775,7 +3775,7 @@ export const peopleDemo = [ city: 'West Timothy', email: 'michele.crawford@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/michele-crawford-21fd4da56c', jobTitle: 'Education officer, environmental', }, @@ -3785,7 +3785,7 @@ export const peopleDemo = [ city: 'Gomezshire', email: 'tommy.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/tommy-brown-a3313774fb', jobTitle: 'Civil engineer, consulting', }, @@ -3795,7 +3795,7 @@ export const peopleDemo = [ city: 'North Amanda', email: 'mary.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/mary-smith-e7451d73fb', jobTitle: 'Commercial/residential surveyor', }, @@ -3805,7 +3805,7 @@ export const peopleDemo = [ city: 'Lake Lauraland', email: 'danielle.dunn@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/danielle-dunn-afadb3743f', jobTitle: 'Higher education lecturer', }, @@ -3815,7 +3815,7 @@ export const peopleDemo = [ city: 'Stephenview', email: 'daniel.ward@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/daniel-ward-9eba345b13', jobTitle: 'Corporate treasurer', }, @@ -3825,7 +3825,7 @@ export const peopleDemo = [ city: 'New Allisonberg', email: 'louis.bailey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/louis-bailey-a7c289633b', jobTitle: 'Actuary', }, @@ -3835,7 +3835,7 @@ export const peopleDemo = [ city: 'Port Joseph', email: 'melvin.rosario@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/melvin-rosario-eeb2165a87', jobTitle: 'Mental health nurse', }, @@ -3845,7 +3845,7 @@ export const peopleDemo = [ city: 'Johnland', email: 'jason.fritz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/jason-fritz-998e404fcc', jobTitle: 'Counselling psychologist', }, @@ -3855,7 +3855,7 @@ export const peopleDemo = [ city: 'Port Frederickmouth', email: 'candice.weber@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/candice-weber-8d1a3aa843', jobTitle: 'Scientific laboratory technician', }, @@ -3865,7 +3865,7 @@ export const peopleDemo = [ city: 'Weeksmouth', email: 'john.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/john-jones-f3a69bf1c7', jobTitle: "Politician's assistant", }, @@ -3875,7 +3875,7 @@ export const peopleDemo = [ city: 'Juliefort', email: 'henry.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/henry-gonzalez-ed4236ae1a', jobTitle: 'Medical secretary', }, @@ -3885,7 +3885,7 @@ export const peopleDemo = [ city: 'Zunigaside', email: 'kaitlyn.brennan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/kaitlyn-brennan-3e779378d9', jobTitle: 'Osteopath', }, @@ -3895,7 +3895,7 @@ export const peopleDemo = [ city: 'Gordontown', email: 'jeffrey.shepard@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/jeffrey-shepard-038f3df4d1', jobTitle: 'Designer, graphic', }, @@ -3905,7 +3905,7 @@ export const peopleDemo = [ city: 'North Kennethfort', email: 'emily.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/emily-smith-3c11276729', jobTitle: 'Claims inspector/assessor', }, @@ -3915,7 +3915,7 @@ export const peopleDemo = [ city: 'West Christian', email: 'richard.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/richard-williams-034896141e', jobTitle: 'Artist', }, @@ -3925,7 +3925,7 @@ export const peopleDemo = [ city: 'Antonioview', email: 'david.cruz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/david-cruz-9821a933f5', jobTitle: 'Lecturer, further education', }, @@ -3935,7 +3935,7 @@ export const peopleDemo = [ city: 'Josephtown', email: 'julie.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/julie-smith-4b6473adc4', jobTitle: 'Conservator, furniture', }, @@ -3945,7 +3945,7 @@ export const peopleDemo = [ city: 'Brittanymouth', email: 'edward.russell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/edward-russell-34445498de', jobTitle: 'Teaching laboratory technician', }, @@ -3955,7 +3955,7 @@ export const peopleDemo = [ city: 'New Biancamouth', email: 'beth.kennedy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/beth-kennedy-956fad5f18', jobTitle: 'Passenger transport manager', }, @@ -3965,7 +3965,7 @@ export const peopleDemo = [ city: 'Millerfort', email: 'craig.maxwell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/craig-maxwell-9b3a04b47e', jobTitle: 'Conservator, furniture', }, @@ -3975,7 +3975,7 @@ export const peopleDemo = [ city: 'Stephaniestad', email: 'christopher.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/christopher-jackson-0ad43cfe80', jobTitle: 'Education officer, community', }, @@ -3985,7 +3985,7 @@ export const peopleDemo = [ city: 'New Mariabury', email: 'jacob.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/jacob-miller-c0d550cede', jobTitle: 'Surveyor, land/geomatics', }, @@ -3995,7 +3995,7 @@ export const peopleDemo = [ city: 'New Kelly', email: 'kevin.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/kevin-williams-41633df75d', jobTitle: 'Doctor, general practice', }, @@ -4005,7 +4005,7 @@ export const peopleDemo = [ city: 'South Isaac', email: 'mary.wiley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/mary-wiley-2c6f70a754', jobTitle: 'Museum/gallery curator', }, @@ -4015,7 +4015,7 @@ export const peopleDemo = [ city: 'Santanafort', email: 'sierra.mccullough@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/sierra-mccullough-9d4780eddd', jobTitle: 'Tour manager', }, @@ -4025,7 +4025,7 @@ export const peopleDemo = [ city: 'Stevenhaven', email: 'michelle.mcknight@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/michelle-mcknight-d4229e04f1', jobTitle: 'Adult guidance worker', }, @@ -4035,7 +4035,7 @@ export const peopleDemo = [ city: 'Carlosfort', email: 'devin.aguilar@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/devin-aguilar-ed9890f7fd', jobTitle: 'Press photographer', }, @@ -4045,7 +4045,7 @@ export const peopleDemo = [ city: 'South Michaelville', email: 'christopher.figueroa@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/christopher-figueroa-8ef81242d0', jobTitle: 'Theatre stage manager', }, @@ -4055,7 +4055,7 @@ export const peopleDemo = [ city: 'Coxberg', email: 'anita.orr@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/anita-orr-7648a03041', jobTitle: 'Air cabin crew', }, @@ -4065,7 +4065,7 @@ export const peopleDemo = [ city: 'Arnoldborough', email: 'richard.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/richard-young-10948fb6af', jobTitle: 'Osteopath', }, @@ -4075,7 +4075,7 @@ export const peopleDemo = [ city: 'New Mariebury', email: 'justin.berry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/justin-berry-4ac0f93944', jobTitle: 'Public house manager', }, @@ -4085,7 +4085,7 @@ export const peopleDemo = [ city: 'New Justinborough', email: 'timothy.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/timothy-davis-6c07a1c0bc', jobTitle: 'Call centre manager', }, @@ -4095,7 +4095,7 @@ export const peopleDemo = [ city: 'Pottermouth', email: 'brian.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/brian-williams-07738aaf00', jobTitle: 'Media buyer', }, @@ -4105,7 +4105,7 @@ export const peopleDemo = [ city: 'Wyattbury', email: 'kyle.carr@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/kyle-carr-ae1d05c89e', jobTitle: 'Environmental education officer', }, @@ -4115,7 +4115,7 @@ export const peopleDemo = [ city: 'New Joshua', email: 'jessica.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/jessica-gonzalez-26ff71c932', jobTitle: 'Acupuncturist', }, @@ -4125,7 +4125,7 @@ export const peopleDemo = [ city: 'West Joseph', email: 'hannah.nguyen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/hannah-nguyen-af672e539c', jobTitle: 'Theatre director', }, @@ -4135,7 +4135,7 @@ export const peopleDemo = [ city: 'East Matthew', email: 'tina.salinas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/tina-salinas-4c6898adb4', jobTitle: 'Office manager', }, @@ -4145,7 +4145,7 @@ export const peopleDemo = [ city: 'Edwardbury', email: 'matthew.king@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/matthew-king-4fcd69e0e2', jobTitle: 'Scientist, forensic', }, @@ -4155,7 +4155,7 @@ export const peopleDemo = [ city: 'Lake Anthonytown', email: 'carrie.mayer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/carrie-mayer-19a613bd93', jobTitle: 'Furniture conservator/restorer', }, @@ -4165,7 +4165,7 @@ export const peopleDemo = [ city: 'Kimberlyport', email: 'alan.guerrero@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/alan-guerrero-35f65ce7ed', jobTitle: 'Catering manager', }, @@ -4175,7 +4175,7 @@ export const peopleDemo = [ city: 'New Kimberly', email: 'alan.edwards@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/alan-edwards-2c5694b583', jobTitle: 'Air traffic controller', }, @@ -4185,7 +4185,7 @@ export const peopleDemo = [ city: 'Lake Christina', email: 'ellen.hughes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/ellen-hughes-d7dcd7cee8', jobTitle: 'Catering manager', }, @@ -4195,7 +4195,7 @@ export const peopleDemo = [ city: 'Port Josephfort', email: 'jennifer.cox@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/jennifer-cox-521071720f', jobTitle: 'Product designer', }, @@ -4205,7 +4205,7 @@ export const peopleDemo = [ city: 'Spencerside', email: 'james.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/james-brown-a872fe7489', jobTitle: 'Theatre manager', }, @@ -4215,7 +4215,7 @@ export const peopleDemo = [ city: 'New Nicholasshire', email: 'kenneth.mason@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/kenneth-mason-7bdf5e7f2b', jobTitle: 'Financial trader', }, @@ -4225,7 +4225,7 @@ export const peopleDemo = [ city: 'Port Jamesport', email: 'kent.mitchell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/kent-mitchell-2685d24cef', jobTitle: 'Horticultural consultant', }, @@ -4235,7 +4235,7 @@ export const peopleDemo = [ city: 'Robertsmouth', email: 'christine.parker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/christine-parker-53b7e92f21', jobTitle: 'Dietitian', }, @@ -4245,7 +4245,7 @@ export const peopleDemo = [ city: 'Marissashire', email: 'christopher.thomas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/christopher-thomas-50eba95625', jobTitle: 'Press photographer', }, @@ -4255,7 +4255,7 @@ export const peopleDemo = [ city: 'Amberborough', email: 'cole.mckenzie@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/cole-mckenzie-0febe188ad', jobTitle: 'Transport planner', }, @@ -4265,7 +4265,7 @@ export const peopleDemo = [ city: 'Hendersonview', email: 'john.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/john-jackson-eaf7698388', jobTitle: 'Engineer, land', }, @@ -4275,7 +4275,7 @@ export const peopleDemo = [ city: 'Christopherport', email: 'denise.gregory@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/denise-gregory-2a012e2939', jobTitle: 'Community arts worker', }, @@ -4285,7 +4285,7 @@ export const peopleDemo = [ city: 'Cassandrastad', email: 'deanna.mays@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/deanna-mays-d96cf68df8', jobTitle: 'Television camera operator', }, @@ -4295,7 +4295,7 @@ export const peopleDemo = [ city: 'Wallaceville', email: 'jennifer.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/jennifer-smith-cff724d712', jobTitle: 'Science writer', }, @@ -4305,7 +4305,7 @@ export const peopleDemo = [ city: 'Lake Justin', email: 'dylan.jimenez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/dylan-jimenez-611f4f7667', jobTitle: 'Runner, broadcasting/film/video', }, @@ -4315,7 +4315,7 @@ export const peopleDemo = [ city: 'East Sarah', email: 'amber.mullins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/amber-mullins-7c2cf23e98', jobTitle: 'Scientist, research (medical)', }, @@ -4325,7 +4325,7 @@ export const peopleDemo = [ city: 'Vegastad', email: 'kirsten.watson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/kirsten-watson-e01dabbf2c', jobTitle: 'Radiographer, therapeutic', }, @@ -4335,7 +4335,7 @@ export const peopleDemo = [ city: 'Lawsonberg', email: 'holly.winters@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/holly-winters-ad7c60377c', jobTitle: 'Public house manager', }, @@ -4345,7 +4345,7 @@ export const peopleDemo = [ city: 'New Angelaport', email: 'matthew.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/matthew-jenkins-d451b82929', jobTitle: 'Neurosurgeon', }, @@ -4355,7 +4355,7 @@ export const peopleDemo = [ city: 'Port Richard', email: 'elizabeth.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/elizabeth-williams-f8ae89860c', jobTitle: 'Ranger/warden', }, @@ -4365,7 +4365,7 @@ export const peopleDemo = [ city: 'Austinfort', email: 'sophia.carpenter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/sophia-carpenter-41f6371dcf', jobTitle: 'Translator', }, @@ -4375,7 +4375,7 @@ export const peopleDemo = [ city: 'Huffmanville', email: 'sarah.duke@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/sarah-duke-9c48b2bc47', jobTitle: 'Holiday representative', }, @@ -4385,7 +4385,7 @@ export const peopleDemo = [ city: 'South Loriport', email: 'colin.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/colin-smith-319020345b', jobTitle: 'Gaffer', }, @@ -4395,7 +4395,7 @@ export const peopleDemo = [ city: 'East Maryland', email: 'christine.baldwin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/christine-baldwin-849d34d1c9', jobTitle: 'Emergency planning/management officer', }, @@ -4405,7 +4405,7 @@ export const peopleDemo = [ city: 'Jeffmouth', email: 'michael.johns@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/michael-johns-d7c1497c08', jobTitle: 'Waste management officer', }, @@ -4415,7 +4415,7 @@ export const peopleDemo = [ city: 'New Jennifer', email: 'jeffery.griffin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/jeffery-griffin-755dd4b413', jobTitle: 'Embryologist, clinical', }, @@ -4425,7 +4425,7 @@ export const peopleDemo = [ city: 'West Leah', email: 'mike.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/mike-hernandez-619531cf07', jobTitle: 'Conservation officer, nature', }, @@ -4435,7 +4435,7 @@ export const peopleDemo = [ city: 'Morafurt', email: 'chelsea.robinson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/chelsea-robinson-65dedb9ed7', jobTitle: 'Restaurant manager, fast food', }, @@ -4445,7 +4445,7 @@ export const peopleDemo = [ city: 'Michaelview', email: 'derek.small@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/derek-small-3b02aeb21a', jobTitle: 'Nutritional therapist', }, @@ -4455,7 +4455,7 @@ export const peopleDemo = [ city: 'Port Chelsea', email: 'robin.miranda@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/robin-miranda-3d69d8721e', jobTitle: 'Logistics and distribution manager', }, @@ -4465,7 +4465,7 @@ export const peopleDemo = [ city: 'Allenstad', email: 'alexander.bryant@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/alexander-bryant-8e40b5c156', jobTitle: 'Museum education officer', }, @@ -4475,7 +4475,7 @@ export const peopleDemo = [ city: 'Lake Lauraville', email: 'jennifer.moody@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/jennifer-moody-92078fa8a0', jobTitle: 'Teacher, adult education', }, @@ -4485,7 +4485,7 @@ export const peopleDemo = [ city: 'Brittneymouth', email: 'kathleen.coleman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/kathleen-coleman-fbba0d93b5', jobTitle: 'Exhibition designer', }, @@ -4495,7 +4495,7 @@ export const peopleDemo = [ city: 'Gomeztown', email: 'miguel.malone@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/miguel-malone-931db4892a', jobTitle: 'Industrial buyer', }, @@ -4505,7 +4505,7 @@ export const peopleDemo = [ city: 'North Tiffany', email: 'eric.kramer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/eric-kramer-911574885e', jobTitle: 'Surveyor, planning and development', }, @@ -4515,7 +4515,7 @@ export const peopleDemo = [ city: 'Christinaberg', email: 'david.harmon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/david-harmon-a02ccbfe74', jobTitle: 'Trade mark attorney', }, @@ -4525,7 +4525,7 @@ export const peopleDemo = [ city: 'New Dustin', email: 'michael.turner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/michael-turner-c7db3d22d3', jobTitle: 'Illustrator', }, @@ -4535,7 +4535,7 @@ export const peopleDemo = [ city: 'Mannfurt', email: 'kim.nelson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/kim-nelson-e9c0c3ac3b', jobTitle: 'Archivist', }, @@ -4545,7 +4545,7 @@ export const peopleDemo = [ city: 'South Matthew', email: 'jason.mcmahon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/jason-mcmahon-17a1cf9c23', jobTitle: 'Oncologist', }, @@ -4555,7 +4555,7 @@ export const peopleDemo = [ city: 'Christinestad', email: 'spencer.mason@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/spencer-mason-511dce37fd', jobTitle: 'Investment analyst', }, @@ -4565,7 +4565,7 @@ export const peopleDemo = [ city: 'North Amanda', email: 'alison.barber@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/alison-barber-c6ac41c30c', jobTitle: 'Tourism officer', }, @@ -4575,7 +4575,7 @@ export const peopleDemo = [ city: 'East Pedro', email: 'alicia.kennedy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/alicia-kennedy-ffd86f7279', jobTitle: 'Civil engineer, consulting', }, @@ -4585,7 +4585,7 @@ export const peopleDemo = [ city: 'East Elizabethside', email: 'edward.parsons@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/edward-parsons-0815f3a265', jobTitle: 'Financial controller', }, @@ -4595,7 +4595,7 @@ export const peopleDemo = [ city: 'Smithville', email: 'justin.petersen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/justin-petersen-5f4849177a', jobTitle: 'Theatre director', }, @@ -4605,7 +4605,7 @@ export const peopleDemo = [ city: 'Port Davidchester', email: 'dawn.dixon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/dawn-dixon-ab7660d36a', jobTitle: 'Musician', }, @@ -4615,7 +4615,7 @@ export const peopleDemo = [ city: 'New Jennifer', email: 'douglas.ward@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/douglas-ward-a7e9b5a4ef', jobTitle: 'Midwife', }, @@ -4625,7 +4625,7 @@ export const peopleDemo = [ city: 'Granttown', email: 'linda.nguyen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/linda-nguyen-9543ff5b4b', jobTitle: 'General practice doctor', }, @@ -4635,7 +4635,7 @@ export const peopleDemo = [ city: 'Courtneystad', email: 'nicole.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/nicole-hernandez-c31e2859c3', jobTitle: 'Teacher, early years/pre', }, @@ -4645,7 +4645,7 @@ export const peopleDemo = [ city: 'South Brandyland', email: 'anne.massey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/anne-massey-68ae91f9e7', jobTitle: 'Engineer, energy', }, @@ -4655,7 +4655,7 @@ export const peopleDemo = [ city: 'Rachelfurt', email: 'jenny.esparza@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/jenny-esparza-0ce6d348d2', jobTitle: 'Trade union research officer', }, @@ -4665,7 +4665,7 @@ export const peopleDemo = [ city: 'Alejandromouth', email: 'robert.ward@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/robert-ward-9dbe3cfea4', jobTitle: 'Air broker', }, @@ -4675,7 +4675,7 @@ export const peopleDemo = [ city: 'Hughesshire', email: 'melissa.farrell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/melissa-farrell-e7db325f4e', jobTitle: 'Sports administrator', }, @@ -4685,7 +4685,7 @@ export const peopleDemo = [ city: 'Johnburgh', email: 'stephen.powers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/stephen-powers-848a849fac', jobTitle: 'Occupational psychologist', }, @@ -4695,7 +4695,7 @@ export const peopleDemo = [ city: 'West Elizabethberg', email: 'robin.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/robin-brown-95dea5c792', jobTitle: 'Publishing rights manager', }, @@ -4705,7 +4705,7 @@ export const peopleDemo = [ city: 'Kimberlyborough', email: 'wanda.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/wanda-moore-846aac522b', jobTitle: 'Chief Technology Officer', }, @@ -4715,7 +4715,7 @@ export const peopleDemo = [ city: 'Robinsonmouth', email: 'danielle.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/danielle-brown-9468c67469', jobTitle: 'Biochemist, clinical', }, @@ -4725,7 +4725,7 @@ export const peopleDemo = [ city: 'Armstrongbury', email: 'timothy.phillips@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/timothy-phillips-f02b8125a6', jobTitle: 'Medical sales representative', }, @@ -4735,7 +4735,7 @@ export const peopleDemo = [ city: 'Elizabethfurt', email: 'daniel.baker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/daniel-baker-b080dba2cf', jobTitle: 'Learning disability nurse', }, @@ -4745,7 +4745,7 @@ export const peopleDemo = [ city: 'Jorgeside', email: 'jason.parker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/jason-parker-c8f6658eb8', jobTitle: 'Public relations officer', }, @@ -4755,7 +4755,7 @@ export const peopleDemo = [ city: 'New Michael', email: 'donald.roy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/donald-roy-81242ff295', jobTitle: 'Ambulance person', }, @@ -4765,7 +4765,7 @@ export const peopleDemo = [ city: 'Karinaberg', email: 'cameron.beck@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/cameron-beck-4a5b8ac2f9', jobTitle: 'Animal technologist', }, @@ -4775,7 +4775,7 @@ export const peopleDemo = [ city: 'West Steven', email: 'christina.carter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/christina-carter-20e5617e72', jobTitle: 'Historic buildings inspector/conservation officer', }, @@ -4785,7 +4785,7 @@ export const peopleDemo = [ city: 'West Melissa', email: 'roy.jackson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/roy-jackson-12ef80ee4f', jobTitle: 'Theatre stage manager', }, @@ -4795,7 +4795,7 @@ export const peopleDemo = [ city: 'Jessicaburgh', email: 'valerie.green@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/valerie-green-c4a084e4af', jobTitle: 'Dramatherapist', }, @@ -4805,7 +4805,7 @@ export const peopleDemo = [ city: 'Wendymouth', email: 'ryan.parker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/ryan-parker-f86f7a2a2c', jobTitle: 'Catering manager', }, @@ -4815,7 +4815,7 @@ export const peopleDemo = [ city: 'East Willie', email: 'spencer.cortez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/spencer-cortez-8653fe9874', jobTitle: 'Logistics and distribution manager', }, @@ -4825,7 +4825,7 @@ export const peopleDemo = [ city: 'Tracyville', email: 'jacqueline.freeman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/jacqueline-freeman-4dc0ff463f', jobTitle: 'English as a second language teacher', }, @@ -4835,7 +4835,7 @@ export const peopleDemo = [ city: 'North Henry', email: 'joanne.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/joanne-hernandez-22617ea91c', jobTitle: 'Research scientist (maths)', }, @@ -4845,7 +4845,7 @@ export const peopleDemo = [ city: 'Cathyberg', email: 'brandon.randolph@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/brandon-randolph-f206b641f3', jobTitle: 'Games developer', }, @@ -4855,7 +4855,7 @@ export const peopleDemo = [ city: 'Port Crystalland', email: 'william.wells@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/william-wells-ddf1c0e80e', jobTitle: 'Teacher, secondary school', }, @@ -4865,7 +4865,7 @@ export const peopleDemo = [ city: 'Pereztown', email: 'monica.wall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/monica-wall-18fcd0b442', jobTitle: 'Clinical molecular geneticist', }, @@ -4875,7 +4875,7 @@ export const peopleDemo = [ city: 'Fernandezport', email: 'patricia.whitehead@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/patricia-whitehead-b76f160790', jobTitle: 'Learning disability nurse', }, @@ -4885,7 +4885,7 @@ export const peopleDemo = [ city: 'Port Jessicatown', email: 'chelsey.cruz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/chelsey-cruz-3927a3b54f', jobTitle: 'Accountant, chartered management', }, @@ -4895,7 +4895,7 @@ export const peopleDemo = [ city: 'Georgeville', email: 'marie.herrera@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/marie-herrera-3c032f2c87', jobTitle: 'Teacher, English as a foreign language', }, @@ -4905,7 +4905,7 @@ export const peopleDemo = [ city: 'South Heatherstad', email: 'gail.russell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/gail-russell-47081f3909', jobTitle: 'Building services engineer', }, @@ -4915,7 +4915,7 @@ export const peopleDemo = [ city: 'Stephanieville', email: 'christopher.whitehead@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/christopher-whitehead-907e5c7217', jobTitle: 'Research officer, trade union', }, @@ -4925,7 +4925,7 @@ export const peopleDemo = [ city: 'Lake Terri', email: 'vicki.gonzales@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/vicki-gonzales-a6b6db11ce', jobTitle: 'Psychologist, counselling', }, @@ -4935,7 +4935,7 @@ export const peopleDemo = [ city: 'Sydneyfurt', email: 'paul.graham@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/paul-graham-0159c4f113', jobTitle: 'Surveyor, commercial/residential', }, @@ -4945,7 +4945,7 @@ export const peopleDemo = [ city: 'Michelleborough', email: 'john.carter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/john-carter-c39f5f879b', jobTitle: 'Cabin crew', }, @@ -4955,7 +4955,7 @@ export const peopleDemo = [ city: 'South Alexandra', email: 'dennis.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/dennis-taylor-81e83d5e15', jobTitle: 'Sports development officer', }, @@ -4965,7 +4965,7 @@ export const peopleDemo = [ city: 'Willietown', email: 'gail.salinas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/gail-salinas-6fd9ac936d', jobTitle: 'Theatre stage manager', }, @@ -4975,7 +4975,7 @@ export const peopleDemo = [ city: 'Kleinfort', email: 'stacey.doyle@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/stacey-doyle-c2b53da3c1', jobTitle: 'Engineer, water', }, @@ -4985,7 +4985,7 @@ export const peopleDemo = [ city: 'Rogersberg', email: 'nicholas.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/nicholas-jones-37d87e9bef', jobTitle: 'Runner, broadcasting/film/video', }, @@ -4995,7 +4995,7 @@ export const peopleDemo = [ city: 'Staceyberg', email: 'sheri.donaldson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/sheri-donaldson-aef36dc097', jobTitle: 'Dentist', }, @@ -5005,7 +5005,7 @@ export const peopleDemo = [ city: 'Theresatown', email: 'christopher.christensen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/christopher-christensen-0405d38686', jobTitle: 'Public relations officer', }, @@ -5015,7 +5015,7 @@ export const peopleDemo = [ city: 'Shortville', email: 'joshua.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/joshua-hernandez-aaf7b631ad', jobTitle: 'Therapist, sports', }, @@ -5025,7 +5025,7 @@ export const peopleDemo = [ city: 'Benjaminland', email: 'ryan.walter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/ryan-walter-3f704e09d0', jobTitle: 'Therapeutic radiographer', }, @@ -5035,7 +5035,7 @@ export const peopleDemo = [ city: 'Mccoyland', email: 'brandy.trevino@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/brandy-trevino-a6f561480a', jobTitle: 'Herpetologist', }, @@ -5045,7 +5045,7 @@ export const peopleDemo = [ city: 'Lake Normanfurt', email: 'john.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/john-martinez-9b2c36ab60', jobTitle: 'Armed forces technical officer', }, @@ -5055,7 +5055,7 @@ export const peopleDemo = [ city: 'Michaelmouth', email: 'jennifer.morris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/jennifer-morris-94daa88e52', jobTitle: 'Homeopath', }, @@ -5065,7 +5065,7 @@ export const peopleDemo = [ city: 'Robinhaven', email: 'amanda.barnett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/amanda-barnett-6714ab7883', jobTitle: 'Clothing/textile technologist', }, @@ -5075,7 +5075,7 @@ export const peopleDemo = [ city: 'Larsonstad', email: 'tanner.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/tanner-miller-dcf9fa91b8', jobTitle: 'Trading standards officer', }, @@ -5085,7 +5085,7 @@ export const peopleDemo = [ city: 'Mikaylamouth', email: 'bobby.sanchez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/bobby-sanchez-b2c15c3790', jobTitle: 'Scientist, marine', }, @@ -5095,7 +5095,7 @@ export const peopleDemo = [ city: 'West Alejandroborough', email: 'brian.cortez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/brian-cortez-77d96367ee', jobTitle: 'Metallurgist', }, @@ -5105,7 +5105,7 @@ export const peopleDemo = [ city: 'Russellville', email: 'misty.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/misty-jenkins-7bb3609670', jobTitle: 'Teacher, primary school', }, @@ -5115,7 +5115,7 @@ export const peopleDemo = [ city: 'Lucasland', email: 'erin.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/erin-hernandez-701db083a7', jobTitle: 'Textile designer', }, @@ -5125,7 +5125,7 @@ export const peopleDemo = [ city: 'Shahville', email: 'victoria.larsen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/victoria-larsen-8bc60c5ec2', jobTitle: 'General practice doctor', }, @@ -5135,7 +5135,7 @@ export const peopleDemo = [ city: 'Port Tonyaview', email: 'brian.diaz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/brian-diaz-97d66d36ca', jobTitle: 'Television floor manager', }, @@ -5145,7 +5145,7 @@ export const peopleDemo = [ city: 'Rebeccaton', email: 'krista.murphy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/krista-murphy-51da3fe5d2', jobTitle: 'Social researcher', }, @@ -5155,7 +5155,7 @@ export const peopleDemo = [ city: 'Grimeston', email: 'calvin.bond@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/calvin-bond-59b4aeacff', jobTitle: 'Early years teacher', }, @@ -5165,7 +5165,7 @@ export const peopleDemo = [ city: 'South Tyler', email: 'terry.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/terry-perez-68f6738ace', jobTitle: 'Teaching laboratory technician', }, @@ -5175,7 +5175,7 @@ export const peopleDemo = [ city: 'Holmeschester', email: 'stephen.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/stephen-wilson-009cf79121', jobTitle: 'Development worker, international aid', }, @@ -5185,7 +5185,7 @@ export const peopleDemo = [ city: 'South Kelly', email: 'lisa.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/lisa-johnson-d9406b185e', jobTitle: 'Archaeologist', }, @@ -5195,7 +5195,7 @@ export const peopleDemo = [ city: 'Jenniferview', email: 'tim.torres@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/tim-torres-9031065f9f', jobTitle: 'Accounting technician', }, @@ -5205,7 +5205,7 @@ export const peopleDemo = [ city: 'West Jamie', email: 'claudia.sosa@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/claudia-sosa-8b95753698', jobTitle: 'Retail buyer', }, @@ -5215,7 +5215,7 @@ export const peopleDemo = [ city: 'New Sandra', email: 'steven.higgins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/steven-higgins-004f3810c4', jobTitle: 'Environmental health practitioner', }, @@ -5225,7 +5225,7 @@ export const peopleDemo = [ city: 'New Amanda', email: 'james.benson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/james-benson-6cc556ac5b', jobTitle: 'Games developer', }, @@ -5235,7 +5235,7 @@ export const peopleDemo = [ city: 'Courtneystad', email: 'tyler.bishop@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/tyler-bishop-b9b8afc542', jobTitle: 'Surveyor, land/geomatics', }, @@ -5245,7 +5245,7 @@ export const peopleDemo = [ city: 'New Jeremy', email: 'monica.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/monica-smith-df2db66e3e', jobTitle: 'Exercise physiologist', }, @@ -5255,7 +5255,7 @@ export const peopleDemo = [ city: 'South Brian', email: 'jillian.carter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/jillian-carter-9a44a80690', jobTitle: 'Forest/woodland manager', }, @@ -5265,7 +5265,7 @@ export const peopleDemo = [ city: 'New Juan', email: 'roberta.graves@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/roberta-graves-5584549fbc', jobTitle: 'Camera operator', }, @@ -5275,7 +5275,7 @@ export const peopleDemo = [ city: 'Tylerfort', email: 'clarence.flores@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/clarence-flores-bc6d7c12e4', jobTitle: 'Publishing copy', }, @@ -5285,7 +5285,7 @@ export const peopleDemo = [ city: 'West Marc', email: 'robert.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/robert-gonzalez-624dd9bd66', jobTitle: 'Geologist, engineering', }, @@ -5295,7 +5295,7 @@ export const peopleDemo = [ city: 'Boltonview', email: 'melissa.lucas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/melissa-lucas-d06ec4efc8', jobTitle: 'Youth worker', }, @@ -5305,7 +5305,7 @@ export const peopleDemo = [ city: 'West Kevinfurt', email: 'lee.lewis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/lee-lewis-c341df06f3', jobTitle: 'Graphic designer', }, @@ -5315,7 +5315,7 @@ export const peopleDemo = [ city: 'Ortizchester', email: 'jessica.preston@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/jessica-preston-011a349bc5', jobTitle: 'Therapist, art', }, @@ -5325,7 +5325,7 @@ export const peopleDemo = [ city: 'North Jason', email: 'henry.west@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/henry-west-168b570375', jobTitle: 'Programme researcher, broadcasting/film/video', }, @@ -5335,7 +5335,7 @@ export const peopleDemo = [ city: 'East Jordan', email: 'kristin.sanchez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/kristin-sanchez-16a334fc5e', jobTitle: 'Hydrologist', }, @@ -5345,7 +5345,7 @@ export const peopleDemo = [ city: 'New Raymondport', email: 'derek.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/derek-davis-919fba4163', jobTitle: 'Logistics and distribution manager', }, @@ -5355,7 +5355,7 @@ export const peopleDemo = [ city: 'Grantside', email: 'dan.gonzales@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/dan-gonzales-83d3f8867f', jobTitle: 'Location manager', }, @@ -5365,7 +5365,7 @@ export const peopleDemo = [ city: 'South Bradley', email: 'edwin.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/edwin-garcia-7bcfaf8b3a', jobTitle: 'Nurse, mental health', }, @@ -5375,7 +5375,7 @@ export const peopleDemo = [ city: 'New Mark', email: 'tonya.hooper@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/tonya-hooper-da5e36c5c5', jobTitle: 'Water engineer', }, @@ -5385,7 +5385,7 @@ export const peopleDemo = [ city: 'South Johnhaven', email: 'jennifer.tate@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/jennifer-tate-ffbd43bb7e', jobTitle: 'Hospital doctor', }, @@ -5395,7 +5395,7 @@ export const peopleDemo = [ city: 'Booneton', email: 'earl.higgins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/earl-higgins-1785700a3f', jobTitle: 'Administrator, sports', }, @@ -5405,7 +5405,7 @@ export const peopleDemo = [ city: 'Johnsonfurt', email: 'sandra.werner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/sandra-werner-ca50c4d94a', jobTitle: 'Administrator', }, @@ -5415,7 +5415,7 @@ export const peopleDemo = [ city: 'North Andre', email: 'brian.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/brian-johnson-1b262bb370', jobTitle: 'Environmental consultant', }, @@ -5425,7 +5425,7 @@ export const peopleDemo = [ city: 'Hoborough', email: 'jacqueline.bell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/jacqueline-bell-41ae313193', jobTitle: 'Advice worker', }, @@ -5435,7 +5435,7 @@ export const peopleDemo = [ city: 'South Robert', email: 'jeffery.gibson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/jeffery-gibson-33288473cf', jobTitle: 'Therapist, occupational', }, @@ -5445,7 +5445,7 @@ export const peopleDemo = [ city: 'Oconnorton', email: 'jacqueline.snyder@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/jacqueline-snyder-cdfcd4794c', jobTitle: 'Press photographer', }, @@ -5455,7 +5455,7 @@ export const peopleDemo = [ city: 'West Sarah', email: 'john.stone@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/john-stone-ed440ef104', jobTitle: 'Advertising account executive', }, @@ -5465,7 +5465,7 @@ export const peopleDemo = [ city: 'New Debraville', email: 'elizabeth.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/elizabeth-allen-f3a164c8d0', jobTitle: 'Animator', }, @@ -5475,7 +5475,7 @@ export const peopleDemo = [ city: 'New Michael', email: 'daniel.knight@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/daniel-knight-71d19ed590', jobTitle: 'Producer, television/film/video', }, @@ -5485,7 +5485,7 @@ export const peopleDemo = [ city: 'Birdland', email: 'whitney.thomas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/whitney-thomas-ef126a0de0', jobTitle: 'Ophthalmologist', }, @@ -5495,7 +5495,7 @@ export const peopleDemo = [ city: 'Krististad', email: 'christina.anderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/christina-anderson-f8458640de', jobTitle: 'Museum education officer', }, @@ -5505,7 +5505,7 @@ export const peopleDemo = [ city: 'West Jamesview', email: 'joseph.peterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/joseph-peterson-0735ee6e8a', jobTitle: 'Actor', }, @@ -5515,7 +5515,7 @@ export const peopleDemo = [ city: 'Katrinabury', email: 'larry.graham@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/larry-graham-ec62249904', jobTitle: 'Surveyor, planning and development', }, @@ -5525,7 +5525,7 @@ export const peopleDemo = [ city: 'South Charlesmouth', email: 'rachael.fox@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/rachael-fox-736e697d7a', jobTitle: 'Senior tax professional/tax inspector', }, @@ -5535,7 +5535,7 @@ export const peopleDemo = [ city: 'West Amyborough', email: 'christopher.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/christopher-wilson-f6db69b44e', jobTitle: 'Lecturer, further education', }, @@ -5545,7 +5545,7 @@ export const peopleDemo = [ city: 'New Connie', email: 'aaron.mccarty@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/aaron-mccarty-4207ebed52', jobTitle: 'Engineer, structural', }, @@ -5555,7 +5555,7 @@ export const peopleDemo = [ city: 'Jamieberg', email: 'albert.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/albert-taylor-715173cf8c', jobTitle: 'Exercise physiologist', }, @@ -5565,7 +5565,7 @@ export const peopleDemo = [ city: 'Port Anaside', email: 'laura.diaz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/laura-diaz-83aa93da5c', jobTitle: 'Clinical embryologist', }, @@ -5575,7 +5575,7 @@ export const peopleDemo = [ city: 'West Michael', email: 'hannah.craig@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/hannah-craig-e5a9be07cf', jobTitle: 'Air traffic controller', }, @@ -5585,7 +5585,7 @@ export const peopleDemo = [ city: 'New Cory', email: 'jessica.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/jessica-smith-59aaca3a47', jobTitle: 'Seismic interpreter', }, @@ -5595,7 +5595,7 @@ export const peopleDemo = [ city: 'Matthewtown', email: 'michael.george@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/michael-george-2f16bc3685', jobTitle: 'Insurance risk surveyor', }, @@ -5605,7 +5605,7 @@ export const peopleDemo = [ city: 'Port Anna', email: 'ronald.hogan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/ronald-hogan-67f4504d1b', jobTitle: 'Engineer, electrical', }, @@ -5615,7 +5615,7 @@ export const peopleDemo = [ city: 'Mariomouth', email: 'elizabeth.wright@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/elizabeth-wright-3d2908f9b7', jobTitle: 'Legal secretary', }, @@ -5625,7 +5625,7 @@ export const peopleDemo = [ city: 'Bowenfort', email: 'thomas.zimmerman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/thomas-zimmerman-83ce4ea20a', jobTitle: 'Advertising art director', }, @@ -5635,7 +5635,7 @@ export const peopleDemo = [ city: 'Lake Christopher', email: 'judith.harris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/judith-harris-c394ff92d4', jobTitle: 'Teacher, secondary school', }, @@ -5645,7 +5645,7 @@ export const peopleDemo = [ city: 'Lake Jeffrey', email: 'james.massey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/james-massey-14adc3c2b2', jobTitle: 'IT sales professional', }, @@ -5655,7 +5655,7 @@ export const peopleDemo = [ city: 'South Jeremyberg', email: 'louis.huynh@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/louis-huynh-8409c77412', jobTitle: 'Education officer, environmental', }, @@ -5665,7 +5665,7 @@ export const peopleDemo = [ city: 'South Angela', email: 'lori.alexander@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/lori-alexander-06fabb279c', jobTitle: 'Diplomatic Services operational officer', }, @@ -5675,7 +5675,7 @@ export const peopleDemo = [ city: 'Crosschester', email: 'anna.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/anna-moore-209cbeb00b', jobTitle: 'Armed forces operational officer', }, @@ -5685,7 +5685,7 @@ export const peopleDemo = [ city: 'Lambville', email: 'richard.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/richard-smith-37150fdba0', jobTitle: 'Administrator, arts', }, @@ -5695,7 +5695,7 @@ export const peopleDemo = [ city: 'Rodriguezstad', email: 'eric.hunter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/eric-hunter-5944420676', jobTitle: 'Osteopath', }, @@ -5705,7 +5705,7 @@ export const peopleDemo = [ city: 'Lake Katherine', email: 'cody.todd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/cody-todd-e59e1908cb', jobTitle: 'Financial planner', }, @@ -5715,7 +5715,7 @@ export const peopleDemo = [ city: 'North Michael', email: 'nicole.patel@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/nicole-patel-d52be83244', jobTitle: 'Glass blower/designer', }, @@ -5725,7 +5725,7 @@ export const peopleDemo = [ city: 'Mitchellside', email: 'eric.rivera@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/eric-rivera-17c826bef0', jobTitle: 'Teacher, primary school', }, @@ -5735,7 +5735,7 @@ export const peopleDemo = [ city: 'Nicholsview', email: 'amy.hall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/amy-hall-5b6ee3fd83', jobTitle: 'Recycling officer', }, @@ -5745,7 +5745,7 @@ export const peopleDemo = [ city: 'Kaitlynton', email: 'randy.trujillo@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/randy-trujillo-c9c04e7445', jobTitle: 'Production engineer', }, @@ -5755,7 +5755,7 @@ export const peopleDemo = [ city: 'South Reneestad', email: 'ashley.conner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/ashley-conner-7e1f59d6d0', jobTitle: 'Chief Technology Officer', }, @@ -5765,7 +5765,7 @@ export const peopleDemo = [ city: 'Lisamouth', email: 'adriana.larsen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/adriana-larsen-790fff0714', jobTitle: 'Financial manager', }, @@ -5775,7 +5775,7 @@ export const peopleDemo = [ city: 'Baileytown', email: 'lindsey.reid@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/lindsey-reid-9408f405f8', jobTitle: 'Geneticist, molecular', }, @@ -5785,7 +5785,7 @@ export const peopleDemo = [ city: 'Clarkton', email: 'sophia.collins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/sophia-collins-65701c7f41', jobTitle: 'Proofreader', }, @@ -5795,7 +5795,7 @@ export const peopleDemo = [ city: 'Lake Robert', email: 'joshua.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/joshua-martin-3db7199eeb', jobTitle: 'Diplomatic Services operational officer', }, @@ -5805,7 +5805,7 @@ export const peopleDemo = [ city: 'New Rebeccaside', email: 'james.adams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/james-adams-fade145464', jobTitle: 'Field trials officer', }, @@ -5815,7 +5815,7 @@ export const peopleDemo = [ city: 'New Kimberly', email: 'maureen.clay@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/maureen-clay-44447f851e', jobTitle: 'Ceramics designer', }, @@ -5825,7 +5825,7 @@ export const peopleDemo = [ city: 'Karenshire', email: 'brenda.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/brenda-moore-431d562885', jobTitle: 'Therapist, nutritional', }, @@ -5835,7 +5835,7 @@ export const peopleDemo = [ city: 'Port Nicoleland', email: 'kathy.glover@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/kathy-glover-1481622628', jobTitle: 'IT consultant', }, @@ -5845,7 +5845,7 @@ export const peopleDemo = [ city: 'Port Gary', email: 'jeffrey.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/jeffrey-jones-74128aa05c', jobTitle: 'Corporate treasurer', }, @@ -5855,7 +5855,7 @@ export const peopleDemo = [ city: 'Millerborough', email: 'dylan.ramirez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/dylan-ramirez-de68206ac6', jobTitle: 'Textile designer', }, @@ -5865,7 +5865,7 @@ export const peopleDemo = [ city: 'Andrewburgh', email: 'derek.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/derek-brown-0bd369600e', jobTitle: 'Maintenance engineer', }, @@ -5875,7 +5875,7 @@ export const peopleDemo = [ city: 'New Amandaville', email: 'nicole.robles@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/nicole-robles-dc92d47af5', jobTitle: 'Customer service manager', }, @@ -5885,7 +5885,7 @@ export const peopleDemo = [ city: 'West Bill', email: 'lauren.murray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/lauren-murray-f56eac5c8b', jobTitle: 'Theatre director', }, @@ -5895,7 +5895,7 @@ export const peopleDemo = [ city: 'Jesusmouth', email: 'vanessa.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/vanessa-jones-83ad40ef01', jobTitle: 'Geophysical data processor', }, @@ -5905,7 +5905,7 @@ export const peopleDemo = [ city: 'Sandersland', email: 'joel.lopez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/joel-lopez-6b55707c2c', jobTitle: 'Civil engineer, contracting', }, @@ -5915,7 +5915,7 @@ export const peopleDemo = [ city: 'North Reneechester', email: 'matthew.peterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/matthew-peterson-55c3973b59', jobTitle: 'Chief Marketing Officer', }, @@ -5925,7 +5925,7 @@ export const peopleDemo = [ city: 'West Sheilaview', email: 'elaine.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/elaine-gonzalez-cb7b12c50a', jobTitle: 'Psychologist, educational', }, @@ -5935,7 +5935,7 @@ export const peopleDemo = [ city: 'New Alexander', email: 'charles.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/charles-jones-dce2f7c6f3', jobTitle: 'Research officer, government', }, @@ -5945,7 +5945,7 @@ export const peopleDemo = [ city: 'New Melissa', email: 'rachel.barton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/rachel-barton-3d81266f5e', jobTitle: 'Cartographer', }, @@ -5955,7 +5955,7 @@ export const peopleDemo = [ city: 'Lake Curtishaven', email: 'alyssa.ellis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/alyssa-ellis-c173bc4670', jobTitle: 'Sales promotion account executive', }, @@ -5965,7 +5965,7 @@ export const peopleDemo = [ city: 'East Ricardo', email: 'patricia.lopez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/patricia-lopez-149ef1e411', jobTitle: 'Consulting civil engineer', }, @@ -5975,7 +5975,7 @@ export const peopleDemo = [ city: 'North Kristen', email: 'scott.moran@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/scott-moran-6acf4736cf', jobTitle: 'Heritage manager', }, @@ -5985,7 +5985,7 @@ export const peopleDemo = [ city: 'New Williamshire', email: 'jerome.morris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/jerome-morris-c94af1fc97', jobTitle: 'Systems analyst', }, @@ -5995,7 +5995,7 @@ export const peopleDemo = [ city: 'Deckerfort', email: 'christopher.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/christopher-davis-4820898e5c', jobTitle: 'Building control surveyor', }, @@ -6005,7 +6005,7 @@ export const peopleDemo = [ city: 'Port Gabrielle', email: 'jessica.downs@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/jessica-downs-9f35d94f5c', jobTitle: 'Solicitor', }, @@ -6015,7 +6015,7 @@ export const peopleDemo = [ city: 'Tonytown', email: 'eric.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/eric-jenkins-aad0386096', jobTitle: 'International aid/development worker', }, @@ -6025,7 +6025,7 @@ export const peopleDemo = [ city: 'South Tammy', email: 'christy.ramsey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/christy-ramsey-5a861e76c3', jobTitle: 'Pharmacist, hospital', }, @@ -6035,7 +6035,7 @@ export const peopleDemo = [ city: 'Lake Sarah', email: 'sarah.evans@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/sarah-evans-f7501659e6', jobTitle: 'Exhibitions officer, museum/gallery', }, @@ -6045,7 +6045,7 @@ export const peopleDemo = [ city: 'Gordonview', email: 'stanley.thomas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/stanley-thomas-76273965ad', jobTitle: 'Production assistant, television', }, @@ -6055,7 +6055,7 @@ export const peopleDemo = [ city: 'Morganbury', email: 'hannah.watts@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/hannah-watts-a4b37f2ae2', jobTitle: 'Learning mentor', }, @@ -6065,7 +6065,7 @@ export const peopleDemo = [ city: 'Christensenville', email: 'michael.maldonado@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/michael-maldonado-94c0630389', jobTitle: 'Drilling engineer', }, @@ -6075,7 +6075,7 @@ export const peopleDemo = [ city: 'New Raymond', email: 'joseph.nguyen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/joseph-nguyen-3f0a744a58', jobTitle: 'Chartered management accountant', }, @@ -6085,7 +6085,7 @@ export const peopleDemo = [ city: 'North Julieberg', email: 'erin.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/erin-garcia-f9967e5d34', jobTitle: 'Science writer', }, @@ -6095,7 +6095,7 @@ export const peopleDemo = [ city: 'Heatherside', email: 'eric.howell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/eric-howell-cfccf703d3', jobTitle: 'Merchandiser, retail', }, @@ -6105,7 +6105,7 @@ export const peopleDemo = [ city: 'Jameshaven', email: 'alexandra.atkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/alexandra-atkins-5fe1b160c9', jobTitle: 'Media planner', }, @@ -6115,7 +6115,7 @@ export const peopleDemo = [ city: 'Lake Alexandra', email: 'raymond.mcdonald@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/raymond-mcdonald-68f29c3dc8', jobTitle: 'Multimedia specialist', }, @@ -6125,7 +6125,7 @@ export const peopleDemo = [ city: 'Karlaburgh', email: 'joseph.barrett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/joseph-barrett-942d11d82f', jobTitle: 'Geophysical data processor', }, @@ -6135,7 +6135,7 @@ export const peopleDemo = [ city: 'Parkerview', email: 'lisa.salazar@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/lisa-salazar-04ed290031', jobTitle: 'Health and safety inspector', }, @@ -6145,7 +6145,7 @@ export const peopleDemo = [ city: 'Port Keithstad', email: 'erica.andrade@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/erica-andrade-64ca983404', jobTitle: 'Lecturer, higher education', }, @@ -6155,7 +6155,7 @@ export const peopleDemo = [ city: 'Courtneyberg', email: 'adam.wright@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/adam-wright-bba29daead', jobTitle: 'Psychotherapist, child', }, @@ -6165,7 +6165,7 @@ export const peopleDemo = [ city: 'New Jaymouth', email: 'michael.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/michael-williams-ab0d587602', jobTitle: 'Agricultural engineer', }, @@ -6175,7 +6175,7 @@ export const peopleDemo = [ city: 'Anthonyfort', email: 'margaret.morales@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/margaret-morales-7c6376f64a', jobTitle: 'Scientist, product/process development', }, @@ -6185,7 +6185,7 @@ export const peopleDemo = [ city: 'Amberfurt', email: 'david.nelson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/david-nelson-a0ef7e99f1', jobTitle: 'Electrical engineer', }, @@ -6195,7 +6195,7 @@ export const peopleDemo = [ city: 'Haleberg', email: 'holly.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/holly-allen-f9240b572e', jobTitle: 'Chief of Staff', }, @@ -6205,7 +6205,7 @@ export const peopleDemo = [ city: 'Carterfurt', email: 'cory.hicks@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/cory-hicks-63ff8f7cf5', jobTitle: 'Charity officer', }, @@ -6215,7 +6215,7 @@ export const peopleDemo = [ city: 'Krystalmouth', email: 'michael.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/michael-johnson-db4d9dc5c3', jobTitle: 'Futures trader', }, @@ -6225,7 +6225,7 @@ export const peopleDemo = [ city: 'Huynhfurt', email: 'ronnie.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/ronnie-martinez-11500b150a', jobTitle: 'Camera operator', }, @@ -6235,7 +6235,7 @@ export const peopleDemo = [ city: 'Charlesstad', email: 'jason.holden@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/jason-holden-1ebdcf4241', jobTitle: 'Automotive engineer', }, @@ -6245,7 +6245,7 @@ export const peopleDemo = [ city: 'East Toddfort', email: 'patrick.gilbert@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/patrick-gilbert-61fd32c01e', jobTitle: 'Learning disability nurse', }, @@ -6255,7 +6255,7 @@ export const peopleDemo = [ city: 'West Derekbury', email: 'sean.white@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/sean-white-dad0c13021', jobTitle: 'Scientist, research (physical sciences)', }, @@ -6265,7 +6265,7 @@ export const peopleDemo = [ city: 'East Timothyfort', email: 'valerie.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/valerie-martinez-1517f8ad33', jobTitle: 'Biomedical engineer', }, @@ -6275,7 +6275,7 @@ export const peopleDemo = [ city: 'Morrisonville', email: 'james.hawkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/james-hawkins-1cfac0a73a', jobTitle: 'Designer, textile', }, @@ -6285,7 +6285,7 @@ export const peopleDemo = [ city: 'East Jeanette', email: 'mckenzie.meyer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/mckenzie-meyer-51b7a8a764', jobTitle: 'Pharmacist, community', }, @@ -6295,7 +6295,7 @@ export const peopleDemo = [ city: 'New Ronaldhaven', email: 'parker.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/parker-young-7d7d0c964f', jobTitle: 'Electronics engineer', }, @@ -6305,7 +6305,7 @@ export const peopleDemo = [ city: 'Charlotteburgh', email: 'john.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/john-johnson-08ef07ffe9', jobTitle: 'Control and instrumentation engineer', }, @@ -6315,7 +6315,7 @@ export const peopleDemo = [ city: 'Williamsshire', email: 'sierra.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/sierra-rodriguez-5808a345e5', jobTitle: 'Sports therapist', }, @@ -6325,7 +6325,7 @@ export const peopleDemo = [ city: 'North Joshua', email: 'patricia.thompson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/patricia-thompson-8942d96ec6', jobTitle: 'Dance movement psychotherapist', }, @@ -6335,7 +6335,7 @@ export const peopleDemo = [ city: 'Lake Paulaport', email: 'valerie.durham@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/valerie-durham-4fef3b9462', jobTitle: 'Graphic designer', }, @@ -6345,7 +6345,7 @@ export const peopleDemo = [ city: 'New Christina', email: 'michael.tucker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/michael-tucker-6c66202d7e', jobTitle: 'Advertising account executive', }, @@ -6355,7 +6355,7 @@ export const peopleDemo = [ city: 'Justinfurt', email: 'martin.hayes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/martin-hayes-7645eada45', jobTitle: 'Occupational therapist', }, @@ -6365,7 +6365,7 @@ export const peopleDemo = [ city: 'New Ericland', email: 'brittany.watkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/brittany-watkins-af3eeda76b', jobTitle: 'Designer, jewellery', }, @@ -6375,7 +6375,7 @@ export const peopleDemo = [ city: 'Perkinsshire', email: 'jacob.dixon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/jacob-dixon-81d8755b18', jobTitle: 'Farm manager', }, @@ -6385,7 +6385,7 @@ export const peopleDemo = [ city: 'New Erin', email: 'sheila.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/sheila-wilson-b61719f6d6', jobTitle: 'Doctor, hospital', }, @@ -6395,7 +6395,7 @@ export const peopleDemo = [ city: 'Hooperside', email: 'lee.oliver@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/lee-oliver-a4da08a2ee', jobTitle: 'Office manager', }, @@ -6405,7 +6405,7 @@ export const peopleDemo = [ city: 'Joeport', email: 'aaron.moreno@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/aaron-moreno-cf9f02bc3b', jobTitle: 'Industrial buyer', }, @@ -6415,7 +6415,7 @@ export const peopleDemo = [ city: 'New Trevorhaven', email: 'diana.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/diana-garcia-62a0597c29', jobTitle: 'Engineer, biomedical', }, @@ -6425,7 +6425,7 @@ export const peopleDemo = [ city: 'South Scottfort', email: 'jonathan.harvey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/jonathan-harvey-8c1d8e255a', jobTitle: 'English as a second language teacher', }, @@ -6435,7 +6435,7 @@ export const peopleDemo = [ city: 'Randallchester', email: 'suzanne.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/suzanne-rodriguez-dffb493dbb', jobTitle: 'Games developer', }, @@ -6445,7 +6445,7 @@ export const peopleDemo = [ city: 'New Nataliechester', email: 'kelsey.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/kelsey-allen-27c8b08501', jobTitle: 'TEFL teacher', }, @@ -6455,7 +6455,7 @@ export const peopleDemo = [ city: 'West Stephen', email: 'alexander.mueller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/alexander-mueller-ce1f7db6fb', jobTitle: 'Chartered legal executive (England and Wales)', }, @@ -6465,7 +6465,7 @@ export const peopleDemo = [ city: 'North Bradleyhaven', email: 'cynthia.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/cynthia-davis-ab0220f93b', jobTitle: 'Designer, fashion/clothing', }, @@ -6475,7 +6475,7 @@ export const peopleDemo = [ city: 'Lake Sheila', email: 'brittany.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/brittany-smith-fb1075699d', jobTitle: 'Environmental manager', }, @@ -6485,7 +6485,7 @@ export const peopleDemo = [ city: 'Bowershaven', email: 'tyler.cook@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/tyler-cook-fe2564bd60', jobTitle: 'Hydrologist', }, @@ -6495,7 +6495,7 @@ export const peopleDemo = [ city: 'East Tiffany', email: 'heather.peck@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/heather-peck-09972b54b0', jobTitle: 'Designer, television/film set', }, @@ -6505,7 +6505,7 @@ export const peopleDemo = [ city: 'East Jennaview', email: 'justin.bender@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/justin-bender-679fb353e6', jobTitle: 'Local government officer', }, @@ -6515,7 +6515,7 @@ export const peopleDemo = [ city: 'South Vincent', email: 'sharon.phillips@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/sharon-phillips-c61cfc876b', jobTitle: 'IT sales professional', }, @@ -6525,7 +6525,7 @@ export const peopleDemo = [ city: 'Port Jamestown', email: 'samuel.bailey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/samuel-bailey-b423dc1293', jobTitle: 'Charity officer', }, @@ -6535,7 +6535,7 @@ export const peopleDemo = [ city: 'New Jeffrey', email: 'thomas.hull@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/thomas-hull-f814264a40', jobTitle: 'Professor Emeritus', }, @@ -6545,7 +6545,7 @@ export const peopleDemo = [ city: 'New Apriltown', email: 'shawn.collins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/shawn-collins-01f97c07d8', jobTitle: 'Chief Marketing Officer', }, @@ -6555,7 +6555,7 @@ export const peopleDemo = [ city: 'New Angelicaborough', email: 'matthew.salas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/matthew-salas-f83a70d28d', jobTitle: 'Scientist, product/process development', }, @@ -6565,7 +6565,7 @@ export const peopleDemo = [ city: 'Burkebury', email: 'sandra.branch@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/sandra-branch-e20208030c', jobTitle: 'Programmer, systems', }, @@ -6575,7 +6575,7 @@ export const peopleDemo = [ city: 'East Bryan', email: 'donald.burns@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/donald-burns-eba0c35180', jobTitle: 'Interior and spatial designer', }, @@ -6585,7 +6585,7 @@ export const peopleDemo = [ city: 'North Karen', email: 'robin.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/robin-allen-bb093171b6', jobTitle: 'Personal assistant', }, @@ -6595,7 +6595,7 @@ export const peopleDemo = [ city: 'East Wesleyview', email: 'andrew.carter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/andrew-carter-940e806c3e', jobTitle: 'Broadcast presenter', }, @@ -6605,7 +6605,7 @@ export const peopleDemo = [ city: 'West Keithfort', email: 'natalie.king@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/natalie-king-f64767c2da', jobTitle: 'Actor', }, @@ -6615,7 +6615,7 @@ export const peopleDemo = [ city: 'Lake Jeremyfurt', email: 'gregory.rosario@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/gregory-rosario-17dfa72dac', jobTitle: 'Adult guidance worker', }, @@ -6625,7 +6625,7 @@ export const peopleDemo = [ city: 'Whiteland', email: 'jeffrey.schultz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/jeffrey-schultz-396167e978', jobTitle: 'Occupational hygienist', }, @@ -6635,7 +6635,7 @@ export const peopleDemo = [ city: 'Teresamouth', email: 'michelle.cook@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/michelle-cook-38eb3ee806', jobTitle: 'Futures trader', }, @@ -6645,7 +6645,7 @@ export const peopleDemo = [ city: 'Melissaport', email: 'billy.hutchinson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/billy-hutchinson-ad8e6c722e', jobTitle: 'Television camera operator', }, @@ -6655,7 +6655,7 @@ export const peopleDemo = [ city: 'Lake Benjamin', email: 'kim.rhodes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/kim-rhodes-acaa899835', jobTitle: 'Engineer, maintenance (IT)', }, @@ -6665,7 +6665,7 @@ export const peopleDemo = [ city: 'Mendezchester', email: 'cristian.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/cristian-garcia-2ebbb7bd5e', jobTitle: 'Clinical cytogeneticist', }, @@ -6675,7 +6675,7 @@ export const peopleDemo = [ city: 'Diazchester', email: 'joseph.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/joseph-rodriguez-de666b5949', jobTitle: 'Colour technologist', }, @@ -6685,7 +6685,7 @@ export const peopleDemo = [ city: 'Port Michaelshire', email: 'dennis.blevins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/dennis-blevins-f1a0ae91b0', jobTitle: 'Engineer, petroleum', }, @@ -6695,7 +6695,7 @@ export const peopleDemo = [ city: 'Samanthaport', email: 'charles.bright@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/charles-bright-08b96bc983', jobTitle: 'Scientist, biomedical', }, @@ -6705,7 +6705,7 @@ export const peopleDemo = [ city: 'Youngmouth', email: 'pamela.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/pamela-moore-cf9a7d2df4', jobTitle: 'Clinical embryologist', }, @@ -6715,7 +6715,7 @@ export const peopleDemo = [ city: 'Garciafort', email: 'andrew.bowen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/andrew-bowen-a1a4379f39', jobTitle: 'Industrial/product designer', }, @@ -6725,7 +6725,7 @@ export const peopleDemo = [ city: 'Port Nicholasfurt', email: 'steven.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/steven-jones-0b032a0b9f', jobTitle: 'Radio producer', }, @@ -6735,7 +6735,7 @@ export const peopleDemo = [ city: 'Lake Scott', email: 'randy.garza@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/randy-garza-e9a6dd8b85', jobTitle: 'Diagnostic radiographer', }, @@ -6745,7 +6745,7 @@ export const peopleDemo = [ city: 'Charleschester', email: 'barbara.wallace@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/barbara-wallace-69672cb7e7', jobTitle: 'Radiation protection practitioner', }, @@ -6755,7 +6755,7 @@ export const peopleDemo = [ city: 'Hardinville', email: 'robert.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/robert-johnson-799ff9ca07', jobTitle: 'Producer, radio', }, @@ -6765,7 +6765,7 @@ export const peopleDemo = [ city: 'South Christina', email: 'daniel.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/daniel-perez-a5274e65fd', jobTitle: 'Product designer', }, @@ -6775,7 +6775,7 @@ export const peopleDemo = [ city: 'Garciaport', email: 'breanna.chapman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/breanna-chapman-6565f5d75b', jobTitle: 'Television camera operator', }, @@ -6785,7 +6785,7 @@ export const peopleDemo = [ city: 'South Kristafurt', email: 'ivan.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/ivan-garcia-5b5da68591', jobTitle: 'Police officer', }, @@ -6795,7 +6795,7 @@ export const peopleDemo = [ city: 'North Lisaburgh', email: 'michelle.thomas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/michelle-thomas-ffd3c66255', jobTitle: 'Ranger/warden', }, @@ -6805,7 +6805,7 @@ export const peopleDemo = [ city: 'New Kelly', email: 'stacey.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/stacey-taylor-bb60b7c3b6', jobTitle: 'Forensic scientist', }, @@ -6815,7 +6815,7 @@ export const peopleDemo = [ city: 'Danielberg', email: 'randall.cohen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/randall-cohen-1b5ce9e43e', jobTitle: 'Freight forwarder', }, @@ -6825,7 +6825,7 @@ export const peopleDemo = [ city: 'North Stevetown', email: 'dennis.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/dennis-johnson-a5cf409bc9', jobTitle: 'Petroleum engineer', }, @@ -6835,7 +6835,7 @@ export const peopleDemo = [ city: 'Lake Jennifer', email: 'scott.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/scott-rodriguez-8a8fffb3b1', jobTitle: 'Neurosurgeon', }, @@ -6845,7 +6845,7 @@ export const peopleDemo = [ city: 'Ortizberg', email: 'katrina.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/katrina-rodriguez-736608f682', jobTitle: 'Glass blower/designer', }, @@ -6855,7 +6855,7 @@ export const peopleDemo = [ city: 'Saraburgh', email: 'thomas.bradley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/thomas-bradley-06ba80922b', jobTitle: 'Automotive engineer', }, @@ -6865,7 +6865,7 @@ export const peopleDemo = [ city: 'Guerrerohaven', email: 'anna.hill@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/anna-hill-ccdca344ff', jobTitle: 'Therapist, sports', }, @@ -6875,7 +6875,7 @@ export const peopleDemo = [ city: 'Nealtown', email: 'karen.pratt@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/karen-pratt-7389b464f8', jobTitle: 'Editor, commissioning', }, @@ -6885,7 +6885,7 @@ export const peopleDemo = [ city: 'Millerport', email: 'casey.garza@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/casey-garza-e265fa80ac', jobTitle: 'Homeopath', }, @@ -6895,7 +6895,7 @@ export const peopleDemo = [ city: 'North Douglastown', email: 'mathew.duran@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/mathew-duran-667faa2205', jobTitle: 'Mechanical engineer', }, @@ -6905,7 +6905,7 @@ export const peopleDemo = [ city: 'Norrisfurt', email: 'michael.watson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/michael-watson-7e29289ceb', jobTitle: 'Scientist, research (physical sciences)', }, @@ -6915,7 +6915,7 @@ export const peopleDemo = [ city: 'Michaelville', email: 'martha.lang@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/martha-lang-37c24a1f79', jobTitle: 'Market researcher', }, @@ -6925,7 +6925,7 @@ export const peopleDemo = [ city: 'Bryanchester', email: 'latasha.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/latasha-perez-a903374657', jobTitle: 'Teacher, primary school', }, @@ -6935,7 +6935,7 @@ export const peopleDemo = [ city: 'New Jamesborough', email: 'james.gordon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/james-gordon-c6aee9e0e7', jobTitle: 'Illustrator', }, @@ -6945,7 +6945,7 @@ export const peopleDemo = [ city: 'Jenniferburgh', email: 'taylor.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/taylor-johnson-505c92efc6', jobTitle: 'Hotel manager', }, @@ -6955,7 +6955,7 @@ export const peopleDemo = [ city: 'Christineville', email: 'george.thompson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/george-thompson-0b7979e007', jobTitle: 'Engineer, mining', }, @@ -6965,7 +6965,7 @@ export const peopleDemo = [ city: 'Millerhaven', email: 'william.gomez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/william-gomez-bb381032a2', jobTitle: 'Medical physicist', }, @@ -6975,7 +6975,7 @@ export const peopleDemo = [ city: 'New Connorbury', email: 'patrick.beck@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/patrick-beck-a53d364316', jobTitle: 'Patent examiner', }, @@ -6985,7 +6985,7 @@ export const peopleDemo = [ city: 'Jeffreyborough', email: 'timothy.parker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/timothy-parker-fa652d4471', jobTitle: 'Therapist, drama', }, @@ -6995,7 +6995,7 @@ export const peopleDemo = [ city: 'Port Emilyside', email: 'nancy.mullen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/nancy-mullen-db4e1644e0', jobTitle: 'Chiropodist', }, @@ -7005,7 +7005,7 @@ export const peopleDemo = [ city: 'Rodrigueztown', email: 'amy.weaver@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/amy-weaver-f6b74416f5', jobTitle: 'Cabin crew', }, @@ -7015,7 +7015,7 @@ export const peopleDemo = [ city: 'North Benjamin', email: 'matthew.crawford@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/matthew-crawford-65e6eb72da', jobTitle: 'Technical author', }, @@ -7025,7 +7025,7 @@ export const peopleDemo = [ city: 'Boydton', email: 'daniel.graham@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/daniel-graham-a67e125f10', jobTitle: 'Technical sales engineer', }, @@ -7035,7 +7035,7 @@ export const peopleDemo = [ city: 'Emilymouth', email: 'teresa.lang@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/teresa-lang-584a181fe2', jobTitle: 'Solicitor, Scotland', }, @@ -7045,7 +7045,7 @@ export const peopleDemo = [ city: 'Lake Andrew', email: 'anthony.brooks@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/anthony-brooks-b32a3564da', jobTitle: 'Aeronautical engineer', }, @@ -7055,7 +7055,7 @@ export const peopleDemo = [ city: 'Keithberg', email: 'thomas.price@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/thomas-price-d5c64ebc73', jobTitle: 'Counselling psychologist', }, @@ -7065,7 +7065,7 @@ export const peopleDemo = [ city: 'Barberhaven', email: 'william.king@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/william-king-f1790dcb3a', jobTitle: 'Medical laboratory scientific officer', }, @@ -7075,7 +7075,7 @@ export const peopleDemo = [ city: 'Rodgerston', email: 'joseph.ramos@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/joseph-ramos-c3050c9ec5', jobTitle: 'English as a foreign language teacher', }, @@ -7085,7 +7085,7 @@ export const peopleDemo = [ city: 'North Ryan', email: 'michael.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/michael-johnson-442bf8f6cc', jobTitle: 'Phytotherapist', }, @@ -7095,7 +7095,7 @@ export const peopleDemo = [ city: 'North Shannon', email: 'lisa.farmer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/lisa-farmer-7e00712bfa', jobTitle: 'Trade mark attorney', }, @@ -7105,7 +7105,7 @@ export const peopleDemo = [ city: 'Annafurt', email: 'beth.tucker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/beth-tucker-366a3e9978', jobTitle: 'Site engineer', }, @@ -7115,7 +7115,7 @@ export const peopleDemo = [ city: 'Gillview', email: 'gerald.olsen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/gerald-olsen-ec232b4b08', jobTitle: 'Seismic interpreter', }, @@ -7125,7 +7125,7 @@ export const peopleDemo = [ city: 'Michellemouth', email: 'donald.turner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/donald-turner-b84598e436', jobTitle: 'Music tutor', }, @@ -7135,7 +7135,7 @@ export const peopleDemo = [ city: 'Millerfurt', email: 'mary.robinson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/mary-robinson-173a75d1ef', jobTitle: 'Public house manager', }, @@ -7145,7 +7145,7 @@ export const peopleDemo = [ city: 'West Melinda', email: 'jason.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/jason-johnson-a909a78a1c', jobTitle: 'Horticulturist, commercial', }, @@ -7155,7 +7155,7 @@ export const peopleDemo = [ city: 'Morrisonfurt', email: 'lisa.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/lisa-jenkins-c78e76770a', jobTitle: 'Counsellor', }, @@ -7165,7 +7165,7 @@ export const peopleDemo = [ city: 'West Heatherhaven', email: 'kevin.cowan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/kevin-cowan-3177fc23bf', jobTitle: 'Recruitment consultant', }, @@ -7175,7 +7175,7 @@ export const peopleDemo = [ city: 'East Michelle', email: 'tina.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/tina-williams-2c3d8afa12', jobTitle: 'Scientist, research (maths)', }, @@ -7185,7 +7185,7 @@ export const peopleDemo = [ city: 'Josephborough', email: 'dustin.macdonald@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/dustin-macdonald-afa9526426', jobTitle: 'Public relations account executive', }, @@ -7195,7 +7195,7 @@ export const peopleDemo = [ city: 'Port Jennifer', email: 'michael.gonzales@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/michael-gonzales-3eb48a8cc2', jobTitle: "Nurse, children's", }, @@ -7205,7 +7205,7 @@ export const peopleDemo = [ city: 'Margaretmouth', email: 'allen.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/allen-miller-5378b7d05d', jobTitle: 'Therapist, speech and language', }, @@ -7215,7 +7215,7 @@ export const peopleDemo = [ city: 'South Alexandra', email: 'meghan.tapia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/meghan-tapia-8d4336f0bc', jobTitle: 'Primary school teacher', }, @@ -7225,7 +7225,7 @@ export const peopleDemo = [ city: 'Lewisfurt', email: 'garrett.kim@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/garrett-kim-f7e6805c01', jobTitle: 'Education administrator', }, @@ -7235,7 +7235,7 @@ export const peopleDemo = [ city: 'New Colleenchester', email: 'heather.chase@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/heather-chase-0a2eb1532b', jobTitle: 'Engineer, civil (consulting)', }, @@ -7245,7 +7245,7 @@ export const peopleDemo = [ city: 'Bradleystad', email: 'kevin.cruz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/kevin-cruz-d3f02830aa', jobTitle: 'Naval architect', }, @@ -7255,7 +7255,7 @@ export const peopleDemo = [ city: 'Jessicaview', email: 'desiree.adkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/desiree-adkins-c85deab253', jobTitle: 'Corporate treasurer', }, @@ -7265,7 +7265,7 @@ export const peopleDemo = [ city: 'Kristinamouth', email: 'teresa.rhodes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/teresa-rhodes-f990f416da', jobTitle: 'Primary school teacher', }, @@ -7275,7 +7275,7 @@ export const peopleDemo = [ city: 'West Jessicaland', email: 'catherine.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/catherine-wilson-7add346581', jobTitle: 'Optician, dispensing', }, @@ -7285,7 +7285,7 @@ export const peopleDemo = [ city: 'Watsontown', email: 'marvin.nelson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/marvin-nelson-fde56b8b5d', jobTitle: 'Operational investment banker', }, @@ -7295,7 +7295,7 @@ export const peopleDemo = [ city: 'Jacksonport', email: 'linda.hull@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/linda-hull-4d41c26e8b', jobTitle: 'Clinical cytogeneticist', }, @@ -7305,7 +7305,7 @@ export const peopleDemo = [ city: 'Beckyfort', email: 'dawn.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/dawn-martin-fe75b2575e', jobTitle: 'Housing manager/officer', }, @@ -7315,7 +7315,7 @@ export const peopleDemo = [ city: 'East Marieshire', email: 'travis.leon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/travis-leon-5c56017c27', jobTitle: 'Further education lecturer', }, @@ -7325,7 +7325,7 @@ export const peopleDemo = [ city: 'Edwardsfurt', email: 'jeffrey.anderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/jeffrey-anderson-1b6caa26b8', jobTitle: 'Merchant navy officer', }, @@ -7335,7 +7335,7 @@ export const peopleDemo = [ city: 'Paulmouth', email: 'jacqueline.gomez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/jacqueline-gomez-0bc243917e', jobTitle: 'Producer, radio', }, @@ -7345,7 +7345,7 @@ export const peopleDemo = [ city: 'Karenburgh', email: 'laura.salazar@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/laura-salazar-b649ef3e65', jobTitle: 'Investment analyst', }, @@ -7355,7 +7355,7 @@ export const peopleDemo = [ city: 'Adamsberg', email: 'jacob.berry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/jacob-berry-b613f63da8', jobTitle: 'Dispensing optician', }, @@ -7365,7 +7365,7 @@ export const peopleDemo = [ city: 'New Michaelton', email: 'justin.cruz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/justin-cruz-e21f40fce4', jobTitle: 'Sports development officer', }, @@ -7375,7 +7375,7 @@ export const peopleDemo = [ city: 'West Jeffrey', email: 'derek.avery@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/derek-avery-2e68141d8b', jobTitle: 'Tax adviser', }, @@ -7385,7 +7385,7 @@ export const peopleDemo = [ city: 'North Kelsey', email: 'julie.richardson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/julie-richardson-c004cc7600', jobTitle: 'Visual merchandiser', }, @@ -7395,7 +7395,7 @@ export const peopleDemo = [ city: 'Edwardchester', email: 'linda.perry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/linda-perry-81b2a5fe77', jobTitle: 'Records manager', }, @@ -7405,7 +7405,7 @@ export const peopleDemo = [ city: 'Davidland', email: 'shannon.johnston@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/shannon-johnston-730f636101', jobTitle: 'Local government officer', }, @@ -7415,7 +7415,7 @@ export const peopleDemo = [ city: 'North Emmamouth', email: 'teresa.peters@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/teresa-peters-79db7ef10b', jobTitle: 'Ecologist', }, @@ -7425,7 +7425,7 @@ export const peopleDemo = [ city: 'Douglasmouth', email: 'vanessa.woods@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/vanessa-woods-6e5bdf7c9a', jobTitle: 'QuickActions analyst', }, @@ -7435,7 +7435,7 @@ export const peopleDemo = [ city: 'Courtneybury', email: 'ashley.ortiz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/ashley-ortiz-5cc26fa72b', jobTitle: 'Surveyor, mining', }, @@ -7445,7 +7445,7 @@ export const peopleDemo = [ city: 'Ryanland', email: 'eric.bailey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/eric-bailey-c789b6c993', jobTitle: 'Financial controller', }, @@ -7455,7 +7455,7 @@ export const peopleDemo = [ city: 'West Linda', email: 'rebecca.palmer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/rebecca-palmer-ef38ef5cea', jobTitle: 'Site engineer', }, @@ -7465,7 +7465,7 @@ export const peopleDemo = [ city: 'Nicholsborough', email: 'lee.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/lee-jones-0adef06cd7', jobTitle: 'Radiographer, therapeutic', }, @@ -7475,7 +7475,7 @@ export const peopleDemo = [ city: 'East Kathrynchester', email: 'samuel.king@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/samuel-king-57c6f39f1c', jobTitle: 'Illustrator', }, @@ -7485,7 +7485,7 @@ export const peopleDemo = [ city: 'Mcphersonport', email: 'timothy.moreno@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/timothy-moreno-eb7ae88f2f', jobTitle: 'Physiotherapist', }, @@ -7495,7 +7495,7 @@ export const peopleDemo = [ city: 'New Makayla', email: 'darlene.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/darlene-jones-e27174b679', jobTitle: 'Health physicist', }, @@ -7505,7 +7505,7 @@ export const peopleDemo = [ city: 'Port Justin', email: 'gregory.liu@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/gregory-liu-aac65508df', jobTitle: 'Psychiatric nurse', }, @@ -7515,7 +7515,7 @@ export const peopleDemo = [ city: 'Donaldbury', email: 'cheryl.chambers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/cheryl-chambers-0f636392e0', jobTitle: 'Education officer, community', }, @@ -7525,7 +7525,7 @@ export const peopleDemo = [ city: 'Shaneton', email: 'mark.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/mark-gonzalez-4546b149d7', jobTitle: 'Adult nurse', }, @@ -7535,7 +7535,7 @@ export const peopleDemo = [ city: 'Roblesport', email: 'douglas.andrews@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/douglas-andrews-7a6a535f81', jobTitle: 'Accountant, chartered management', }, @@ -7545,7 +7545,7 @@ export const peopleDemo = [ city: 'East Lisaburgh', email: 'stephanie.porter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/stephanie-porter-74bdb68326', jobTitle: 'Computer games developer', }, @@ -7555,7 +7555,7 @@ export const peopleDemo = [ city: 'Alexischester', email: 'meghan.campbell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/meghan-campbell-5098a6f7a9', jobTitle: 'Theatre manager', }, @@ -7565,7 +7565,7 @@ export const peopleDemo = [ city: 'North William', email: 'caitlin.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/caitlin-martin-93755bb8ba', jobTitle: 'Scientist, research (maths)', }, @@ -7575,7 +7575,7 @@ export const peopleDemo = [ city: 'North Sean', email: 'kimberly.terry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/kimberly-terry-5f017ebb4b', jobTitle: 'Surveyor, building', }, @@ -7585,7 +7585,7 @@ export const peopleDemo = [ city: 'Lake Amandaborough', email: 'levi.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/levi-smith-4d6387a547', jobTitle: 'Mental health nurse', }, @@ -7595,7 +7595,7 @@ export const peopleDemo = [ city: 'Lake Paigeborough', email: 'tracy.alvarez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/tracy-alvarez-633fc7a383', jobTitle: 'Environmental health practitioner', }, @@ -7605,7 +7605,7 @@ export const peopleDemo = [ city: 'Owensstad', email: 'david.gonzales@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/david-gonzales-6df036cad1', jobTitle: 'Legal secretary', }, @@ -7615,7 +7615,7 @@ export const peopleDemo = [ city: 'East Thomasbury', email: 'lisa.tran@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/lisa-tran-e0115b5653', jobTitle: 'Therapist, speech and language', }, @@ -7625,7 +7625,7 @@ export const peopleDemo = [ city: 'Williamhaven', email: 'kristin.pearson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/kristin-pearson-e0ebe90624', jobTitle: 'Editor, film/video', }, @@ -7635,7 +7635,7 @@ export const peopleDemo = [ city: 'North Lindsey', email: 'bruce.wood@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/bruce-wood-e61cf3a298', jobTitle: 'Charity fundraiser', }, @@ -7645,7 +7645,7 @@ export const peopleDemo = [ city: 'Kristinshire', email: 'stephanie.stout@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/stephanie-stout-da19425869', jobTitle: 'Conservator, furniture', }, @@ -7655,7 +7655,7 @@ export const peopleDemo = [ city: 'Daisyburgh', email: 'denise.sandoval@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/denise-sandoval-07f4d63a26', jobTitle: 'Immunologist', }, @@ -7665,7 +7665,7 @@ export const peopleDemo = [ city: 'North Ryanmouth', email: 'christopher.clarke@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/christopher-clarke-44a16b8bf4', jobTitle: 'Stage manager', }, @@ -7675,7 +7675,7 @@ export const peopleDemo = [ city: 'Barnesburgh', email: 'kimberly.jefferson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/kimberly-jefferson-ff1550e548', jobTitle: 'Advertising account executive', }, @@ -7685,7 +7685,7 @@ export const peopleDemo = [ city: 'East Austin', email: 'jeffrey.hunt@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/jeffrey-hunt-faa3d941ee', jobTitle: 'Customer service manager', }, @@ -7695,7 +7695,7 @@ export const peopleDemo = [ city: 'South Gregorytown', email: 'nichole.lowery@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/nichole-lowery-fb08af1201', jobTitle: 'Planning and development surveyor', }, @@ -7705,7 +7705,7 @@ export const peopleDemo = [ city: 'Lauraburgh', email: 'daniel.wiley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/daniel-wiley-9ed6e2002f', jobTitle: 'Surveyor, mining', }, @@ -7715,7 +7715,7 @@ export const peopleDemo = [ city: 'Mitchellbury', email: 'elizabeth.watson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/elizabeth-watson-037218b4e1', jobTitle: 'Journalist, broadcasting', }, @@ -7725,7 +7725,7 @@ export const peopleDemo = [ city: 'South Natalieport', email: 'sandra.bailey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/sandra-bailey-74db8eab37', jobTitle: 'Minerals surveyor', }, @@ -7735,7 +7735,7 @@ export const peopleDemo = [ city: 'New Kristin', email: 'andrew.henson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/andrew-henson-23fb7e5d05', jobTitle: 'Biomedical scientist', }, @@ -7745,7 +7745,7 @@ export const peopleDemo = [ city: 'Wallston', email: 'samantha.alexander@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/samantha-alexander-4a24cc632b', jobTitle: 'Production assistant, television', }, @@ -7755,7 +7755,7 @@ export const peopleDemo = [ city: 'Lake Rachel', email: 'jeanette.nichols@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/jeanette-nichols-a24214c373', jobTitle: 'Engineer, biomedical', }, @@ -7765,7 +7765,7 @@ export const peopleDemo = [ city: 'Susanbury', email: 'kevin.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/kevin-williams-0165f2638b', jobTitle: 'Broadcast journalist', }, @@ -7775,7 +7775,7 @@ export const peopleDemo = [ city: 'Jacquelineshire', email: 'brenda.harper@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/brenda-harper-30b61b982b', jobTitle: 'Public relations account executive', }, @@ -7785,7 +7785,7 @@ export const peopleDemo = [ city: 'Brownbury', email: 'shawn.jenkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/shawn-jenkins-c839f2afce', jobTitle: 'Optometrist', }, @@ -7795,7 +7795,7 @@ export const peopleDemo = [ city: 'Brendaport', email: 'michelle.bush@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/michelle-bush-2e2e3d23ee', jobTitle: 'Teacher, English as a foreign language', }, @@ -7805,7 +7805,7 @@ export const peopleDemo = [ city: 'East David', email: 'melanie.gilbert@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/melanie-gilbert-01fe35dd5f', jobTitle: 'Trading standards officer', }, @@ -7815,7 +7815,7 @@ export const peopleDemo = [ city: 'Gomezville', email: 'brandon.sanders@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/brandon-sanders-4661fbd2df', jobTitle: 'Radiation protection practitioner', }, @@ -7825,7 +7825,7 @@ export const peopleDemo = [ city: 'Emilyside', email: 'samantha.hicks@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/samantha-hicks-d7b99728fd', jobTitle: 'Firefighter', }, @@ -7835,7 +7835,7 @@ export const peopleDemo = [ city: 'Adamchester', email: 'joann.booth@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/joann-booth-c081ce2c43', jobTitle: 'Geoscientist', }, @@ -7845,7 +7845,7 @@ export const peopleDemo = [ city: 'Juliehaven', email: 'robert.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/robert-hernandez-5e65b16f59', jobTitle: 'Nurse, learning disability', }, @@ -7855,7 +7855,7 @@ export const peopleDemo = [ city: 'West Nicoleshire', email: 'jeremy.stewart@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/jeremy-stewart-27f2b87ae6', jobTitle: 'Chartered public finance accountant', }, @@ -7865,7 +7865,7 @@ export const peopleDemo = [ city: 'West Tracy', email: 'lisa.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/lisa-brown-01aa1694a4', jobTitle: 'Patent attorney', }, @@ -7875,7 +7875,7 @@ export const peopleDemo = [ city: 'East Aaron', email: 'kristine.benson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/kristine-benson-ee3307c3e8', jobTitle: 'Financial risk analyst', }, @@ -7885,7 +7885,7 @@ export const peopleDemo = [ city: 'Steeleport', email: 'stephanie.carter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/stephanie-carter-aa789505dc', jobTitle: 'Patent attorney', }, @@ -7895,7 +7895,7 @@ export const peopleDemo = [ city: 'Lake Brianmouth', email: 'benjamin.castro@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/benjamin-castro-5609ebd89b', jobTitle: 'Surveyor, mining', }, @@ -7905,7 +7905,7 @@ export const peopleDemo = [ city: 'Stevenshire', email: 'ryan.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/ryan-davis-04fe8f1d38', jobTitle: 'Therapeutic radiographer', }, @@ -7915,7 +7915,7 @@ export const peopleDemo = [ city: 'Port Robert', email: 'david.rhodes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/david-rhodes-b30501dc23', jobTitle: 'Leisure centre manager', }, @@ -7925,7 +7925,7 @@ export const peopleDemo = [ city: 'New Toni', email: 'elizabeth.evans@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/elizabeth-evans-1c62f6e072', jobTitle: 'Emergency planning/management officer', }, @@ -7935,7 +7935,7 @@ export const peopleDemo = [ city: 'North Craigside', email: 'kenneth.solis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/kenneth-solis-abc002b3d7', jobTitle: 'Operational researcher', }, @@ -7945,7 +7945,7 @@ export const peopleDemo = [ city: 'West Jonathanside', email: 'barbara.hudson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/barbara-hudson-d52a7f47e3', jobTitle: 'Engineer, manufacturing systems', }, @@ -7955,7 +7955,7 @@ export const peopleDemo = [ city: 'New Raymond', email: 'kelly.hooper@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/kelly-hooper-9a2d9e03bd', jobTitle: 'Automotive engineer', }, @@ -7965,7 +7965,7 @@ export const peopleDemo = [ city: 'Archerhaven', email: 'shannon.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/shannon-brown-a9e4eebc94', jobTitle: 'Scientist, forensic', }, @@ -7975,7 +7975,7 @@ export const peopleDemo = [ city: 'North Tinamouth', email: 'lucas.price@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/lucas-price-8220b81a6d', jobTitle: 'Health promotion specialist', }, @@ -7985,7 +7985,7 @@ export const peopleDemo = [ city: 'Campbellburgh', email: 'theodore.booth@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/theodore-booth-28b999e5b3', jobTitle: 'Camera operator', }, @@ -7995,7 +7995,7 @@ export const peopleDemo = [ city: 'Lake Mariahmouth', email: 'christopher.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/christopher-johnson-9e6100ff7b', jobTitle: 'Art therapist', }, @@ -8005,7 +8005,7 @@ export const peopleDemo = [ city: 'Lake Williamburgh', email: 'sara.higgins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/sara-higgins-3f7fd986f9', jobTitle: 'IT sales professional', }, @@ -8015,7 +8015,7 @@ export const peopleDemo = [ city: 'Kathrynton', email: 'kelly.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/kelly-brown-759dbe09e0', jobTitle: 'Psychologist, clinical', }, @@ -8025,7 +8025,7 @@ export const peopleDemo = [ city: 'Bethanymouth', email: 'andrea.weaver@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/andrea-weaver-6d792fc29a', jobTitle: 'Ceramics designer', }, @@ -8035,7 +8035,7 @@ export const peopleDemo = [ city: 'Lake Veronica', email: 'david.ford@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/david-ford-ed83f54167', jobTitle: 'Designer, furniture', }, @@ -8045,7 +8045,7 @@ export const peopleDemo = [ city: 'Darrellshire', email: 'elizabeth.scott@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/elizabeth-scott-56456b1569', jobTitle: 'Government social research officer', }, @@ -8055,7 +8055,7 @@ export const peopleDemo = [ city: 'Shawnside', email: 'haley.rodriguez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/haley-rodriguez-5998488ad7', jobTitle: 'Librarian, public', }, @@ -8065,7 +8065,7 @@ export const peopleDemo = [ city: 'New Alexander', email: 'joshua.harris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/joshua-harris-c48d311bee', jobTitle: 'Energy engineer', }, @@ -8075,7 +8075,7 @@ export const peopleDemo = [ city: 'Brownshire', email: 'ellen.mcdaniel@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/ellen-mcdaniel-4308564cbb', jobTitle: 'Rural practice surveyor', }, @@ -8085,7 +8085,7 @@ export const peopleDemo = [ city: 'Joshuastad', email: 'anthony.macias@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/anthony-macias-cd01ea615c', jobTitle: 'Estate manager/land agent', }, @@ -8095,7 +8095,7 @@ export const peopleDemo = [ city: 'South Martinstad', email: 'samantha.bell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/samantha-bell-3246e99ce4', jobTitle: 'Armed forces technical officer', }, @@ -8105,7 +8105,7 @@ export const peopleDemo = [ city: 'Wileyland', email: 'roger.king@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/roger-king-91a87f58dd', jobTitle: 'Programme researcher, broadcasting/film/video', }, @@ -8115,7 +8115,7 @@ export const peopleDemo = [ city: 'East Richard', email: 'logan.kim@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/logan-kim-24c952a76b', jobTitle: 'Ranger/warden', }, @@ -8125,7 +8125,7 @@ export const peopleDemo = [ city: 'Michelemouth', email: 'nicole.bass@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/nicole-bass-3323b936fa', jobTitle: 'Commercial/residential surveyor', }, @@ -8135,7 +8135,7 @@ export const peopleDemo = [ city: 'East Allison', email: 'tony.dean@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/tony-dean-6a37678e19', jobTitle: 'Tax inspector', }, @@ -8145,7 +8145,7 @@ export const peopleDemo = [ city: 'Lindseyton', email: 'mercedes.green@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/mercedes-green-4b158a8688', jobTitle: 'Clinical cytogeneticist', }, @@ -8155,7 +8155,7 @@ export const peopleDemo = [ city: 'Mayerfurt', email: 'stephen.owens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/stephen-owens-8863b05296', jobTitle: 'Pathologist', }, @@ -8165,7 +8165,7 @@ export const peopleDemo = [ city: 'Lunaport', email: 'nathan.williamson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/nathan-williamson-ffdb40e0ae', jobTitle: 'Publishing copy', }, @@ -8175,7 +8175,7 @@ export const peopleDemo = [ city: 'Port Jackieshire', email: 'anthony.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/anthony-davis-c8f6c37766', jobTitle: 'Ship broker', }, @@ -8185,7 +8185,7 @@ export const peopleDemo = [ city: 'New Angelaburgh', email: 'kathleen.stewart@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/kathleen-stewart-3c65492da0', jobTitle: 'Economist', }, @@ -8195,7 +8195,7 @@ export const peopleDemo = [ city: 'Port Jamesfort', email: 'victoria.ruiz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/victoria-ruiz-a5f8ac2c75', jobTitle: 'IT trainer', }, @@ -8205,7 +8205,7 @@ export const peopleDemo = [ city: 'Johnsonmouth', email: 'danielle.ibarra@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/danielle-ibarra-d11e8407ab', jobTitle: 'Nurse, adult', }, @@ -8215,7 +8215,7 @@ export const peopleDemo = [ city: 'Elizabethburgh', email: 'meghan.delgado@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/meghan-delgado-7954afab5b', jobTitle: 'Psychiatric nurse', }, @@ -8225,7 +8225,7 @@ export const peopleDemo = [ city: 'West Andrewfort', email: 'lauren.skinner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/lauren-skinner-447a3bcd28', jobTitle: 'Psychologist, prison and probation services', }, @@ -8235,7 +8235,7 @@ export const peopleDemo = [ city: 'North Angela', email: 'jesse.underwood@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/jesse-underwood-072dbeda4c', jobTitle: 'Pharmacist, hospital', }, @@ -8245,7 +8245,7 @@ export const peopleDemo = [ city: 'Port Jennifer', email: 'antonio.gentry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/antonio-gentry-0f29dc0871', jobTitle: 'Magazine features editor', }, @@ -8255,7 +8255,7 @@ export const peopleDemo = [ city: 'South Saraport', email: 'gabriela.murphy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/gabriela-murphy-e59b96e98f', jobTitle: 'Exercise physiologist', }, @@ -8265,7 +8265,7 @@ export const peopleDemo = [ city: 'Mooreville', email: 'kyle.kramer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/kyle-kramer-3412f7e41b', jobTitle: 'Exhibitions officer, museum/gallery', }, @@ -8275,7 +8275,7 @@ export const peopleDemo = [ city: 'Brownmouth', email: 'daniel.burton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/daniel-burton-c79414d37b', jobTitle: 'Commercial horticulturist', }, @@ -8285,7 +8285,7 @@ export const peopleDemo = [ city: 'Lake Laurahaven', email: 'mark.stevens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/mark-stevens-8d430f5e85', jobTitle: 'Data processing manager', }, @@ -8295,7 +8295,7 @@ export const peopleDemo = [ city: 'Wuhaven', email: 'kevin.lawson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/kevin-lawson-3923eb16c8', jobTitle: 'Radiographer, therapeutic', }, @@ -8305,7 +8305,7 @@ export const peopleDemo = [ city: 'New Cathymouth', email: 'christopher.larson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/christopher-larson-c3acf6c87e', jobTitle: 'Research scientist (life sciences)', }, @@ -8315,7 +8315,7 @@ export const peopleDemo = [ city: 'Oliviaside', email: 'james.ward@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/james-ward-bd728eec7b', jobTitle: 'Chief Marketing Officer', }, @@ -8325,7 +8325,7 @@ export const peopleDemo = [ city: 'Victoriamouth', email: 'thomas.ramirez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/thomas-ramirez-2aeaabdca6', jobTitle: 'Claims inspector/assessor', }, @@ -8335,7 +8335,7 @@ export const peopleDemo = [ city: 'Linside', email: 'makayla.schmitt@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/makayla-schmitt-5c93328d2c', jobTitle: 'Web designer', }, @@ -8345,7 +8345,7 @@ export const peopleDemo = [ city: 'Kristyville', email: 'andrew.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/andrew-miller-f2fe0c545b', jobTitle: 'Environmental education officer', }, @@ -8355,7 +8355,7 @@ export const peopleDemo = [ city: 'East Preston', email: 'matthew.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/matthew-perez-e0f9b3e4cb', jobTitle: 'Ranger/warden', }, @@ -8365,7 +8365,7 @@ export const peopleDemo = [ city: 'Jenniferhaven', email: 'molly.peterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/molly-peterson-9556e9927d', jobTitle: 'Environmental consultant', }, @@ -8375,7 +8375,7 @@ export const peopleDemo = [ city: 'Baileyfort', email: 'eric.kennedy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/eric-kennedy-f791a22768', jobTitle: 'Garment/textile technologist', }, @@ -8385,7 +8385,7 @@ export const peopleDemo = [ city: 'South Sharon', email: 'daniel.nguyen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/daniel-nguyen-9ecda3ed3b', jobTitle: 'Analytical chemist', }, @@ -8395,7 +8395,7 @@ export const peopleDemo = [ city: 'West Melanie', email: 'edward.washington@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/edward-washington-611f90b992', jobTitle: 'Field trials officer', }, @@ -8405,7 +8405,7 @@ export const peopleDemo = [ city: 'Calebville', email: 'stephanie.phillips@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/stephanie-phillips-7959a94ad5', jobTitle: 'Ecologist', }, @@ -8415,7 +8415,7 @@ export const peopleDemo = [ city: 'New Heatherfort', email: 'francisco.leach@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/francisco-leach-ae0211a300', jobTitle: 'Photographer', }, @@ -8425,7 +8425,7 @@ export const peopleDemo = [ city: 'Stokesstad', email: 'lisa.gutierrez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/lisa-gutierrez-372eee2535', jobTitle: 'Community development worker', }, @@ -8435,7 +8435,7 @@ export const peopleDemo = [ city: 'Gregoryville', email: 'robert.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/robert-martinez-a6ae2c5b1f', jobTitle: 'Secretary/administrator', }, @@ -8445,7 +8445,7 @@ export const peopleDemo = [ city: 'Smithtown', email: 'courtney.kelley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/courtney-kelley-950cfddf8b', jobTitle: 'Environmental education officer', }, @@ -8455,7 +8455,7 @@ export const peopleDemo = [ city: 'New Lori', email: 'samuel.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/samuel-davis-dd4df4491c', jobTitle: 'Engineer, technical sales', }, @@ -8465,7 +8465,7 @@ export const peopleDemo = [ city: 'Davidport', email: 'paul.kim@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/paul-kim-01704924f8', jobTitle: 'Optometrist', }, @@ -8475,7 +8475,7 @@ export const peopleDemo = [ city: 'Johnsonbury', email: 'samantha.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/samantha-jones-2abb1198e0', jobTitle: 'Medical secretary', }, @@ -8485,7 +8485,7 @@ export const peopleDemo = [ city: 'Brianshire', email: 'daniel.buchanan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/daniel-buchanan-962af5531b', jobTitle: 'Surveyor, land/geomatics', }, @@ -8495,7 +8495,7 @@ export const peopleDemo = [ city: 'Lake Emily', email: 'sherry.oliver@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/sherry-oliver-f3bbba4a94', jobTitle: 'Buyer, industrial', }, @@ -8505,7 +8505,7 @@ export const peopleDemo = [ city: 'Brittanyport', email: 'richard.burton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/richard-burton-eba4d16199', jobTitle: 'Trading standards officer', }, @@ -8515,7 +8515,7 @@ export const peopleDemo = [ city: 'South Matthew', email: 'larry.floyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/larry-floyd-a72834d039', jobTitle: 'Herpetologist', }, @@ -8525,7 +8525,7 @@ export const peopleDemo = [ city: 'North Briana', email: 'abigail.garrett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/abigail-garrett-fc4de32453', jobTitle: 'Training and development officer', }, @@ -8535,7 +8535,7 @@ export const peopleDemo = [ city: 'Hardyton', email: 'craig.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/craig-miller-e5dbce647e', jobTitle: 'Architect', }, @@ -8545,7 +8545,7 @@ export const peopleDemo = [ city: 'Donnaton', email: 'christina.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/christina-garcia-7ba1c75253', jobTitle: 'Radio producer', }, @@ -8555,7 +8555,7 @@ export const peopleDemo = [ city: 'Chadmouth', email: 'lynn.gallagher@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/lynn-gallagher-03ab6cdaae', jobTitle: 'Management consultant', }, @@ -8565,7 +8565,7 @@ export const peopleDemo = [ city: 'Tiffanystad', email: 'veronica.oliver@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/veronica-oliver-e46c83d82a', jobTitle: 'Theatre stage manager', }, @@ -8575,7 +8575,7 @@ export const peopleDemo = [ city: 'Amandamouth', email: 'julie.stevenson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/julie-stevenson-482959b900', jobTitle: 'Pharmacist, community', }, @@ -8585,7 +8585,7 @@ export const peopleDemo = [ city: 'New Julie', email: 'kathleen.gardner@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/kathleen-gardner-b144fa40bf', jobTitle: 'Research scientist (life sciences)', }, @@ -8595,7 +8595,7 @@ export const peopleDemo = [ city: 'West Brittany', email: 'james.ward@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/james-ward-9590de76ac', jobTitle: 'Psychologist, clinical', }, @@ -8605,7 +8605,7 @@ export const peopleDemo = [ city: 'Mccannchester', email: 'brandon.baker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/brandon-baker-79f5bc09a5', jobTitle: 'Geologist, wellsite', }, @@ -8615,7 +8615,7 @@ export const peopleDemo = [ city: 'Williamchester', email: 'cheyenne.stevens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/cheyenne-stevens-9102d355f5', jobTitle: 'Freight forwarder', }, @@ -8625,7 +8625,7 @@ export const peopleDemo = [ city: 'New Traviston', email: 'nicholas.chaney@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/nicholas-chaney-3bd29a6f7f', jobTitle: 'Civil engineer, contracting', }, @@ -8635,7 +8635,7 @@ export const peopleDemo = [ city: 'North Courtney', email: 'robert.allen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/robert-allen-d4a0e6e38b', jobTitle: 'Media buyer', }, @@ -8645,7 +8645,7 @@ export const peopleDemo = [ city: 'Mcknightberg', email: 'steven.walters@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/steven-walters-cecce0a460', jobTitle: 'Sports administrator', }, @@ -8655,7 +8655,7 @@ export const peopleDemo = [ city: 'West Jasonville', email: 'alexandra.rivera@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/alexandra-rivera-82738fef4b', jobTitle: 'Scientist, research (maths)', }, @@ -8665,7 +8665,7 @@ export const peopleDemo = [ city: 'West Samuelmouth', email: 'robert.doyle@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/robert-doyle-5012cb4f96', jobTitle: 'Early years teacher', }, @@ -8675,7 +8675,7 @@ export const peopleDemo = [ city: 'Jeremiahside', email: 'melinda.graves@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/melinda-graves-4b5dc7a605', jobTitle: 'Scientific laboratory technician', }, @@ -8685,7 +8685,7 @@ export const peopleDemo = [ city: 'North Christopher', email: 'phillip.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/phillip-johnson-1dbd354784', jobTitle: 'Librarian, public', }, @@ -8695,7 +8695,7 @@ export const peopleDemo = [ city: 'Averyfurt', email: 'kristin.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/kristin-garcia-667a8d3bf0', jobTitle: 'Maintenance engineer', }, @@ -8705,7 +8705,7 @@ export const peopleDemo = [ city: 'Lake Jeanside', email: 'randy.white@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/randy-white-158e05e8b0', jobTitle: 'Financial risk analyst', }, @@ -8715,7 +8715,7 @@ export const peopleDemo = [ city: 'Caitlinmouth', email: 'david.woods@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/david-woods-46b2d4b34e', jobTitle: 'Emergency planning/management officer', }, @@ -8725,7 +8725,7 @@ export const peopleDemo = [ city: 'Rebeccafurt', email: 'taylor.humphrey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/taylor-humphrey-7143e1cb93', jobTitle: 'Catering manager', }, @@ -8735,7 +8735,7 @@ export const peopleDemo = [ city: 'Stevenview', email: 'emily.evans@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/emily-evans-df56e8e3d2', jobTitle: 'Engineer, materials', }, @@ -8745,7 +8745,7 @@ export const peopleDemo = [ city: 'West Rachel', email: 'mike.weber@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/mike-weber-d97f9c1f39', jobTitle: 'Health service manager', }, @@ -8755,7 +8755,7 @@ export const peopleDemo = [ city: 'Eileentown', email: 'bonnie.anderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/bonnie-anderson-e54636e584', jobTitle: 'Town planner', }, @@ -8765,7 +8765,7 @@ export const peopleDemo = [ city: 'Port Dawn', email: 'tyler.barnett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/tyler-barnett-39213ade04', jobTitle: 'Industrial buyer', }, @@ -8775,7 +8775,7 @@ export const peopleDemo = [ city: 'Nathanielburgh', email: 'brenda.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/brenda-brown-dbcd1ad947', jobTitle: 'Secretary, company', }, @@ -8785,7 +8785,7 @@ export const peopleDemo = [ city: 'Lake Wendymouth', email: 'matthew.mills@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/matthew-mills-0b84ace87f', jobTitle: 'Nutritional therapist', }, @@ -8795,7 +8795,7 @@ export const peopleDemo = [ city: 'Jesseport', email: 'bradley.henderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/bradley-henderson-8d96b99f04', jobTitle: "Politician's assistant", }, @@ -8805,7 +8805,7 @@ export const peopleDemo = [ city: 'New Joe', email: 'christopher.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/christopher-wilson-04a5992bf8', jobTitle: 'Historic buildings inspector/conservation officer', }, @@ -8815,7 +8815,7 @@ export const peopleDemo = [ city: 'Williambury', email: 'janet.cooper@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/janet-cooper-5b3b8ebf22', jobTitle: 'Mental health nurse', }, @@ -8825,7 +8825,7 @@ export const peopleDemo = [ city: 'Wilsonton', email: 'ashlee.barajas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/ashlee-barajas-7db08816c2', jobTitle: 'Designer, television/film set', }, @@ -8835,7 +8835,7 @@ export const peopleDemo = [ city: 'Halefort', email: 'amanda.valenzuela@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/amanda-valenzuela-30589a563a', jobTitle: 'Engineer, automotive', }, @@ -8845,7 +8845,7 @@ export const peopleDemo = [ city: 'West Ianstad', email: 'charles.evans@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/charles-evans-afcfd375f0', jobTitle: 'Financial planner', }, @@ -8855,7 +8855,7 @@ export const peopleDemo = [ city: 'Myersberg', email: 'patricia.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/patricia-martinez-a6795704a8', jobTitle: 'Chartered public finance accountant', }, @@ -8865,7 +8865,7 @@ export const peopleDemo = [ city: 'East Nicholas', email: 'andrea.byrd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/andrea-byrd-6adea7eafa', jobTitle: 'Ecologist', }, @@ -8875,7 +8875,7 @@ export const peopleDemo = [ city: 'Walkerfurt', email: 'martin.hebert@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/martin-hebert-0fdd8cb20a', jobTitle: 'Statistician', }, @@ -8885,7 +8885,7 @@ export const peopleDemo = [ city: 'Michaelmouth', email: 'joyce.mathis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/joyce-mathis-4aefab1ba3', jobTitle: 'Multimedia specialist', }, @@ -8895,7 +8895,7 @@ export const peopleDemo = [ city: 'Lake Matthewmouth', email: 'charles.ray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/charles-ray-d81d2e4cf7', jobTitle: 'Communications engineer', }, @@ -8905,7 +8905,7 @@ export const peopleDemo = [ city: 'Brittanyhaven', email: 'amanda.vega@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/amanda-vega-bc1c79f067', jobTitle: 'Occupational therapist', }, @@ -8915,7 +8915,7 @@ export const peopleDemo = [ city: 'Richardchester', email: 'kathryn.freeman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/kathryn-freeman-5cbbc22506', jobTitle: 'Wellsite geologist', }, @@ -8925,7 +8925,7 @@ export const peopleDemo = [ city: 'South Robinberg', email: 'ryan.chambers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/ryan-chambers-2ce1a65a0c', jobTitle: 'Agricultural consultant', }, @@ -8935,7 +8935,7 @@ export const peopleDemo = [ city: 'Boydburgh', email: 'dustin.carr@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/dustin-carr-aea35977e8', jobTitle: 'Ergonomist', }, @@ -8945,7 +8945,7 @@ export const peopleDemo = [ city: 'Joannport', email: 'eugene.sims@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/eugene-sims-35b50bccf6', jobTitle: 'Banker', }, @@ -8955,7 +8955,7 @@ export const peopleDemo = [ city: 'Darrenmouth', email: 'gwendolyn.glover@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/gwendolyn-glover-c850972ca2', jobTitle: 'Geochemist', }, @@ -8965,7 +8965,7 @@ export const peopleDemo = [ city: 'Woodberg', email: 'kevin.oconnell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/kevin-oconnell-366df31264', jobTitle: 'Editor, commissioning', }, @@ -8975,7 +8975,7 @@ export const peopleDemo = [ city: 'Port Ginatown', email: 'mark.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/mark-williams-34678a412b', jobTitle: 'Education administrator', }, @@ -8985,7 +8985,7 @@ export const peopleDemo = [ city: 'West Heatherbury', email: 'jack.reed@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/jack-reed-624aef385a', jobTitle: 'Corporate treasurer', }, @@ -8995,7 +8995,7 @@ export const peopleDemo = [ city: 'West Dannyside', email: 'anthony.green@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/anthony-green-26a1d57a62', jobTitle: 'Land', }, @@ -9005,7 +9005,7 @@ export const peopleDemo = [ city: 'Cameronton', email: 'louis.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/louis-johnson-dec8e3a5a2', jobTitle: 'Pathologist', }, @@ -9015,7 +9015,7 @@ export const peopleDemo = [ city: 'South Scott', email: 'fernando.stephens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/fernando-stephens-d825082895', jobTitle: 'Sports development officer', }, @@ -9025,7 +9025,7 @@ export const peopleDemo = [ city: 'Sampsonville', email: 'tammy.soto@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/tammy-soto-dc33b99453', jobTitle: 'Consulting civil engineer', }, @@ -9035,7 +9035,7 @@ export const peopleDemo = [ city: 'Mcdonaldside', email: 'anthony.clay@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/anthony-clay-d0ffc44035', jobTitle: 'Site engineer', }, @@ -9045,7 +9045,7 @@ export const peopleDemo = [ city: 'South Deanbury', email: 'jennifer.haney@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/jennifer-haney-fe08f83150', jobTitle: 'Engineer, broadcasting (operations)', }, @@ -9055,7 +9055,7 @@ export const peopleDemo = [ city: 'West Jillian', email: 'kevin.harris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/kevin-harris-5184b24b32', jobTitle: 'Presenter, broadcasting', }, @@ -9065,7 +9065,7 @@ export const peopleDemo = [ city: 'Jeffreyland', email: 'allison.crawford@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/allison-crawford-67b761b025', jobTitle: 'Homeopath', }, @@ -9075,7 +9075,7 @@ export const peopleDemo = [ city: 'South Rebeccaburgh', email: 'stacey.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/stacey-garcia-9250901bae', jobTitle: 'Bonds trader', }, @@ -9085,7 +9085,7 @@ export const peopleDemo = [ city: 'Johnport', email: 'stacey.romero@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/stacey-romero-7728909deb', jobTitle: 'Civil Service fast streamer', }, @@ -9095,7 +9095,7 @@ export const peopleDemo = [ city: 'Danielfort', email: 'joseph.bell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/joseph-bell-0f0d64c86b', jobTitle: 'Historic buildings inspector/conservation officer', }, @@ -9105,7 +9105,7 @@ export const peopleDemo = [ city: 'Jerrybury', email: 'nicholas.edwards@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/nicholas-edwards-208dfccc7e', jobTitle: 'Community development worker', }, @@ -9115,7 +9115,7 @@ export const peopleDemo = [ city: 'North Ashleyburgh', email: 'brian.freeman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/brian-freeman-f01b205c86', jobTitle: 'Social researcher', }, @@ -9125,7 +9125,7 @@ export const peopleDemo = [ city: 'Alvaradoberg', email: 'christine.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/christine-johnson-20bfd043a5', jobTitle: 'Systems analyst', }, @@ -9135,7 +9135,7 @@ export const peopleDemo = [ city: 'East Anthonychester', email: 'christine.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/christine-brown-cf7634fc0b', jobTitle: 'Museum/gallery exhibitions officer', }, @@ -9145,7 +9145,7 @@ export const peopleDemo = [ city: 'Port Brian', email: 'grant.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/grant-brown-b0072f7d7c', jobTitle: 'Financial trader', }, @@ -9155,7 +9155,7 @@ export const peopleDemo = [ city: 'Thomasmouth', email: 'megan.robinson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/megan-robinson-ee228aab40', jobTitle: 'Cartographer', }, @@ -9165,7 +9165,7 @@ export const peopleDemo = [ city: 'South Nathan', email: 'ronald.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/ronald-smith-9a58c743fc', jobTitle: 'Engineer, aeronautical', }, @@ -9175,7 +9175,7 @@ export const peopleDemo = [ city: 'Lake Nicolefurt', email: 'tonya.chandler@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/tonya-chandler-8cba0ccb14', jobTitle: 'Surveyor, building', }, @@ -9185,7 +9185,7 @@ export const peopleDemo = [ city: 'West John', email: 'jose.jacobs@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/jose-jacobs-e5dd5c613f', jobTitle: 'Environmental consultant', }, @@ -9195,7 +9195,7 @@ export const peopleDemo = [ city: 'Port Charlesfurt', email: 'william.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/william-johnson-fffd051063', jobTitle: 'Oceanographer', }, @@ -9205,7 +9205,7 @@ export const peopleDemo = [ city: 'East Thomas', email: 'jason.mitchell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/jason-mitchell-d3f1d47944', jobTitle: 'Hydrographic surveyor', }, @@ -9215,7 +9215,7 @@ export const peopleDemo = [ city: 'Lake Nathan', email: 'wendy.soto@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/wendy-soto-507c1d708f', jobTitle: 'Metallurgist', }, @@ -9225,7 +9225,7 @@ export const peopleDemo = [ city: 'North Isaac', email: 'patrick.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/patrick-jones-33ee1b90f5', jobTitle: 'Engineer, biomedical', }, @@ -9235,7 +9235,7 @@ export const peopleDemo = [ city: 'Bowenbury', email: 'vanessa.ingram@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/vanessa-ingram-dd5d4ea14e', jobTitle: 'Hospital pharmacist', }, @@ -9245,7 +9245,7 @@ export const peopleDemo = [ city: 'Stanleymouth', email: 'walter.rhodes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/walter-rhodes-8cff091883', jobTitle: 'Actuary', }, @@ -9255,7 +9255,7 @@ export const peopleDemo = [ city: 'Gibsonfort', email: 'heather.cardenas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/heather-cardenas-0a0d52106f', jobTitle: 'Sport and exercise psychologist', }, @@ -9265,7 +9265,7 @@ export const peopleDemo = [ city: 'North Janicebury', email: 'nathan.schwartz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/nathan-schwartz-db3bb93189', jobTitle: 'Merchandiser, retail', }, @@ -9275,7 +9275,7 @@ export const peopleDemo = [ city: 'New Michaelberg', email: 'roger.gill@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/roger-gill-367ad4e3fc', jobTitle: 'Lighting technician, broadcasting/film/video', }, @@ -9285,7 +9285,7 @@ export const peopleDemo = [ city: 'New Jerryfort', email: 'cynthia.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/cynthia-taylor-c8c2e63cb4', jobTitle: 'Biomedical scientist', }, @@ -9295,7 +9295,7 @@ export const peopleDemo = [ city: 'Christopherberg', email: 'jenna.rojas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/jenna-rojas-01776ae7b4', jobTitle: 'Clinical biochemist', }, @@ -9305,7 +9305,7 @@ export const peopleDemo = [ city: 'Williamview', email: 'trevor.chase@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/trevor-chase-c2e65df749', jobTitle: 'Lobbyist', }, @@ -9315,7 +9315,7 @@ export const peopleDemo = [ city: 'Charlesberg', email: 'scott.murphy@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/scott-murphy-2e879fc86d', jobTitle: 'Commissioning editor', }, @@ -9325,7 +9325,7 @@ export const peopleDemo = [ city: 'New Laura', email: 'zachary.thornton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/zachary-thornton-303f2657b8', jobTitle: 'Wellsite geologist', }, @@ -9335,7 +9335,7 @@ export const peopleDemo = [ city: 'Kimberlychester', email: 'richard.aguirre@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/richard-aguirre-40f00db472', jobTitle: 'Barista', }, @@ -9345,7 +9345,7 @@ export const peopleDemo = [ city: 'Danielhaven', email: 'mckenzie.black@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/mckenzie-black-5e934e9a1d', jobTitle: 'Scientist, biomedical', }, @@ -9355,7 +9355,7 @@ export const peopleDemo = [ city: 'Port Chadport', email: 'jacqueline.randall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/jacqueline-randall-38109939cb', jobTitle: 'Retail manager', }, @@ -9365,7 +9365,7 @@ export const peopleDemo = [ city: 'Port William', email: 'sheri.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/sheri-taylor-df26d6d5ee', jobTitle: 'Commercial art gallery manager', }, @@ -9375,7 +9375,7 @@ export const peopleDemo = [ city: 'East Christinaburgh', email: 'brandon.acevedo@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/brandon-acevedo-32a27033d1', jobTitle: 'Investment banker, corporate', }, @@ -9385,7 +9385,7 @@ export const peopleDemo = [ city: 'Wintersside', email: 'katherine.best@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/katherine-best-9e3dcb6aa0', jobTitle: 'Corporate treasurer', }, @@ -9395,7 +9395,7 @@ export const peopleDemo = [ city: 'Lake Betty', email: 'daniel.adams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/daniel-adams-fbc7ca02b9', jobTitle: 'Research officer, trade union', }, @@ -9405,7 +9405,7 @@ export const peopleDemo = [ city: 'East Robertfurt', email: 'elizabeth.vega@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/elizabeth-vega-4755e545bd', jobTitle: 'Investment banker, corporate', }, @@ -9415,7 +9415,7 @@ export const peopleDemo = [ city: 'Ronaldland', email: 'jennifer.kim@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/jennifer-kim-e40f68f1d3', jobTitle: 'Counselling psychologist', }, @@ -9425,7 +9425,7 @@ export const peopleDemo = [ city: 'Jenniferstad', email: 'stephen.saunders@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/stephen-saunders-d055c5642e', jobTitle: 'Biomedical engineer', }, @@ -9435,7 +9435,7 @@ export const peopleDemo = [ city: 'South Amanda', email: 'ashley.prince@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/ashley-prince-59d88abe81', jobTitle: 'Ophthalmologist', }, @@ -9445,7 +9445,7 @@ export const peopleDemo = [ city: 'Port Charlesfurt', email: 'paul.mckay@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/paul-mckay-00b47ec261', jobTitle: 'Physiotherapist', }, @@ -9455,7 +9455,7 @@ export const peopleDemo = [ city: 'East Lauraview', email: 'shelby.hughes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/shelby-hughes-7c90e603de', jobTitle: 'Video editor', }, @@ -9465,7 +9465,7 @@ export const peopleDemo = [ city: 'North Keith', email: 'cheryl.townsend@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/cheryl-townsend-62c7a27460', jobTitle: 'Marketing executive', }, @@ -9475,7 +9475,7 @@ export const peopleDemo = [ city: 'Angelaborough', email: 'brianna.peck@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/brianna-peck-ab76fe8301', jobTitle: 'Mudlogger', }, @@ -9485,7 +9485,7 @@ export const peopleDemo = [ city: 'New Destiny', email: 'andrew.ford@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/andrew-ford-1ddb22e213', jobTitle: 'Interpreter', }, @@ -9495,7 +9495,7 @@ export const peopleDemo = [ city: 'Port Grace', email: 'robert.brown@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/robert-brown-28cac6c157', jobTitle: 'Public house manager', }, @@ -9505,7 +9505,7 @@ export const peopleDemo = [ city: 'East Melvinberg', email: 'joy.richards@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/joy-richards-f0af7d1ee4', jobTitle: 'Tree surgeon', }, @@ -9515,7 +9515,7 @@ export const peopleDemo = [ city: 'South Jamesview', email: 'jessica.hogan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/jessica-hogan-54aff81ceb', jobTitle: 'Land', }, @@ -9525,7 +9525,7 @@ export const peopleDemo = [ city: 'Lisaport', email: 'lisa.watson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/lisa-watson-0cae086726', jobTitle: 'Graphic designer', }, @@ -9535,7 +9535,7 @@ export const peopleDemo = [ city: 'Billyfurt', email: 'christine.morton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/christine-morton-3d83c68241', jobTitle: 'Stage manager', }, @@ -9545,7 +9545,7 @@ export const peopleDemo = [ city: 'Ballfurt', email: 'brian.lewis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/brian-lewis-de68c08ac1', jobTitle: 'Trade union research officer', }, @@ -9555,7 +9555,7 @@ export const peopleDemo = [ city: 'North Dylanbury', email: 'matthew.fernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/matthew-fernandez-1cd32d9567', jobTitle: 'Lecturer, further education', }, @@ -9565,7 +9565,7 @@ export const peopleDemo = [ city: 'Masseyfurt', email: 'linda.mitchell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/linda-mitchell-825783bf1c', jobTitle: 'Mechanical engineer', }, @@ -9575,7 +9575,7 @@ export const peopleDemo = [ city: 'East Scott', email: 'colin.walker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/colin-walker-bc08fcfaa1', jobTitle: 'Surveyor, mining', }, @@ -9585,7 +9585,7 @@ export const peopleDemo = [ city: 'South Georgehaven', email: 'robert.gray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/robert-gray-2c8a3e5f3a', jobTitle: 'Development worker, international aid', }, @@ -9595,7 +9595,7 @@ export const peopleDemo = [ city: 'Lozanofurt', email: 'natalie.lawrence@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/natalie-lawrence-d8c06e2d82', jobTitle: 'Therapeutic radiographer', }, @@ -9605,7 +9605,7 @@ export const peopleDemo = [ city: 'South Michaelbury', email: 'mark.castro@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/mark-castro-4184957d96', jobTitle: 'Engineer, control and instrumentation', }, @@ -9615,7 +9615,7 @@ export const peopleDemo = [ city: 'Boothville', email: 'melissa.molina@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/melissa-molina-aae4218215', jobTitle: 'Psychologist, clinical', }, @@ -9625,7 +9625,7 @@ export const peopleDemo = [ city: 'Murrayburgh', email: 'terry.melendez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/terry-melendez-692cdf776c', jobTitle: 'Public relations officer', }, @@ -9635,7 +9635,7 @@ export const peopleDemo = [ city: 'Madisonbury', email: 'charlene.beck@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/charlene-beck-810d5075ae', jobTitle: 'Tax inspector', }, @@ -9645,7 +9645,7 @@ export const peopleDemo = [ city: 'West Geoffrey', email: 'yvonne.avila@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/yvonne-avila-cd93548e92', jobTitle: 'Merchant navy officer', }, @@ -9655,7 +9655,7 @@ export const peopleDemo = [ city: 'Lake Kathy', email: 'andrea.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/andrea-garcia-21bf3350f1', jobTitle: 'Contracting civil engineer', }, @@ -9665,7 +9665,7 @@ export const peopleDemo = [ city: 'Daniellehaven', email: 'mary.goodman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/mary-goodman-97470c0612', jobTitle: 'Metallurgist', }, @@ -9675,7 +9675,7 @@ export const peopleDemo = [ city: 'Lake Ryanbury', email: 'shelly.powers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/shelly-powers-8790890d27', jobTitle: 'Glass blower/designer', }, @@ -9685,7 +9685,7 @@ export const peopleDemo = [ city: 'Morgantown', email: 'holly.hensley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/holly-hensley-111036f6da', jobTitle: 'Geophysicist/field seismologist', }, @@ -9695,7 +9695,7 @@ export const peopleDemo = [ city: 'New Kelly', email: 'christina.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/christina-davis-2b07b44392', jobTitle: 'Chiropodist', }, @@ -9705,7 +9705,7 @@ export const peopleDemo = [ city: 'Port Markhaven', email: 'adam.cochran@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/adam-cochran-1657e82dbf', jobTitle: 'Communications engineer', }, @@ -9715,7 +9715,7 @@ export const peopleDemo = [ city: 'Johnsonland', email: 'katherine.abbott@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/katherine-abbott-ec30f06ab2', jobTitle: 'Solicitor, Scotland', }, @@ -9725,7 +9725,7 @@ export const peopleDemo = [ city: 'Jameshaven', email: 'jenna.mendez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/jenna-mendez-38ef424a7f', jobTitle: 'Automotive engineer', }, @@ -9735,7 +9735,7 @@ export const peopleDemo = [ city: 'Lake Ronald', email: 'steven.barnes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/steven-barnes-f5d1f0c993', jobTitle: 'Occupational hygienist', }, @@ -9745,7 +9745,7 @@ export const peopleDemo = [ city: 'Spencefort', email: 'ashley.manning@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/ashley-manning-8b3cc7cf6f', jobTitle: 'Firefighter', }, @@ -9755,7 +9755,7 @@ export const peopleDemo = [ city: 'Amytown', email: 'david.peterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/david-peterson-296c6d71cf', jobTitle: 'Scientist, water quality', }, @@ -9765,7 +9765,7 @@ export const peopleDemo = [ city: 'East Michael', email: 'patrick.ellis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/patrick-ellis-984f4db9b8', jobTitle: 'Graphic designer', }, @@ -9775,7 +9775,7 @@ export const peopleDemo = [ city: 'Shepherdburgh', email: 'james.sullivan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/james-sullivan-711645bcdd', jobTitle: 'Chief Strategy Officer', }, @@ -9785,7 +9785,7 @@ export const peopleDemo = [ city: 'East Rodneyshire', email: 'jeffrey.beck@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/jeffrey-beck-a78371285a', jobTitle: 'Corporate investment banker', }, @@ -9795,7 +9795,7 @@ export const peopleDemo = [ city: 'Tammyville', email: 'joyce.phillips@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/joyce-phillips-781047cb11', jobTitle: 'Community pharmacist', }, @@ -9805,7 +9805,7 @@ export const peopleDemo = [ city: 'Port Daniellemouth', email: 'nicholas.sanchez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/nicholas-sanchez-ba28dda9a9', jobTitle: 'Technical sales engineer', }, @@ -9815,7 +9815,7 @@ export const peopleDemo = [ city: 'Port Brandonberg', email: 'john.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/john-perez-55c123ef5d', jobTitle: 'Research scientist (physical sciences)', }, @@ -9825,7 +9825,7 @@ export const peopleDemo = [ city: 'Taraview', email: 'tammy.mueller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/tammy-mueller-69c65883c9', jobTitle: 'Energy manager', }, @@ -9835,7 +9835,7 @@ export const peopleDemo = [ city: 'Santiagochester', email: 'susan.wong@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/susan-wong-9581ecf892', jobTitle: 'Lobbyist', }, @@ -9845,7 +9845,7 @@ export const peopleDemo = [ city: 'Lake Charles', email: 'jacob.lutz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/jacob-lutz-29f1197777', jobTitle: 'Energy engineer', }, @@ -9855,7 +9855,7 @@ export const peopleDemo = [ city: 'Port Tammyshire', email: 'mark.cruz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/mark-cruz-88019b5101', jobTitle: 'Pilot, airline', }, @@ -9865,7 +9865,7 @@ export const peopleDemo = [ city: 'East Matthewtown', email: 'sharon.soto@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/sharon-soto-288840ca64', jobTitle: 'Therapist, drama', }, @@ -9875,7 +9875,7 @@ export const peopleDemo = [ city: 'Whitefurt', email: 'maria.rodgers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/maria-rodgers-3b3e0df751', jobTitle: 'Merchandiser, retail', }, @@ -9885,7 +9885,7 @@ export const peopleDemo = [ city: 'Hernandezchester', email: 'scott.norton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/scott-norton-fa28a83774', jobTitle: 'Museum/gallery exhibitions officer', }, @@ -9895,7 +9895,7 @@ export const peopleDemo = [ city: 'East David', email: 'caitlin.harper@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/caitlin-harper-13e7507d0b', jobTitle: 'Animal nutritionist', }, @@ -9905,7 +9905,7 @@ export const peopleDemo = [ city: 'East Robertburgh', email: 'elizabeth.newman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/elizabeth-newman-b3274ecf1c', jobTitle: 'Curator', }, @@ -9915,7 +9915,7 @@ export const peopleDemo = [ city: 'Joshualand', email: 'rebecca.knight@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/rebecca-knight-f9ba229de1', jobTitle: 'Marine scientist', }, @@ -9925,7 +9925,7 @@ export const peopleDemo = [ city: 'Port Tammyside', email: 'rebecca.henry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/rebecca-henry-8af895981f', jobTitle: 'Graphic designer', }, @@ -9935,7 +9935,7 @@ export const peopleDemo = [ city: 'Leslieberg', email: 'douglas.mccall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/douglas-mccall-bbacffb65f', jobTitle: 'Producer, television/film/video', }, @@ -9945,7 +9945,7 @@ export const peopleDemo = [ city: 'South Thomas', email: 'shelia.mcneil@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/shelia-mcneil-09a1a630f5', jobTitle: 'Surveyor, quantity', }, @@ -9955,7 +9955,7 @@ export const peopleDemo = [ city: 'Port George', email: 'diana.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/diana-moore-bfde41a990', jobTitle: 'Public house manager', }, @@ -9965,7 +9965,7 @@ export const peopleDemo = [ city: 'North Dennis', email: 'andrea.gregory@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/andrea-gregory-1bed92c29b', jobTitle: 'Clinical cytogeneticist', }, @@ -9975,7 +9975,7 @@ export const peopleDemo = [ city: 'Port Michael', email: 'sandra.houston@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/sandra-houston-5eb9930bec', jobTitle: 'Research scientist (maths)', }, @@ -9985,7 +9985,7 @@ export const peopleDemo = [ city: 'West Judyfort', email: 'christina.rangel@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/christina-rangel-14f26a977d', jobTitle: 'Immunologist', }, @@ -9995,7 +9995,7 @@ export const peopleDemo = [ city: 'Smithchester', email: 'ruben.aguilar@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/ruben-aguilar-abaeafdd06', jobTitle: 'Counselling psychologist', }, @@ -10005,7 +10005,7 @@ export const peopleDemo = [ city: 'Smithstad', email: 'briana.townsend@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/briana-townsend-b3b34529fe', jobTitle: 'Cytogeneticist', }, @@ -10015,7 +10015,7 @@ export const peopleDemo = [ city: 'Lake Kimstad', email: 'william.thompson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/william-thompson-94dc40f1de', jobTitle: 'Equities trader', }, @@ -10025,7 +10025,7 @@ export const peopleDemo = [ city: 'Donaldburgh', email: 'ashley.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/ashley-martinez-a406dfe50c', jobTitle: 'Ecologist', }, @@ -10035,7 +10035,7 @@ export const peopleDemo = [ city: 'Karlland', email: 'carla.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/carla-wilson-d059473869', jobTitle: 'Media buyer', }, @@ -10045,7 +10045,7 @@ export const peopleDemo = [ city: 'Lesterfort', email: 'robert.contreras@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/robert-contreras-50f852ba89', jobTitle: 'Ecologist', }, @@ -10055,7 +10055,7 @@ export const peopleDemo = [ city: 'Mahoneyhaven', email: 'denise.burton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/denise-burton-3a79ce7f23', jobTitle: 'Clothing/textile technologist', }, @@ -10065,7 +10065,7 @@ export const peopleDemo = [ city: 'East Sydneymouth', email: 'cassidy.mckee@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/cassidy-mckee-1189782a1e', jobTitle: 'Psychotherapist, child', }, @@ -10075,7 +10075,7 @@ export const peopleDemo = [ city: 'East Kaitlinborough', email: 'angela.torres@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/angela-torres-1c99a88f63', jobTitle: 'Agricultural engineer', }, @@ -10085,7 +10085,7 @@ export const peopleDemo = [ city: 'Millerbury', email: 'shirley.hall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/shirley-hall-68f8f386ca', jobTitle: 'Editorial assistant', }, @@ -10095,7 +10095,7 @@ export const peopleDemo = [ city: 'New Todd', email: 'robert.arroyo@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/robert-arroyo-956557b51d', jobTitle: 'Counselling psychologist', }, @@ -10105,7 +10105,7 @@ export const peopleDemo = [ city: 'Kennedystad', email: 'kurt.moon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/kurt-moon-c7b749a5da', jobTitle: 'Cabin crew', }, @@ -10115,7 +10115,7 @@ export const peopleDemo = [ city: 'North Krystal', email: 'nicholas.bradshaw@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/nicholas-bradshaw-414f2727de', jobTitle: 'Financial controller', }, @@ -10125,7 +10125,7 @@ export const peopleDemo = [ city: 'Gregoryport', email: 'tyler.murray@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/tyler-murray-244f09586c', jobTitle: 'Psychotherapist, child', }, @@ -10135,7 +10135,7 @@ export const peopleDemo = [ city: 'Desireebury', email: 'shawn.lowery@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/shawn-lowery-7baa4120f2', jobTitle: 'Chartered management accountant', }, @@ -10145,7 +10145,7 @@ export const peopleDemo = [ city: 'Gabrielborough', email: 'jessica.ward@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/jessica-ward-c7e5de5066', jobTitle: 'Acupuncturist', }, @@ -10155,7 +10155,7 @@ export const peopleDemo = [ city: 'Williamsfort', email: 'james.vazquez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/james-vazquez-edc804602a', jobTitle: 'Housing manager/officer', }, @@ -10165,7 +10165,7 @@ export const peopleDemo = [ city: 'Joshuamouth', email: 'jeffrey.moyer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/jeffrey-moyer-82eb5b4ba8', jobTitle: 'Herpetologist', }, @@ -10175,7 +10175,7 @@ export const peopleDemo = [ city: 'Fosterview', email: 'matthew.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/matthew-moore-13cab3f864', jobTitle: 'Secondary school teacher', }, @@ -10185,7 +10185,7 @@ export const peopleDemo = [ city: 'Mariamouth', email: 'crystal.pena@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/crystal-pena-620fa626fd', jobTitle: 'Therapist, occupational', }, @@ -10195,7 +10195,7 @@ export const peopleDemo = [ city: 'Michaelfort', email: 'ann.mclaughlin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/ann-mclaughlin-cfe670a52c', jobTitle: 'Quality manager', }, @@ -10205,7 +10205,7 @@ export const peopleDemo = [ city: 'Navarromouth', email: 'corey.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/corey-jones-2b4ff4b6c5', jobTitle: 'Nutritional therapist', }, @@ -10215,7 +10215,7 @@ export const peopleDemo = [ city: 'Lake James', email: 'james.boyer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/james-boyer-dcd4131baf', jobTitle: 'Management consultant', }, @@ -10225,7 +10225,7 @@ export const peopleDemo = [ city: 'Lake Mary', email: 'karen.schroeder@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/karen-schroeder-9ded3a0da2', jobTitle: 'Chiropodist', }, @@ -10235,7 +10235,7 @@ export const peopleDemo = [ city: 'Robinsonchester', email: 'ashley.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/ashley-johnson-0d0a8f76d0', jobTitle: 'Presenter, broadcasting', }, @@ -10245,7 +10245,7 @@ export const peopleDemo = [ city: 'West Andrea', email: 'susan.stevens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/susan-stevens-75236edd7c', jobTitle: 'Psychologist, sport and exercise', }, @@ -10255,7 +10255,7 @@ export const peopleDemo = [ city: 'Stacyhaven', email: 'jennifer.deleon@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/jennifer-deleon-92f894acba', jobTitle: 'Development worker, community', }, @@ -10265,7 +10265,7 @@ export const peopleDemo = [ city: 'Port Jessica', email: 'kelsey.lopez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/kelsey-lopez-90ffaeecdc', jobTitle: 'English as a foreign language teacher', }, @@ -10275,7 +10275,7 @@ export const peopleDemo = [ city: 'Lake Larry', email: 'jill.hammond@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/jill-hammond-4f87a8c4fd', jobTitle: 'Designer, jewellery', }, @@ -10285,7 +10285,7 @@ export const peopleDemo = [ city: 'Pearsonside', email: 'joseph.wu@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/joseph-wu-f673a42110', jobTitle: 'Naval architect', }, @@ -10295,7 +10295,7 @@ export const peopleDemo = [ city: 'Walkerberg', email: 'melissa.walker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/melissa-walker-34f194896b', jobTitle: 'Meteorologist', }, @@ -10305,7 +10305,7 @@ export const peopleDemo = [ city: 'North Williamstad', email: 'lisa.mcdaniel@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/lisa-mcdaniel-9ab62dfbef', jobTitle: 'Investment banker, corporate', }, @@ -10315,7 +10315,7 @@ export const peopleDemo = [ city: 'West Sandra', email: 'ruben.robinson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/ruben-robinson-0da78d81dc', jobTitle: 'Environmental health practitioner', }, @@ -10325,7 +10325,7 @@ export const peopleDemo = [ city: 'New Benjamin', email: 'david.novak@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/david-novak-c3c6f0fb18', jobTitle: 'Pharmacologist', }, @@ -10335,7 +10335,7 @@ export const peopleDemo = [ city: 'Faulknerport', email: 'carl.osborne@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/carl-osborne-4aa9f429ec', jobTitle: 'Arts administrator', }, @@ -10345,7 +10345,7 @@ export const peopleDemo = [ city: 'Campbellbury', email: 'jennifer.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/jennifer-moore-0468307bea', jobTitle: 'Water engineer', }, @@ -10355,7 +10355,7 @@ export const peopleDemo = [ city: 'Frederickland', email: 'nicolas.walton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/nicolas-walton-41b1798348', jobTitle: 'Clinical molecular geneticist', }, @@ -10365,7 +10365,7 @@ export const peopleDemo = [ city: 'East Steven', email: 'robert.vega@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/robert-vega-c9f7c2fc77', jobTitle: 'Engineer, automotive', }, @@ -10375,7 +10375,7 @@ export const peopleDemo = [ city: 'South Sandrafurt', email: 'emily.morrison@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/emily-morrison-dfcf31f9c8', jobTitle: 'Learning disability nurse', }, @@ -10385,7 +10385,7 @@ export const peopleDemo = [ city: 'Hodgesfort', email: 'michael.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/michael-johnson-67b413c0fb', jobTitle: 'Consulting civil engineer', }, @@ -10395,7 +10395,7 @@ export const peopleDemo = [ city: 'Port Brian', email: 'chris.miller@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/chris-miller-e5fd1642f6', jobTitle: 'Artist', }, @@ -10405,7 +10405,7 @@ export const peopleDemo = [ city: 'Allisonborough', email: 'ebony.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/ebony-jones-46514a3944', jobTitle: 'Armed forces training and education officer', }, @@ -10415,7 +10415,7 @@ export const peopleDemo = [ city: 'Rickyshire', email: 'bonnie.mcintyre@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/bonnie-mcintyre-a8669d6f36', jobTitle: 'Gaffer', }, @@ -10425,7 +10425,7 @@ export const peopleDemo = [ city: 'New Sarah', email: 'tom.dawson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/tom-dawson-f004272b8c', jobTitle: 'Mechanical engineer', }, @@ -10435,7 +10435,7 @@ export const peopleDemo = [ city: 'West Melissa', email: 'sharon.weber@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/sharon-weber-793caf2e63', jobTitle: 'Magazine journalist', }, @@ -10445,7 +10445,7 @@ export const peopleDemo = [ city: 'West Jenniferstad', email: 'rodney.lewis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/rodney-lewis-ec6d41e8de', jobTitle: 'Scientist, research (physical sciences)', }, @@ -10455,7 +10455,7 @@ export const peopleDemo = [ city: 'Rossport', email: 'jordan.norton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/jordan-norton-0c59185977', jobTitle: 'Armed forces training and education officer', }, @@ -10465,7 +10465,7 @@ export const peopleDemo = [ city: 'Jonesland', email: 'stephen.kramer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/stephen-kramer-3c9febe618', jobTitle: 'Journalist, broadcasting', }, @@ -10475,7 +10475,7 @@ export const peopleDemo = [ city: 'East Stefanie', email: 'hannah.miles@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/hannah-miles-a7dda4b311', jobTitle: 'Retail manager', }, @@ -10485,7 +10485,7 @@ export const peopleDemo = [ city: 'West Luis', email: 'matthew.gomez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/matthew-gomez-138bea0818', jobTitle: 'Landscape architect', }, @@ -10495,7 +10495,7 @@ export const peopleDemo = [ city: 'West Jameshaven', email: 'brian.ashley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/brian-ashley-97ba5b196c', jobTitle: 'Clinical psychologist', }, @@ -10505,7 +10505,7 @@ export const peopleDemo = [ city: 'Port Debraburgh', email: 'tyler.silva@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/tyler-silva-13e0220240', jobTitle: 'Podiatrist', }, @@ -10515,7 +10515,7 @@ export const peopleDemo = [ city: 'Walterberg', email: 'timothy.neal@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/timothy-neal-dc5fb8cee9', jobTitle: 'Media buyer', }, @@ -10525,7 +10525,7 @@ export const peopleDemo = [ city: 'Port Jody', email: 'melanie.mora@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/melanie-mora-2690667b9b', jobTitle: 'Personal assistant', }, @@ -10535,7 +10535,7 @@ export const peopleDemo = [ city: 'East Lindsay', email: 'michael.gonzalez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/michael-gonzalez-06594d52b5', jobTitle: 'Waste management officer', }, @@ -10545,7 +10545,7 @@ export const peopleDemo = [ city: 'Ramosborough', email: 'anthony.moran@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/anthony-moran-1b9585391d', jobTitle: 'Accountant, chartered certified', }, @@ -10555,7 +10555,7 @@ export const peopleDemo = [ city: 'New Anthonyhaven', email: 'kelly.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/kelly-taylor-a64148e880', jobTitle: 'Nutritional therapist', }, @@ -10565,7 +10565,7 @@ export const peopleDemo = [ city: 'East Brandy', email: 'jesse.lawson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/jesse-lawson-b8e1d0a3a6', jobTitle: 'Industrial/product designer', }, @@ -10575,7 +10575,7 @@ export const peopleDemo = [ city: 'Bauerburgh', email: 'kaylee.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/kaylee-wilson-816a279b72', jobTitle: 'Chartered management accountant', }, @@ -10585,7 +10585,7 @@ export const peopleDemo = [ city: 'New Rileystad', email: 'ernest.benson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/ernest-benson-62c7191ff3', jobTitle: 'Data scientist', }, @@ -10595,7 +10595,7 @@ export const peopleDemo = [ city: 'Martinberg', email: 'anthony.garcia@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/anthony-garcia-1d2ba220ca', jobTitle: 'Brewing technologist', }, @@ -10605,7 +10605,7 @@ export const peopleDemo = [ city: 'South Alexisview', email: 'karen.moody@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/karen-moody-f6f256a833', jobTitle: 'Conservation officer, nature', }, @@ -10615,7 +10615,7 @@ export const peopleDemo = [ city: 'North Russell', email: 'erin.jimenez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/erin-jimenez-0cc5ce0c10', jobTitle: 'Press sub', }, @@ -10625,7 +10625,7 @@ export const peopleDemo = [ city: 'West Ricardo', email: 'daniel.boyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/daniel-boyd-2fa5991fba', jobTitle: 'Telecommunications researcher', }, @@ -10635,7 +10635,7 @@ export const peopleDemo = [ city: 'West Jillianchester', email: 'robert.garrett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/robert-garrett-2fade0517f', jobTitle: 'Immigration officer', }, @@ -10645,7 +10645,7 @@ export const peopleDemo = [ city: 'Morrisonchester', email: 'david.vazquez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/david-vazquez-852b6ab31b', jobTitle: 'Purchasing manager', }, @@ -10655,7 +10655,7 @@ export const peopleDemo = [ city: 'Michaelchester', email: 'zachary.simmons@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/zachary-simmons-3b73fdab08', jobTitle: 'Education officer, museum', }, @@ -10665,7 +10665,7 @@ export const peopleDemo = [ city: 'Tiffanyside', email: 'melissa.wilson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/melissa-wilson-8479abddcc', jobTitle: 'General practice doctor', }, @@ -10675,7 +10675,7 @@ export const peopleDemo = [ city: 'Webertown', email: 'michelle.thompson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/michelle-thompson-bf79635c87', jobTitle: 'Counsellor', }, @@ -10685,7 +10685,7 @@ export const peopleDemo = [ city: 'Rodneyburgh', email: 'donald.stephens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/donald-stephens-7f7cf70def', jobTitle: 'Private music teacher', }, @@ -10695,7 +10695,7 @@ export const peopleDemo = [ city: 'Paulachester', email: 'marcus.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/marcus-smith-3a27ce898a', jobTitle: 'Clinical biochemist', }, @@ -10705,7 +10705,7 @@ export const peopleDemo = [ city: 'Robertbury', email: 'norma.watkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/norma-watkins-61bad82fae', jobTitle: 'Firefighter', }, @@ -10715,7 +10715,7 @@ export const peopleDemo = [ city: 'Rodneyfurt', email: 'jody.morales@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/jody-morales-9f975a16d2', jobTitle: 'Administrator, education', }, @@ -10725,7 +10725,7 @@ export const peopleDemo = [ city: 'North Barbaraville', email: 'ronald.cox@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/ronald-cox-f10c08c550', jobTitle: 'Sports coach', }, @@ -10735,7 +10735,7 @@ export const peopleDemo = [ city: 'Flemingmouth', email: 'jennifer.horn@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/jennifer-horn-7eade4bcb9', jobTitle: 'Product designer', }, @@ -10745,7 +10745,7 @@ export const peopleDemo = [ city: 'Johnside', email: 'joseph.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/joseph-jones-dcec25d4d9', jobTitle: 'Research scientist (maths)', }, @@ -10755,7 +10755,7 @@ export const peopleDemo = [ city: 'Lake Dean', email: 'cody.blevins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/cody-blevins-57df0cb073', jobTitle: 'Clinical psychologist', }, @@ -10765,7 +10765,7 @@ export const peopleDemo = [ city: 'Hamiltonstad', email: 'allison.hickman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/allison-hickman-07a1fac3cf', jobTitle: 'Writer', }, @@ -10775,7 +10775,7 @@ export const peopleDemo = [ city: 'West Brendahaven', email: 'david.everett@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/david-everett-5a8004bbc1', jobTitle: 'Archaeologist', }, @@ -10785,7 +10785,7 @@ export const peopleDemo = [ city: 'Kariport', email: 'james.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/james-smith-ccd3177ab7', jobTitle: 'Film/video editor', }, @@ -10795,7 +10795,7 @@ export const peopleDemo = [ city: 'South Angela', email: 'chad.stevens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/chad-stevens-babda7a962', jobTitle: 'Higher education careers adviser', }, @@ -10805,7 +10805,7 @@ export const peopleDemo = [ city: 'Hillview', email: 'nicole.campbell@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/nicole-campbell-47b63850d5', jobTitle: 'Plant breeder/geneticist', }, @@ -10815,7 +10815,7 @@ export const peopleDemo = [ city: 'Ochoaberg', email: 'eric.johnson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/eric-johnson-be7867ea1f', jobTitle: 'Bookseller', }, @@ -10825,7 +10825,7 @@ export const peopleDemo = [ city: 'Jenniferburgh', email: 'judith.ortiz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/judith-ortiz-d3ecd8a548', jobTitle: 'Publishing rights manager', }, @@ -10835,7 +10835,7 @@ export const peopleDemo = [ city: 'Adamside', email: 'evan.floyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/evan-floyd-73bc237c29', jobTitle: 'Chief Operating Officer', }, @@ -10845,7 +10845,7 @@ export const peopleDemo = [ city: 'Cassandraview', email: 'deborah.myers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/deborah-myers-0014fb575f', jobTitle: 'Engineer, energy', }, @@ -10855,7 +10855,7 @@ export const peopleDemo = [ city: 'Zunigaside', email: 'jonathan.valdez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/jonathan-valdez-cca544261b', jobTitle: 'Manufacturing engineer', }, @@ -10865,7 +10865,7 @@ export const peopleDemo = [ city: 'Mcculloughborough', email: 'marie.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/marie-davis-50223e7489', jobTitle: 'Garment/textile technologist', }, @@ -10875,7 +10875,7 @@ export const peopleDemo = [ city: 'Jacobburgh', email: 'brent.mcpherson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/brent-mcpherson-365c1d18d0', jobTitle: 'Data scientist', }, @@ -10885,7 +10885,7 @@ export const peopleDemo = [ city: 'Smithborough', email: 'catherine.kim@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/catherine-kim-c3dc33b8f1', jobTitle: 'Manufacturing systems engineer', }, @@ -10895,7 +10895,7 @@ export const peopleDemo = [ city: 'Port Matthew', email: 'evan.hanson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/evan-hanson-93ee351985', jobTitle: 'Art therapist', }, @@ -10905,7 +10905,7 @@ export const peopleDemo = [ city: 'Fritzport', email: 'natalie.cooper@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/natalie-cooper-043bdf4ec3', jobTitle: 'Land', }, @@ -10915,7 +10915,7 @@ export const peopleDemo = [ city: 'Port Erin', email: 'jacqueline.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/jacqueline-martin-3a858a2278', jobTitle: 'Product manager', }, @@ -10925,7 +10925,7 @@ export const peopleDemo = [ city: 'South Angela', email: 'ashley.harrington@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/ashley-harrington-cdb4eef042', jobTitle: 'Immunologist', }, @@ -10935,7 +10935,7 @@ export const peopleDemo = [ city: 'West Reginald', email: 'heather.jones@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/heather-jones-6accdac7b2', jobTitle: 'Warehouse manager', }, @@ -10945,7 +10945,7 @@ export const peopleDemo = [ city: 'East Davidstad', email: 'corey.martin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/corey-martin-e69cbef278', jobTitle: 'Loss adjuster, chartered', }, @@ -10955,7 +10955,7 @@ export const peopleDemo = [ city: 'Jamesberg', email: 'christine.scott@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/christine-scott-001224382b', jobTitle: 'Historic buildings inspector/conservation officer', }, @@ -10965,7 +10965,7 @@ export const peopleDemo = [ city: 'Lake Brianville', email: 'alicia.ball@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/alicia-ball-3870ff3969', jobTitle: 'Presenter, broadcasting', }, @@ -10975,7 +10975,7 @@ export const peopleDemo = [ city: 'Port Diane', email: 'antonio.ferguson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/antonio-ferguson-995e342c4b', jobTitle: 'Production assistant, radio', }, @@ -10985,7 +10985,7 @@ export const peopleDemo = [ city: 'Kevinstad', email: 'joseph.baldwin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/joseph-baldwin-03fcc63126', jobTitle: 'Heritage manager', }, @@ -10995,7 +10995,7 @@ export const peopleDemo = [ city: 'Fostertown', email: 'devin.lopez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/devin-lopez-c9e1d4f2c8', jobTitle: 'Surveyor, insurance', }, @@ -11005,7 +11005,7 @@ export const peopleDemo = [ city: 'Gibsonstad', email: 'victoria.weber@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-1.png', linkedinUrl: '/in/victoria-weber-1d629bb105', jobTitle: 'Probation officer', }, @@ -11015,7 +11015,7 @@ export const peopleDemo = [ city: 'Olsontown', email: 'erica.lamb@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-2.png', linkedinUrl: '/in/erica-lamb-ccc3ee1ef0', jobTitle: 'Environmental health practitioner', }, @@ -11025,7 +11025,7 @@ export const peopleDemo = [ city: 'Justinville', email: 'regina.rivera@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', linkedinUrl: '/in/regina-rivera-80ffb29755', jobTitle: 'Psychologist, sport and exercise', }, @@ -11035,7 +11035,7 @@ export const peopleDemo = [ city: 'East Zoeview', email: 'sarah.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-4.png', linkedinUrl: '/in/sarah-hernandez-1e2803fbdb', jobTitle: 'Engineer, water', }, @@ -11045,7 +11045,7 @@ export const peopleDemo = [ city: 'Simonchester', email: 'jessica.graham@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-5.png', linkedinUrl: '/in/jessica-graham-c96b6e62d8', jobTitle: 'Water engineer', }, @@ -11055,7 +11055,7 @@ export const peopleDemo = [ city: 'Lake Scottville', email: 'john.ritter@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-6.png', linkedinUrl: '/in/john-ritter-eb8a8f542a', jobTitle: 'Dancer', }, @@ -11065,7 +11065,7 @@ export const peopleDemo = [ city: 'South Charles', email: 'jessica.bruce@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-7.png', linkedinUrl: '/in/jessica-bruce-a0631e1611', jobTitle: 'Community development worker', }, @@ -11075,7 +11075,7 @@ export const peopleDemo = [ city: 'Clintonberg', email: 'sara.larsen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-8.png', linkedinUrl: '/in/sara-larsen-cc08b21030', jobTitle: 'Computer games developer', }, @@ -11085,7 +11085,7 @@ export const peopleDemo = [ city: 'North Daniellestad', email: 'eric.ellison@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-9.png', linkedinUrl: '/in/eric-ellison-d200f0e1b9', jobTitle: 'Farm manager', }, @@ -11095,7 +11095,7 @@ export const peopleDemo = [ city: 'Alvaradomouth', email: 'katrina.butler@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-10.png', linkedinUrl: '/in/katrina-butler-3042d9be2a', jobTitle: 'Sales executive', }, @@ -11105,7 +11105,7 @@ export const peopleDemo = [ city: 'North Stefanieton', email: 'michelle.powers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-11.png', linkedinUrl: '/in/michelle-powers-1f5eda1b79', jobTitle: 'Textile designer', }, @@ -11115,7 +11115,7 @@ export const peopleDemo = [ city: 'North Nichole', email: 'jessica.baker@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-12.png', linkedinUrl: '/in/jessica-baker-eeaf05a650', jobTitle: 'Operational researcher', }, @@ -11125,7 +11125,7 @@ export const peopleDemo = [ city: 'Bernardmouth', email: 'cory.cooper@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-13.png', linkedinUrl: '/in/cory-cooper-7be856494d', jobTitle: 'Community development worker', }, @@ -11135,7 +11135,7 @@ export const peopleDemo = [ city: 'New Stephanie', email: 'brittany.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-14.png', linkedinUrl: '/in/brittany-williams-95380b6e0a', jobTitle: 'Engineer, control and instrumentation', }, @@ -11145,7 +11145,7 @@ export const peopleDemo = [ city: 'North Benjamin', email: 'jessica.hinton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-15.png', linkedinUrl: '/in/jessica-hinton-fa6b8fd2e2', jobTitle: 'Plant breeder/geneticist', }, @@ -11155,7 +11155,7 @@ export const peopleDemo = [ city: 'West Mark', email: 'natalie.ochoa@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-16.png', linkedinUrl: '/in/natalie-ochoa-b182dc5873', jobTitle: 'Museum/gallery conservator', }, @@ -11165,7 +11165,7 @@ export const peopleDemo = [ city: 'Clineview', email: 'kristine.warren@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-17.png', linkedinUrl: '/in/kristine-warren-a4821ef1b4', jobTitle: 'Senior tax professional/tax inspector', }, @@ -11175,7 +11175,7 @@ export const peopleDemo = [ city: 'Cynthiaburgh', email: 'lindsey.dalton@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-18.png', linkedinUrl: '/in/lindsey-dalton-70af5be384', jobTitle: 'Child psychotherapist', }, @@ -11185,7 +11185,7 @@ export const peopleDemo = [ city: 'Thomasmouth', email: 'jennifer.morgan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-19.png', linkedinUrl: '/in/jennifer-morgan-ab5acc70fc', jobTitle: 'Quantity surveyor', }, @@ -11195,7 +11195,7 @@ export const peopleDemo = [ city: 'West Andrewhaven', email: 'bryan.harris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-20.png', linkedinUrl: '/in/bryan-harris-f858052f2c', jobTitle: 'Civil engineer, consulting', }, @@ -11205,7 +11205,7 @@ export const peopleDemo = [ city: 'Elizabethmouth', email: 'norma.adkins@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-21.png', linkedinUrl: '/in/norma-adkins-6297907f60', jobTitle: 'Forest/woodland manager', }, @@ -11215,7 +11215,7 @@ export const peopleDemo = [ city: 'Lake Annview', email: 'kara.perry@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-22.png', linkedinUrl: '/in/kara-perry-3dddc5ee1c', jobTitle: 'Waste management officer', }, @@ -11225,7 +11225,7 @@ export const peopleDemo = [ city: 'South Cynthiaberg', email: 'nicole.kelly@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-23.png', linkedinUrl: '/in/nicole-kelly-793e9bc70f', jobTitle: 'Therapist, music', }, @@ -11235,7 +11235,7 @@ export const peopleDemo = [ city: 'Brucetown', email: 'annette.long@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-24.png', linkedinUrl: '/in/annette-long-45da6c37e7', jobTitle: 'Community arts worker', }, @@ -11245,7 +11245,7 @@ export const peopleDemo = [ city: 'Deniseport', email: 'john.stewart@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-25.png', linkedinUrl: '/in/john-stewart-50fd4b2b37', jobTitle: 'Land/geomatics surveyor', }, @@ -11255,7 +11255,7 @@ export const peopleDemo = [ city: 'South Mark', email: 'gregory.larson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-26.png', linkedinUrl: '/in/gregory-larson-4f52726447', jobTitle: 'Translator', }, @@ -11265,7 +11265,7 @@ export const peopleDemo = [ city: 'North Rhonda', email: 'wanda.herrera@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-27.png', linkedinUrl: '/in/wanda-herrera-10f674edf1', jobTitle: 'Therapeutic radiographer', }, @@ -11275,7 +11275,7 @@ export const peopleDemo = [ city: 'Mcconnellland', email: 'sarah.davis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-28.png', linkedinUrl: '/in/sarah-davis-c63949c87f', jobTitle: 'Architect', }, @@ -11285,7 +11285,7 @@ export const peopleDemo = [ city: 'East William', email: 'devin.snow@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-29.png', linkedinUrl: '/in/devin-snow-b8a26f7352', jobTitle: 'Social researcher', }, @@ -11295,7 +11295,7 @@ export const peopleDemo = [ city: 'Nicholeside', email: 'gina.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-30.png', linkedinUrl: '/in/gina-hernandez-40ea9940fd', jobTitle: 'Advice worker', }, @@ -11305,7 +11305,7 @@ export const peopleDemo = [ city: 'West Nicholas', email: 'ronnie.watson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-31.png', linkedinUrl: '/in/ronnie-watson-5e8ffd4706', jobTitle: 'Press sub', }, @@ -11315,7 +11315,7 @@ export const peopleDemo = [ city: 'Davisside', email: 'mark.duran@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-32.png', linkedinUrl: '/in/mark-duran-0a02b4a8ee', jobTitle: 'Operational researcher', }, @@ -11325,7 +11325,7 @@ export const peopleDemo = [ city: 'Candiceborough', email: 'shawn.wolfe@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-33.png', linkedinUrl: '/in/shawn-wolfe-3b9f538b13', jobTitle: 'Research officer, government', }, @@ -11335,7 +11335,7 @@ export const peopleDemo = [ city: 'Sanchezville', email: 'mark.welch@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-34.png', linkedinUrl: '/in/mark-welch-3a104608c4', jobTitle: 'Chief Marketing Officer', }, @@ -11345,7 +11345,7 @@ export const peopleDemo = [ city: 'Josephbury', email: 'james.holland@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-35.png', linkedinUrl: '/in/james-holland-a99089ebf7', jobTitle: 'Engineer, building services', }, @@ -11355,7 +11355,7 @@ export const peopleDemo = [ city: 'Lake Denisebury', email: 'jennifer.weber@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-36.png', linkedinUrl: '/in/jennifer-weber-384a97d0de', jobTitle: 'Engineer, maintenance', }, @@ -11365,7 +11365,7 @@ export const peopleDemo = [ city: 'Port April', email: 'beth.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-37.png', linkedinUrl: '/in/beth-hernandez-7708149061', jobTitle: 'Runner, broadcasting/film/video', }, @@ -11375,7 +11375,7 @@ export const peopleDemo = [ city: 'Port Tina', email: 'eric.barnes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-38.png', linkedinUrl: '/in/eric-barnes-332ab94dce', jobTitle: 'Patent attorney', }, @@ -11385,7 +11385,7 @@ export const peopleDemo = [ city: 'Mezaborough', email: 'ryan.richardson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-39.png', linkedinUrl: '/in/ryan-richardson-36096d0c1f', jobTitle: 'Cartographer', }, @@ -11395,7 +11395,7 @@ export const peopleDemo = [ city: 'Lake Shawn', email: 'brandy.cowan@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', linkedinUrl: '/in/brandy-cowan-bab6874e38', jobTitle: 'Patent attorney', }, @@ -11405,7 +11405,7 @@ export const peopleDemo = [ city: 'East Sherylstad', email: 'zachary.jensen@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-41.png', linkedinUrl: '/in/zachary-jensen-ad43305058', jobTitle: 'Nurse, adult', }, @@ -11415,7 +11415,7 @@ export const peopleDemo = [ city: 'West Elizabeth', email: 'carrie.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-42.png', linkedinUrl: '/in/carrie-taylor-c50c9da449', jobTitle: 'Designer, furniture', }, @@ -11425,7 +11425,7 @@ export const peopleDemo = [ city: 'Ronaldfort', email: 'timothy.williams@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-43.png', linkedinUrl: '/in/timothy-williams-0bd12cc799', jobTitle: 'Editor, magazine features', }, @@ -11435,7 +11435,7 @@ export const peopleDemo = [ city: 'South Michaelfurt', email: 'peter.rodgers@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-44.png', linkedinUrl: '/in/peter-rodgers-2b96b2d840', jobTitle: 'Scientific laboratory technician', }, @@ -11445,7 +11445,7 @@ export const peopleDemo = [ city: 'West Christopherview', email: 'julie.taylor@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-45.png', linkedinUrl: '/in/julie-taylor-fa514d063e', jobTitle: 'Careers information officer', }, @@ -11455,7 +11455,7 @@ export const peopleDemo = [ city: 'Amberton', email: 'samuel.ortiz@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-46.png', linkedinUrl: '/in/samuel-ortiz-fd07e1761a', jobTitle: 'Insurance broker', }, @@ -11465,7 +11465,7 @@ export const peopleDemo = [ city: 'West Elizabethfurt', email: 'kevin.lucas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', linkedinUrl: '/in/kevin-lucas-3789d78a4d', jobTitle: 'Therapist, occupational', }, @@ -11475,7 +11475,7 @@ export const peopleDemo = [ city: 'Jonathanhaven', email: 'alexis.hernandez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-48.png', linkedinUrl: '/in/alexis-hernandez-9b63f9db08', jobTitle: 'Journalist, magazine', }, @@ -11485,7 +11485,7 @@ export const peopleDemo = [ city: 'Marymouth', email: 'sophia.wood@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-49.png', linkedinUrl: '/in/sophia-wood-e6ff6bda50', jobTitle: 'Database administrator', }, @@ -11495,7 +11495,7 @@ export const peopleDemo = [ city: 'Jeffreyview', email: 'lori.hunt@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-50.png', linkedinUrl: '/in/lori-hunt-242e73a5d1', jobTitle: 'Financial planner', }, @@ -11505,7 +11505,7 @@ export const peopleDemo = [ city: 'West Alicia', email: 'dennis.stark@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-51.png', linkedinUrl: '/in/dennis-stark-aac95d0674', jobTitle: 'Quality manager', }, @@ -11515,7 +11515,7 @@ export const peopleDemo = [ city: 'Port Juliamouth', email: 'robert.smith@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-52.png', linkedinUrl: '/in/robert-smith-31ba372c60', jobTitle: 'Estate manager/land agent', }, @@ -11525,7 +11525,7 @@ export const peopleDemo = [ city: 'South Linda', email: 'megan.hughes@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-53.png', linkedinUrl: '/in/megan-hughes-d7607985f9', jobTitle: 'Materials engineer', }, @@ -11535,7 +11535,7 @@ export const peopleDemo = [ city: 'Jamesberg', email: 'kristine.osborne@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-54.png', linkedinUrl: '/in/kristine-osborne-bb389c0df3', jobTitle: 'Dietitian', }, @@ -11545,7 +11545,7 @@ export const peopleDemo = [ city: 'Lake Marcus', email: 'brandy.thomas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-55.png', linkedinUrl: '/in/brandy-thomas-9d8f298d17', jobTitle: 'Copy', }, @@ -11555,7 +11555,7 @@ export const peopleDemo = [ city: 'Jorgeton', email: 'brad.long@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-56.png', linkedinUrl: '/in/brad-long-f8d735beb2', jobTitle: 'Personal assistant', }, @@ -11565,7 +11565,7 @@ export const peopleDemo = [ city: 'Davisstad', email: 'caleb.stevens@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-57.png', linkedinUrl: '/in/caleb-stevens-f00c3e5dd3', jobTitle: 'Multimedia specialist', }, @@ -11575,7 +11575,7 @@ export const peopleDemo = [ city: 'Loweryland', email: 'matthew.wall@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-58.png', linkedinUrl: '/in/matthew-wall-90d1a29c8e', jobTitle: 'Chartered legal executive (England and Wales)', }, @@ -11585,7 +11585,7 @@ export const peopleDemo = [ city: 'South Lisa', email: 'cynthia.cook@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-59.png', linkedinUrl: '/in/cynthia-cook-25e7c0ba3c', jobTitle: 'Radiographer, diagnostic', }, @@ -11595,7 +11595,7 @@ export const peopleDemo = [ city: 'North Jasminebury', email: 'lisa.tate@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-60.png', linkedinUrl: '/in/lisa-tate-9456c0ae0b', jobTitle: 'Lecturer, higher education', }, @@ -11605,7 +11605,7 @@ export const peopleDemo = [ city: 'South Jeffrey', email: 'gloria.chapman@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-61.png', linkedinUrl: '/in/gloria-chapman-c537bef76d', jobTitle: 'Building services engineer', }, @@ -11615,7 +11615,7 @@ export const peopleDemo = [ city: 'Victoriaport', email: 'connie.lewis@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-62.png', linkedinUrl: '/in/connie-lewis-c6b975976f', jobTitle: 'Insurance risk surveyor', }, @@ -11625,7 +11625,7 @@ export const peopleDemo = [ city: 'Dylanberg', email: 'gary.harris@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-63.png', linkedinUrl: '/in/gary-harris-619dfa4ebb', jobTitle: 'Research officer, trade union', }, @@ -11635,7 +11635,7 @@ export const peopleDemo = [ city: 'Lake Daniel', email: 'sharon.berger@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-64.png', linkedinUrl: '/in/sharon-berger-2da41916fd', jobTitle: 'Therapeutic radiographer', }, @@ -11645,7 +11645,7 @@ export const peopleDemo = [ city: 'Steveborough', email: 'michael.russo@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-65.png', linkedinUrl: '/in/michael-russo-a3ba403e9b', jobTitle: 'Production engineer', }, @@ -11655,7 +11655,7 @@ export const peopleDemo = [ city: 'New Randall', email: 'michael.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-66.png', linkedinUrl: '/in/michael-young-800f26276e', jobTitle: 'Charity fundraiser', }, @@ -11665,7 +11665,7 @@ export const peopleDemo = [ city: 'Michelleberg', email: 'devin.ramsey@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-67.png', linkedinUrl: '/in/devin-ramsey-428f24b810', jobTitle: 'Educational psychologist', }, @@ -11675,7 +11675,7 @@ export const peopleDemo = [ city: 'Keithville', email: 'sara.lee@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', linkedinUrl: '/in/sara-lee-becddd74f2', jobTitle: 'Equality and diversity officer', }, @@ -11685,7 +11685,7 @@ export const peopleDemo = [ city: 'Port Stephanie', email: 'robin.stark@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-69.png', linkedinUrl: '/in/robin-stark-868b21526e', jobTitle: 'Production engineer', }, @@ -11695,7 +11695,7 @@ export const peopleDemo = [ city: 'Samanthafort', email: 'sergio.burns@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-70.png', linkedinUrl: '/in/sergio-burns-994728d1d7', jobTitle: 'Fitness centre manager', }, @@ -11705,7 +11705,7 @@ export const peopleDemo = [ city: 'Combsfurt', email: 'lisa.haas@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-71.png', linkedinUrl: '/in/lisa-haas-e3c3871c8d', jobTitle: 'Oceanographer', }, @@ -11715,7 +11715,7 @@ export const peopleDemo = [ city: 'East Amandaville', email: 'courtney.donaldson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-72.png', linkedinUrl: '/in/courtney-donaldson-dd063d66c2', jobTitle: 'Sports therapist', }, @@ -11725,7 +11725,7 @@ export const peopleDemo = [ city: 'Jeffreyview', email: 'ashley.conrad@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-73.png', linkedinUrl: '/in/ashley-conrad-96b6f83928', jobTitle: 'Manufacturing engineer', }, @@ -11735,7 +11735,7 @@ export const peopleDemo = [ city: 'Bentonland', email: 'tim.levine@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-74.png', linkedinUrl: '/in/tim-levine-01557d92c6', jobTitle: 'Animal nutritionist', }, @@ -11745,7 +11745,7 @@ export const peopleDemo = [ city: 'Port Erinburgh', email: 'michelle.martinez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-75.png', linkedinUrl: '/in/michelle-martinez-905bf6439c', jobTitle: 'TEFL teacher', }, @@ -11755,7 +11755,7 @@ export const peopleDemo = [ city: 'New Jean', email: 'jennifer.rose@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-76.png', linkedinUrl: '/in/jennifer-rose-162f9c1d7b', jobTitle: 'Logistics and distribution manager', }, @@ -11765,7 +11765,7 @@ export const peopleDemo = [ city: 'Martinezmouth', email: 'casey.greer@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-77.png', linkedinUrl: '/in/casey-greer-e3d7510c16', jobTitle: 'Editor, commissioning', }, @@ -11775,7 +11775,7 @@ export const peopleDemo = [ city: 'South Sandra', email: 'crystal.mclaughlin@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-78.png', linkedinUrl: '/in/crystal-mclaughlin-7b67938d55', jobTitle: 'Chiropodist', }, @@ -11785,7 +11785,7 @@ export const peopleDemo = [ city: 'North Joshua', email: 'rachel.floyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-79.png', linkedinUrl: '/in/rachel-floyd-554e320b10', jobTitle: 'Therapist, drama', }, @@ -11795,7 +11795,7 @@ export const peopleDemo = [ city: 'South Joy', email: 'shannon.anderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-80.png', linkedinUrl: '/in/shannon-anderson-7e555790c5', jobTitle: 'Mudlogger', }, @@ -11805,7 +11805,7 @@ export const peopleDemo = [ city: 'Webbstad', email: 'catherine.white@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-81.png', linkedinUrl: '/in/catherine-white-759d1e61da', jobTitle: 'Barista', }, @@ -11815,7 +11815,7 @@ export const peopleDemo = [ city: 'Port Davidton', email: 'matthew.fisher@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-82.png', linkedinUrl: '/in/matthew-fisher-4f49bc00ee', jobTitle: 'Systems developer', }, @@ -11825,7 +11825,7 @@ export const peopleDemo = [ city: 'Woodsborough', email: 'tracy.leonard@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-83.png', linkedinUrl: '/in/tracy-leonard-b9b7beae66', jobTitle: 'Psychologist, sport and exercise', }, @@ -11835,7 +11835,7 @@ export const peopleDemo = [ city: 'South Michaelville', email: 'jenna.moore@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-84.png', linkedinUrl: '/in/jenna-moore-c17c91ef24', jobTitle: 'Senior tax professional/tax inspector', }, @@ -11845,7 +11845,7 @@ export const peopleDemo = [ city: 'West Edwardchester', email: 'alice.edwards@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-85.png', linkedinUrl: '/in/alice-edwards-aefc82ee0c', jobTitle: 'Engineer, water', }, @@ -11855,7 +11855,7 @@ export const peopleDemo = [ city: 'Reynoldsview', email: 'theresa.orozco@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-86.png', linkedinUrl: '/in/theresa-orozco-4a2ec9a601', jobTitle: 'Journalist, magazine', }, @@ -11865,7 +11865,7 @@ export const peopleDemo = [ city: 'Richardberg', email: 'samantha.hicks@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-87.png', linkedinUrl: '/in/samantha-hicks-16b4d5470f', jobTitle: 'Herpetologist', }, @@ -11875,7 +11875,7 @@ export const peopleDemo = [ city: 'Lauriemouth', email: 'brian.finley@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-88.png', linkedinUrl: '/in/brian-finley-aa52351d68', jobTitle: 'Youth worker', }, @@ -11885,7 +11885,7 @@ export const peopleDemo = [ city: 'Juliebury', email: 'kevin.black@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', linkedinUrl: '/in/kevin-black-ada9f2fada', jobTitle: 'Teacher, adult education', }, @@ -11895,7 +11895,7 @@ export const peopleDemo = [ city: 'Kevinborough', email: 'eric.peterson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-90.png', linkedinUrl: '/in/eric-peterson-2e15a4a39c', jobTitle: 'Ranger/warden', }, @@ -11905,7 +11905,7 @@ export const peopleDemo = [ city: 'New Ronaldview', email: 'samantha.steele@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-91.png', linkedinUrl: '/in/samantha-steele-a6e15143ce', jobTitle: 'Designer, industrial/product', }, @@ -11915,7 +11915,7 @@ export const peopleDemo = [ city: 'West Brandonville', email: 'tiffany.boyd@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-92.png', linkedinUrl: '/in/tiffany-boyd-5875da90d3', jobTitle: 'Editor, film/video', }, @@ -11925,7 +11925,7 @@ export const peopleDemo = [ city: 'Rochafurt', email: 'larry.johnston@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-93.png', linkedinUrl: '/in/larry-johnston-2639a97c12', jobTitle: 'Horticulturist, amenity', }, @@ -11935,7 +11935,7 @@ export const peopleDemo = [ city: 'Lake Rhonda', email: 'vanessa.villanueva@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-94.png', linkedinUrl: '/in/vanessa-villanueva-cf67adba5d', jobTitle: 'Child psychotherapist', }, @@ -11945,7 +11945,7 @@ export const peopleDemo = [ city: 'East Christophermouth', email: 'danielle.gutierrez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-95.png', linkedinUrl: '/in/danielle-gutierrez-bae65dfff7', jobTitle: 'Technical author', }, @@ -11955,7 +11955,7 @@ export const peopleDemo = [ city: 'Cookland', email: 'pamela.anderson@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-96.png', linkedinUrl: '/in/pamela-anderson-79a31bf795', jobTitle: 'Geoscientist', }, @@ -11965,7 +11965,7 @@ export const peopleDemo = [ city: 'Fryeville', email: 'linda.young@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-97.png', linkedinUrl: '/in/linda-young-912fcbd8df', jobTitle: 'Ship broker', }, @@ -11975,7 +11975,7 @@ export const peopleDemo = [ city: 'Davidmouth', email: 'rodney.orr@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-98.png', linkedinUrl: '/in/rodney-orr-ae717c2f34', jobTitle: 'Best boy', }, @@ -11985,7 +11985,7 @@ export const peopleDemo = [ city: 'Laurenfurt', email: 'ashley.perez@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-99.png', linkedinUrl: '/in/ashley-perez-9375a8f7c7', jobTitle: 'Chemical engineer', }, @@ -11995,7 +11995,7 @@ export const peopleDemo = [ city: 'Travisfurt', email: 'abigail.scott@example.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-100.png', linkedinUrl: '/in/abigail-scott-34179b2995', jobTitle: 'Radiographer, diagnostic', }, diff --git a/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/company.ts b/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/company.ts index 9899ffd1ca44..6061cef071a6 100644 --- a/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/company.ts +++ b/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/company.ts @@ -10,7 +10,12 @@ export const companyPrefillData = async ( .into(`${schemaName}.company`, [ 'name', 'domainName', - 'address', + 'addressAddressStreet1', + 'addressAddressStreet2', + 'addressAddressCity', + 'addressAddressState', + 'addressAddressPostcode', + 'addressAddressCountry', 'employees', 'position', ]) @@ -19,35 +24,60 @@ export const companyPrefillData = async ( { name: 'Airbnb', domainName: 'airbnb.com', - address: 'San Francisco', + addressAddressStreet1: '888 Brannan St', + addressAddressStreet2: null, + addressAddressCity: 'San Francisco', + addressAddressState: 'CA', + addressAddressPostcode: '94103', + addressAddressCountry: 'United States', employees: 5000, position: 1, }, { name: 'Qonto', domainName: 'qonto.com', - address: 'San Francisco', + addressAddressStreet1: '18 rue de navarrin', + addressAddressStreet2: null, + addressAddressCity: 'Paris', + addressAddressState: null, + addressAddressPostcode: '75009', + addressAddressCountry: 'France', employees: 800, position: 2, }, { name: 'Stripe', domainName: 'stripe.com', - address: 'San Francisco', + addressAddressStreet1: 'Eutaw Street', + addressAddressStreet2: null, + addressAddressCity: 'Dublin', + addressAddressState: null, + addressAddressPostcode: null, + addressAddressCountry: 'Ireland', employees: 8000, position: 3, }, { name: 'Figma', domainName: 'figma.com', - address: 'San Francisco', + addressAddressStreet1: '760 Market St', + addressAddressStreet2: 'Floor 10', + addressAddressCity: 'San Francisco', + addressAddressState: null, + addressAddressPostcode: '94102', + addressAddressCountry: 'United States', employees: 800, position: 4, }, { name: 'Notion', domainName: 'notion.com', - address: 'San Francisco', + addressAddressStreet1: '2300 Harrison St', + addressAddressStreet2: null, + addressAddressCity: 'San Francisco', + addressAddressState: 'CA', + addressAddressPostcode: '94110', + addressAddressCountry: 'United States', employees: 400, position: 5, }, diff --git a/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/person.ts b/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/person.ts index 68d82ebb1eb9..5c4a0212bff0 100644 --- a/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/person.ts +++ b/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/person.ts @@ -23,7 +23,7 @@ export const personPrefillData = async ( city: 'San Francisco', email: 'chesky@airbnb.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-3.png', position: 1, }, { @@ -32,7 +32,7 @@ export const personPrefillData = async ( city: 'Paris', email: 'prot@qonto.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-89.png', position: 2, }, { @@ -41,7 +41,7 @@ export const personPrefillData = async ( city: 'San Francisco', email: 'collison@stripe.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-47.png', position: 3, }, { @@ -50,7 +50,7 @@ export const personPrefillData = async ( city: 'San Francisco', email: 'field@figma.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-40.png', position: 4, }, { @@ -59,7 +59,7 @@ export const personPrefillData = async ( city: 'San Francisco', email: 'zhao@notion.com', avatarUrl: - '', + 'https://twentyhq.github.io/placeholder-images/people/image-68.png', position: 5, }, ]) diff --git a/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/view-opportunity-fields.ts b/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/view-opportunity-fields.ts index 7e7ed6b59354..baa08a95f7ea 100644 --- a/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/view-opportunity-fields.ts +++ b/packages/twenty-server/src/engine/workspace-manager/standard-objects-prefill-data/view-opportunity-fields.ts @@ -47,16 +47,6 @@ export const viewOpportunityFields = ( isVisible: true, size: 150, }, - { - fieldMetadataId: - objectMetadataMap[STANDARD_OBJECT_IDS.opportunity].fields[ - OPPORTUNITY_STANDARD_FIELD_IDS.probability - ], - viewId: viewId, - position: 4, - isVisible: true, - size: 150, - }, { fieldMetadataId: objectMetadataMap[STANDARD_OBJECT_IDS.opportunity].fields[ diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/clean-inactive-workspaces.command.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/clean-inactive-workspaces.command.ts index 822fd2e94fd1..79b805c607cf 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/clean-inactive-workspaces.command.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/clean-inactive-workspaces.command.ts @@ -1,7 +1,6 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner, Option } from 'nest-commander'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { CleanInactiveWorkspaceJob } from 'src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.job'; @@ -16,7 +15,7 @@ export type CleanInactiveWorkspacesCommandOptions = { }) export class CleanInactiveWorkspacesCommand extends CommandRunner { constructor( - @Inject(MessageQueue.taskAssignedQueue) + @InjectMessageQueue(MessageQueue.cronQueue) private readonly messageQueueService: MessageQueueService, ) { super(); diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/delete-incomplete-workspaces.command.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/delete-workspaces.command.ts similarity index 56% rename from packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/delete-incomplete-workspaces.command.ts rename to packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/delete-workspaces.command.ts index 30c980c4e0c1..2bbc17b75878 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/delete-incomplete-workspaces.command.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/delete-workspaces.command.ts @@ -2,26 +2,29 @@ import { InjectRepository } from '@nestjs/typeorm'; import { Logger } from '@nestjs/common'; import { Command, CommandRunner, Option } from 'nest-commander'; -import { FindOptionsWhere, In, Repository } from 'typeorm'; +import { In, Repository } from 'typeorm'; -import { WorkspaceService } from 'src/engine/core-modules/workspace/services/workspace.service'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; -import { getDryRunLogHeader } from 'src/utils/get-dry-run-log-header'; import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; +import { getDryRunLogHeader } from 'src/utils/get-dry-run-log-header'; +import { WorkspaceService } from 'src/engine/core-modules/workspace/services/workspace.service'; +import { LoadServiceWithWorkspaceContext } from 'src/engine/twenty-orm/context/load-service-with-workspace.context'; -type DeleteIncompleteWorkspacesCommandOptions = { +type DeleteWorkspacesCommandOptions = { dryRun?: boolean; - workspaceIds?: string[]; + workspaceIds: string[]; }; @Command({ - name: 'workspace:delete-incomplete', - description: 'Delete incomplete workspaces', + name: 'workspace:delete', + description: 'Delete workspace', }) -export class DeleteIncompleteWorkspacesCommand extends CommandRunner { - private readonly logger = new Logger(DeleteIncompleteWorkspacesCommand.name); +export class DeleteWorkspacesCommand extends CommandRunner { + private readonly logger = new Logger(DeleteWorkspacesCommand.name); + constructor( private readonly workspaceService: WorkspaceService, + private readonly loadServiceWithWorkspaceContext: LoadServiceWithWorkspaceContext, @InjectRepository(Workspace, 'core') private readonly workspaceRepository: Repository<Workspace>, private readonly dataSourceService: DataSourceService, @@ -41,7 +44,7 @@ export class DeleteIncompleteWorkspacesCommand extends CommandRunner { @Option({ flags: '-w, --workspace-ids [workspace_ids]', description: 'comma separated workspace ids', - required: false, + required: true, }) parseWorkspaceIds(value: string): string[] { return value.split(','); @@ -49,41 +52,43 @@ export class DeleteIncompleteWorkspacesCommand extends CommandRunner { async run( _passedParam: string[], - options: DeleteIncompleteWorkspacesCommandOptions, + options: DeleteWorkspacesCommandOptions, ): Promise<void> { - const where: FindOptionsWhere<Workspace> = { - subscriptionStatus: 'incomplete', - }; + const workspaces = await this.workspaceRepository.find({ + where: { id: In(options.workspaceIds) }, + }); - if (options.workspaceIds) { - where.id = In(options.workspaceIds); - } - - const incompleteWorkspaces = await this.workspaceRepository.findBy(where); const dataSources = await this.dataSourceService.getManyDataSourceMetadata(); + const workspaceIdsWithSchema = dataSources.map( (dataSource) => dataSource.workspaceId, ); - const incompleteWorkspacesToDelete = incompleteWorkspaces.filter( - (incompleteWorkspace) => - workspaceIdsWithSchema.includes(incompleteWorkspace.id), + + const workspacesToDelete = workspaces.filter((Workspace) => + workspaceIdsWithSchema.includes(Workspace.id), ); - if (incompleteWorkspacesToDelete.length) { + if (workspacesToDelete.length) { this.logger.log( - `Running Deleting incomplete workspaces on ${incompleteWorkspacesToDelete.length} workspaces`, + `Running Deleting workspaces on ${workspacesToDelete.length} workspaces`, ); } - for (const incompleteWorkspace of incompleteWorkspacesToDelete) { + for (const workspace of workspacesToDelete) { this.logger.log( `${getDryRunLogHeader(options.dryRun)}Deleting workspace ${ - incompleteWorkspace.id - } name: '${incompleteWorkspace.displayName}'`, + workspace.id + } name: '${workspace.displayName}'`, ); + const workspaceServiceInstance = + await this.loadServiceWithWorkspaceContext.load( + this.workspaceService, + workspace.id, + ); + if (!options.dryRun) { - await this.workspaceService.softDeleteWorkspace(incompleteWorkspace.id); + await workspaceServiceInstance.softDeleteWorkspace(workspace.id); } } } diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/start-clean-inactive-workspaces.cron.command.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/start-clean-inactive-workspaces.cron.command.ts index 62b7f0fa0e4d..daee64136210 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/start-clean-inactive-workspaces.cron.command.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/start-clean-inactive-workspaces.cron.command.ts @@ -1,7 +1,6 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner } from 'nest-commander'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { cleanInactiveWorkspaceCronPattern } from 'src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.cron.pattern'; @@ -13,7 +12,7 @@ import { CleanInactiveWorkspaceJob } from 'src/engine/workspace-manager/workspac }) export class StartCleanInactiveWorkspacesCronCommand extends CommandRunner { constructor( - @Inject(MessageQueue.cronQueue) + @InjectMessageQueue(MessageQueue.cronQueue) private readonly messageQueueService: MessageQueueService, ) { super(); diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/stop-clean-inactive-workspaces.cron.command.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/stop-clean-inactive-workspaces.cron.command.ts index 1dd6dd85e650..0fa47c6fb59c 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/stop-clean-inactive-workspaces.cron.command.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/commands/stop-clean-inactive-workspaces.cron.command.ts @@ -1,7 +1,6 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner } from 'nest-commander'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { cleanInactiveWorkspaceCronPattern } from 'src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.cron.pattern'; @@ -13,7 +12,7 @@ import { CleanInactiveWorkspaceJob } from 'src/engine/workspace-manager/workspac }) export class StopCleanInactiveWorkspacesCronCommand extends CommandRunner { constructor( - @Inject(MessageQueue.cronQueue) + @InjectMessageQueue(MessageQueue.cronQueue) private readonly messageQueueService: MessageQueueService, ) { super(); diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.job.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.job.ts index 7d74648dccf9..95f9fc9a4f17 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.job.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/crons/clean-inactive-workspace.job.ts @@ -1,4 +1,4 @@ -import { Injectable, Logger } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; import { render } from '@react-email/render'; import { In } from 'typeorm'; @@ -7,8 +7,6 @@ import { DeleteInactiveWorkspaceEmail, } from 'twenty-emails'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { ObjectMetadataService } from 'src/engine/metadata-modules/object-metadata/object-metadata.service'; import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; import { TypeORMService } from 'src/database/typeorm/typeorm.service'; @@ -20,6 +18,9 @@ import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadat import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; import { CleanInactiveWorkspacesCommandOptions } from 'src/engine/workspace-manager/workspace-cleaner/commands/clean-inactive-workspaces.command'; import { getDryRunLogHeader } from 'src/utils/get-dry-run-log-header'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; const MILLISECONDS_IN_ONE_DAY = 1000 * 3600 * 24; @@ -28,10 +29,8 @@ type WorkspaceToDeleteData = { daysSinceInactive: number; }; -@Injectable() -export class CleanInactiveWorkspaceJob - implements MessageQueueJob<CleanInactiveWorkspacesCommandOptions> -{ +@Processor(MessageQueue.cronQueue) +export class CleanInactiveWorkspaceJob { private readonly logger = new Logger(CleanInactiveWorkspaceJob.name); private readonly inactiveDaysBeforeDelete; private readonly inactiveDaysBeforeEmail; @@ -193,6 +192,7 @@ export class CleanInactiveWorkspaceJob }); } + @Process(CleanInactiveWorkspaceJob.name) async handle(data: CleanInactiveWorkspacesCommandOptions): Promise<void> { const isDryRun = data.dryRun || false; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/workspace-cleaner.module.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/workspace-cleaner.module.ts index 6bf7eb0136ad..681755be16a0 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/workspace-cleaner.module.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-cleaner/workspace-cleaner.module.ts @@ -2,12 +2,12 @@ import { Module } from '@nestjs/common'; import { TypeOrmModule } from '@nestjs/typeorm'; import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; -import { DeleteIncompleteWorkspacesCommand } from 'src/engine/workspace-manager/workspace-cleaner/commands/delete-incomplete-workspaces.command'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { CleanInactiveWorkspacesCommand } from 'src/engine/workspace-manager/workspace-cleaner/commands/clean-inactive-workspaces.command'; import { StartCleanInactiveWorkspacesCronCommand } from 'src/engine/workspace-manager/workspace-cleaner/commands/start-clean-inactive-workspaces.cron.command'; import { StopCleanInactiveWorkspacesCronCommand } from 'src/engine/workspace-manager/workspace-cleaner/commands/stop-clean-inactive-workspaces.cron.command'; import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; +import { DeleteWorkspacesCommand } from 'src/engine/workspace-manager/workspace-cleaner/commands/delete-workspaces.command'; @Module({ imports: [ @@ -16,7 +16,7 @@ import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-s DataSourceModule, ], providers: [ - DeleteIncompleteWorkspacesCommand, + DeleteWorkspacesCommand, CleanInactiveWorkspacesCommand, StartCleanInactiveWorkspacesCronCommand, StopCleanInactiveWorkspacesCronCommand, diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-manager.module.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-manager.module.ts index a3afeb00cc42..53bba06ce186 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-manager.module.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-manager.module.ts @@ -4,21 +4,30 @@ import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-s import { ObjectMetadataModule } from 'src/engine/metadata-modules/object-metadata/object-metadata.module'; import { WorkspaceMigrationModule } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; import { WorkspaceHealthModule } from 'src/engine/workspace-manager/workspace-health/workspace-health.module'; +import { WorkspaceStatusModule } from 'src/engine/workspace-manager/workspace-status/workspace-manager.module'; +import { + WorkspaceSyncMetadataModule, +} from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; import { WorkspaceManagerService } from './workspace-manager.service'; +import { FieldMetadataModule } from 'src/engine/metadata-modules/field-metadata/field-metadata.module'; +import { RelationMetadataModule } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.module'; @Module({ imports: [ WorkspaceDataSourceModule, WorkspaceMigrationModule, ObjectMetadataModule, + FieldMetadataModule, + RelationMetadataModule, DataSourceModule, WorkspaceSyncMetadataModule, WorkspaceHealthModule, + WorkspaceStatusModule, ], exports: [WorkspaceManagerService], providers: [WorkspaceManagerService], }) -export class WorkspaceManagerModule {} +export class WorkspaceManagerModule { +} diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-manager.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-manager.service.ts index c7d75b6440e8..f49be7866353 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-manager.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-manager.service.ts @@ -1,13 +1,23 @@ import { Injectable } from '@nestjs/common'; +import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; import { ObjectMetadataService } from 'src/engine/metadata-modules/object-metadata/object-metadata.service'; import { WorkspaceMigrationService } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.service'; -import { standardObjectsPrefillData } from 'src/engine/workspace-manager/standard-objects-prefill-data/standard-objects-prefill-data'; -import { demoObjectsPrefillData } from 'src/engine/workspace-manager/demo-objects-prefill-data/demo-objects-prefill-data'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; -import { WorkspaceSyncMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service'; +import { + demoObjectsPrefillData, +} from 'src/engine/workspace-manager/demo-objects-prefill-data/demo-objects-prefill-data'; +import { + standardObjectsPrefillData, +} from 'src/engine/workspace-manager/standard-objects-prefill-data/standard-objects-prefill-data'; +import { + WorkspaceSyncMetadataService, +} from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service'; +import * as process from 'node:process'; +import { prefillWorkspaceWithFunnelminkFSMObjects } from 'src/funnelmink/funnelmink-objects-prefill-data'; +import { FieldMetadataService } from 'src/engine/metadata-modules/field-metadata/field-metadata.service'; +import { RelationMetadataService } from '../metadata-modules/relation-metadata/relation-metadata.service'; @Injectable() export class WorkspaceManagerService { @@ -17,7 +27,10 @@ export class WorkspaceManagerService { private readonly objectMetadataService: ObjectMetadataService, private readonly dataSourceService: DataSourceService, private readonly workspaceSyncMetadataService: WorkspaceSyncMetadataService, - ) {} + private readonly fieldMetadataService: FieldMetadataService, + private readonly relationMetadataService: RelationMetadataService, + ) { + } /** * Init a workspace by creating a new data source and running all migrations @@ -47,6 +60,17 @@ export class WorkspaceManagerService { dataSourceMetadata, workspaceId, ); + + if (process.env.FUNNELMINK_PREFILL_NEW_WORKSPACES_WITH_FSM_OBJECTS === 'true') { + await prefillWorkspaceWithFunnelminkFSMObjects( + dataSourceMetadata, + workspaceId, + this.workspaceDataSourceService, + this.objectMetadataService, + this.fieldMetadataService, + this.relationMetadataService, + ); + } } /** diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-migration-builder/factories/index.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-builder/factories/index.ts index 50f96d394829..5d5e66ac5310 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-migration-builder/factories/index.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-builder/factories/index.ts @@ -1,3 +1,5 @@ +import { WorkspaceMigrationIndexFactory } from 'src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-index.factory'; + import { WorkspaceMigrationObjectFactory } from './workspace-migration-object.factory'; import { WorkspaceMigrationFieldFactory } from './workspace-migration-field.factory'; import { WorkspaceMigrationRelationFactory } from './workspace-migration-relation.factory'; @@ -6,4 +8,5 @@ export const workspaceMigrationBuilderFactories = [ WorkspaceMigrationObjectFactory, WorkspaceMigrationFieldFactory, WorkspaceMigrationRelationFactory, + WorkspaceMigrationIndexFactory, ]; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-index.factory.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-index.factory.ts new file mode 100644 index 000000000000..0392041edc93 --- /dev/null +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-index.factory.ts @@ -0,0 +1,156 @@ +import { Injectable } from '@nestjs/common'; + +import { WorkspaceMigrationBuilderAction } from 'src/engine/workspace-manager/workspace-migration-builder/interfaces/workspace-migration-builder-action.interface'; + +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { + WorkspaceMigrationEntity, + WorkspaceMigrationIndexActionType, + WorkspaceMigrationTableActionType, +} from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; +import { computeObjectTargetTable } from 'src/engine/utils/compute-object-target-table.util'; +import { generateMigrationName } from 'src/engine/metadata-modules/workspace-migration/utils/generate-migration-name.util'; +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; + +@Injectable() +export class WorkspaceMigrationIndexFactory { + constructor() {} + + async create( + originalObjectMetadataCollection: ObjectMetadataEntity[], + indexMetadataCollection: IndexMetadataEntity[], + action: WorkspaceMigrationBuilderAction, + ): Promise<Partial<WorkspaceMigrationEntity>[]> { + const originalObjectMetadataMap = Object.fromEntries( + originalObjectMetadataCollection.map((obj) => [obj.id, obj]), + ); + + const indexMetadataByObjectMetadataMap = new Map< + ObjectMetadataEntity, + IndexMetadataEntity[] + >(); + + indexMetadataCollection.forEach((currentIndexMetadata) => { + const objectMetadata = + originalObjectMetadataMap[currentIndexMetadata.objectMetadataId]; + + if (!objectMetadata) { + throw new Error( + `Object metadata with id ${currentIndexMetadata.objectMetadataId} not found`, + ); + } + + if (!indexMetadataByObjectMetadataMap.has(objectMetadata)) { + indexMetadataByObjectMetadataMap.set(objectMetadata, []); + } + + indexMetadataByObjectMetadataMap + ?.get(objectMetadata) + ?.push(currentIndexMetadata); + }); + + switch (action) { + case WorkspaceMigrationBuilderAction.CREATE: + return this.createIndexMigration(indexMetadataByObjectMetadataMap); + case WorkspaceMigrationBuilderAction.DELETE: + return this.deleteIndexMigration(indexMetadataByObjectMetadataMap); + default: + return []; + } + } + + private async createIndexMigration( + indexMetadataByObjectMetadataMap: Map< + ObjectMetadataEntity, + IndexMetadataEntity[] + >, + ): Promise<Partial<WorkspaceMigrationEntity>[]> { + const workspaceMigrations: Partial<WorkspaceMigrationEntity>[] = []; + + for (const [ + objectMetadata, + indexMetadataCollection, + ] of indexMetadataByObjectMetadataMap) { + const targetTable = computeObjectTargetTable(objectMetadata); + + const fieldsById = Object.fromEntries( + objectMetadata.fields.map((field) => [field.id, field]), + ); + + const indexes = indexMetadataCollection.map((indexMetadata) => ({ + name: indexMetadata.name, + action: WorkspaceMigrationIndexActionType.CREATE, + columns: indexMetadata.indexFieldMetadatas + .sort((a, b) => a.order - b.order) + .map((indexFieldMetadata) => { + const fieldMetadata = + fieldsById[indexFieldMetadata.fieldMetadataId]; + + if (!fieldMetadata) { + throw new Error( + `Field metadata with id ${indexFieldMetadata.fieldMetadataId} not found in object metadata with id ${objectMetadata.id}`, + ); + } + + return fieldMetadata.name; + }), + })); + + workspaceMigrations.push({ + workspaceId: objectMetadata.workspaceId, + name: generateMigrationName( + `create-${objectMetadata.nameSingular}-indexes`, + ), + isCustom: false, + migrations: [ + { + name: targetTable, + action: WorkspaceMigrationTableActionType.ALTER_INDEXES, + indexes, + }, + ], + }); + } + + return workspaceMigrations; + } + + private async deleteIndexMigration( + indexMetadataByObjectMetadataMap: Map< + ObjectMetadataEntity, + IndexMetadataEntity[] + >, + ): Promise<Partial<WorkspaceMigrationEntity>[]> { + const workspaceMigrations: Partial<WorkspaceMigrationEntity>[] = []; + + for (const [ + objectMetadata, + indexMetadataCollection, + ] of indexMetadataByObjectMetadataMap) { + const targetTable = computeObjectTargetTable(objectMetadata); + + const indexes = indexMetadataCollection.map((indexMetadata) => ({ + name: indexMetadata.name, + action: WorkspaceMigrationIndexActionType.DROP, + columns: [], + })); + + workspaceMigrations.push({ + workspaceId: objectMetadata.workspaceId, + name: generateMigrationName( + `delete-${objectMetadata.nameSingular}-indexes`, + ), + isCustom: false, + migrations: [ + { + name: targetTable, + action: WorkspaceMigrationTableActionType.ALTER_INDEXES, + indexes, + }, + ], + }); + } + + return workspaceMigrations; + } +} diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/services/workspace-migration-enum.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/services/workspace-migration-enum.service.ts index 05ebc0370482..03dd078fa6a6 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/services/workspace-migration-enum.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/services/workspace-migration-enum.service.ts @@ -49,10 +49,6 @@ export class WorkspaceMigrationEnumService { typeof enumValue !== 'string', ); - if (!columnDefinition.isNullable && !columnDefinition.defaultValue) { - columnDefinition.defaultValue = serializeDefaultValue(enumValues[0]); - } - const oldColumnName = `${columnDefinition.columnName}_old_${v4()}`; // Rename old column diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.service.ts index 560d9fff4995..c239462a2dbc 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.service.ts @@ -5,6 +5,7 @@ import { Table, TableColumn, TableForeignKey, + TableIndex, TableUnique, } from 'typeorm'; @@ -20,6 +21,8 @@ import { WorkspaceMigrationColumnDropRelation, WorkspaceMigrationTableActionType, WorkspaceMigrationForeignTable, + WorkspaceMigrationIndexAction, + WorkspaceMigrationIndexActionType, } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; import { WorkspaceMigrationEnumService } from 'src/engine/workspace-manager/workspace-migration-runner/services/workspace-migration-enum.service'; @@ -137,6 +140,7 @@ export class WorkspaceMigrationRunnerService { tableMigration.columns, ); } + break; } case WorkspaceMigrationTableActionType.DROP: @@ -163,6 +167,17 @@ export class WorkspaceMigrationRunnerService { tableMigration.columns, ); break; + + case WorkspaceMigrationTableActionType.ALTER_INDEXES: + if (tableMigration.indexes && tableMigration.indexes.length > 0) { + await this.handleIndexesChanges( + queryRunner, + schemaName, + tableMigration.newName ?? tableMigration.name, + tableMigration.indexes, + ); + } + break; default: throw new Error( `Migration table action ${tableMigration.action} not supported`, @@ -170,6 +185,32 @@ export class WorkspaceMigrationRunnerService { } } + private async handleIndexesChanges( + queryRunner: QueryRunner, + schemaName: string, + tableName: string, + indexes: WorkspaceMigrationIndexAction[], + ) { + for (const index of indexes) { + switch (index.action) { + case WorkspaceMigrationIndexActionType.CREATE: + await queryRunner.createIndex( + `${schemaName}.${tableName}`, + new TableIndex({ + name: index.name, + columnNames: index.columns, + }), + ); + break; + case WorkspaceMigrationIndexActionType.DROP: + await queryRunner.dropIndex(`${schemaName}.${tableName}`, index.name); + break; + default: + throw new Error(`Migration index action not supported`); + } + } + } + /** * Creates a table for a given schema and table name * diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-status/services/workspace-status.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-status/services/workspace-status.service.ts new file mode 100644 index 000000000000..4792e62edd8f --- /dev/null +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-status/services/workspace-status.service.ts @@ -0,0 +1,74 @@ +import { Injectable } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; + +import { Any, Repository } from 'typeorm'; + +import { + BillingSubscription, + SubscriptionStatus, +} from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; +import { + FeatureFlagEntity, + FeatureFlagKeys, +} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; + +@Injectable() +export class WorkspaceStatusService { + constructor( + private readonly environmentService: EnvironmentService, + @InjectRepository(Workspace, 'core') + private readonly workspaceRepository: Repository<Workspace>, + @InjectRepository(BillingSubscription, 'core') + private readonly billingSubscriptionRepository: Repository<BillingSubscription>, + @InjectRepository(FeatureFlagEntity, 'core') + private readonly featureFlagRepository: Repository<FeatureFlagEntity>, + ) {} + + async getActiveWorkspaceIds(): Promise<string[]> { + const workspaces = await this.workspaceRepository.find(); + const workspaceIds = workspaces.map((workspace) => workspace.id); + + if (!this.environmentService.get('IS_BILLING_ENABLED')) { + return workspaceIds; + } + + const billingSubscriptionForWorkspaces = + await this.billingSubscriptionRepository.find({ + where: { + workspaceId: Any(workspaceIds), + status: Any([ + SubscriptionStatus.PastDue, + SubscriptionStatus.Active, + SubscriptionStatus.Trialing, + ]), + }, + }); + + const workspaceIdsWithActiveSubscription = + billingSubscriptionForWorkspaces.map( + (billingSubscription) => billingSubscription.workspaceId, + ); + + const freeAccessEnabledFeatureFlagForWorkspace = + await this.featureFlagRepository.find({ + where: { + workspaceId: Any(workspaceIds), + key: FeatureFlagKeys.IsFreeAccessEnabled, + value: true, + }, + }); + + const workspaceIdsWithFreeAccessEnabled = + freeAccessEnabledFeatureFlagForWorkspace.map( + (featureFlag) => featureFlag.workspaceId, + ); + + return workspaceIds.filter( + (workspaceId) => + workspaceIdsWithActiveSubscription.includes(workspaceId) || + workspaceIdsWithFreeAccessEnabled.includes(workspaceId), + ); + } +} diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-status/workspace-manager.module.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-status/workspace-manager.module.ts new file mode 100644 index 000000000000..ac39a9e0c322 --- /dev/null +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-status/workspace-manager.module.ts @@ -0,0 +1,21 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { BillingSubscription } from 'src/engine/core-modules/billing/entities/billing-subscription.entity'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module'; +import { WorkspaceStatusService } from 'src/engine/workspace-manager/workspace-status/services/workspace-status.service'; + +@Module({ + imports: [ + EnvironmentModule, + TypeOrmModule.forFeature( + [Workspace, BillingSubscription, FeatureFlagEntity], + 'core', + ), + ], + exports: [WorkspaceStatusService], + providers: [WorkspaceStatusService], +}) +export class WorkspaceStatusModule {} diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/add-standard-id.command.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/add-standard-id.command.ts index be810ced1e81..f351cfabf54f 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/add-standard-id.command.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/add-standard-id.command.ts @@ -4,13 +4,13 @@ import { InjectDataSource } from '@nestjs/typeorm'; import { Command, CommandRunner, Option } from 'nest-commander'; import { DataSource } from 'typeorm'; -import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { standardObjectMetadataDefinitions } from 'src/engine/workspace-manager/workspace-sync-metadata/standard-objects'; -import { StandardObjectFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory'; -import { computeStandardObject } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-object.util'; -import { StandardFieldFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-field.factory'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { CustomWorkspaceEntity } from 'src/engine/twenty-orm/custom.workspace-entity'; +import { StandardFieldFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-field.factory'; +import { StandardObjectFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory'; +import { standardObjectMetadataDefinitions } from 'src/engine/workspace-manager/workspace-sync-metadata/standard-objects'; +import { computeStandardFields } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-fields.util'; interface RunCommandOptions { workspaceId?: string; @@ -58,7 +58,10 @@ export class AddStandardIdCommand extends CommandRunner { IS_AIRTABLE_INTEGRATION_ENABLED: true, IS_POSTGRESQL_INTEGRATION_ENABLED: true, IS_STRIPE_INTEGRATION_ENABLED: false, - IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED: true, + IS_COPILOT_ENABLED: false, + IS_MESSAGING_ALIAS_FETCHING_ENABLED: true, + IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED: true, + IS_FREE_ACCESS_ENABLED: false, }, ); const standardFieldMetadataCollection = this.standardFieldFactory.create( @@ -73,7 +76,10 @@ export class AddStandardIdCommand extends CommandRunner { IS_AIRTABLE_INTEGRATION_ENABLED: true, IS_POSTGRESQL_INTEGRATION_ENABLED: true, IS_STRIPE_INTEGRATION_ENABLED: false, - IS_CONTACT_CREATION_FOR_SENT_AND_RECEIVED_EMAILS_ENABLED: true, + IS_COPILOT_ENABLED: false, + IS_MESSAGING_ALIAS_FETCHING_ENABLED: true, + IS_GOOGLE_CALENDAR_SYNC_V2_ENABLED: true, + IS_FREE_ACCESS_ENABLED: false, }, ); @@ -117,11 +123,8 @@ export class AddStandardIdCommand extends CommandRunner { continue; } - const computedStandardObjectMetadata = computeStandardObject( - standardObjectMetadata ?? { - ...originalObjectMetadata, - fields: standardFieldMetadataCollection, - }, + const computedStandardFieldMetadataCollection = computeStandardFields( + standardFieldMetadataCollection, originalObjectMetadata, customObjectMetadataCollection, ); @@ -129,13 +132,13 @@ export class AddStandardIdCommand extends CommandRunner { if (!originalObjectMetadata.isCustom) { updateObjectMetadataCollection.push({ id: originalObjectMetadata.id, - standardId: computedStandardObjectMetadata.standardId, + standardId: originalObjectMetadata.standardId, }); } for (const fieldMetadata of originalObjectMetadata.fields) { const standardFieldMetadata = - computedStandardObjectMetadata.fields.find( + computedStandardFieldMetadataCollection.find( (field) => field.name === fieldMetadata.name && !field.isCustom, ); diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/sync-workspace-metadata.command.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/sync-workspace-metadata.command.ts index b6d15d677bd5..46313f69571d 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/sync-workspace-metadata.command.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/sync-workspace-metadata.command.ts @@ -1,19 +1,20 @@ import { Logger } from '@nestjs/common'; +import isEmpty from 'lodash.isempty'; import { Command, CommandRunner, Option } from 'nest-commander'; import { DataSourceService } from 'src/engine/metadata-modules/data-source/data-source.service'; -import { WorkspaceSyncMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service'; import { WorkspaceHealthService } from 'src/engine/workspace-manager/workspace-health/workspace-health.service'; -import { WorkspaceService } from 'src/engine/core-modules/workspace/services/workspace.service'; +import { WorkspaceStatusService } from 'src/engine/workspace-manager/workspace-status/services/workspace-status.service'; +import { WorkspaceSyncMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service'; import { SyncWorkspaceLoggerService } from './services/sync-workspace-logger.service'; // TODO: implement dry-run interface RunWorkspaceMigrationsOptions { - workspaceId?: string; dryRun?: boolean; force?: boolean; + workspaceId?: string; } @Command({ @@ -28,7 +29,7 @@ export class SyncWorkspaceMetadataCommand extends CommandRunner { private readonly workspaceHealthService: WorkspaceHealthService, private readonly dataSourceService: DataSourceService, private readonly syncWorkspaceLoggerService: SyncWorkspaceLoggerService, - private readonly workspaceService: WorkspaceService, + private readonly workspaceStatusService: WorkspaceStatusService, ) { super(); } @@ -37,9 +38,20 @@ export class SyncWorkspaceMetadataCommand extends CommandRunner { _passedParam: string[], options: RunWorkspaceMigrationsOptions, ): Promise<void> { - const workspaceIds = options.workspaceId - ? [options.workspaceId] - : await this.workspaceService.getWorkspaceIds(); + // TODO: re-implement load index from workspaceService, this is breaking the logger + let workspaceIds = options.workspaceId ? [options.workspaceId] : []; + + if (isEmpty(workspaceIds)) { + const activeWorkspaceIds = + await this.workspaceStatusService.getActiveWorkspaceIds(); + + workspaceIds = activeWorkspaceIds; + this.logger.log( + `Attempting to sync ${activeWorkspaceIds.length} workspaces.`, + ); + } + + const errorsDuringSync: string[] = []; for (const workspaceId of workspaceIds) { try { @@ -60,7 +72,7 @@ export class SyncWorkspaceMetadataCommand extends CommandRunner { 'Please use `workspace:health` command to check issues and fix them before running this command.', ); - return; + continue; } this.logger.warn( @@ -78,28 +90,46 @@ export class SyncWorkspaceMetadataCommand extends CommandRunner { ); } - const dataSourceMetadata = - await this.dataSourceService.getLastDataSourceMetadataFromWorkspaceIdOrFail( - workspaceId, - ); + try { + const dataSourceMetadata = + await this.dataSourceService.getLastDataSourceMetadataFromWorkspaceIdOrFail( + workspaceId, + ); - const { storage, workspaceMigrations } = - await this.workspaceSyncMetadataService.synchronize( - { + const { storage, workspaceMigrations } = + await this.workspaceSyncMetadataService.synchronize( + { + workspaceId, + dataSourceId: dataSourceMetadata.id, + }, + { applyChanges: !options.dryRun }, + ); + + if (options.dryRun) { + await this.syncWorkspaceLoggerService.saveLogs( workspaceId, - dataSourceId: dataSourceMetadata.id, - }, - { applyChanges: !options.dryRun }, + storage, + workspaceMigrations, + ); + } + } catch (error) { + errorsDuringSync.push( + `Failed to synchronize workspace ${workspaceId}: ${error.message}`, ); - if (options.dryRun) { - await this.syncWorkspaceLoggerService.saveLogs( - workspaceId, - storage, - workspaceMigrations, - ); + continue; } } + + this.logger.log( + `Finished synchronizing all active workspaces (${ + workspaceIds.length + } workspaces). ${ + errorsDuringSync.length > 0 + ? 'Errors during sync:\n' + errorsDuringSync.join('.\n') + : '' + }`, + ); } @Option({ diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/workspace-sync-metadata-commands.module.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/workspace-sync-metadata-commands.module.ts index 8e30f0c418fb..1178966443ab 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/workspace-sync-metadata-commands.module.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/commands/workspace-sync-metadata-commands.module.ts @@ -1,12 +1,15 @@ import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; import { DataSourceModule } from 'src/engine/metadata-modules/data-source/data-source.module'; -import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; import { WorkspaceHealthModule } from 'src/engine/workspace-manager/workspace-health/workspace-health.module'; -import { WorkspaceModule } from 'src/engine/core-modules/workspace/workspace.module'; +import { WorkspaceStatusModule } from 'src/engine/workspace-manager/workspace-status/workspace-manager.module'; import { AddStandardIdCommand } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/add-standard-id.command'; import { ConvertRecordPositionsToIntegers } from 'src/engine/workspace-manager/workspace-sync-metadata/commands/convert-record-positions-to-integers.command'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { WorkspaceSyncMetadataModule } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module'; import { SyncWorkspaceMetadataCommand } from './sync-workspace-metadata.command'; @@ -19,6 +22,8 @@ import { SyncWorkspaceLoggerService } from './services/sync-workspace-logger.ser WorkspaceModule, DataSourceModule, WorkspaceDataSourceModule, + TypeOrmModule.forFeature([Workspace], 'core'), + WorkspaceStatusModule, ], providers: [ SyncWorkspaceMetadataCommand, diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/__tests__/workspace-field.comparator.spec.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/__tests__/workspace-field.comparator.spec.ts index e4deb708c3a6..3df3f57398fe 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/__tests__/workspace-field.comparator.spec.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/__tests__/workspace-field.comparator.spec.ts @@ -36,7 +36,7 @@ describe('WorkspaceFieldComparator', () => { ], } as any; - const result = comparator.compare(original, standard); + const result = comparator.compare('', original.fields, standard.fields); expect(result).toEqual([ { @@ -65,7 +65,7 @@ describe('WorkspaceFieldComparator', () => { ], } as any; - const result = comparator.compare(original, standard); + const result = comparator.compare('', original.fields, standard.fields); expect(result).toEqual([ { @@ -88,7 +88,7 @@ describe('WorkspaceFieldComparator', () => { } as any; const standard = { fields: [] } as any; - const result = comparator.compare(original, standard); + const result = comparator.compare('', original.fields, standard.fields); expect(result).toEqual([ { @@ -108,7 +108,7 @@ describe('WorkspaceFieldComparator', () => { fields: [createMockFieldMetadata({ standardId: '1' })], } as any; - const result = comparator.compare(original, standard); + const result = comparator.compare('', original.fields, standard.fields); expect(result).toHaveLength(0); }); diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/index.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/index.ts index 3c95f7babe3f..2f29fbf41f5b 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/index.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/index.ts @@ -1,3 +1,5 @@ +import { WorkspaceIndexComparator } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-index.comparator'; + import { WorkspaceFieldComparator } from './workspace-field.comparator'; import { WorkspaceObjectComparator } from './workspace-object.comparator'; import { WorkspaceRelationComparator } from './workspace-relation.comparator'; @@ -6,4 +8,5 @@ export const workspaceSyncMetadataComparators = [ WorkspaceFieldComparator, WorkspaceObjectComparator, WorkspaceRelationComparator, + WorkspaceIndexComparator, ]; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-field.comparator.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-field.comparator.ts index eaf55f33a446..7e51a398f9be 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-field.comparator.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-field.comparator.ts @@ -7,14 +7,12 @@ import { FieldComparatorResult, } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface'; import { ComputedPartialFieldMetadata } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-field-metadata.interface'; -import { ComputedPartialWorkspaceEntity } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-object-metadata.interface'; -import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; -import { transformMetadataForComparison } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators/utils/transform-metadata-for-comparison.util'; import { FieldMetadataEntity, FieldMetadataType, } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { transformMetadataForComparison } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators/utils/transform-metadata-for-comparison.util'; const commonFieldPropertiesToIgnore = [ 'id', @@ -30,13 +28,20 @@ const commonFieldPropertiesToIgnore = [ const fieldPropertiesToStringify = ['defaultValue'] as const; +const shouldSkipFieldCreation = ( + standardFieldMetadata: ComputedPartialFieldMetadata | undefined, +) => { + return standardFieldMetadata?.isCustom; +}; + @Injectable() export class WorkspaceFieldComparator { constructor() {} public compare( - originalObjectMetadata: ObjectMetadataEntity, - standardObjectMetadata: ComputedPartialWorkspaceEntity, + originalObjectMetadataId: string, + originalFieldMetadataCollection: FieldMetadataEntity[], + standardFieldMetadataCollection: ComputedPartialFieldMetadata[], ): FieldComparatorResult[] { const result: FieldComparatorResult[] = []; const fieldPropertiesToUpdateMap: Record< @@ -46,7 +51,7 @@ export class WorkspaceFieldComparator { // Double security to only compare non-custom fields const filteredOriginalFieldCollection = - originalObjectMetadata.fields.filter((field) => !field.isCustom); + originalFieldMetadataCollection.filter((field) => !field.isCustom); const originalFieldMetadataMap = transformMetadataForComparison( filteredOriginalFieldCollection, { @@ -73,7 +78,7 @@ export class WorkspaceFieldComparator { }, ); const standardFieldMetadataMap = transformMetadataForComparison( - standardObjectMetadata.fields, + standardFieldMetadataCollection, { shouldIgnoreProperty: (property, originalMetadata) => { if (commonFieldPropertiesToIgnore.includes(property)) { @@ -109,20 +114,19 @@ export class WorkspaceFieldComparator { const findField = ( field: ComputedPartialFieldMetadata | FieldMetadataEntity, ) => { - if (field.isCustom) { - return field.name === fieldName; - } - return field.standardId === fieldName; }; // Object shouldn't have thousands of fields, so we can use find here const standardFieldMetadata = - standardObjectMetadata.fields.find(findField); + standardFieldMetadataCollection.find(findField); const originalFieldMetadata = - originalObjectMetadata.fields.find(findField); + originalFieldMetadataCollection.find(findField); switch (difference.type) { case 'CREATE': { + if (shouldSkipFieldCreation(standardFieldMetadata)) { + break; + } if (!standardFieldMetadata) { throw new Error( `Field ${fieldName} not found in standardObjectMetadata`, @@ -133,7 +137,7 @@ export class WorkspaceFieldComparator { action: ComparatorAction.CREATE, object: { ...standardFieldMetadata, - objectMetadataId: originalObjectMetadata.id, + objectMetadataId: originalObjectMetadataId, }, }); break; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-index.comparator.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-index.comparator.ts new file mode 100644 index 000000000000..5fbf2ad9756d --- /dev/null +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-index.comparator.ts @@ -0,0 +1,90 @@ +import { Injectable } from '@nestjs/common'; + +import diff from 'microdiff'; + +import { + IndexComparatorResult, + ComparatorAction, +} from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface'; + +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; +import { transformMetadataForComparison } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators/utils/transform-metadata-for-comparison.util'; + +const propertiesToIgnore = ['createdAt', 'updatedAt', 'indexFieldMetadatas']; + +@Injectable() +export class WorkspaceIndexComparator { + constructor() {} + + compare( + originalIndexMetadataCollection: IndexMetadataEntity[], + standardIndexMetadataCollection: Partial<IndexMetadataEntity>[], + ): IndexComparatorResult[] { + const results: IndexComparatorResult[] = []; + + // Create a map of standard relations + const standardIndexMetadataMap = transformMetadataForComparison( + standardIndexMetadataCollection, + { + keyFactory(indexMetadata) { + return `${indexMetadata.name}`; + }, + }, + ); + + const originalIndexMetadataCollectionWithColumns = + originalIndexMetadataCollection.map((indexMetadata) => { + return { + ...indexMetadata, + columns: indexMetadata.indexFieldMetadatas.map( + (indexFieldMetadata) => indexFieldMetadata.fieldMetadata.name, + ), + indexFieldMetadatas: undefined, + }; + }); + + // Create a filtered map of original relations + // We filter out 'id' later because we need it to remove the relation from DB + const originalIndexMetadataMap = transformMetadataForComparison( + originalIndexMetadataCollectionWithColumns, + { + shouldIgnoreProperty: (property) => + propertiesToIgnore.includes(property), + keyFactory(indexMetadata) { + return `${indexMetadata.name}`; + }, + }, + ); + + // Compare indexes + const indexesDifferences = diff( + originalIndexMetadataMap, + standardIndexMetadataMap, + ); + + for (const difference of indexesDifferences) { + switch (difference.type) { + case 'CREATE': { + results.push({ + action: ComparatorAction.CREATE, + object: difference.value, + }); + break; + } + case 'REMOVE': { + if (difference.path[difference.path.length - 1] !== 'id') { + results.push({ + action: ComparatorAction.DELETE, + object: difference.oldValue, + }); + } + break; + } + default: + break; + } + } + + return results; + } +} diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-object.comparator.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-object.comparator.ts index 9ab7f2f27f0d..eaa37da7cf84 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-object.comparator.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-object.comparator.ts @@ -27,8 +27,8 @@ export class WorkspaceObjectComparator { constructor() {} public compare( - originalObjectMetadata: ObjectMetadataEntity | undefined, - standardObjectMetadata: ComputedPartialWorkspaceEntity, + originalObjectMetadata: Omit<ObjectMetadataEntity, 'fields'> | undefined, + standardObjectMetadata: Omit<ComputedPartialWorkspaceEntity, 'fields'>, ): ObjectComparatorResult { // If the object doesn't exist in the original metadata, we need to create it if (!originalObjectMetadata) { diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids.ts index 21910d0a758e..0778c007c325 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids.ts @@ -67,10 +67,14 @@ export const CALENDAR_CHANNEL_STANDARD_FIELD_IDS = { handle: '20202020-1d08-420a-9aa7-22e0f298232d', visibility: '20202020-1b07-4796-9f01-d626bab7ca4d', isContactAutoCreationEnabled: '20202020-50fb-404b-ba28-369911a3793a', + contactAutoCreationPolicy: '20202020-b55d-447d-b4df-226319058775', isSyncEnabled: '20202020-fe19-4818-8854-21f7b1b43395', syncCursor: '20202020-bac2-4852-a5cb-7a7898992b70', calendarChannelEventAssociations: '20202020-afb0-4a9f-979f-2d5087d71d09', throttleFailureCount: '20202020-525c-4b76-b9bd-0dd57fd11d61', + syncStatus: '20202020-7116-41da-8b4b-035975c4eb6a', + syncStage: '20202020-6246-42e6-b5cd-003bd921782c', + syncStageStartedAt: '20202020-a934-46f1-a8e7-9568b1e3a53e', }; export const CALENDAR_EVENT_PARTICIPANT_STANDARD_FIELD_IDS = { @@ -110,7 +114,8 @@ export const COMMENT_STANDARD_FIELD_IDS = { export const COMPANY_STANDARD_FIELD_IDS = { name: '20202020-4d99-4e2e-a84c-4a27837b1ece', domainName: '20202020-0c28-43d8-8ba5-3659924d3489', - address: '20202020-a82a-4ee2-96cc-a18a3259d953', + address_deprecated: '20202020-a82a-4ee2-96cc-a18a3259d953', + address: '20202020-c5ce-4adc-b7b6-9c0979fc55e7', employees: '20202020-8965-464a-8a75-74bafc152a0b', linkedinLink: '20202020-ebeb-4beb-b9ad-6848036fb451', xLink: '20202020-6f64-4fd9-9580-9c1991c7d8c3', @@ -136,6 +141,7 @@ export const CONNECTED_ACCOUNT_STANDARD_FIELD_IDS = { authFailedAt: '20202020-d268-4c6b-baff-400d402b430a', messageChannels: '20202020-24f7-4362-8468-042204d1e445', calendarChannels: '20202020-af4a-47bb-99ec-51911c1d3977', + handleAliases: '20202020-8a3d-46be-814f-6228af16c47b', }; export const EVENT_STANDARD_FIELD_IDS = { @@ -152,6 +158,7 @@ export const AUDIT_LOGS_STANDARD_FIELD_IDS = { properties: '20202020-5d36-470e-8fad-d56ea3ab2fd0', context: '20202020-b9d1-4058-9a75-7469cab5ca8c', objectName: '20202020-76ba-4c47-b7e5-96034005d00a', + objectMetadataId: '20202020-127b-409d-9864-0ec44aa9ed98', recordId: '20202020-c578-4acf-bf94-eb53b035cea2', workspaceMember: '20202020-6e96-4300-b3f5-67a707147385', }; @@ -202,6 +209,9 @@ export const MESSAGE_CHANNEL_STANDARD_FIELD_IDS = { connectedAccount: '20202020-49a2-44a4-b470-282c0440d15d', type: '20202020-ae95-42d9-a3f1-797a2ea22122', isContactAutoCreationEnabled: '20202020-fabd-4f14-b7c6-3310f6d132c6', + contactAutoCreationPolicy: '20202020-fc0e-4ba6-b259-a66ca89cfa38', + excludeNonProfessionalEmails: '20202020-1df5-445d-b4f3-2413ad178431', + excludeGroupEmails: '20202020-45a0-4be4-9164-5820a6a109fb', messageChannelMessageAssociations: '20202020-49b8-4766-88fd-75f1e21b3d5f', isSyncEnabled: '20202020-d9a6-48e9-990b-b97fdf22e8dd', syncCursor: '20202020-79d1-41cf-b738-bcf5ed61e256', @@ -241,7 +251,7 @@ export const OPPORTUNITY_STANDARD_FIELD_IDS = { name: '20202020-8609-4f65-a2d9-44009eb422b5', amount: '20202020-583e-4642-8533-db761d5fa82f', closeDate: '20202020-527e-44d6-b1ac-c4158d307b97', - probability: '20202020-69d4-45f3-9703-690b09fafcf0', + probabilityDeprecated: '20202020-69d4-45f3-9703-690b09fafcf0', stage: '20202020-6f76-477d-8551-28cd65b2b4b9', position: '20202020-806d-493a-bbc6-6313e62958e2', pointOfContact: '20202020-8dfb-42fc-92b6-01afb759ed16', diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/index.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/index.ts index 72185436922b..958bf346cae7 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/index.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/index.ts @@ -1,3 +1,5 @@ +import { StandardIndexFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-index.factory'; + import { FeatureFlagFactory } from './feature-flags.factory'; import { StandardFieldFactory } from './standard-field.factory'; import { StandardObjectFactory } from './standard-object.factory'; @@ -8,4 +10,5 @@ export const workspaceSyncMetadataFactories = [ StandardFieldFactory, StandardObjectFactory, StandardRelationFactory, + StandardIndexFactory, ]; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-field.factory.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-field.factory.ts index 07bb1ef059d2..eff6dedbdb0f 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-field.factory.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-field.factory.ts @@ -16,6 +16,7 @@ import { metadataArgsStorage } from 'src/engine/twenty-orm/storage/metadata-args import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { createDeterministicUuid } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/create-deterministic-uuid.util'; +import { getJoinColumn } from 'src/engine/twenty-orm/utils/get-join-column.util'; @Injectable() export class StandardFieldFactory { @@ -23,10 +24,53 @@ export class StandardFieldFactory { target: typeof BaseWorkspaceEntity, context: WorkspaceSyncContext, workspaceFeatureFlagsMap: FeatureFlagMap, - ): Array<PartialFieldMetadata | PartialComputedFieldMetadata> { + ): (PartialFieldMetadata | PartialComputedFieldMetadata)[]; + + create( + targets: (typeof BaseWorkspaceEntity)[], + context: WorkspaceSyncContext, + workspaceFeatureFlagsMap: FeatureFlagMap, // Map of standardId to field metadata + ): Map<string, (PartialFieldMetadata | PartialComputedFieldMetadata)[]>; + + create( + targetOrTargets: + | typeof BaseWorkspaceEntity + | (typeof BaseWorkspaceEntity)[], + context: WorkspaceSyncContext, + workspaceFeatureFlagsMap: FeatureFlagMap, + ): + | (PartialFieldMetadata | PartialComputedFieldMetadata)[] + | Map<string, (PartialFieldMetadata | PartialComputedFieldMetadata)[]> { + if (Array.isArray(targetOrTargets)) { + return targetOrTargets.reduce((acc, target) => { + const workspaceEntityMetadataArgs = + metadataArgsStorage.filterEntities(target); + + if (!workspaceEntityMetadataArgs) { + return acc; + } + + if ( + isGatedAndNotEnabled( + workspaceEntityMetadataArgs.gate, + workspaceFeatureFlagsMap, + ) + ) { + return acc; + } + + acc.set( + workspaceEntityMetadataArgs.standardId, + this.create(target, context, workspaceFeatureFlagsMap), + ); + + return acc; + }, new Map<string, (PartialFieldMetadata | PartialComputedFieldMetadata)[]>()); + } + const workspaceEntityMetadataArgs = - metadataArgsStorage.filterEntities(target); - const metadataCollections = this.collectMetadata(target); + metadataArgsStorage.filterEntities(targetOrTargets); + const metadataCollections = this.collectMetadata(targetOrTargets); return [ ...this.processMetadata( @@ -118,7 +162,7 @@ export class StandardFieldFactory { options: workspaceFieldMetadataArgs.options, workspaceId: context.workspaceId, isNullable: workspaceFieldMetadataArgs.isNullable, - isCustom: false, + isCustom: workspaceFieldMetadataArgs.isDeprecated ? true : false, isSystem: workspaceEntityMetadataArgs?.isSystem || workspaceFieldMetadataArgs.isSystem, @@ -139,6 +183,14 @@ export class StandardFieldFactory { const foreignKeyStandardId = createDeterministicUuid( workspaceRelationMetadataArgs.standardId, ); + const joinColumnMetadataArgsCollection = + metadataArgsStorage.filterJoinColumns( + workspaceRelationMetadataArgs.target, + ); + const joinColumn = getJoinColumn( + joinColumnMetadataArgsCollection, + workspaceRelationMetadataArgs, + ); if ( isGatedAndNotEnabled( @@ -149,11 +201,11 @@ export class StandardFieldFactory { return []; } - if (workspaceRelationMetadataArgs.joinColumn) { + if (joinColumn) { fieldMetadataCollection.push({ type: FieldMetadataType.UUID, standardId: foreignKeyStandardId, - name: workspaceRelationMetadataArgs.joinColumn, + name: joinColumn, label: `${workspaceRelationMetadataArgs.label} id (foreign key)`, description: `${workspaceRelationMetadataArgs.description} id foreign key`, icon: workspaceRelationMetadataArgs.icon, diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-index.factory.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-index.factory.ts new file mode 100644 index 000000000000..83e450cbe90b --- /dev/null +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-index.factory.ts @@ -0,0 +1,69 @@ +import { Injectable } from '@nestjs/common'; + +import { FeatureFlagMap } from 'src/engine/core-modules/feature-flag/interfaces/feature-flag-map.interface'; +import { PartialIndexMetadata } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-index-metadata.interface'; +import { WorkspaceSyncContext } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/workspace-sync-context.interface'; + +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { metadataArgsStorage } from 'src/engine/twenty-orm/storage/metadata-args.storage'; + +@Injectable() +export class StandardIndexFactory { + create( + standardObjectMetadataDefinitions: (typeof BaseWorkspaceEntity)[], + context: WorkspaceSyncContext, + originalObjectMetadataMap: Record<string, ObjectMetadataEntity>, + workspaceFeatureFlagsMap: FeatureFlagMap, + ): Partial<IndexMetadataEntity>[] { + return standardObjectMetadataDefinitions.flatMap((standardObjectMetadata) => + this.createIndexMetadata( + standardObjectMetadata, + context, + originalObjectMetadataMap, + workspaceFeatureFlagsMap, + ), + ); + } + + private createIndexMetadata( + target: typeof BaseWorkspaceEntity, + context: WorkspaceSyncContext, + originalObjectMetadataMap: Record<string, ObjectMetadataEntity>, + _workspaceFeatureFlagsMap: FeatureFlagMap, + ): Partial<IndexMetadataEntity>[] { + const workspaceEntity = metadataArgsStorage.filterEntities(target); + + if (!workspaceEntity) { + throw new Error( + `Object metadata decorator not found, can't parse ${target.name}`, + ); + } + + const workspaceIndexMetadataArgsCollection = + metadataArgsStorage.filterIndexes(target); + + return workspaceIndexMetadataArgsCollection.map( + (workspaceIndexMetadataArgs) => { + const objectMetadata = + originalObjectMetadataMap[workspaceEntity.nameSingular]; + + if (!objectMetadata) { + throw new Error( + `Object metadata not found for ${workspaceEntity.nameSingular}`, + ); + } + + const indexMetadata: PartialIndexMetadata = { + workspaceId: context.workspaceId, + objectMetadataId: objectMetadata.id, + name: workspaceIndexMetadataArgs.name, + columns: workspaceIndexMetadataArgs.columns, + }; + + return indexMetadata; + }, + ); + } +} diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory.ts index 9a907f3c3d65..72e29753e992 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory.ts @@ -8,17 +8,13 @@ import { isGatedAndNotEnabled } from 'src/engine/workspace-manager/workspace-syn import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; import { metadataArgsStorage } from 'src/engine/twenty-orm/storage/metadata-args.storage'; -import { StandardFieldFactory } from './standard-field.factory'; - @Injectable() export class StandardObjectFactory { - constructor(private readonly standardFieldFactory: StandardFieldFactory) {} - create( standardObjectMetadataDefinitions: (typeof BaseWorkspaceEntity)[], context: WorkspaceSyncContext, workspaceFeatureFlagsMap: FeatureFlagMap, - ): PartialWorkspaceEntity[] { + ): Omit<PartialWorkspaceEntity, 'fields'>[] { return standardObjectMetadataDefinitions .map((metadata) => this.createObjectMetadata(metadata, context, workspaceFeatureFlagsMap), @@ -30,7 +26,7 @@ export class StandardObjectFactory { target: typeof BaseWorkspaceEntity, context: WorkspaceSyncContext, workspaceFeatureFlagsMap: FeatureFlagMap, - ): PartialWorkspaceEntity | undefined { + ): Omit<PartialWorkspaceEntity, 'fields'> | undefined { const workspaceEntityMetadataArgs = metadataArgsStorage.filterEntities(target); @@ -49,12 +45,6 @@ export class StandardObjectFactory { return undefined; } - const fields = this.standardFieldFactory.create( - target, - context, - workspaceFeatureFlagsMap, - ); - return { ...workspaceEntityMetadataArgs, // TODO: Remove targetTableName when we remove the old metadata @@ -64,7 +54,6 @@ export class StandardObjectFactory { isCustom: false, isRemote: false, isSystem: workspaceEntityMetadataArgs.isSystem ?? false, - fields, }; } } diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface.ts index efc061bc73ac..b4279b5d2397 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface.ts @@ -1,5 +1,6 @@ import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { RelationMetadataEntity } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; import { ComputedPartialFieldMetadata } from './partial-field-metadata.interface'; import { ComputedPartialWorkspaceEntity } from './partial-object-metadata.interface'; @@ -32,9 +33,9 @@ export interface ComparatorDeleteResult<T> { export type ObjectComparatorResult = | ComparatorSkipResult - | ComparatorCreateResult<ComputedPartialWorkspaceEntity> + | ComparatorCreateResult<Omit<ComputedPartialWorkspaceEntity, 'fields'>> | ComparatorUpdateResult< - Partial<ComputedPartialWorkspaceEntity> & { id: string } + Partial<Omit<ComputedPartialWorkspaceEntity, 'fields'>> & { id: string } >; export type FieldComparatorResult = @@ -49,3 +50,8 @@ export type RelationComparatorResult = | ComparatorCreateResult<Partial<RelationMetadataEntity>> | ComparatorDeleteResult<RelationMetadataEntity> | ComparatorUpdateResult<Partial<RelationMetadataEntity>>; + +export type IndexComparatorResult = + | ComparatorCreateResult<Partial<IndexMetadataEntity>> + | ComparatorUpdateResult<Partial<IndexMetadataEntity>> + | ComparatorDeleteResult<IndexMetadataEntity>; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-index-metadata.interface.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-index-metadata.interface.ts new file mode 100644 index 000000000000..26939eaa740a --- /dev/null +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-index-metadata.interface.ts @@ -0,0 +1,8 @@ +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; + +export type PartialIndexMetadata = Omit< + IndexMetadataEntity, + 'id' | 'objectMetadata' | 'indexFieldMetadatas' | 'createdAt' | 'updatedAt' +> & { + columns: string[]; +}; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-metadata-updater.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-metadata-updater.service.ts index 5ba73cbcc203..1ae597c2d985 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-metadata-updater.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-metadata-updater.service.ts @@ -11,6 +11,7 @@ import { v4 as uuidV4 } from 'uuid'; import { DeepPartial } from 'typeorm/common/DeepPartial'; import { PartialFieldMetadata } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-field-metadata.interface'; +import { PartialIndexMetadata } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-index-metadata.interface'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { @@ -22,6 +23,7 @@ import { FieldMetadataComplexOption } from 'src/engine/metadata-modules/field-me import { WorkspaceSyncStorage } from 'src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage'; import { FieldMetadataUpdate } from 'src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-field.factory'; import { ObjectMetadataUpdate } from 'src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-object.factory'; +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; @Injectable() export class WorkspaceMetadataUpdaterService { @@ -45,9 +47,6 @@ export class WorkspaceMetadataUpdaterService { storage.objectMetadataCreateCollection.map((objectMetadata) => ({ ...objectMetadata, isActive: true, - fields: objectMetadata.fields.map((field) => - this.prepareFieldMetadataForCreation(field), - ), })) as DeepPartial<ObjectMetadataEntity>[], ); const identifiers = createdPartialObjectMetadataCollection.map( @@ -126,16 +125,6 @@ export class WorkspaceMetadataUpdaterService { updatedFieldMetadataCollection: FieldMetadataUpdate[]; }> { const fieldMetadataRepository = manager.getRepository(FieldMetadataEntity); - - /** - * Create field metadata - */ - const createdFieldMetadataCollection = await fieldMetadataRepository.save( - storage.fieldMetadataCreateCollection.map((field) => - this.prepareFieldMetadataForCreation(field), - ) as DeepPartial<FieldMetadataEntity>[], - ); - /** * Update field metadata */ @@ -146,6 +135,15 @@ export class WorkspaceMetadataUpdaterService { 'workspaceId', ]); + /** + * Create field metadata + */ + const createdFieldMetadataCollection = await fieldMetadataRepository.save( + storage.fieldMetadataCreateCollection.map((field) => + this.prepareFieldMetadataForCreation(field), + ) as DeepPartial<FieldMetadataEntity>[], + ); + /** * Delete field metadata */ @@ -230,6 +228,69 @@ export class WorkspaceMetadataUpdaterService { }; } + async updateIndexMetadata( + manager: EntityManager, + storage: WorkspaceSyncStorage, + originalObjectMetadataCollection: ObjectMetadataEntity[], + ): Promise<{ + createdIndexMetadataCollection: IndexMetadataEntity[]; + }> { + const indexMetadataRepository = manager.getRepository(IndexMetadataEntity); + + const convertIndexMetadataForSaving = ( + indexMetadata: PartialIndexMetadata, + ) => { + const convertIndexFieldMetadataForSaving = ( + column: string, + order: number, + ) => { + const fieldMetadata = originalObjectMetadataCollection + .find((object) => object.id === indexMetadata.objectMetadataId) + ?.fields.find((field) => column === field.name); + + if (!fieldMetadata) { + throw new Error(` + Field metadata not found for column ${column} in object ${indexMetadata.objectMetadataId} + `); + } + + return { + fieldMetadataId: fieldMetadata.id, + order, + }; + }; + + return { + ...indexMetadata, + indexFieldMetadatas: indexMetadata.columns.map((column, index) => + convertIndexFieldMetadataForSaving(column, index), + ), + }; + }; + + /** + * Create index metadata + */ + const createdIndexMetadataCollection = await indexMetadataRepository.save( + storage.indexMetadataCreateCollection.map(convertIndexMetadataForSaving), + ); + + /** + * Delete index metadata + */ + if (storage.indexMetadataDeleteCollection.length > 0) { + await indexMetadataRepository.delete( + storage.indexMetadataDeleteCollection.map( + (indexMetadata) => indexMetadata.id, + ), + ); + } + + return { + createdIndexMetadataCollection, + }; + } + /** * Update entities in the database * @param manager EntityManager diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-field-metadata.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-field-metadata.service.ts index 4fe25b628ce2..a926b9525ba9 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-field-metadata.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-field-metadata.service.ts @@ -3,7 +3,10 @@ import { Injectable, Logger } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { WorkspaceSyncContext } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/workspace-sync-context.interface'; -import { ComparatorAction } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface'; +import { + ComparatorAction, + FieldComparatorResult, +} from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface'; import { FeatureFlagMap } from 'src/engine/core-modules/feature-flag/interfaces/feature-flag-map.interface'; import { WorkspaceMigrationBuilderAction } from 'src/engine/workspace-manager/workspace-migration-builder/interfaces/workspace-migration-builder-action.interface'; @@ -15,7 +18,9 @@ import { WorkspaceSyncStorage } from 'src/engine/workspace-manager/workspace-syn import { WorkspaceMigrationFieldFactory } from 'src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-field.factory'; import { StandardFieldFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-field.factory'; import { CustomWorkspaceEntity } from 'src/engine/twenty-orm/custom.workspace-entity'; -import { computeStandardObject } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-object.util'; +import { computeStandardFields } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-fields.util'; +import { standardObjectMetadataDefinitions } from 'src/engine/workspace-manager/workspace-sync-metadata/standard-objects'; +import { mapObjectMetadataByUniqueIdentifier } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/sync-metadata.util'; @Injectable() export class WorkspaceSyncFieldMetadataService { @@ -43,60 +48,28 @@ export class WorkspaceSyncFieldMetadataService { where: { workspaceId: context.workspaceId, // We're only interested in standard fields - fields: { isCustom: false }, }, relations: ['dataSource', 'fields'], }); - - // Filter out custom objects const customObjectMetadataCollection = originalObjectMetadataCollection.filter( (objectMetadata) => objectMetadata.isCustom, ); - // Create standard field metadata collection - const standardFieldMetadataCollection = this.standardFieldFactory.create( - CustomWorkspaceEntity, + await this.synchronizeStandardObjectFields( context, + originalObjectMetadataCollection, + customObjectMetadataCollection, + storage, workspaceFeatureFlagsMap, ); - // Loop over all standard objects and compare them with the objects in DB - for (const customObjectMetadata of customObjectMetadataCollection) { - // Also, maybe it's better to refactor a bit and move generation part into a separate module ? - const standardObjectMetadata = computeStandardObject( - { - ...customObjectMetadata, - fields: standardFieldMetadataCollection, - }, - customObjectMetadata, - ); - - /** - * COMPARE FIELD METADATA - */ - const fieldComparatorResults = this.workspaceFieldComparator.compare( - customObjectMetadata, - standardObjectMetadata, - ); - - for (const fieldComparatorResult of fieldComparatorResults) { - switch (fieldComparatorResult.action) { - case ComparatorAction.CREATE: { - storage.addCreateFieldMetadata(fieldComparatorResult.object); - break; - } - case ComparatorAction.UPDATE: { - storage.addUpdateFieldMetadata(fieldComparatorResult.object); - break; - } - case ComparatorAction.DELETE: { - storage.addDeleteFieldMetadata(fieldComparatorResult.object); - break; - } - } - } - } + await this.synchronizeCustomObjectFields( + context, + customObjectMetadataCollection, + storage, + workspaceFeatureFlagsMap, + ); this.logger.log('Updating workspace metadata'); @@ -137,4 +110,110 @@ export class WorkspaceSyncFieldMetadataService { ...deleteFieldWorkspaceMigrations, ]; } + + /** + * This can be optimized to avoid import of standardObjectFactory here. + * We should refactor the logic of the factory, so this one only create the objects and not the fields. + * Then standardFieldFactory should be used to create the fields of standard objects. + */ + private async synchronizeStandardObjectFields( + context: WorkspaceSyncContext, + originalObjectMetadataCollection: ObjectMetadataEntity[], + customObjectMetadataCollection: ObjectMetadataEntity[], + storage: WorkspaceSyncStorage, + workspaceFeatureFlagsMap: FeatureFlagMap, + ): Promise<void> { + // Create standard field metadata map + const standardObjectStandardFieldMetadataMap = + this.standardFieldFactory.create( + standardObjectMetadataDefinitions, + context, + workspaceFeatureFlagsMap, + ); + + // Create map of original and standard object metadata by standard ids + const originalObjectMetadataMap = mapObjectMetadataByUniqueIdentifier( + originalObjectMetadataCollection, + ); + + // Loop over all standard objects and compare them with the objects in DB + for (const [ + standardObjectId, + standardFieldMetadataCollection, + ] of standardObjectStandardFieldMetadataMap) { + const originalObjectMetadata = + originalObjectMetadataMap[standardObjectId]; + const computedStandardFieldMetadataCollection = computeStandardFields( + standardFieldMetadataCollection, + originalObjectMetadata, + // We need to provide this for generated relations with custom objects + customObjectMetadataCollection, + ); + + const fieldComparatorResults = this.workspaceFieldComparator.compare( + originalObjectMetadata.id, + originalObjectMetadata.fields, + computedStandardFieldMetadataCollection, + ); + + this.storeComparatorResults(fieldComparatorResults, storage); + } + } + + private async synchronizeCustomObjectFields( + context: WorkspaceSyncContext, + customObjectMetadataCollection: ObjectMetadataEntity[], + storage: WorkspaceSyncStorage, + workspaceFeatureFlagsMap: FeatureFlagMap, + ): Promise<void> { + // Create standard field metadata collection + const customObjectStandardFieldMetadataCollection = + this.standardFieldFactory.create( + CustomWorkspaceEntity, + context, + workspaceFeatureFlagsMap, + ); + + // Loop over all custom objects from the DB and compare their fields with standard fields + for (const customObjectMetadata of customObjectMetadataCollection) { + // Also, maybe it's better to refactor a bit and move generation part into a separate module ? + const standardFieldMetadataCollection = computeStandardFields( + customObjectStandardFieldMetadataCollection, + customObjectMetadata, + ); + + /** + * COMPARE FIELD METADATA + */ + const fieldComparatorResults = this.workspaceFieldComparator.compare( + customObjectMetadata.id, + customObjectMetadata.fields, + standardFieldMetadataCollection, + ); + + this.storeComparatorResults(fieldComparatorResults, storage); + } + } + + private storeComparatorResults( + fieldComparatorResults: FieldComparatorResult[], + storage: WorkspaceSyncStorage, + ): void { + for (const fieldComparatorResult of fieldComparatorResults) { + switch (fieldComparatorResult.action) { + case ComparatorAction.CREATE: { + storage.addCreateFieldMetadata(fieldComparatorResult.object); + break; + } + case ComparatorAction.UPDATE: { + storage.addUpdateFieldMetadata(fieldComparatorResult.object); + break; + } + case ComparatorAction.DELETE: { + storage.addDeleteFieldMetadata(fieldComparatorResult.object); + break; + } + } + } + } } diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-index-metadata.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-index-metadata.service.ts new file mode 100644 index 000000000000..4f7a30ef9d25 --- /dev/null +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-index-metadata.service.ts @@ -0,0 +1,119 @@ +import { Injectable, Logger } from '@nestjs/common'; + +import { EntityManager } from 'typeorm'; + +import { FeatureFlagMap } from 'src/engine/core-modules/feature-flag/interfaces/feature-flag-map.interface'; +import { WorkspaceSyncContext } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/workspace-sync-context.interface'; +import { ComparatorAction } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/comparator.interface'; +import { WorkspaceMigrationBuilderAction } from 'src/engine/workspace-manager/workspace-migration-builder/interfaces/workspace-migration-builder-action.interface'; + +import { WorkspaceMigrationEntity } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; +import { WorkspaceSyncStorage } from 'src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage'; +import { StandardIndexFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-index.factory'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { mapObjectMetadataByUniqueIdentifier } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/sync-metadata.util'; +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; +import { standardObjectMetadataDefinitions } from 'src/engine/workspace-manager/workspace-sync-metadata/standard-objects'; +import { WorkspaceIndexComparator } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-index.comparator'; +import { WorkspaceMetadataUpdaterService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-metadata-updater.service'; +import { WorkspaceMigrationIndexFactory } from 'src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-index.factory'; + +@Injectable() +export class WorkspaceSyncIndexMetadataService { + private readonly logger = new Logger(WorkspaceSyncIndexMetadataService.name); + + constructor( + private readonly standardIndexFactory: StandardIndexFactory, + private readonly workspaceIndexComparator: WorkspaceIndexComparator, + private readonly workspaceMetadataUpdaterService: WorkspaceMetadataUpdaterService, + private readonly workspaceMigrationIndexFactory: WorkspaceMigrationIndexFactory, + ) {} + + async synchronize( + context: WorkspaceSyncContext, + manager: EntityManager, + storage: WorkspaceSyncStorage, + workspaceFeatureFlagsMap: FeatureFlagMap, + ): Promise<Partial<WorkspaceMigrationEntity>[]> { + this.logger.log('Syncing index metadata'); + + const objectMetadataRepository = + manager.getRepository(ObjectMetadataEntity); + + // Retrieve object metadata collection from DB + const originalObjectMetadataCollection = + await objectMetadataRepository.find({ + where: { + workspaceId: context.workspaceId, + // We're only interested in standard fields + fields: { isCustom: false }, + isCustom: false, + }, + relations: ['dataSource', 'fields', 'indexes'], + }); + + // Create map of object metadata & field metadata by unique identifier + const originalObjectMetadataMap = mapObjectMetadataByUniqueIdentifier( + originalObjectMetadataCollection, + // Relation are based on the singular name + (objectMetadata) => objectMetadata.nameSingular, + ); + + const indexMetadataRepository = manager.getRepository(IndexMetadataEntity); + + const originalIndexMetadataCollection = await indexMetadataRepository.find({ + where: { + workspaceId: context.workspaceId, + }, + relations: ['indexFieldMetadatas.fieldMetadata'], + }); + + // Generate index metadata from models + const standardIndexMetadataCollection = this.standardIndexFactory.create( + standardObjectMetadataDefinitions, + context, + originalObjectMetadataMap, + workspaceFeatureFlagsMap, + ); + + const indexComparatorResults = this.workspaceIndexComparator.compare( + originalIndexMetadataCollection, + standardIndexMetadataCollection, + ); + + for (const indexComparatorResult of indexComparatorResults) { + if (indexComparatorResult.action === ComparatorAction.CREATE) { + storage.addCreateIndexMetadata(indexComparatorResult.object); + } else if (indexComparatorResult.action === ComparatorAction.DELETE) { + storage.addDeleteIndexMetadata(indexComparatorResult.object); + } + } + + const metadataIndexUpdaterResult = + await this.workspaceMetadataUpdaterService.updateIndexMetadata( + manager, + storage, + originalObjectMetadataCollection, + ); + + // Create migrations + const createIndexWorkspaceMigrations = + await this.workspaceMigrationIndexFactory.create( + originalObjectMetadataCollection, + metadataIndexUpdaterResult.createdIndexMetadataCollection, + WorkspaceMigrationBuilderAction.CREATE, + ); + + const deleteIndexWorkspaceMigrations = + await this.workspaceMigrationIndexFactory.create( + originalObjectMetadataCollection, + storage.indexMetadataDeleteCollection, + WorkspaceMigrationBuilderAction.DELETE, + ); + + return [ + ...createIndexWorkspaceMigrations, + ...deleteIndexWorkspaceMigrations, + ]; + } +} diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-object-metadata.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-object-metadata.service.ts index 147389b18918..34d456d19d95 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-object-metadata.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-object-metadata.service.ts @@ -12,11 +12,9 @@ import { mapObjectMetadataByUniqueIdentifier } from 'src/engine/workspace-manage import { WorkspaceMigrationEntity } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; import { StandardObjectFactory } from 'src/engine/workspace-manager/workspace-sync-metadata/factories/standard-object.factory'; import { WorkspaceObjectComparator } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-object.comparator'; -import { WorkspaceFieldComparator } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators/workspace-field.comparator'; import { WorkspaceMetadataUpdaterService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-metadata-updater.service'; import { WorkspaceSyncStorage } from 'src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage'; import { WorkspaceMigrationObjectFactory } from 'src/engine/workspace-manager/workspace-migration-builder/factories/workspace-migration-object.factory'; -import { computeStandardObject } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-object.util'; import { standardObjectMetadataDefinitions } from 'src/engine/workspace-manager/workspace-sync-metadata/standard-objects'; @Injectable() @@ -26,7 +24,6 @@ export class WorkspaceSyncObjectMetadataService { constructor( private readonly standardObjectFactory: StandardObjectFactory, private readonly workspaceObjectComparator: WorkspaceObjectComparator, - private readonly workspaceFieldComparator: WorkspaceFieldComparator, private readonly workspaceMetadataUpdaterService: WorkspaceMetadataUpdaterService, private readonly workspaceMigrationObjectFactory: WorkspaceMigrationObjectFactory, ) {} @@ -49,10 +46,6 @@ export class WorkspaceSyncObjectMetadataService { }, relations: ['dataSource', 'fields'], }); - const customObjectMetadataCollection = - originalObjectMetadataCollection.filter( - (objectMetadata) => objectMetadata.isCustom, - ); // Create standard object metadata collection const standardObjectMetadataCollection = this.standardObjectFactory.create( @@ -87,11 +80,8 @@ export class WorkspaceSyncObjectMetadataService { for (const standardObjectId in standardObjectMetadataMap) { const originalObjectMetadata = originalObjectMetadataMap[standardObjectId]; - const standardObjectMetadata = computeStandardObject( - standardObjectMetadataMap[standardObjectId], - originalObjectMetadata, - customObjectMetadataCollection, - ); + const standardObjectMetadata = + standardObjectMetadataMap[standardObjectId]; /** * COMPARE OBJECT METADATA @@ -109,31 +99,6 @@ export class WorkspaceSyncObjectMetadataService { if (objectComparatorResult.action === ComparatorAction.UPDATE) { storage.addUpdateObjectMetadata(objectComparatorResult.object); } - - /** - * COMPARE FIELD METADATA - */ - const fieldComparatorResults = this.workspaceFieldComparator.compare( - originalObjectMetadata, - standardObjectMetadata, - ); - - for (const fieldComparatorResult of fieldComparatorResults) { - switch (fieldComparatorResult.action) { - case ComparatorAction.CREATE: { - storage.addCreateFieldMetadata(fieldComparatorResult.object); - break; - } - case ComparatorAction.UPDATE: { - storage.addUpdateFieldMetadata(fieldComparatorResult.object); - break; - } - case ComparatorAction.DELETE: { - storage.addDeleteFieldMetadata(fieldComparatorResult.object); - break; - } - } - } } this.logger.log('Updating workspace metadata'); diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-relation-metadata.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-relation-metadata.service.ts index 86d46f01a853..ba14f3634e93 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-relation-metadata.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-relation-metadata.service.ts @@ -46,7 +46,6 @@ export class WorkspaceSyncRelationMetadataService { await objectMetadataRepository.find({ where: { workspaceId: context.workspaceId, - fields: { isCustom: false }, }, relations: ['dataSource', 'fields'], }); diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/standard-objects/index.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/standard-objects/index.ts index 39ef93760e7f..43d591382253 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/standard-objects/index.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/standard-objects/index.ts @@ -1,33 +1,34 @@ import { ActivityTargetWorkspaceEntity } from 'src/modules/activity/standard-objects/activity-target.workspace-entity'; import { ActivityWorkspaceEntity } from 'src/modules/activity/standard-objects/activity.workspace-entity'; +import { CommentWorkspaceEntity } from 'src/modules/activity/standard-objects/comment.workspace-entity'; import { ApiKeyWorkspaceEntity } from 'src/modules/api-key/standard-objects/api-key.workspace-entity'; import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objects/attachment.workspace-entity'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { CommentWorkspaceEntity } from 'src/modules/activity/standard-objects/comment.workspace-entity'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/company.workspace-entity'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { FavoriteWorkspaceEntity } from 'src/modules/favorite/standard-objects/favorite.workspace-entity'; +import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; +import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; +import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; +import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; import { OpportunityWorkspaceEntity } from 'src/modules/opportunity/standard-objects/opportunity.workspace-entity'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; +import { AuditLogWorkspaceEntity } from 'src/modules/timeline/standard-objects/audit-log.workspace-entity'; +import { BehavioralEventWorkspaceEntity } from 'src/modules/timeline/standard-objects/behavioral-event.workspace-entity'; +import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; import { ViewFieldWorkspaceEntity } from 'src/modules/view/standard-objects/view-field.workspace-entity'; import { ViewFilterWorkspaceEntity } from 'src/modules/view/standard-objects/view-filter.workspace-entity'; import { ViewSortWorkspaceEntity } from 'src/modules/view/standard-objects/view-sort.workspace-entity'; import { ViewWorkspaceEntity } from 'src/modules/view/standard-objects/view.workspace-entity'; import { WebhookWorkspaceEntity } from 'src/modules/webhook/standard-objects/webhook.workspace-entity'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { AuditLogWorkspaceEntity } from 'src/modules/timeline/standard-objects/audit-log.workspace-entity'; -import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; -import { BehavioralEventWorkspaceEntity } from 'src/modules/timeline/standard-objects/behavioral-event.workspace-entity'; -import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; -import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; -import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; -import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; -import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; +// TODO: Maybe we should automate this with the DiscoverService of Nest.JS export const standardObjectMetadataDefinitions = [ ActivityTargetWorkspaceEntity, ActivityWorkspaceEntity, diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage.ts index 6d0a7a9d39ae..49bc4b176fcd 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage.ts @@ -4,12 +4,17 @@ import { ComputedPartialFieldMetadata } from 'src/engine/workspace-manager/works import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { RelationMetadataEntity } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { IndexMetadataEntity } from 'src/engine/metadata-modules/index-metadata/index-metadata.entity'; export class WorkspaceSyncStorage { // Object metadata - private readonly _objectMetadataCreateCollection: ComputedPartialWorkspaceEntity[] = - []; - private readonly _objectMetadataUpdateCollection: (Partial<ComputedPartialWorkspaceEntity> & { + private readonly _objectMetadataCreateCollection: Omit< + ComputedPartialWorkspaceEntity, + 'fields' + >[] = []; + private readonly _objectMetadataUpdateCollection: (Partial< + Omit<ComputedPartialWorkspaceEntity, 'fields'> + > & { id: string; })[] = []; private readonly _objectMetadataDeleteCollection: ObjectMetadataEntity[] = []; @@ -25,10 +30,17 @@ export class WorkspaceSyncStorage { // Relation metadata private readonly _relationMetadataCreateCollection: Partial<RelationMetadataEntity>[] = []; + private readonly _relationMetadataUpdateCollection: Partial<RelationMetadataEntity>[] = + []; private readonly _relationMetadataDeleteCollection: RelationMetadataEntity[] = []; - private readonly _relationMetadataUpdateCollection: Partial<RelationMetadataEntity>[] = + + // Index metadata + private readonly _indexMetadataCreateCollection: Partial<IndexMetadataEntity>[] = []; + private readonly _indexMetadataUpdateCollection: Partial<IndexMetadataEntity>[] = + []; + private readonly _indexMetadataDeleteCollection: IndexMetadataEntity[] = []; constructor() {} @@ -68,7 +80,21 @@ export class WorkspaceSyncStorage { return this._relationMetadataDeleteCollection; } - addCreateObjectMetadata(object: ComputedPartialWorkspaceEntity) { + get indexMetadataCreateCollection() { + return this._indexMetadataCreateCollection; + } + + get indexMetadataUpdateCollection() { + return this._indexMetadataUpdateCollection; + } + + get indexMetadataDeleteCollection() { + return this._indexMetadataDeleteCollection; + } + + addCreateObjectMetadata( + object: Omit<ComputedPartialWorkspaceEntity, 'fields'>, + ) { this._objectMetadataCreateCollection.push(object); } @@ -107,4 +133,16 @@ export class WorkspaceSyncStorage { addDeleteRelationMetadata(relation: RelationMetadataEntity) { this._relationMetadataDeleteCollection.push(relation); } + + addCreateIndexMetadata(index: Partial<IndexMetadataEntity>) { + this._indexMetadataCreateCollection.push(index); + } + + addUpdateIndexMetadata(index: Partial<IndexMetadataEntity>) { + this._indexMetadataUpdateCollection.push(index); + } + + addDeleteIndexMetadata(index: IndexMetadataEntity) { + this._indexMetadataDeleteCollection.push(index); + } } diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/types/object-record.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/types/object-record.ts deleted file mode 100644 index d517a6aeda88..000000000000 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/types/object-record.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { ObjectLiteral } from 'typeorm'; - -import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; - -export type ObjectRecord<T extends ObjectLiteral> = { - [K in keyof T as T[K] extends BaseWorkspaceEntity - ? `${Extract<K, string>}Id` - : K]: T[K] extends BaseWorkspaceEntity - ? string - : T[K] extends BaseWorkspaceEntity[] - ? string[] - : T[K]; -} & { - [K in keyof T]: T[K] extends BaseWorkspaceEntity - ? ObjectRecord<T[K]> - : T[K] extends BaseWorkspaceEntity[] - ? ObjectRecord<T[K][number]>[] - : T[K]; -}; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-object.util.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-fields.util.ts similarity index 81% rename from packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-object.util.ts rename to packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-fields.util.ts index c45503534a77..7f1a1247bdce 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-object.util.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/utils/compute-standard-fields.util.ts @@ -1,8 +1,8 @@ import { - ComputedPartialWorkspaceEntity, - PartialWorkspaceEntity, -} from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-object-metadata.interface'; -import { ComputedPartialFieldMetadata } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-field-metadata.interface'; + ComputedPartialFieldMetadata, + PartialComputedFieldMetadata, + PartialFieldMetadata, +} from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/partial-field-metadata.interface'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; @@ -11,16 +11,18 @@ import { createRelationDeterministicUuid, } from 'src/engine/workspace-manager/workspace-sync-metadata/utils/create-deterministic-uuid.util'; -export const computeStandardObject = ( - standardObjectMetadata: Omit<PartialWorkspaceEntity, 'standardId'> & { - standardId: string | null; - }, +export const computeStandardFields = ( + standardFieldMetadataCollection: ( + | PartialFieldMetadata + | PartialComputedFieldMetadata + )[], originalObjectMetadata: ObjectMetadataEntity, customObjectMetadataCollection: ObjectMetadataEntity[] = [], -): ComputedPartialWorkspaceEntity => { +): ComputedPartialFieldMetadata[] => { const fields: ComputedPartialFieldMetadata[] = []; - for (const partialFieldMetadata of standardObjectMetadata.fields) { + for (const partialFieldMetadata of standardFieldMetadataCollection) { + // Relation from standard object to custom object if ('argsFactory' in partialFieldMetadata) { // Compute standard fields of custom object for (const customObjectMetadata of customObjectMetadataCollection) { @@ -63,6 +65,7 @@ export const computeStandardObject = ( }); } } else { + // Relation from standard object to standard object const labelText = typeof partialFieldMetadata.label === 'function' ? partialFieldMetadata.label(originalObjectMetadata) @@ -80,8 +83,5 @@ export const computeStandardObject = ( } } - return { - ...standardObjectMetadata, - fields, - }; + return fields; }; diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module.ts index 7e8d046cbe7e..e6f2d15a1858 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.module.ts @@ -5,17 +5,18 @@ import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature- import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { RelationMetadataEntity } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; import { WorkspaceMigrationEntity } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; +import { WorkspaceMigrationBuilderModule } from 'src/engine/workspace-manager/workspace-migration-builder/workspace-migration-builder.module'; import { WorkspaceMigrationRunnerModule } from 'src/engine/workspace-manager/workspace-migration-runner/workspace-migration-runner.module'; -import { WorkspaceSyncMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service'; -import { workspaceSyncMetadataFactories } from 'src/engine/workspace-manager/workspace-sync-metadata/factories'; import { workspaceSyncMetadataComparators } from 'src/engine/workspace-manager/workspace-sync-metadata/comparators'; +import { workspaceSyncMetadataFactories } from 'src/engine/workspace-manager/workspace-sync-metadata/factories'; import { WorkspaceMetadataUpdaterService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-metadata-updater.service'; +import { WorkspaceSyncFieldMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-field-metadata.service'; +import { WorkspaceSyncIndexMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-index-metadata.service'; import { WorkspaceSyncObjectMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-object-metadata.service'; import { WorkspaceSyncRelationMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-relation-metadata.service'; -import { WorkspaceSyncFieldMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-field-metadata.service'; -import { WorkspaceMigrationBuilderModule } from 'src/engine/workspace-manager/workspace-migration-builder/workspace-migration-builder.module'; -import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.module'; +import { WorkspaceSyncMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service'; @Module({ imports: [ @@ -41,6 +42,7 @@ import { WorkspaceCacheVersionModule } from 'src/engine/metadata-modules/workspa WorkspaceSyncRelationMetadataService, WorkspaceSyncFieldMetadataService, WorkspaceSyncMetadataService, + WorkspaceSyncIndexMetadataService, ], exports: [...workspaceSyncMetadataFactories, WorkspaceSyncMetadataService], }) diff --git a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service.ts b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service.ts index 8b4db8d36ae4..b233be4e3747 100644 --- a/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service.ts +++ b/packages/twenty-server/src/engine/workspace-manager/workspace-sync-metadata/workspace-sync-metadata.service.ts @@ -13,6 +13,7 @@ import { WorkspaceSyncFieldMetadataService } from 'src/engine/workspace-manager/ import { WorkspaceSyncStorage } from 'src/engine/workspace-manager/workspace-sync-metadata/storage/workspace-sync.storage'; import { WorkspaceMigrationEntity } from 'src/engine/metadata-modules/workspace-migration/workspace-migration.entity'; import { WorkspaceCacheVersionService } from 'src/engine/metadata-modules/workspace-cache-version/workspace-cache-version.service'; +import { WorkspaceSyncIndexMetadataService } from 'src/engine/workspace-manager/workspace-sync-metadata/services/workspace-sync-index-metadata.service'; interface SynchronizeOptions { applyChanges?: boolean; @@ -31,6 +32,7 @@ export class WorkspaceSyncMetadataService { private readonly workspaceSyncRelationMetadataService: WorkspaceSyncRelationMetadataService, private readonly workspaceSyncFieldMetadataService: WorkspaceSyncFieldMetadataService, private readonly workspaceCacheVersionService: WorkspaceCacheVersionService, + private readonly workspaceSyncIndexMetadataService: WorkspaceSyncIndexMetadataService, ) {} /** @@ -97,11 +99,21 @@ export class WorkspaceSyncMetadataService { workspaceFeatureFlagsMap, ); + // 4 - Sync standard indexes on standard objects + const workspaceIndexMigrations = + await this.workspaceSyncIndexMetadataService.synchronize( + context, + manager, + storage, + workspaceFeatureFlagsMap, + ); + // Save workspace migrations into the database workspaceMigrations = await workspaceMigrationRepository.save([ ...workspaceObjectMigrations, ...workspaceFieldMigrations, ...workspaceRelationMigrations, + ...workspaceIndexMigrations, ]); // If we're running a dry run, rollback the transaction and do not execute migrations diff --git a/packages/twenty-server/src/funnelmink/funnelmink-constants.ts b/packages/twenty-server/src/funnelmink/funnelmink-constants.ts new file mode 100644 index 000000000000..92fae36f36ed --- /dev/null +++ b/packages/twenty-server/src/funnelmink/funnelmink-constants.ts @@ -0,0 +1,35 @@ +export const FUNNELMINK_ICONS = { + // objects + company: 'IconBuildingSkyscraper', + person: 'IconUser', + crew: 'IconCar', + equipment: 'IconBackhoe', + job: 'IconPlant', + material: 'IconPaperBag', + service: 'IconHeartHandshake', + workorder: 'IconChecklist', + member: 'IconUserCircle', + + // fields + activityTargets: 'IconCheckbox', + address: 'IconMap', + annualRecurringRevenue: 'IconMoneybag', + attachments: 'IconFileImport', + avatarUrl: 'IconFileUpload', + calendarEventParticipants: 'IconCalendar', + createdAt: 'IconCalendar', + description: 'IconInfoCircle', + domainName: 'IconGlobe', + email: 'IconMail', + favorite: 'IconHeart', + idealCustomerProfile: 'IconTarget', + jobTitle: 'IconBriefcase', + linkedinLink: 'IconBrandLinkedin', + name: 'IconArticle', + phone: 'IconPhone', + position: 'IconHierarchy2', + stage: 'IconTargetArrow', + stickyNote: 'IconNote', + timelineActivities: 'IconTimelineEvent', + xLink: 'IconBrandX', +}; diff --git a/packages/twenty-server/src/funnelmink/funnelmink-objects-prefill-data.ts b/packages/twenty-server/src/funnelmink/funnelmink-objects-prefill-data.ts new file mode 100644 index 000000000000..b866a3d78a03 --- /dev/null +++ b/packages/twenty-server/src/funnelmink/funnelmink-objects-prefill-data.ts @@ -0,0 +1,357 @@ +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; +import { ObjectMetadataService } from 'src/engine/metadata-modules/object-metadata/object-metadata.service'; +import { FUNNELMINK_ICONS } from 'src/funnelmink/funnelmink-constants'; +import { CreateObjectInput } from 'src/engine/metadata-modules/object-metadata/dtos/create-object.input'; +import { CreateFieldInput } from 'src/engine/metadata-modules/field-metadata/dtos/create-field.input'; +import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { FieldMetadataService } from 'src/engine/metadata-modules/field-metadata/field-metadata.service'; +import { RelationMetadataService } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.service'; +import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { CreateRelationInput } from 'src/engine/metadata-modules/relation-metadata/dtos/create-relation.input'; + +// fm TODO: this needs to be fault tolerant and reusable +// - in the future, if a user messes with their data model, we can offer to regen missing objects (and/or fields) +// - as this function iterates, it should catch and log errors without breaking out of the loop + +export const prefillWorkspaceWithFunnelminkFSMObjects = async ( + workspaceId: string, + workspaceDataSourceService: WorkspaceDataSourceService, + objectMetadataService: ObjectMetadataService, + fieldMetadataService: FieldMetadataService, + relationMetadataService: RelationMetadataService, +) => { + const workspaceDataSource = + await workspaceDataSourceService.connectToWorkspaceDataSource(workspaceId); + + if (!workspaceDataSource) { + throw new Error('Could not connect to workspace data source'); + } + + // loop through create funnelmink objects and fields + await createFunnelminkObjects( + workspaceId, + objectMetadataService, + fieldMetadataService, + ); + + await createFunnelminkRelations( + workspaceId, + objectMetadataService, + relationMetadataService, + ); + + // fm TODO: prefill data (see `standard-objects-prefill-data.ts`) + + // fm TODO: prefill views (see `standard-objects-prefill-data.ts`) +}; + +const createFunnelminkObjects = async ( + workspaceId: string, + objectMetadataService: ObjectMetadataService, + fieldMetadataService: FieldMetadataService, +) => { + const metadatas: ObjectMetadataEntity[] = []; + + // loop through, create objects, fields and relations + for (const object of fsmObjects) { + const input: CreateObjectInput = { + dataSourceId: '', + workspaceId: workspaceId, + nameSingular: object.nameSingular, + namePlural: object.namePlural, + labelSingular: object.labelSingular, + labelPlural: object.labelPlural, + description: object.description, + icon: object.icon, + isRemote: object.isRemote, + }; + const objectMetadata = await objectMetadataService.createOne(input); + + console.log( + `[fm] Created object ${object.nameSingular} with ID ${objectMetadata.id}`, + ); + for (const field of object.fields) { + const fieldInput: CreateFieldInput = { + workspaceId: workspaceId, + objectMetadataId: objectMetadata.id, + name: field.name, + label: field.label, + type: field.type, + icon: field.icon, + isRemoteCreation: false, + }; + + await fieldMetadataService.createOne(fieldInput); + } + metadatas.push(objectMetadata); + } +}; + +const createFunnelminkRelations = async ( + workspaceId: string, + objectMetadataService: ObjectMetadataService, + relationMetadataService: RelationMetadataService, +) => { + const objectMetadatas = + await objectMetadataService.findManyWithinWorkspace(workspaceId); + + for (const relation of fsmRelationships) { + const parentMetadata = objectMetadatas.find( + (metadata) => metadata.namePlural === relation.fromName, + ); + let toName = relation.toName; + + if (relation.toName === 'crewLead') { + toName = 'workspaceMembers'; + } + const childMetadata = objectMetadatas.find( + (metadata) => metadata.namePlural === toName, + ); + + if (!parentMetadata || !childMetadata) { + throw new Error( + `Could not find object metadata for ${relation.fromName} or ${relation.toName}`, + ); + } + + const input: CreateRelationInput = { + fromDescription: relation.description, + fromIcon: relation.fromIcon, + fromLabel: relation.fromLabel, + fromName: relation.fromName, + fromObjectMetadataId: childMetadata.id, + relationType: relation.type, + toIcon: relation.toIcon, + toLabel: relation.toLabel, + toName: relation.toName, + toObjectMetadataId: parentMetadata.id, + workspaceId: workspaceId, + }; + + console.log( + `[fm] Creating relationship from ${parentMetadata.nameSingular} to ${relation.toName}`, + ); + await relationMetadataService.createOne(input); + console.log( + `[fm] Created relationship from ${parentMetadata.nameSingular} to ${relation.toName}`, + ); + } +}; + +const fsmObjects = [ + { + nameSingular: 'workorder', + namePlural: 'workorders', + labelSingular: 'Work Order', + labelPlural: 'Work Orders', + description: 'A Work Order', + isRemote: false, + icon: FUNNELMINK_ICONS.workorder, + fields: [ + { + name: 'stickyNote', + label: 'Sticky Note', + icon: FUNNELMINK_ICONS.stickyNote, + type: FieldMetadataType.TEXT, + }, + { + name: 'description', + label: 'Description', + icon: FUNNELMINK_ICONS.description, + type: FieldMetadataType.TEXT, + }, + ], + }, + { + nameSingular: 'service', + namePlural: 'services', + labelSingular: 'Service', + labelPlural: 'Services', + description: 'A Service', + isRemote: false, + icon: FUNNELMINK_ICONS.service, + fields: [ + { + name: 'stickyNote', + label: 'Sticky Note', + icon: FUNNELMINK_ICONS.stickyNote, + type: FieldMetadataType.TEXT, + }, + { + name: 'description', + label: 'Description', + icon: FUNNELMINK_ICONS.description, + type: FieldMetadataType.TEXT, + }, + ], + }, + { + nameSingular: 'crew', + namePlural: 'crews', + labelSingular: 'Crew', + labelPlural: 'Crews', + description: 'A Crew', + isRemote: false, + icon: FUNNELMINK_ICONS.crew, + fields: [ + { + name: 'stickyNote', + label: 'Sticky Note', + icon: FUNNELMINK_ICONS.stickyNote, + type: FieldMetadataType.TEXT, + }, + ], + }, + { + nameSingular: 'equipment', + namePlural: 'equipments', + labelSingular: 'Equipment', + labelPlural: 'Equipment', + description: 'Equipment', + isRemote: false, + icon: FUNNELMINK_ICONS.equipment, + fields: [ + { + name: 'stickyNote', + label: 'Sticky Note', + icon: FUNNELMINK_ICONS.stickyNote, + type: FieldMetadataType.TEXT, + }, + ], + }, + { + nameSingular: 'material', + namePlural: 'materials', + labelSingular: 'Material', + labelPlural: 'Materials', + description: 'Material', + isRemote: false, + icon: FUNNELMINK_ICONS.material, + fields: [ + { + name: 'stickyNote', + label: 'Sticky Note', + icon: FUNNELMINK_ICONS.stickyNote, + type: FieldMetadataType.TEXT, + }, + ], + }, + { + nameSingular: 'job', + namePlural: 'jobs', + labelSingular: 'Job', + labelPlural: 'Jobs', + description: 'Job', + isRemote: false, + icon: FUNNELMINK_ICONS.job, + fields: [ + { + name: 'stickyNote', + label: 'Sticky Note', + icon: FUNNELMINK_ICONS.stickyNote, + type: FieldMetadataType.TEXT, + }, + { + name: 'description', + label: 'Description', + icon: FUNNELMINK_ICONS.description, + type: FieldMetadataType.TEXT, + }, + ], + }, +]; + +const fsmRelationships = [ + { + description: 'The Work Orders for this Company', + fromIcon: FUNNELMINK_ICONS.workorder, + fromLabel: 'Work Orders', + fromName: 'workorders', + toIcon: FUNNELMINK_ICONS.company, + toLabel: 'Company', + toName: 'companies', + type: RelationMetadataType.ONE_TO_MANY, + }, + { + description: 'The Work Orders for this Person', + fromIcon: FUNNELMINK_ICONS.workorder, + fromLabel: 'Work Orders', + fromName: 'workorders', + toIcon: FUNNELMINK_ICONS.person, + toLabel: 'Person', + toName: 'people', + type: RelationMetadataType.ONE_TO_MANY, + }, + { + description: 'The Jobs this Work Order represents', + fromIcon: FUNNELMINK_ICONS.job, + fromLabel: 'Jobs', + fromName: 'jobs', + toIcon: FUNNELMINK_ICONS.workorder, + toLabel: 'Work Orders', + toName: 'workorders', + type: RelationMetadataType.ONE_TO_MANY, + }, + // { + // description: 'The Jobs where this Service is performed', + // fromIcon: FUNNELMINK_ICONS.service, + // fromLabel: 'Services', + // fromName: 'services', + // toIcon: FUNNELMINK_ICONS.job, + // toLabel: 'Jobs', + // toName: 'jobs', + // type: RelationMetadataType.MANY_TO_MANY, + // }, + { + description: 'The Jobs this Crew is assigned to', + fromIcon: FUNNELMINK_ICONS.job, + fromLabel: 'Jobs', + fromName: 'jobs', + toIcon: FUNNELMINK_ICONS.crew, + toLabel: 'Crew', + toName: 'crews', + type: RelationMetadataType.ONE_TO_MANY, + }, + { + description: 'The Crew Lead', + fromIcon: FUNNELMINK_ICONS.crew, + fromLabel: 'Crews', + fromName: 'crews', + toIcon: FUNNELMINK_ICONS.member, + toLabel: 'Crew Lead', + toName: 'crewLead', + type: RelationMetadataType.ONE_TO_MANY, + }, + // { + // description: 'The Crew Members', + // fromIcon: FUNNELMINK_ICONS.crew, + // fromLabel: 'Crews', + // fromName: 'crews', + // toIcon: FUNNELMINK_ICONS.member, + // toLabel: 'Crew Members', + // toName: 'crewMembers', + // type: RelationMetadataType.MANY_TO_MANY, + // }, + // { + // description: 'The Jobs this Equipment is used for', + // fromIcon: FUNNELMINK_ICONS.equipment, + // fromLabel: 'Equipment', + // fromName: 'equipments', + // toIcon: FUNNELMINK_ICONS.job, + // toLabel: 'Jobs', + // toName: 'jobs', + // type: RelationMetadataType.MANY_TO_MANY, + // }, + // + // { + // description: 'The Jobs this Material is used for', + // fromIcon: FUNNELMINK_ICONS.material, + // fromLabel: 'Materials', + // fromName: 'materials', + // toIcon: FUNNELMINK_ICONS.job, + // toLabel: 'Jobs', + // toName: 'jobs', + // type: RelationMetadataType.MANY_TO_MANY, + // }, +]; diff --git a/packages/twenty-server/src/modules/activity/repositories/comment.repository.ts b/packages/twenty-server/src/modules/activity/repositories/comment.repository.ts deleted file mode 100644 index e6bf07b2b8a5..000000000000 --- a/packages/twenty-server/src/modules/activity/repositories/comment.repository.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; - -@Injectable() -export class CommentRepository { - constructor( - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - ) {} - - async deleteByAuthorId(authorId: string, workspaceId: string): Promise<void> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."comment" WHERE "authorId" = $1`, - [authorId], - workspaceId, - ); - } -} diff --git a/packages/twenty-server/src/modules/activity/standard-objects/activity-target.workspace-entity.ts b/packages/twenty-server/src/modules/activity/standard-objects/activity-target.workspace-entity.ts index c99325d137f5..dd2ef18cb94e 100644 --- a/packages/twenty-server/src/modules/activity/standard-objects/activity-target.workspace-entity.ts +++ b/packages/twenty-server/src/modules/activity/standard-objects/activity-target.workspace-entity.ts @@ -14,6 +14,7 @@ import { WorkspaceDynamicRelation } from 'src/engine/twenty-orm/decorators/works import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.activityTarget, @@ -31,12 +32,14 @@ export class ActivityTargetWorkspaceEntity extends BaseWorkspaceEntity { label: 'Activity', description: 'ActivityTarget activity', icon: 'IconNotes', - joinColumn: 'activityId', inverseSideTarget: () => ActivityWorkspaceEntity, inverseSideFieldKey: 'activityTargets', }) @WorkspaceIsNullable() - activity: Relation<ActivityWorkspaceEntity>; + activity: Relation<ActivityWorkspaceEntity> | null; + + @WorkspaceJoinColumn('activity') + activityId: string | null; @WorkspaceRelation({ standardId: ACTIVITY_TARGET_STANDARD_FIELD_IDS.person, @@ -44,12 +47,14 @@ export class ActivityTargetWorkspaceEntity extends BaseWorkspaceEntity { label: 'Person', description: 'ActivityTarget person', icon: 'IconUser', - joinColumn: 'personId', inverseSideTarget: () => PersonWorkspaceEntity, inverseSideFieldKey: 'activityTargets', }) @WorkspaceIsNullable() - person: Relation<PersonWorkspaceEntity>; + person: Relation<PersonWorkspaceEntity> | null; + + @WorkspaceJoinColumn('person') + personId: string | null; @WorkspaceRelation({ standardId: ACTIVITY_TARGET_STANDARD_FIELD_IDS.company, @@ -57,12 +62,14 @@ export class ActivityTargetWorkspaceEntity extends BaseWorkspaceEntity { label: 'Company', description: 'ActivityTarget company', icon: 'IconBuildingSkyscraper', - joinColumn: 'companyId', inverseSideTarget: () => CompanyWorkspaceEntity, inverseSideFieldKey: 'activityTargets', }) @WorkspaceIsNullable() - company: Relation<CompanyWorkspaceEntity>; + company: Relation<CompanyWorkspaceEntity> | null; + + @WorkspaceJoinColumn('company') + companyId: string | null; @WorkspaceRelation({ standardId: ACTIVITY_TARGET_STANDARD_FIELD_IDS.opportunity, @@ -70,12 +77,14 @@ export class ActivityTargetWorkspaceEntity extends BaseWorkspaceEntity { label: 'Opportunity', description: 'ActivityTarget opportunity', icon: 'IconTargetArrow', - joinColumn: 'opportunityId', inverseSideTarget: () => OpportunityWorkspaceEntity, inverseSideFieldKey: 'activityTargets', }) @WorkspaceIsNullable() - opportunity: Relation<OpportunityWorkspaceEntity>; + opportunity: Relation<OpportunityWorkspaceEntity> | null; + + @WorkspaceJoinColumn('opportunity') + opportunityId: string | null; @WorkspaceDynamicRelation({ type: RelationMetadataType.MANY_TO_ONE, diff --git a/packages/twenty-server/src/modules/activity/standard-objects/activity.workspace-entity.ts b/packages/twenty-server/src/modules/activity/standard-objects/activity.workspace-entity.ts index 8c6460eb45e3..f8d81dfebca2 100644 --- a/packages/twenty-server/src/modules/activity/standard-objects/activity.workspace-entity.ts +++ b/packages/twenty-server/src/modules/activity/standard-objects/activity.workspace-entity.ts @@ -17,6 +17,7 @@ import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.activity, @@ -64,7 +65,7 @@ export class ActivityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconCalendarEvent', }) @WorkspaceIsNullable() - reminderAt: Date; + reminderAt: Date | null; @WorkspaceField({ standardId: ACTIVITY_STANDARD_FIELD_IDS.dueAt, @@ -74,7 +75,7 @@ export class ActivityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconCalendarEvent', }) @WorkspaceIsNullable() - dueAt: Date; + dueAt: Date | null; @WorkspaceField({ standardId: ACTIVITY_STANDARD_FIELD_IDS.completedAt, @@ -84,7 +85,7 @@ export class ActivityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconCheck', }) @WorkspaceIsNullable() - completedAt: Date; + completedAt: Date | null; @WorkspaceRelation({ standardId: ACTIVITY_STANDARD_FIELD_IDS.activityTargets, @@ -131,10 +132,12 @@ export class ActivityWorkspaceEntity extends BaseWorkspaceEntity { inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'authoredActivities', onDelete: RelationOnDeleteAction.SET_NULL, - joinColumn: 'authorId', }) @WorkspaceIsNullable() - author: Relation<WorkspaceMemberWorkspaceEntity>; + author: Relation<WorkspaceMemberWorkspaceEntity> | null; + + @WorkspaceJoinColumn('author') + authorId: string | null; @WorkspaceRelation({ standardId: ACTIVITY_STANDARD_FIELD_IDS.assignee, @@ -145,8 +148,10 @@ export class ActivityWorkspaceEntity extends BaseWorkspaceEntity { inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'assignedActivities', onDelete: RelationOnDeleteAction.SET_NULL, - joinColumn: 'assigneeId', }) @WorkspaceIsNullable() - assignee: Relation<WorkspaceMemberWorkspaceEntity>; + assignee: Relation<WorkspaceMemberWorkspaceEntity> | null; + + @WorkspaceJoinColumn('assignee') + assigneeId: string | null; } diff --git a/packages/twenty-server/src/modules/activity/standard-objects/comment.workspace-entity.ts b/packages/twenty-server/src/modules/activity/standard-objects/comment.workspace-entity.ts index b198e36da399..a572f5d074ad 100644 --- a/packages/twenty-server/src/modules/activity/standard-objects/comment.workspace-entity.ts +++ b/packages/twenty-server/src/modules/activity/standard-objects/comment.workspace-entity.ts @@ -11,6 +11,7 @@ import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.comment, @@ -37,21 +38,25 @@ export class CommentWorkspaceEntity extends BaseWorkspaceEntity { label: 'Author', description: 'Comment author', icon: 'IconCircleUser', - joinColumn: 'authorId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'authoredComments', }) author: Relation<WorkspaceMemberWorkspaceEntity>; + @WorkspaceJoinColumn('author') + authorId: string; + @WorkspaceRelation({ standardId: COMMENT_STANDARD_FIELD_IDS.activity, type: RelationMetadataType.MANY_TO_ONE, label: 'Activity', description: 'Comment activity', icon: 'IconNotes', - joinColumn: 'activityId', inverseSideTarget: () => ActivityWorkspaceEntity, inverseSideFieldKey: 'comments', }) activity: Relation<ActivityWorkspaceEntity>; + + @WorkspaceJoinColumn('activity') + activityId: string; } diff --git a/packages/twenty-server/src/modules/api-key/standard-objects/api-key.workspace-entity.ts b/packages/twenty-server/src/modules/api-key/standard-objects/api-key.workspace-entity.ts index 40159e093a85..82ca9d107d6b 100644 --- a/packages/twenty-server/src/modules/api-key/standard-objects/api-key.workspace-entity.ts +++ b/packages/twenty-server/src/modules/api-key/standard-objects/api-key.workspace-entity.ts @@ -45,5 +45,5 @@ export class ApiKeyWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconCalendar', }) @WorkspaceIsNullable() - revokedAt?: Date; + revokedAt?: Date | null; } diff --git a/packages/twenty-server/src/modules/attachment/repositories/attachment.repository.ts b/packages/twenty-server/src/modules/attachment/repositories/attachment.repository.ts deleted file mode 100644 index 2d340d2cba0d..000000000000 --- a/packages/twenty-server/src/modules/attachment/repositories/attachment.repository.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; - -@Injectable() -export class AttachmentRepository { - constructor( - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - ) {} - - async deleteByAuthorId(authorId: string, workspaceId: string): Promise<void> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."attachment" WHERE "authorId" = $1`, - [authorId], - workspaceId, - ); - } -} diff --git a/packages/twenty-server/src/modules/attachment/standard-objects/attachment.workspace-entity.ts b/packages/twenty-server/src/modules/attachment/standard-objects/attachment.workspace-entity.ts index 42af628b9e03..a4ea0b2e7d14 100644 --- a/packages/twenty-server/src/modules/attachment/standard-objects/attachment.workspace-entity.ts +++ b/packages/twenty-server/src/modules/attachment/standard-objects/attachment.workspace-entity.ts @@ -18,6 +18,7 @@ import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metad import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; import { WorkspaceDynamicRelation } from 'src/engine/twenty-orm/decorators/workspace-dynamic-relation.decorator'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.attachment, @@ -63,24 +64,28 @@ export class AttachmentWorkspaceEntity extends BaseWorkspaceEntity { label: 'Author', description: 'Attachment author', icon: 'IconCircleUser', - joinColumn: 'authorId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'authoredAttachments', }) author: Relation<WorkspaceMemberWorkspaceEntity>; + @WorkspaceJoinColumn('author') + authorId: string; + @WorkspaceRelation({ standardId: ATTACHMENT_STANDARD_FIELD_IDS.activity, type: RelationMetadataType.MANY_TO_ONE, label: 'Activity', description: 'Attachment activity', icon: 'IconNotes', - joinColumn: 'activityId', inverseSideTarget: () => ActivityWorkspaceEntity, inverseSideFieldKey: 'attachments', }) @WorkspaceIsNullable() - activity: Relation<ActivityWorkspaceEntity>; + activity: Relation<ActivityWorkspaceEntity> | null; + + @WorkspaceJoinColumn('activity') + activityId: string | null; @WorkspaceRelation({ standardId: ATTACHMENT_STANDARD_FIELD_IDS.person, @@ -88,12 +93,14 @@ export class AttachmentWorkspaceEntity extends BaseWorkspaceEntity { label: 'Person', description: 'Attachment person', icon: 'IconUser', - joinColumn: 'personId', inverseSideTarget: () => PersonWorkspaceEntity, inverseSideFieldKey: 'attachments', }) @WorkspaceIsNullable() - person: Relation<PersonWorkspaceEntity>; + person: Relation<PersonWorkspaceEntity> | null; + + @WorkspaceJoinColumn('person') + personId: string | null; @WorkspaceRelation({ standardId: ATTACHMENT_STANDARD_FIELD_IDS.company, @@ -101,12 +108,14 @@ export class AttachmentWorkspaceEntity extends BaseWorkspaceEntity { label: 'Company', description: 'Attachment company', icon: 'IconBuildingSkyscraper', - joinColumn: 'companyId', inverseSideTarget: () => CompanyWorkspaceEntity, inverseSideFieldKey: 'attachments', }) @WorkspaceIsNullable() - company: Relation<CompanyWorkspaceEntity>; + company: Relation<CompanyWorkspaceEntity> | null; + + @WorkspaceJoinColumn('company') + companyId: string | null; @WorkspaceRelation({ standardId: ATTACHMENT_STANDARD_FIELD_IDS.opportunity, @@ -114,12 +123,14 @@ export class AttachmentWorkspaceEntity extends BaseWorkspaceEntity { label: 'Opportunity', description: 'Attachment opportunity', icon: 'IconBuildingSkyscraper', - joinColumn: 'opportunityId', inverseSideTarget: () => OpportunityWorkspaceEntity, inverseSideFieldKey: 'attachments', }) @WorkspaceIsNullable() - opportunity: Relation<OpportunityWorkspaceEntity>; + opportunity: Relation<OpportunityWorkspaceEntity> | null; + + @WorkspaceJoinColumn('opportunity') + opportunityId: string | null; @WorkspaceDynamicRelation({ type: RelationMetadataType.MANY_TO_ONE, diff --git a/packages/twenty-server/src/modules/connected-account/services/blocklist/blocklist-validation.module.ts b/packages/twenty-server/src/modules/blocklist/blocklist-validation-manager/blocklist-validation-manager.module.ts similarity index 64% rename from packages/twenty-server/src/modules/connected-account/services/blocklist/blocklist-validation.module.ts rename to packages/twenty-server/src/modules/blocklist/blocklist-validation-manager/blocklist-validation-manager.module.ts index a76112fd8fc9..6d4678dc210e 100644 --- a/packages/twenty-server/src/modules/connected-account/services/blocklist/blocklist-validation.module.ts +++ b/packages/twenty-server/src/modules/blocklist/blocklist-validation-manager/blocklist-validation-manager.module.ts @@ -1,8 +1,8 @@ import { Module } from '@nestjs/common'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { BlocklistValidationService } from 'src/modules/connected-account/services/blocklist/blocklist-validation.service'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; +import { BlocklistValidationService } from 'src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; @Module({ @@ -15,4 +15,4 @@ import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/sta providers: [BlocklistValidationService], exports: [BlocklistValidationService], }) -export class BlocklistValidationModule {} +export class BlocklistValidationManagerModule {} diff --git a/packages/twenty-server/src/modules/connected-account/services/blocklist/blocklist-validation.service.ts b/packages/twenty-server/src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service.ts similarity index 95% rename from packages/twenty-server/src/modules/connected-account/services/blocklist/blocklist-validation.service.ts rename to packages/twenty-server/src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service.ts index efa8d727997a..eb518f3f4c99 100644 --- a/packages/twenty-server/src/modules/connected-account/services/blocklist/blocklist-validation.service.ts +++ b/packages/twenty-server/src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service.ts @@ -9,8 +9,8 @@ import { import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { isDomain } from 'src/engine/utils/is-domain'; -import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; +import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; diff --git a/packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-create-many.pre-query.hook.ts b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-create-many.pre-query.hook.ts similarity index 51% rename from packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-create-many.pre-query.hook.ts rename to packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-create-many.pre-query.hook.ts index 30e025de1079..b8118ec12aa9 100644 --- a/packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-create-many.pre-query.hook.ts +++ b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-create-many.pre-query.hook.ts @@ -1,15 +1,16 @@ -import { Injectable } from '@nestjs/common'; - -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; import { CreateManyResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; import { BlocklistItem, BlocklistValidationService, -} from 'src/modules/connected-account/services/blocklist/blocklist-validation.service'; +} from 'src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service'; -@Injectable() -export class BlocklistCreateManyPreQueryHook implements WorkspacePreQueryHook { +@WorkspaceQueryHook(`blocklist.createMany`) +export class BlocklistCreateManyPreQueryHook + implements WorkspaceQueryHookInstance +{ constructor( private readonly blocklistValidationService: BlocklistValidationService, ) {} diff --git a/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-query-hook.module.ts b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-query-hook.module.ts new file mode 100644 index 000000000000..3a03459c85ec --- /dev/null +++ b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-query-hook.module.ts @@ -0,0 +1,16 @@ +import { Module } from '@nestjs/common'; + +import { BlocklistValidationManagerModule } from 'src/modules/blocklist/blocklist-validation-manager/blocklist-validation-manager.module'; +import { BlocklistCreateManyPreQueryHook } from 'src/modules/blocklist/query-hooks/blocklist-create-many.pre-query.hook'; +import { BlocklistUpdateManyPreQueryHook } from 'src/modules/blocklist/query-hooks/blocklist-update-many.pre-query.hook'; +import { BlocklistUpdateOnePreQueryHook } from 'src/modules/blocklist/query-hooks/blocklist-update-one.pre-query.hook'; + +@Module({ + imports: [BlocklistValidationManagerModule], + providers: [ + BlocklistCreateManyPreQueryHook, + BlocklistUpdateManyPreQueryHook, + BlocklistUpdateOnePreQueryHook, + ], +}) +export class BlocklistQueryHookModule {} diff --git a/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-update-many.pre-query.hook.ts b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-update-many.pre-query.hook.ts new file mode 100644 index 000000000000..28be74cfc6dc --- /dev/null +++ b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-update-many.pre-query.hook.ts @@ -0,0 +1,16 @@ +import { MethodNotAllowedException } from '@nestjs/common'; + +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; + +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; + +@WorkspaceQueryHook(`blocklist.updateMany`) +export class BlocklistUpdateManyPreQueryHook + implements WorkspaceQueryHookInstance +{ + constructor() {} + + async execute(): Promise<void> { + throw new MethodNotAllowedException('Method not allowed.'); + } +} diff --git a/packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-update-one.pre-query.hook.ts b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-update-one.pre-query.hook.ts similarity index 51% rename from packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-update-one.pre-query.hook.ts rename to packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-update-one.pre-query.hook.ts index 28c39a43b2e9..b9fdaf247bd7 100644 --- a/packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-update-one.pre-query.hook.ts +++ b/packages/twenty-server/src/modules/blocklist/query-hooks/blocklist-update-one.pre-query.hook.ts @@ -1,15 +1,16 @@ -import { Injectable } from '@nestjs/common'; - -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; import { UpdateOneResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; import { BlocklistItem, BlocklistValidationService, -} from 'src/modules/connected-account/services/blocklist/blocklist-validation.service'; +} from 'src/modules/blocklist/blocklist-validation-manager/services/blocklist-validation.service'; -@Injectable() -export class BlocklistUpdateOnePreQueryHook implements WorkspacePreQueryHook { +@WorkspaceQueryHook(`blocklist.updateOne`) +export class BlocklistUpdateOnePreQueryHook + implements WorkspaceQueryHookInstance +{ constructor( private readonly blocklistValidationService: BlocklistValidationService, ) {} diff --git a/packages/twenty-server/src/modules/connected-account/repositories/blocklist.repository.ts b/packages/twenty-server/src/modules/blocklist/repositories/blocklist.repository.ts similarity index 84% rename from packages/twenty-server/src/modules/connected-account/repositories/blocklist.repository.ts rename to packages/twenty-server/src/modules/blocklist/repositories/blocklist.repository.ts index 99ec3ecbb6a1..755997e23ff1 100644 --- a/packages/twenty-server/src/modules/connected-account/repositories/blocklist.repository.ts +++ b/packages/twenty-server/src/modules/blocklist/repositories/blocklist.repository.ts @@ -3,8 +3,7 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; @Injectable() export class BlocklistRepository { @@ -16,7 +15,7 @@ export class BlocklistRepository { id: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<BlocklistWorkspaceEntity> | null> { + ): Promise<BlocklistWorkspaceEntity | null> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -39,7 +38,7 @@ export class BlocklistRepository { workspaceMemberId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<BlocklistWorkspaceEntity>[]> { + ): Promise<BlocklistWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -56,7 +55,7 @@ export class BlocklistRepository { handle: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<BlocklistWorkspaceEntity>[]> { + ): Promise<BlocklistWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); diff --git a/packages/twenty-server/src/modules/connected-account/standard-objects/blocklist.workspace-entity.ts b/packages/twenty-server/src/modules/blocklist/standard-objects/blocklist.workspace-entity.ts similarity index 92% rename from packages/twenty-server/src/modules/connected-account/standard-objects/blocklist.workspace-entity.ts rename to packages/twenty-server/src/modules/blocklist/standard-objects/blocklist.workspace-entity.ts index fce1a5eb201d..323f211a5e35 100644 --- a/packages/twenty-server/src/modules/connected-account/standard-objects/blocklist.workspace-entity.ts +++ b/packages/twenty-server/src/modules/blocklist/standard-objects/blocklist.workspace-entity.ts @@ -11,6 +11,7 @@ import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/work import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.blocklist, @@ -38,9 +39,11 @@ export class BlocklistWorkspaceEntity extends BaseWorkspaceEntity { label: 'WorkspaceMember', description: 'WorkspaceMember', icon: 'IconCircleUser', - joinColumn: 'workspaceMemberId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'blocklist', }) workspaceMember: Relation<WorkspaceMemberWorkspaceEntity>; + + @WorkspaceJoinColumn('workspaceMember') + workspaceMemberId: string; } diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service.ts b/packages/twenty-server/src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service.ts similarity index 97% rename from packages/twenty-server/src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service.ts rename to packages/twenty-server/src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service.ts index aa6eb282a977..ef22e2a1cb3a 100644 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service.ts +++ b/packages/twenty-server/src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service.ts @@ -7,6 +7,8 @@ import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/work import { PersonRepository } from 'src/modules/person/repositories/person.repository'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; +// TODO: Move inside person module and workspace-member module + @Injectable() export class AddPersonIdAndWorkspaceMemberIdService { constructor( diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/utils/__tests__/is-email-blocklisted.util.spec.ts b/packages/twenty-server/src/modules/calendar-messaging-participant-manager/utils/__tests__/is-email-blocklisted.util.spec.ts similarity index 97% rename from packages/twenty-server/src/modules/calendar-messaging-participant/utils/__tests__/is-email-blocklisted.util.spec.ts rename to packages/twenty-server/src/modules/calendar-messaging-participant-manager/utils/__tests__/is-email-blocklisted.util.spec.ts index c7c6af34c30f..4d5cf12553a7 100644 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/utils/__tests__/is-email-blocklisted.util.spec.ts +++ b/packages/twenty-server/src/modules/calendar-messaging-participant-manager/utils/__tests__/is-email-blocklisted.util.spec.ts @@ -1,4 +1,4 @@ -import { isEmailBlocklisted } from 'src/modules/calendar-messaging-participant/utils/is-email-blocklisted.util'; +import { isEmailBlocklisted } from 'src/modules/calendar-messaging-participant-manager/utils/is-email-blocklisted.util'; describe('isEmailBlocklisted', () => { it('should return true if email is blocklisted', () => { diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/utils/is-email-blocklisted.util.ts b/packages/twenty-server/src/modules/calendar-messaging-participant-manager/utils/is-email-blocklisted.util.ts similarity index 91% rename from packages/twenty-server/src/modules/calendar-messaging-participant/utils/is-email-blocklisted.util.ts rename to packages/twenty-server/src/modules/calendar-messaging-participant-manager/utils/is-email-blocklisted.util.ts index af21bbe1ed96..1ba64ee19fc6 100644 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/utils/is-email-blocklisted.util.ts +++ b/packages/twenty-server/src/modules/calendar-messaging-participant-manager/utils/is-email-blocklisted.util.ts @@ -1,3 +1,5 @@ +// TODO: Move inside blocklist module + export const isEmailBlocklisted = ( channelHandle: string, email: string | null | undefined, diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/calendar-messaging-participant-job.module.ts b/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/calendar-messaging-participant-job.module.ts deleted file mode 100644 index a2c338dace65..000000000000 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/calendar-messaging-participant-job.module.ts +++ /dev/null @@ -1,21 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { MatchParticipantJob } from 'src/modules/calendar-messaging-participant/jobs/match-participant.job'; -import { UnmatchParticipantJob } from 'src/modules/calendar-messaging-participant/jobs/unmatch-participant.job'; -import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module'; -import { MessagingCommonModule } from 'src/modules/messaging/common/messaging-common.module'; - -@Module({ - imports: [CalendarEventParticipantModule, MessagingCommonModule], - providers: [ - { - provide: MatchParticipantJob.name, - useClass: MatchParticipantJob, - }, - { - provide: UnmatchParticipantJob.name, - useClass: UnmatchParticipantJob, - }, - ], -}) -export class CalendarMessagingParticipantJobModule {} diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/match-participant.job.ts b/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/match-participant.job.ts deleted file mode 100644 index 8f9015bfcd24..000000000000 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/match-participant.job.ts +++ /dev/null @@ -1,41 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { CalendarEventParticipantService } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service'; -import { MessagingMessageParticipantService } from 'src/modules/messaging/common/services/messaging-message-participant.service'; - -export type MatchParticipantJobData = { - workspaceId: string; - email: string; - personId?: string; - workspaceMemberId?: string; -}; - -@Injectable() -export class MatchParticipantJob - implements MessageQueueJob<MatchParticipantJobData> -{ - constructor( - private readonly messageParticipantService: MessagingMessageParticipantService, - private readonly calendarEventParticipantService: CalendarEventParticipantService, - ) {} - - async handle(data: MatchParticipantJobData): Promise<void> { - const { workspaceId, email, personId, workspaceMemberId } = data; - - await this.messageParticipantService.matchMessageParticipants( - workspaceId, - email, - personId, - workspaceMemberId, - ); - - await this.calendarEventParticipantService.matchCalendarEventParticipants( - workspaceId, - email, - personId, - workspaceMemberId, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/unmatch-participant.job.ts b/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/unmatch-participant.job.ts deleted file mode 100644 index 11fee2cda22f..000000000000 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/jobs/unmatch-participant.job.ts +++ /dev/null @@ -1,41 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { CalendarEventParticipantService } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service'; -import { MessagingMessageParticipantService } from 'src/modules/messaging/common/services/messaging-message-participant.service'; - -export type UnmatchParticipantJobData = { - workspaceId: string; - email: string; - personId?: string; - workspaceMemberId?: string; -}; - -@Injectable() -export class UnmatchParticipantJob - implements MessageQueueJob<UnmatchParticipantJobData> -{ - constructor( - private readonly messageParticipantService: MessagingMessageParticipantService, - private readonly calendarEventParticipantService: CalendarEventParticipantService, - ) {} - - async handle(data: UnmatchParticipantJobData): Promise<void> { - const { workspaceId, email, personId, workspaceMemberId } = data; - - await this.messageParticipantService.unmatchMessageParticipants( - workspaceId, - email, - personId, - workspaceMemberId, - ); - - await this.calendarEventParticipantService.unmatchCalendarEventParticipants( - workspaceId, - email, - personId, - workspaceMemberId, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.module.ts b/packages/twenty-server/src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.module.ts deleted file mode 100644 index eac7dd9c797d..000000000000 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.module.ts +++ /dev/null @@ -1,16 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; -import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; - -@Module({ - imports: [ - WorkspaceDataSourceModule, - ObjectMetadataRepositoryModule.forFeature([PersonWorkspaceEntity]), - ], - providers: [AddPersonIdAndWorkspaceMemberIdService], - exports: [AddPersonIdAndWorkspaceMemberIdService], -}) -export class AddPersonIdAndWorkspaceMemberIdModule {} diff --git a/packages/twenty-server/src/modules/calendar/blocklist-manager/calendar-blocklist-manager.module.ts b/packages/twenty-server/src/modules/calendar/blocklist-manager/calendar-blocklist-manager.module.ts new file mode 100644 index 000000000000..57497f1d279e --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/blocklist-manager/calendar-blocklist-manager.module.ts @@ -0,0 +1,18 @@ +import { Module } from '@nestjs/common'; + +import { BlocklistItemDeleteCalendarEventsJob } from 'src/modules/calendar/blocklist-manager/jobs/blocklist-item-delete-calendar-events.job'; +import { BlocklistReimportCalendarEventsJob } from 'src/modules/calendar/blocklist-manager/jobs/blocklist-reimport-calendar-events.job'; +import { CalendarBlocklistListener } from 'src/modules/calendar/blocklist-manager/listeners/calendar-blocklist.listener'; +import { CalendarEventCleanerModule } from 'src/modules/calendar/calendar-event-cleaner/calendar-event-cleaner.module'; +import { CalendarEventImportManagerModule } from 'src/modules/calendar/calendar-event-import-manager/calendar-event-import-manager.module'; + +@Module({ + imports: [CalendarEventCleanerModule, CalendarEventImportManagerModule], + providers: [ + CalendarBlocklistListener, + BlocklistItemDeleteCalendarEventsJob, + BlocklistReimportCalendarEventsJob, + ], + exports: [], +}) +export class CalendarBlocklistManagerModule {} diff --git a/packages/twenty-server/src/modules/calendar/blocklist-manager/jobs/blocklist-item-delete-calendar-events.job.ts b/packages/twenty-server/src/modules/calendar/blocklist-manager/jobs/blocklist-item-delete-calendar-events.job.ts new file mode 100644 index 000000000000..7091c127a000 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/blocklist-manager/jobs/blocklist-item-delete-calendar-events.job.ts @@ -0,0 +1,101 @@ +import { Logger, Scope } from '@nestjs/common'; + +import { Any, ILike } from 'typeorm'; + +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { CalendarEventCleanerService } from 'src/modules/calendar/calendar-event-cleaner/services/calendar-event-cleaner.service'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; + +export type BlocklistItemDeleteCalendarEventsJobData = { + workspaceId: string; + blocklistItemId: string; +}; + +@Processor({ + queueName: MessageQueue.calendarQueue, + scope: Scope.REQUEST, +}) +export class BlocklistItemDeleteCalendarEventsJob { + private readonly logger = new Logger( + BlocklistItemDeleteCalendarEventsJob.name, + ); + + constructor( + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, + @InjectWorkspaceRepository(CalendarChannelEventAssociationWorkspaceEntity) + private readonly calendarChannelEventAssociationRepository: WorkspaceRepository<CalendarChannelEventAssociationWorkspaceEntity>, + @InjectObjectMetadataRepository(BlocklistWorkspaceEntity) + private readonly blocklistRepository: BlocklistRepository, + private readonly calendarEventCleanerService: CalendarEventCleanerService, + ) {} + + @Process(BlocklistItemDeleteCalendarEventsJob.name) + async handle(data: BlocklistItemDeleteCalendarEventsJobData): Promise<void> { + const { workspaceId, blocklistItemId } = data; + + const blocklistItem = await this.blocklistRepository.getById( + blocklistItemId, + workspaceId, + ); + + if (!blocklistItem) { + this.logger.log( + `Blocklist item with id ${blocklistItemId} not found in workspace ${workspaceId}`, + ); + + return; + } + + const { handle, workspaceMemberId } = blocklistItem; + + this.logger.log( + `Deleting calendar events from ${handle} in workspace ${workspaceId} for workspace member ${workspaceMemberId}`, + ); + + if (!workspaceMemberId) { + throw new Error( + `Workspace member ID is undefined for blocklist item ${blocklistItemId} in workspace ${workspaceId}`, + ); + } + + const calendarChannels = await this.calendarChannelRepository.find({ + where: { + connectedAccount: { + accountOwnerId: workspaceMemberId, + }, + }, + }); + + const calendarChannelIds = calendarChannels.map(({ id }) => id); + + const isHandleDomain = handle.startsWith('@'); + + await this.calendarChannelEventAssociationRepository.delete({ + calendarEvent: { + calendarEventParticipants: { + handle: isHandleDomain ? ILike(`%${handle}`) : handle, + }, + calendarChannelEventAssociations: { + calendarChannelId: Any(calendarChannelIds), + }, + }, + }); + + await this.calendarEventCleanerService.cleanWorkspaceCalendarEvents( + workspaceId, + ); + + this.logger.log( + `Deleted calendar events from handle ${handle} in workspace ${workspaceId} for workspace member ${workspaceMemberId}`, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/blocklist-manager/jobs/blocklist-reimport-calendar-events.job.ts b/packages/twenty-server/src/modules/calendar/blocklist-manager/jobs/blocklist-reimport-calendar-events.job.ts new file mode 100644 index 000000000000..4eef5ecdb1a9 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/blocklist-manager/jobs/blocklist-reimport-calendar-events.job.ts @@ -0,0 +1,61 @@ +import { Scope } from '@nestjs/common'; + +import { Any } from 'typeorm'; + +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { + CalendarChannelSyncStage, + CalendarChannelWorkspaceEntity, +} from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; + +export type BlocklistReimportCalendarEventsJobData = { + workspaceId: string; + workspaceMemberId: string; +}; + +@Processor({ + queueName: MessageQueue.calendarQueue, + scope: Scope.REQUEST, +}) +export class BlocklistReimportCalendarEventsJob { + constructor( + @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) + private readonly connectedAccountRepository: ConnectedAccountRepository, + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, + ) {} + + @Process(BlocklistReimportCalendarEventsJob.name) + async handle(data: BlocklistReimportCalendarEventsJobData): Promise<void> { + const { workspaceId, workspaceMemberId } = data; + + const connectedAccounts = + await this.connectedAccountRepository.getAllByWorkspaceMemberId( + workspaceMemberId, + workspaceId, + ); + + if (!connectedAccounts || connectedAccounts.length === 0) { + return; + } + + await this.calendarChannelRepository.update( + { + connectedAccountId: Any( + connectedAccounts.map((connectedAccount) => connectedAccount.id), + ), + }, + { + syncStage: + CalendarChannelSyncStage.FULL_CALENDAR_EVENT_LIST_FETCH_PENDING, + }, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/listeners/calendar-blocklist.listener.ts b/packages/twenty-server/src/modules/calendar/blocklist-manager/listeners/calendar-blocklist.listener.ts similarity index 82% rename from packages/twenty-server/src/modules/calendar/listeners/calendar-blocklist.listener.ts rename to packages/twenty-server/src/modules/calendar/blocklist-manager/listeners/calendar-blocklist.listener.ts index c1ed4f2d98eb..9c1c54c5fab3 100644 --- a/packages/twenty-server/src/modules/calendar/listeners/calendar-blocklist.listener.ts +++ b/packages/twenty-server/src/modules/calendar/blocklist-manager/listeners/calendar-blocklist.listener.ts @@ -1,25 +1,26 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { - BlocklistItemDeleteCalendarEventsJobData, BlocklistItemDeleteCalendarEventsJob, -} from 'src/modules/calendar/jobs/blocklist-item-delete-calendar-events.job'; + BlocklistItemDeleteCalendarEventsJobData, +} from 'src/modules/calendar/blocklist-manager/jobs/blocklist-item-delete-calendar-events.job'; import { - BlocklistReimportCalendarEventsJobData, BlocklistReimportCalendarEventsJob, -} from 'src/modules/calendar/jobs/blocklist-reimport-calendar-events.job'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; + BlocklistReimportCalendarEventsJobData, +} from 'src/modules/calendar/blocklist-manager/jobs/blocklist-reimport-calendar-events.job'; @Injectable() export class CalendarBlocklistListener { constructor( - @Inject(MessageQueue.calendarQueue) + @InjectMessageQueue(MessageQueue.calendarQueue) private readonly messageQueueService: MessageQueueService, ) {} @@ -45,7 +46,6 @@ export class CalendarBlocklistListener { { workspaceId: payload.workspaceId, workspaceMemberId: payload.properties.before.workspaceMember.id, - handle: payload.properties.before.handle, }, ); } @@ -67,7 +67,6 @@ export class CalendarBlocklistListener { { workspaceId: payload.workspaceId, workspaceMemberId: payload.properties.after.workspaceMember.id, - handle: payload.properties.before.handle, }, ); } diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/calendar-event-cleaner.module.ts b/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/calendar-event-cleaner.module.ts new file mode 100644 index 000000000000..50da2766148f --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/calendar-event-cleaner.module.ts @@ -0,0 +1,16 @@ +import { Module } from '@nestjs/common'; + +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { DeleteConnectedAccountAssociatedCalendarDataJob } from 'src/modules/calendar/calendar-event-cleaner/jobs/delete-connected-account-associated-calendar-data.job'; +import { CalendarEventCleanerService } from 'src/modules/calendar/calendar-event-cleaner/services/calendar-event-cleaner.service'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; + +@Module({ + imports: [TwentyORMModule.forFeature([CalendarEventWorkspaceEntity])], + providers: [ + CalendarEventCleanerService, + DeleteConnectedAccountAssociatedCalendarDataJob, + ], + exports: [CalendarEventCleanerService], +}) +export class CalendarEventCleanerModule {} diff --git a/packages/twenty-server/src/modules/calendar/jobs/delete-connected-account-associated-calendar-data.job.ts b/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/jobs/delete-connected-account-associated-calendar-data.job.ts similarity index 62% rename from packages/twenty-server/src/modules/calendar/jobs/delete-connected-account-associated-calendar-data.job.ts rename to packages/twenty-server/src/modules/calendar/calendar-event-cleaner/jobs/delete-connected-account-associated-calendar-data.job.ts index 2d41ada2fa43..481814595542 100644 --- a/packages/twenty-server/src/modules/calendar/jobs/delete-connected-account-associated-calendar-data.job.ts +++ b/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/jobs/delete-connected-account-associated-calendar-data.job.ts @@ -1,19 +1,17 @@ -import { Injectable, Logger } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { CalendarEventCleanerService } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { CalendarEventCleanerService } from 'src/modules/calendar/calendar-event-cleaner/services/calendar-event-cleaner.service'; export type DeleteConnectedAccountAssociatedCalendarDataJobData = { workspaceId: string; connectedAccountId: string; }; -@Injectable() -export class DeleteConnectedAccountAssociatedCalendarDataJob - implements - MessageQueueJob<DeleteConnectedAccountAssociatedCalendarDataJobData> -{ +@Processor(MessageQueue.calendarQueue) +export class DeleteConnectedAccountAssociatedCalendarDataJob { private readonly logger = new Logger( DeleteConnectedAccountAssociatedCalendarDataJob.name, ); @@ -22,6 +20,7 @@ export class DeleteConnectedAccountAssociatedCalendarDataJob private readonly calendarEventCleanerService: CalendarEventCleanerService, ) {} + @Process(DeleteConnectedAccountAssociatedCalendarDataJob.name) async handle( data: DeleteConnectedAccountAssociatedCalendarDataJobData, ): Promise<void> { diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/services/calendar-event-cleaner.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/services/calendar-event-cleaner.service.ts new file mode 100644 index 000000000000..0248957a612a --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-cleaner/services/calendar-event-cleaner.service.ts @@ -0,0 +1,40 @@ +import { Injectable } from '@nestjs/common'; + +import { Any, IsNull } from 'typeorm'; + +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { deleteUsingPagination } from 'src/modules/messaging/message-cleaner/utils/delete-using-pagination.util'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; + +@Injectable() +export class CalendarEventCleanerService { + constructor( + @InjectWorkspaceRepository(CalendarEventWorkspaceEntity) + private readonly calendarEventRepository: WorkspaceRepository<CalendarEventWorkspaceEntity>, + ) {} + + public async cleanWorkspaceCalendarEvents(workspaceId: string) { + await deleteUsingPagination( + workspaceId, + 500, + async (limit, offset) => { + const nonAssociatedCalendarEvents = + await this.calendarEventRepository.find({ + where: { + calendarChannelEventAssociations: { + id: IsNull(), + }, + }, + take: limit, + skip: offset, + }); + + return nonAssociatedCalendarEvents.map(({ id }) => id); + }, + async (ids) => { + await this.calendarEventRepository.delete({ id: Any(ids) }); + }, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/calendar-event-import-manager.module.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/calendar-event-import-manager.module.ts new file mode 100644 index 000000000000..cfcee7cad740 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/calendar-event-import-manager.module.ts @@ -0,0 +1,69 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { BillingModule } from 'src/engine/core-modules/billing/billing.module'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; +import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { CalendarEventCleanerModule } from 'src/modules/calendar/calendar-event-cleaner/calendar-event-cleaner.module'; +import { CalendarEventListFetchCronCommand } from 'src/modules/calendar/calendar-event-import-manager/crons/commands/calendar-event-list-fetch.cron.command'; +import { CalendarEventListFetchCronJob } from 'src/modules/calendar/calendar-event-import-manager/crons/jobs/calendar-event-list-fetch.cron.job'; +import { GoogleCalendarDriverModule } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/google-calendar-driver.module'; +import { CalendarEventListFetchJob } from 'src/modules/calendar/calendar-event-import-manager/jobs/calendar-event-list-fetch.job'; +import { CalendarChannelSyncStatusService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-channel-sync-status.service'; +import { CalendarEventImportErrorHandlerService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-event-import-error-handling.service'; +import { CalendarEventsImportService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-events-import.service'; +import { CalendarGetCalendarEventsService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-get-events.service'; +import { CalendarSaveEventsService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-save-events.service'; +import { CalendarEventParticipantManagerModule } from 'src/modules/calendar/calendar-event-participant-manager/calendar-event-participant-manager.module'; +import { CalendarCommonModule } from 'src/modules/calendar/common/calendar-common.module'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; +import { RefreshAccessTokenManagerModule } from 'src/modules/connected-account/refresh-access-token-manager/refresh-access-token-manager.module'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; + +@Module({ + imports: [ + TwentyORMModule.forFeature([ + CalendarEventWorkspaceEntity, + CalendarChannelWorkspaceEntity, + CalendarChannelEventAssociationWorkspaceEntity, + CalendarEventParticipantWorkspaceEntity, + ]), + ObjectMetadataRepositoryModule.forFeature([ + ConnectedAccountWorkspaceEntity, + BlocklistWorkspaceEntity, + PersonWorkspaceEntity, + WorkspaceMemberWorkspaceEntity, + ]), + CalendarEventParticipantManagerModule, + TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), + TypeOrmModule.forFeature([DataSourceEntity], 'metadata'), + WorkspaceDataSourceModule, + CalendarEventCleanerModule, + GoogleCalendarDriverModule, + BillingModule, + RefreshAccessTokenManagerModule, + CalendarCommonModule, + CalendarEventParticipantManagerModule, + ], + providers: [ + CalendarChannelSyncStatusService, + CalendarEventsImportService, + CalendarEventImportErrorHandlerService, + CalendarGetCalendarEventsService, + CalendarSaveEventsService, + CalendarEventListFetchCronJob, + CalendarEventListFetchCronCommand, + CalendarEventListFetchJob, + ], + exports: [CalendarEventsImportService], +}) +export class CalendarEventImportManagerModule {} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/constants/calendar-throttle-duration.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/constants/calendar-throttle-duration.ts new file mode 100644 index 000000000000..64028026a9ab --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/constants/calendar-throttle-duration.ts @@ -0,0 +1 @@ +export const CALENDAR_THROTTLE_DURATION = 1000 * 60 * 1; // 1 minute diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/constants/calendar-throttle-max-attempts.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/constants/calendar-throttle-max-attempts.ts new file mode 100644 index 000000000000..f6a3866380a0 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/constants/calendar-throttle-max-attempts.ts @@ -0,0 +1 @@ +export const CALENDAR_THROTTLE_MAX_ATTEMPTS = 4; diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/crons/commands/calendar-event-list-fetch.cron.command.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/crons/commands/calendar-event-list-fetch.cron.command.ts new file mode 100644 index 000000000000..87d139aa48d9 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/crons/commands/calendar-event-list-fetch.cron.command.ts @@ -0,0 +1,31 @@ +import { Command, CommandRunner } from 'nest-commander'; + +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { CalendarEventListFetchCronJob } from 'src/modules/calendar/calendar-event-import-manager/crons/jobs/calendar-event-list-fetch.cron.job'; + +const CALENDAR_EVENTS_IMPORT_CRON_PATTERN = '*/5 * * * *'; + +@Command({ + name: 'cron:calendar:calendar-event-list-fetch', + description: 'Starts a cron job to fetch the calendar event list', +}) +export class CalendarEventListFetchCronCommand extends CommandRunner { + constructor( + @InjectMessageQueue(MessageQueue.cronQueue) + private readonly messageQueueService: MessageQueueService, + ) { + super(); + } + + async run(): Promise<void> { + await this.messageQueueService.addCron<undefined>( + CalendarEventListFetchCronJob.name, + undefined, + { + repeat: { pattern: CALENDAR_EVENTS_IMPORT_CRON_PATTERN }, + }, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/crons/jobs/calendar-event-list-fetch.cron.job.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/crons/jobs/calendar-event-list-fetch.cron.job.ts new file mode 100644 index 000000000000..30c6fad2e9de --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/crons/jobs/calendar-event-list-fetch.cron.job.ts @@ -0,0 +1,78 @@ +import { InjectRepository } from '@nestjs/typeorm'; + +import { Any, In, Repository } from 'typeorm'; + +import { BillingService } from 'src/engine/core-modules/billing/billing.service'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; +import { TwentyORMManager } from 'src/engine/twenty-orm/twenty-orm.manager'; +import { + CalendarEventListFetchJob, + CalendarEventsImportJobData, +} from 'src/modules/calendar/calendar-event-import-manager/jobs/calendar-event-list-fetch.job'; +import { + CalendarChannelSyncStage, + CalendarChannelWorkspaceEntity, +} from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; + +@Processor({ + queueName: MessageQueue.cronQueue, +}) +export class CalendarEventListFetchCronJob { + constructor( + @InjectRepository(DataSourceEntity, 'metadata') + private readonly dataSourceRepository: Repository<DataSourceEntity>, + @InjectMessageQueue(MessageQueue.calendarQueue) + private readonly messageQueueService: MessageQueueService, + private readonly billingService: BillingService, + private readonly twentyORMManager: TwentyORMManager, + ) {} + + @Process(CalendarEventListFetchCronJob.name) + async handle(): Promise<void> { + const workspaceIds = + await this.billingService.getActiveSubscriptionWorkspaceIds(); + + const dataSources = await this.dataSourceRepository.find({ + where: { + workspaceId: In(workspaceIds), + }, + }); + + const workspaceIdsWithDataSources = new Set( + dataSources.map((dataSource) => dataSource.workspaceId), + ); + + for (const workspaceId of workspaceIdsWithDataSources) { + const calendarChannelRepository = + await this.twentyORMManager.getRepositoryForWorkspace( + workspaceId, + CalendarChannelWorkspaceEntity, + ); + + const calendarChannels = await calendarChannelRepository.find({ + where: { + isSyncEnabled: true, + syncStage: Any([ + CalendarChannelSyncStage.FULL_CALENDAR_EVENT_LIST_FETCH_PENDING, + CalendarChannelSyncStage.PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING, + ]), + }, + }); + + for (const calendarChannel of calendarChannels) { + await this.messageQueueService.add<CalendarEventsImportJobData>( + CalendarEventListFetchJob.name, + { + calendarChannelId: calendarChannel.id, + workspaceId, + }, + ); + } + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/google-calendar-driver.module.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/google-calendar-driver.module.ts new file mode 100644 index 000000000000..46b209ef9af4 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/google-calendar-driver.module.ts @@ -0,0 +1,13 @@ +import { Module } from '@nestjs/common'; + +import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module'; +import { GoogleCalendarClientProvider } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/providers/google-calendar.provider'; +import { GoogleCalendarGetEventsService } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/services/google-calendar-get-events.service'; +import { OAuth2ClientManagerModule } from 'src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module'; + +@Module({ + imports: [EnvironmentModule, OAuth2ClientManagerModule], + providers: [GoogleCalendarClientProvider, GoogleCalendarGetEventsService], + exports: [GoogleCalendarGetEventsService], +}) +export class GoogleCalendarDriverModule {} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/providers/google-calendar.provider.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/providers/google-calendar.provider.ts new file mode 100644 index 000000000000..e91758494902 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/providers/google-calendar.provider.ts @@ -0,0 +1,30 @@ +import { Injectable } from '@nestjs/common'; + +import { calendar_v3 as calendarV3, google } from 'googleapis'; + +import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Injectable() +export class GoogleCalendarClientProvider { + constructor( + private readonly oAuth2ClientManagerService: OAuth2ClientManagerService, + ) {} + + public async getGoogleCalendarClient( + connectedAccount: Pick< + ConnectedAccountWorkspaceEntity, + 'provider' | 'refreshToken' + >, + ): Promise<calendarV3.Calendar> { + const oAuth2Client = + await this.oAuth2ClientManagerService.getOAuth2Client(connectedAccount); + + const googleCalendarClient = google.calendar({ + version: 'v3', + auth: oAuth2Client, + }); + + return googleCalendarClient; + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/services/google-calendar-get-events.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/services/google-calendar-get-events.service.ts new file mode 100644 index 000000000000..20f3ec63704d --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/services/google-calendar-get-events.service.ts @@ -0,0 +1,110 @@ +import { Injectable } from '@nestjs/common'; + +import { GaxiosError } from 'gaxios'; +import { calendar_v3 as calendarV3 } from 'googleapis'; + +import { GoogleCalendarClientProvider } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/providers/google-calendar.provider'; +import { GoogleCalendarError } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/types/google-calendar-error.type'; +import { formatGoogleCalendarEvents } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/format-google-calendar-event.util'; +import { parseGaxiosError } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-gaxios-error.util'; +import { parseGoogleCalendarError } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-google-calendar-error.util'; +import { GetCalendarEventsResponse } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-get-events.service'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Injectable() +export class GoogleCalendarGetEventsService { + constructor( + private readonly googleCalendarClientProvider: GoogleCalendarClientProvider, + ) {} + + public async getCalendarEvents( + connectedAccount: Pick< + ConnectedAccountWorkspaceEntity, + 'provider' | 'refreshToken' | 'id' + >, + syncCursor?: string, + ): Promise<GetCalendarEventsResponse> { + const googleCalendarClient = + await this.googleCalendarClientProvider.getGoogleCalendarClient( + connectedAccount, + ); + + let nextSyncToken: string | null | undefined; + let nextPageToken: string | undefined; + const events: calendarV3.Schema$Event[] = []; + + let hasMoreEvents = true; + + while (hasMoreEvents) { + const googleCalendarEvents = await googleCalendarClient.events + .list({ + calendarId: 'primary', + maxResults: 500, + syncToken: syncCursor, + pageToken: nextPageToken, + showDeleted: true, + }) + .catch(async (error: GaxiosError) => { + this.handleError(error); + + return { + data: { + items: [], + nextSyncToken: undefined, + nextPageToken: undefined, + }, + }; + }); + + nextSyncToken = googleCalendarEvents.data.nextSyncToken; + nextPageToken = googleCalendarEvents.data.nextPageToken || undefined; + + const { items } = googleCalendarEvents.data; + + if (!items || items.length === 0) { + break; + } + + events.push(...items); + + if (!nextPageToken) { + hasMoreEvents = false; + } + } + + return { + calendarEvents: formatGoogleCalendarEvents(events), + nextSyncCursor: nextSyncToken || '', + }; + } + + private handleError(error: GaxiosError) { + if ( + error.code && + [ + 'ECONNRESET', + 'ENOTFOUND', + 'ECONNABORTED', + 'ETIMEDOUT', + 'ERR_NETWORK', + ].includes(error.code) + ) { + throw parseGaxiosError(error); + } + if (error.response?.status !== 410) { + const googleCalendarError: GoogleCalendarError = { + code: error.response?.status, + reason: + error.response?.data?.error?.errors?.[0].reason || + error.response?.data?.error || + '', + message: + error.response?.data?.error?.errors?.[0].message || + error.response?.data?.error_description || + '', + }; + + throw parseGoogleCalendarError(googleCalendarError); + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/types/google-calendar-error.type.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/types/google-calendar-error.type.ts new file mode 100644 index 000000000000..b007431ad384 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/types/google-calendar-error.type.ts @@ -0,0 +1,5 @@ +export type GoogleCalendarError = { + code?: number; + reason: string; + message: string; +}; diff --git a/packages/twenty-server/src/modules/calendar/utils/format-google-calendar-event.util.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/format-google-calendar-event.util.ts similarity index 82% rename from packages/twenty-server/src/modules/calendar/utils/format-google-calendar-event.util.ts rename to packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/format-google-calendar-event.util.ts index bb8a4523a37d..06f97e3c585c 100644 --- a/packages/twenty-server/src/modules/calendar/utils/format-google-calendar-event.util.ts +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/format-google-calendar-event.util.ts @@ -1,16 +1,17 @@ import { calendar_v3 as calendarV3 } from 'googleapis'; -import { v4 } from 'uuid'; -import { CalendarEventParticipantResponseStatus } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { CalendarEventWithParticipants } from 'src/modules/calendar/types/calendar-event'; +import { CalendarEventParticipantResponseStatus } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventWithParticipants } from 'src/modules/calendar/common/types/calendar-event'; -export const formatGoogleCalendarEvent = ( +export const formatGoogleCalendarEvents = ( + events: calendarV3.Schema$Event[], +): CalendarEventWithParticipants[] => { + return events.map(formatGoogleCalendarEvent); +}; + +const formatGoogleCalendarEvent = ( event: calendarV3.Schema$Event, - iCalUIDCalendarEventIdMap: Map<string, string>, ): CalendarEventWithParticipants => { - const id = - (event.iCalUID && iCalUIDCalendarEventIdMap.get(event.iCalUID)) ?? v4(); - const formatResponseStatus = (status: string | null | undefined) => { switch (status) { case 'accepted': @@ -25,7 +26,6 @@ export const formatGoogleCalendarEvent = ( }; return { - id, title: event.summary ?? '', isCanceled: event.status === 'cancelled', isFullDay: event.start?.dateTime == null, @@ -44,12 +44,11 @@ export const formatGoogleCalendarEvent = ( recurringEventExternalId: event.recurringEventId ?? '', participants: event.attendees?.map((attendee) => ({ - calendarEventId: id, - iCalUID: event.iCalUID ?? '', handle: attendee.email ?? '', displayName: attendee.displayName ?? '', isOrganizer: attendee.organizer === true, responseStatus: formatResponseStatus(attendee.responseStatus), })) ?? [], + status: event.status ?? '', }; }; diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-gaxios-error.util.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-gaxios-error.util.ts new file mode 100644 index 000000000000..4245ff82fe22 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-gaxios-error.util.ts @@ -0,0 +1,28 @@ +import { GaxiosError } from 'gaxios'; + +import { + CalendarEventError, + CalendarEventErrorCode, +} from 'src/modules/calendar/calendar-event-import-manager/types/calendar-event-error.type'; + +export const parseGaxiosError = (error: GaxiosError): CalendarEventError => { + const { code } = error; + + switch (code) { + case 'ECONNRESET': + case 'ENOTFOUND': + case 'ECONNABORTED': + case 'ETIMEDOUT': + case 'ERR_NETWORK': + return { + code: CalendarEventErrorCode.TEMPORARY_ERROR, + message: error.message, + }; + + default: + return { + code: CalendarEventErrorCode.UNKNOWN, + message: error.message, + }; + } +}; diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-google-calendar-error.util.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-google-calendar-error.util.ts new file mode 100644 index 000000000000..9e5667c629ca --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/utils/parse-google-calendar-error.util.ts @@ -0,0 +1,86 @@ +import { GoogleCalendarError } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/types/google-calendar-error.type'; +import { + CalendarEventError, + CalendarEventErrorCode, +} from 'src/modules/calendar/calendar-event-import-manager/types/calendar-event-error.type'; + +export const parseGoogleCalendarError = ( + error: GoogleCalendarError, +): CalendarEventError => { + const { code, reason, message } = error; + + switch (code) { + case 400: + if (reason === 'invalid_grant') { + return { + code: CalendarEventErrorCode.INSUFFICIENT_PERMISSIONS, + message, + }; + } + if (reason === 'failedPrecondition') { + return { + code: CalendarEventErrorCode.TEMPORARY_ERROR, + message, + }; + } + + return { + code: CalendarEventErrorCode.UNKNOWN, + message, + }; + + case 404: + return { + code: CalendarEventErrorCode.NOT_FOUND, + message, + }; + + case 429: + return { + code: CalendarEventErrorCode.TEMPORARY_ERROR, + message, + }; + + case 403: + if ( + reason === 'rateLimitExceeded' || + reason === 'userRateLimitExceeded' + ) { + return { + code: CalendarEventErrorCode.TEMPORARY_ERROR, + message, + }; + } else { + return { + code: CalendarEventErrorCode.INSUFFICIENT_PERMISSIONS, + message, + }; + } + + case 401: + return { + code: CalendarEventErrorCode.INSUFFICIENT_PERMISSIONS, + message, + }; + case 500: + if (reason === 'backendError') { + return { + code: CalendarEventErrorCode.TEMPORARY_ERROR, + message, + }; + } else { + return { + code: CalendarEventErrorCode.UNKNOWN, + message, + }; + } + + default: + break; + } + + return { + code: CalendarEventErrorCode.UNKNOWN, + message, + }; +}; diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/jobs/calendar-event-list-fetch.job.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/jobs/calendar-event-list-fetch.job.ts new file mode 100644 index 000000000000..34ba97feaef0 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/jobs/calendar-event-list-fetch.job.ts @@ -0,0 +1,92 @@ +import { Scope } from '@nestjs/common'; + +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { CalendarEventsImportService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-events-import.service'; +import { isThrottled } from 'src/modules/connected-account/utils/is-throttled'; +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { + CalendarChannelSyncStage, + CalendarChannelWorkspaceEntity, +} from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; + +export type CalendarEventsImportJobData = { + calendarChannelId: string; + workspaceId: string; +}; + +@Processor({ + queueName: MessageQueue.calendarQueue, + scope: Scope.REQUEST, +}) +export class CalendarEventListFetchJob { + constructor( + private readonly calendarEventsImportService: CalendarEventsImportService, + @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) + private readonly connectedAccountRepository: ConnectedAccountRepository, + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, + ) {} + + @Process(CalendarEventListFetchJob.name) + async handle(data: CalendarEventsImportJobData): Promise<void> { + const { workspaceId, calendarChannelId } = data; + + const calendarChannel = await this.calendarChannelRepository.findOne({ + where: { + id: calendarChannelId, + isSyncEnabled: true, + }, + }); + + if (!calendarChannel) { + return; + } + + if ( + isThrottled( + calendarChannel.syncStageStartedAt, + calendarChannel.throttleFailureCount, + ) + ) { + return; + } + + const connectedAccount = + await this.connectedAccountRepository.getConnectedAccountOrThrow( + workspaceId, + calendarChannel.connectedAccountId, + ); + + switch (calendarChannel.syncStage) { + case CalendarChannelSyncStage.FULL_CALENDAR_EVENT_LIST_FETCH_PENDING: + await this.calendarChannelRepository.update(calendarChannelId, { + syncCursor: '', + syncStageStartedAt: null, + }); + + await this.calendarEventsImportService.processCalendarEventsImport( + calendarChannel, + connectedAccount, + workspaceId, + ); + break; + + case CalendarChannelSyncStage.PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING: + await this.calendarEventsImportService.processCalendarEventsImport( + calendarChannel, + connectedAccount, + workspaceId, + ); + break; + + default: + break; + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-channel-sync-status.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-channel-sync-status.service.ts new file mode 100644 index 000000000000..0d6dd28fcc0c --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-channel-sync-status.service.ts @@ -0,0 +1,118 @@ +import { Injectable } from '@nestjs/common'; + +import { CacheStorageService } from 'src/engine/integrations/cache-storage/cache-storage.service'; +import { InjectCacheStorage } from 'src/engine/integrations/cache-storage/decorators/cache-storage.decorator'; +import { CacheStorageNamespace } from 'src/engine/integrations/cache-storage/types/cache-storage-namespace.enum'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { + CalendarChannelSyncStage, + CalendarChannelSyncStatus, + CalendarChannelWorkspaceEntity, +} from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; + +@Injectable() +export class CalendarChannelSyncStatusService { + constructor( + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, + @InjectCacheStorage(CacheStorageNamespace.Calendar) + private readonly cacheStorage: CacheStorageService, + ) {} + + public async scheduleFullCalendarEventListFetch(calendarChannelId: string) { + await this.calendarChannelRepository.update(calendarChannelId, { + syncStage: + CalendarChannelSyncStage.FULL_CALENDAR_EVENT_LIST_FETCH_PENDING, + }); + } + + public async schedulePartialCalendarEventListFetch( + calendarChannelId: string, + ) { + await this.calendarChannelRepository.update(calendarChannelId, { + syncStage: + CalendarChannelSyncStage.PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING, + }); + } + + public async markAsCalendarEventListFetchOngoing(calendarChannelId: string) { + await this.calendarChannelRepository.update(calendarChannelId, { + syncStage: CalendarChannelSyncStage.CALENDAR_EVENT_LIST_FETCH_ONGOING, + syncStatus: CalendarChannelSyncStatus.ONGOING, + syncStageStartedAt: new Date().toISOString(), + }); + } + + public async resetAndScheduleFullCalendarEventListFetch( + calendarChannelId: string, + workspaceId: string, + ) { + await this.cacheStorage.del( + `calendar-events-to-import:${workspaceId}:google-calendar:${calendarChannelId}`, + ); + + await this.calendarChannelRepository.update(calendarChannelId, { + syncCursor: '', + syncStageStartedAt: null, + throttleFailureCount: 0, + }); + + await this.scheduleFullCalendarEventListFetch(calendarChannelId); + } + + public async scheduleCalendarEventsImport(calendarChannelId: string) { + await this.calendarChannelRepository.update(calendarChannelId, { + syncStage: CalendarChannelSyncStage.CALENDAR_EVENTS_IMPORT_PENDING, + }); + } + + public async markAsCalendarEventsImportOngoing(calendarChannelId: string) { + await this.calendarChannelRepository.update(calendarChannelId, { + syncStage: CalendarChannelSyncStage.CALENDAR_EVENTS_IMPORT_ONGOING, + syncStatus: CalendarChannelSyncStatus.ONGOING, + }); + } + + public async markAsCompletedAndSchedulePartialMessageListFetch( + calendarChannelId: string, + ) { + await this.calendarChannelRepository.update(calendarChannelId, { + syncStage: + CalendarChannelSyncStage.PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING, + syncStatus: CalendarChannelSyncStatus.ACTIVE, + throttleFailureCount: 0, + syncStageStartedAt: null, + }); + + await this.schedulePartialCalendarEventListFetch(calendarChannelId); + } + + public async markAsFailedUnknownAndFlushCalendarEventsToImport( + calendarChannelId: string, + workspaceId: string, + ) { + await this.cacheStorage.del( + `calendar-events-to-import:${workspaceId}:google-calendar:${calendarChannelId}`, + ); + + await this.calendarChannelRepository.update(calendarChannelId, { + syncStatus: CalendarChannelSyncStatus.FAILED_UNKNOWN, + syncStage: CalendarChannelSyncStage.FAILED, + }); + } + + public async markAsFailedInsufficientPermissionsAndFlushCalendarEventsToImport( + calendarChannelId: string, + workspaceId: string, + ) { + await this.cacheStorage.del( + `calendar-events-to-import:${workspaceId}:google-calendar:${calendarChannelId}`, + ); + + await this.calendarChannelRepository.update(calendarChannelId, { + syncStatus: CalendarChannelSyncStatus.FAILED_INSUFFICIENT_PERMISSIONS, + syncStage: CalendarChannelSyncStage.FAILED, + }); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-event-import-error-handling.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-event-import-error-handling.service.ts new file mode 100644 index 000000000000..93c8b9238f80 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-event-import-error-handling.service.ts @@ -0,0 +1,144 @@ +import { Injectable } from '@nestjs/common'; + +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { CALENDAR_THROTTLE_MAX_ATTEMPTS } from 'src/modules/calendar/calendar-event-import-manager/constants/calendar-throttle-max-attempts'; +import { CalendarChannelSyncStatusService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-channel-sync-status.service'; +import { CalendarEventError } from 'src/modules/calendar/calendar-event-import-manager/types/calendar-event-error.type'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; + +export enum CalendarEventImportSyncStep { + FULL_CALENDAR_EVENT_LIST_FETCH = 'FULL_CALENDAR_EVENT_LIST_FETCH', + PARTIAL_CALENDAR_EVENT_LIST_FETCH = 'PARTIAL_CALENDAR_EVENT_LIST_FETCH', + CALENDAR_EVENTS_IMPORT = 'CALENDAR_EVENTS_IMPORT', +} + +@Injectable() +export class CalendarEventImportErrorHandlerService { + constructor( + private readonly calendarChannelSyncStatusService: CalendarChannelSyncStatusService, + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, + ) {} + + public async handleError( + error: CalendarEventError, + syncStep: CalendarEventImportSyncStep, + calendarChannel: Pick< + CalendarChannelWorkspaceEntity, + 'id' | 'throttleFailureCount' + >, + workspaceId: string, + ): Promise<void> { + switch (error.code) { + case 'NOT_FOUND': + await this.handleNotFoundError(syncStep, calendarChannel, workspaceId); + break; + case 'TEMPORARY_ERROR': + await this.handleTemporaryError(syncStep, calendarChannel, workspaceId); + break; + case 'INSUFFICIENT_PERMISSIONS': + await this.handleInsufficientPermissionsError( + calendarChannel, + workspaceId, + ); + break; + case 'UNKNOWN': + await this.handleUnknownError(error, calendarChannel, workspaceId); + break; + } + } + + private async handleTemporaryError( + syncStep: CalendarEventImportSyncStep, + calendarChannel: Pick< + CalendarChannelWorkspaceEntity, + 'id' | 'throttleFailureCount' + >, + workspaceId: string, + ): Promise<void> { + if ( + calendarChannel.throttleFailureCount >= CALENDAR_THROTTLE_MAX_ATTEMPTS + ) { + await this.calendarChannelSyncStatusService.markAsFailedUnknownAndFlushCalendarEventsToImport( + calendarChannel.id, + workspaceId, + ); + + return; + } + + await this.calendarChannelRepository.increment( + { + id: calendarChannel.id, + }, + 'throttleFailureCount', + 1, + ); + + switch (syncStep) { + case CalendarEventImportSyncStep.FULL_CALENDAR_EVENT_LIST_FETCH: + await this.calendarChannelSyncStatusService.scheduleFullCalendarEventListFetch( + calendarChannel.id, + ); + break; + + case CalendarEventImportSyncStep.PARTIAL_CALENDAR_EVENT_LIST_FETCH: + await this.calendarChannelSyncStatusService.schedulePartialCalendarEventListFetch( + calendarChannel.id, + ); + break; + + case CalendarEventImportSyncStep.CALENDAR_EVENTS_IMPORT: + await this.calendarChannelSyncStatusService.scheduleCalendarEventsImport( + calendarChannel.id, + ); + break; + + default: + break; + } + } + + private async handleInsufficientPermissionsError( + calendarChannel: Pick<CalendarChannelWorkspaceEntity, 'id'>, + workspaceId: string, + ): Promise<void> { + await this.calendarChannelSyncStatusService.markAsFailedInsufficientPermissionsAndFlushCalendarEventsToImport( + calendarChannel.id, + workspaceId, + ); + } + + private async handleUnknownError( + error: CalendarEventError, + calendarChannel: Pick<CalendarChannelWorkspaceEntity, 'id'>, + workspaceId: string, + ): Promise<void> { + await this.calendarChannelSyncStatusService.markAsFailedUnknownAndFlushCalendarEventsToImport( + calendarChannel.id, + workspaceId, + ); + + throw new Error( + `Unknown error occurred while importing calendar events for calendar channel ${calendarChannel.id} in workspace ${workspaceId}: ${error.message}`, + ); + } + + private async handleNotFoundError( + syncStep: CalendarEventImportSyncStep, + calendarChannel: Pick<CalendarChannelWorkspaceEntity, 'id'>, + workspaceId: string, + ): Promise<void> { + if ( + syncStep === CalendarEventImportSyncStep.FULL_CALENDAR_EVENT_LIST_FETCH + ) { + return; + } + + await this.calendarChannelSyncStatusService.resetAndScheduleFullCalendarEventListFetch( + calendarChannel.id, + workspaceId, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-events-import.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-events-import.service.ts new file mode 100644 index 000000000000..da1dfdddb860 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-events-import.service.ts @@ -0,0 +1,144 @@ +import { Injectable } from '@nestjs/common'; + +import { Any } from 'typeorm'; + +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { CalendarEventCleanerService } from 'src/modules/calendar/calendar-event-cleaner/services/calendar-event-cleaner.service'; +import { CalendarChannelSyncStatusService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-channel-sync-status.service'; +import { + CalendarEventImportErrorHandlerService, + CalendarEventImportSyncStep, +} from 'src/modules/calendar/calendar-event-import-manager/services/calendar-event-import-error-handling.service'; +import { + CalendarGetCalendarEventsService, + GetCalendarEventsResponse, +} from 'src/modules/calendar/calendar-event-import-manager/services/calendar-get-events.service'; +import { CalendarSaveEventsService } from 'src/modules/calendar/calendar-event-import-manager/services/calendar-save-events.service'; +import { filterEventsAndReturnCancelledEvents } from 'src/modules/calendar/calendar-event-import-manager/utils/filter-events.util'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { + CalendarChannelSyncStage, + CalendarChannelWorkspaceEntity, +} from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Injectable() +export class CalendarEventsImportService { + constructor( + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, + @InjectWorkspaceRepository(CalendarChannelEventAssociationWorkspaceEntity) + private readonly calendarChannelEventAssociationRepository: WorkspaceRepository<CalendarChannelEventAssociationWorkspaceEntity>, + @InjectObjectMetadataRepository(BlocklistWorkspaceEntity) + private readonly blocklistRepository: BlocklistRepository, + private readonly calendarEventCleanerService: CalendarEventCleanerService, + private readonly calendarChannelSyncStatusService: CalendarChannelSyncStatusService, + private readonly getCalendarEventsService: CalendarGetCalendarEventsService, + private readonly calendarSaveEventsService: CalendarSaveEventsService, + private readonly calendarEventImportErrorHandlerService: CalendarEventImportErrorHandlerService, + ) {} + + public async processCalendarEventsImport( + calendarChannel: CalendarChannelWorkspaceEntity, + connectedAccount: ConnectedAccountWorkspaceEntity, + workspaceId: string, + ): Promise<void> { + const syncStep = + calendarChannel.syncStage === + CalendarChannelSyncStage.FULL_CALENDAR_EVENT_LIST_FETCH_PENDING + ? CalendarEventImportSyncStep.FULL_CALENDAR_EVENT_LIST_FETCH + : CalendarEventImportSyncStep.PARTIAL_CALENDAR_EVENT_LIST_FETCH; + + await this.calendarChannelSyncStatusService.markAsCalendarEventListFetchOngoing( + calendarChannel.id, + ); + let calendarEvents: GetCalendarEventsResponse['calendarEvents'] = []; + let nextSyncCursor: GetCalendarEventsResponse['nextSyncCursor'] = ''; + + try { + const getCalendarEventsResponse = + await this.getCalendarEventsService.getCalendarEvents( + connectedAccount, + calendarChannel.syncCursor, + ); + + calendarEvents = getCalendarEventsResponse.calendarEvents; + nextSyncCursor = getCalendarEventsResponse.nextSyncCursor; + } catch (error) { + await this.calendarEventImportErrorHandlerService.handleError( + error, + syncStep, + calendarChannel, + workspaceId, + ); + + return; + } + + if (!calendarEvents || calendarEvents?.length === 0) { + await this.calendarChannelRepository.update( + { + id: calendarChannel.id, + }, + { + syncCursor: nextSyncCursor, + }, + ); + + await this.calendarChannelSyncStatusService.schedulePartialCalendarEventListFetch( + calendarChannel.id, + ); + } + + const blocklist = await this.blocklistRepository.getByWorkspaceMemberId( + connectedAccount.accountOwnerId, + workspaceId, + ); + + const { filteredEvents, cancelledEvents } = + filterEventsAndReturnCancelledEvents( + calendarChannel, + calendarEvents, + blocklist.map((blocklist) => blocklist.handle), + ); + + const cancelledEventExternalIds = cancelledEvents.map( + (event) => event.externalId, + ); + + await this.calendarSaveEventsService.saveCalendarEventsAndEnqueueContactCreationJob( + filteredEvents, + calendarChannel, + connectedAccount, + workspaceId, + ); + + await this.calendarChannelEventAssociationRepository.delete({ + eventExternalId: Any(cancelledEventExternalIds), + calendarChannel: { + id: calendarChannel.id, + }, + }); + + await this.calendarEventCleanerService.cleanWorkspaceCalendarEvents( + workspaceId, + ); + + await this.calendarChannelRepository.update( + { + id: calendarChannel.id, + }, + { + syncCursor: nextSyncCursor, + }, + ); + + await this.calendarChannelSyncStatusService.markAsCompletedAndSchedulePartialMessageListFetch( + calendarChannel.id, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-get-events.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-get-events.service.ts new file mode 100644 index 000000000000..d9fa121101c5 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-get-events.service.ts @@ -0,0 +1,37 @@ +import { Injectable } from '@nestjs/common'; + +import { GoogleCalendarGetEventsService as GoogleCalendarGetCalendarEventsService } from 'src/modules/calendar/calendar-event-import-manager/drivers/google-calendar/services/google-calendar-get-events.service'; +import { CalendarEventWithParticipants } from 'src/modules/calendar/common/types/calendar-event'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +export type GetCalendarEventsResponse = { + calendarEvents: CalendarEventWithParticipants[]; + nextSyncCursor: string; +}; + +@Injectable() +export class CalendarGetCalendarEventsService { + constructor( + private readonly googleCalendarGetCalendarEventsService: GoogleCalendarGetCalendarEventsService, + ) {} + + public async getCalendarEvents( + connectedAccount: Pick< + ConnectedAccountWorkspaceEntity, + 'provider' | 'refreshToken' | 'id' + >, + syncCursor?: string, + ): Promise<GetCalendarEventsResponse> { + switch (connectedAccount.provider) { + case 'google': + return this.googleCalendarGetCalendarEventsService.getCalendarEvents( + connectedAccount, + syncCursor, + ); + default: + throw new Error( + `Provider ${connectedAccount.provider} is not supported.`, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-save-events.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-save-events.service.ts new file mode 100644 index 000000000000..52ea333888fb --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/services/calendar-save-events.service.ts @@ -0,0 +1,159 @@ +import { Injectable } from '@nestjs/common'; +import { EventEmitter2 } from '@nestjs/event-emitter'; + +import { Any } from 'typeorm'; + +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { WorkspaceDataSource } from 'src/engine/twenty-orm/datasource/workspace.datasource'; +import { InjectWorkspaceDatasource } from 'src/engine/twenty-orm/decorators/inject-workspace-datasource.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { injectIdsInCalendarEvents } from 'src/modules/calendar/calendar-event-import-manager/utils/inject-ids-in-calendar-events.util'; +import { CalendarEventParticipantService } from 'src/modules/calendar/calendar-event-participant-manager/services/calendar-event-participant.service'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; +import { CalendarEventWithParticipants } from 'src/modules/calendar/common/types/calendar-event'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { + CreateCompanyAndContactJob, + CreateCompanyAndContactJobData, +} from 'src/modules/contact-creation-manager/jobs/create-company-and-contact.job'; + +@Injectable() +export class CalendarSaveEventsService { + constructor( + @InjectWorkspaceRepository(CalendarEventWorkspaceEntity) + private readonly calendarEventRepository: WorkspaceRepository<CalendarEventWorkspaceEntity>, + @InjectWorkspaceRepository(CalendarChannelEventAssociationWorkspaceEntity) + private readonly calendarChannelEventAssociationRepository: WorkspaceRepository<CalendarChannelEventAssociationWorkspaceEntity>, + @InjectWorkspaceDatasource() + private readonly workspaceDataSource: WorkspaceDataSource, + private readonly calendarEventParticipantService: CalendarEventParticipantService, + @InjectMessageQueue(MessageQueue.contactCreationQueue) + private readonly messageQueueService: MessageQueueService, + private readonly eventEmitter: EventEmitter2, + ) {} + + public async saveCalendarEventsAndEnqueueContactCreationJob( + filteredEvents: CalendarEventWithParticipants[], + calendarChannel: CalendarChannelWorkspaceEntity, + connectedAccount: ConnectedAccountWorkspaceEntity, + workspaceId: string, + ): Promise<void> { + const existingCalendarEvents = await this.calendarEventRepository.find({ + where: { + iCalUID: Any(filteredEvents.map((event) => event.iCalUID as string)), + }, + }); + + const iCalUIDCalendarEventIdMap = new Map( + existingCalendarEvents.map((calendarEvent) => [ + calendarEvent.iCalUID, + calendarEvent.id, + ]), + ); + + const calendarEventsWithIds = injectIdsInCalendarEvents( + filteredEvents, + iCalUIDCalendarEventIdMap, + ); + + // TODO: When we will be able to add unicity contraint on iCalUID, we will do a INSERT ON CONFLICT DO UPDATE + + const existingEventsICalUIDs = existingCalendarEvents.map( + (calendarEvent) => calendarEvent.iCalUID, + ); + + const eventsToSave = calendarEventsWithIds.filter( + (calendarEvent) => + !existingEventsICalUIDs.includes(calendarEvent.iCalUID), + ); + + const eventsToUpdate = calendarEventsWithIds.filter((calendarEvent) => + existingEventsICalUIDs.includes(calendarEvent.iCalUID), + ); + + const existingCalendarChannelEventAssociations = + await this.calendarChannelEventAssociationRepository.find({ + where: { + eventExternalId: Any( + calendarEventsWithIds.map((calendarEvent) => calendarEvent.id), + ), + calendarChannel: { + id: calendarChannel.id, + }, + }, + }); + + const calendarChannelEventAssociationsToSave = calendarEventsWithIds + .filter( + (calendarEvent) => + !existingCalendarChannelEventAssociations.some( + (association) => association.eventExternalId === calendarEvent.id, + ), + ) + .map((calendarEvent) => ({ + calendarEventId: calendarEvent.id, + eventExternalId: calendarEvent.externalId, + calendarChannelId: calendarChannel.id, + })); + + const participantsToSave = eventsToSave.flatMap( + (event) => event.participants, + ); + + const participantsToUpdate = eventsToUpdate.flatMap( + (event) => event.participants, + ); + + const savedCalendarEventParticipantsToEmit: CalendarEventParticipantWorkspaceEntity[] = + []; + + await this.workspaceDataSource?.transaction(async (transactionManager) => { + await this.calendarEventRepository.save( + eventsToSave, + {}, + transactionManager, + ); + + await this.calendarEventRepository.save( + eventsToUpdate, + {}, + transactionManager, + ); + + await this.calendarChannelEventAssociationRepository.save( + calendarChannelEventAssociationsToSave, + {}, + transactionManager, + ); + + await this.calendarEventParticipantService.upsertAndDeleteCalendarEventParticipants( + participantsToSave, + participantsToUpdate, + transactionManager, + ); + }); + + this.eventEmitter.emit(`calendarEventParticipant.matched`, { + workspaceId, + workspaceMemberId: connectedAccount.accountOwnerId, + calendarEventParticipants: savedCalendarEventParticipantsToEmit, + }); + + if (calendarChannel.isContactAutoCreationEnabled) { + await this.messageQueueService.add<CreateCompanyAndContactJobData>( + CreateCompanyAndContactJob.name, + { + workspaceId, + connectedAccount, + contactsToCreate: participantsToSave, + }, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/types/calendar-event-error.type.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/types/calendar-event-error.type.ts new file mode 100644 index 000000000000..7bf1b56f8d18 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/types/calendar-event-error.type.ts @@ -0,0 +1,11 @@ +export enum CalendarEventErrorCode { + NOT_FOUND = 'NOT_FOUND', + TEMPORARY_ERROR = 'TEMPORARY_ERROR', + INSUFFICIENT_PERMISSIONS = 'INSUFFICIENT_PERMISSIONS', + UNKNOWN = 'UNKNOWN', +} + +export interface CalendarEventError { + message: string; + code: CalendarEventErrorCode; +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/filter-events.util.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/filter-events.util.ts new file mode 100644 index 000000000000..b3e7e0d1d221 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/filter-events.util.ts @@ -0,0 +1,40 @@ +import { filterOutBlocklistedEvents } from 'src/modules/calendar/calendar-event-import-manager/utils/filter-out-blocklisted-events.util'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventWithParticipants } from 'src/modules/calendar/common/types/calendar-event'; + +export const filterEventsAndReturnCancelledEvents = ( + calendarChannel: Pick<CalendarChannelWorkspaceEntity, 'handle'>, + events: CalendarEventWithParticipants[], + blocklist: string[], +): { + filteredEvents: CalendarEventWithParticipants[]; + cancelledEvents: CalendarEventWithParticipants[]; +} => { + const filteredEvents = filterOutBlocklistedEvents( + calendarChannel.handle, + events, + blocklist, + ); + + return filteredEvents.reduce( + ( + acc: { + filteredEvents: CalendarEventWithParticipants[]; + cancelledEvents: CalendarEventWithParticipants[]; + }, + event, + ) => { + if (event.status === 'cancelled') { + acc.cancelledEvents.push(event); + } else { + acc.filteredEvents.push(event); + } + + return acc; + }, + { + filteredEvents: [], + cancelledEvents: [], + }, + ); +}; diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/filter-out-blocklisted-events.util.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/filter-out-blocklisted-events.util.ts new file mode 100644 index 000000000000..b28f5801c51c --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/filter-out-blocklisted-events.util.ts @@ -0,0 +1,19 @@ +import { isEmailBlocklisted } from 'src/modules/calendar-messaging-participant-manager/utils/is-email-blocklisted.util'; +import { CalendarEventWithParticipants } from 'src/modules/calendar/common/types/calendar-event'; + +export const filterOutBlocklistedEvents = ( + calendarChannelHandle: string, + events: CalendarEventWithParticipants[], + blocklist: string[], +) => { + return events.filter((event) => { + if (!event.participants) { + return true; + } + + return event.participants.every( + (attendee) => + !isEmailBlocklisted(calendarChannelHandle, attendee.handle, blocklist), + ); + }); +}; diff --git a/packages/twenty-server/src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/get-flattened-values-and-values-string-for-batch-raw-query.util.ts similarity index 100% rename from packages/twenty-server/src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util.ts rename to packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/get-flattened-values-and-values-string-for-batch-raw-query.util.ts diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/inject-ids-in-calendar-events.util.ts b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/inject-ids-in-calendar-events.util.ts new file mode 100644 index 000000000000..b41e329dc0f4 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-import-manager/utils/inject-ids-in-calendar-events.util.ts @@ -0,0 +1,31 @@ +import { v4 } from 'uuid'; + +import { + CalendarEventWithParticipants, + CalendarEventWithParticipantsAndCalendarEventId, +} from 'src/modules/calendar/common/types/calendar-event'; + +export const injectIdsInCalendarEvents = ( + calendarEvents: CalendarEventWithParticipants[], + iCalUIDCalendarEventIdMap: Map<string, string>, +): CalendarEventWithParticipantsAndCalendarEventId[] => { + return calendarEvents.map((calendarEvent) => { + const id = iCalUIDCalendarEventIdMap.get(calendarEvent.iCalUID) ?? v4(); + + return injectIdInCalendarEvent(calendarEvent, id); + }); +}; + +const injectIdInCalendarEvent = ( + calendarEvent: CalendarEventWithParticipants, + id: string, +): CalendarEventWithParticipantsAndCalendarEventId => { + return { + ...calendarEvent, + id, + participants: calendarEvent.participants.map((participant) => ({ + ...participant, + calendarEventId: id, + })), + }; +}; diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/calendar-event-participant-manager.module.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/calendar-event-participant-manager.module.ts new file mode 100644 index 000000000000..609c83429af4 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/calendar-event-participant-manager.module.ts @@ -0,0 +1,46 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { FieldMetadataEntity } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; +import { CalendarCreateCompanyAndContactAfterSyncJob } from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-create-company-and-contact-after-sync.job'; +import { CalendarEventParticipantMatchParticipantJob } from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-match-participant.job'; +import { CalendarEventParticipantUnmatchParticipantJob } from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-unmatch-participant.job'; +import { CalendarEventParticipantPersonListener } from 'src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-person.listener'; +import { CalendarEventParticipantWorkspaceMemberListener } from 'src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-workspace-member.listener'; +import { CalendarEventParticipantListener } from 'src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant.listener'; +import { CalendarEventParticipantService } from 'src/modules/calendar/calendar-event-participant-manager/services/calendar-event-participant.service'; +import { CalendarCommonModule } from 'src/modules/calendar/common/calendar-common.module'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { ContactCreationManagerModule } from 'src/modules/contact-creation-manager/contact-creation-manager.module'; +import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; + +@Module({ + imports: [ + WorkspaceDataSourceModule, + TwentyORMModule.forFeature([CalendarEventParticipantWorkspaceEntity]), + ObjectMetadataRepositoryModule.forFeature([PersonWorkspaceEntity]), + TypeOrmModule.forFeature( + [ObjectMetadataEntity, FieldMetadataEntity], + 'metadata', + ), + ContactCreationManagerModule, + CalendarCommonModule, + ], + providers: [ + CalendarEventParticipantService, + CalendarCreateCompanyAndContactAfterSyncJob, + CalendarEventParticipantMatchParticipantJob, + CalendarEventParticipantUnmatchParticipantJob, + CalendarEventParticipantListener, + CalendarEventParticipantPersonListener, + CalendarEventParticipantWorkspaceMemberListener, + AddPersonIdAndWorkspaceMemberIdService, + ], + exports: [CalendarEventParticipantService], +}) +export class CalendarEventParticipantManagerModule {} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-create-company-and-contact-after-sync.job.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-create-company-and-contact-after-sync.job.ts new file mode 100644 index 000000000000..372dde104575 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-create-company-and-contact-after-sync.job.ts @@ -0,0 +1,99 @@ +import { Logger, Scope } from '@nestjs/common'; + +import { IsNull } from 'typeorm'; + +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CreateCompanyAndContactService } from 'src/modules/contact-creation-manager/services/create-company-and-contact.service'; + +export type CalendarCreateCompanyAndContactAfterSyncJobData = { + workspaceId: string; + calendarChannelId: string; +}; + +@Processor({ + queueName: MessageQueue.calendarQueue, + scope: Scope.REQUEST, +}) +export class CalendarCreateCompanyAndContactAfterSyncJob { + private readonly logger = new Logger( + CalendarCreateCompanyAndContactAfterSyncJob.name, + ); + constructor( + private readonly createCompanyAndContactService: CreateCompanyAndContactService, + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, + @InjectWorkspaceRepository(CalendarEventParticipantWorkspaceEntity) + private readonly calendarEventParticipantRepository: WorkspaceRepository<CalendarEventParticipantWorkspaceEntity>, + ) {} + + @Process(CalendarCreateCompanyAndContactAfterSyncJob.name) + async handle( + data: CalendarCreateCompanyAndContactAfterSyncJobData, + ): Promise<void> { + this.logger.log( + `create contacts and companies after sync for workspace ${data.workspaceId} and calendarChannel ${data.calendarChannelId}`, + ); + const { workspaceId, calendarChannelId } = data; + + const calendarChannel = await this.calendarChannelRepository.findOne({ + where: { + id: calendarChannelId, + }, + relations: ['connectedAccount.accountOwner'], + }); + + if (!calendarChannel) { + throw new Error( + `Calendar channel with id ${calendarChannelId} not found in workspace ${workspaceId}`, + ); + } + + const { handle, isContactAutoCreationEnabled, connectedAccount } = + calendarChannel; + + if (!isContactAutoCreationEnabled || !handle) { + return; + } + + if (!connectedAccount) { + throw new Error( + `Connected account not found in workspace ${workspaceId}`, + ); + } + + const calendarEventParticipantsWithoutPersonIdAndWorkspaceMemberId = + await this.calendarEventParticipantRepository.find({ + where: { + calendarEvent: { + calendarChannelEventAssociations: { + calendarChannelId, + }, + calendarEventParticipants: { + person: IsNull(), + workspaceMember: IsNull(), + }, + }, + }, + relations: [ + 'calendarEvent.calendarChannelEventAssociations', + 'calendarEvent.calendarEventParticipants', + ], + }); + + await this.createCompanyAndContactService.createCompaniesAndContactsAndUpdateParticipants( + connectedAccount, + calendarEventParticipantsWithoutPersonIdAndWorkspaceMemberId, + workspaceId, + ); + + this.logger.log( + `create contacts and companies after sync for workspace ${data.workspaceId} and calendarChannel ${data.calendarChannelId} done`, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-match-participant.job.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-match-participant.job.ts new file mode 100644 index 000000000000..e8dc28c0c117 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-match-participant.job.ts @@ -0,0 +1,37 @@ +import { Scope } from '@nestjs/common'; + +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { CalendarEventParticipantService } from 'src/modules/calendar/calendar-event-participant-manager/services/calendar-event-participant.service'; + +export type CalendarEventParticipantMatchParticipantJobData = { + workspaceId: string; + email: string; + personId?: string; + workspaceMemberId?: string; +}; + +@Processor({ + queueName: MessageQueue.calendarQueue, + scope: Scope.REQUEST, +}) +export class CalendarEventParticipantMatchParticipantJob { + constructor( + private readonly calendarEventParticipantService: CalendarEventParticipantService, + ) {} + + @Process(CalendarEventParticipantMatchParticipantJob.name) + async handle( + data: CalendarEventParticipantMatchParticipantJobData, + ): Promise<void> { + const { workspaceId, email, personId, workspaceMemberId } = data; + + await this.calendarEventParticipantService.matchCalendarEventParticipants( + workspaceId, + email, + personId, + workspaceMemberId, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-unmatch-participant.job.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-unmatch-participant.job.ts new file mode 100644 index 000000000000..9aa9d3e38ec7 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-unmatch-participant.job.ts @@ -0,0 +1,37 @@ +import { Scope } from '@nestjs/common'; + +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { CalendarEventParticipantService } from 'src/modules/calendar/calendar-event-participant-manager/services/calendar-event-participant.service'; + +export type CalendarEventParticipantUnmatchParticipantJobData = { + workspaceId: string; + email: string; + personId?: string; + workspaceMemberId?: string; +}; + +@Processor({ + queueName: MessageQueue.calendarQueue, + scope: Scope.REQUEST, +}) +export class CalendarEventParticipantUnmatchParticipantJob { + constructor( + private readonly calendarEventParticipantService: CalendarEventParticipantService, + ) {} + + @Process(CalendarEventParticipantUnmatchParticipantJob.name) + async handle( + data: CalendarEventParticipantUnmatchParticipantJobData, + ): Promise<void> { + const { workspaceId, email, personId, workspaceMemberId } = data; + + await this.calendarEventParticipantService.unmatchCalendarEventParticipants( + workspaceId, + email, + personId, + workspaceMemberId, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-person.listener.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-person.listener.ts new file mode 100644 index 000000000000..7b369af572d6 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-person.listener.ts @@ -0,0 +1,74 @@ +import { Injectable } from '@nestjs/common'; +import { OnEvent } from '@nestjs/event-emitter'; + +import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; +import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; +import { objectRecordChangedProperties as objectRecordUpdateEventChangedProperties } from 'src/engine/integrations/event-emitter/utils/object-record-changed-properties.util'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { + CalendarEventParticipantMatchParticipantJobData, + CalendarEventParticipantMatchParticipantJob, +} from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-match-participant.job'; +import { + CalendarEventParticipantUnmatchParticipantJobData, + CalendarEventParticipantUnmatchParticipantJob, +} from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-unmatch-participant.job'; +import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; + +@Injectable() +export class CalendarEventParticipantPersonListener { + constructor( + @InjectMessageQueue(MessageQueue.calendarQueue) + private readonly messageQueueService: MessageQueueService, + ) {} + + @OnEvent('person.created') + async handleCreatedEvent( + payload: ObjectRecordCreateEvent<PersonWorkspaceEntity>, + ) { + if (payload.properties.after.email === null) { + return; + } + + await this.messageQueueService.add<CalendarEventParticipantMatchParticipantJobData>( + CalendarEventParticipantMatchParticipantJob.name, + { + workspaceId: payload.workspaceId, + email: payload.properties.after.email, + personId: payload.recordId, + }, + ); + } + + @OnEvent('person.updated') + async handleUpdatedEvent( + payload: ObjectRecordUpdateEvent<PersonWorkspaceEntity>, + ) { + if ( + objectRecordUpdateEventChangedProperties( + payload.properties.before, + payload.properties.after, + ).includes('email') + ) { + await this.messageQueueService.add<CalendarEventParticipantUnmatchParticipantJobData>( + CalendarEventParticipantUnmatchParticipantJob.name, + { + workspaceId: payload.workspaceId, + email: payload.properties.before.email, + personId: payload.recordId, + }, + ); + + await this.messageQueueService.add<CalendarEventParticipantMatchParticipantJobData>( + CalendarEventParticipantMatchParticipantJob.name, + { + workspaceId: payload.workspaceId, + email: payload.properties.after.email, + personId: payload.recordId, + }, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-workspace-member.listener.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-workspace-member.listener.ts new file mode 100644 index 000000000000..1f84fce0dc2f --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant-workspace-member.listener.ts @@ -0,0 +1,74 @@ +import { Injectable } from '@nestjs/common'; +import { OnEvent } from '@nestjs/event-emitter'; + +import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; +import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; +import { objectRecordChangedProperties as objectRecordUpdateEventChangedProperties } from 'src/engine/integrations/event-emitter/utils/object-record-changed-properties.util'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { + CalendarEventParticipantMatchParticipantJob, + CalendarEventParticipantMatchParticipantJobData, +} from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-match-participant.job'; +import { + CalendarEventParticipantUnmatchParticipantJobData, + CalendarEventParticipantUnmatchParticipantJob, +} from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-event-participant-unmatch-participant.job'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; + +@Injectable() +export class CalendarEventParticipantWorkspaceMemberListener { + constructor( + @InjectMessageQueue(MessageQueue.calendarQueue) + private readonly messageQueueService: MessageQueueService, + ) {} + + @OnEvent('workspaceMember.created') + async handleCreatedEvent( + payload: ObjectRecordCreateEvent<WorkspaceMemberWorkspaceEntity>, + ) { + if (payload.properties.after.userEmail === null) { + return; + } + + await this.messageQueueService.add<CalendarEventParticipantMatchParticipantJobData>( + CalendarEventParticipantMatchParticipantJob.name, + { + workspaceId: payload.workspaceId, + email: payload.properties.after.userEmail, + workspaceMemberId: payload.properties.after.id, + }, + ); + } + + @OnEvent('workspaceMember.updated') + async handleUpdatedEvent( + payload: ObjectRecordUpdateEvent<WorkspaceMemberWorkspaceEntity>, + ) { + if ( + objectRecordUpdateEventChangedProperties<WorkspaceMemberWorkspaceEntity>( + payload.properties.before, + payload.properties.after, + ).includes('userEmail') + ) { + await this.messageQueueService.add<CalendarEventParticipantUnmatchParticipantJobData>( + CalendarEventParticipantUnmatchParticipantJob.name, + { + workspaceId: payload.workspaceId, + email: payload.properties.before.userEmail, + personId: payload.recordId, + }, + ); + + await this.messageQueueService.add<CalendarEventParticipantMatchParticipantJobData>( + CalendarEventParticipantMatchParticipantJob.name, + { + workspaceId: payload.workspaceId, + email: payload.properties.after.userEmail, + workspaceMemberId: payload.recordId, + }, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/listeners/calendar-event-participant.listener.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant.listener.ts similarity index 88% rename from packages/twenty-server/src/modules/calendar/listeners/calendar-event-participant.listener.ts rename to packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant.listener.ts index 79025878c586..b28a73033990 100644 --- a/packages/twenty-server/src/modules/calendar/listeners/calendar-event-participant.listener.ts +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/listeners/calendar-event-participant.listener.ts @@ -7,10 +7,9 @@ import { Repository } from 'typeorm'; import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository'; import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; @Injectable() export class CalendarEventParticipantListener { @@ -25,8 +24,8 @@ export class CalendarEventParticipantListener { @OnEvent('calendarEventParticipant.matched') public async handleCalendarEventParticipantMatchedEvent(payload: { workspaceId: string; - userId: string; - calendarEventParticipants: ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]; + workspaceMemberId: string; + calendarEventParticipants: CalendarEventParticipantWorkspaceEntity[]; }): Promise<void> { const calendarEventParticipants = payload.calendarEventParticipants ?? []; @@ -59,7 +58,7 @@ export class CalendarEventParticipantListener { properties: null, objectName: 'calendarEvent', recordId: participant.personId, - workspaceMemberId: payload.userId, + workspaceMemberId: payload.workspaceMemberId, workspaceId: payload.workspaceId, linkedObjectMetadataId: calendarEventObjectMetadata.id, linkedRecordId: participant.calendarEventId, diff --git a/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/services/calendar-event-participant.service.ts b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/services/calendar-event-participant.service.ts new file mode 100644 index 000000000000..346e727d99e2 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/calendar-event-participant-manager/services/calendar-event-participant.service.ts @@ -0,0 +1,192 @@ +import { Injectable } from '@nestjs/common'; +import { EventEmitter2 } from '@nestjs/event-emitter'; + +import { isDefined } from 'class-validator'; +import differenceWith from 'lodash.differencewith'; +import { Any } from 'typeorm'; + +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventParticipantWithCalendarEventId } from 'src/modules/calendar/common/types/calendar-event'; + +@Injectable() +export class CalendarEventParticipantService { + constructor( + @InjectWorkspaceRepository(CalendarEventParticipantWorkspaceEntity) + private readonly calendarEventParticipantRepository: WorkspaceRepository<CalendarEventParticipantWorkspaceEntity>, + private readonly eventEmitter: EventEmitter2, + ) {} + + public async upsertAndDeleteCalendarEventParticipants( + participantsToSave: CalendarEventParticipantWithCalendarEventId[], + participantsToUpdate: CalendarEventParticipantWithCalendarEventId[], + transactionManager?: any, + ): Promise<void> { + const existingCalendarEventParticipants = + await this.calendarEventParticipantRepository.find({ + where: { + calendarEventId: Any( + participantsToUpdate + .map((participant) => participant.calendarEventId) + .filter(isDefined), + ), + }, + }); + + const { calendarEventParticipantsToUpdate, newCalendarEventParticipants } = + participantsToUpdate.reduce( + (acc, calendarEventParticipant) => { + const existingCalendarEventParticipant = + existingCalendarEventParticipants.find( + (existingCalendarEventParticipant) => + existingCalendarEventParticipant.handle === + calendarEventParticipant.handle && + existingCalendarEventParticipant.calendarEventId === + calendarEventParticipant.calendarEventId, + ); + + if (existingCalendarEventParticipant) { + acc.calendarEventParticipantsToUpdate.push( + calendarEventParticipant, + ); + } else { + acc.newCalendarEventParticipants.push(calendarEventParticipant); + } + + return acc; + }, + { + calendarEventParticipantsToUpdate: + [] as CalendarEventParticipantWithCalendarEventId[], + newCalendarEventParticipants: + [] as CalendarEventParticipantWithCalendarEventId[], + }, + ); + + const calendarEventParticipantsToDelete = differenceWith( + existingCalendarEventParticipants, + participantsToUpdate, + (existingCalendarEventParticipant, participantToUpdate) => + existingCalendarEventParticipant.handle === + participantToUpdate.handle && + existingCalendarEventParticipant.calendarEventId === + participantToUpdate.calendarEventId, + ); + + await this.calendarEventParticipantRepository.delete( + { + id: Any( + calendarEventParticipantsToDelete.map( + (calendarEventParticipant) => calendarEventParticipant.id, + ), + ), + }, + transactionManager, + ); + + for (const calendarEventParticipantToUpdate of calendarEventParticipantsToUpdate) { + await this.calendarEventParticipantRepository.update( + { + calendarEventId: calendarEventParticipantToUpdate.calendarEventId, + handle: calendarEventParticipantToUpdate.handle, + }, + { + ...calendarEventParticipantToUpdate, + }, + transactionManager, + ); + } + + participantsToSave.push(...newCalendarEventParticipants); + + await this.calendarEventParticipantRepository.save( + participantsToSave, + {}, + transactionManager, + ); + } + + public async matchCalendarEventParticipants( + workspaceId: string, + email: string, + personId?: string, + workspaceMemberId?: string, + ) { + const calendarEventParticipantsToUpdate = + await this.calendarEventParticipantRepository.find({ + where: { + handle: email, + }, + }); + + const calendarEventParticipantIdsToUpdate = + calendarEventParticipantsToUpdate.map((participant) => participant.id); + + if (personId) { + await this.calendarEventParticipantRepository.update( + { + id: Any(calendarEventParticipantIdsToUpdate), + }, + { + person: { + id: personId, + }, + }, + ); + + const updatedCalendarEventParticipants = + await this.calendarEventParticipantRepository.find({ + where: { + id: Any(calendarEventParticipantIdsToUpdate), + }, + }); + + this.eventEmitter.emit(`calendarEventParticipant.matched`, { + workspaceId, + workspaceMemberId: null, + calendarEventParticipants: updatedCalendarEventParticipants, + }); + } + if (workspaceMemberId) { + await this.calendarEventParticipantRepository.update( + { + id: Any(calendarEventParticipantIdsToUpdate), + }, + { + workspaceMember: { + id: workspaceMemberId, + }, + }, + ); + } + } + + public async unmatchCalendarEventParticipants( + workspaceId: string, + handle: string, + personId?: string, + workspaceMemberId?: string, + ) { + if (personId) { + await this.calendarEventParticipantRepository.update( + { + handle, + }, + { + person: null, + }, + ); + } + if (workspaceMemberId) { + await this.calendarEventParticipantRepository.update( + { + handle, + }, + { + workspaceMember: null, + }, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/calendar/calendar.module.ts b/packages/twenty-server/src/modules/calendar/calendar.module.ts index d6a63b5a206e..de5759c1849c 100644 --- a/packages/twenty-server/src/modules/calendar/calendar.module.ts +++ b/packages/twenty-server/src/modules/calendar/calendar.module.ts @@ -1,27 +1,20 @@ import { Module } from '@nestjs/common'; -import { TypeOrmModule } from '@nestjs/typeorm'; -import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { CalendarBlocklistListener } from 'src/modules/calendar/listeners/calendar-blocklist.listener'; -import { CalendarChannelListener } from 'src/modules/calendar/listeners/calendar-channel.listener'; -import { CalendarEventParticipantListener } from 'src/modules/calendar/listeners/calendar-event-participant.listener'; -import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; +import { CalendarBlocklistManagerModule } from 'src/modules/calendar/blocklist-manager/calendar-blocklist-manager.module'; +import { CalendarEventCleanerModule } from 'src/modules/calendar/calendar-event-cleaner/calendar-event-cleaner.module'; +import { CalendarEventImportManagerModule } from 'src/modules/calendar/calendar-event-import-manager/calendar-event-import-manager.module'; +import { CalendarEventParticipantManagerModule } from 'src/modules/calendar/calendar-event-participant-manager/calendar-event-participant-manager.module'; +import { CalendarCommonModule } from 'src/modules/calendar/common/calendar-common.module'; @Module({ imports: [ - WorkspaceDataSourceModule, - ObjectMetadataRepositoryModule.forFeature([ - TimelineActivityWorkspaceEntity, - ]), - TypeOrmModule.forFeature([ObjectMetadataEntity], 'metadata'), - ], - providers: [ - CalendarChannelListener, - CalendarBlocklistListener, - CalendarEventParticipantListener, + CalendarBlocklistManagerModule, + CalendarEventCleanerModule, + CalendarEventImportManagerModule, + CalendarEventParticipantManagerModule, + CalendarCommonModule, ], + providers: [], exports: [], }) export class CalendarModule {} diff --git a/packages/twenty-server/src/modules/calendar/commands/calendar-commands.module.ts b/packages/twenty-server/src/modules/calendar/commands/calendar-commands.module.ts deleted file mode 100644 index 20b187004c8b..000000000000 --- a/packages/twenty-server/src/modules/calendar/commands/calendar-commands.module.ts +++ /dev/null @@ -1,19 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { GoogleCalendarSyncCommand } from 'src/modules/calendar/commands/google-calendar-sync.command'; -import { WorkspaceGoogleCalendarSyncModule } from 'src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.module'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; - -@Module({ - imports: [ - ObjectMetadataRepositoryModule.forFeature([ - ConnectedAccountWorkspaceEntity, - CalendarChannelWorkspaceEntity, - ]), - WorkspaceGoogleCalendarSyncModule, - ], - providers: [GoogleCalendarSyncCommand], -}) -export class CalendarCommandsModule {} diff --git a/packages/twenty-server/src/modules/calendar/commands/google-calendar-sync.command.ts b/packages/twenty-server/src/modules/calendar/commands/google-calendar-sync.command.ts deleted file mode 100644 index f605ad74256c..000000000000 --- a/packages/twenty-server/src/modules/calendar/commands/google-calendar-sync.command.ts +++ /dev/null @@ -1,40 +0,0 @@ -import { Command, CommandRunner, Option } from 'nest-commander'; - -import { WorkspaceGoogleCalendarSyncService } from 'src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.service'; - -interface GoogleCalendarSyncOptions { - workspaceId: string; -} - -@Command({ - name: 'workspace:google-calendar-sync', - description: - 'Start google calendar sync for all workspaceMembers in a workspace.', -}) -export class GoogleCalendarSyncCommand extends CommandRunner { - constructor( - private readonly workspaceGoogleCalendarSyncService: WorkspaceGoogleCalendarSyncService, - ) { - super(); - } - - @Option({ - flags: '-w, --workspace-id [workspace_id]', - description: 'workspace id', - required: true, - }) - parseWorkspaceId(value: string): string { - return value; - } - - async run( - _passedParam: string[], - options: GoogleCalendarSyncOptions, - ): Promise<void> { - await this.workspaceGoogleCalendarSyncService.startWorkspaceGoogleCalendarSync( - options.workspaceId, - ); - - return; - } -} diff --git a/packages/twenty-server/src/modules/calendar/common/calendar-common.module.ts b/packages/twenty-server/src/modules/calendar/common/calendar-common.module.ts new file mode 100644 index 000000000000..3d3ccecd2d0e --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/common/calendar-common.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; + +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; + +@Module({ + imports: [WorkspaceDataSourceModule], + providers: [AddPersonIdAndWorkspaceMemberIdService], + exports: [], +}) +export class CalendarCommonModule {} diff --git a/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook.ts b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook.ts new file mode 100644 index 000000000000..8660401f1248 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook.ts @@ -0,0 +1,52 @@ +import { BadRequestException, NotFoundException, Scope } from '@nestjs/common'; + +import { FindManyResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; + +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; +import { CanAccessCalendarEventService } from 'src/modules/calendar/common/query-hooks/calendar-event/services/can-access-calendar-event.service'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; + +@WorkspaceQueryHook({ + key: `calendarEvent.findMany`, + scope: Scope.REQUEST, +}) +export class CalendarEventFindManyPreQueryHook + implements WorkspaceQueryHookInstance +{ + constructor( + @InjectWorkspaceRepository(CalendarChannelEventAssociationWorkspaceEntity) + private readonly calendarChannelEventAssociationRepository: WorkspaceRepository<CalendarChannelEventAssociationWorkspaceEntity>, + private readonly canAccessCalendarEventService: CanAccessCalendarEventService, + ) {} + + async execute( + userId: string, + workspaceId: string, + payload: FindManyResolverArgs, + ): Promise<void> { + if (!payload?.filter?.id?.eq) { + throw new BadRequestException('id filter is required'); + } + + const calendarChannelCalendarEventAssociations = + await this.calendarChannelEventAssociationRepository.find({ + where: { + calendarEventId: payload?.filter?.id?.eq, + }, + relations: ['calendarChannel.connectedAccount'], + }); + + if (calendarChannelCalendarEventAssociations.length === 0) { + throw new NotFoundException(); + } + + await this.canAccessCalendarEventService.canAccessCalendarEvent( + userId, + workspaceId, + calendarChannelCalendarEventAssociations, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook.ts b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook.ts new file mode 100644 index 000000000000..add123542be9 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook.ts @@ -0,0 +1,53 @@ +import { BadRequestException, NotFoundException, Scope } from '@nestjs/common'; + +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; +import { FindOneResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; + +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { CanAccessCalendarEventService } from 'src/modules/calendar/common/query-hooks/calendar-event/services/can-access-calendar-event.service'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; + +@WorkspaceQueryHook({ + key: `calendarEvent.findOne`, + scope: Scope.REQUEST, +}) +export class CalendarEventFindOnePreQueryHook + implements WorkspaceQueryHookInstance +{ + constructor( + @InjectWorkspaceRepository(CalendarChannelEventAssociationWorkspaceEntity) + private readonly calendarChannelEventAssociationRepository: WorkspaceRepository<CalendarChannelEventAssociationWorkspaceEntity>, + private readonly canAccessCalendarEventService: CanAccessCalendarEventService, + ) {} + + async execute( + userId: string, + workspaceId: string, + payload: FindOneResolverArgs, + ): Promise<void> { + if (!payload?.filter?.id?.eq) { + throw new BadRequestException('id filter is required'); + } + + // TODO: Re-implement this using twenty ORM + const calendarChannelCalendarEventAssociations = + await this.calendarChannelEventAssociationRepository.find({ + where: { + calendarEventId: payload?.filter?.id?.eq, + }, + relations: ['calendarChannel.connectedAccount'], + }); + + if (calendarChannelCalendarEventAssociations.length === 0) { + throw new NotFoundException(); + } + + await this.canAccessCalendarEventService.canAccessCalendarEvent( + userId, + workspaceId, + calendarChannelCalendarEventAssociations, + ); + } +} diff --git a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/services/can-access-calendar-event.service.ts b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/services/can-access-calendar-event.service.ts similarity index 72% rename from packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/services/can-access-calendar-event.service.ts rename to packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/services/can-access-calendar-event.service.ts index 0a971c32dcbf..fcd05354a713 100644 --- a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/services/can-access-calendar-event.service.ts +++ b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-event/services/can-access-calendar-event.service.ts @@ -1,25 +1,26 @@ import { ForbiddenException, Injectable } from '@nestjs/common'; import groupBy from 'lodash.groupby'; +import { Any } from 'typeorm'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { - CalendarChannelWorkspaceEntity, - CalendarChannelVisibility, -} from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { + CalendarChannelWorkspaceEntity, + CalendarChannelVisibility, +} from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; @Injectable() export class CanAccessCalendarEventService { constructor( - @InjectObjectMetadataRepository(CalendarChannelWorkspaceEntity) - private readonly calendarChannelRepository: CalendarChannelRepository, + @InjectWorkspaceRepository(CalendarChannelWorkspaceEntity) + private readonly calendarChannelRepository: WorkspaceRepository<CalendarChannelWorkspaceEntity>, @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) private readonly connectedAccountRepository: ConnectedAccountRepository, @InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity) @@ -29,14 +30,17 @@ export class CanAccessCalendarEventService { public async canAccessCalendarEvent( userId: string, workspaceId: string, - calendarChannelCalendarEventAssociations: ObjectRecord<CalendarChannelEventAssociationWorkspaceEntity>[], + calendarChannelCalendarEventAssociations: CalendarChannelEventAssociationWorkspaceEntity[], ) { - const calendarChannels = await this.calendarChannelRepository.getByIds( - calendarChannelCalendarEventAssociations.map( - (association) => association.calendarChannelId, - ), - workspaceId, - ); + const calendarChannels = await this.calendarChannelRepository.find({ + where: { + id: Any( + calendarChannelCalendarEventAssociations.map( + (association) => association.calendarChannel.id, + ), + ), + }, + }); const calendarChannelsGroupByVisibility = groupBy( calendarChannels, diff --git a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-query-hook.module.ts b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-query-hook.module.ts similarity index 66% rename from packages/twenty-server/src/modules/calendar/query-hooks/calendar-query-hook.module.ts rename to packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-query-hook.module.ts index 0bed2914aa0d..e3a16adf896b 100644 --- a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-query-hook.module.ts +++ b/packages/twenty-server/src/modules/calendar/common/query-hooks/calendar-query-hook.module.ts @@ -3,31 +3,28 @@ import { Module } from '@nestjs/common'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { CalendarEventFindManyPreQueryHook } from 'src/modules/calendar/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook'; -import { CalendarEventFindOnePreQueryHook } from 'src/modules/calendar/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook'; -import { CanAccessCalendarEventService } from 'src/modules/calendar/query-hooks/calendar-event/services/can-access-calendar-event.service'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { CalendarEventFindManyPreQueryHook } from 'src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook'; +import { CalendarEventFindOnePreQueryHook } from 'src/modules/calendar/common/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook'; +import { CanAccessCalendarEventService } from 'src/modules/calendar/common/query-hooks/calendar-event/services/can-access-calendar-event.service'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; @Module({ imports: [ - ObjectMetadataRepositoryModule.forFeature([ + TwentyORMModule.forFeature([ CalendarChannelEventAssociationWorkspaceEntity, CalendarChannelWorkspaceEntity, + ]), + ObjectMetadataRepositoryModule.forFeature([ ConnectedAccountWorkspaceEntity, WorkspaceMemberWorkspaceEntity, ]), ], providers: [ CanAccessCalendarEventService, - { - provide: CalendarEventFindOnePreQueryHook.name, - useClass: CalendarEventFindOnePreQueryHook, - }, - { - provide: CalendarEventFindManyPreQueryHook.name, - useClass: CalendarEventFindManyPreQueryHook, - }, + CalendarEventFindOnePreQueryHook, + CalendarEventFindManyPreQueryHook, ], }) export class CalendarQueryHookModule {} diff --git a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity.ts b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity.ts similarity index 88% rename from packages/twenty-server/src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity.ts rename to packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity.ts index bf1e82aa7210..f1475da3b012 100644 --- a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity.ts +++ b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity.ts @@ -3,7 +3,6 @@ import { Relation } from 'src/engine/workspace-manager/workspace-sync-metadata/i import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { CALENDAR_CHANNEL_EVENT_ASSOCIATION_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; @@ -11,7 +10,9 @@ import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-re import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.calendarChannelEventAssociation, @@ -41,12 +42,14 @@ export class CalendarChannelEventAssociationWorkspaceEntity extends BaseWorkspac label: 'Channel ID', description: 'Channel ID', icon: 'IconCalendar', - joinColumn: 'calendarChannelId', inverseSideTarget: () => CalendarChannelWorkspaceEntity, inverseSideFieldKey: 'calendarChannelEventAssociations', }) calendarChannel: Relation<CalendarChannelWorkspaceEntity>; + @WorkspaceJoinColumn('calendarChannel') + calendarChannelId: string; + @WorkspaceRelation({ standardId: CALENDAR_CHANNEL_EVENT_ASSOCIATION_STANDARD_FIELD_IDS.calendarEvent, @@ -54,9 +57,11 @@ export class CalendarChannelEventAssociationWorkspaceEntity extends BaseWorkspac label: 'Event ID', description: 'Event ID', icon: 'IconCalendar', - joinColumn: 'calendarEventId', inverseSideTarget: () => CalendarEventWorkspaceEntity, inverseSideFieldKey: 'calendarChannelEventAssociations', }) calendarEvent: Relation<CalendarEventWorkspaceEntity>; + + @WorkspaceJoinColumn('calendarEvent') + calendarEventId: string; } diff --git a/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity.ts b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity.ts new file mode 100644 index 000000000000..88a549847726 --- /dev/null +++ b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity.ts @@ -0,0 +1,301 @@ +import { Relation } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/relation.interface'; + +import { + RelationMetadataType, + RelationOnDeleteAction, +} from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; +import { CALENDAR_CHANNEL_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; +import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; +import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; +import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; +import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; +import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; +import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; + +export enum CalendarChannelVisibility { + METADATA = 'METADATA', + SHARE_EVERYTHING = 'SHARE_EVERYTHING', +} + +export enum CalendarChannelSyncStatus { + NOT_SYNCED = 'NOT_SYNCED', + ONGOING = 'ONGOING', + ACTIVE = 'ACTIVE', + FAILED_INSUFFICIENT_PERMISSIONS = 'FAILED_INSUFFICIENT_PERMISSIONS', + FAILED_UNKNOWN = 'FAILED_UNKNOWN', +} + +export enum CalendarChannelSyncStage { + FULL_CALENDAR_EVENT_LIST_FETCH_PENDING = 'FULL_CALENDAR_EVENT_LIST_FETCH_PENDING', + PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING = 'PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING', + CALENDAR_EVENT_LIST_FETCH_ONGOING = 'CALENDAR_EVENT_LIST_FETCH_ONGOING', + CALENDAR_EVENTS_IMPORT_PENDING = 'CALENDAR_EVENTS_IMPORT_PENDING', + CALENDAR_EVENTS_IMPORT_ONGOING = 'CALENDAR_EVENTS_IMPORT_ONGOING', + FAILED = 'FAILED', +} + +export enum CalendarChannelContactAutoCreationPolicy { + AS_PARTICIPANT_AND_ORGANIZER = 'AS_PARTICIPANT_AND_ORGANIZER', + AS_PARTICIPANT = 'AS_PARTICIPANT', + AS_ORGANIZER = 'AS_ORGANIZER', + NONE = 'NONE', +} + +@WorkspaceEntity({ + standardId: STANDARD_OBJECT_IDS.calendarChannel, + namePlural: 'calendarChannels', + labelSingular: 'Calendar Channel', + labelPlural: 'Calendar Channels', + description: 'Calendar Channels', + icon: 'IconCalendar', +}) +@WorkspaceIsSystem() +@WorkspaceIsNotAuditLogged() +export class CalendarChannelWorkspaceEntity extends BaseWorkspaceEntity { + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.handle, + type: FieldMetadataType.TEXT, + label: 'Handle', + description: 'Handle', + icon: 'IconAt', + }) + handle: string; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.syncStatus, + type: FieldMetadataType.SELECT, + label: 'Sync status', + description: 'Sync status', + icon: 'IconStatusChange', + options: [ + { + value: CalendarChannelSyncStatus.ONGOING, + label: 'Ongoing', + position: 1, + color: 'yellow', + }, + { + value: CalendarChannelSyncStatus.NOT_SYNCED, + label: 'Not Synced', + position: 2, + color: 'blue', + }, + { + value: CalendarChannelSyncStatus.ACTIVE, + label: 'Active', + position: 3, + color: 'green', + }, + { + value: CalendarChannelSyncStatus.FAILED_INSUFFICIENT_PERMISSIONS, + label: 'Failed Insufficient Permissions', + position: 4, + color: 'red', + }, + { + value: CalendarChannelSyncStatus.FAILED_UNKNOWN, + label: 'Failed Unknown', + position: 5, + color: 'red', + }, + ], + }) + @WorkspaceIsNullable() + syncStatus: CalendarChannelSyncStatus | null; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.syncStage, + type: FieldMetadataType.SELECT, + label: 'Sync stage', + description: 'Sync stage', + icon: 'IconStatusChange', + options: [ + { + value: CalendarChannelSyncStage.FULL_CALENDAR_EVENT_LIST_FETCH_PENDING, + label: 'Full calendar event list fetch pending', + position: 0, + color: 'blue', + }, + { + value: + CalendarChannelSyncStage.PARTIAL_CALENDAR_EVENT_LIST_FETCH_PENDING, + label: 'Partial calendar event list fetch pending', + position: 1, + color: 'blue', + }, + { + value: CalendarChannelSyncStage.CALENDAR_EVENT_LIST_FETCH_ONGOING, + label: 'Calendar event list fetch ongoing', + position: 2, + color: 'orange', + }, + { + value: CalendarChannelSyncStage.CALENDAR_EVENTS_IMPORT_PENDING, + label: 'Calendar events import pending', + position: 3, + color: 'blue', + }, + { + value: CalendarChannelSyncStage.CALENDAR_EVENTS_IMPORT_ONGOING, + label: 'Calendar events import ongoing', + position: 4, + color: 'orange', + }, + { + value: CalendarChannelSyncStage.FAILED, + label: 'Failed', + position: 5, + color: 'red', + }, + ], + defaultValue: `'${CalendarChannelSyncStage.FULL_CALENDAR_EVENT_LIST_FETCH_PENDING}'`, + }) + syncStage: CalendarChannelSyncStage; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.visibility, + type: FieldMetadataType.SELECT, + label: 'Visibility', + description: 'Visibility', + icon: 'IconEyeglass', + options: [ + { + value: CalendarChannelVisibility.METADATA, + label: 'Metadata', + position: 0, + color: 'green', + }, + { + value: CalendarChannelVisibility.SHARE_EVERYTHING, + label: 'Share Everything', + position: 1, + color: 'orange', + }, + ], + defaultValue: `'${CalendarChannelVisibility.SHARE_EVERYTHING}'`, + }) + visibility: string; + + @WorkspaceField({ + standardId: + CALENDAR_CHANNEL_STANDARD_FIELD_IDS.isContactAutoCreationEnabled, + type: FieldMetadataType.BOOLEAN, + label: 'Is Contact Auto Creation Enabled', + description: 'Is Contact Auto Creation Enabled', + icon: 'IconUserCircle', + defaultValue: true, + }) + isContactAutoCreationEnabled: boolean; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.contactAutoCreationPolicy, + type: FieldMetadataType.SELECT, + label: 'Contact auto creation policy', + description: + 'Automatically create records for people you participated with in an event.', + icon: 'IconUserCircle', + options: [ + { + value: + CalendarChannelContactAutoCreationPolicy.AS_PARTICIPANT_AND_ORGANIZER, + label: 'As Participant and Organizer', + color: 'green', + position: 0, + }, + { + value: CalendarChannelContactAutoCreationPolicy.AS_PARTICIPANT, + label: 'As Participant', + color: 'orange', + position: 1, + }, + { + value: CalendarChannelContactAutoCreationPolicy.AS_ORGANIZER, + label: 'As Organizer', + color: 'blue', + position: 2, + }, + { + value: CalendarChannelContactAutoCreationPolicy.NONE, + label: 'None', + color: 'red', + position: 3, + }, + ], + defaultValue: `'${CalendarChannelContactAutoCreationPolicy.AS_PARTICIPANT_AND_ORGANIZER}'`, + }) + contactAutoCreationPolicy: CalendarChannelContactAutoCreationPolicy; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.isSyncEnabled, + type: FieldMetadataType.BOOLEAN, + label: 'Is Sync Enabled', + description: 'Is Sync Enabled', + icon: 'IconRefresh', + defaultValue: true, + }) + isSyncEnabled: boolean; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.syncCursor, + type: FieldMetadataType.TEXT, + label: 'Sync Cursor', + description: + 'Sync Cursor. Used for syncing events from the calendar provider', + icon: 'IconReload', + }) + syncCursor: string; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.syncStageStartedAt, + type: FieldMetadataType.DATE_TIME, + label: 'Sync stage started at', + description: 'Sync stage started at', + icon: 'IconHistory', + }) + @WorkspaceIsNullable() + syncStageStartedAt: string | null; + + @WorkspaceField({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.throttleFailureCount, + type: FieldMetadataType.NUMBER, + label: 'Throttle Failure Count', + description: 'Throttle Failure Count', + icon: 'IconX', + defaultValue: 0, + }) + throttleFailureCount: number; + + @WorkspaceRelation({ + standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.connectedAccount, + type: RelationMetadataType.MANY_TO_ONE, + label: 'Connected Account', + description: 'Connected Account', + icon: 'IconUserCircle', + inverseSideTarget: () => ConnectedAccountWorkspaceEntity, + inverseSideFieldKey: 'calendarChannels', + }) + connectedAccount: Relation<ConnectedAccountWorkspaceEntity>; + + @WorkspaceJoinColumn('connectedAccount') + connectedAccountId: string; + + @WorkspaceRelation({ + standardId: + CALENDAR_CHANNEL_STANDARD_FIELD_IDS.calendarChannelEventAssociations, + type: RelationMetadataType.ONE_TO_MANY, + label: 'Calendar Channel Event Associations', + description: 'Calendar Channel Event Associations', + icon: 'IconCalendar', + inverseSideTarget: () => CalendarChannelEventAssociationWorkspaceEntity, + onDelete: RelationOnDeleteAction.CASCADE, + }) + calendarChannelEventAssociations: Relation< + CalendarChannelEventAssociationWorkspaceEntity[] + >; +} diff --git a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity.ts b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity.ts similarity index 91% rename from packages/twenty-server/src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity.ts rename to packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity.ts index ee271b437998..4d8727a50fcf 100644 --- a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity.ts +++ b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity.ts @@ -3,7 +3,6 @@ import { Relation } from 'src/engine/workspace-manager/workspace-sync-metadata/i import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; import { CALENDAR_EVENT_PARTICIPANT_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; @@ -14,6 +13,8 @@ import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; export enum CalendarEventParticipantResponseStatus { NEEDS_ACTION = 'NEEDS_ACTION', @@ -103,24 +104,28 @@ export class CalendarEventParticipantWorkspaceEntity extends BaseWorkspaceEntity label: 'Event ID', description: 'Event ID', icon: 'IconCalendar', - joinColumn: 'calendarEventId', inverseSideTarget: () => CalendarEventWorkspaceEntity, inverseSideFieldKey: 'calendarEventParticipants', }) calendarEvent: Relation<CalendarEventWorkspaceEntity>; + @WorkspaceJoinColumn('calendarEvent') + calendarEventId: string; + @WorkspaceRelation({ standardId: CALENDAR_EVENT_PARTICIPANT_STANDARD_FIELD_IDS.person, type: RelationMetadataType.MANY_TO_ONE, label: 'Person', description: 'Person', icon: 'IconUser', - joinColumn: 'personId', inverseSideTarget: () => PersonWorkspaceEntity, inverseSideFieldKey: 'calendarEventParticipants', }) @WorkspaceIsNullable() - person: Relation<PersonWorkspaceEntity>; + person: Relation<PersonWorkspaceEntity> | null; + + @WorkspaceJoinColumn('person') + personId: string | null; @WorkspaceRelation({ standardId: CALENDAR_EVENT_PARTICIPANT_STANDARD_FIELD_IDS.workspaceMember, @@ -128,10 +133,12 @@ export class CalendarEventParticipantWorkspaceEntity extends BaseWorkspaceEntity label: 'Workspace Member', description: 'Workspace Member', icon: 'IconUser', - joinColumn: 'workspaceMemberId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'calendarEventParticipants', }) @WorkspaceIsNullable() - workspaceMember: Relation<WorkspaceMemberWorkspaceEntity>; + workspaceMember: Relation<WorkspaceMemberWorkspaceEntity> | null; + + @WorkspaceJoinColumn('workspaceMember') + workspaceMemberId: string | null; } diff --git a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-event.workspace-entity.ts b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-event.workspace-entity.ts similarity index 96% rename from packages/twenty-server/src/modules/calendar/standard-objects/calendar-event.workspace-entity.ts rename to packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-event.workspace-entity.ts index d8d9052d27b3..563c85391690 100644 --- a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-event.workspace-entity.ts +++ b/packages/twenty-server/src/modules/calendar/common/standard-objects/calendar-event.workspace-entity.ts @@ -5,8 +5,6 @@ import { RelationMetadataType, RelationOnDeleteAction, } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; import { CALENDAR_EVENT_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; import { LinkMetadata } from 'src/engine/metadata-modules/field-metadata/composite-types/link.composite-type'; @@ -17,6 +15,8 @@ import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/work import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; +import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel-event-association.workspace-entity'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.calendarEvent, @@ -44,6 +44,7 @@ export class CalendarEventWorkspaceEntity extends BaseWorkspaceEntity { label: 'Is canceled', description: 'Is canceled', icon: 'IconCalendarCancel', + defaultValue: false, }) isCanceled: boolean; @@ -53,6 +54,7 @@ export class CalendarEventWorkspaceEntity extends BaseWorkspaceEntity { label: 'Is Full Day', description: 'Is Full Day', icon: 'Icon24Hours', + defaultValue: false, }) isFullDay: boolean; diff --git a/packages/twenty-server/src/modules/calendar/types/calendar-event.ts b/packages/twenty-server/src/modules/calendar/common/types/calendar-event.ts similarity index 54% rename from packages/twenty-server/src/modules/calendar/types/calendar-event.ts rename to packages/twenty-server/src/modules/calendar/common/types/calendar-event.ts index c2d7bd596603..00f4c82ac5c5 100644 --- a/packages/twenty-server/src/modules/calendar/types/calendar-event.ts +++ b/packages/twenty-server/src/modules/calendar/common/types/calendar-event.ts @@ -1,21 +1,21 @@ -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; +import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event.workspace-entity'; export type CalendarEvent = Omit< - ObjectRecord<CalendarEventWorkspaceEntity>, + CalendarEventWorkspaceEntity, | 'createdAt' | 'updatedAt' | 'calendarChannelEventAssociations' | 'calendarEventParticipants' | 'conferenceLink' + | 'id' > & { conferenceLinkLabel: string; conferenceLinkUrl: string; }; export type CalendarEventParticipant = Omit< - ObjectRecord<CalendarEventParticipantWorkspaceEntity>, + CalendarEventParticipantWorkspaceEntity, | 'id' | 'createdAt' | 'updatedAt' @@ -24,15 +24,23 @@ export type CalendarEventParticipant = Omit< | 'person' | 'workspaceMember' | 'calendarEvent' -> & { - iCalUID: string; -}; + | 'calendarEventId' +>; + +export type CalendarEventParticipantWithCalendarEventId = + CalendarEventParticipant & { + calendarEventId: string; + }; export type CalendarEventWithParticipants = CalendarEvent & { externalId: string; participants: CalendarEventParticipant[]; + status: string; }; -export type CalendarEventParticipantWithId = CalendarEventParticipant & { +export type CalendarEventWithParticipantsAndCalendarEventId = CalendarEvent & { id: string; + externalId: string; + participants: CalendarEventParticipantWithCalendarEventId[]; + status: string; }; diff --git a/packages/twenty-server/src/modules/calendar/crons/commands/calendar-cron-commands.module.ts b/packages/twenty-server/src/modules/calendar/crons/commands/calendar-cron-commands.module.ts deleted file mode 100644 index 30df8fe7daa5..000000000000 --- a/packages/twenty-server/src/modules/calendar/crons/commands/calendar-cron-commands.module.ts +++ /dev/null @@ -1,8 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { GoogleCalendarSyncCronCommand } from 'src/modules/calendar/crons/commands/google-calendar-sync.cron.command'; - -@Module({ - providers: [GoogleCalendarSyncCronCommand], -}) -export class CalendarCronCommandsModule {} diff --git a/packages/twenty-server/src/modules/calendar/crons/commands/google-calendar-sync.cron.command.ts b/packages/twenty-server/src/modules/calendar/crons/commands/google-calendar-sync.cron.command.ts deleted file mode 100644 index 90eeed4a0596..000000000000 --- a/packages/twenty-server/src/modules/calendar/crons/commands/google-calendar-sync.cron.command.ts +++ /dev/null @@ -1,32 +0,0 @@ -import { Inject } from '@nestjs/common'; - -import { Command, CommandRunner } from 'nest-commander'; - -import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; -import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { GoogleCalendarSyncCronJob } from 'src/modules/calendar/crons/jobs/google-calendar-sync.cron.job'; - -const GOOGLE_CALENDAR_SYNC_CRON_PATTERN = '*/5 * * * *'; - -@Command({ - name: 'cron:calendar:google-calendar-sync', - description: 'Starts a cron job to sync google calendar for all workspaces.', -}) -export class GoogleCalendarSyncCronCommand extends CommandRunner { - constructor( - @Inject(MessageQueue.cronQueue) - private readonly messageQueueService: MessageQueueService, - ) { - super(); - } - - async run(): Promise<void> { - await this.messageQueueService.addCron<undefined>( - GoogleCalendarSyncCronJob.name, - undefined, - { - repeat: { pattern: GOOGLE_CALENDAR_SYNC_CRON_PATTERN }, - }, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/crons/jobs/calendar-cron-job.module.ts b/packages/twenty-server/src/modules/calendar/crons/jobs/calendar-cron-job.module.ts deleted file mode 100644 index 04f57299fa88..000000000000 --- a/packages/twenty-server/src/modules/calendar/crons/jobs/calendar-cron-job.module.ts +++ /dev/null @@ -1,23 +0,0 @@ -import { Module } from '@nestjs/common'; -import { TypeOrmModule } from '@nestjs/typeorm'; - -import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; -import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; -import { GoogleCalendarSyncCronJob } from 'src/modules/calendar/crons/jobs/google-calendar-sync.cron.job'; -import { WorkspaceGoogleCalendarSyncModule } from 'src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.module'; - -@Module({ - imports: [ - TypeOrmModule.forFeature([Workspace, FeatureFlagEntity], 'core'), - TypeOrmModule.forFeature([DataSourceEntity], 'metadata'), - WorkspaceGoogleCalendarSyncModule, - ], - providers: [ - { - provide: GoogleCalendarSyncCronJob.name, - useClass: GoogleCalendarSyncCronJob, - }, - ], -}) -export class CalendarCronJobModule {} diff --git a/packages/twenty-server/src/modules/calendar/crons/jobs/google-calendar-sync.cron.job.ts b/packages/twenty-server/src/modules/calendar/crons/jobs/google-calendar-sync.cron.job.ts deleted file mode 100644 index 7c2072c41e1c..000000000000 --- a/packages/twenty-server/src/modules/calendar/crons/jobs/google-calendar-sync.cron.job.ts +++ /dev/null @@ -1,52 +0,0 @@ -import { Injectable } from '@nestjs/common'; -import { InjectRepository } from '@nestjs/typeorm'; - -import { Repository, In } from 'typeorm'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; -import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; -import { WorkspaceGoogleCalendarSyncService } from 'src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.service'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; - -@Injectable() -export class GoogleCalendarSyncCronJob implements MessageQueueJob<undefined> { - constructor( - @InjectRepository(Workspace, 'core') - private readonly workspaceRepository: Repository<Workspace>, - @InjectRepository(DataSourceEntity, 'metadata') - private readonly dataSourceRepository: Repository<DataSourceEntity>, - private readonly workspaceGoogleCalendarSyncService: WorkspaceGoogleCalendarSyncService, - private readonly environmentService: EnvironmentService, - ) {} - - async handle(): Promise<void> { - const workspaceIds = ( - await this.workspaceRepository.find({ - where: this.environmentService.get('IS_BILLING_ENABLED') - ? { - subscriptionStatus: In(['active', 'trialing', 'past_due']), - } - : {}, - select: ['id'], - }) - ).map((workspace) => workspace.id); - - const dataSources = await this.dataSourceRepository.find({ - where: { - workspaceId: In(workspaceIds), - }, - }); - - const workspaceIdsWithDataSources = new Set( - dataSources.map((dataSource) => dataSource.workspaceId), - ); - - for (const workspaceId of workspaceIdsWithDataSources) { - await this.workspaceGoogleCalendarSyncService.startWorkspaceGoogleCalendarSync( - workspaceId, - ); - } - } -} diff --git a/packages/twenty-server/src/modules/calendar/jobs/blocklist-item-delete-calendar-events.job.ts b/packages/twenty-server/src/modules/calendar/jobs/blocklist-item-delete-calendar-events.job.ts deleted file mode 100644 index a3196a02a45d..000000000000 --- a/packages/twenty-server/src/modules/calendar/jobs/blocklist-item-delete-calendar-events.job.ts +++ /dev/null @@ -1,83 +0,0 @@ -import { Injectable, Logger } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CalendarChannelEventAssociationRepository } from 'src/modules/calendar/repositories/calendar-channel-event-association.repository'; -import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; -import { CalendarEventCleanerService } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; - -export type BlocklistItemDeleteCalendarEventsJobData = { - workspaceId: string; - blocklistItemId: string; -}; - -@Injectable() -export class BlocklistItemDeleteCalendarEventsJob - implements MessageQueueJob<BlocklistItemDeleteCalendarEventsJobData> -{ - private readonly logger = new Logger( - BlocklistItemDeleteCalendarEventsJob.name, - ); - - constructor( - @InjectObjectMetadataRepository(CalendarChannelWorkspaceEntity) - private readonly calendarChannelRepository: CalendarChannelRepository, - @InjectObjectMetadataRepository( - CalendarChannelEventAssociationWorkspaceEntity, - ) - private readonly calendarChannelEventAssociationRepository: CalendarChannelEventAssociationRepository, - @InjectObjectMetadataRepository(BlocklistWorkspaceEntity) - private readonly blocklistRepository: BlocklistRepository, - private readonly calendarEventCleanerService: CalendarEventCleanerService, - ) {} - - async handle(data: BlocklistItemDeleteCalendarEventsJobData): Promise<void> { - const { workspaceId, blocklistItemId } = data; - - const blocklistItem = await this.blocklistRepository.getById( - blocklistItemId, - workspaceId, - ); - - if (!blocklistItem) { - this.logger.log( - `Blocklist item with id ${blocklistItemId} not found in workspace ${workspaceId}`, - ); - - return; - } - - const { handle, workspaceMemberId } = blocklistItem; - - this.logger.log( - `Deleting calendar events from ${handle} in workspace ${workspaceId} for workspace member ${workspaceMemberId}`, - ); - - const calendarChannels = - await this.calendarChannelRepository.getIdsByWorkspaceMemberId( - workspaceMemberId, - workspaceId, - ); - - const calendarChannelIds = calendarChannels.map(({ id }) => id); - - await this.calendarChannelEventAssociationRepository.deleteByCalendarEventParticipantHandleAndCalendarChannelIds( - handle, - calendarChannelIds, - workspaceId, - ); - - await this.calendarEventCleanerService.cleanWorkspaceCalendarEvents( - workspaceId, - ); - - this.logger.log( - `Deleted calendar events from handle ${handle} in workspace ${workspaceId} for workspace member ${workspaceMemberId}`, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/jobs/blocklist-reimport-calendar-events.job.ts b/packages/twenty-server/src/modules/calendar/jobs/blocklist-reimport-calendar-events.job.ts deleted file mode 100644 index 85660dae7d5a..000000000000 --- a/packages/twenty-server/src/modules/calendar/jobs/blocklist-reimport-calendar-events.job.ts +++ /dev/null @@ -1,59 +0,0 @@ -import { Injectable, Logger } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service'; -import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; - -export type BlocklistReimportCalendarEventsJobData = { - workspaceId: string; - workspaceMemberId: string; - handle: string; -}; - -@Injectable() -export class BlocklistReimportCalendarEventsJob - implements MessageQueueJob<BlocklistReimportCalendarEventsJobData> -{ - private readonly logger = new Logger(BlocklistReimportCalendarEventsJob.name); - - constructor( - @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) - private readonly connectedAccountRepository: ConnectedAccountRepository, - private readonly googleCalendarSyncService: GoogleCalendarSyncService, - ) {} - - async handle(data: BlocklistReimportCalendarEventsJobData): Promise<void> { - const { workspaceId, workspaceMemberId, handle } = data; - - this.logger.log( - `Reimporting calendar events from handle ${handle} in workspace ${workspaceId} for workspace member ${workspaceMemberId}`, - ); - - const connectedAccount = - await this.connectedAccountRepository.getAllByWorkspaceMemberId( - workspaceMemberId, - workspaceId, - ); - - if (!connectedAccount || connectedAccount.length === 0) { - this.logger.error( - `No connected account found for workspace member ${workspaceMemberId} in workspace ${workspaceId}`, - ); - - return; - } - - await this.googleCalendarSyncService.startGoogleCalendarSync( - workspaceId, - connectedAccount[0].id, - handle, - ); - - this.logger.log( - `Reimporting calendar events from ${handle} in workspace ${workspaceId} for workspace member ${workspaceMemberId} done`, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/jobs/calendar-create-company-and-contact-after-sync.job.ts b/packages/twenty-server/src/modules/calendar/jobs/calendar-create-company-and-contact-after-sync.job.ts deleted file mode 100644 index 3c1b9046b9a5..000000000000 --- a/packages/twenty-server/src/modules/calendar/jobs/calendar-create-company-and-contact-after-sync.job.ts +++ /dev/null @@ -1,89 +0,0 @@ -import { Injectable, Logger } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; -import { CalendarEventParticipantRepository } from 'src/modules/calendar/repositories/calendar-event-participant.repository'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service'; -import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; - -export type CalendarCreateCompanyAndContactAfterSyncJobData = { - workspaceId: string; - calendarChannelId: string; -}; - -@Injectable() -export class CalendarCreateCompanyAndContactAfterSyncJob - implements MessageQueueJob<CalendarCreateCompanyAndContactAfterSyncJobData> -{ - private readonly logger = new Logger( - CalendarCreateCompanyAndContactAfterSyncJob.name, - ); - constructor( - private readonly createCompanyAndContactService: CreateCompanyAndContactService, - @InjectObjectMetadataRepository(CalendarChannelWorkspaceEntity) - private readonly calendarChannelService: CalendarChannelRepository, - @InjectObjectMetadataRepository(CalendarEventParticipantWorkspaceEntity) - private readonly calendarEventParticipantRepository: CalendarEventParticipantRepository, - @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) - private readonly connectedAccountRepository: ConnectedAccountRepository, - ) {} - - async handle( - data: CalendarCreateCompanyAndContactAfterSyncJobData, - ): Promise<void> { - this.logger.log( - `create contacts and companies after sync for workspace ${data.workspaceId} and calendarChannel ${data.calendarChannelId}`, - ); - const { workspaceId, calendarChannelId } = data; - - const calendarChannels = await this.calendarChannelService.getByIds( - [calendarChannelId], - workspaceId, - ); - - if (calendarChannels.length === 0) { - throw new Error( - `Calendar channel with id ${calendarChannelId} not found in workspace ${workspaceId}`, - ); - } - - const { handle, isContactAutoCreationEnabled, connectedAccountId } = - calendarChannels[0]; - - if (!isContactAutoCreationEnabled || !handle) { - return; - } - - const connectedAccount = await this.connectedAccountRepository.getById( - connectedAccountId, - workspaceId, - ); - - if (!connectedAccount) { - throw new Error( - `Connected account with id ${connectedAccountId} not found in workspace ${workspaceId}`, - ); - } - - const calendarEventParticipantsWithoutPersonIdAndWorkspaceMemberId = - await this.calendarEventParticipantRepository.getByCalendarChannelIdWithoutPersonIdAndWorkspaceMemberId( - calendarChannelId, - workspaceId, - ); - - await this.createCompanyAndContactService.createCompaniesAndContactsAndUpdateParticipants( - connectedAccount, - calendarEventParticipantsWithoutPersonIdAndWorkspaceMemberId, - workspaceId, - ); - - this.logger.log( - `create contacts and companies after sync for workspace ${data.workspaceId} and calendarChannel ${data.calendarChannelId} done`, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/jobs/calendar-job.module.ts b/packages/twenty-server/src/modules/calendar/jobs/calendar-job.module.ts deleted file mode 100644 index 4888d98fc0f8..000000000000 --- a/packages/twenty-server/src/modules/calendar/jobs/calendar-job.module.ts +++ /dev/null @@ -1,57 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { BlocklistItemDeleteCalendarEventsJob } from 'src/modules/calendar/jobs/blocklist-item-delete-calendar-events.job'; -import { BlocklistReimportCalendarEventsJob } from 'src/modules/calendar/jobs/blocklist-reimport-calendar-events.job'; -import { CalendarCreateCompanyAndContactAfterSyncJob } from 'src/modules/calendar/jobs/calendar-create-company-and-contact-after-sync.job'; -import { DeleteConnectedAccountAssociatedCalendarDataJob } from 'src/modules/calendar/jobs/delete-connected-account-associated-calendar-data.job'; -import { GoogleCalendarSyncJob } from 'src/modules/calendar/jobs/google-calendar-sync.job'; -import { CalendarEventCleanerModule } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module'; -import { GoogleCalendarSyncModule } from 'src/modules/calendar/services/google-calendar-sync/google-calendar-sync.module'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { AutoCompaniesAndContactsCreationModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.module'; -import { GoogleAPIRefreshAccessTokenModule } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.module'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; - -@Module({ - imports: [ - ObjectMetadataRepositoryModule.forFeature([ - CalendarChannelWorkspaceEntity, - CalendarChannelEventAssociationWorkspaceEntity, - CalendarEventParticipantWorkspaceEntity, - ConnectedAccountWorkspaceEntity, - BlocklistWorkspaceEntity, - ]), - CalendarEventCleanerModule, - AutoCompaniesAndContactsCreationModule, - GoogleCalendarSyncModule, - GoogleAPIRefreshAccessTokenModule, - GoogleCalendarSyncModule, - ], - providers: [ - { - provide: BlocklistItemDeleteCalendarEventsJob.name, - useClass: BlocklistItemDeleteCalendarEventsJob, - }, - { - provide: BlocklistReimportCalendarEventsJob.name, - useClass: BlocklistReimportCalendarEventsJob, - }, - { - provide: GoogleCalendarSyncJob.name, - useClass: GoogleCalendarSyncJob, - }, - { - provide: CalendarCreateCompanyAndContactAfterSyncJob.name, - useClass: CalendarCreateCompanyAndContactAfterSyncJob, - }, - { - provide: DeleteConnectedAccountAssociatedCalendarDataJob.name, - useClass: DeleteConnectedAccountAssociatedCalendarDataJob, - }, - ], -}) -export class CalendarJobModule {} diff --git a/packages/twenty-server/src/modules/calendar/jobs/google-calendar-sync.job.ts b/packages/twenty-server/src/modules/calendar/jobs/google-calendar-sync.job.ts deleted file mode 100644 index dd7a3362d3a6..000000000000 --- a/packages/twenty-server/src/modules/calendar/jobs/google-calendar-sync.job.ts +++ /dev/null @@ -1,47 +0,0 @@ -import { Injectable, Logger } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service'; -import { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service'; - -export type GoogleCalendarSyncJobData = { - workspaceId: string; - connectedAccountId: string; -}; - -@Injectable() -export class GoogleCalendarSyncJob - implements MessageQueueJob<GoogleCalendarSyncJobData> -{ - private readonly logger = new Logger(GoogleCalendarSyncJob.name); - - constructor( - private readonly googleAPIsRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService, - private readonly googleCalendarSyncService: GoogleCalendarSyncService, - ) {} - - async handle(data: GoogleCalendarSyncJobData): Promise<void> { - this.logger.log( - `google calendar sync for workspace ${data.workspaceId} and account ${data.connectedAccountId}`, - ); - try { - await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken( - data.workspaceId, - data.connectedAccountId, - ); - } catch (e) { - this.logger.error( - `Error refreshing access token for connected account ${data.connectedAccountId} in workspace ${data.workspaceId}`, - e, - ); - - return; - } - - await this.googleCalendarSyncService.startGoogleCalendarSync( - data.workspaceId, - data.connectedAccountId, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook.ts b/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook.ts deleted file mode 100644 index a5c8b06aa649..000000000000 --- a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/calendar-event-find-many.pre-query.hook.ts +++ /dev/null @@ -1,52 +0,0 @@ -import { - BadRequestException, - Injectable, - NotFoundException, -} from '@nestjs/common'; - -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; -import { FindManyResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; - -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { CalendarChannelEventAssociationRepository } from 'src/modules/calendar/repositories/calendar-channel-event-association.repository'; -import { CanAccessCalendarEventService } from 'src/modules/calendar/query-hooks/calendar-event/services/can-access-calendar-event.service'; - -@Injectable() -export class CalendarEventFindManyPreQueryHook - implements WorkspacePreQueryHook -{ - constructor( - @InjectObjectMetadataRepository( - CalendarChannelEventAssociationWorkspaceEntity, - ) - private readonly calendarChannelEventAssociationRepository: CalendarChannelEventAssociationRepository, - private readonly canAccessCalendarEventService: CanAccessCalendarEventService, - ) {} - - async execute( - userId: string, - workspaceId: string, - payload: FindManyResolverArgs, - ): Promise<void> { - if (!payload?.filter?.id?.eq) { - throw new BadRequestException('id filter is required'); - } - - const calendarChannelCalendarEventAssociations = - await this.calendarChannelEventAssociationRepository.getByCalendarEventIds( - [payload?.filter?.id?.eq], - workspaceId, - ); - - if (calendarChannelCalendarEventAssociations.length === 0) { - throw new NotFoundException(); - } - - await this.canAccessCalendarEventService.canAccessCalendarEvent( - userId, - workspaceId, - calendarChannelCalendarEventAssociations, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook.ts b/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook.ts deleted file mode 100644 index 082e1b280185..000000000000 --- a/packages/twenty-server/src/modules/calendar/query-hooks/calendar-event/calendar-event-find-one.pre-query-hook.ts +++ /dev/null @@ -1,50 +0,0 @@ -import { - BadRequestException, - Injectable, - NotFoundException, -} from '@nestjs/common'; - -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; -import { FindOneResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; - -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CanAccessCalendarEventService } from 'src/modules/calendar/query-hooks/calendar-event/services/can-access-calendar-event.service'; -import { CalendarChannelEventAssociationRepository } from 'src/modules/calendar/repositories/calendar-channel-event-association.repository'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; - -@Injectable() -export class CalendarEventFindOnePreQueryHook implements WorkspacePreQueryHook { - constructor( - @InjectObjectMetadataRepository( - CalendarChannelEventAssociationWorkspaceEntity, - ) - private readonly calendarChannelEventAssociationRepository: CalendarChannelEventAssociationRepository, - private readonly canAccessCalendarEventService: CanAccessCalendarEventService, - ) {} - - async execute( - userId: string, - workspaceId: string, - payload: FindOneResolverArgs, - ): Promise<void> { - if (!payload?.filter?.id?.eq) { - throw new BadRequestException('id filter is required'); - } - - const calendarChannelCalendarEventAssociations = - await this.calendarChannelEventAssociationRepository.getByCalendarEventIds( - [payload?.filter?.id?.eq], - workspaceId, - ); - - if (calendarChannelCalendarEventAssociations.length === 0) { - throw new NotFoundException(); - } - - await this.canAccessCalendarEventService.canAccessCalendarEvent( - userId, - workspaceId, - calendarChannelCalendarEventAssociations, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/repositories/calendar-channel-event-association.repository.ts b/packages/twenty-server/src/modules/calendar/repositories/calendar-channel-event-association.repository.ts deleted file mode 100644 index 89a9a28b486f..000000000000 --- a/packages/twenty-server/src/modules/calendar/repositories/calendar-channel-event-association.repository.ts +++ /dev/null @@ -1,205 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { EntityManager } from 'typeorm'; - -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; - -@Injectable() -export class CalendarChannelEventAssociationRepository { - constructor( - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - ) {} - - public async getByEventExternalIdsAndCalendarChannelId( - eventExternalIds: string[], - calendarChannelId: string, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarChannelEventAssociationWorkspaceEntity>[]> { - if (eventExternalIds.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarChannelEventAssociation" - WHERE "eventExternalId" = ANY($1) AND "calendarChannelId" = $2`, - [eventExternalIds, calendarChannelId], - workspaceId, - transactionManager, - ); - } - - public async deleteByEventExternalIdsAndCalendarChannelId( - eventExternalIds: string[], - calendarChannelId: string, - workspaceId: string, - transactionManager?: EntityManager, - ) { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."calendarChannelEventAssociation" WHERE "eventExternalId" = ANY($1) AND "calendarChannelId" = $2`, - [eventExternalIds, calendarChannelId], - workspaceId, - transactionManager, - ); - } - - public async getByCalendarChannelIds( - calendarChannelIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarChannelEventAssociationWorkspaceEntity>[]> { - if (calendarChannelIds.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarChannelEventAssociation" - WHERE "calendarChannelId" = ANY($1)`, - [calendarChannelIds], - workspaceId, - transactionManager, - ); - } - - public async deleteByCalendarChannelIds( - calendarChannelIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ) { - if (calendarChannelIds.length === 0) { - return; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."calendarChannelEventAssociation" WHERE "calendarChannelId" = ANY($1)`, - [calendarChannelIds], - workspaceId, - transactionManager, - ); - } - - public async deleteByIds( - ids: string[], - workspaceId: string, - transactionManager?: EntityManager, - ) { - if (ids.length === 0) { - return; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."calendarChannelEventAssociation" WHERE "id" = ANY($1)`, - [ids], - workspaceId, - transactionManager, - ); - } - - public async getByCalendarEventIds( - calendarEventIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarChannelEventAssociationWorkspaceEntity>[]> { - if (calendarEventIds.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarChannelEventAssociation" - WHERE "calendarEventId" = ANY($1)`, - [calendarEventIds], - workspaceId, - transactionManager, - ); - } - - public async saveCalendarChannelEventAssociations( - calendarChannelEventAssociations: Omit< - ObjectRecord<CalendarChannelEventAssociationWorkspaceEntity>, - 'id' | 'createdAt' | 'updatedAt' | 'calendarChannel' | 'calendarEvent' - >[], - workspaceId: string, - transactionManager?: EntityManager, - ) { - if (calendarChannelEventAssociations.length === 0) { - return; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const { - flattenedValues: calendarChannelEventAssociationValues, - valuesString, - } = getFlattenedValuesAndValuesStringForBatchRawQuery( - calendarChannelEventAssociations, - { - calendarChannelId: 'uuid', - calendarEventId: 'uuid', - eventExternalId: 'text', - }, - ); - - await this.workspaceDataSourceService.executeRawQuery( - `INSERT INTO ${dataSourceSchema}."calendarChannelEventAssociation" ("calendarChannelId", "calendarEventId", "eventExternalId") - VALUES ${valuesString}`, - calendarChannelEventAssociationValues, - workspaceId, - transactionManager, - ); - } - - public async deleteByCalendarEventParticipantHandleAndCalendarChannelIds( - calendarEventParticipantHandle: string, - calendarChannelIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ) { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const isHandleDomain = calendarEventParticipantHandle.startsWith('@'); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."calendarChannelEventAssociation" - WHERE "id" IN ( - SELECT "calendarChannelEventAssociation"."id" - FROM ${dataSourceSchema}."calendarChannelEventAssociation" "calendarChannelEventAssociation" - JOIN ${dataSourceSchema}."calendarEvent" "calendarEvent" ON "calendarChannelEventAssociation"."calendarEventId" = "calendarEvent"."id" - JOIN ${dataSourceSchema}."calendarEventParticipant" "calendarEventParticipant" ON "calendarEvent"."id" = "calendarEventParticipant"."calendarEventId" - WHERE "calendarEventParticipant"."handle" ${ - isHandleDomain ? 'ILIKE' : '=' - } $1 AND "calendarChannelEventAssociation"."calendarChannelId" = ANY($2) - )`, - [ - isHandleDomain - ? `%${calendarEventParticipantHandle}` - : calendarEventParticipantHandle, - calendarChannelIds, - ], - workspaceId, - transactionManager, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/repositories/calendar-channel.repository.ts b/packages/twenty-server/src/modules/calendar/repositories/calendar-channel.repository.ts deleted file mode 100644 index d0cde3ca82fd..000000000000 --- a/packages/twenty-server/src/modules/calendar/repositories/calendar-channel.repository.ts +++ /dev/null @@ -1,135 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { EntityManager } from 'typeorm'; - -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; - -@Injectable() -export class CalendarChannelRepository { - constructor( - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - ) {} - - public async getAll( - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarChannelWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarChannel"`, - [], - workspaceId, - transactionManager, - ); - } - - public async create( - calendarChannel: Pick< - ObjectRecord<CalendarChannelWorkspaceEntity>, - 'id' | 'connectedAccountId' | 'handle' | 'visibility' - >, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<void> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `INSERT INTO ${dataSourceSchema}."calendarChannel" (id, "connectedAccountId", "handle", "visibility") VALUES ($1, $2, $3, $4)`, - [ - calendarChannel.id, - calendarChannel.connectedAccountId, - calendarChannel.handle, - calendarChannel.visibility, - ], - workspaceId, - transactionManager, - ); - } - - public async getByConnectedAccountId( - connectedAccountId: string, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarChannelWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarChannel" WHERE "connectedAccountId" = $1 LIMIT 1`, - [connectedAccountId], - workspaceId, - transactionManager, - ); - } - - public async getFirstByConnectedAccountId( - connectedAccountId: string, - workspaceId: string, - ): Promise<ObjectRecord<CalendarChannelWorkspaceEntity> | undefined> { - const calendarChannels = await this.getByConnectedAccountId( - connectedAccountId, - workspaceId, - ); - - return calendarChannels[0]; - } - - public async getByIds( - ids: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarChannelWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarChannel" WHERE "id" = ANY($1)`, - [ids], - workspaceId, - transactionManager, - ); - } - - public async getIdsByWorkspaceMemberId( - workspaceMemberId: string, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarChannelWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const calendarChannelIds = - await this.workspaceDataSourceService.executeRawQuery( - `SELECT "calendarChannel".id FROM ${dataSourceSchema}."calendarChannel" "calendarChannel" - JOIN ${dataSourceSchema}."connectedAccount" ON "calendarChannel"."connectedAccountId" = ${dataSourceSchema}."connectedAccount"."id" - WHERE ${dataSourceSchema}."connectedAccount"."accountOwnerId" = $1`, - [workspaceMemberId], - workspaceId, - transactionManager, - ); - - return calendarChannelIds; - } - - public async updateSyncCursor( - syncCursor: string | null, - calendarChannelId: string, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<void> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarChannel" SET "syncCursor" = $1 WHERE "id" = $2`, - [syncCursor || '', calendarChannelId], - workspaceId, - transactionManager, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/repositories/calendar-event-participant.repository.ts b/packages/twenty-server/src/modules/calendar/repositories/calendar-event-participant.repository.ts deleted file mode 100644 index f095e100e7d0..000000000000 --- a/packages/twenty-server/src/modules/calendar/repositories/calendar-event-participant.repository.ts +++ /dev/null @@ -1,305 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { EntityManager } from 'typeorm'; -import differenceWith from 'lodash.differencewith'; - -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; -import { - CalendarEventParticipant, - CalendarEventParticipantWithId, -} from 'src/modules/calendar/types/calendar-event'; - -@Injectable() -export class CalendarEventParticipantRepository { - constructor( - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - ) {} - - public async getByHandles( - handles: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarEventParticipant" WHERE "handle" = ANY($1)`, - [handles], - workspaceId, - transactionManager, - ); - } - - public async updateParticipantsPersonId( - participantIds: string[], - personId: string, - workspaceId: string, - transactionManager?: EntityManager, - ) { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEventParticipant" SET "personId" = $1 WHERE "id" = ANY($2)`, - [personId, participantIds], - workspaceId, - transactionManager, - ); - } - - public async updateParticipantsPersonIdAndReturn( - participantIds: string[], - personId: string, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEventParticipant" SET "personId" = $1 WHERE "id" = ANY($2) RETURNING *`, - [personId, participantIds], - workspaceId, - transactionManager, - ); - } - - public async updateParticipantsWorkspaceMemberId( - participantIds: string[], - workspaceMemberId: string, - workspaceId: string, - transactionManager?: EntityManager, - ) { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEventParticipant" SET "workspaceMemberId" = $1 WHERE "id" = ANY($2)`, - [workspaceMemberId, participantIds], - workspaceId, - transactionManager, - ); - } - - public async removePersonIdByHandle( - handle: string, - workspaceId: string, - transactionManager?: EntityManager, - ) { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEventParticipant" SET "personId" = NULL WHERE "handle" = $1`, - [handle], - workspaceId, - transactionManager, - ); - } - - public async removeWorkspaceMemberIdByHandle( - handle: string, - workspaceId: string, - transactionManager?: EntityManager, - ) { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEventParticipant" SET "workspaceMemberId" = NULL WHERE "handle" = $1`, - [handle], - workspaceId, - transactionManager, - ); - } - - public async getByIds( - calendarEventParticipantIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]> { - if (calendarEventParticipantIds.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarEventParticipant" WHERE "id" = ANY($1)`, - [calendarEventParticipantIds], - workspaceId, - transactionManager, - ); - } - - public async getByCalendarEventIds( - calendarEventIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]> { - if (calendarEventIds.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarEventParticipant" WHERE "calendarEventId" = ANY($1)`, - [calendarEventIds], - workspaceId, - transactionManager, - ); - } - - public async deleteByIds( - calendarEventParticipantIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<void> { - if (calendarEventParticipantIds.length === 0) { - return; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."calendarEventParticipant" WHERE "id" = ANY($1)`, - [calendarEventParticipantIds], - workspaceId, - transactionManager, - ); - } - - public async updateCalendarEventParticipantsAndReturnNewOnes( - calendarEventParticipants: CalendarEventParticipant[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<CalendarEventParticipant[]> { - if (calendarEventParticipants.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const existingCalendarEventParticipants = await this.getByCalendarEventIds( - calendarEventParticipants.map( - (calendarEventParticipant) => calendarEventParticipant.calendarEventId, - ), - workspaceId, - transactionManager, - ); - - const calendarEventParticipantsToDelete = differenceWith( - existingCalendarEventParticipants, - calendarEventParticipants, - (existingCalendarEventParticipant, calendarEventParticipant) => - existingCalendarEventParticipant.handle === - calendarEventParticipant.handle, - ); - - const newCalendarEventParticipants = differenceWith( - calendarEventParticipants, - existingCalendarEventParticipants, - (calendarEventParticipant, existingCalendarEventParticipant) => - calendarEventParticipant.handle === - existingCalendarEventParticipant.handle, - ); - - await this.deleteByIds( - calendarEventParticipantsToDelete.map( - (calendarEventParticipant) => calendarEventParticipant.id, - ), - workspaceId, - transactionManager, - ); - - const { flattenedValues, valuesString } = - getFlattenedValuesAndValuesStringForBatchRawQuery( - calendarEventParticipants, - { - calendarEventId: 'uuid', - handle: 'text', - displayName: 'text', - isOrganizer: 'boolean', - responseStatus: `${dataSourceSchema}."calendarEventParticipant_responseStatus_enum"`, - }, - ); - - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEventParticipant" AS "calendarEventParticipant" - SET "displayName" = "newValues"."displayName", - "isOrganizer" = "newValues"."isOrganizer", - "responseStatus" = "newValues"."responseStatus" - FROM (VALUES ${valuesString}) AS "newValues"("calendarEventId", "handle", "displayName", "isOrganizer", "responseStatus") - WHERE "calendarEventParticipant"."handle" = "newValues"."handle" - AND "calendarEventParticipant"."calendarEventId" = "newValues"."calendarEventId"`, - flattenedValues, - workspaceId, - transactionManager, - ); - - return newCalendarEventParticipants; - } - - public async getWithoutPersonIdAndWorkspaceMemberId( - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<CalendarEventParticipantWithId[]> { - if (!workspaceId) { - throw new Error('WorkspaceId is required'); - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const calendarEventParticipants: CalendarEventParticipantWithId[] = - await this.workspaceDataSourceService.executeRawQuery( - `SELECT "calendarEventParticipant".* - FROM ${dataSourceSchema}."calendarEventParticipant" AS "calendarEventParticipant" - WHERE "calendarEventParticipant"."personId" IS NULL - AND "calendarEventParticipant"."workspaceMemberId" IS NULL`, - [], - workspaceId, - transactionManager, - ); - - return calendarEventParticipants; - } - - public async getByCalendarChannelIdWithoutPersonIdAndWorkspaceMemberId( - calendarChannelId: string, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<CalendarEventParticipantWithId[]> { - if (!workspaceId) { - throw new Error('WorkspaceId is required'); - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const calendarEventParticipants: CalendarEventParticipantWithId[] = - await this.workspaceDataSourceService.executeRawQuery( - `SELECT "calendarEventParticipant".* - FROM ${dataSourceSchema}."calendarEventParticipant" AS "calendarEventParticipant" - LEFT JOIN ${dataSourceSchema}."calendarEvent" AS "calendarEvent" ON "calendarEventParticipant"."calendarEventId" = "calendarEvent"."id" - LEFT JOIN ${dataSourceSchema}."calendarChannelEventAssociation" AS "calendarChannelEventAssociation" ON "calendarEvent"."id" = "calendarChannelEventAssociation"."calendarEventId" - WHERE "calendarChannelEventAssociation"."calendarChannelId" = $1 - AND "calendarEventParticipant"."personId" IS NULL - AND "calendarEventParticipant"."workspaceMemberId" IS NULL`, - [calendarChannelId], - workspaceId, - transactionManager, - ); - - return calendarEventParticipants; - } -} diff --git a/packages/twenty-server/src/modules/calendar/repositories/calendar-event.repository.ts b/packages/twenty-server/src/modules/calendar/repositories/calendar-event.repository.ts deleted file mode 100644 index f6f28918dd1d..000000000000 --- a/packages/twenty-server/src/modules/calendar/repositories/calendar-event.repository.ts +++ /dev/null @@ -1,227 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { EntityManager } from 'typeorm'; - -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; -import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; -import { CalendarEvent } from 'src/modules/calendar/types/calendar-event'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; - -@Injectable() -export class CalendarEventRepository { - constructor( - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - ) {} - - public async getByIds( - calendarEventIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventWorkspaceEntity>[]> { - if (calendarEventIds.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarEvent" WHERE "id" = ANY($1)`, - [calendarEventIds], - workspaceId, - transactionManager, - ); - } - - public async getByICalUIDs( - iCalUIDs: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventWorkspaceEntity>[]> { - if (iCalUIDs.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - return await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."calendarEvent" WHERE "iCalUID" = ANY($1)`, - [iCalUIDs], - workspaceId, - transactionManager, - ); - } - - public async deleteByIds( - calendarEventIds: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<void> { - if (calendarEventIds.length === 0) { - return; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - await this.workspaceDataSourceService.executeRawQuery( - `DELETE FROM ${dataSourceSchema}."calendarEvent" WHERE "id" = ANY($1)`, - [calendarEventIds], - workspaceId, - transactionManager, - ); - } - - public async getNonAssociatedCalendarEventIdsPaginated( - limit: number, - offset: number, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const nonAssociatedCalendarEvents = - await this.workspaceDataSourceService.executeRawQuery( - `SELECT m.id FROM ${dataSourceSchema}."calendarEvent" m - LEFT JOIN ${dataSourceSchema}."calendarChannelEventAssociation" ccea - ON m.id = ccea."calendarEventId" - WHERE ccea.id IS NULL - LIMIT $1 OFFSET $2`, - [limit, offset], - workspaceId, - transactionManager, - ); - - return nonAssociatedCalendarEvents.map(({ id }) => id); - } - - public async getICalUIDCalendarEventIdMap( - iCalUIDs: string[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<Map<string, string>> { - if (iCalUIDs.length === 0) { - return new Map(); - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const calendarEvents: - | { - id: string; - iCalUID: string; - }[] - | undefined = await this.workspaceDataSourceService.executeRawQuery( - `SELECT id, "iCalUID" FROM ${dataSourceSchema}."calendarEvent" WHERE "iCalUID" = ANY($1)`, - [iCalUIDs], - workspaceId, - transactionManager, - ); - - const iCalUIDsCalendarEventIdsMap = new Map<string, string>(); - - calendarEvents?.forEach((calendarEvent) => { - iCalUIDsCalendarEventIdsMap.set(calendarEvent.iCalUID, calendarEvent.id); - }); - - return iCalUIDsCalendarEventIdsMap; - } - - public async saveCalendarEvents( - calendarEvents: CalendarEvent[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<void> { - if (calendarEvents.length === 0) { - return; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const { flattenedValues, valuesString } = - getFlattenedValuesAndValuesStringForBatchRawQuery(calendarEvents, { - id: 'uuid', - title: 'text', - isCanceled: 'boolean', - isFullDay: 'boolean', - startsAt: 'timestamptz', - endsAt: 'timestamptz', - externalCreatedAt: 'timestamptz', - externalUpdatedAt: 'timestamptz', - description: 'text', - location: 'text', - iCalUID: 'text', - conferenceSolution: 'text', - conferenceLinkLabel: 'text', - conferenceLinkUrl: 'text', - recurringEventExternalId: 'text', - }); - - await this.workspaceDataSourceService.executeRawQuery( - `INSERT INTO ${dataSourceSchema}."calendarEvent" ("id", "title", "isCanceled", "isFullDay", "startsAt", "endsAt", "externalCreatedAt", "externalUpdatedAt", "description", "location", "iCalUID", "conferenceSolution", "conferenceLinkLabel", "conferenceLinkUrl", "recurringEventExternalId") VALUES ${valuesString}`, - flattenedValues, - workspaceId, - transactionManager, - ); - } - - public async updateCalendarEvents( - calendarEvents: CalendarEvent[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<void> { - if (calendarEvents.length === 0) { - return; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const { flattenedValues, valuesString } = - getFlattenedValuesAndValuesStringForBatchRawQuery(calendarEvents, { - title: 'text', - isCanceled: 'boolean', - isFullDay: 'boolean', - startsAt: 'timestamptz', - endsAt: 'timestamptz', - externalCreatedAt: 'timestamptz', - externalUpdatedAt: 'timestamptz', - description: 'text', - location: 'text', - iCalUID: 'text', - conferenceSolution: 'text', - conferenceLinkLabel: 'text', - conferenceLinkUrl: 'text', - recurringEventExternalId: 'text', - }); - - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEvent" AS "calendarEvent" - SET "title" = "newData"."title", - "isCanceled" = "newData"."isCanceled", - "isFullDay" = "newData"."isFullDay", - "startsAt" = "newData"."startsAt", - "endsAt" = "newData"."endsAt", - "externalCreatedAt" = "newData"."externalCreatedAt", - "externalUpdatedAt" = "newData"."externalUpdatedAt", - "description" = "newData"."description", - "location" = "newData"."location", - "conferenceSolution" = "newData"."conferenceSolution", - "conferenceLinkLabel" = "newData"."conferenceLinkLabel", - "conferenceLinkUrl" = "newData"."conferenceLinkUrl", - "recurringEventExternalId" = "newData"."recurringEventExternalId" - FROM (VALUES ${valuesString}) - AS "newData"("title", "isCanceled", "isFullDay", "startsAt", "endsAt", "externalCreatedAt", "externalUpdatedAt", "description", "location", "iCalUID", "conferenceSolution", "conferenceLinkLabel", "conferenceLinkUrl", "recurringEventExternalId") - WHERE "calendarEvent"."iCalUID" = "newData"."iCalUID"`, - flattenedValues, - workspaceId, - transactionManager, - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module.ts b/packages/twenty-server/src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module.ts deleted file mode 100644 index 644371db16f6..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { CalendarEventCleanerService } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; - -@Module({ - imports: [ - ObjectMetadataRepositoryModule.forFeature([CalendarEventWorkspaceEntity]), - ], - providers: [CalendarEventCleanerService], - exports: [CalendarEventCleanerService], -}) -export class CalendarEventCleanerModule {} diff --git a/packages/twenty-server/src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service.ts b/packages/twenty-server/src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service.ts deleted file mode 100644 index 9f3f5d520298..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CalendarEventRepository } from 'src/modules/calendar/repositories/calendar-event.repository'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; -import { deleteUsingPagination } from 'src/modules/messaging/message-cleaner/utils/delete-using-pagination.util'; - -@Injectable() -export class CalendarEventCleanerService { - constructor( - @InjectObjectMetadataRepository(CalendarEventWorkspaceEntity) - private readonly calendarEventRepository: CalendarEventRepository, - ) {} - - public async cleanWorkspaceCalendarEvents(workspaceId: string) { - await deleteUsingPagination( - workspaceId, - 500, - this.calendarEventRepository.getNonAssociatedCalendarEventIdsPaginated.bind( - this.calendarEventRepository, - ), - this.calendarEventRepository.deleteByIds.bind( - this.calendarEventRepository, - ), - ); - } -} diff --git a/packages/twenty-server/src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module.ts b/packages/twenty-server/src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module.ts deleted file mode 100644 index 4eabeedb997a..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module.ts +++ /dev/null @@ -1,18 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { AddPersonIdAndWorkspaceMemberIdModule } from 'src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.module'; -import { CalendarEventParticipantService } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service'; -import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; - -@Module({ - imports: [ - WorkspaceDataSourceModule, - ObjectMetadataRepositoryModule.forFeature([PersonWorkspaceEntity]), - AddPersonIdAndWorkspaceMemberIdModule, - ], - providers: [CalendarEventParticipantService], - exports: [CalendarEventParticipantService], -}) -export class CalendarEventParticipantModule {} diff --git a/packages/twenty-server/src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service.ts b/packages/twenty-server/src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service.ts deleted file mode 100644 index 4b524de0dedc..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service.ts +++ /dev/null @@ -1,185 +0,0 @@ -import { Injectable } from '@nestjs/common'; -import { EventEmitter2 } from '@nestjs/event-emitter'; - -import { EntityManager } from 'typeorm'; - -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { PersonRepository } from 'src/modules/person/repositories/person.repository'; -import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; -import { CalendarEventParticipant } from 'src/modules/calendar/types/calendar-event'; -import { CalendarEventParticipantRepository } from 'src/modules/calendar/repositories/calendar-event-participant.repository'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; - -@Injectable() -export class CalendarEventParticipantService { - constructor( - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - @InjectObjectMetadataRepository(CalendarEventParticipantWorkspaceEntity) - private readonly calendarEventParticipantRepository: CalendarEventParticipantRepository, - @InjectObjectMetadataRepository(PersonWorkspaceEntity) - private readonly personRepository: PersonRepository, - private readonly addPersonIdAndWorkspaceMemberIdService: AddPersonIdAndWorkspaceMemberIdService, - private readonly eventEmitter: EventEmitter2, - ) {} - - public async updateCalendarEventParticipantsAfterPeopleCreation( - createdPeople: ObjectRecord<PersonWorkspaceEntity>[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]> { - const participants = - await this.calendarEventParticipantRepository.getByHandles( - createdPeople.map((person) => person.email), - workspaceId, - ); - - if (!participants) return []; - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const handles = participants.map((participant) => participant.handle); - - const participantPersonIds = await this.personRepository.getByEmails( - handles, - workspaceId, - transactionManager, - ); - - const calendarEventParticipantsToUpdate = participants.map( - (participant) => ({ - id: participant.id, - personId: participantPersonIds.find( - (e: { id: string; email: string }) => e.email === participant.handle, - )?.id, - }), - ); - - if (calendarEventParticipantsToUpdate.length === 0) return []; - - const { flattenedValues, valuesString } = - getFlattenedValuesAndValuesStringForBatchRawQuery( - calendarEventParticipantsToUpdate, - { - id: 'uuid', - personId: 'uuid', - }, - ); - - return ( - await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."calendarEventParticipant" AS "calendarEventParticipant" SET "personId" = "data"."personId" - FROM (VALUES ${valuesString}) AS "data"("id", "personId") - WHERE "calendarEventParticipant"."id" = "data"."id" - RETURNING *`, - flattenedValues, - workspaceId, - transactionManager, - ) - ).flat(); - } - - public async saveCalendarEventParticipants( - calendarEventParticipants: CalendarEventParticipant[], - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<CalendarEventParticipantWorkspaceEntity>[]> { - if (calendarEventParticipants.length === 0) { - return []; - } - - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const calendarEventParticipantsToSave = - await this.addPersonIdAndWorkspaceMemberIdService.addPersonIdAndWorkspaceMemberId( - calendarEventParticipants, - workspaceId, - transactionManager, - ); - - const { flattenedValues, valuesString } = - getFlattenedValuesAndValuesStringForBatchRawQuery( - calendarEventParticipantsToSave, - { - calendarEventId: 'uuid', - handle: 'text', - displayName: 'text', - isOrganizer: 'boolean', - responseStatus: `${dataSourceSchema}."calendarEventParticipant_responseStatus_enum"`, - personId: 'uuid', - workspaceMemberId: 'uuid', - }, - ); - - return await this.workspaceDataSourceService.executeRawQuery( - `INSERT INTO ${dataSourceSchema}."calendarEventParticipant" ("calendarEventId", "handle", "displayName", "isOrganizer", "responseStatus", "personId", "workspaceMemberId") VALUES ${valuesString} - RETURNING *`, - flattenedValues, - workspaceId, - transactionManager, - ); - } - - public async matchCalendarEventParticipants( - workspaceId: string, - email: string, - personId?: string, - workspaceMemberId?: string, - ) { - const calendarEventParticipantsToUpdate = - await this.calendarEventParticipantRepository.getByHandles( - [email], - workspaceId, - ); - - const calendarEventParticipantIdsToUpdate = - calendarEventParticipantsToUpdate.map((participant) => participant.id); - - if (personId) { - const updatedCalendarEventParticipants = - await this.calendarEventParticipantRepository.updateParticipantsPersonIdAndReturn( - calendarEventParticipantIdsToUpdate, - personId, - workspaceId, - ); - - this.eventEmitter.emit(`calendarEventParticipant.matched`, { - workspaceId, - userId: null, - calendarEventParticipants: updatedCalendarEventParticipants, - }); - } - if (workspaceMemberId) { - await this.calendarEventParticipantRepository.updateParticipantsWorkspaceMemberId( - calendarEventParticipantIdsToUpdate, - workspaceMemberId, - workspaceId, - ); - } - } - - public async unmatchCalendarEventParticipants( - workspaceId: string, - handle: string, - personId?: string, - workspaceMemberId?: string, - ) { - if (personId) { - await this.calendarEventParticipantRepository.removePersonIdByHandle( - handle, - workspaceId, - ); - } - if (workspaceMemberId) { - await this.calendarEventParticipantRepository.removeWorkspaceMemberIdByHandle( - handle, - workspaceId, - ); - } - } -} diff --git a/packages/twenty-server/src/modules/calendar/services/google-calendar-sync/google-calendar-sync.module.ts b/packages/twenty-server/src/modules/calendar/services/google-calendar-sync/google-calendar-sync.module.ts deleted file mode 100644 index fcd5e495ce19..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/google-calendar-sync/google-calendar-sync.module.ts +++ /dev/null @@ -1,41 +0,0 @@ -import { Module } from '@nestjs/common'; -import { TypeOrmModule } from '@nestjs/typeorm'; - -import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { CalendarEventCleanerModule } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.module'; -import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module'; -import { GoogleCalendarSyncService } from 'src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service'; -import { CalendarProvidersModule } from 'src/modules/calendar/services/providers/calendar-providers.module'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; - -@Module({ - imports: [ - CalendarProvidersModule, - ObjectMetadataRepositoryModule.forFeature([ - ConnectedAccountWorkspaceEntity, - CalendarEventWorkspaceEntity, - CalendarChannelWorkspaceEntity, - CalendarChannelEventAssociationWorkspaceEntity, - CalendarEventParticipantWorkspaceEntity, - BlocklistWorkspaceEntity, - PersonWorkspaceEntity, - WorkspaceMemberWorkspaceEntity, - ]), - CalendarEventParticipantModule, - TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), - WorkspaceDataSourceModule, - CalendarEventCleanerModule, - ], - providers: [GoogleCalendarSyncService], - exports: [GoogleCalendarSyncService], -}) -export class GoogleCalendarSyncModule {} diff --git a/packages/twenty-server/src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service.ts b/packages/twenty-server/src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service.ts deleted file mode 100644 index d0c8663d86d4..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/google-calendar-sync/google-calendar-sync.service.ts +++ /dev/null @@ -1,534 +0,0 @@ -import { Injectable, Logger } from '@nestjs/common'; -import { InjectRepository } from '@nestjs/typeorm'; -import { EventEmitter2 } from '@nestjs/event-emitter'; - -import { Repository } from 'typeorm'; -import { calendar_v3 as calendarV3 } from 'googleapis'; -import { GaxiosError } from 'gaxios'; - -import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; -import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository'; -import { - FeatureFlagEntity, - FeatureFlagKeys, -} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { GoogleCalendarClientProvider } from 'src/modules/calendar/services/providers/google-calendar/google-calendar.provider'; -import { CalendarChannelEventAssociationRepository } from 'src/modules/calendar/repositories/calendar-channel-event-association.repository'; -import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { CalendarEventRepository } from 'src/modules/calendar/repositories/calendar-event.repository'; -import { formatGoogleCalendarEvent } from 'src/modules/calendar/utils/format-google-calendar-event.util'; -import { CalendarEventParticipantRepository } from 'src/modules/calendar/repositories/calendar-event-participant.repository'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CalendarEventWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event.workspace-entity'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; -import { CalendarEventCleanerService } from 'src/modules/calendar/services/calendar-event-cleaner/calendar-event-cleaner.service'; -import { CalendarEventParticipantService } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service'; -import { CalendarEventWithParticipants } from 'src/modules/calendar/types/calendar-event'; -import { filterOutBlocklistedEvents } from 'src/modules/calendar/utils/filter-out-blocklisted-events.util'; -import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; -import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; -import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { - CreateCompanyAndContactJob, - CreateCompanyAndContactJobData, -} from 'src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; - -@Injectable() -export class GoogleCalendarSyncService { - private readonly logger = new Logger(GoogleCalendarSyncService.name); - - constructor( - private readonly googleCalendarClientProvider: GoogleCalendarClientProvider, - @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) - private readonly connectedAccountRepository: ConnectedAccountRepository, - @InjectObjectMetadataRepository(CalendarEventWorkspaceEntity) - private readonly calendarEventRepository: CalendarEventRepository, - @InjectObjectMetadataRepository(CalendarChannelWorkspaceEntity) - private readonly calendarChannelRepository: CalendarChannelRepository, - @InjectObjectMetadataRepository( - CalendarChannelEventAssociationWorkspaceEntity, - ) - private readonly calendarChannelEventAssociationRepository: CalendarChannelEventAssociationRepository, - @InjectObjectMetadataRepository(CalendarEventParticipantWorkspaceEntity) - private readonly calendarEventParticipantsRepository: CalendarEventParticipantRepository, - @InjectObjectMetadataRepository(BlocklistWorkspaceEntity) - private readonly blocklistRepository: BlocklistRepository, - @InjectRepository(FeatureFlagEntity, 'core') - private readonly featureFlagRepository: Repository<FeatureFlagEntity>, - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - private readonly calendarEventCleanerService: CalendarEventCleanerService, - private readonly calendarEventParticipantsService: CalendarEventParticipantService, - @InjectMessageQueue(MessageQueue.emailQueue) - private readonly messageQueueService: MessageQueueService, - private readonly eventEmitter: EventEmitter2, - ) {} - - public async startGoogleCalendarSync( - workspaceId: string, - connectedAccountId: string, - emailOrDomainToReimport?: string, - ): Promise<void> { - const connectedAccount = await this.connectedAccountRepository.getById( - connectedAccountId, - workspaceId, - ); - - if (!connectedAccount) { - return; - } - - const refreshToken = connectedAccount.refreshToken; - const workspaceMemberId = connectedAccount.accountOwnerId; - - if (!refreshToken) { - throw new Error( - `No refresh token found for connected account ${connectedAccountId} in workspace ${workspaceId} during sync`, - ); - } - - const calendarChannel = - await this.calendarChannelRepository.getFirstByConnectedAccountId( - connectedAccountId, - workspaceId, - ); - - const syncToken = calendarChannel?.syncCursor || undefined; - - if (!calendarChannel) { - return; - } - - const calendarChannelId = calendarChannel.id; - - const { events, nextSyncToken } = await this.getEventsFromGoogleCalendar( - refreshToken, - workspaceId, - connectedAccountId, - emailOrDomainToReimport, - syncToken, - ); - - if (!events || events?.length === 0) { - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} done with nothing to import.`, - ); - - return; - } - - const blocklist = await this.getBlocklist(workspaceMemberId, workspaceId); - - let filteredEvents = filterOutBlocklistedEvents( - calendarChannel.handle, - events, - blocklist, - ).filter((event) => event.status !== 'cancelled'); - - if (emailOrDomainToReimport) { - filteredEvents = filteredEvents.filter( - (event) => - event.attendees?.some( - (attendee) => attendee.email?.endsWith(emailOrDomainToReimport), - ), - ); - } - - const cancelledEventExternalIds = filteredEvents - .filter((event) => event.status === 'cancelled') - .map((event) => event.id as string); - - const iCalUIDCalendarEventIdMap = - await this.calendarEventRepository.getICalUIDCalendarEventIdMap( - filteredEvents.map((calendarEvent) => calendarEvent.iCalUID as string), - workspaceId, - ); - - const formattedEvents = filteredEvents.map((event) => - formatGoogleCalendarEvent(event, iCalUIDCalendarEventIdMap), - ); - - // TODO: When we will be able to add unicity contraint on iCalUID, we will do a INSERT ON CONFLICT DO UPDATE - - let startTime = Date.now(); - - const existingEvents = await this.calendarEventRepository.getByICalUIDs( - formattedEvents.map((event) => event.iCalUID), - workspaceId, - ); - - const existingEventsICalUIDs = existingEvents.map((event) => event.iCalUID); - - let endTime = Date.now(); - - const eventsToSave = formattedEvents.filter( - (event) => !existingEventsICalUIDs.includes(event.iCalUID), - ); - - const eventsToUpdate = formattedEvents.filter((event) => - existingEventsICalUIDs.includes(event.iCalUID), - ); - - startTime = Date.now(); - - const existingCalendarChannelEventAssociations = - await this.calendarChannelEventAssociationRepository.getByEventExternalIdsAndCalendarChannelId( - formattedEvents.map((event) => event.externalId), - calendarChannelId, - workspaceId, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: getting existing calendar channel event associations in ${ - endTime - startTime - }ms.`, - ); - - const calendarChannelEventAssociationsToSave = formattedEvents - .filter( - (event) => - !existingCalendarChannelEventAssociations.some( - (association) => association.eventExternalId === event.id, - ), - ) - .map((event) => ({ - calendarEventId: event.id, - eventExternalId: event.externalId, - calendarChannelId, - })); - - if (events.length > 0) { - await this.saveGoogleCalendarEvents( - eventsToSave, - eventsToUpdate, - calendarChannelEventAssociationsToSave, - connectedAccount, - calendarChannel, - workspaceId, - ); - - startTime = Date.now(); - - await this.calendarChannelEventAssociationRepository.deleteByEventExternalIdsAndCalendarChannelId( - cancelledEventExternalIds, - calendarChannelId, - workspaceId, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: deleting calendar channel event associations in ${ - endTime - startTime - }ms.`, - ); - - startTime = Date.now(); - - await this.calendarEventCleanerService.cleanWorkspaceCalendarEvents( - workspaceId, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: cleaning calendar events in ${ - endTime - startTime - }ms.`, - ); - } else { - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} done with nothing to import.`, - ); - } - - if (!nextSyncToken) { - throw new Error( - `No next sync token found for connected account ${connectedAccountId} in workspace ${workspaceId} during sync`, - ); - } - - startTime = Date.now(); - - await this.calendarChannelRepository.updateSyncCursor( - nextSyncToken, - calendarChannel.id, - workspaceId, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId}: updating sync cursor in ${ - endTime - startTime - }ms.`, - ); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} ${ - syncToken ? `and ${syncToken} syncToken ` : '' - }done.`, - ); - } - - public async getBlocklist(workspaceMemberId: string, workspaceId: string) { - const isBlocklistEnabledFeatureFlag = - await this.featureFlagRepository.findOneBy({ - workspaceId, - key: FeatureFlagKeys.IsBlocklistEnabled, - value: true, - }); - - const isBlocklistEnabled = - isBlocklistEnabledFeatureFlag && isBlocklistEnabledFeatureFlag.value; - - const blocklist = isBlocklistEnabled - ? await this.blocklistRepository.getByWorkspaceMemberId( - workspaceMemberId, - workspaceId, - ) - : []; - - return blocklist.map((blocklist) => blocklist.handle); - } - - public async getEventsFromGoogleCalendar( - refreshToken: string, - workspaceId: string, - connectedAccountId: string, - emailOrDomainToReimport?: string, - syncToken?: string, - ): Promise<{ - events: calendarV3.Schema$Event[]; - nextSyncToken: string | null | undefined; - }> { - const googleCalendarClient = - await this.googleCalendarClientProvider.getGoogleCalendarClient( - refreshToken, - ); - - const startTime = Date.now(); - - let nextSyncToken: string | null | undefined; - let nextPageToken: string | undefined; - const events: calendarV3.Schema$Event[] = []; - - let hasMoreEvents = true; - - while (hasMoreEvents) { - const googleCalendarEvents = await googleCalendarClient.events - .list({ - calendarId: 'primary', - maxResults: 500, - syncToken: emailOrDomainToReimport ? undefined : syncToken, - pageToken: nextPageToken, - q: emailOrDomainToReimport, - showDeleted: true, - }) - .catch(async (error: GaxiosError) => { - if (error.response?.status !== 410) { - throw error; - } - - await this.calendarChannelRepository.updateSyncCursor( - null, - connectedAccountId, - workspaceId, - ); - - this.logger.log( - `Sync token is no longer valid for connected account ${connectedAccountId} in workspace ${workspaceId}, resetting sync cursor.`, - ); - - return { - data: { - items: [], - nextSyncToken: undefined, - nextPageToken: undefined, - }, - }; - }); - - nextSyncToken = googleCalendarEvents.data.nextSyncToken; - nextPageToken = googleCalendarEvents.data.nextPageToken || undefined; - - const { items } = googleCalendarEvents.data; - - if (!items || items.length === 0) { - break; - } - - events.push(...items); - - if (!nextPageToken) { - hasMoreEvents = false; - } - } - - const endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${connectedAccountId} getting events list in ${ - endTime - startTime - }ms.`, - ); - - return { events, nextSyncToken }; - } - - public async saveGoogleCalendarEvents( - eventsToSave: CalendarEventWithParticipants[], - eventsToUpdate: CalendarEventWithParticipants[], - calendarChannelEventAssociationsToSave: { - calendarEventId: string; - eventExternalId: string; - calendarChannelId: string; - }[], - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, - calendarChannel: CalendarChannelWorkspaceEntity, - workspaceId: string, - ): Promise<void> { - const dataSourceMetadata = - await this.workspaceDataSourceService.connectToWorkspaceDataSource( - workspaceId, - ); - - const participantsToSave = eventsToSave.flatMap( - (event) => event.participants, - ); - - const participantsToUpdate = eventsToUpdate.flatMap( - (event) => event.participants, - ); - - let startTime: number; - let endTime: number; - - const savedCalendarEventParticipantsToEmit: ObjectRecord<CalendarEventParticipantWorkspaceEntity>[] = - []; - - try { - await dataSourceMetadata?.transaction(async (transactionManager) => { - startTime = Date.now(); - - await this.calendarEventRepository.saveCalendarEvents( - eventsToSave, - workspaceId, - transactionManager, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${ - connectedAccount.id - }: saving ${eventsToSave.length} events in ${endTime - startTime}ms.`, - ); - - startTime = Date.now(); - - await this.calendarEventRepository.updateCalendarEvents( - eventsToUpdate, - workspaceId, - transactionManager, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${ - connectedAccount.id - }: updating ${eventsToUpdate.length} events in ${ - endTime - startTime - }ms.`, - ); - - startTime = Date.now(); - - await this.calendarChannelEventAssociationRepository.saveCalendarChannelEventAssociations( - calendarChannelEventAssociationsToSave, - workspaceId, - transactionManager, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${ - connectedAccount.id - }: saving calendar channel event associations in ${ - endTime - startTime - }ms.`, - ); - - startTime = Date.now(); - - const newCalendarEventParticipants = - await this.calendarEventParticipantsRepository.updateCalendarEventParticipantsAndReturnNewOnes( - participantsToUpdate, - workspaceId, - transactionManager, - ); - - endTime = Date.now(); - - participantsToSave.push(...newCalendarEventParticipants); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${ - connectedAccount.id - }: updating participants in ${endTime - startTime}ms.`, - ); - - startTime = Date.now(); - - const savedCalendarEventParticipants = - await this.calendarEventParticipantsService.saveCalendarEventParticipants( - participantsToSave, - workspaceId, - transactionManager, - ); - - savedCalendarEventParticipantsToEmit.push( - ...savedCalendarEventParticipants, - ); - - endTime = Date.now(); - - this.logger.log( - `google calendar sync for workspace ${workspaceId} and account ${ - connectedAccount.id - }: saving participants in ${endTime - startTime}ms.`, - ); - }); - - this.eventEmitter.emit(`calendarEventParticipant.matched`, { - workspaceId, - userId: connectedAccount.accountOwnerId, - calendarEventParticipants: savedCalendarEventParticipantsToEmit, - }); - - if (calendarChannel.isContactAutoCreationEnabled) { - await this.messageQueueService.add<CreateCompanyAndContactJobData>( - CreateCompanyAndContactJob.name, - { - workspaceId, - connectedAccount, - contactsToCreate: participantsToSave, - }, - ); - } - } catch (error) { - this.logger.error( - `Error during google calendar sync for workspace ${workspaceId} and account ${connectedAccount.id}: ${error.message}`, - ); - } - } -} diff --git a/packages/twenty-server/src/modules/calendar/services/providers/calendar-providers.module.ts b/packages/twenty-server/src/modules/calendar/services/providers/calendar-providers.module.ts deleted file mode 100644 index 67f8064f7adf..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/providers/calendar-providers.module.ts +++ /dev/null @@ -1,11 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module'; -import { GoogleCalendarClientProvider } from 'src/modules/calendar/services/providers/google-calendar/google-calendar.provider'; - -@Module({ - imports: [EnvironmentModule], - providers: [GoogleCalendarClientProvider], - exports: [GoogleCalendarClientProvider], -}) -export class CalendarProvidersModule {} diff --git a/packages/twenty-server/src/modules/calendar/services/providers/google-calendar/google-calendar.provider.ts b/packages/twenty-server/src/modules/calendar/services/providers/google-calendar/google-calendar.provider.ts deleted file mode 100644 index 6e2faa04e613..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/providers/google-calendar/google-calendar.provider.ts +++ /dev/null @@ -1,44 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { OAuth2Client } from 'google-auth-library'; -import { calendar_v3 as calendarV3, google } from 'googleapis'; - -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; - -@Injectable() -export class GoogleCalendarClientProvider { - constructor(private readonly environmentService: EnvironmentService) {} - - public async getGoogleCalendarClient( - refreshToken: string, - ): Promise<calendarV3.Calendar> { - const oAuth2Client = await this.getOAuth2Client(refreshToken); - - const googleCalendarClient = google.calendar({ - version: 'v3', - auth: oAuth2Client, - }); - - return googleCalendarClient; - } - - private async getOAuth2Client(refreshToken: string): Promise<OAuth2Client> { - const googleCalendarClientId = this.environmentService.get( - 'AUTH_GOOGLE_CLIENT_ID', - ); - const googleCalendarClientSecret = this.environmentService.get( - 'AUTH_GOOGLE_CLIENT_SECRET', - ); - - const oAuth2Client = new google.auth.OAuth2( - googleCalendarClientId, - googleCalendarClientSecret, - ); - - oAuth2Client.setCredentials({ - refresh_token: refreshToken, - }); - - return oAuth2Client; - } -} diff --git a/packages/twenty-server/src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.module.ts b/packages/twenty-server/src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.module.ts deleted file mode 100644 index 877de36e89d5..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.module.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { WorkspaceGoogleCalendarSyncService } from 'src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.service'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; - -@Module({ - imports: [ - ObjectMetadataRepositoryModule.forFeature([CalendarChannelWorkspaceEntity]), - ], - providers: [WorkspaceGoogleCalendarSyncService], - exports: [WorkspaceGoogleCalendarSyncService], -}) -export class WorkspaceGoogleCalendarSyncModule {} diff --git a/packages/twenty-server/src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.service.ts b/packages/twenty-server/src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.service.ts deleted file mode 100644 index 8d68c01d88c8..000000000000 --- a/packages/twenty-server/src/modules/calendar/services/workspace-google-calendar-sync/workspace-google-calendar-sync.service.ts +++ /dev/null @@ -1,46 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; -import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; -import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { - GoogleCalendarSyncJobData, - GoogleCalendarSyncJob, -} from 'src/modules/calendar/jobs/google-calendar-sync.job'; -import { CalendarChannelRepository } from 'src/modules/calendar/repositories/calendar-channel.repository'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; - -@Injectable() -export class WorkspaceGoogleCalendarSyncService { - constructor( - @InjectObjectMetadataRepository(CalendarChannelWorkspaceEntity) - private readonly calendarChannelRepository: CalendarChannelRepository, - @InjectMessageQueue(MessageQueue.calendarQueue) - private readonly messageQueueService: MessageQueueService, - ) {} - - public async startWorkspaceGoogleCalendarSync( - workspaceId: string, - ): Promise<void> { - const calendarChannels = - await this.calendarChannelRepository.getAll(workspaceId); - - for (const calendarChannel of calendarChannels) { - if (!calendarChannel?.isSyncEnabled) { - continue; - } - - await this.messageQueueService.add<GoogleCalendarSyncJobData>( - GoogleCalendarSyncJob.name, - { - workspaceId, - connectedAccountId: calendarChannel.connectedAccountId, - }, - { - retryLimit: 2, - }, - ); - } - } -} diff --git a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-channel.workspace-entity.ts b/packages/twenty-server/src/modules/calendar/standard-objects/calendar-channel.workspace-entity.ts deleted file mode 100644 index 85aed3b43c9e..000000000000 --- a/packages/twenty-server/src/modules/calendar/standard-objects/calendar-channel.workspace-entity.ts +++ /dev/null @@ -1,134 +0,0 @@ -import { Relation } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/relation.interface'; - -import { - RelationMetadataType, - RelationOnDeleteAction, -} from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; -import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { CALENDAR_CHANNEL_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; -import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { CalendarChannelEventAssociationWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel-event-association.workspace-entity'; -import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; -import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; -import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; -import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; -import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; -import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; - -export enum CalendarChannelVisibility { - METADATA = 'METADATA', - SHARE_EVERYTHING = 'SHARE_EVERYTHING', -} - -@WorkspaceEntity({ - standardId: STANDARD_OBJECT_IDS.calendarChannel, - namePlural: 'calendarChannels', - labelSingular: 'Calendar Channel', - labelPlural: 'Calendar Channels', - description: 'Calendar Channels', - icon: 'IconCalendar', -}) -@WorkspaceIsSystem() -@WorkspaceIsNotAuditLogged() -export class CalendarChannelWorkspaceEntity extends BaseWorkspaceEntity { - @WorkspaceField({ - standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.handle, - type: FieldMetadataType.TEXT, - label: 'Handle', - description: 'Handle', - icon: 'IconAt', - }) - handle: string; - - @WorkspaceField({ - standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.visibility, - type: FieldMetadataType.SELECT, - label: 'Visibility', - description: 'Visibility', - icon: 'IconEyeglass', - options: [ - { - value: CalendarChannelVisibility.METADATA, - label: 'Metadata', - position: 0, - color: 'green', - }, - { - value: CalendarChannelVisibility.SHARE_EVERYTHING, - label: 'Share Everything', - position: 1, - color: 'orange', - }, - ], - defaultValue: `'${CalendarChannelVisibility.SHARE_EVERYTHING}'`, - }) - visibility: string; - - @WorkspaceField({ - standardId: - CALENDAR_CHANNEL_STANDARD_FIELD_IDS.isContactAutoCreationEnabled, - type: FieldMetadataType.BOOLEAN, - label: 'Is Contact Auto Creation Enabled', - description: 'Is Contact Auto Creation Enabled', - icon: 'IconUserCircle', - defaultValue: true, - }) - isContactAutoCreationEnabled: boolean; - - @WorkspaceField({ - standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.isSyncEnabled, - type: FieldMetadataType.BOOLEAN, - label: 'Is Sync Enabled', - description: 'Is Sync Enabled', - icon: 'IconRefresh', - defaultValue: true, - }) - isSyncEnabled: boolean; - - @WorkspaceField({ - standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.syncCursor, - type: FieldMetadataType.TEXT, - label: 'Sync Cursor', - description: - 'Sync Cursor. Used for syncing events from the calendar provider', - icon: 'IconReload', - }) - syncCursor: string; - - @WorkspaceField({ - standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.throttleFailureCount, - type: FieldMetadataType.NUMBER, - label: 'Throttle Failure Count', - description: 'Throttle Failure Count', - icon: 'IconX', - defaultValue: 0, - }) - throttleFailureCount: number; - - @WorkspaceRelation({ - standardId: CALENDAR_CHANNEL_STANDARD_FIELD_IDS.connectedAccount, - type: RelationMetadataType.MANY_TO_ONE, - label: 'Connected Account', - description: 'Connected Account', - icon: 'IconUserCircle', - joinColumn: 'connectedAccountId', - inverseSideTarget: () => ConnectedAccountWorkspaceEntity, - inverseSideFieldKey: 'calendarChannels', - }) - connectedAccount: Relation<ConnectedAccountWorkspaceEntity>; - - @WorkspaceRelation({ - standardId: - CALENDAR_CHANNEL_STANDARD_FIELD_IDS.calendarChannelEventAssociations, - type: RelationMetadataType.ONE_TO_MANY, - label: 'Calendar Channel Event Associations', - description: 'Calendar Channel Event Associations', - icon: 'IconCalendar', - inverseSideTarget: () => CalendarChannelEventAssociationWorkspaceEntity, - onDelete: RelationOnDeleteAction.CASCADE, - }) - calendarChannelEventAssociations: Relation< - CalendarChannelEventAssociationWorkspaceEntity[] - >; -} diff --git a/packages/twenty-server/src/modules/calendar/utils/filter-out-blocklisted-events.util.ts b/packages/twenty-server/src/modules/calendar/utils/filter-out-blocklisted-events.util.ts deleted file mode 100644 index 067e49ea5020..000000000000 --- a/packages/twenty-server/src/modules/calendar/utils/filter-out-blocklisted-events.util.ts +++ /dev/null @@ -1,20 +0,0 @@ -import { calendar_v3 as calendarV3 } from 'googleapis'; - -import { isEmailBlocklisted } from 'src/modules/calendar-messaging-participant/utils/is-email-blocklisted.util'; - -export const filterOutBlocklistedEvents = ( - calendarChannelHandle: string, - events: calendarV3.Schema$Event[], - blocklist: string[], -) => { - return events.filter((event) => { - if (!event.attendees) { - return true; - } - - return event.attendees.every( - (attendee) => - !isEmailBlocklisted(calendarChannelHandle, attendee.email, blocklist), - ); - }); -}; diff --git a/packages/twenty-server/src/modules/calendar/utils/google-calendar-search-filter.util.ts b/packages/twenty-server/src/modules/calendar/utils/google-calendar-search-filter.util.ts deleted file mode 100644 index e9dc788d9f92..000000000000 --- a/packages/twenty-server/src/modules/calendar/utils/google-calendar-search-filter.util.ts +++ /dev/null @@ -1,9 +0,0 @@ -export const googleCalendarSearchFilterExcludeEmails = ( - emails: string[], -): string | undefined => { - if (emails.length === 0) { - return undefined; - } - - return `-(${emails.join(', ')})`; -}; diff --git a/packages/twenty-server/src/modules/company/repositories/company.repository.ts b/packages/twenty-server/src/modules/company/repositories/company.repository.ts index 716a7a5bc95e..471db67709ae 100644 --- a/packages/twenty-server/src/modules/company/repositories/company.repository.ts +++ b/packages/twenty-server/src/modules/company/repositories/company.repository.ts @@ -67,7 +67,7 @@ export class CompanyRepository { ); await this.workspaceDataSourceService.executeRawQuery( - `INSERT INTO ${dataSourceSchema}.company (id, "domainName", name, address, position) + `INSERT INTO ${dataSourceSchema}.company (id, "domainName", name, "addressAddressCity", position) VALUES ($1, $2, $3, $4, $5)`, [ companyToCreate.id, diff --git a/packages/twenty-server/src/modules/company/standard-objects/company.workspace-entity.ts b/packages/twenty-server/src/modules/company/standard-objects/company.workspace-entity.ts index 867b11e023e9..8f2d68f5dcee 100644 --- a/packages/twenty-server/src/modules/company/standard-objects/company.workspace-entity.ts +++ b/packages/twenty-server/src/modules/company/standard-objects/company.workspace-entity.ts @@ -1,3 +1,5 @@ +import { Address } from 'nodemailer/lib/mailer'; + import { Relation } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/relation.interface'; import { CurrencyMetadata } from 'src/engine/metadata-modules/field-metadata/composite-types/currency.composite-type'; @@ -7,6 +9,14 @@ import { RelationMetadataType, RelationOnDeleteAction, } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; +import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; +import { WorkspaceIsDeprecated } from 'src/engine/twenty-orm/decorators/workspace-is-deprecated.decorator'; +import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; +import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { COMPANY_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; import { ActivityTargetWorkspaceEntity } from 'src/modules/activity/standard-objects/activity-target.workspace-entity'; @@ -14,14 +24,8 @@ import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objec import { FavoriteWorkspaceEntity } from 'src/modules/favorite/standard-objects/favorite.workspace-entity'; import { OpportunityWorkspaceEntity } from 'src/modules/opportunity/standard-objects/opportunity.workspace-entity'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; -import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; -import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; -import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; -import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; -import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; -import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.company, @@ -51,15 +55,6 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { }) domainName?: string; - @WorkspaceField({ - standardId: COMPANY_STANDARD_FIELD_IDS.address, - type: FieldMetadataType.TEXT, - label: 'Address', - description: 'The company address', - icon: 'IconMap', - }) - address: string; - @WorkspaceField({ standardId: COMPANY_STANDARD_FIELD_IDS.employees, type: FieldMetadataType.NUMBER, @@ -68,7 +63,7 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconUsers', }) @WorkspaceIsNullable() - employees: number; + employees: number | null; @WorkspaceField({ standardId: COMPANY_STANDARD_FIELD_IDS.linkedinLink, @@ -78,7 +73,7 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconBrandLinkedin', }) @WorkspaceIsNullable() - linkedinLink: LinkMetadata; + linkedinLink: LinkMetadata | null; @WorkspaceField({ standardId: COMPANY_STANDARD_FIELD_IDS.xLink, @@ -88,7 +83,7 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconBrandX', }) @WorkspaceIsNullable() - xLink: LinkMetadata; + xLink: LinkMetadata | null; @WorkspaceField({ standardId: COMPANY_STANDARD_FIELD_IDS.annualRecurringRevenue, @@ -99,7 +94,17 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconMoneybag', }) @WorkspaceIsNullable() - annualRecurringRevenue: CurrencyMetadata; + annualRecurringRevenue: CurrencyMetadata | null; + + @WorkspaceField({ + standardId: COMPANY_STANDARD_FIELD_IDS.address, + type: FieldMetadataType.ADDRESS, + label: 'Address', + description: 'Address of the company', + icon: 'IconMap', + }) + @WorkspaceIsNullable() + address: Address; @WorkspaceField({ standardId: COMPANY_STANDARD_FIELD_IDS.idealCustomerProfile, @@ -121,7 +126,7 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { }) @WorkspaceIsSystem() @WorkspaceIsNullable() - position: number; + position: number | null; // Relations @WorkspaceRelation({ @@ -143,13 +148,15 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { description: 'Your team member responsible for managing the company account', icon: 'IconUserCircle', - joinColumn: 'accountOwnerId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'accountOwnerForCompanies', onDelete: RelationOnDeleteAction.SET_NULL, }) @WorkspaceIsNullable() - accountOwner: Relation<WorkspaceMemberWorkspaceEntity>; + accountOwner: Relation<WorkspaceMemberWorkspaceEntity> | null; + + @WorkspaceJoinColumn('accountOwner') + accountOwnerId: string | null; @WorkspaceRelation({ standardId: COMPANY_STANDARD_FIELD_IDS.activityTargets, @@ -212,4 +219,16 @@ export class CompanyWorkspaceEntity extends BaseWorkspaceEntity { @WorkspaceIsNullable() @WorkspaceIsSystem() timelineActivities: Relation<TimelineActivityWorkspaceEntity[]>; + + @WorkspaceField({ + standardId: COMPANY_STANDARD_FIELD_IDS.address_deprecated, + type: FieldMetadataType.TEXT, + label: 'Address (deprecated) ', + description: + 'Address of the company - deprecated in favor of new address field', + icon: 'IconMap', + }) + @WorkspaceIsDeprecated() + @WorkspaceIsNullable() + addressOld: string; } diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.module.ts b/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.module.ts deleted file mode 100644 index 5e6d0fa68a73..000000000000 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.module.ts +++ /dev/null @@ -1,33 +0,0 @@ -import { Module } from '@nestjs/common'; -import { TypeOrmModule } from '@nestjs/typeorm'; - -import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service'; -import { CreateCompanyModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.module'; -import { CreateContactModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.module'; -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; -import { CalendarEventParticipantModule } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.module'; -import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { MessagingCommonModule } from 'src/modules/messaging/common/messaging-common.module'; - -@Module({ - imports: [ - CreateContactModule, - CreateCompanyModule, - ObjectMetadataRepositoryModule.forFeature([ - PersonWorkspaceEntity, - WorkspaceMemberWorkspaceEntity, - CalendarEventParticipantWorkspaceEntity, - ]), - MessagingCommonModule, - WorkspaceDataSourceModule, - CalendarEventParticipantModule, - TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), - ], - providers: [CreateCompanyAndContactService], - exports: [CreateCompanyAndContactService], -}) -export class AutoCompaniesAndContactsCreationModule {} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.module.ts b/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.module.ts deleted file mode 100644 index c93c537ec395..000000000000 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.module.ts +++ /dev/null @@ -1,14 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/company.workspace-entity'; -import { CreateCompanyService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.service'; - -@Module({ - imports: [ - ObjectMetadataRepositoryModule.forFeature([CompanyWorkspaceEntity]), - ], - providers: [CreateCompanyService], - exports: [CreateCompanyService], -}) -export class CreateCompanyModule {} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.module.ts b/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.module.ts deleted file mode 100644 index b24586e3b9ae..000000000000 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.module.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { CreateContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.service'; -import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; - -@Module({ - imports: [ObjectMetadataRepositoryModule.forFeature([PersonWorkspaceEntity])], - providers: [CreateContactService], - exports: [CreateContactService], -}) -export class CreateContactModule {} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/jobs/auto-companies-and-contacts-creation-job.module.ts b/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/jobs/auto-companies-and-contacts-creation-job.module.ts deleted file mode 100644 index f65e1965b7f0..000000000000 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/jobs/auto-companies-and-contacts-creation-job.module.ts +++ /dev/null @@ -1,15 +0,0 @@ -import { Module } from '@nestjs/common'; - -import { AutoCompaniesAndContactsCreationModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.module'; -import { CreateCompanyAndContactJob } from 'src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job'; - -@Module({ - imports: [AutoCompaniesAndContactsCreationModule], - providers: [ - { - provide: CreateCompanyAndContactJob.name, - useClass: CreateCompanyAndContactJob, - }, - ], -}) -export class AutoCompaniesAndContactsCreationJobModule {} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service.ts b/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service.ts deleted file mode 100644 index 7fd46851031f..000000000000 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service.ts +++ /dev/null @@ -1,184 +0,0 @@ -import { Injectable } from '@nestjs/common'; -import { EventEmitter2 } from '@nestjs/event-emitter'; - -import { EntityManager } from 'typeorm'; -import compact from 'lodash.compact'; - -import { getDomainNameFromHandle } from 'src/modules/calendar-messaging-participant/utils/get-domain-name-from-handle.util'; -import { CreateCompanyService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.service'; -import { CreateContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.service'; -import { PersonRepository } from 'src/modules/person/repositories/person.repository'; -import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; -import { isWorkEmail } from 'src/utils/is-work-email'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; -import { getUniqueContactsAndHandles } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util'; -import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type'; -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { CalendarEventParticipantService } from 'src/modules/calendar/services/calendar-event-participant/calendar-event-participant.service'; -import { filterOutContactsFromCompanyOrWorkspace } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/filter-out-contacts-from-company-or-workspace.util'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { MessagingMessageParticipantService } from 'src/modules/messaging/common/services/messaging-message-participant.service'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; - -@Injectable() -export class CreateCompanyAndContactService { - constructor( - private readonly createContactService: CreateContactService, - private readonly createCompaniesService: CreateCompanyService, - @InjectObjectMetadataRepository(PersonWorkspaceEntity) - private readonly personRepository: PersonRepository, - @InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity) - private readonly workspaceMemberRepository: WorkspaceMemberRepository, - private readonly workspaceDataSourceService: WorkspaceDataSourceService, - private readonly messageParticipantService: MessagingMessageParticipantService, - private readonly calendarEventParticipantService: CalendarEventParticipantService, - private readonly eventEmitter: EventEmitter2, - ) {} - - async createCompaniesAndPeople( - connectedAccountHandle: string, - contactsToCreate: Contacts, - workspaceId: string, - transactionManager?: EntityManager, - ): Promise<ObjectRecord<PersonWorkspaceEntity>[]> { - if (!contactsToCreate || contactsToCreate.length === 0) { - return []; - } - - // TODO: This is a feature that may be implemented in the future - const isContactAutoCreationForNonWorkEmailsEnabled = false; - - const workspaceMembers = - await this.workspaceMemberRepository.getAllByWorkspaceId( - workspaceId, - transactionManager, - ); - - const contactsToCreateFromOtherCompanies = - filterOutContactsFromCompanyOrWorkspace( - contactsToCreate, - connectedAccountHandle, - workspaceMembers, - ); - - const { uniqueContacts, uniqueHandles } = getUniqueContactsAndHandles( - contactsToCreateFromOtherCompanies, - ); - - if (uniqueHandles.length === 0) { - return []; - } - - const alreadyCreatedContacts = await this.personRepository.getByEmails( - uniqueHandles, - workspaceId, - transactionManager, - ); - - const alreadyCreatedContactEmails: string[] = alreadyCreatedContacts?.map( - ({ email }) => email, - ); - - const filteredContactsToCreate = uniqueContacts.filter( - (participant) => - !alreadyCreatedContactEmails.includes(participant.handle) && - participant.handle.includes('@') && - (isContactAutoCreationForNonWorkEmailsEnabled || - isWorkEmail(participant.handle)), - ); - - const filteredContactsToCreateWithCompanyDomainNames = - filteredContactsToCreate?.map((participant) => ({ - handle: participant.handle, - displayName: participant.displayName, - companyDomainName: isWorkEmail(participant.handle) - ? getDomainNameFromHandle(participant.handle) - : undefined, - })); - - const domainNamesToCreate = compact( - filteredContactsToCreateWithCompanyDomainNames.map( - (participant) => participant.companyDomainName, - ), - ); - - const companiesObject = await this.createCompaniesService.createCompanies( - domainNamesToCreate, - workspaceId, - transactionManager, - ); - - const formattedContactsToCreate = - filteredContactsToCreateWithCompanyDomainNames.map((contact) => ({ - handle: contact.handle, - displayName: contact.displayName, - companyId: - contact.companyDomainName && contact.companyDomainName !== '' - ? companiesObject[contact.companyDomainName] - : undefined, - })); - - return await this.createContactService.createPeople( - formattedContactsToCreate, - workspaceId, - transactionManager, - ); - } - - async createCompaniesAndContactsAndUpdateParticipants( - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, - contactsToCreate: Contacts, - workspaceId: string, - ) { - const { dataSource: workspaceDataSource } = - await this.workspaceDataSourceService.connectedToWorkspaceDataSourceAndReturnMetadata( - workspaceId, - ); - - let updatedMessageParticipants: ObjectRecord<MessageParticipantWorkspaceEntity>[] = - []; - let updatedCalendarEventParticipants: ObjectRecord<CalendarEventParticipantWorkspaceEntity>[] = - []; - - await workspaceDataSource?.transaction( - async (transactionManager: EntityManager) => { - const createdPeople = await this.createCompaniesAndPeople( - connectedAccount.handle, - contactsToCreate, - workspaceId, - transactionManager, - ); - - updatedMessageParticipants = - await this.messageParticipantService.updateMessageParticipantsAfterPeopleCreation( - createdPeople, - workspaceId, - transactionManager, - ); - - updatedCalendarEventParticipants = - await this.calendarEventParticipantService.updateCalendarEventParticipantsAfterPeopleCreation( - createdPeople, - workspaceId, - transactionManager, - ); - }, - ); - - this.eventEmitter.emit(`messageParticipant.matched`, { - workspaceId, - userId: connectedAccount.accountOwnerId, - messageParticipants: updatedMessageParticipants, - }); - - this.eventEmitter.emit(`calendarEventParticipant.matched`, { - workspaceId, - userId: connectedAccount.accountOwnerId, - calendarEventParticipants: updatedCalendarEventParticipants, - }); - } -} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/filter-out-contacts-from-company-or-workspace.util.ts b/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/filter-out-contacts-from-company-or-workspace.util.ts deleted file mode 100644 index eadb0e9c134b..000000000000 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/filter-out-contacts-from-company-or-workspace.util.ts +++ /dev/null @@ -1,27 +0,0 @@ -import { getDomainNameFromHandle } from 'src/modules/calendar-messaging-participant/utils/get-domain-name-from-handle.util'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type'; - -export function filterOutContactsFromCompanyOrWorkspace( - contacts: Contacts, - selfHandle: string, - workspaceMembers: ObjectRecord<WorkspaceMemberWorkspaceEntity>[], -): Contacts { - const selfDomainName = getDomainNameFromHandle(selfHandle); - - const workspaceMembersMap = workspaceMembers.reduce( - (map, workspaceMember) => { - map[workspaceMember.userEmail] = true; - - return map; - }, - new Map<string, boolean>(), - ); - - return contacts.filter( - (contact) => - getDomainNameFromHandle(contact.handle) !== selfDomainName && - !workspaceMembersMap[contact.handle], - ); -} diff --git a/packages/twenty-server/src/modules/connected-account/email-alias-manager/drivers/google/google-email-alias-manager.service.ts b/packages/twenty-server/src/modules/connected-account/email-alias-manager/drivers/google/google-email-alias-manager.service.ts new file mode 100644 index 000000000000..5de07e5e1a48 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/email-alias-manager/drivers/google/google-email-alias-manager.service.ts @@ -0,0 +1,43 @@ +import { Injectable } from '@nestjs/common'; + +import { google } from 'googleapis'; + +import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Injectable() +export class GoogleEmailAliasManagerService { + constructor( + private readonly oAuth2ClientManagerService: OAuth2ClientManagerService, + ) {} + + public async getHandleAliases( + connectedAccount: ConnectedAccountWorkspaceEntity, + ) { + const oAuth2Client = + await this.oAuth2ClientManagerService.getOAuth2Client(connectedAccount); + + const people = google.people({ + version: 'v1', + auth: oAuth2Client, + }); + + const emailsResponse = await people.people.get({ + resourceName: 'people/me', + personFields: 'emailAddresses', + }); + + const emailAddresses = emailsResponse.data.emailAddresses; + + const handleAliases = + emailAddresses + ?.filter((emailAddress) => { + return emailAddress.metadata?.primary !== true; + }) + .map((emailAddress) => { + return emailAddress.value || ''; + }) || []; + + return handleAliases; + } +} diff --git a/packages/twenty-server/src/modules/connected-account/email-alias-manager/email-alias-manager.module.ts b/packages/twenty-server/src/modules/connected-account/email-alias-manager/email-alias-manager.module.ts new file mode 100644 index 000000000000..e1678e3d7ab6 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/email-alias-manager/email-alias-manager.module.ts @@ -0,0 +1,19 @@ +import { Module } from '@nestjs/common'; + +import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; +import { GoogleEmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/drivers/google/google-email-alias-manager.service'; +import { EmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/services/email-alias-manager.service'; +import { OAuth2ClientManagerModule } from 'src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Module({ + imports: [ + ObjectMetadataRepositoryModule.forFeature([ + ConnectedAccountWorkspaceEntity, + ]), + OAuth2ClientManagerModule, + ], + providers: [EmailAliasManagerService, GoogleEmailAliasManagerService], + exports: [EmailAliasManagerService], +}) +export class EmailAliasManagerModule {} diff --git a/packages/twenty-server/src/modules/connected-account/email-alias-manager/services/email-alias-manager.service.ts b/packages/twenty-server/src/modules/connected-account/email-alias-manager/services/email-alias-manager.service.ts new file mode 100644 index 000000000000..50a855ba31a7 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/email-alias-manager/services/email-alias-manager.service.ts @@ -0,0 +1,41 @@ +import { Injectable } from '@nestjs/common'; + +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { GoogleEmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/drivers/google/google-email-alias-manager.service'; +import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Injectable() +export class EmailAliasManagerService { + constructor( + @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) + private readonly connectedAccountRepository: ConnectedAccountRepository, + private readonly googleEmailAliasManagerService: GoogleEmailAliasManagerService, + ) {} + + public async refreshHandleAliases( + connectedAccount: ConnectedAccountWorkspaceEntity, + workspaceId: string, + ) { + let handleAliases: string[]; + + switch (connectedAccount.provider) { + case 'google': + handleAliases = + await this.googleEmailAliasManagerService.getHandleAliases( + connectedAccount, + ); + break; + default: + throw new Error( + `Email alias manager for provider ${connectedAccount.provider} is not implemented`, + ); + } + + await this.connectedAccountRepository.updateHandleAliases( + handleAliases, + connectedAccount.id, + workspaceId, + ); + } +} diff --git a/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/drivers/google/google-oauth2-client-manager.service.ts b/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/drivers/google/google-oauth2-client-manager.service.ts new file mode 100644 index 000000000000..ef577eafc702 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/drivers/google/google-oauth2-client-manager.service.ts @@ -0,0 +1,29 @@ +import { Injectable } from '@nestjs/common'; + +import { OAuth2Client } from 'google-auth-library'; +import { google } from 'googleapis'; + +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; + +@Injectable() +export class GoogleOAuth2ClientManagerService { + constructor(private readonly environmentService: EnvironmentService) {} + + public async getOAuth2Client(refreshToken: string): Promise<OAuth2Client> { + const gmailClientId = this.environmentService.get('AUTH_GOOGLE_CLIENT_ID'); + const gmailClientSecret = this.environmentService.get( + 'AUTH_GOOGLE_CLIENT_SECRET', + ); + + const oAuth2Client = new google.auth.OAuth2( + gmailClientId, + gmailClientSecret, + ); + + oAuth2Client.setCredentials({ + refresh_token: refreshToken, + }); + + return oAuth2Client; + } +} diff --git a/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module.ts b/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module.ts new file mode 100644 index 000000000000..23c65eba1157 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; + +import { GoogleOAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/drivers/google/google-oauth2-client-manager.service'; +import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service'; + +@Module({ + imports: [], + providers: [OAuth2ClientManagerService, GoogleOAuth2ClientManagerService], + exports: [OAuth2ClientManagerService], +}) +export class OAuth2ClientManagerModule {} diff --git a/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service.ts b/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service.ts new file mode 100644 index 000000000000..9585ce5a8da6 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service.ts @@ -0,0 +1,33 @@ +import { Injectable } from '@nestjs/common'; + +import { OAuth2Client } from 'google-auth-library'; + +import { GoogleOAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/drivers/google/google-oauth2-client-manager.service'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Injectable() +export class OAuth2ClientManagerService { + constructor( + private readonly googleOAuth2ClientManagerService: GoogleOAuth2ClientManagerService, + ) {} + + public async getOAuth2Client( + connectedAccount: Pick< + ConnectedAccountWorkspaceEntity, + 'provider' | 'refreshToken' + >, + ): Promise<OAuth2Client> { + const { refreshToken } = connectedAccount; + + switch (connectedAccount.provider) { + case 'google': + return this.googleOAuth2ClientManagerService.getOAuth2Client( + refreshToken, + ); + default: + throw new Error( + `OAuth2 client manager for provider ${connectedAccount.provider} is not implemented`, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-update-many.pre-query.hook.ts b/packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-update-many.pre-query.hook.ts deleted file mode 100644 index 1ce12a5350df..000000000000 --- a/packages/twenty-server/src/modules/connected-account/query-hooks/blocklist/blocklist-update-many.pre-query.hook.ts +++ /dev/null @@ -1,12 +0,0 @@ -import { Injectable, MethodNotAllowedException } from '@nestjs/common'; - -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; - -@Injectable() -export class BlocklistUpdateManyPreQueryHook implements WorkspacePreQueryHook { - constructor() {} - - async execute(): Promise<void> { - throw new MethodNotAllowedException('Method not allowed.'); - } -} diff --git a/packages/twenty-server/src/modules/connected-account/query-hooks/connected-account-delete-one.pre-query.hook.ts b/packages/twenty-server/src/modules/connected-account/query-hooks/connected-account-delete-one.pre-query.hook.ts new file mode 100644 index 000000000000..9296263df119 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/query-hooks/connected-account-delete-one.pre-query.hook.ts @@ -0,0 +1,43 @@ +import { EventEmitter2 } from '@nestjs/event-emitter'; + +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; +import { DeleteOneResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; + +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; + +@WorkspaceQueryHook(`connectedAccount.deleteOne`) +export class ConnectedAccountDeleteOnePreQueryHook + implements WorkspaceQueryHookInstance +{ + constructor( + @InjectWorkspaceRepository(MessageChannelWorkspaceEntity) + private readonly messageChannelRepository: WorkspaceRepository<MessageChannelWorkspaceEntity>, + private eventEmitter: EventEmitter2, + ) {} + + async execute( + _userId: string, + workspaceId: string, + payload: DeleteOneResolverArgs, + ): Promise<void> { + const connectedAccountId = payload.id; + + const messageChannels = await this.messageChannelRepository.findBy({ + connectedAccountId, + }); + + messageChannels.forEach((messageChannel) => { + this.eventEmitter.emit('messageChannel.deleted', { + workspaceId, + recordId: messageChannel.id, + } satisfies Pick< + ObjectRecordDeleteEvent<MessageChannelWorkspaceEntity>, + 'workspaceId' | 'recordId' + >); + }); + } +} diff --git a/packages/twenty-server/src/modules/connected-account/query-hooks/connected-account-query-hook.module.ts b/packages/twenty-server/src/modules/connected-account/query-hooks/connected-account-query-hook.module.ts index c711b4b567c8..57aaa6408be0 100644 --- a/packages/twenty-server/src/modules/connected-account/query-hooks/connected-account-query-hook.module.ts +++ b/packages/twenty-server/src/modules/connected-account/query-hooks/connected-account-query-hook.module.ts @@ -1,25 +1,11 @@ import { Module } from '@nestjs/common'; -import { BlocklistCreateManyPreQueryHook } from 'src/modules/connected-account/query-hooks/blocklist/blocklist-create-many.pre-query.hook'; -import { BlocklistUpdateManyPreQueryHook } from 'src/modules/connected-account/query-hooks/blocklist/blocklist-update-many.pre-query.hook'; -import { BlocklistUpdateOnePreQueryHook } from 'src/modules/connected-account/query-hooks/blocklist/blocklist-update-one.pre-query.hook'; -import { BlocklistValidationModule } from 'src/modules/connected-account/services/blocklist/blocklist-validation.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { ConnectedAccountDeleteOnePreQueryHook } from 'src/modules/connected-account/query-hooks/connected-account-delete-one.pre-query.hook'; +import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; @Module({ - imports: [BlocklistValidationModule], - providers: [ - { - provide: BlocklistCreateManyPreQueryHook.name, - useClass: BlocklistCreateManyPreQueryHook, - }, - { - provide: BlocklistUpdateManyPreQueryHook.name, - useClass: BlocklistUpdateManyPreQueryHook, - }, - { - provide: BlocklistUpdateOnePreQueryHook.name, - useClass: BlocklistUpdateOnePreQueryHook, - }, - ], + imports: [TwentyORMModule.forFeature([MessageChannelWorkspaceEntity])], + providers: [ConnectedAccountDeleteOnePreQueryHook], }) export class ConnectedAccountQueryHookModule {} diff --git a/packages/twenty-server/src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.module.ts b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/drivers/google/google-api-refresh-access-token.module.ts similarity index 87% rename from packages/twenty-server/src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.module.ts rename to packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/drivers/google/google-api-refresh-access-token.module.ts index 0c3180c09a77..14529ba0b745 100644 --- a/packages/twenty-server/src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.module.ts +++ b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/drivers/google/google-api-refresh-access-token.module.ts @@ -1,7 +1,7 @@ import { Module } from '@nestjs/common'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service'; +import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/refresh-access-token-manager/drivers/google/services/google-api-refresh-access-token.service'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { MessagingCommonModule } from 'src/modules/messaging/common/messaging-common.module'; diff --git a/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/drivers/google/services/google-api-refresh-access-token.service.ts b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/drivers/google/services/google-api-refresh-access-token.service.ts new file mode 100644 index 000000000000..3a5d015be990 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/drivers/google/services/google-api-refresh-access-token.service.ts @@ -0,0 +1,29 @@ +import { Injectable } from '@nestjs/common'; + +import axios from 'axios'; + +import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; + +@Injectable() +export class GoogleAPIRefreshAccessTokenService { + constructor(private readonly environmentService: EnvironmentService) {} + + async refreshAccessToken(refreshToken: string): Promise<string> { + const response = await axios.post( + 'https://oauth2.googleapis.com/token', + { + client_id: this.environmentService.get('AUTH_GOOGLE_CLIENT_ID'), + client_secret: this.environmentService.get('AUTH_GOOGLE_CLIENT_SECRET'), + refresh_token: refreshToken, + grant_type: 'refresh_token', + }, + { + headers: { + 'Content-Type': 'application/json', + }, + }, + ); + + return response.data.access_token; + } +} diff --git a/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/refresh-access-token-manager.module.ts b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/refresh-access-token-manager.module.ts new file mode 100644 index 000000000000..e8eb170b5d3a --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/refresh-access-token-manager.module.ts @@ -0,0 +1,11 @@ +import { Module } from '@nestjs/common'; + +import { GoogleAPIRefreshAccessTokenModule } from 'src/modules/connected-account/refresh-access-token-manager/drivers/google/google-api-refresh-access-token.module'; +import { RefreshAccessTokenService } from 'src/modules/connected-account/refresh-access-token-manager/services/refresh-access-token.service'; + +@Module({ + imports: [GoogleAPIRefreshAccessTokenModule], + providers: [RefreshAccessTokenService], + exports: [RefreshAccessTokenService], +}) +export class RefreshAccessTokenManagerModule {} diff --git a/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/services/refresh-access-token.service.ts b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/services/refresh-access-token.service.ts new file mode 100644 index 000000000000..7f381750fc51 --- /dev/null +++ b/packages/twenty-server/src/modules/connected-account/refresh-access-token-manager/services/refresh-access-token.service.ts @@ -0,0 +1,62 @@ +import { Injectable } from '@nestjs/common'; + +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/refresh-access-token-manager/drivers/google/services/google-api-refresh-access-token.service'; +import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; + +@Injectable() +export class RefreshAccessTokenService { + constructor( + private readonly googleAPIRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService, + @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) + private readonly connectedAccountRepository: ConnectedAccountRepository, + ) {} + + async refreshAndSaveAccessToken( + connectedAccount: ConnectedAccountWorkspaceEntity, + workspaceId: string, + ): Promise<string> { + const refreshToken = connectedAccount.refreshToken; + + if (!refreshToken) { + throw new Error( + `No refresh token found for connected account ${connectedAccount.id} in workspace ${workspaceId}`, + ); + } + const accessToken = await this.refreshAccessToken( + connectedAccount, + refreshToken, + ); + + await this.connectedAccountRepository.updateAccessToken( + accessToken, + connectedAccount.id, + workspaceId, + ); + + await this.connectedAccountRepository.updateAccessToken( + accessToken, + connectedAccount.id, + workspaceId, + ); + + return accessToken; + } + + async refreshAccessToken( + connectedAccount: ConnectedAccountWorkspaceEntity, + refreshToken: string, + ): Promise<string> { + switch (connectedAccount.provider) { + case 'google': + return this.googleAPIRefreshAccessTokenService.refreshAccessToken( + refreshToken, + ); + default: + throw new Error( + `Provider ${connectedAccount.provider} is not supported.`, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/connected-account/repositories/connected-account.repository.ts b/packages/twenty-server/src/modules/connected-account/repositories/connected-account.repository.ts index 1658befee36b..ae68fd68ab0c 100644 --- a/packages/twenty-server/src/modules/connected-account/repositories/connected-account.repository.ts +++ b/packages/twenty-server/src/modules/connected-account/repositories/connected-account.repository.ts @@ -4,7 +4,6 @@ import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; @Injectable() export class ConnectedAccountRepository { @@ -15,7 +14,7 @@ export class ConnectedAccountRepository { public async getAll( workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>[]> { + ): Promise<ConnectedAccountWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -31,7 +30,7 @@ export class ConnectedAccountRepository { connectedAccountIds: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>[]> { + ): Promise<ConnectedAccountWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -47,7 +46,7 @@ export class ConnectedAccountRepository { workspaceMemberId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>[] | undefined> { + ): Promise<ConnectedAccountWorkspaceEntity[] | undefined> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -66,7 +65,7 @@ export class ConnectedAccountRepository { userId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>[] | undefined> { + ): Promise<ConnectedAccountWorkspaceEntity[] | undefined> { const schemaExists = await this.workspaceDataSourceService.checkSchemaExists(workspaceId); @@ -102,7 +101,7 @@ export class ConnectedAccountRepository { workspaceMemberId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>[] | undefined> { + ): Promise<ConnectedAccountWorkspaceEntity[] | undefined> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -119,7 +118,7 @@ export class ConnectedAccountRepository { public async create( connectedAccount: Pick< - ObjectRecord<ConnectedAccountWorkspaceEntity>, + ConnectedAccountWorkspaceEntity, | 'id' | 'handle' | 'provider' @@ -129,7 +128,7 @@ export class ConnectedAccountRepository { >, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>> { + ): Promise<ConnectedAccountWorkspaceEntity> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -170,7 +169,7 @@ export class ConnectedAccountRepository { connectedAccountId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity> | undefined> { + ): Promise<ConnectedAccountWorkspaceEntity | undefined> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -189,7 +188,7 @@ export class ConnectedAccountRepository { connectedAccountId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>> { + ): Promise<ConnectedAccountWorkspaceEntity> { const connectedAccount = await this.getById( connectedAccountId, workspaceId, @@ -293,7 +292,7 @@ export class ConnectedAccountRepository { public async getConnectedAccountOrThrow( workspaceId: string, connectedAccountId: string, - ): Promise<ObjectRecord<ConnectedAccountWorkspaceEntity>> { + ): Promise<ConnectedAccountWorkspaceEntity> { const connectedAccount = await this.getById( connectedAccountId, workspaceId, @@ -307,4 +306,22 @@ export class ConnectedAccountRepository { return connectedAccount; } + + public async updateHandleAliases( + handleAliases: string[], + connectedAccountId: string, + workspaceId: string, + transactionManager?: EntityManager, + ) { + const dataSourceSchema = + this.workspaceDataSourceService.getSchemaName(workspaceId); + + await this.workspaceDataSourceService.executeRawQuery( + `UPDATE ${dataSourceSchema}."connectedAccount" SET "handleAliases" = $1 WHERE "id" = $2`, + // TODO: modify handleAliases to be of fieldmetadatatype array + [handleAliases.join(','), connectedAccountId], + workspaceId, + transactionManager, + ); + } } diff --git a/packages/twenty-server/src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service.ts b/packages/twenty-server/src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service.ts deleted file mode 100644 index dc687e1a4c19..000000000000 --- a/packages/twenty-server/src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service.ts +++ /dev/null @@ -1,108 +0,0 @@ -import { Injectable } from '@nestjs/common'; - -import axios from 'axios'; - -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; -import { MessagingTelemetryService } from 'src/modules/messaging/common/services/messaging-telemetry.service'; -import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; -import { MessagingChannelSyncStatusService } from 'src/modules/messaging/common/services/messaging-channel-sync-status.service'; - -@Injectable() -export class GoogleAPIRefreshAccessTokenService { - constructor( - private readonly environmentService: EnvironmentService, - @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) - private readonly connectedAccountRepository: ConnectedAccountRepository, - @InjectObjectMetadataRepository(MessageChannelWorkspaceEntity) - private readonly messageChannelRepository: MessageChannelRepository, - private readonly messagingTelemetryService: MessagingTelemetryService, - private readonly messagingChannelSyncStatusService: MessagingChannelSyncStatusService, - ) {} - - async refreshAndSaveAccessToken( - workspaceId: string, - connectedAccountId: string, - ): Promise<void> { - const connectedAccount = await this.connectedAccountRepository.getById( - connectedAccountId, - workspaceId, - ); - - if (!connectedAccount) { - throw new Error( - `No connected account found for ${connectedAccountId} in workspace ${workspaceId}`, - ); - } - - const refreshToken = connectedAccount.refreshToken; - - if (!refreshToken) { - throw new Error( - `No refresh token found for connected account ${connectedAccountId} in workspace ${workspaceId}`, - ); - } - - try { - const accessToken = await this.refreshAccessToken(refreshToken); - - await this.connectedAccountRepository.updateAccessToken( - accessToken, - connectedAccountId, - workspaceId, - ); - } catch (error) { - const messageChannel = - await this.messageChannelRepository.getFirstByConnectedAccountId( - connectedAccountId, - workspaceId, - ); - - if (!messageChannel) { - throw new Error( - `No message channel found for connected account ${connectedAccountId} in workspace ${workspaceId}`, - ); - } - - await this.messagingTelemetryService.track({ - eventName: `refresh_token.error.insufficient_permissions`, - workspaceId, - connectedAccountId: messageChannel.connectedAccountId, - messageChannelId: messageChannel.id, - message: `${error.code}: ${error.reason}`, - }); - - await this.messagingChannelSyncStatusService.markAsFailedInsufficientPermissionsAndFlushMessagesToImport( - messageChannel.id, - workspaceId, - ); - - await this.connectedAccountRepository.updateAuthFailedAt( - messageChannel.connectedAccountId, - workspaceId, - ); - } - } - - async refreshAccessToken(refreshToken: string): Promise<string> { - const response = await axios.post( - 'https://oauth2.googleapis.com/token', - { - client_id: this.environmentService.get('AUTH_GOOGLE_CLIENT_ID'), - client_secret: this.environmentService.get('AUTH_GOOGLE_CLIENT_SECRET'), - refresh_token: refreshToken, - grant_type: 'refresh_token', - }, - { - headers: { - 'Content-Type': 'application/json', - }, - }, - ); - - return response.data.access_token; - } -} diff --git a/packages/twenty-server/src/modules/connected-account/standard-objects/connected-account.workspace-entity.ts b/packages/twenty-server/src/modules/connected-account/standard-objects/connected-account.workspace-entity.ts index 481f319703e3..6fb6ad15c407 100644 --- a/packages/twenty-server/src/modules/connected-account/standard-objects/connected-account.workspace-entity.ts +++ b/packages/twenty-server/src/modules/connected-account/standard-objects/connected-account.workspace-entity.ts @@ -5,18 +5,19 @@ import { RelationMetadataType, RelationOnDeleteAction, } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; -import { CONNECTED_ACCOUNT_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; -import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-channel.workspace-entity'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; -import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; -import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; -import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; +import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; +import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; +import { CONNECTED_ACCOUNT_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; +import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; export enum ConnectedAccountProvider { GOOGLE = 'google', @@ -86,7 +87,16 @@ export class ConnectedAccountWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconX', }) @WorkspaceIsNullable() - authFailedAt: Date; + authFailedAt: Date | null; + + @WorkspaceField({ + standardId: CONNECTED_ACCOUNT_STANDARD_FIELD_IDS.handleAliases, + type: FieldMetadataType.TEXT, + label: 'Handle Aliases', + description: 'Handle Aliases', + icon: 'IconMail', + }) + handleAliases: string; @WorkspaceRelation({ standardId: CONNECTED_ACCOUNT_STANDARD_FIELD_IDS.accountOwner, @@ -94,12 +104,14 @@ export class ConnectedAccountWorkspaceEntity extends BaseWorkspaceEntity { label: 'Account Owner', description: 'Account Owner', icon: 'IconUserCircle', - joinColumn: 'accountOwnerId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'connectedAccounts', }) accountOwner: Relation<WorkspaceMemberWorkspaceEntity>; + @WorkspaceJoinColumn('accountOwner') + accountOwnerId: string; + @WorkspaceRelation({ standardId: CONNECTED_ACCOUNT_STANDARD_FIELD_IDS.messageChannels, type: RelationMetadataType.ONE_TO_MANY, diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/utils/is-throttled.ts b/packages/twenty-server/src/modules/connected-account/utils/is-throttled.ts similarity index 100% rename from packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/utils/is-throttled.ts rename to packages/twenty-server/src/modules/connected-account/utils/is-throttled.ts diff --git a/packages/twenty-server/src/modules/contact-creation-manager/constants/contacts-creation-batch-size.constant.ts b/packages/twenty-server/src/modules/contact-creation-manager/constants/contacts-creation-batch-size.constant.ts new file mode 100644 index 000000000000..3e96e531719f --- /dev/null +++ b/packages/twenty-server/src/modules/contact-creation-manager/constants/contacts-creation-batch-size.constant.ts @@ -0,0 +1 @@ +export const CONTACTS_CREATION_BATCH_SIZE = 100; diff --git a/packages/twenty-server/src/modules/contact-creation-manager/contact-creation-manager.module.ts b/packages/twenty-server/src/modules/contact-creation-manager/contact-creation-manager.module.ts new file mode 100644 index 000000000000..fdb994efa77d --- /dev/null +++ b/packages/twenty-server/src/modules/contact-creation-manager/contact-creation-manager.module.ts @@ -0,0 +1,37 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/company.workspace-entity'; +import { AutoCompaniesAndContactsCreationCalendarChannelListener } from 'src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-calendar-channel.listener'; +import { AutoCompaniesAndContactsCreationMessageChannelListener } from 'src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-message-channel.listener'; +import { CreateCompanyAndContactService } from 'src/modules/contact-creation-manager/services/create-company-and-contact.service'; +import { CreateCompanyService } from 'src/modules/contact-creation-manager/services/create-company.service'; +import { CreateContactService } from 'src/modules/contact-creation-manager/services/create-contact.service'; +import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; + +@Module({ + imports: [ + ObjectMetadataRepositoryModule.forFeature([ + PersonWorkspaceEntity, + WorkspaceMemberWorkspaceEntity, + CompanyWorkspaceEntity, + ]), + WorkspaceDataSourceModule, + TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), + TypeOrmModule.forFeature([ObjectMetadataEntity], 'metadata'), + ], + providers: [ + CreateCompanyService, + CreateContactService, + CreateCompanyAndContactService, + AutoCompaniesAndContactsCreationMessageChannelListener, + AutoCompaniesAndContactsCreationCalendarChannelListener, + ], + exports: [CreateCompanyAndContactService], +}) +export class ContactCreationManagerModule {} diff --git a/packages/twenty-server/src/modules/contact-creation-manager/jobs/auto-companies-and-contacts-creation-job.module.ts b/packages/twenty-server/src/modules/contact-creation-manager/jobs/auto-companies-and-contacts-creation-job.module.ts new file mode 100644 index 000000000000..a67a9e498b81 --- /dev/null +++ b/packages/twenty-server/src/modules/contact-creation-manager/jobs/auto-companies-and-contacts-creation-job.module.ts @@ -0,0 +1,10 @@ +import { Module } from '@nestjs/common'; + +import { ContactCreationManagerModule } from 'src/modules/contact-creation-manager/contact-creation-manager.module'; +import { CreateCompanyAndContactJob } from 'src/modules/contact-creation-manager/jobs/create-company-and-contact.job'; + +@Module({ + imports: [ContactCreationManagerModule], + providers: [CreateCompanyAndContactJob], +}) +export class AutoCompaniesAndContactsCreationJobModule {} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job.ts b/packages/twenty-server/src/modules/contact-creation-manager/jobs/create-company-and-contact.job.ts similarity index 57% rename from packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job.ts rename to packages/twenty-server/src/modules/contact-creation-manager/jobs/create-company-and-contact.job.ts index 7c31e16e81ab..822d14299795 100644 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/jobs/create-company-and-contact.job.ts @@ -1,28 +1,25 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { CreateCompanyAndContactService } from 'src/modules/contact-creation-manager/services/create-company-and-contact.service'; export type CreateCompanyAndContactJobData = { workspaceId: string; - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>; + connectedAccount: ConnectedAccountWorkspaceEntity; contactsToCreate: { displayName: string; handle: string; }[]; }; -@Injectable() -export class CreateCompanyAndContactJob - implements MessageQueueJob<CreateCompanyAndContactJobData> -{ +@Processor(MessageQueue.contactCreationQueue) +export class CreateCompanyAndContactJob { constructor( private readonly createCompanyAndContactService: CreateCompanyAndContactService, ) {} + @Process(CreateCompanyAndContactJob.name) async handle(data: CreateCompanyAndContactJobData): Promise<void> { const { workspaceId, connectedAccount, contactsToCreate } = data; diff --git a/packages/twenty-server/src/modules/calendar/listeners/calendar-channel.listener.ts b/packages/twenty-server/src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-calendar-channel.listener.ts similarity index 78% rename from packages/twenty-server/src/modules/calendar/listeners/calendar-channel.listener.ts rename to packages/twenty-server/src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-calendar-channel.listener.ts index 3d1e28a557ab..51104de273c4 100644 --- a/packages/twenty-server/src/modules/calendar/listeners/calendar-channel.listener.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-calendar-channel.listener.ts @@ -1,20 +1,21 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; import { objectRecordChangedProperties } from 'src/engine/integrations/event-emitter/utils/object-record-changed-properties.util'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { CalendarCreateCompanyAndContactAfterSyncJobData, CalendarCreateCompanyAndContactAfterSyncJob, -} from 'src/modules/calendar/jobs/calendar-create-company-and-contact-after-sync.job'; +} from 'src/modules/calendar/calendar-event-participant-manager/jobs/calendar-create-company-and-contact-after-sync.job'; import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; @Injectable() -export class CalendarChannelListener { +export class AutoCompaniesAndContactsCreationCalendarChannelListener { constructor( - @Inject(MessageQueue.calendarQueue) + @InjectMessageQueue(MessageQueue.calendarQueue) private readonly messageQueueService: MessageQueueService, ) {} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/listeners/messaging-message-channel.listener.ts b/packages/twenty-server/src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-message-channel.listener.ts similarity index 78% rename from packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/listeners/messaging-message-channel.listener.ts rename to packages/twenty-server/src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-message-channel.listener.ts index 6b3cde3987be..f1d6362d45a9 100644 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/listeners/messaging-message-channel.listener.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/listeners/auto-companies-and-contacts-creation-message-channel.listener.ts @@ -1,20 +1,21 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; import { objectRecordChangedProperties } from 'src/engine/integrations/event-emitter/utils/object-record-changed-properties.util'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { MessagingCreateCompanyAndContactAfterSyncJobData, MessagingCreateCompanyAndContactAfterSyncJob, -} from 'src/modules/messaging/message-participants-manager/jobs/messaging-create-company-and-contact-after-sync.job'; +} from 'src/modules/messaging/message-participant-manager/jobs/messaging-create-company-and-contact-after-sync.job'; @Injectable() -export class MessagingMessageChannelListener { +export class AutoCompaniesAndContactsCreationMessageChannelListener { constructor( - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.contactCreationQueue) private readonly messageQueueService: MessageQueueService, ) {} diff --git a/packages/twenty-server/src/modules/contact-creation-manager/services/create-company-and-contact.service.ts b/packages/twenty-server/src/modules/contact-creation-manager/services/create-company-and-contact.service.ts new file mode 100644 index 000000000000..14a984cb27d8 --- /dev/null +++ b/packages/twenty-server/src/modules/contact-creation-manager/services/create-company-and-contact.service.ts @@ -0,0 +1,169 @@ +import { Injectable } from '@nestjs/common'; +import { EventEmitter2 } from '@nestjs/event-emitter'; +import { InjectRepository } from '@nestjs/typeorm'; + +import chunk from 'lodash.chunk'; +import compact from 'lodash.compact'; +import { EntityManager, Repository } from 'typeorm'; + +import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { CONTACTS_CREATION_BATCH_SIZE } from 'src/modules/contact-creation-manager/constants/contacts-creation-batch-size.constant'; +import { CreateCompanyService } from 'src/modules/contact-creation-manager/services/create-company.service'; +import { CreateContactService } from 'src/modules/contact-creation-manager/services/create-contact.service'; +import { Contact } from 'src/modules/contact-creation-manager/types/contact.type'; +import { filterOutSelfAndContactsFromCompanyOrWorkspace } from 'src/modules/contact-creation-manager/utils/filter-out-contacts-from-company-or-workspace.util'; +import { getDomainNameFromHandle } from 'src/modules/contact-creation-manager/utils/get-domain-name-from-handle.util'; +import { getUniqueContactsAndHandles } from 'src/modules/contact-creation-manager/utils/get-unique-contacts-and-handles.util'; +import { PersonRepository } from 'src/modules/person/repositories/person.repository'; +import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; +import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; +import { isWorkEmail } from 'src/utils/is-work-email'; + +@Injectable() +export class CreateCompanyAndContactService { + constructor( + private readonly createContactService: CreateContactService, + private readonly createCompaniesService: CreateCompanyService, + @InjectObjectMetadataRepository(PersonWorkspaceEntity) + private readonly personRepository: PersonRepository, + @InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity) + private readonly workspaceMemberRepository: WorkspaceMemberRepository, + private readonly eventEmitter: EventEmitter2, + @InjectRepository(ObjectMetadataEntity, 'metadata') + private readonly objectMetadataRepository: Repository<ObjectMetadataEntity>, + ) {} + + private async createCompaniesAndPeople( + connectedAccount: ConnectedAccountWorkspaceEntity, + contactsToCreate: Contact[], + workspaceId: string, + transactionManager?: EntityManager, + ): Promise<PersonWorkspaceEntity[]> { + if (!contactsToCreate || contactsToCreate.length === 0) { + return []; + } + + const workspaceMembers = + await this.workspaceMemberRepository.getAllByWorkspaceId( + workspaceId, + transactionManager, + ); + + const contactsToCreateFromOtherCompanies = + filterOutSelfAndContactsFromCompanyOrWorkspace( + contactsToCreate, + connectedAccount, + workspaceMembers, + ); + + const { uniqueContacts, uniqueHandles } = getUniqueContactsAndHandles( + contactsToCreateFromOtherCompanies, + ); + + if (uniqueHandles.length === 0) { + return []; + } + + const alreadyCreatedContacts = await this.personRepository.getByEmails( + uniqueHandles, + workspaceId, + transactionManager, + ); + + const alreadyCreatedContactEmails: string[] = alreadyCreatedContacts?.map( + ({ email }) => email, + ); + + const filteredContactsToCreate = uniqueContacts.filter( + (participant) => + !alreadyCreatedContactEmails.includes(participant.handle) && + participant.handle.includes('@'), + ); + + const filteredContactsToCreateWithCompanyDomainNames = + filteredContactsToCreate?.map((participant) => ({ + handle: participant.handle, + displayName: participant.displayName, + companyDomainName: isWorkEmail(participant.handle) + ? getDomainNameFromHandle(participant.handle) + : undefined, + })); + + const domainNamesToCreate = compact( + filteredContactsToCreateWithCompanyDomainNames.map( + (participant) => participant.companyDomainName, + ), + ); + + const companiesObject = await this.createCompaniesService.createCompanies( + domainNamesToCreate, + workspaceId, + transactionManager, + ); + + const formattedContactsToCreate = + filteredContactsToCreateWithCompanyDomainNames.map((contact) => ({ + handle: contact.handle, + displayName: contact.displayName, + companyId: + contact.companyDomainName && contact.companyDomainName !== '' + ? companiesObject[contact.companyDomainName] + : undefined, + })); + + return await this.createContactService.createPeople( + formattedContactsToCreate, + workspaceId, + transactionManager, + ); + } + + async createCompaniesAndContactsAndUpdateParticipants( + connectedAccount: ConnectedAccountWorkspaceEntity, + contactsToCreate: Contact[], + workspaceId: string, + ) { + const contactsBatches = chunk( + contactsToCreate, + CONTACTS_CREATION_BATCH_SIZE, + ); + + // TODO: Remove this when events are emitted directly inside TwentyORM + + const objectMetadata = await this.objectMetadataRepository.findOne({ + where: { + standardId: STANDARD_OBJECT_IDS.person, + workspaceId, + }, + }); + + if (!objectMetadata) { + throw new Error('Object metadata not found'); + } + + for (const contactsBatch of contactsBatches) { + const createdPeople = await this.createCompaniesAndPeople( + connectedAccount, + contactsBatch, + workspaceId, + ); + + for (const createdPerson of createdPeople) { + this.eventEmitter.emit('person.created', { + name: 'person.created', + workspaceId, + recordId: createdPerson.id, + objectMetadata, + properties: { + after: createdPerson, + }, + } satisfies ObjectRecordCreateEvent<any>); + } + } + } +} diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.service.ts b/packages/twenty-server/src/modules/contact-creation-manager/services/create-company.service.ts similarity index 92% rename from packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.service.ts rename to packages/twenty-server/src/modules/contact-creation-manager/services/create-company.service.ts index 2721623e948b..266bdad128bd 100644 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-company/create-company.service.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/services/create-company.service.ts @@ -1,13 +1,13 @@ import { Injectable } from '@nestjs/common'; +import axios, { AxiosInstance } from 'axios'; import { EntityManager } from 'typeorm'; import { v4 } from 'uuid'; -import axios, { AxiosInstance } from 'axios'; -import { CompanyRepository } from 'src/modules/company/repositories/company.repository'; -import { getCompanyNameFromDomainName } from 'src/modules/calendar-messaging-participant/utils/get-company-name-from-domain-name.util'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { CompanyRepository } from 'src/modules/company/repositories/company.repository'; import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/company.workspace-entity'; +import { getCompanyNameFromDomainName } from 'src/modules/contact-creation-manager/utils/get-company-name-from-domain-name.util'; @Injectable() export class CreateCompanyService { private readonly httpService: AxiosInstance; @@ -76,7 +76,7 @@ export class CreateCompanyService { return companiesObject; } - async createCompany( + private async createCompany( domainName: string, workspaceId: string, transactionManager?: EntityManager, @@ -99,7 +99,7 @@ export class CreateCompanyService { return companyId; } - async getCompanyInfoFromDomainName(domainName: string): Promise<{ + private async getCompanyInfoFromDomainName(domainName: string): Promise<{ name: string; city: string; }> { diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.service.ts b/packages/twenty-server/src/modules/contact-creation-manager/services/create-contact.service.ts similarity index 85% rename from packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.service.ts rename to packages/twenty-server/src/modules/contact-creation-manager/services/create-contact.service.ts index a8a121526b83..775525c5af77 100644 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/create-contact/create-contact.service.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/services/create-contact.service.ts @@ -3,11 +3,10 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { v4 } from 'uuid'; -import { PersonRepository } from 'src/modules/person/repositories/person.repository'; -import { getFirstNameAndLastNameFromHandleAndDisplayName } from 'src/modules/calendar-messaging-participant/utils/get-first-name-and-last-name-from-handle-and-display-name.util'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { getFirstNameAndLastNameFromHandleAndDisplayName } from 'src/modules/contact-creation-manager/utils/get-first-name-and-last-name-from-handle-and-display-name.util'; +import { PersonRepository } from 'src/modules/person/repositories/person.repository'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; type ContactToCreate = { handle: string; @@ -30,7 +29,7 @@ export class CreateContactService { private readonly personRepository: PersonRepository, ) {} - public formatContacts( + private formatContacts( contactsToCreate: ContactToCreate[], ): FormattedContactToCreate[] { return contactsToCreate.map((contact) => { @@ -55,7 +54,7 @@ export class CreateContactService { contactsToCreate: ContactToCreate[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<PersonWorkspaceEntity>[]> { + ): Promise<PersonWorkspaceEntity[]> { if (contactsToCreate.length === 0) return []; const formattedContacts = this.formatContacts(contactsToCreate); diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type.ts b/packages/twenty-server/src/modules/contact-creation-manager/types/contact.type.ts similarity index 66% rename from packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type.ts rename to packages/twenty-server/src/modules/contact-creation-manager/types/contact.type.ts index 0cd41a5b768c..40424311387a 100644 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/types/contact.type.ts @@ -2,5 +2,3 @@ export type Contact = { handle: string; displayName: string; }; - -export type Contacts = Contact[]; diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/__tests__/get-unique-contacts-and-handles.spec.ts b/packages/twenty-server/src/modules/contact-creation-manager/utils/__tests__/get-unique-contacts-and-handles.spec.ts similarity index 74% rename from packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/__tests__/get-unique-contacts-and-handles.spec.ts rename to packages/twenty-server/src/modules/contact-creation-manager/utils/__tests__/get-unique-contacts-and-handles.spec.ts index 87537a8e88b4..7bafaa4aaddc 100644 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/__tests__/get-unique-contacts-and-handles.spec.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/utils/__tests__/get-unique-contacts-and-handles.spec.ts @@ -1,9 +1,9 @@ -import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type'; -import { getUniqueContactsAndHandles } from 'src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util'; +import { Contact } from 'src/modules/contact-creation-manager/types/contact.type'; +import { getUniqueContactsAndHandles } from 'src/modules/contact-creation-manager/utils/get-unique-contacts-and-handles.util'; describe('getUniqueContactsAndHandles', () => { it('should return empty arrays when contacts is empty', () => { - const contacts: Contacts = []; + const contacts: Contact[] = []; const result = getUniqueContactsAndHandles(contacts); expect(result.uniqueContacts).toEqual([]); @@ -11,7 +11,7 @@ describe('getUniqueContactsAndHandles', () => { }); it('should return unique contacts and handles', () => { - const contacts: Contacts = [ + const contacts: Contact[] = [ { handle: 'john@twenty.com', displayName: 'John Doe' }, { handle: 'john@twenty.com', displayName: 'John Doe' }, { handle: 'jane@twenty.com', displayName: 'Jane Smith' }, diff --git a/packages/twenty-server/src/modules/contact-creation-manager/utils/filter-out-contacts-from-company-or-workspace.util.ts b/packages/twenty-server/src/modules/contact-creation-manager/utils/filter-out-contacts-from-company-or-workspace.util.ts new file mode 100644 index 000000000000..42ecfd59d094 --- /dev/null +++ b/packages/twenty-server/src/modules/contact-creation-manager/utils/filter-out-contacts-from-company-or-workspace.util.ts @@ -0,0 +1,30 @@ +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { Contact } from 'src/modules/contact-creation-manager/types/contact.type'; +import { getDomainNameFromHandle } from 'src/modules/contact-creation-manager/utils/get-domain-name-from-handle.util'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; + +export function filterOutSelfAndContactsFromCompanyOrWorkspace( + contacts: Contact[], + connectedAccount: ConnectedAccountWorkspaceEntity, + workspaceMembers: WorkspaceMemberWorkspaceEntity[], +): Contact[] { + const selfDomainName = getDomainNameFromHandle(connectedAccount.handle); + + const handleAliases = connectedAccount.handleAliases?.split(',') || []; + + const workspaceMembersMap = workspaceMembers.reduce( + (map, workspaceMember) => { + map[workspaceMember.userEmail] = true; + + return map; + }, + new Map<string, boolean>(), + ); + + return contacts.filter( + (contact) => + getDomainNameFromHandle(contact.handle) !== selfDomainName && + !workspaceMembersMap[contact.handle] && + !handleAliases.includes(contact.handle), + ); +} diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/utils/get-company-name-from-domain-name.util.ts b/packages/twenty-server/src/modules/contact-creation-manager/utils/get-company-name-from-domain-name.util.ts similarity index 100% rename from packages/twenty-server/src/modules/calendar-messaging-participant/utils/get-company-name-from-domain-name.util.ts rename to packages/twenty-server/src/modules/contact-creation-manager/utils/get-company-name-from-domain-name.util.ts diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/utils/get-domain-name-from-handle.util.ts b/packages/twenty-server/src/modules/contact-creation-manager/utils/get-domain-name-from-handle.util.ts similarity index 100% rename from packages/twenty-server/src/modules/calendar-messaging-participant/utils/get-domain-name-from-handle.util.ts rename to packages/twenty-server/src/modules/contact-creation-manager/utils/get-domain-name-from-handle.util.ts diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/utils/get-first-name-and-last-name-from-handle-and-display-name.util.ts b/packages/twenty-server/src/modules/contact-creation-manager/utils/get-first-name-and-last-name-from-handle-and-display-name.util.ts similarity index 100% rename from packages/twenty-server/src/modules/calendar-messaging-participant/utils/get-first-name-and-last-name-from-handle-and-display-name.util.ts rename to packages/twenty-server/src/modules/contact-creation-manager/utils/get-first-name-and-last-name-from-handle-and-display-name.util.ts diff --git a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util.ts b/packages/twenty-server/src/modules/contact-creation-manager/utils/get-unique-contacts-and-handles.util.ts similarity index 64% rename from packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util.ts rename to packages/twenty-server/src/modules/contact-creation-manager/utils/get-unique-contacts-and-handles.util.ts index a6296e3d982a..26cd2e892628 100644 --- a/packages/twenty-server/src/modules/connected-account/auto-companies-and-contacts-creation/utils/get-unique-contacts-and-handles.util.ts +++ b/packages/twenty-server/src/modules/contact-creation-manager/utils/get-unique-contacts-and-handles.util.ts @@ -1,10 +1,10 @@ import uniq from 'lodash.uniq'; import uniqBy from 'lodash.uniqby'; -import { Contacts } from 'src/modules/connected-account/auto-companies-and-contacts-creation/types/contact.type'; +import { Contact } from 'src/modules/contact-creation-manager/types/contact.type'; -export function getUniqueContactsAndHandles(contacts: Contacts): { - uniqueContacts: Contacts; +export function getUniqueContactsAndHandles(contacts: Contact[]): { + uniqueContacts: Contact[]; uniqueHandles: string[]; } { if (contacts.length === 0) { diff --git a/packages/twenty-server/src/modules/favorite/standard-objects/favorite.workspace-entity.ts b/packages/twenty-server/src/modules/favorite/standard-objects/favorite.workspace-entity.ts index 6c1f81b6b3b3..705c62d32f71 100644 --- a/packages/twenty-server/src/modules/favorite/standard-objects/favorite.workspace-entity.ts +++ b/packages/twenty-server/src/modules/favorite/standard-objects/favorite.workspace-entity.ts @@ -17,6 +17,7 @@ import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-re import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; import { WorkspaceDynamicRelation } from 'src/engine/twenty-orm/decorators/workspace-dynamic-relation.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.favorite, @@ -46,24 +47,28 @@ export class FavoriteWorkspaceEntity extends BaseWorkspaceEntity { label: 'Workspace Member', description: 'Favorite workspace member', icon: 'IconCircleUser', - joinColumn: 'workspaceMemberId', inverseSideFieldKey: 'favorites', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, }) workspaceMember: Relation<WorkspaceMemberWorkspaceEntity>; + @WorkspaceJoinColumn('workspaceMember') + workspaceMemberId: string; + @WorkspaceRelation({ standardId: FAVORITE_STANDARD_FIELD_IDS.person, type: RelationMetadataType.MANY_TO_ONE, label: 'Person', description: 'Favorite person', icon: 'IconUser', - joinColumn: 'personId', inverseSideTarget: () => PersonWorkspaceEntity, inverseSideFieldKey: 'favorites', }) @WorkspaceIsNullable() - person: Relation<PersonWorkspaceEntity>; + person: Relation<PersonWorkspaceEntity> | null; + + @WorkspaceJoinColumn('person') + personId: string; @WorkspaceRelation({ standardId: FAVORITE_STANDARD_FIELD_IDS.company, @@ -71,12 +76,14 @@ export class FavoriteWorkspaceEntity extends BaseWorkspaceEntity { label: 'Company', description: 'Favorite company', icon: 'IconBuildingSkyscraper', - joinColumn: 'companyId', inverseSideTarget: () => CompanyWorkspaceEntity, inverseSideFieldKey: 'favorites', }) @WorkspaceIsNullable() - company: Relation<CompanyWorkspaceEntity>; + company: Relation<CompanyWorkspaceEntity> | null; + + @WorkspaceJoinColumn('company') + companyId: string; @WorkspaceRelation({ standardId: FAVORITE_STANDARD_FIELD_IDS.opportunity, @@ -84,12 +91,14 @@ export class FavoriteWorkspaceEntity extends BaseWorkspaceEntity { label: 'Opportunity', description: 'Favorite opportunity', icon: 'IconTargetArrow', - joinColumn: 'opportunityId', inverseSideTarget: () => OpportunityWorkspaceEntity, inverseSideFieldKey: 'favorites', }) @WorkspaceIsNullable() - opportunity: Relation<OpportunityWorkspaceEntity>; + opportunity: Relation<OpportunityWorkspaceEntity> | null; + + @WorkspaceJoinColumn('opportunity') + opportunityId: string; @WorkspaceDynamicRelation({ type: RelationMetadataType.MANY_TO_ONE, diff --git a/packages/twenty-server/src/modules/messaging/blocklist-manager/jobs/messaging-blocklist-item-delete-messages.job.ts b/packages/twenty-server/src/modules/messaging/blocklist-manager/jobs/messaging-blocklist-item-delete-messages.job.ts index 3a52e42deb17..a2941c679d1e 100644 --- a/packages/twenty-server/src/modules/messaging/blocklist-manager/jobs/messaging-blocklist-item-delete-messages.job.ts +++ b/packages/twenty-server/src/modules/messaging/blocklist-manager/jobs/messaging-blocklist-item-delete-messages.job.ts @@ -1,10 +1,11 @@ -import { Injectable, Logger } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { Logger, Scope } from '@nestjs/common'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; +import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; @@ -16,10 +17,11 @@ export type BlocklistItemDeleteMessagesJobData = { blocklistItemId: string; }; -@Injectable() -export class BlocklistItemDeleteMessagesJob - implements MessageQueueJob<BlocklistItemDeleteMessagesJobData> -{ +@Processor({ + queueName: MessageQueue.messagingQueue, + scope: Scope.REQUEST, +}) +export class BlocklistItemDeleteMessagesJob { private readonly logger = new Logger(BlocklistItemDeleteMessagesJob.name); constructor( @@ -34,6 +36,7 @@ export class BlocklistItemDeleteMessagesJob private readonly threadCleanerService: MessagingMessageCleanerService, ) {} + @Process(BlocklistItemDeleteMessagesJob.name) async handle(data: BlocklistItemDeleteMessagesJobData): Promise<void> { const { workspaceId, blocklistItemId } = data; @@ -56,6 +59,12 @@ export class BlocklistItemDeleteMessagesJob `Deleting messages from ${handle} in workspace ${workspaceId} for workspace member ${workspaceMemberId}`, ); + if (!workspaceMemberId) { + throw new Error( + `Workspace member ID is not defined for blocklist item ${blocklistItemId} in workspace ${workspaceId}`, + ); + } + const messageChannels = await this.messageChannelRepository.getIdsByWorkspaceMemberId( workspaceMemberId, diff --git a/packages/twenty-server/src/modules/messaging/blocklist-manager/listeners/messaging-blocklist.listener.ts b/packages/twenty-server/src/modules/messaging/blocklist-manager/listeners/messaging-blocklist.listener.ts index c852bb9edb41..60433a2902fc 100644 --- a/packages/twenty-server/src/modules/messaging/blocklist-manager/listeners/messaging-blocklist.listener.ts +++ b/packages/twenty-server/src/modules/messaging/blocklist-manager/listeners/messaging-blocklist.listener.ts @@ -1,19 +1,20 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; +import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; -import { - BlocklistItemDeleteMessagesJobData, - BlocklistItemDeleteMessagesJob, -} from 'src/modules/messaging/blocklist-manager/jobs/messaging-blocklist-item-delete-messages.job'; -import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { + BlocklistItemDeleteMessagesJob, + BlocklistItemDeleteMessagesJobData, +} from 'src/modules/messaging/blocklist-manager/jobs/messaging-blocklist-item-delete-messages.job'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessagingChannelSyncStatusService } from 'src/modules/messaging/common/services/messaging-channel-sync-status.service'; import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; @@ -21,7 +22,7 @@ import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/stan @Injectable() export class MessagingBlocklistListener { constructor( - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.messagingQueue) private readonly messageQueueService: MessageQueueService, @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) private readonly connectedAccountRepository: ConnectedAccountRepository, diff --git a/packages/twenty-server/src/modules/messaging/common/messaging-common.module.ts b/packages/twenty-server/src/modules/messaging/common/messaging-common.module.ts index f13575d948d3..a700e79a7e6e 100644 --- a/packages/twenty-server/src/modules/messaging/common/messaging-common.module.ts +++ b/packages/twenty-server/src/modules/messaging/common/messaging-common.module.ts @@ -6,14 +6,12 @@ import { AnalyticsModule } from 'src/engine/core-modules/analytics/analytics.mod import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { AddPersonIdAndWorkspaceMemberIdModule } from 'src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.module'; +import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; import { MessagingChannelSyncStatusService } from 'src/modules/messaging/common/services/messaging-channel-sync-status.service'; import { MessagingErrorHandlingService } from 'src/modules/messaging/common/services/messaging-error-handling.service'; import { MessagingFetchByBatchesService } from 'src/modules/messaging/common/services/messaging-fetch-by-batch.service'; -import { MessagingMessageParticipantService } from 'src/modules/messaging/common/services/messaging-message-participant.service'; import { MessagingMessageThreadService } from 'src/modules/messaging/common/services/messaging-message-thread.service'; import { MessagingMessageService } from 'src/modules/messaging/common/services/messaging-message.service'; -import { MessagingSaveMessagesAndEnqueueContactCreationService } from 'src/modules/messaging/common/services/messaging-save-messages-and-enqueue-contact-creation.service'; import { MessagingTelemetryService } from 'src/modules/messaging/common/services/messaging-telemetry.service'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; @@ -34,26 +32,22 @@ import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/perso MessageThreadWorkspaceEntity, ]), TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), - AddPersonIdAndWorkspaceMemberIdModule, ], providers: [ MessagingMessageService, MessagingMessageThreadService, - MessagingSaveMessagesAndEnqueueContactCreationService, MessagingErrorHandlingService, MessagingTelemetryService, MessagingChannelSyncStatusService, - MessagingMessageParticipantService, MessagingFetchByBatchesService, + AddPersonIdAndWorkspaceMemberIdService, ], exports: [ MessagingMessageService, MessagingMessageThreadService, - MessagingSaveMessagesAndEnqueueContactCreationService, MessagingErrorHandlingService, MessagingTelemetryService, MessagingChannelSyncStatusService, - MessagingMessageParticipantService, MessagingFetchByBatchesService, ], }) diff --git a/packages/twenty-server/src/modules/messaging/common/query-hooks/message/can-access-message-thread.service.ts b/packages/twenty-server/src/modules/messaging/common/query-hooks/message/can-access-message-thread.service.ts index 095523ee6358..f79c972c6100 100644 --- a/packages/twenty-server/src/modules/messaging/common/query-hooks/message/can-access-message-thread.service.ts +++ b/packages/twenty-server/src/modules/messaging/common/query-hooks/message/can-access-message-thread.service.ts @@ -9,6 +9,7 @@ import { MessageChannelRepository } from 'src/modules/messaging/common/repositor import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; +import { isDefined } from 'src/utils/is-defined'; export class CanAccessMessageThreadService { constructor( @@ -46,7 +47,9 @@ export class CanAccessMessageThreadService { const messageChannelsConnectedAccounts = await this.connectedAccountRepository.getByIds( - messageChannels.map((channel) => channel.connectedAccountId), + messageChannels + .map((channel) => channel.connectedAccountId) + .filter(isDefined), workspaceId, ); diff --git a/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-many.pre-query.hook.ts b/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-many.pre-query.hook.ts index 8e82c578afe2..0e2f2a5d79a7 100644 --- a/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-many.pre-query.hook.ts +++ b/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-many.pre-query.hook.ts @@ -1,19 +1,16 @@ -import { - BadRequestException, - Injectable, - NotFoundException, -} from '@nestjs/common'; +import { BadRequestException, NotFoundException } from '@nestjs/common'; -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; import { FindManyResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { CanAccessMessageThreadService } from 'src/modules/messaging/common/query-hooks/message/can-access-message-thread.service'; import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; -@Injectable() -export class MessageFindManyPreQueryHook implements WorkspacePreQueryHook { +@WorkspaceQueryHook(`message.findMany`) +export class MessageFindManyPreQueryHook implements WorkspaceQueryHookInstance { constructor( @InjectObjectMetadataRepository( MessageChannelMessageAssociationWorkspaceEntity, diff --git a/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-one.pre-query-hook.ts b/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-one.pre-query-hook.ts index 2140ac01081a..713e024894f6 100644 --- a/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-one.pre-query-hook.ts +++ b/packages/twenty-server/src/modules/messaging/common/query-hooks/message/message-find-one.pre-query-hook.ts @@ -1,16 +1,17 @@ /* eslint-disable @typescript-eslint/no-unused-vars */ -import { Injectable, NotFoundException } from '@nestjs/common'; +import { NotFoundException } from '@nestjs/common'; -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; import { FindOneResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { CanAccessMessageThreadService } from 'src/modules/messaging/common/query-hooks/message/can-access-message-thread.service'; import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; -@Injectable() -export class MessageFindOnePreQueryHook implements WorkspacePreQueryHook { +@WorkspaceQueryHook(`message.findOne`) +export class MessageFindOnePreQueryHook implements WorkspaceQueryHookInstance { constructor( @InjectObjectMetadataRepository( MessageChannelMessageAssociationWorkspaceEntity, diff --git a/packages/twenty-server/src/modules/messaging/common/query-hooks/messaging-query-hook.module.ts b/packages/twenty-server/src/modules/messaging/common/query-hooks/messaging-query-hook.module.ts index f27adf462af7..4268de828017 100644 --- a/packages/twenty-server/src/modules/messaging/common/query-hooks/messaging-query-hook.module.ts +++ b/packages/twenty-server/src/modules/messaging/common/query-hooks/messaging-query-hook.module.ts @@ -20,14 +20,8 @@ import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/stan ], providers: [ CanAccessMessageThreadService, - { - provide: MessageFindOnePreQueryHook.name, - useClass: MessageFindOnePreQueryHook, - }, - { - provide: MessageFindManyPreQueryHook.name, - useClass: MessageFindManyPreQueryHook, - }, + MessageFindOnePreQueryHook, + MessageFindManyPreQueryHook, ], }) export class MessagingQueryHookModule {} diff --git a/packages/twenty-server/src/modules/messaging/common/repositories/message-channel-message-association.repository.ts b/packages/twenty-server/src/modules/messaging/common/repositories/message-channel-message-association.repository.ts index 8699a7670f3a..e9b0ffa0d56a 100644 --- a/packages/twenty-server/src/modules/messaging/common/repositories/message-channel-message-association.repository.ts +++ b/packages/twenty-server/src/modules/messaging/common/repositories/message-channel-message-association.repository.ts @@ -3,7 +3,6 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; @Injectable() @@ -17,7 +16,7 @@ export class MessageChannelMessageAssociationRepository { messageChannelId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelMessageAssociationWorkspaceEntity>[]> { + ): Promise<MessageChannelMessageAssociationWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -130,7 +129,7 @@ export class MessageChannelMessageAssociationRepository { messageChannelIds: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelMessageAssociationWorkspaceEntity>[]> { + ): Promise<MessageChannelMessageAssociationWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -183,7 +182,7 @@ export class MessageChannelMessageAssociationRepository { messageThreadExternalIds: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelMessageAssociationWorkspaceEntity>[]> { + ): Promise<MessageChannelMessageAssociationWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -200,7 +199,7 @@ export class MessageChannelMessageAssociationRepository { messageThreadExternalId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelMessageAssociationWorkspaceEntity> | null> { + ): Promise<MessageChannelMessageAssociationWorkspaceEntity | null> { const existingMessageChannelMessageAssociations = await this.getByMessageThreadExternalIds( [messageThreadExternalId], @@ -222,7 +221,7 @@ export class MessageChannelMessageAssociationRepository { messageIds: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelMessageAssociationWorkspaceEntity>[]> { + ): Promise<MessageChannelMessageAssociationWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -239,7 +238,7 @@ export class MessageChannelMessageAssociationRepository { messageThreadId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelMessageAssociationWorkspaceEntity>[]> { + ): Promise<MessageChannelMessageAssociationWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); diff --git a/packages/twenty-server/src/modules/messaging/common/repositories/message-channel.repository.ts b/packages/twenty-server/src/modules/messaging/common/repositories/message-channel.repository.ts index a3a8c727ca11..3e46454fb732 100644 --- a/packages/twenty-server/src/modules/messaging/common/repositories/message-channel.repository.ts +++ b/packages/twenty-server/src/modules/messaging/common/repositories/message-channel.repository.ts @@ -3,7 +3,6 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { MessageChannelWorkspaceEntity, MessageChannelSyncStatus, @@ -18,7 +17,7 @@ export class MessageChannelRepository { public async create( messageChannel: Pick< - ObjectRecord<MessageChannelWorkspaceEntity>, + MessageChannelWorkspaceEntity, | 'id' | 'connectedAccountId' | 'type' @@ -57,7 +56,11 @@ export class MessageChannelRepository { this.workspaceDataSourceService.getSchemaName(workspaceId); await this.workspaceDataSourceService.executeRawQuery( - `UPDATE ${dataSourceSchema}."messageChannel" SET "syncStatus" = NULL, "syncCursor" = '', "syncStageStartedAt" = NULL + `UPDATE ${dataSourceSchema}."messageChannel" + SET "syncStatus" = NULL, + "syncStage" = '${MessageChannelSyncStage.FULL_MESSAGE_LIST_FETCH_PENDING}', + "syncCursor" = '', + "syncStageStartedAt" = NULL WHERE "connectedAccountId" = $1`, [connectedAccountId], workspaceId, @@ -68,7 +71,7 @@ export class MessageChannelRepository { public async getAll( workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelWorkspaceEntity>[]> { + ): Promise<MessageChannelWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -84,7 +87,7 @@ export class MessageChannelRepository { connectedAccountId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelWorkspaceEntity>[]> { + ): Promise<MessageChannelWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -99,7 +102,7 @@ export class MessageChannelRepository { public async getFirstByConnectedAccountIdOrFail( connectedAccountId: string, workspaceId: string, - ): Promise<ObjectRecord<MessageChannelWorkspaceEntity>> { + ): Promise<MessageChannelWorkspaceEntity> { const messageChannel = await this.getFirstByConnectedAccountId( connectedAccountId, workspaceId, @@ -118,7 +121,7 @@ export class MessageChannelRepository { connectedAccountId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelWorkspaceEntity> | undefined> { + ): Promise<MessageChannelWorkspaceEntity | undefined> { const messageChannels = await this.getByConnectedAccountId( connectedAccountId, workspaceId, @@ -132,7 +135,7 @@ export class MessageChannelRepository { ids: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelWorkspaceEntity>[]> { + ): Promise<MessageChannelWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -148,7 +151,7 @@ export class MessageChannelRepository { id: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelWorkspaceEntity>> { + ): Promise<MessageChannelWorkspaceEntity> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -167,7 +170,7 @@ export class MessageChannelRepository { workspaceMemberId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageChannelWorkspaceEntity>[]> { + ): Promise<MessageChannelWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); diff --git a/packages/twenty-server/src/modules/messaging/common/repositories/message-participant.repository.ts b/packages/twenty-server/src/modules/messaging/common/repositories/message-participant.repository.ts index 7d6515ae4a3b..2cb2214f1168 100644 --- a/packages/twenty-server/src/modules/messaging/common/repositories/message-participant.repository.ts +++ b/packages/twenty-server/src/modules/messaging/common/repositories/message-participant.repository.ts @@ -3,7 +3,6 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; import { ParticipantWithId } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message'; @@ -17,7 +16,7 @@ export class MessageParticipantRepository { handles: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageParticipantWorkspaceEntity>[]> { + ): Promise<MessageParticipantWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); diff --git a/packages/twenty-server/src/modules/messaging/common/repositories/message.repository.ts b/packages/twenty-server/src/modules/messaging/common/repositories/message.repository.ts index ed75638c1f01..5f1d3eb25a4c 100644 --- a/packages/twenty-server/src/modules/messaging/common/repositories/message.repository.ts +++ b/packages/twenty-server/src/modules/messaging/common/repositories/message.repository.ts @@ -3,7 +3,6 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; @Injectable() @@ -40,7 +39,7 @@ export class MessageRepository { headerMessageId: string, workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageWorkspaceEntity> | null> { + ): Promise<MessageWorkspaceEntity | null> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -62,7 +61,7 @@ export class MessageRepository { messageIds: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageWorkspaceEntity>[]> { + ): Promise<MessageWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -94,7 +93,7 @@ export class MessageRepository { messageThreadIds: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageWorkspaceEntity>[]> { + ): Promise<MessageWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); diff --git a/packages/twenty-server/src/modules/messaging/common/services/messaging-channel-sync-status.service.ts b/packages/twenty-server/src/modules/messaging/common/services/messaging-channel-sync-status.service.ts index bf472695a8ec..0e3d3fe179c1 100644 --- a/packages/twenty-server/src/modules/messaging/common/services/messaging-channel-sync-status.service.ts +++ b/packages/twenty-server/src/modules/messaging/common/services/messaging-channel-sync-status.service.ts @@ -57,12 +57,10 @@ export class MessagingChannelSyncStatusService { messageChannelId: string, workspaceId: string, ) { - await this.cacheStorage.setPop( + await this.cacheStorage.del( `messages-to-import:${workspaceId}:gmail:${messageChannelId}`, ); - // TODO: remove nextPageToken from cache - await this.messageChannelRepository.resetSyncCursor( messageChannelId, workspaceId, @@ -126,7 +124,7 @@ export class MessagingChannelSyncStatusService { messageChannelId: string, workspaceId: string, ) { - await this.cacheStorage.setPop( + await this.cacheStorage.del( `messages-to-import:${workspaceId}:gmail:${messageChannelId}`, ); @@ -147,7 +145,7 @@ export class MessagingChannelSyncStatusService { messageChannelId: string, workspaceId: string, ) { - await this.cacheStorage.setPop( + await this.cacheStorage.del( `messages-to-import:${workspaceId}:gmail:${messageChannelId}`, ); diff --git a/packages/twenty-server/src/modules/messaging/common/services/messaging-error-handling.service.ts b/packages/twenty-server/src/modules/messaging/common/services/messaging-error-handling.service.ts index c8cdce1d02a9..2ab9cb8ac185 100644 --- a/packages/twenty-server/src/modules/messaging/common/services/messaging-error-handling.service.ts +++ b/packages/twenty-server/src/modules/messaging/common/services/messaging-error-handling.service.ts @@ -3,7 +3,6 @@ import { Injectable } from '@nestjs/common'; import snakeCase from 'lodash.snakecase'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { MessagingTelemetryService } from 'src/modules/messaging/common/services/messaging-telemetry.service'; @@ -36,7 +35,7 @@ export class MessagingErrorHandlingService { public async handleGmailError( error: GmailError, syncStep: SyncStep, - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, workspaceId: string, ): Promise<void> { const { code, reason } = error; @@ -51,6 +50,21 @@ export class MessagingErrorHandlingService { workspaceId, ); } + if (reason === 'failedPrecondition') { + await this.handleFailedPrecondition( + error, + syncStep, + messageChannel, + workspaceId, + ); + } else { + await this.handleUnknownError( + error, + syncStep, + messageChannel, + workspaceId, + ); + } break; case 404: await this.handleNotFound(error, syncStep, messageChannel, workspaceId); @@ -141,7 +155,7 @@ export class MessagingErrorHandlingService { private async handleRateLimitExceeded( error: GmailError, syncStep: SyncStep, - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, workspaceId: string, ): Promise<void> { await this.messagingTelemetryService.track({ @@ -152,50 +166,30 @@ export class MessagingErrorHandlingService { message: `${error.code}: ${error.reason}`, }); - if ( - messageChannel.throttleFailureCount >= MESSAGING_THROTTLE_MAX_ATTEMPTS - ) { - await this.messagingChannelSyncStatusService.markAsFailedUnknownAndFlushMessagesToImport( - messageChannel.id, - workspaceId, - ); - - return; - } - - await this.throttle(messageChannel, workspaceId); - - switch (syncStep) { - case 'full-message-list-fetch': - await this.messagingChannelSyncStatusService.scheduleFullMessageListFetch( - messageChannel.id, - workspaceId, - ); - break; - - case 'partial-message-list-fetch': - await this.messagingChannelSyncStatusService.schedulePartialMessageListFetch( - messageChannel.id, - workspaceId, - ); - break; + await this.handleThrottle(syncStep, messageChannel, workspaceId); + } - case 'messages-import': - await this.messagingChannelSyncStatusService.scheduleMessagesImport( - messageChannel.id, - workspaceId, - ); - break; + private async handleFailedPrecondition( + error: GmailError, + syncStep: SyncStep, + messageChannel: MessageChannelWorkspaceEntity, + workspaceId: string, + ): Promise<void> { + await this.messagingTelemetryService.track({ + eventName: `${snakeCase(syncStep)}.error.failed_precondition`, + workspaceId, + connectedAccountId: messageChannel.connectedAccountId, + messageChannelId: messageChannel.id, + message: `${error.code}: ${error.reason}`, + }); - default: - break; - } + await this.handleThrottle(syncStep, messageChannel, workspaceId); } private async handleInsufficientPermissions( error: GmailError, syncStep: SyncStep, - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, workspaceId: string, ): Promise<void> { await this.messagingTelemetryService.track({ @@ -211,6 +205,12 @@ export class MessagingErrorHandlingService { workspaceId, ); + if (!messageChannel.connectedAccountId) { + throw new Error( + `Connected account ID is not defined for message channel ${messageChannel.id} in workspace ${workspaceId}`, + ); + } + await this.connectedAccountRepository.updateAuthFailedAt( messageChannel.connectedAccountId, workspaceId, @@ -220,7 +220,7 @@ export class MessagingErrorHandlingService { private async handleNotFound( error: GmailError, syncStep: SyncStep, - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, workspaceId: string, ): Promise<void> { if (syncStep === 'messages-import') { @@ -241,8 +241,53 @@ export class MessagingErrorHandlingService { ); } + private async handleThrottle( + syncStep: SyncStep, + messageChannel: MessageChannelWorkspaceEntity, + workspaceId: string, + ): Promise<void> { + if ( + messageChannel.throttleFailureCount >= MESSAGING_THROTTLE_MAX_ATTEMPTS + ) { + await this.messagingChannelSyncStatusService.markAsFailedUnknownAndFlushMessagesToImport( + messageChannel.id, + workspaceId, + ); + + return; + } + + await this.throttle(messageChannel, workspaceId); + + switch (syncStep) { + case 'full-message-list-fetch': + await this.messagingChannelSyncStatusService.scheduleFullMessageListFetch( + messageChannel.id, + workspaceId, + ); + break; + + case 'partial-message-list-fetch': + await this.messagingChannelSyncStatusService.schedulePartialMessageListFetch( + messageChannel.id, + workspaceId, + ); + break; + + case 'messages-import': + await this.messagingChannelSyncStatusService.scheduleMessagesImport( + messageChannel.id, + workspaceId, + ); + break; + + default: + break; + } + } + private async throttle( - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, workspaceId: string, ): Promise<void> { await this.messageChannelRepository.incrementThrottleFailureCount( @@ -258,4 +303,28 @@ export class MessagingErrorHandlingService { message: `Increment throttle failure count to ${messageChannel.throttleFailureCount}`, }); } + + private async handleUnknownError( + error: GmailError, + syncStep: SyncStep, + messageChannel: MessageChannelWorkspaceEntity, + workspaceId: string, + ): Promise<void> { + await this.messagingTelemetryService.track({ + eventName: `${snakeCase(syncStep)}.error.unknown`, + workspaceId, + connectedAccountId: messageChannel.connectedAccountId, + messageChannelId: messageChannel.id, + message: `${error.code}: ${error.reason}`, + }); + + await this.messagingChannelSyncStatusService.markAsFailedUnknownAndFlushMessagesToImport( + messageChannel.id, + workspaceId, + ); + + throw new Error( + `Unhandled Gmail error code ${error.code} with reason ${error.reason}`, + ); + } } diff --git a/packages/twenty-server/src/modules/messaging/common/services/messaging-fetch-by-batch.service.ts b/packages/twenty-server/src/modules/messaging/common/services/messaging-fetch-by-batch.service.ts index 354ab92659f0..978850bb9629 100644 --- a/packages/twenty-server/src/modules/messaging/common/services/messaging-fetch-by-batch.service.ts +++ b/packages/twenty-server/src/modules/messaging/common/services/messaging-fetch-by-batch.service.ts @@ -5,25 +5,31 @@ import { AxiosResponse } from 'axios'; import { GmailMessageParsedResponse } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message-parsed-response'; import { BatchQueries } from 'src/modules/messaging/message-import-manager/types/batch-queries'; +import { createQueriesFromMessageIds } from 'src/modules/messaging/message-import-manager/utils/create-queries-from-message-ids.util'; @Injectable() export class MessagingFetchByBatchesService { constructor(private readonly httpService: HttpService) {} async fetchAllByBatches( - queries: BatchQueries, + messageIds: string[], accessToken: string, boundary: string, - ): Promise<AxiosResponse<any, any>[]> { - const batchLimit = 50; + ): Promise<{ + messageIdsByBatch: string[][]; + batchResponses: AxiosResponse<any, any>[]; + }> { + const batchLimit = 20; let batchOffset = 0; let batchResponses: AxiosResponse<any, any>[] = []; - while (batchOffset < queries.length) { + const messageIdsByBatch: string[][] = []; + + while (batchOffset < messageIds.length) { const batchResponse = await this.fetchBatch( - queries, + messageIds, accessToken, batchOffset, batchLimit, @@ -32,19 +38,25 @@ export class MessagingFetchByBatchesService { batchResponses = batchResponses.concat(batchResponse); + messageIdsByBatch.push( + messageIds.slice(batchOffset, batchOffset + batchLimit), + ); + batchOffset += batchLimit; } - return batchResponses; + return { messageIdsByBatch, batchResponses }; } async fetchBatch( - queries: BatchQueries, + messageIds: string[], accessToken: string, batchOffset: number, batchLimit: number, boundary: string, ): Promise<AxiosResponse<any, any>> { + const queries = createQueriesFromMessageIds(messageIds); + const limitedQueries = queries.slice(batchOffset, batchOffset + batchLimit); const response = await this.httpService.axiosRef.post( diff --git a/packages/twenty-server/src/modules/messaging/common/services/messaging-message.service.ts b/packages/twenty-server/src/modules/messaging/common/services/messaging-message.service.ts index 4b475d095d7d..e96d2ab3044d 100644 --- a/packages/twenty-server/src/modules/messaging/common/services/messaging-message.service.ts +++ b/packages/twenty-server/src/modules/messaging/common/services/messaging-message.service.ts @@ -1,27 +1,22 @@ -import { Injectable, Logger } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; -import { DataSource, EntityManager } from 'typeorm'; +import { EntityManager } from 'typeorm'; import { v4 } from 'uuid'; -import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository'; -import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessageThreadRepository } from 'src/modules/messaging/common/repositories/message-thread.repository'; import { MessageRepository } from 'src/modules/messaging/common/repositories/message.repository'; +import { MessagingMessageThreadService } from 'src/modules/messaging/common/services/messaging-message-thread.service'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; -import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; import { GmailMessage } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message'; -import { MessagingMessageThreadService } from 'src/modules/messaging/common/services/messaging-message-thread.service'; @Injectable() export class MessagingMessageService { - private readonly logger = new Logger(MessagingMessageService.name); - constructor( private readonly workspaceDataSourceService: WorkspaceDataSourceService, @InjectObjectMetadataRepository( @@ -30,8 +25,6 @@ export class MessagingMessageService { private readonly messageChannelMessageAssociationRepository: MessageChannelMessageAssociationRepository, @InjectObjectMetadataRepository(MessageWorkspaceEntity) private readonly messageRepository: MessageRepository, - @InjectObjectMetadataRepository(MessageChannelWorkspaceEntity) - private readonly messageChannelRepository: MessageChannelRepository, @InjectObjectMetadataRepository(MessageThreadWorkspaceEntity) private readonly messageThreadRepository: MessageThreadRepository, private readonly messageThreadService: MessagingMessageThreadService, @@ -39,7 +32,7 @@ export class MessagingMessageService { public async saveMessagesWithinTransaction( messages: GmailMessage[], - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, + connectedAccount: ConnectedAccountWorkspaceEntity, gmailMessageChannelId: string, workspaceId: string, transactionManager: EntityManager, @@ -68,6 +61,12 @@ export class MessagingMessageService { transactionManager, ); + if (!savedOrExistingMessageThreadId) { + throw new Error( + `No message thread found for message ${message.headerMessageId} in workspace ${workspaceId} in saveMessages`, + ); + } + const savedOrExistingMessageId = await this.saveMessageOrReturnExistingMessage( message, @@ -96,102 +95,10 @@ export class MessagingMessageService { return messageExternalIdsAndIdsMap; } - public async saveMessages( - messages: GmailMessage[], - workspaceDataSource: DataSource, - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, - gmailMessageChannelId: string, - workspaceId: string, - ): Promise<Map<string, string>> { - const messageExternalIdsAndIdsMap = new Map<string, string>(); - - try { - let keepImporting = true; - - for (const message of messages) { - if (!keepImporting) { - break; - } - - await workspaceDataSource?.transaction( - async (manager: EntityManager) => { - const gmailMessageChannel = - await this.messageChannelRepository.getByIds( - [gmailMessageChannelId], - workspaceId, - manager, - ); - - if (gmailMessageChannel.length === 0) { - this.logger.error( - `No message channel found for connected account ${connectedAccount.id} in workspace ${workspaceId} in saveMessages`, - ); - - keepImporting = false; - - return; - } - - const existingMessageChannelMessageAssociationsCount = - await this.messageChannelMessageAssociationRepository.countByMessageExternalIdsAndMessageChannelId( - [message.externalId], - gmailMessageChannelId, - workspaceId, - manager, - ); - - if (existingMessageChannelMessageAssociationsCount > 0) { - return; - } - - // TODO: This does not handle all thread merging use cases and might create orphan threads. - const savedOrExistingMessageThreadId = - await this.messageThreadService.saveMessageThreadOrReturnExistingMessageThread( - message.headerMessageId, - message.messageThreadExternalId, - workspaceId, - manager, - ); - - const savedOrExistingMessageId = - await this.saveMessageOrReturnExistingMessage( - message, - savedOrExistingMessageThreadId, - connectedAccount, - workspaceId, - manager, - ); - - messageExternalIdsAndIdsMap.set( - message.externalId, - savedOrExistingMessageId, - ); - - await this.messageChannelMessageAssociationRepository.insert( - gmailMessageChannelId, - savedOrExistingMessageId, - message.externalId, - savedOrExistingMessageThreadId, - message.messageThreadExternalId, - workspaceId, - manager, - ); - }, - ); - } - } catch (error) { - throw new Error( - `Error saving connected account ${connectedAccount.id} messages to workspace ${workspaceId}: ${error.message}`, - ); - } - - return messageExternalIdsAndIdsMap; - } - private async saveMessageOrReturnExistingMessage( message: GmailMessage, messageThreadId: string, - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, + connectedAccount: ConnectedAccountWorkspaceEntity, workspaceId: string, manager: EntityManager, ): Promise<string> { @@ -209,7 +116,10 @@ export class MessagingMessageService { const newMessageId = v4(); const messageDirection = - connectedAccount.handle === message.fromHandle ? 'outgoing' : 'incoming'; + connectedAccount.handle === message.fromHandle || + connectedAccount.handleAliases?.includes(message.fromHandle) + ? 'outgoing' + : 'incoming'; const receivedAt = new Date(parseInt(message.internalDate)); diff --git a/packages/twenty-server/src/modules/messaging/common/services/messaging-telemetry.service.ts b/packages/twenty-server/src/modules/messaging/common/services/messaging-telemetry.service.ts index ae9cec0fa136..0234f7fbcb4e 100644 --- a/packages/twenty-server/src/modules/messaging/common/services/messaging-telemetry.service.ts +++ b/packages/twenty-server/src/modules/messaging/common/services/messaging-telemetry.service.ts @@ -5,7 +5,7 @@ import { EnvironmentService } from 'src/engine/integrations/environment/environm type MessagingTelemetryTrackInput = { eventName: string; - workspaceId: string; + workspaceId?: string; userId?: string; connectedAccountId?: string; messageChannelId?: string; diff --git a/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity.ts b/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity.ts index 91e9fa828209..0e0171766c54 100644 --- a/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity.ts +++ b/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity.ts @@ -14,6 +14,7 @@ import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metad import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.messageChannelMessageAssociation, @@ -35,7 +36,7 @@ export class MessageChannelMessageAssociationWorkspaceEntity extends BaseWorkspa icon: 'IconHash', }) @WorkspaceIsNullable() - messageExternalId: string; + messageExternalId: string | null; @WorkspaceField({ standardId: @@ -46,7 +47,7 @@ export class MessageChannelMessageAssociationWorkspaceEntity extends BaseWorkspa icon: 'IconHash', }) @WorkspaceIsNullable() - messageThreadExternalId: string; + messageThreadExternalId: string | null; @WorkspaceRelation({ standardId: @@ -55,12 +56,14 @@ export class MessageChannelMessageAssociationWorkspaceEntity extends BaseWorkspa label: 'Message Channel Id', description: 'Message Channel Id', icon: 'IconHash', - joinColumn: 'messageChannelId', inverseSideTarget: () => MessageChannelWorkspaceEntity, inverseSideFieldKey: 'messageChannelMessageAssociations', }) @WorkspaceIsNullable() - messageChannel: Relation<MessageChannelWorkspaceEntity>; + messageChannel: Relation<MessageChannelWorkspaceEntity> | null; + + @WorkspaceJoinColumn('messageChannel') + messageChannelId: string; @WorkspaceRelation({ standardId: MESSAGE_CHANNEL_MESSAGE_ASSOCIATION_STANDARD_FIELD_IDS.message, @@ -68,12 +71,14 @@ export class MessageChannelMessageAssociationWorkspaceEntity extends BaseWorkspa label: 'Message Id', description: 'Message Id', icon: 'IconHash', - joinColumn: 'messageId', inverseSideTarget: () => MessageWorkspaceEntity, inverseSideFieldKey: 'messageChannelMessageAssociations', }) @WorkspaceIsNullable() - message: Relation<MessageWorkspaceEntity>; + message: Relation<MessageWorkspaceEntity> | null; + + @WorkspaceJoinColumn('message') + messageId: string; @WorkspaceRelation({ standardId: @@ -82,10 +87,12 @@ export class MessageChannelMessageAssociationWorkspaceEntity extends BaseWorkspa label: 'Message Thread Id', description: 'Message Thread Id', icon: 'IconHash', - joinColumn: 'messageThreadId', inverseSideTarget: () => MessageThreadWorkspaceEntity, inverseSideFieldKey: 'messageChannelMessageAssociations', }) @WorkspaceIsNullable() - messageThread: Relation<MessageThreadWorkspaceEntity>; + messageThread: Relation<MessageThreadWorkspaceEntity> | null; + + @WorkspaceJoinColumn('messageThread') + messageThreadId: string; } diff --git a/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel.workspace-entity.ts b/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel.workspace-entity.ts index 92175c6afbf8..565e705095dc 100644 --- a/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel.workspace-entity.ts +++ b/packages/twenty-server/src/modules/messaging/common/standard-objects/message-channel.workspace-entity.ts @@ -5,25 +5,20 @@ import { RelationMetadataType, RelationOnDeleteAction, } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; -import { MESSAGE_CHANNEL_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; -import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; -import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; -import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; +import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; +import { MESSAGE_CHANNEL_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; +import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; export enum MessageChannelSyncStatus { - // TO BE DEPRECATED - PENDING = 'PENDING', - SUCCEEDED = 'SUCCEEDED', - FAILED = 'FAILED', - - // NEW STATUSES NOT_SYNCED = 'NOT_SYNCED', ONGOING = 'ONGOING', COMPLETED = 'COMPLETED', @@ -51,6 +46,12 @@ export enum MessageChannelType { SMS = 'sms', } +export enum MessageChannelContactAutoCreationPolicy { + SENT_AND_RECEIVED = 'SENT_AND_RECEIVED', + SENT = 'SENT', + NONE = 'NONE', +} + @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.messageChannel, namePlural: 'messageChannels', @@ -125,6 +126,7 @@ export class MessageChannelWorkspaceEntity extends BaseWorkspaceEntity { }) type: string; + // TODO: Deprecate this field and migrate data to contactAutoCreationFor @WorkspaceField({ standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.isContactAutoCreationEnabled, type: FieldMetadataType.BOOLEAN, @@ -135,6 +137,57 @@ export class MessageChannelWorkspaceEntity extends BaseWorkspaceEntity { }) isContactAutoCreationEnabled: boolean; + @WorkspaceField({ + standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.contactAutoCreationPolicy, + type: FieldMetadataType.SELECT, + label: 'Contact auto creation policy', + description: + 'Automatically create People records when receiving or sending emails', + icon: 'IconUserCircle', + options: [ + { + value: MessageChannelContactAutoCreationPolicy.SENT_AND_RECEIVED, + label: 'Sent and Received', + position: 0, + color: 'green', + }, + { + value: MessageChannelContactAutoCreationPolicy.SENT, + label: 'Sent', + position: 1, + color: 'blue', + }, + { + value: MessageChannelContactAutoCreationPolicy.NONE, + label: 'None', + position: 2, + color: 'red', + }, + ], + defaultValue: `'${MessageChannelContactAutoCreationPolicy.SENT}'`, + }) + contactAutoCreationPolicy: MessageChannelContactAutoCreationPolicy; + + @WorkspaceField({ + standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.excludeNonProfessionalEmails, + type: FieldMetadataType.BOOLEAN, + label: 'Exclude non professional emails', + description: 'Exclude non professional emails', + icon: 'IconBriefcase', + defaultValue: true, + }) + excludeNonProfessionalEmails: boolean; + + @WorkspaceField({ + standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.excludeGroupEmails, + type: FieldMetadataType.BOOLEAN, + label: 'Exclude group emails', + description: 'Exclude group emails', + icon: 'IconUsersGroup', + defaultValue: true, + }) + excludeGroupEmails: boolean; + @WorkspaceField({ standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.isSyncEnabled, type: FieldMetadataType.BOOLEAN, @@ -162,7 +215,7 @@ export class MessageChannelWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconHistory', }) @WorkspaceIsNullable() - syncedAt: string; + syncedAt: string | null; @WorkspaceField({ standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.syncStatus, @@ -171,26 +224,6 @@ export class MessageChannelWorkspaceEntity extends BaseWorkspaceEntity { description: 'Sync status', icon: 'IconStatusChange', options: [ - // TO BE DEPRECATED: PENDING, SUCCEEDED, FAILED - { - value: MessageChannelSyncStatus.PENDING, - label: 'Pending', - position: 0, - color: 'blue', - }, - { - value: MessageChannelSyncStatus.SUCCEEDED, - label: 'Succeeded', - position: 2, - color: 'green', - }, - { - value: MessageChannelSyncStatus.FAILED, - label: 'Failed', - position: 3, - color: 'red', - }, - // NEW STATUSES { value: MessageChannelSyncStatus.ONGOING, label: 'Ongoing', @@ -224,7 +257,7 @@ export class MessageChannelWorkspaceEntity extends BaseWorkspaceEntity { ], }) @WorkspaceIsNullable() - syncStatus: MessageChannelSyncStatus; + syncStatus: MessageChannelSyncStatus | null; @WorkspaceField({ standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.syncStage, @@ -282,7 +315,7 @@ export class MessageChannelWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconHistory', }) @WorkspaceIsNullable() - syncStageStartedAt: string; + syncStageStartedAt: string | null; @WorkspaceField({ standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.throttleFailureCount, @@ -300,12 +333,14 @@ export class MessageChannelWorkspaceEntity extends BaseWorkspaceEntity { label: 'Connected Account', description: 'Connected Account', icon: 'IconUserCircle', - joinColumn: 'connectedAccountId', inverseSideTarget: () => ConnectedAccountWorkspaceEntity, inverseSideFieldKey: 'messageChannels', }) connectedAccount: Relation<ConnectedAccountWorkspaceEntity>; + @WorkspaceJoinColumn('connectedAccount') + connectedAccountId: string; + @WorkspaceRelation({ standardId: MESSAGE_CHANNEL_STANDARD_FIELD_IDS.messageChannelMessageAssociations, diff --git a/packages/twenty-server/src/modules/messaging/common/standard-objects/message-participant.workspace-entity.ts b/packages/twenty-server/src/modules/messaging/common/standard-objects/message-participant.workspace-entity.ts index b8c1e09f6447..0821bf32c012 100644 --- a/packages/twenty-server/src/modules/messaging/common/standard-objects/message-participant.workspace-entity.ts +++ b/packages/twenty-server/src/modules/messaging/common/standard-objects/message-participant.workspace-entity.ts @@ -14,6 +14,7 @@ import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-re import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.messageParticipant, @@ -66,24 +67,28 @@ export class MessageParticipantWorkspaceEntity extends BaseWorkspaceEntity { label: 'Message', description: 'Message', icon: 'IconMessage', - joinColumn: 'messageId', inverseSideTarget: () => MessageWorkspaceEntity, inverseSideFieldKey: 'messageParticipants', }) message: Relation<MessageWorkspaceEntity>; + @WorkspaceJoinColumn('message') + messageId: string; + @WorkspaceRelation({ standardId: MESSAGE_PARTICIPANT_STANDARD_FIELD_IDS.person, type: RelationMetadataType.MANY_TO_ONE, label: 'Person', description: 'Person', icon: 'IconUser', - joinColumn: 'personId', inverseSideTarget: () => PersonWorkspaceEntity, inverseSideFieldKey: 'messageParticipants', }) @WorkspaceIsNullable() - person: Relation<PersonWorkspaceEntity>; + person: Relation<PersonWorkspaceEntity> | null; + + @WorkspaceJoinColumn('person') + personId: string | null; @WorkspaceRelation({ standardId: MESSAGE_PARTICIPANT_STANDARD_FIELD_IDS.workspaceMember, @@ -91,10 +96,12 @@ export class MessageParticipantWorkspaceEntity extends BaseWorkspaceEntity { label: 'Workspace Member', description: 'Workspace member', icon: 'IconCircleUser', - joinColumn: 'workspaceMemberId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'messageParticipants', }) @WorkspaceIsNullable() - workspaceMember: Relation<WorkspaceMemberWorkspaceEntity>; + workspaceMember: Relation<WorkspaceMemberWorkspaceEntity> | null; + + @WorkspaceJoinColumn('workspaceMember') + workspaceMemberId: string | null; } diff --git a/packages/twenty-server/src/modules/messaging/common/standard-objects/message.workspace-entity.ts b/packages/twenty-server/src/modules/messaging/common/standard-objects/message.workspace-entity.ts index a3ad1ea48c8c..a9475a96d6ce 100644 --- a/packages/twenty-server/src/modules/messaging/common/standard-objects/message.workspace-entity.ts +++ b/packages/twenty-server/src/modules/messaging/common/standard-objects/message.workspace-entity.ts @@ -17,6 +17,7 @@ import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-re import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.message, @@ -78,7 +79,7 @@ export class MessageWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconCalendar', }) @WorkspaceIsNullable() - receivedAt: string; + receivedAt: string | null; @WorkspaceRelation({ standardId: MESSAGE_STANDARD_FIELD_IDS.messageThread, @@ -86,13 +87,15 @@ export class MessageWorkspaceEntity extends BaseWorkspaceEntity { label: 'Message Thread Id', description: 'Message Thread Id', icon: 'IconHash', - joinColumn: 'messageThreadId', inverseSideTarget: () => MessageThreadWorkspaceEntity, inverseSideFieldKey: 'messages', onDelete: RelationOnDeleteAction.CASCADE, }) @WorkspaceIsNullable() - messageThread: Relation<MessageThreadWorkspaceEntity>; + messageThread: Relation<MessageThreadWorkspaceEntity> | null; + + @WorkspaceJoinColumn('messageThread') + messageThreadId: string | null; @WorkspaceRelation({ standardId: MESSAGE_STANDARD_FIELD_IDS.messageParticipants, diff --git a/packages/twenty-server/src/modules/messaging/message-cleaner/jobs/messaging-connected-account-deletion-cleanup.job.ts b/packages/twenty-server/src/modules/messaging/message-cleaner/jobs/messaging-connected-account-deletion-cleanup.job.ts index 5cba9355cb84..c9cc4e9e993e 100644 --- a/packages/twenty-server/src/modules/messaging/message-cleaner/jobs/messaging-connected-account-deletion-cleanup.job.ts +++ b/packages/twenty-server/src/modules/messaging/message-cleaner/jobs/messaging-connected-account-deletion-cleanup.job.ts @@ -1,18 +1,20 @@ -import { Injectable, Logger } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { Logger, Scope } from '@nestjs/common'; import { MessagingMessageCleanerService } from 'src/modules/messaging/message-cleaner/services/messaging-message-cleaner.service'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; export type MessagingConnectedAccountDeletionCleanupJobData = { workspaceId: string; connectedAccountId: string; }; -@Injectable() -export class MessagingConnectedAccountDeletionCleanupJob - implements MessageQueueJob<MessagingConnectedAccountDeletionCleanupJobData> -{ +@Processor({ + queueName: MessageQueue.messagingQueue, + scope: Scope.REQUEST, +}) +export class MessagingConnectedAccountDeletionCleanupJob { private readonly logger = new Logger( MessagingConnectedAccountDeletionCleanupJob.name, ); @@ -21,6 +23,7 @@ export class MessagingConnectedAccountDeletionCleanupJob private readonly messageCleanerService: MessagingMessageCleanerService, ) {} + @Process(MessagingConnectedAccountDeletionCleanupJob.name) async handle( data: MessagingConnectedAccountDeletionCleanupJobData, ): Promise<void> { diff --git a/packages/twenty-server/src/modules/messaging/message-cleaner/listeners/messaging-message-cleaner-connected-account.listener.ts b/packages/twenty-server/src/modules/messaging/message-cleaner/listeners/messaging-message-cleaner-connected-account.listener.ts index 24d5deb2974c..8880ad5d10d4 100644 --- a/packages/twenty-server/src/modules/messaging/message-cleaner/listeners/messaging-message-cleaner-connected-account.listener.ts +++ b/packages/twenty-server/src/modules/messaging/message-cleaner/listeners/messaging-message-cleaner-connected-account.listener.ts @@ -1,24 +1,26 @@ -import { Inject } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { - DeleteConnectedAccountAssociatedCalendarDataJobData, - DeleteConnectedAccountAssociatedCalendarDataJob, -} from 'src/modules/calendar/jobs/delete-connected-account-associated-calendar-data.job'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { MessagingConnectedAccountDeletionCleanupJob, MessagingConnectedAccountDeletionCleanupJobData, } from 'src/modules/messaging/message-cleaner/jobs/messaging-connected-account-deletion-cleanup.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { + DeleteConnectedAccountAssociatedCalendarDataJobData, + DeleteConnectedAccountAssociatedCalendarDataJob, +} from 'src/modules/calendar/calendar-event-cleaner/jobs/delete-connected-account-associated-calendar-data.job'; +@Injectable() export class MessagingMessageCleanerConnectedAccountListener { constructor( - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.messagingQueue) private readonly messageQueueService: MessageQueueService, - @Inject(MessageQueue.calendarQueue) + @InjectMessageQueue(MessageQueue.calendarQueue) private readonly calendarQueueService: MessageQueueService, ) {} diff --git a/packages/twenty-server/src/modules/messaging/message-cleaner/messaging-message-cleaner.module.ts b/packages/twenty-server/src/modules/messaging/message-cleaner/messaging-message-cleaner.module.ts index 4562bd75bad6..ae4221a85e85 100644 --- a/packages/twenty-server/src/modules/messaging/message-cleaner/messaging-message-cleaner.module.ts +++ b/packages/twenty-server/src/modules/messaging/message-cleaner/messaging-message-cleaner.module.ts @@ -1,6 +1,6 @@ import { Module } from '@nestjs/common'; -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; import { MessagingConnectedAccountDeletionCleanupJob } from 'src/modules/messaging/message-cleaner/jobs/messaging-connected-account-deletion-cleanup.job'; @@ -9,17 +9,14 @@ import { MessagingMessageCleanerService } from 'src/modules/messaging/message-cl @Module({ imports: [ - ObjectMetadataRepositoryModule.forFeature([ + TwentyORMModule.forFeature([ MessageWorkspaceEntity, MessageThreadWorkspaceEntity, ]), ], providers: [ MessagingMessageCleanerService, - { - provide: MessagingConnectedAccountDeletionCleanupJob.name, - useClass: MessagingConnectedAccountDeletionCleanupJob, - }, + MessagingConnectedAccountDeletionCleanupJob, MessagingMessageCleanerConnectedAccountListener, ], exports: [MessagingMessageCleanerService], diff --git a/packages/twenty-server/src/modules/messaging/message-cleaner/services/messaging-message-cleaner.service.ts b/packages/twenty-server/src/modules/messaging/message-cleaner/services/messaging-message-cleaner.service.ts index 19a9d6b2d185..21b9f5b28185 100644 --- a/packages/twenty-server/src/modules/messaging/message-cleaner/services/messaging-message-cleaner.service.ts +++ b/packages/twenty-server/src/modules/messaging/message-cleaner/services/messaging-message-cleaner.service.ts @@ -1,8 +1,9 @@ import { Injectable } from '@nestjs/common'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { MessageThreadRepository } from 'src/modules/messaging/common/repositories/message-thread.repository'; -import { MessageRepository } from 'src/modules/messaging/common/repositories/message.repository'; +import { EntityManager } from 'typeorm'; + +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; import { MessageThreadWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-thread.workspace-entity'; import { MessageWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message.workspace-entity'; import { deleteUsingPagination } from 'src/modules/messaging/message-cleaner/utils/delete-using-pagination.util'; @@ -10,31 +11,74 @@ import { deleteUsingPagination } from 'src/modules/messaging/message-cleaner/uti @Injectable() export class MessagingMessageCleanerService { constructor( - @InjectObjectMetadataRepository(MessageWorkspaceEntity) - private readonly messageRepository: MessageRepository, - @InjectObjectMetadataRepository(MessageThreadWorkspaceEntity) - private readonly messageThreadRepository: MessageThreadRepository, + @InjectWorkspaceRepository(MessageWorkspaceEntity) + private readonly messageRepository: WorkspaceRepository<MessageWorkspaceEntity>, + @InjectWorkspaceRepository(MessageThreadWorkspaceEntity) + private readonly messageThreadRepository: WorkspaceRepository<MessageThreadWorkspaceEntity>, ) {} public async cleanWorkspaceThreads(workspaceId: string) { await deleteUsingPagination( workspaceId, 500, - this.messageRepository.getNonAssociatedMessageIdsPaginated.bind( - this.messageRepository, - ), - this.messageRepository.deleteByIds.bind(this.messageRepository), + async ( + limit: number, + offset: number, + workspaceId: string, + transactionManager?: EntityManager, + ) => { + const nonAssociatedMessages = await this.messageRepository.find( + { + where: { + messageChannelMessageAssociations: [], + }, + take: limit, + skip: offset, + relations: ['messageChannelMessageAssociations'], + }, + transactionManager, + ); + + return nonAssociatedMessages.map(({ id }) => id); + }, + async ( + ids: string[], + workspaceId: string, + transactionManager?: EntityManager, + ) => { + await this.messageRepository.delete(ids, transactionManager); + }, ); await deleteUsingPagination( workspaceId, 500, - this.messageThreadRepository.getOrphanThreadIdsPaginated.bind( - this.messageThreadRepository, - ), - this.messageThreadRepository.deleteByIds.bind( - this.messageThreadRepository, - ), + async ( + limit: number, + offset: number, + workspaceId: string, + transactionManager?: EntityManager, + ) => { + const orphanThreads = await this.messageThreadRepository.find( + { + where: { + messages: [], + }, + take: limit, + skip: offset, + }, + transactionManager, + ); + + return orphanThreads.map(({ id }) => id); + }, + async ( + ids: string[], + workspaceId: string, + transactionManager?: EntityManager, + ) => { + await this.messageThreadRepository.delete(ids, transactionManager); + }, ); } } diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/commands/messaging-single-message-import.command.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/commands/messaging-single-message-import.command.ts new file mode 100644 index 000000000000..c55465f7dedc --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/commands/messaging-single-message-import.command.ts @@ -0,0 +1,69 @@ +import { Command, CommandRunner, Option } from 'nest-commander'; + +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { + MessagingAddSingleMessageToCacheForImportJob, + MessagingAddSingleMessageToCacheForImportJobData, +} from 'src/modules/messaging/message-import-manager/jobs/messaging-add-single-message-to-cache-for-import.job'; + +type MessagingSingleMessageImportCommandOptions = { + messageExternalId: string; + messageChannelId: string; + workspaceId: string; +}; + +@Command({ + name: 'messaging:single-message-import', + description: 'Enqueue a job to schedule the import of a single message', +}) +export class MessagingSingleMessageImportCommand extends CommandRunner { + constructor( + @InjectMessageQueue(MessageQueue.messagingQueue) + private readonly messageQueueService: MessageQueueService, + ) { + super(); + } + + async run( + _passedParam: string[], + options: MessagingSingleMessageImportCommandOptions, + ): Promise<void> { + await this.messageQueueService.add<MessagingAddSingleMessageToCacheForImportJobData>( + MessagingAddSingleMessageToCacheForImportJob.name, + { + messageExternalId: options.messageExternalId, + messageChannelId: options.messageChannelId, + workspaceId: options.workspaceId, + }, + ); + } + + @Option({ + flags: '-m, --message-external-id [message_external_id]', + description: 'Message external ID', + required: true, + }) + parseMessageId(value: string): string { + return value; + } + + @Option({ + flags: '-M, --message-channel-id [message_channel_id]', + description: 'Message channel ID', + required: true, + }) + parseMessageChannelId(value: string): string { + return value; + } + + @Option({ + flags: '-w, --workspace-id [workspace_id]', + description: 'Workspace ID', + required: true, + }) + parseWorkspaceId(value: string): string { + return value; + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-message-list-fetch.cron.command.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-message-list-fetch.cron.command.ts index 585762d2cb98..d8e2253bd1d1 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-message-list-fetch.cron.command.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-message-list-fetch.cron.command.ts @@ -1,7 +1,6 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner } from 'nest-commander'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { MessagingMessageListFetchCronJob } from 'src/modules/messaging/message-import-manager/crons/jobs/messaging-message-list-fetch.cron.job'; @@ -15,7 +14,7 @@ const MESSAGING_MESSAGE_LIST_FETCH_CRON_PATTERN = '*/5 * * * *'; }) export class MessagingMessageListFetchCronCommand extends CommandRunner { constructor( - @Inject(MessageQueue.cronQueue) + @InjectMessageQueue(MessageQueue.cronQueue) private readonly messageQueueService: MessageQueueService, ) { super(); diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-messages-import.cron.command.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-messages-import.cron.command.ts index aae9df3723b2..206f2d47815b 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-messages-import.cron.command.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-messages-import.cron.command.ts @@ -1,7 +1,6 @@ -import { Inject } from '@nestjs/common'; - import { Command, CommandRunner } from 'nest-commander'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { MessagingMessagesImportCronJob } from 'src/modules/messaging/message-import-manager/crons/jobs/messaging-messages-import.cron.job'; @@ -12,7 +11,7 @@ import { MessagingMessagesImportCronJob } from 'src/modules/messaging/message-im }) export class MessagingMessagesImportCronCommand extends CommandRunner { constructor( - @Inject(MessageQueue.cronQueue) + @InjectMessageQueue(MessageQueue.cronQueue) private readonly messageQueueService: MessageQueueService, ) { super(); @@ -24,7 +23,7 @@ export class MessagingMessagesImportCronCommand extends CommandRunner { undefined, { repeat: { - every: 5000, + every: 30000, }, }, ); diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-ongoing-stale.cron.command.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-ongoing-stale.cron.command.ts new file mode 100644 index 000000000000..ed77d79555b8 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/commands/messaging-ongoing-stale.cron.command.ts @@ -0,0 +1,32 @@ +import { Command, CommandRunner } from 'nest-commander'; + +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { MessagingOngoingStaleCronJob } from 'src/modules/messaging/message-import-manager/crons/jobs/messaging-ongoing-stale.cron.job'; + +const MESSAGING_ONGOING_STALE_CRON_PATTERN = '0 * * * *'; + +@Command({ + name: 'cron:messaging:ongoing-stale', + description: + 'Starts a cron job to check for stale ongoing message imports and put them back to pending', +}) +export class MessagingOngoingStaleCronCommand extends CommandRunner { + constructor( + @InjectMessageQueue(MessageQueue.cronQueue) + private readonly messageQueueService: MessageQueueService, + ) { + super(); + } + + async run(): Promise<void> { + await this.messageQueueService.addCron<undefined>( + MessagingOngoingStaleCronJob.name, + undefined, + { + repeat: { pattern: MESSAGING_ONGOING_STALE_CRON_PATTERN }, + }, + ); + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-message-list-fetch.cron.job.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-message-list-fetch.cron.job.ts index 9a2cb54740df..0e8811b6b148 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-message-list-fetch.cron.job.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-message-list-fetch.cron.job.ts @@ -1,16 +1,12 @@ -import { Inject, Injectable, Logger } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository, In } from 'typeorm'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessageChannelSyncStage, @@ -20,36 +16,29 @@ import { MessagingMessageListFetchJobData, MessagingMessageListFetchJob, } from 'src/modules/messaging/message-import-manager/jobs/messaging-message-list-fetch.job'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { BillingService } from 'src/engine/core-modules/billing/billing.service'; -@Injectable() -export class MessagingMessageListFetchCronJob - implements MessageQueueJob<undefined> -{ +@Processor(MessageQueue.cronQueue) +export class MessagingMessageListFetchCronJob { private readonly logger = new Logger(MessagingMessageListFetchCronJob.name); constructor( - @InjectRepository(Workspace, 'core') - private readonly workspaceRepository: Repository<Workspace>, @InjectRepository(DataSourceEntity, 'metadata') private readonly dataSourceRepository: Repository<DataSourceEntity>, - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.messagingQueue) private readonly messageQueueService: MessageQueueService, @InjectObjectMetadataRepository(MessageChannelWorkspaceEntity) private readonly messageChannelRepository: MessageChannelRepository, - private readonly environmentService: EnvironmentService, + private readonly billingService: BillingService, ) {} + @Process(MessagingMessageListFetchCronJob.name) async handle(): Promise<void> { - const workspaceIds = ( - await this.workspaceRepository.find({ - where: this.environmentService.get('IS_BILLING_ENABLED') - ? { - subscriptionStatus: In(['active', 'trialing', 'past_due']), - } - : {}, - select: ['id'], - }) - ).map((workspace) => workspace.id); + const workspaceIds = + await this.billingService.getActiveSubscriptionWorkspaceIds(); const dataSources = await this.dataSourceRepository.find({ where: { diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-messages-import.cron.job.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-messages-import.cron.job.ts index 2faafc14a8c5..4be6ce2e1214 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-messages-import.cron.job.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-messages-import.cron.job.ts @@ -1,53 +1,44 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository, In } from 'typeorm'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { MessagingMessagesImportJobData, MessagingMessagesImportJob, } from 'src/modules/messaging/message-import-manager/jobs/messaging-messages-import.job'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { MessageChannelSyncStage, MessageChannelWorkspaceEntity, } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { BillingService } from 'src/engine/core-modules/billing/billing.service'; + +@Processor(MessageQueue.cronQueue) +export class MessagingMessagesImportCronJob { + private readonly logger = new Logger(MessagingMessagesImportCronJob.name); -@Injectable() -export class MessagingMessagesImportCronJob - implements MessageQueueJob<undefined> -{ constructor( - @InjectRepository(Workspace, 'core') - private readonly workspaceRepository: Repository<Workspace>, @InjectRepository(DataSourceEntity, 'metadata') private readonly dataSourceRepository: Repository<DataSourceEntity>, - private readonly environmentService: EnvironmentService, - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.messagingQueue) private readonly messageQueueService: MessageQueueService, @InjectObjectMetadataRepository(MessageChannelWorkspaceEntity) private readonly messageChannelRepository: MessageChannelRepository, + private readonly billingService: BillingService, ) {} + @Process(MessagingMessagesImportCronJob.name) async handle(): Promise<void> { - const workspaceIds = ( - await this.workspaceRepository.find({ - where: this.environmentService.get('IS_BILLING_ENABLED') - ? { - subscriptionStatus: In(['active', 'trialing', 'past_due']), - } - : {}, - select: ['id'], - }) - ).map((workspace) => workspace.id); + const workspaceIds = + await this.billingService.getActiveSubscriptionWorkspaceIds(); const dataSources = await this.dataSourceRepository.find({ where: { diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-ongoing-stale.cron.job.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-ongoing-stale.cron.job.ts new file mode 100644 index 000000000000..bc1bce31ce1e --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/crons/jobs/messaging-ongoing-stale.cron.job.ts @@ -0,0 +1,51 @@ +import { InjectRepository } from '@nestjs/typeorm'; + +import { Repository, In } from 'typeorm'; + +import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { + MessagingOngoingStaleJobData, + MessagingOngoingStaleJob, +} from 'src/modules/messaging/message-import-manager/jobs/messaging-ongoing-stale.job'; +import { BillingService } from 'src/engine/core-modules/billing/billing.service'; + +@Processor(MessageQueue.cronQueue) +export class MessagingOngoingStaleCronJob { + constructor( + @InjectRepository(DataSourceEntity, 'metadata') + private readonly dataSourceRepository: Repository<DataSourceEntity>, + @InjectMessageQueue(MessageQueue.messagingQueue) + private readonly messageQueueService: MessageQueueService, + private readonly billingService: BillingService, + ) {} + + @Process(MessagingOngoingStaleCronJob.name) + async handle(): Promise<void> { + const workspaceIds = + await this.billingService.getActiveSubscriptionWorkspaceIds(); + + const dataSources = await this.dataSourceRepository.find({ + where: { + workspaceId: In(workspaceIds), + }, + }); + + const workspaceIdsWithDataSources = new Set( + dataSources.map((dataSource) => dataSource.workspaceId), + ); + + for (const workspaceId of workspaceIdsWithDataSources) { + await this.messageQueueService.add<MessagingOngoingStaleJobData>( + MessagingOngoingStaleJob.name, + { + workspaceId, + }, + ); + } + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/constants/messaging-gmail-users-messages-get-batch-size.constant.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/constants/messaging-gmail-users-messages-get-batch-size.constant.ts index 81255f383742..fdf015b35a78 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/constants/messaging-gmail-users-messages-get-batch-size.constant.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/constants/messaging-gmail-users-messages-get-batch-size.constant.ts @@ -1 +1 @@ -export const MESSAGING_GMAIL_USERS_MESSAGES_GET_BATCH_SIZE = 20; +export const MESSAGING_GMAIL_USERS_MESSAGES_GET_BATCH_SIZE = 100; diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/messaging-gmail-driver.module.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/messaging-gmail-driver.module.ts index 380e4534efb1..1056c25cc362 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/messaging-gmail-driver.module.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/messaging-gmail-driver.module.ts @@ -2,10 +2,14 @@ import { Module } from '@nestjs/common'; import { TypeOrmModule } from '@nestjs/typeorm'; import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { FeatureFlagModule } from 'src/engine/core-modules/feature-flag/feature-flag.module'; import { EnvironmentModule } from 'src/engine/integrations/environment/environment.module'; import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { GoogleAPIRefreshAccessTokenModule } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.module'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { EmailAliasManagerModule } from 'src/modules/connected-account/email-alias-manager/email-alias-manager.module'; +import { OAuth2ClientManagerModule } from 'src/modules/connected-account/oauth2-client-manager/oauth2-client-manager.module'; +import { RefreshAccessTokenManagerModule } from 'src/modules/connected-account/refresh-access-token-manager/refresh-access-token-manager.module'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { MessagingCommonModule } from 'src/modules/messaging/common/messaging-common.module'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; @@ -17,10 +21,12 @@ import { MessagingGmailFullMessageListFetchService } from 'src/modules/messaging import { MessagingGmailHistoryService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-history.service'; import { MessagingGmailMessagesImportService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-messages-import.service'; import { MessagingGmailPartialMessageListFetchService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-partial-message-list-fetch.service'; +import { MessagingSaveMessagesAndEnqueueContactCreationService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-save-messages-and-enqueue-contact-creation.service'; +import { MessageParticipantManagerModule } from 'src/modules/messaging/message-participant-manager/message-participant-manager.module'; @Module({ imports: [ - GoogleAPIRefreshAccessTokenModule, + RefreshAccessTokenManagerModule, EnvironmentModule, ObjectMetadataRepositoryModule.forFeature([ ConnectedAccountWorkspaceEntity, @@ -30,6 +36,11 @@ import { MessagingGmailPartialMessageListFetchService } from 'src/modules/messag ]), MessagingCommonModule, TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), + OAuth2ClientManagerModule, + EmailAliasManagerModule, + FeatureFlagModule, + WorkspaceDataSourceModule, + MessageParticipantManagerModule, ], providers: [ MessagingGmailClientProvider, @@ -39,6 +50,7 @@ import { MessagingGmailPartialMessageListFetchService } from 'src/modules/messag MessagingGmailFullMessageListFetchService, MessagingGmailMessagesImportService, MessagingGmailFetchMessageIdsToExcludeService, + MessagingSaveMessagesAndEnqueueContactCreationService, ], exports: [ MessagingGmailClientProvider, diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/providers/messaging-gmail-client.provider.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/providers/messaging-gmail-client.provider.ts index 130e335621a1..834fedca46f7 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/providers/messaging-gmail-client.provider.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/providers/messaging-gmail-client.provider.ts @@ -1,16 +1,24 @@ import { Injectable } from '@nestjs/common'; -import { OAuth2Client } from 'google-auth-library'; import { gmail_v1, google } from 'googleapis'; -import { EnvironmentService } from 'src/engine/integrations/environment/environment.service'; +import { OAuth2ClientManagerService } from 'src/modules/connected-account/oauth2-client-manager/services/oauth2-client-manager.service'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; @Injectable() export class MessagingGmailClientProvider { - constructor(private readonly environmentService: EnvironmentService) {} - - public async getGmailClient(refreshToken: string): Promise<gmail_v1.Gmail> { - const oAuth2Client = await this.getOAuth2Client(refreshToken); + constructor( + private readonly oAuth2ClientManagerService: OAuth2ClientManagerService, + ) {} + + public async getGmailClient( + connectedAccount: Pick< + ConnectedAccountWorkspaceEntity, + 'provider' | 'refreshToken' + >, + ): Promise<gmail_v1.Gmail> { + const oAuth2Client = + await this.oAuth2ClientManagerService.getOAuth2Client(connectedAccount); const gmailClient = google.gmail({ version: 'v1', @@ -19,22 +27,4 @@ export class MessagingGmailClientProvider { return gmailClient; } - - private async getOAuth2Client(refreshToken: string): Promise<OAuth2Client> { - const gmailClientId = this.environmentService.get('AUTH_GOOGLE_CLIENT_ID'); - const gmailClientSecret = this.environmentService.get( - 'AUTH_GOOGLE_CLIENT_SECRET', - ); - - const oAuth2Client = new google.auth.OAuth2( - gmailClientId, - gmailClientSecret, - ); - - oAuth2Client.setCredentials({ - refresh_token: refreshToken, - }); - - return oAuth2Client; - } } diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-fetch-messages-by-batches.service.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-fetch-messages-by-batches.service.ts index 1e0c0c6e979c..df3c96bb3e0a 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-fetch-messages-by-batches.service.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-fetch-messages-by-batches.service.ts @@ -7,12 +7,8 @@ import { gmail_v1 } from 'googleapis'; import { assert, assertNotNull } from 'src/utils/assert'; import { GmailMessage } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message'; -import { MessageQuery } from 'src/modules/messaging/message-import-manager/types/message-or-thread-query'; import { formatAddressObjectAsParticipants } from 'src/modules/messaging/message-import-manager/utils/format-address-object-as-participants.util'; import { MessagingFetchByBatchesService } from 'src/modules/messaging/common/services/messaging-fetch-by-batch.service'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; @Injectable() export class MessagingGmailFetchMessagesByBatchesService { @@ -22,46 +18,34 @@ export class MessagingGmailFetchMessagesByBatchesService { constructor( private readonly fetchByBatchesService: MessagingFetchByBatchesService, - @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) - private readonly connectedAccountRepository: ConnectedAccountRepository, ) {} async fetchAllMessages( - queries: MessageQuery[], + messageIds: string[], + accessToken: string, connectedAccountId: string, workspaceId: string, ): Promise<GmailMessage[]> { let startTime = Date.now(); - const connectedAccount = await this.connectedAccountRepository.getById( - connectedAccountId, - workspaceId, - ); - - if (!connectedAccount) { - throw new Error( - `Connected account ${connectedAccountId} not found in workspace ${workspaceId}`, + const { messageIdsByBatch, batchResponses } = + await this.fetchByBatchesService.fetchAllByBatches( + messageIds, + accessToken, + 'batch_gmail_messages', ); - } - - const accessToken = connectedAccount.accessToken; - - const batchResponses = await this.fetchByBatchesService.fetchAllByBatches( - queries, - accessToken, - 'batch_gmail_messages', - ); let endTime = Date.now(); this.logger.log( `Messaging import for workspace ${workspaceId} and account ${connectedAccountId} fetching ${ - queries.length + messageIds.length } messages in ${endTime - startTime}ms`, ); startTime = Date.now(); const formattedResponse = this.formatBatchResponsesAsGmailMessages( + messageIdsByBatch, batchResponses, workspaceId, connectedAccountId, @@ -71,7 +55,7 @@ export class MessagingGmailFetchMessagesByBatchesService { this.logger.log( `Messaging import for workspace ${workspaceId} and account ${connectedAccountId} formatting ${ - queries.length + messageIds.length } messages in ${endTime - startTime}ms`, ); @@ -79,6 +63,7 @@ export class MessagingGmailFetchMessagesByBatchesService { } private formatBatchResponseAsGmailMessage( + messageIds: string[], responseCollection: AxiosResponse<any, any>, workspaceId: string, connectedAccountId: string, @@ -90,94 +75,92 @@ export class MessagingGmailFetchMessagesByBatchesService { return str.replace(/\0/g, ''); }; - const formattedResponse = parsedResponses.map( - (response): GmailMessage | null => { - if ('error' in response) { - if (response.error.code === 404) { - return null; - } - - throw response.error; - } - - const { - historyId, - id, - threadId, - internalDate, - subject, - from, - to, - cc, - bcc, - headerMessageId, - text, - attachments, - deliveredTo, - } = this.parseGmailMessage(response); - - if (!from) { - this.logger.log( - `From value is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, - ); - - return null; - } - - if (!to && !deliveredTo && !bcc && !cc) { - this.logger.log( - `To, Delivered-To, Bcc or Cc value is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, - ); - - return null; - } - - if (!headerMessageId) { - this.logger.log( - `Message-ID is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, - ); - + const formattedResponse = parsedResponses.map((response, index) => { + if ('error' in response) { + if (response.error.code === 404) { return null; } - if (!threadId) { - this.logger.log( - `Thread Id is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, - ); - - return null; - } - - const participants = [ - ...formatAddressObjectAsParticipants(from, 'from'), - ...formatAddressObjectAsParticipants(to ?? deliveredTo, 'to'), - ...formatAddressObjectAsParticipants(cc, 'cc'), - ...formatAddressObjectAsParticipants(bcc, 'bcc'), - ]; - - let textWithoutReplyQuotations = text; - - if (text) { - textWithoutReplyQuotations = planer.extractFrom(text, 'text/plain'); - } - - const messageFromGmail: GmailMessage = { - historyId, - externalId: id, - headerMessageId, - subject: subject || '', - messageThreadExternalId: threadId, - internalDate, - fromHandle: from[0].address || '', - fromDisplayName: from[0].name || '', - participants, - text: sanitizeString(textWithoutReplyQuotations || ''), - attachments, - }; - - return messageFromGmail; - }, - ); + throw { ...response.error, messageId: messageIds[index] }; + } + + const { + historyId, + id, + threadId, + internalDate, + subject, + from, + to, + cc, + bcc, + headerMessageId, + text, + attachments, + deliveredTo, + } = this.parseGmailMessage(response); + + if (!from) { + this.logger.log( + `From value is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, + ); + + return null; + } + + if (!to && !deliveredTo && !bcc && !cc) { + this.logger.log( + `To, Delivered-To, Bcc or Cc value is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, + ); + + return null; + } + + if (!headerMessageId) { + this.logger.log( + `Message-ID is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, + ); + + return null; + } + + if (!threadId) { + this.logger.log( + `Thread Id is missing while importing message #${id} in workspace ${workspaceId} and account ${connectedAccountId}`, + ); + + return null; + } + + const participants = [ + ...formatAddressObjectAsParticipants(from, 'from'), + ...formatAddressObjectAsParticipants(to ?? deliveredTo, 'to'), + ...formatAddressObjectAsParticipants(cc, 'cc'), + ...formatAddressObjectAsParticipants(bcc, 'bcc'), + ]; + + let textWithoutReplyQuotations = text; + + if (text) { + textWithoutReplyQuotations = planer.extractFrom(text, 'text/plain'); + } + + const messageFromGmail: GmailMessage = { + historyId, + externalId: id, + headerMessageId, + subject: subject || '', + messageThreadExternalId: threadId, + internalDate, + fromHandle: from[0].address || '', + fromDisplayName: from[0].name || '', + participants, + text: sanitizeString(textWithoutReplyQuotations || ''), + attachments, + }; + + return messageFromGmail; + }); const filteredMessages = formattedResponse.filter((message) => assertNotNull(message), @@ -187,12 +170,14 @@ export class MessagingGmailFetchMessagesByBatchesService { } private formatBatchResponsesAsGmailMessages( + messageIdsByBatch: string[][], batchResponses: AxiosResponse<any, any>[], workspaceId: string, connectedAccountId: string, ): GmailMessage[] { - const messageBatches = batchResponses.map((response) => { + const messageBatches = batchResponses.map((response, index) => { return this.formatBatchResponseAsGmailMessage( + messageIdsByBatch[index], response, workspaceId, connectedAccountId, diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-full-message-list-fetch.service.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-full-message-list-fetch.service.ts index 079baf1ac9cf..12eeda48a37f 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-full-message-list-fetch.service.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-full-message-list-fetch.service.ts @@ -9,7 +9,6 @@ import { InjectCacheStorage } from 'src/engine/integrations/cache-storage/decora import { CacheStorageNamespace } from 'src/engine/integrations/cache-storage/types/cache-storage-namespace.enum'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; @@ -23,8 +22,6 @@ import { MessagingChannelSyncStatusService } from 'src/modules/messaging/common/ import { MessagingGmailClientProvider } from 'src/modules/messaging/message-import-manager/drivers/gmail/providers/messaging-gmail-client.provider'; import { MESSAGING_GMAIL_USERS_MESSAGES_LIST_MAX_RESULT } from 'src/modules/messaging/message-import-manager/drivers/gmail/constants/messaging-gmail-users-messages-list-max-result.constant'; import { MESSAGING_GMAIL_EXCLUDED_CATEGORIES } from 'src/modules/messaging/message-import-manager/drivers/gmail/constants/messaging-gmail-excluded-categories'; -import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service'; -import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; @Injectable() export class MessagingGmailFullMessageListFetchService { @@ -42,16 +39,13 @@ export class MessagingGmailFullMessageListFetchService { MessageChannelMessageAssociationWorkspaceEntity, ) private readonly messageChannelMessageAssociationRepository: MessageChannelMessageAssociationRepository, - @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) - private readonly connectedAccountRepository: ConnectedAccountRepository, private readonly messagingChannelSyncStatusService: MessagingChannelSyncStatusService, private readonly gmailErrorHandlingService: MessagingErrorHandlingService, - private readonly googleAPIsRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService, ) {} public async processMessageListFetch( - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, + connectedAccount: ConnectedAccountWorkspaceEntity, workspaceId: string, ) { await this.messagingChannelSyncStatusService.markAsMessagesListFetchOngoing( @@ -59,27 +53,8 @@ export class MessagingGmailFullMessageListFetchService { workspaceId, ); - await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken( - workspaceId, - connectedAccount.id, - ); - - const refreshedConnectedAccount = - await this.connectedAccountRepository.getById( - connectedAccount.id, - workspaceId, - ); - - if (!refreshedConnectedAccount) { - throw new Error( - `Connected account ${connectedAccount.id} not found in workspace ${workspaceId}`, - ); - } - const gmailClient: gmail_v1.Gmail = - await this.gmailClientProvider.getGmailClient( - refreshedConnectedAccount.refreshToken, - ); + await this.gmailClientProvider.getGmailClient(connectedAccount); const { error: gmailError } = await this.fetchAllMessageIdsFromGmailAndStoreInCache( diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-messages-import.service.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-messages-import.service.ts index 8b6e462426ac..4d87f09b358d 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-messages-import.service.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-messages-import.service.ts @@ -1,27 +1,29 @@ import { Injectable, Logger } from '@nestjs/common'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { FeatureFlagKeys } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { IsFeatureEnabledService } from 'src/engine/core-modules/feature-flag/services/is-feature-enabled.service'; +import { CacheStorageService } from 'src/engine/integrations/cache-storage/cache-storage.service'; import { InjectCacheStorage } from 'src/engine/integrations/cache-storage/decorators/cache-storage.decorator'; import { CacheStorageNamespace } from 'src/engine/integrations/cache-storage/types/cache-storage-namespace.enum'; -import { CacheStorageService } from 'src/engine/integrations/cache-storage/cache-storage.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; -import { GoogleAPIRefreshAccessTokenService } from 'src/modules/connected-account/services/google-api-refresh-access-token/google-api-refresh-access-token.service'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { BlocklistRepository } from 'src/modules/connected-account/repositories/blocklist.repository'; +import { BlocklistRepository } from 'src/modules/blocklist/repositories/blocklist.repository'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { EmailAliasManagerService } from 'src/modules/connected-account/email-alias-manager/services/email-alias-manager.service'; +import { RefreshAccessTokenService } from 'src/modules/connected-account/refresh-access-token-manager/services/refresh-access-token.service'; +import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; +import { MessagingChannelSyncStatusService } from 'src/modules/messaging/common/services/messaging-channel-sync-status.service'; +import { MessagingErrorHandlingService } from 'src/modules/messaging/common/services/messaging-error-handling.service'; import { MessagingTelemetryService } from 'src/modules/messaging/common/services/messaging-telemetry.service'; import { - MessageChannelWorkspaceEntity, MessageChannelSyncStage, + MessageChannelWorkspaceEntity, } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; -import { createQueriesFromMessageIds } from 'src/modules/messaging/message-import-manager/utils/create-queries-from-message-ids.util'; -import { filterEmails } from 'src/modules/messaging/message-import-manager/utils/filter-emails.util'; -import { MessagingChannelSyncStatusService } from 'src/modules/messaging/common/services/messaging-channel-sync-status.service'; import { MESSAGING_GMAIL_USERS_MESSAGES_GET_BATCH_SIZE } from 'src/modules/messaging/message-import-manager/drivers/gmail/constants/messaging-gmail-users-messages-get-batch-size.constant'; import { MessagingGmailFetchMessagesByBatchesService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-fetch-messages-by-batches.service'; -import { MessagingErrorHandlingService } from 'src/modules/messaging/common/services/messaging-error-handling.service'; -import { MessagingSaveMessagesAndEnqueueContactCreationService } from 'src/modules/messaging/common/services/messaging-save-messages-and-enqueue-contact-creation.service'; -import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; +import { MessagingSaveMessagesAndEnqueueContactCreationService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-save-messages-and-enqueue-contact-creation.service'; +import { filterEmails } from 'src/modules/messaging/message-import-manager/utils/filter-emails.util'; @Injectable() export class MessagingGmailMessagesImportService { @@ -36,17 +38,21 @@ export class MessagingGmailMessagesImportService { private readonly messagingChannelSyncStatusService: MessagingChannelSyncStatusService, private readonly saveMessagesAndEnqueueContactCreationService: MessagingSaveMessagesAndEnqueueContactCreationService, private readonly gmailErrorHandlingService: MessagingErrorHandlingService, - private readonly googleAPIsRefreshAccessTokenService: GoogleAPIRefreshAccessTokenService, + private readonly refreshAccessTokenService: RefreshAccessTokenService, private readonly messagingTelemetryService: MessagingTelemetryService, @InjectObjectMetadataRepository(BlocklistWorkspaceEntity) private readonly blocklistRepository: BlocklistRepository, @InjectObjectMetadataRepository(MessageChannelWorkspaceEntity) private readonly messageChannelRepository: MessageChannelRepository, + private readonly emailAliasManagerService: EmailAliasManagerService, + private readonly isFeatureEnabledService: IsFeatureEnabledService, + @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) + private readonly connectedAccountRepository: ConnectedAccountRepository, ) {} async processMessageBatchImport( - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, + connectedAccount: ConnectedAccountWorkspaceEntity, workspaceId: string, ) { if ( @@ -72,10 +78,59 @@ export class MessagingGmailMessagesImportService { workspaceId, ); - await this.googleAPIsRefreshAccessTokenService.refreshAndSaveAccessToken( - workspaceId, - connectedAccount.id, - ); + let accessToken: string; + + try { + accessToken = + await this.refreshAccessTokenService.refreshAndSaveAccessToken( + connectedAccount, + workspaceId, + ); + } catch (error) { + await this.messagingTelemetryService.track({ + eventName: `refresh_token.error.insufficient_permissions`, + workspaceId, + connectedAccountId: messageChannel.connectedAccountId, + messageChannelId: messageChannel.id, + message: `${error.code}: ${error.reason}`, + }); + + await this.messagingChannelSyncStatusService.markAsFailedInsufficientPermissionsAndFlushMessagesToImport( + messageChannel.id, + workspaceId, + ); + + await this.connectedAccountRepository.updateAuthFailedAt( + messageChannel.connectedAccountId, + workspaceId, + ); + + return; + } + + if ( + await this.isFeatureEnabledService.isFeatureEnabled( + FeatureFlagKeys.IsMessagingAliasFetchingEnabled, + workspaceId, + ) + ) { + try { + await this.emailAliasManagerService.refreshHandleAliases( + connectedAccount, + workspaceId, + ); + } catch (error) { + await this.gmailErrorHandlingService.handleGmailError( + { + code: error.code, + reason: error.message, + }, + 'messages-import', + messageChannel, + workspaceId, + ); + } + } const messageIdsToFetch = (await this.cacheStorage.setPop( @@ -95,12 +150,11 @@ export class MessagingGmailMessagesImportService { ); } - const messageQueries = createQueriesFromMessageIds(messageIdsToFetch); - try { const allMessages = await this.fetchMessagesByBatchesService.fetchAllMessages( - messageQueries, + messageIdsToFetch, + accessToken, connectedAccount.id, workspaceId, ); @@ -152,6 +206,14 @@ export class MessagingGmailMessagesImportService { workspaceId, ); } catch (error) { + this.logger.log( + `Messaging import for messageId ${ + error.messageId + }, workspace ${workspaceId} and connected account ${ + connectedAccount.id + } failed with error: ${JSON.stringify(error)}`, + ); + await this.cacheStorage.setAdd( `messages-to-import:${workspaceId}:gmail:${messageChannel.id}`, messageIdsToFetch, @@ -180,7 +242,7 @@ export class MessagingGmailMessagesImportService { } private async trackMessageImportCompleted( - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, workspaceId: string, ) { await this.messagingTelemetryService.track({ diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-partial-message-list-fetch.service.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-partial-message-list-fetch.service.ts index 654150d5b1da..069180e5d7e1 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-partial-message-list-fetch.service.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-partial-message-list-fetch.service.ts @@ -7,7 +7,6 @@ import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/s import { CacheStorageService } from 'src/engine/integrations/cache-storage/cache-storage.service'; import { InjectCacheStorage } from 'src/engine/integrations/cache-storage/decorators/cache-storage.decorator'; import { CacheStorageNamespace } from 'src/engine/integrations/cache-storage/types/cache-storage-namespace.enum'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { MessageChannelMessageAssociationRepository } from 'src/modules/messaging/common/repositories/message-channel-message-association.repository'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessageChannelMessageAssociationWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel-message-association.workspace-entity'; @@ -41,8 +40,8 @@ export class MessagingGmailPartialMessageListFetchService { ) {} public async processMessageListFetch( - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, + connectedAccount: ConnectedAccountWorkspaceEntity, workspaceId: string, ): Promise<void> { await this.messagingChannelSyncStatusService.markAsMessagesListFetchOngoing( @@ -53,9 +52,7 @@ export class MessagingGmailPartialMessageListFetchService { const lastSyncHistoryId = messageChannel.syncCursor; const gmailClient: gmail_v1.Gmail = - await this.gmailClientProvider.getGmailClient( - connectedAccount.refreshToken, - ); + await this.gmailClientProvider.getGmailClient(connectedAccount); const { history, historyId, error } = await this.gmailGetHistoryService.getHistory( diff --git a/packages/twenty-server/src/modules/messaging/common/services/messaging-save-messages-and-enqueue-contact-creation.service.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-save-messages-and-enqueue-contact-creation.service.ts similarity index 57% rename from packages/twenty-server/src/modules/messaging/common/services/messaging-save-messages-and-enqueue-contact-creation.service.ts rename to packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-save-messages-and-enqueue-contact-creation.service.ts index 163c4427b7c4..f408e1c3d548 100644 --- a/packages/twenty-server/src/modules/messaging/common/services/messaging-save-messages-and-enqueue-contact-creation.service.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-save-messages-and-enqueue-contact-creation.service.ts @@ -1,37 +1,39 @@ -import { Injectable, Inject } from '@nestjs/common'; -import { InjectRepository } from '@nestjs/typeorm'; +import { Injectable } from '@nestjs/common'; import { EventEmitter2 } from '@nestjs/event-emitter'; +import { InjectRepository } from '@nestjs/typeorm'; import { EntityManager, Repository } from 'typeorm'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { - CreateCompanyAndContactJobData, CreateCompanyAndContactJob, -} from 'src/modules/connected-account/auto-companies-and-contacts-creation/jobs/create-company-and-contact.job'; + CreateCompanyAndContactJobData, +} from 'src/modules/contact-creation-manager/jobs/create-company-and-contact.job'; +import { MessagingMessageService } from 'src/modules/messaging/common/services/messaging-message.service'; import { - FeatureFlagEntity, - FeatureFlagKeys, -} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; + MessageChannelContactAutoCreationPolicy, + MessageChannelWorkspaceEntity, +} from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; import { GmailMessage, Participant, ParticipantWithMessageId, } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message'; -import { MessagingMessageService } from 'src/modules/messaging/common/services/messaging-message.service'; -import { MessagingMessageParticipantService } from 'src/modules/messaging/common/services/messaging-message-participant.service'; -import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; +import { MessagingMessageParticipantService } from 'src/modules/messaging/message-participant-manager/services/messaging-message-participant.service'; +import { isGroupEmail } from 'src/utils/is-group-email'; +import { isWorkEmail } from 'src/utils/is-work-email'; @Injectable() export class MessagingSaveMessagesAndEnqueueContactCreationService { constructor( private readonly workspaceDataSourceService: WorkspaceDataSourceService, - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.contactCreationQueue) private readonly messageQueueService: MessageQueueService, private readonly messageService: MessagingMessageService, private readonly messageParticipantService: MessagingMessageParticipantService, @@ -42,8 +44,8 @@ export class MessagingSaveMessagesAndEnqueueContactCreationService { async saveMessagesAndEnqueueContactCreationJob( messagesToSave: GmailMessage[], - messageChannel: ObjectRecord<MessageChannelWorkspaceEntity>, - connectedAccount: ObjectRecord<ConnectedAccountWorkspaceEntity>, + messageChannel: MessageChannelWorkspaceEntity, + connectedAccount: ConnectedAccountWorkspaceEntity, workspaceId: string, ) { const workspaceDataSource = @@ -51,18 +53,9 @@ export class MessagingSaveMessagesAndEnqueueContactCreationService { workspaceId, ); - const isContactCreationForSentAndReceivedEmailsEnabledFeatureFlag = - await this.featureFlagRepository.findOneBy({ - workspaceId: workspaceId, - key: FeatureFlagKeys.IsContactCreationForSentAndReceivedEmailsEnabled, - value: true, - }); + const handleAliases = connectedAccount.handleAliases?.split(',') || []; - const isContactCreationForSentAndReceivedEmailsEnabled = - isContactCreationForSentAndReceivedEmailsEnabledFeatureFlag?.value; - - let savedMessageParticipants: ObjectRecord<MessageParticipantWorkspaceEntity>[] = - []; + let savedMessageParticipants: MessageParticipantWorkspaceEntity[] = []; const participantsWithMessageId = await workspaceDataSource?.transaction( async (transactionManager: EntityManager) => { @@ -81,15 +74,43 @@ export class MessagingSaveMessagesAndEnqueueContactCreationService { const messageId = messageExternalIdsAndIdsMap.get(message.externalId); return messageId - ? message.participants.map((participant: Participant) => ({ - ...participant, - messageId, - shouldCreateContact: - messageChannel.isContactAutoCreationEnabled && - (isContactCreationForSentAndReceivedEmailsEnabled || - message.participants.find((p) => p.role === 'from') - ?.handle === connectedAccount.handle), - })) + ? message.participants.map((participant: Participant) => { + const fromHandle = + message.participants.find((p) => p.role === 'from')?.handle || + ''; + + const isMessageSentByConnectedAccount = + handleAliases.includes(fromHandle) || + fromHandle === connectedAccount.handle; + + const isParticipantConnectedAccount = + handleAliases.includes(participant.handle) || + participant.handle === connectedAccount.handle; + + const isExcludedByNonProfessionalEmails = + messageChannel.excludeNonProfessionalEmails && + !isWorkEmail(participant.handle); + + const isExcludedByGroupEmails = + messageChannel.excludeGroupEmails && + isGroupEmail(participant.handle); + + const shouldCreateContact = + !isParticipantConnectedAccount && + !isExcludedByNonProfessionalEmails && + !isExcludedByGroupEmails && + (messageChannel.contactAutoCreationPolicy === + MessageChannelContactAutoCreationPolicy.SENT_AND_RECEIVED || + (messageChannel.contactAutoCreationPolicy === + MessageChannelContactAutoCreationPolicy.SENT && + isMessageSentByConnectedAccount)); + + return { + ...participant, + messageId, + shouldCreateContact, + }; + }) : []; }); @@ -106,7 +127,7 @@ export class MessagingSaveMessagesAndEnqueueContactCreationService { this.eventEmitter.emit(`messageParticipant.matched`, { workspaceId, - userId: connectedAccount.accountOwnerId, + workspaceMemberId: connectedAccount.accountOwnerId, messageParticipants: savedMessageParticipants, }); diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-add-single-message-to-cache-for-import.job.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-add-single-message-to-cache-for-import.job.ts new file mode 100644 index 000000000000..d81bba9cd096 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-add-single-message-to-cache-for-import.job.ts @@ -0,0 +1,32 @@ +import { CacheStorageService } from 'src/engine/integrations/cache-storage/cache-storage.service'; +import { InjectCacheStorage } from 'src/engine/integrations/cache-storage/decorators/cache-storage.decorator'; +import { CacheStorageNamespace } from 'src/engine/integrations/cache-storage/types/cache-storage-namespace.enum'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; + +export type MessagingAddSingleMessageToCacheForImportJobData = { + messageExternalId: string; + messageChannelId: string; + workspaceId: string; +}; + +@Processor(MessageQueue.messagingQueue) +export class MessagingAddSingleMessageToCacheForImportJob { + constructor( + @InjectCacheStorage(CacheStorageNamespace.Messaging) + private readonly cacheStorage: CacheStorageService, + ) {} + + @Process(MessagingAddSingleMessageToCacheForImportJob.name) + async handle( + data: MessagingAddSingleMessageToCacheForImportJobData, + ): Promise<void> { + const { messageExternalId, messageChannelId, workspaceId } = data; + + await this.cacheStorage.setAdd( + `messages-to-import:${workspaceId}:gmail:${messageChannelId}`, + [messageExternalId], + ); + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-clean-cache.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-clean-cache.ts new file mode 100644 index 000000000000..8c2bc681f240 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-clean-cache.ts @@ -0,0 +1,38 @@ +import { Logger } from '@nestjs/common'; + +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { InjectCacheStorage } from 'src/engine/integrations/cache-storage/decorators/cache-storage.decorator'; +import { CacheStorageNamespace } from 'src/engine/integrations/cache-storage/types/cache-storage-namespace.enum'; +import { CacheStorageService } from 'src/engine/integrations/cache-storage/cache-storage.service'; + +export type MessagingCleanCacheJobData = { + workspaceId: string; + messageChannelId: string; +}; + +@Processor(MessageQueue.messagingQueue) +export class MessagingCleanCacheJob { + private readonly logger = new Logger(MessagingCleanCacheJob.name); + + constructor( + @InjectCacheStorage(CacheStorageNamespace.Messaging) + private readonly cacheStorage: CacheStorageService, + ) {} + + @Process(MessagingCleanCacheJob.name) + async handle(data: MessagingCleanCacheJobData): Promise<void> { + this.logger.log( + `Deleting message channel ${data.messageChannelId} associated cache in workspace ${data.workspaceId}`, + ); + + await this.cacheStorage.del( + `messages-to-import:${data.workspaceId}:gmail:${data.messageChannelId}`, + ); + + this.logger.log( + `Deleted message channel ${data.messageChannelId} associated cache in workspace ${data.workspaceId}`, + ); + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-message-list-fetch.job.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-message-list-fetch.job.ts index 4cd4b40e5651..2a468fa493dd 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-message-list-fetch.job.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-message-list-fetch.job.ts @@ -1,7 +1,8 @@ -import { Injectable, Logger } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { Logger, Scope } from '@nestjs/common'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; @@ -13,17 +14,18 @@ import { } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { MessagingGmailFullMessageListFetchService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-full-message-list-fetch.service'; import { MessagingGmailPartialMessageListFetchService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-partial-message-list-fetch.service'; -import { isThrottled } from 'src/modules/messaging/message-import-manager/drivers/gmail/utils/is-throttled'; +import { isThrottled } from 'src/modules/connected-account/utils/is-throttled'; export type MessagingMessageListFetchJobData = { messageChannelId: string; workspaceId: string; }; -@Injectable() -export class MessagingMessageListFetchJob - implements MessageQueueJob<MessagingMessageListFetchJobData> -{ +@Processor({ + queueName: MessageQueue.messagingQueue, + scope: Scope.REQUEST, +}) +export class MessagingMessageListFetchJob { private readonly logger = new Logger(MessagingMessageListFetchJob.name); constructor( @@ -36,6 +38,7 @@ export class MessagingMessageListFetchJob private readonly messagingTelemetryService: MessagingTelemetryService, ) {} + @Process(MessagingMessageListFetchJob.name) async handle(data: MessagingMessageListFetchJobData): Promise<void> { const { messageChannelId, workspaceId } = data; diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-messages-import.job.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-messages-import.job.ts index 246f5f8b1f5e..b4cc6ae0b87d 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-messages-import.job.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-messages-import.job.ts @@ -1,7 +1,8 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; +import { Scope } from '@nestjs/common'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; @@ -12,17 +13,18 @@ import { MessageChannelWorkspaceEntity, } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { MessagingGmailMessagesImportService } from 'src/modules/messaging/message-import-manager/drivers/gmail/services/messaging-gmail-messages-import.service'; -import { isThrottled } from 'src/modules/messaging/message-import-manager/drivers/gmail/utils/is-throttled'; +import { isThrottled } from 'src/modules/connected-account/utils/is-throttled'; export type MessagingMessagesImportJobData = { messageChannelId: string; workspaceId: string; }; -@Injectable() -export class MessagingMessagesImportJob - implements MessageQueueJob<MessagingMessagesImportJobData> -{ +@Processor({ + queueName: MessageQueue.messagingQueue, + scope: Scope.REQUEST, +}) +export class MessagingMessagesImportJob { constructor( @InjectObjectMetadataRepository(ConnectedAccountWorkspaceEntity) private readonly connectedAccountRepository: ConnectedAccountRepository, @@ -32,6 +34,7 @@ export class MessagingMessagesImportJob private readonly messagingTelemetryService: MessagingTelemetryService, ) {} + @Process(MessagingMessagesImportJob.name) async handle(data: MessagingMessagesImportJobData): Promise<void> { const { messageChannelId, workspaceId } = data; diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-ongoing-stale.job.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-ongoing-stale.job.ts new file mode 100644 index 000000000000..0eb3c2aa12f8 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/jobs/messaging-ongoing-stale.job.ts @@ -0,0 +1,74 @@ +import { Logger, Scope } from '@nestjs/common'; + +import { In } from 'typeorm'; + +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { + MessageChannelSyncStage, + MessageChannelWorkspaceEntity, +} from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { isSyncStale } from 'src/modules/messaging/message-import-manager/utils/is-sync-stale.util'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { MessagingChannelSyncStatusService } from 'src/modules/messaging/common/services/messaging-channel-sync-status.service'; + +export type MessagingOngoingStaleJobData = { + workspaceId: string; +}; + +@Processor({ + queueName: MessageQueue.messagingQueue, + scope: Scope.REQUEST, +}) +export class MessagingOngoingStaleJob { + private readonly logger = new Logger(MessagingOngoingStaleJob.name); + constructor( + @InjectWorkspaceRepository(MessageChannelWorkspaceEntity) + private readonly messageChannelRepository: WorkspaceRepository<MessageChannelWorkspaceEntity>, + private readonly messagingChannelSyncStatusService: MessagingChannelSyncStatusService, + ) {} + + @Process(MessagingOngoingStaleJob.name) + async handle(data: MessagingOngoingStaleJobData): Promise<void> { + const { workspaceId } = data; + + const messageChannels = await this.messageChannelRepository.find({ + where: { + syncStage: In([ + MessageChannelSyncStage.MESSAGES_IMPORT_ONGOING, + MessageChannelSyncStage.MESSAGE_LIST_FETCH_ONGOING, + ]), + }, + }); + + for (const messageChannel of messageChannels) { + if ( + messageChannel.syncStageStartedAt && + isSyncStale(messageChannel.syncStageStartedAt) + ) { + this.logger.log( + `Sync for message channel ${messageChannel.id} and workspace ${workspaceId} is stale. Setting sync stage to MESSAGES_IMPORT_PENDING`, + ); + + switch (messageChannel.syncStage) { + case MessageChannelSyncStage.MESSAGE_LIST_FETCH_ONGOING: + await this.messagingChannelSyncStatusService.schedulePartialMessageListFetch( + messageChannel.id, + workspaceId, + ); + break; + case MessageChannelSyncStage.MESSAGES_IMPORT_ONGOING: + await this.messagingChannelSyncStatusService.scheduleMessagesImport( + messageChannel.id, + workspaceId, + ); + break; + default: + break; + } + } + } + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/listeners/messaging-import-manager-message-channel.listener.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/listeners/messaging-import-manager-message-channel.listener.ts new file mode 100644 index 000000000000..d3f00c95fc42 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/listeners/messaging-import-manager-message-channel.listener.ts @@ -0,0 +1,33 @@ +import { Injectable } from '@nestjs/common'; +import { OnEvent } from '@nestjs/event-emitter'; + +import { ObjectRecordDeleteEvent } from 'src/engine/integrations/event-emitter/types/object-record-delete.event'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { + MessagingCleanCacheJob, + MessagingCleanCacheJobData, +} from 'src/modules/messaging/message-import-manager/jobs/messaging-clean-cache'; + +@Injectable() +export class MessagingMessageImportManagerMessageChannelListener { + constructor( + @InjectMessageQueue(MessageQueue.messagingQueue) + private readonly messageQueueService: MessageQueueService, + ) {} + + @OnEvent('messageChannel.deleted') + async handleDeletedEvent( + payload: ObjectRecordDeleteEvent<MessageChannelWorkspaceEntity>, + ) { + await this.messageQueueService.add<MessagingCleanCacheJobData>( + MessagingCleanCacheJob.name, + { + workspaceId: payload.workspaceId, + messageChannelId: payload.recordId, + }, + ); + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/messaging-import-manager.module.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/messaging-import-manager.module.ts index b0b6362c0f35..b2f951bc9789 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/messaging-import-manager.module.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/messaging-import-manager.module.ts @@ -3,41 +3,50 @@ import { TypeOrmModule } from '@nestjs/typeorm'; import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; import { MessagingCommonModule } from 'src/modules/messaging/common/messaging-common.module'; +import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { MessagingSingleMessageImportCommand } from 'src/modules/messaging/message-import-manager/commands/messaging-single-message-import.command'; import { MessagingMessageListFetchCronCommand } from 'src/modules/messaging/message-import-manager/crons/commands/messaging-message-list-fetch.cron.command'; import { MessagingMessagesImportCronCommand } from 'src/modules/messaging/message-import-manager/crons/commands/messaging-messages-import.cron.command'; +import { MessagingOngoingStaleCronCommand } from 'src/modules/messaging/message-import-manager/crons/commands/messaging-ongoing-stale.cron.command'; import { MessagingMessageListFetchCronJob } from 'src/modules/messaging/message-import-manager/crons/jobs/messaging-message-list-fetch.cron.job'; import { MessagingMessagesImportCronJob } from 'src/modules/messaging/message-import-manager/crons/jobs/messaging-messages-import.cron.job'; +import { MessagingOngoingStaleCronJob } from 'src/modules/messaging/message-import-manager/crons/jobs/messaging-ongoing-stale.cron.job'; import { MessagingGmailDriverModule } from 'src/modules/messaging/message-import-manager/drivers/gmail/messaging-gmail-driver.module'; +import { MessagingAddSingleMessageToCacheForImportJob } from 'src/modules/messaging/message-import-manager/jobs/messaging-add-single-message-to-cache-for-import.job'; +import { MessagingCleanCacheJob } from 'src/modules/messaging/message-import-manager/jobs/messaging-clean-cache'; import { MessagingMessageListFetchJob } from 'src/modules/messaging/message-import-manager/jobs/messaging-message-list-fetch.job'; import { MessagingMessagesImportJob } from 'src/modules/messaging/message-import-manager/jobs/messaging-messages-import.job'; +import { MessagingOngoingStaleJob } from 'src/modules/messaging/message-import-manager/jobs/messaging-ongoing-stale.job'; +import { MessagingMessageImportManagerMessageChannelListener } from 'src/modules/messaging/message-import-manager/listeners/messaging-import-manager-message-channel.listener'; +import { BillingModule } from 'src/engine/core-modules/billing/billing.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; @Module({ imports: [ + WorkspaceDataSourceModule, MessagingGmailDriverModule, MessagingCommonModule, TypeOrmModule.forFeature([Workspace], 'core'), TypeOrmModule.forFeature([DataSourceEntity], 'metadata'), + TwentyORMModule.forFeature([MessageChannelWorkspaceEntity]), + BillingModule, ], providers: [ MessagingMessageListFetchCronCommand, MessagingMessagesImportCronCommand, - { - provide: MessagingMessageListFetchJob.name, - useClass: MessagingMessageListFetchJob, - }, - { - provide: MessagingMessagesImportJob.name, - useClass: MessagingMessagesImportJob, - }, - { - provide: MessagingMessageListFetchCronJob.name, - useClass: MessagingMessageListFetchCronJob, - }, - { - provide: MessagingMessagesImportCronJob.name, - useClass: MessagingMessagesImportCronJob, - }, + MessagingOngoingStaleCronCommand, + MessagingSingleMessageImportCommand, + MessagingMessageListFetchJob, + MessagingMessagesImportJob, + MessagingOngoingStaleJob, + MessagingMessageListFetchCronJob, + MessagingMessagesImportCronJob, + MessagingOngoingStaleCronJob, + MessagingAddSingleMessageToCacheForImportJob, + MessagingMessageImportManagerMessageChannelListener, + MessagingCleanCacheJob, ], exports: [], }) diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/utils/__tests__/is-sync-stale.util.spec.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/utils/__tests__/is-sync-stale.util.spec.ts new file mode 100644 index 000000000000..9935eb610812 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/utils/__tests__/is-sync-stale.util.spec.ts @@ -0,0 +1,34 @@ +import { MESSAGING_IMPORT_ONGOING_SYNC_TIMEOUT } from 'src/modules/messaging/message-import-manager/constants/messaging-import-ongoing-sync-timeout.constant'; +import { isSyncStale } from 'src/modules/messaging/message-import-manager/utils/is-sync-stale.util'; + +jest.useFakeTimers().setSystemTime(new Date('2024-01-01')); + +describe('isSyncStale', () => { + it('should return true if sync is stale', () => { + const syncStageStartedAt = new Date( + Date.now() - MESSAGING_IMPORT_ONGOING_SYNC_TIMEOUT - 1, + ).toISOString(); + + const result = isSyncStale(syncStageStartedAt); + + expect(result).toBe(true); + }); + + it('should return false if sync is not stale', () => { + const syncStageStartedAt = new Date( + Date.now() - MESSAGING_IMPORT_ONGOING_SYNC_TIMEOUT + 1, + ).toISOString(); + + const result = isSyncStale(syncStageStartedAt); + + expect(result).toBe(false); + }); + + it('should return false if syncStageStartedAt is invalid', () => { + const syncStageStartedAt = 'invalid-date'; + + expect(() => { + isSyncStale(syncStageStartedAt); + }).toThrow('Invalid date format'); + }); +}); diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/utils/filter-emails.util.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/utils/filter-emails.util.ts index 3bd9819f27d4..b87a36146cb1 100644 --- a/packages/twenty-server/src/modules/messaging/message-import-manager/utils/filter-emails.util.ts +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/utils/filter-emails.util.ts @@ -1,4 +1,4 @@ -import { isEmailBlocklisted } from 'src/modules/calendar-messaging-participant/utils/is-email-blocklisted.util'; +import { isEmailBlocklisted } from 'src/modules/calendar-messaging-participant-manager/utils/is-email-blocklisted.util'; import { GmailMessage } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message'; // Todo: refactor this into several utils @@ -9,7 +9,7 @@ export const filterEmails = ( ) => { return filterOutBlocklistedMessages( messageChannelHandle, - filterOutIcsAttachments(filterOutNonPersonalEmails(messages)), + filterOutIcsAttachments(messages), blocklist, ); }; @@ -46,22 +46,3 @@ const filterOutIcsAttachments = (messages: GmailMessage[]) => { ); }); }; - -const isPersonEmail = (email: string): boolean => { - const nonPersonalPattern = - /noreply|no-reply|do_not_reply|no\.reply|^(info@|contact@|hello@|support@|feedback@|service@|help@|invites@|invite@|welcome@|alerts@|team@|notifications@|notification@|news@)/; - - return !nonPersonalPattern.test(email); -}; - -const filterOutNonPersonalEmails = (messages: GmailMessage[]) => { - return messages.filter((message) => { - if (!message.participants) { - return true; - } - - return message.participants.every((participant) => - isPersonEmail(participant.handle), - ); - }); -}; diff --git a/packages/twenty-server/src/modules/messaging/message-import-manager/utils/is-sync-stale.util.ts b/packages/twenty-server/src/modules/messaging/message-import-manager/utils/is-sync-stale.util.ts new file mode 100644 index 000000000000..7a1ae5e2029f --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-import-manager/utils/is-sync-stale.util.ts @@ -0,0 +1,13 @@ +import { MESSAGING_IMPORT_ONGOING_SYNC_TIMEOUT } from 'src/modules/messaging/message-import-manager/constants/messaging-import-ongoing-sync-timeout.constant'; + +export const isSyncStale = (syncStageStartedAt: string): boolean => { + const syncStageStartedTime = new Date(syncStageStartedAt).getTime(); + + if (isNaN(syncStageStartedTime)) { + throw new Error('Invalid date format'); + } + + return ( + Date.now() - syncStageStartedTime > MESSAGING_IMPORT_ONGOING_SYNC_TIMEOUT + ); +}; diff --git a/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/message-participant-match-participant.job.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/message-participant-match-participant.job.ts new file mode 100644 index 000000000000..a9f486ebf1d3 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/message-participant-match-participant.job.ts @@ -0,0 +1,35 @@ +import { Scope } from '@nestjs/common'; + +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessagingMessageParticipantService } from 'src/modules/messaging/message-participant-manager/services/messaging-message-participant.service'; + +export type MessageParticipantMatchParticipantJobData = { + workspaceId: string; + email: string; + personId?: string; + workspaceMemberId?: string; +}; + +@Processor({ + queueName: MessageQueue.messagingQueue, + scope: Scope.REQUEST, +}) +export class MessageParticipantMatchParticipantJob { + constructor( + private readonly messageParticipantService: MessagingMessageParticipantService, + ) {} + + @Process(MessageParticipantMatchParticipantJob.name) + async handle(data: MessageParticipantMatchParticipantJobData): Promise<void> { + const { workspaceId, email, personId, workspaceMemberId } = data; + + await this.messageParticipantService.matchMessageParticipants( + workspaceId, + email, + personId, + workspaceMemberId, + ); + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/message-participant-unmatch-participant.job.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/message-participant-unmatch-participant.job.ts new file mode 100644 index 000000000000..2be146301bff --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/message-participant-unmatch-participant.job.ts @@ -0,0 +1,37 @@ +import { Scope } from '@nestjs/common'; + +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessagingMessageParticipantService } from 'src/modules/messaging/message-participant-manager/services/messaging-message-participant.service'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; + +export type MessageParticipantUnmatchParticipantJobData = { + workspaceId: string; + email: string; + personId?: string; + workspaceMemberId?: string; +}; + +@Processor({ + queueName: MessageQueue.messagingQueue, + scope: Scope.REQUEST, +}) +export class MessageParticipantUnmatchParticipantJob { + constructor( + private readonly messageParticipantService: MessagingMessageParticipantService, + ) {} + + @Process(MessageParticipantUnmatchParticipantJob.name) + async handle( + data: MessageParticipantUnmatchParticipantJobData, + ): Promise<void> { + const { workspaceId, email, personId, workspaceMemberId } = data; + + await this.messageParticipantService.unmatchMessageParticipants( + workspaceId, + email, + personId, + workspaceMemberId, + ); + } +} diff --git a/packages/twenty-server/src/modules/messaging/message-participants-manager/jobs/messaging-create-company-and-contact-after-sync.job.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/messaging-create-company-and-contact-after-sync.job.ts similarity index 62% rename from packages/twenty-server/src/modules/messaging/message-participants-manager/jobs/messaging-create-company-and-contact-after-sync.job.ts rename to packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/messaging-create-company-and-contact-after-sync.job.ts index 268fd74ccebf..c22b4b7d3d59 100644 --- a/packages/twenty-server/src/modules/messaging/message-participants-manager/jobs/messaging-create-company-and-contact-after-sync.job.ts +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/jobs/messaging-create-company-and-contact-after-sync.job.ts @@ -1,32 +1,31 @@ -import { Injectable, Logger } from '@nestjs/common'; +import { Logger } from '@nestjs/common'; import { InjectRepository } from '@nestjs/typeorm'; import { Repository } from 'typeorm'; -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - -import { - FeatureFlagEntity, - FeatureFlagKeys, -} from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CreateCompanyAndContactService } from 'src/modules/connected-account/auto-companies-and-contacts-creation/services/create-company-and-contact.service'; +import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; +import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; +import { CreateCompanyAndContactService } from 'src/modules/contact-creation-manager/services/create-company-and-contact.service'; import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; import { MessageParticipantRepository } from 'src/modules/messaging/common/repositories/message-participant.repository'; -import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { + MessageChannelContactAutoCreationPolicy, + MessageChannelWorkspaceEntity, +} from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; -import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; -import { ConnectedAccountRepository } from 'src/modules/connected-account/repositories/connected-account.repository'; export type MessagingCreateCompanyAndContactAfterSyncJobData = { workspaceId: string; messageChannelId: string; }; -@Injectable() -export class MessagingCreateCompanyAndContactAfterSyncJob - implements MessageQueueJob<MessagingCreateCompanyAndContactAfterSyncJobData> -{ +@Processor(MessageQueue.messagingQueue) +export class MessagingCreateCompanyAndContactAfterSyncJob { private readonly logger = new Logger( MessagingCreateCompanyAndContactAfterSyncJob.name, ); @@ -42,6 +41,7 @@ export class MessagingCreateCompanyAndContactAfterSyncJob private readonly connectedAccountRepository: ConnectedAccountRepository, ) {} + @Process(MessagingCreateCompanyAndContactAfterSyncJob.name) async handle( data: MessagingCreateCompanyAndContactAfterSyncJobData, ): Promise<void> { @@ -55,10 +55,11 @@ export class MessagingCreateCompanyAndContactAfterSyncJob workspaceId, ); - const { isContactAutoCreationEnabled, connectedAccountId } = - messageChannel[0]; + const { contactAutoCreationPolicy, connectedAccountId } = messageChannel[0]; - if (!isContactAutoCreationEnabled) { + if ( + contactAutoCreationPolicy === MessageChannelContactAutoCreationPolicy.NONE + ) { return; } @@ -73,25 +74,17 @@ export class MessagingCreateCompanyAndContactAfterSyncJob ); } - const isContactCreationForSentAndReceivedEmailsEnabledFeatureFlag = - await this.featureFlagRepository.findOneBy({ - workspaceId: workspaceId, - key: FeatureFlagKeys.IsContactCreationForSentAndReceivedEmailsEnabled, - value: true, - }); - - const isContactCreationForSentAndReceivedEmailsEnabled = - isContactCreationForSentAndReceivedEmailsEnabledFeatureFlag?.value; - - const contactsToCreate = isContactCreationForSentAndReceivedEmailsEnabled - ? await this.messageParticipantRepository.getByMessageChannelIdWithoutPersonIdAndWorkspaceMemberId( - messageChannelId, - workspaceId, - ) - : await this.messageParticipantRepository.getByMessageChannelIdWithoutPersonIdAndWorkspaceMemberIdAndMessageOutgoing( - messageChannelId, - workspaceId, - ); + const contactsToCreate = + contactAutoCreationPolicy === + MessageChannelContactAutoCreationPolicy.SENT_AND_RECEIVED + ? await this.messageParticipantRepository.getByMessageChannelIdWithoutPersonIdAndWorkspaceMemberId( + messageChannelId, + workspaceId, + ) + : await this.messageParticipantRepository.getByMessageChannelIdWithoutPersonIdAndWorkspaceMemberIdAndMessageOutgoing( + messageChannelId, + workspaceId, + ); await this.createCompanyAndContactService.createCompaniesAndContactsAndUpdateParticipants( connectedAccount, diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/listeners/participant-person.listener.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant-person.listener.ts similarity index 63% rename from packages/twenty-server/src/modules/calendar-messaging-participant/listeners/participant-person.listener.ts rename to packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant-person.listener.ts index 67fa7e7dfa86..521c92585054 100644 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/listeners/participant-person.listener.ts +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant-person.listener.ts @@ -1,25 +1,26 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; import { objectRecordChangedProperties as objectRecordUpdateEventChangedProperties } from 'src/engine/integrations/event-emitter/utils/object-record-changed-properties.util'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { - MatchParticipantJobData, - MatchParticipantJob, -} from 'src/modules/calendar-messaging-participant/jobs/match-participant.job'; + MessageParticipantMatchParticipantJobData, + MessageParticipantMatchParticipantJob, +} from 'src/modules/messaging/message-participant-manager/jobs/message-participant-match-participant.job'; import { - UnmatchParticipantJobData, - UnmatchParticipantJob, -} from 'src/modules/calendar-messaging-participant/jobs/unmatch-participant.job'; + MessageParticipantUnmatchParticipantJobData, + MessageParticipantUnmatchParticipantJob, +} from 'src/modules/messaging/message-participant-manager/jobs/message-participant-unmatch-participant.job'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; @Injectable() -export class ParticipantPersonListener { +export class MessageParticipantPersonListener { constructor( - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.messagingQueue) private readonly messageQueueService: MessageQueueService, ) {} @@ -31,8 +32,8 @@ export class ParticipantPersonListener { return; } - await this.messageQueueService.add<MatchParticipantJobData>( - MatchParticipantJob.name, + await this.messageQueueService.add<MessageParticipantMatchParticipantJobData>( + MessageParticipantMatchParticipantJob.name, { workspaceId: payload.workspaceId, email: payload.properties.after.email, @@ -51,8 +52,8 @@ export class ParticipantPersonListener { payload.properties.after, ).includes('email') ) { - await this.messageQueueService.add<UnmatchParticipantJobData>( - UnmatchParticipantJob.name, + await this.messageQueueService.add<MessageParticipantUnmatchParticipantJobData>( + MessageParticipantUnmatchParticipantJob.name, { workspaceId: payload.workspaceId, email: payload.properties.before.email, @@ -60,8 +61,8 @@ export class ParticipantPersonListener { }, ); - await this.messageQueueService.add<MatchParticipantJobData>( - MatchParticipantJob.name, + await this.messageQueueService.add<MessageParticipantMatchParticipantJobData>( + MessageParticipantMatchParticipantJob.name, { workspaceId: payload.workspaceId, email: payload.properties.after.email, diff --git a/packages/twenty-server/src/modules/calendar-messaging-participant/listeners/participant-workspace-member.listener.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant-workspace-member.listener.ts similarity index 62% rename from packages/twenty-server/src/modules/calendar-messaging-participant/listeners/participant-workspace-member.listener.ts rename to packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant-workspace-member.listener.ts index 6c8de2e44652..a7dde2e18433 100644 --- a/packages/twenty-server/src/modules/calendar-messaging-participant/listeners/participant-workspace-member.listener.ts +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant-workspace-member.listener.ts @@ -1,25 +1,26 @@ -import { Inject, Injectable } from '@nestjs/common'; +import { Injectable } from '@nestjs/common'; import { OnEvent } from '@nestjs/event-emitter'; import { ObjectRecordCreateEvent } from 'src/engine/integrations/event-emitter/types/object-record-create.event'; import { ObjectRecordUpdateEvent } from 'src/engine/integrations/event-emitter/types/object-record-update.event'; import { objectRecordChangedProperties as objectRecordUpdateEventChangedProperties } from 'src/engine/integrations/event-emitter/utils/object-record-changed-properties.util'; +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; import { - MatchParticipantJobData, - MatchParticipantJob, -} from 'src/modules/calendar-messaging-participant/jobs/match-participant.job'; + MessageParticipantMatchParticipantJobData, + MessageParticipantMatchParticipantJob, +} from 'src/modules/messaging/message-participant-manager/jobs/message-participant-match-participant.job'; import { - UnmatchParticipantJobData, - UnmatchParticipantJob, -} from 'src/modules/calendar-messaging-participant/jobs/unmatch-participant.job'; + MessageParticipantUnmatchParticipantJobData, + MessageParticipantUnmatchParticipantJob, +} from 'src/modules/messaging/message-participant-manager/jobs/message-participant-unmatch-participant.job'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; @Injectable() -export class ParticipantWorkspaceMemberListener { +export class MessageParticipantWorkspaceMemberListener { constructor( - @Inject(MessageQueue.messagingQueue) + @InjectMessageQueue(MessageQueue.messagingQueue) private readonly messageQueueService: MessageQueueService, ) {} @@ -31,8 +32,8 @@ export class ParticipantWorkspaceMemberListener { return; } - await this.messageQueueService.add<MatchParticipantJobData>( - MatchParticipantJob.name, + await this.messageQueueService.add<MessageParticipantMatchParticipantJobData>( + MessageParticipantMatchParticipantJob.name, { workspaceId: payload.workspaceId, email: payload.properties.after.userEmail, @@ -46,13 +47,13 @@ export class ParticipantWorkspaceMemberListener { payload: ObjectRecordUpdateEvent<WorkspaceMemberWorkspaceEntity>, ) { if ( - objectRecordUpdateEventChangedProperties( + objectRecordUpdateEventChangedProperties<WorkspaceMemberWorkspaceEntity>( payload.properties.before, payload.properties.after, ).includes('userEmail') ) { - await this.messageQueueService.add<UnmatchParticipantJobData>( - UnmatchParticipantJob.name, + await this.messageQueueService.add<MessageParticipantUnmatchParticipantJobData>( + MessageParticipantUnmatchParticipantJob.name, { workspaceId: payload.workspaceId, email: payload.properties.before.userEmail, @@ -60,8 +61,8 @@ export class ParticipantWorkspaceMemberListener { }, ); - await this.messageQueueService.add<MatchParticipantJobData>( - MatchParticipantJob.name, + await this.messageQueueService.add<MessageParticipantMatchParticipantJobData>( + MessageParticipantMatchParticipantJob.name, { workspaceId: payload.workspaceId, email: payload.properties.after.userEmail, diff --git a/packages/twenty-server/src/modules/messaging/message-participants-manager/listeners/message-participant.listener.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant.listener.ts similarity index 91% rename from packages/twenty-server/src/modules/messaging/message-participants-manager/listeners/message-participant.listener.ts rename to packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant.listener.ts index 92853ba5d345..a2fd781c1e07 100644 --- a/packages/twenty-server/src/modules/messaging/message-participants-manager/listeners/message-participant.listener.ts +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/listeners/message-participant.listener.ts @@ -9,7 +9,6 @@ import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repos import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { TimelineActivityRepository } from 'src/modules/timeline/repositiories/timeline-activity.repository'; import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; @Injectable() @@ -25,8 +24,8 @@ export class MessageParticipantListener { @OnEvent('messageParticipant.matched') public async handleMessageParticipantMatched(payload: { workspaceId: string; - userId: string; - messageParticipants: ObjectRecord<MessageParticipantWorkspaceEntity>[]; + workspaceMemberId: string; + messageParticipants: MessageParticipantWorkspaceEntity[]; }): Promise<void> { const messageParticipants = payload.messageParticipants ?? []; @@ -60,7 +59,7 @@ export class MessageParticipantListener { properties: null, objectName: 'message', recordId: participant.personId, - workspaceMemberId: payload.userId, + workspaceMemberId: payload.workspaceMemberId, workspaceId: payload.workspaceId, linkedObjectMetadataId: messageObjectMetadata.id, linkedRecordId: participant.messageId, diff --git a/packages/twenty-server/src/modules/messaging/message-participant-manager/message-participant-manager.module.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/message-participant-manager.module.ts new file mode 100644 index 000000000000..bbac91a84687 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/message-participant-manager.module.ts @@ -0,0 +1,48 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { AnalyticsModule } from 'src/engine/core-modules/analytics/analytics.module'; +import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; +import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; +import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; +import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; +import { CalendarChannelWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-channel.workspace-entity'; +import { ContactCreationManagerModule } from 'src/modules/contact-creation-manager/contact-creation-manager.module'; +import { MessagingCommonModule } from 'src/modules/messaging/common/messaging-common.module'; +import { MessageParticipantMatchParticipantJob } from 'src/modules/messaging/message-participant-manager/jobs/message-participant-match-participant.job'; +import { MessageParticipantUnmatchParticipantJob } from 'src/modules/messaging/message-participant-manager/jobs/message-participant-unmatch-participant.job'; +import { MessagingCreateCompanyAndContactAfterSyncJob } from 'src/modules/messaging/message-participant-manager/jobs/messaging-create-company-and-contact-after-sync.job'; +import { MessageParticipantPersonListener } from 'src/modules/messaging/message-participant-manager/listeners/message-participant-person.listener'; +import { MessageParticipantWorkspaceMemberListener } from 'src/modules/messaging/message-participant-manager/listeners/message-participant-workspace-member.listener'; +import { MessageParticipantListener } from 'src/modules/messaging/message-participant-manager/listeners/message-participant.listener'; +import { MessagingMessageParticipantService } from 'src/modules/messaging/message-participant-manager/services/messaging-message-participant.service'; +import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; + +@Module({ + imports: [ + TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), + AnalyticsModule, + ContactCreationManagerModule, + WorkspaceDataSourceModule, + ObjectMetadataRepositoryModule.forFeature([ + TimelineActivityWorkspaceEntity, + ]), + TypeOrmModule.forFeature([ObjectMetadataEntity], 'metadata'), + TwentyORMModule.forFeature([CalendarChannelWorkspaceEntity]), + MessagingCommonModule, + ], + providers: [ + MessagingMessageParticipantService, + MessageParticipantMatchParticipantJob, + MessageParticipantUnmatchParticipantJob, + MessagingCreateCompanyAndContactAfterSyncJob, + MessageParticipantListener, + MessageParticipantPersonListener, + MessageParticipantWorkspaceMemberListener, + AddPersonIdAndWorkspaceMemberIdService, + ], + exports: [MessagingMessageParticipantService], +}) +export class MessageParticipantManagerModule {} diff --git a/packages/twenty-server/src/modules/messaging/common/services/messaging-message-participant.service.ts b/packages/twenty-server/src/modules/messaging/message-participant-manager/services/messaging-message-participant.service.ts similarity index 89% rename from packages/twenty-server/src/modules/messaging/common/services/messaging-message-participant.service.ts rename to packages/twenty-server/src/modules/messaging/message-participant-manager/services/messaging-message-participant.service.ts index 78a5623bbaac..24eaec161986 100644 --- a/packages/twenty-server/src/modules/messaging/common/services/messaging-message-participant.service.ts +++ b/packages/twenty-server/src/modules/messaging/message-participant-manager/services/messaging-message-participant.service.ts @@ -7,15 +7,12 @@ import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repos import { PersonRepository } from 'src/modules/person/repositories/person.repository'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; -import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; +import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/calendar-event-import-manager/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; +import { AddPersonIdAndWorkspaceMemberIdService } from 'src/modules/calendar-messaging-participant-manager/services/add-person-id-and-workspace-member-id/add-person-id-and-workspace-member-id.service'; import { MessageParticipantRepository } from 'src/modules/messaging/common/repositories/message-participant.repository'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; import { ParticipantWithMessageId } from 'src/modules/messaging/message-import-manager/drivers/gmail/types/gmail-message'; -// Todo: this is not the right place for this file. The code needs to be refactored in term of business modules with a precise scope. -// Putting it here to avoid circular dependencies for now. @Injectable() export class MessagingMessageParticipantService { constructor( @@ -29,10 +26,10 @@ export class MessagingMessageParticipantService { ) {} public async updateMessageParticipantsAfterPeopleCreation( - createdPeople: ObjectRecord<PersonWorkspaceEntity>[], + createdPeople: PersonWorkspaceEntity[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageParticipantWorkspaceEntity>[]> { + ): Promise<MessageParticipantWorkspaceEntity[]> { const participants = await this.messageParticipantRepository.getByHandles( createdPeople.map((person) => person.email), workspaceId, @@ -87,7 +84,7 @@ export class MessagingMessageParticipantService { participants: ParticipantWithMessageId[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<MessageParticipantWorkspaceEntity>[]> { + ): Promise<MessageParticipantWorkspaceEntity[]> { if (!participants) return []; const dataSourceSchema = @@ -149,7 +146,7 @@ export class MessagingMessageParticipantService { this.eventEmitter.emit(`messageParticipant.matched`, { workspaceId, - userId: null, + workspaceMemberId: null, messageParticipants: updatedMessageParticipants, }); } diff --git a/packages/twenty-server/src/modules/messaging/message-participants-manager/messaging-participants-manager.module.ts b/packages/twenty-server/src/modules/messaging/message-participants-manager/messaging-participants-manager.module.ts deleted file mode 100644 index 589d3eb47794..000000000000 --- a/packages/twenty-server/src/modules/messaging/message-participants-manager/messaging-participants-manager.module.ts +++ /dev/null @@ -1,35 +0,0 @@ -import { Module } from '@nestjs/common'; -import { TypeOrmModule } from '@nestjs/typeorm'; - -import { AnalyticsModule } from 'src/engine/core-modules/analytics/analytics.module'; -import { FeatureFlagEntity } from 'src/engine/core-modules/feature-flag/feature-flag.entity'; -import { ObjectMetadataEntity } from 'src/engine/metadata-modules/object-metadata/object-metadata.entity'; -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; -import { WorkspaceDataSourceModule } from 'src/engine/workspace-datasource/workspace-datasource.module'; -import { AutoCompaniesAndContactsCreationModule } from 'src/modules/connected-account/auto-companies-and-contacts-creation/auto-companies-and-contacts-creation.module'; -import { MessagingGmailDriverModule } from 'src/modules/messaging/message-import-manager/drivers/gmail/messaging-gmail-driver.module'; -import { MessagingCreateCompanyAndContactAfterSyncJob } from 'src/modules/messaging/message-participants-manager/jobs/messaging-create-company-and-contact-after-sync.job'; -import { MessageParticipantListener } from 'src/modules/messaging/message-participants-manager/listeners/message-participant.listener'; -import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; - -@Module({ - imports: [ - TypeOrmModule.forFeature([FeatureFlagEntity], 'core'), - AnalyticsModule, - MessagingGmailDriverModule, - AutoCompaniesAndContactsCreationModule, - WorkspaceDataSourceModule, - ObjectMetadataRepositoryModule.forFeature([ - TimelineActivityWorkspaceEntity, - ]), - TypeOrmModule.forFeature([ObjectMetadataEntity], 'metadata'), - ], - providers: [ - { - provide: MessagingCreateCompanyAndContactAfterSyncJob.name, - useClass: MessagingCreateCompanyAndContactAfterSyncJob, - }, - MessageParticipantListener, - ], -}) -export class MessaginParticipantsManagerModule {} diff --git a/packages/twenty-server/src/modules/messaging/messaging.module.ts b/packages/twenty-server/src/modules/messaging/messaging.module.ts index 0797e3853b64..24376567ca96 100644 --- a/packages/twenty-server/src/modules/messaging/messaging.module.ts +++ b/packages/twenty-server/src/modules/messaging/messaging.module.ts @@ -3,14 +3,16 @@ import { Module } from '@nestjs/common'; import { MessagingBlocklistManagerModule } from 'src/modules/messaging/blocklist-manager/messaging-blocklist-manager.module'; import { MessagingMessageCleanerModule } from 'src/modules/messaging/message-cleaner/messaging-message-cleaner.module'; import { MessagingImportManagerModule } from 'src/modules/messaging/message-import-manager/messaging-import-manager.module'; -import { MessaginParticipantsManagerModule } from 'src/modules/messaging/message-participants-manager/messaging-participants-manager.module'; +import { MessageParticipantManagerModule } from 'src/modules/messaging/message-participant-manager/message-participant-manager.module'; +import { MessagingMonitoringModule } from 'src/modules/messaging/monitoring/messaging-monitoring.module'; @Module({ imports: [ MessagingImportManagerModule, MessagingMessageCleanerModule, - MessaginParticipantsManagerModule, + MessageParticipantManagerModule, MessagingBlocklistManagerModule, + MessagingMonitoringModule, ], providers: [], exports: [], diff --git a/packages/twenty-server/src/modules/messaging/monitoring/crons/commands/messaging-message-channel-sync-status-monitoring.cron.command.ts b/packages/twenty-server/src/modules/messaging/monitoring/crons/commands/messaging-message-channel-sync-status-monitoring.cron.command.ts new file mode 100644 index 000000000000..d90fd0b5be45 --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/monitoring/crons/commands/messaging-message-channel-sync-status-monitoring.cron.command.ts @@ -0,0 +1,36 @@ +import { Command, CommandRunner } from 'nest-commander'; + +import { InjectMessageQueue } from 'src/engine/integrations/message-queue/decorators/message-queue.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; +import { MessagingMessageChannelSyncStatusMonitoringCronJob } from 'src/modules/messaging/monitoring/crons/jobs/messaging-message-channel-sync-status-monitoring.cron'; + +const MESSAGING_MESSAGE_CHANNEL_SYNC_STATUS_MONITORING_CRON_PATTERN = + '2/10 * * * *'; //Every 10 minutes, starting at 2 minutes past the hour + +@Command({ + name: 'cron:messaging:monitoring:message-channel-sync-status', + description: + 'Starts a cron job to monitor the sync status of message channels', +}) +export class MessagingMessageChannelSyncStatusMonitoringCronCommand extends CommandRunner { + constructor( + @InjectMessageQueue(MessageQueue.cronQueue) + private readonly messageQueueService: MessageQueueService, + ) { + super(); + } + + async run(): Promise<void> { + await this.messageQueueService.addCron<undefined>( + MessagingMessageChannelSyncStatusMonitoringCronJob.name, + undefined, + { + repeat: { + pattern: + MESSAGING_MESSAGE_CHANNEL_SYNC_STATUS_MONITORING_CRON_PATTERN, + }, + }, + ); + } +} diff --git a/packages/twenty-server/src/modules/messaging/monitoring/crons/jobs/messaging-message-channel-sync-status-monitoring.cron.ts b/packages/twenty-server/src/modules/messaging/monitoring/crons/jobs/messaging-message-channel-sync-status-monitoring.cron.ts new file mode 100644 index 000000000000..a0384d52829c --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/monitoring/crons/jobs/messaging-message-channel-sync-status-monitoring.cron.ts @@ -0,0 +1,77 @@ +import { Logger } from '@nestjs/common'; +import { InjectRepository } from '@nestjs/typeorm'; + +import { Repository, In } from 'typeorm'; +import snakeCase from 'lodash.snakecase'; + +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; +import { MessageChannelRepository } from 'src/modules/messaging/common/repositories/message-channel.repository'; +import { MessageChannelWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-channel.workspace-entity'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; +import { MessagingTelemetryService } from 'src/modules/messaging/common/services/messaging-telemetry.service'; +import { BillingService } from 'src/engine/core-modules/billing/billing.service'; + +@Processor(MessageQueue.cronQueue) +export class MessagingMessageChannelSyncStatusMonitoringCronJob { + private readonly logger = new Logger( + MessagingMessageChannelSyncStatusMonitoringCronJob.name, + ); + + constructor( + @InjectRepository(Workspace, 'core') + private readonly workspaceRepository: Repository<Workspace>, + @InjectRepository(DataSourceEntity, 'metadata') + private readonly dataSourceRepository: Repository<DataSourceEntity>, + @InjectObjectMetadataRepository(MessageChannelWorkspaceEntity) + private readonly messageChannelRepository: MessageChannelRepository, + private readonly billingService: BillingService, + private readonly messagingTelemetryService: MessagingTelemetryService, + ) {} + + @Process(MessagingMessageChannelSyncStatusMonitoringCronJob.name) + async handle(): Promise<void> { + this.logger.log('Starting message channel sync status monitoring...'); + + await this.messagingTelemetryService.track({ + eventName: 'message_channel.monitoring.sync_status.start', + message: 'Starting message channel sync status monitoring', + }); + + const workspaceIds = + await this.billingService.getActiveSubscriptionWorkspaceIds(); + + const dataSources = await this.dataSourceRepository.find({ + where: { + workspaceId: In(workspaceIds), + }, + }); + + const workspaceIdsWithDataSources = new Set( + dataSources.map((dataSource) => dataSource.workspaceId), + ); + + for (const workspaceId of workspaceIdsWithDataSources) { + const messageChannels = + await this.messageChannelRepository.getAll(workspaceId); + + for (const messageChannel of messageChannels) { + if (!messageChannel.syncStatus) { + continue; + } + await this.messagingTelemetryService.track({ + eventName: `message_channel.monitoring.sync_status.${snakeCase( + messageChannel.syncStatus, + )}`, + workspaceId, + connectedAccountId: messageChannel.connectedAccountId, + messageChannelId: messageChannel.id, + message: messageChannel.syncStatus, + }); + } + } + } +} diff --git a/packages/twenty-server/src/modules/messaging/monitoring/messaging-monitoring.module.ts b/packages/twenty-server/src/modules/messaging/monitoring/messaging-monitoring.module.ts new file mode 100644 index 000000000000..d41ed75e413e --- /dev/null +++ b/packages/twenty-server/src/modules/messaging/monitoring/messaging-monitoring.module.ts @@ -0,0 +1,24 @@ +import { Module } from '@nestjs/common'; +import { TypeOrmModule } from '@nestjs/typeorm'; + +import { Workspace } from 'src/engine/core-modules/workspace/workspace.entity'; +import { DataSourceEntity } from 'src/engine/metadata-modules/data-source/data-source.entity'; +import { MessagingCommonModule } from 'src/modules/messaging/common/messaging-common.module'; +import { MessagingMessageChannelSyncStatusMonitoringCronCommand } from 'src/modules/messaging/monitoring/crons/commands/messaging-message-channel-sync-status-monitoring.cron.command'; +import { MessagingMessageChannelSyncStatusMonitoringCronJob } from 'src/modules/messaging/monitoring/crons/jobs/messaging-message-channel-sync-status-monitoring.cron'; +import { BillingModule } from 'src/engine/core-modules/billing/billing.module'; + +@Module({ + imports: [ + MessagingCommonModule, + BillingModule, + TypeOrmModule.forFeature([Workspace], 'core'), + TypeOrmModule.forFeature([DataSourceEntity], 'metadata'), + ], + providers: [ + MessagingMessageChannelSyncStatusMonitoringCronCommand, + MessagingMessageChannelSyncStatusMonitoringCronJob, + ], + exports: [], +}) +export class MessagingMonitoringModule {} diff --git a/packages/twenty-server/src/modules/opportunity/standard-objects/opportunity.workspace-entity.ts b/packages/twenty-server/src/modules/opportunity/standard-objects/opportunity.workspace-entity.ts index 5ec506eb57c9..670ef37a069d 100644 --- a/packages/twenty-server/src/modules/opportunity/standard-objects/opportunity.workspace-entity.ts +++ b/packages/twenty-server/src/modules/opportunity/standard-objects/opportunity.workspace-entity.ts @@ -6,6 +6,15 @@ import { RelationMetadataType, RelationOnDeleteAction, } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; +import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; +import { WorkspaceIsDeprecated } from 'src/engine/twenty-orm/decorators/workspace-is-deprecated.decorator'; +import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; +import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; +import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { OPPORTUNITY_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; import { ActivityTargetWorkspaceEntity } from 'src/modules/activity/standard-objects/activity-target.workspace-entity'; @@ -14,13 +23,6 @@ import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/com import { FavoriteWorkspaceEntity } from 'src/modules/favorite/standard-objects/favorite.workspace-entity'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; -import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; -import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; -import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; -import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; -import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; -import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; -import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.opportunity, @@ -49,7 +51,7 @@ export class OpportunityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconCurrencyDollar', }) @WorkspaceIsNullable() - amount: CurrencyMetadata; + amount: CurrencyMetadata | null; @WorkspaceField({ standardId: OPPORTUNITY_STANDARD_FIELD_IDS.closeDate, @@ -59,17 +61,7 @@ export class OpportunityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconCalendarEvent', }) @WorkspaceIsNullable() - closeDate: Date; - - @WorkspaceField({ - standardId: OPPORTUNITY_STANDARD_FIELD_IDS.probability, - type: FieldMetadataType.TEXT, - label: 'Probability', - description: 'Opportunity probability', - icon: 'IconProgressCheck', - defaultValue: "'0'", - }) - probability: string; + closeDate: Date | null; @WorkspaceField({ standardId: OPPORTUNITY_STANDARD_FIELD_IDS.stage, @@ -102,7 +94,7 @@ export class OpportunityWorkspaceEntity extends BaseWorkspaceEntity { }) @WorkspaceIsSystem() @WorkspaceIsNullable() - position: number; + position: number | null; @WorkspaceRelation({ standardId: OPPORTUNITY_STANDARD_FIELD_IDS.pointOfContact, @@ -110,13 +102,15 @@ export class OpportunityWorkspaceEntity extends BaseWorkspaceEntity { label: 'Point of Contact', description: 'Opportunity point of contact', icon: 'IconUser', - joinColumn: 'pointOfContactId', inverseSideTarget: () => PersonWorkspaceEntity, inverseSideFieldKey: 'pointOfContactForOpportunities', onDelete: RelationOnDeleteAction.SET_NULL, }) @WorkspaceIsNullable() - pointOfContact: Relation<PersonWorkspaceEntity>; + pointOfContact: Relation<PersonWorkspaceEntity> | null; + + @WorkspaceJoinColumn('pointOfContact') + pointOfContactId: string | null; @WorkspaceRelation({ standardId: OPPORTUNITY_STANDARD_FIELD_IDS.company, @@ -124,13 +118,15 @@ export class OpportunityWorkspaceEntity extends BaseWorkspaceEntity { label: 'Company', description: 'Opportunity company', icon: 'IconBuildingSkyscraper', - joinColumn: 'companyId', inverseSideTarget: () => CompanyWorkspaceEntity, inverseSideFieldKey: 'opportunities', onDelete: RelationOnDeleteAction.SET_NULL, }) @WorkspaceIsNullable() - company: Relation<CompanyWorkspaceEntity>; + company: Relation<CompanyWorkspaceEntity> | null; + + @WorkspaceJoinColumn('company') + companyId: string | null; @WorkspaceRelation({ standardId: OPPORTUNITY_STANDARD_FIELD_IDS.favorites, @@ -180,4 +176,15 @@ export class OpportunityWorkspaceEntity extends BaseWorkspaceEntity { }) @WorkspaceIsNullable() timelineActivities: Relation<TimelineActivityWorkspaceEntity[]>; + + @WorkspaceField({ + standardId: OPPORTUNITY_STANDARD_FIELD_IDS.probabilityDeprecated, + type: FieldMetadataType.TEXT, + label: 'Probability', + description: 'Opportunity probability', + icon: 'IconProgressCheck', + defaultValue: "'0'", + }) + @WorkspaceIsDeprecated() + probability: string; } diff --git a/packages/twenty-server/src/modules/person/repositories/person.repository.ts b/packages/twenty-server/src/modules/person/repositories/person.repository.ts index 57e1464fc955..adb40f1cbdb3 100644 --- a/packages/twenty-server/src/modules/person/repositories/person.repository.ts +++ b/packages/twenty-server/src/modules/person/repositories/person.repository.ts @@ -3,9 +3,8 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; import { PersonWorkspaceEntity } from 'src/modules/person/standard-objects/person.workspace-entity'; -import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; +import { getFlattenedValuesAndValuesStringForBatchRawQuery } from 'src/modules/calendar/calendar-event-import-manager/utils/get-flattened-values-and-values-string-for-batch-raw-query.util'; @Injectable() export class PersonRepository { @@ -17,7 +16,7 @@ export class PersonRepository { emails: string[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<PersonWorkspaceEntity>[]> { + ): Promise<PersonWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -56,7 +55,7 @@ export class PersonRepository { }[], workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<PersonWorkspaceEntity>[]> { + ): Promise<PersonWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); diff --git a/packages/twenty-server/src/modules/person/standard-objects/person.workspace-entity.ts b/packages/twenty-server/src/modules/person/standard-objects/person.workspace-entity.ts index 83353993dfb1..0d1704aed68f 100644 --- a/packages/twenty-server/src/modules/person/standard-objects/person.workspace-entity.ts +++ b/packages/twenty-server/src/modules/person/standard-objects/person.workspace-entity.ts @@ -11,7 +11,6 @@ import { PERSON_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspac import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; import { ActivityTargetWorkspaceEntity } from 'src/modules/activity/standard-objects/activity-target.workspace-entity'; import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objects/attachment.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/company.workspace-entity'; import { FavoriteWorkspaceEntity } from 'src/modules/favorite/standard-objects/favorite.workspace-entity'; import { OpportunityWorkspaceEntity } from 'src/modules/opportunity/standard-objects/opportunity.workspace-entity'; @@ -23,6 +22,8 @@ import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace- import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.person, @@ -41,7 +42,7 @@ export class PersonWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconUser', }) @WorkspaceIsNullable() - name: FullNameMetadata; + name: FullNameMetadata | null; @WorkspaceField({ standardId: PERSON_STANDARD_FIELD_IDS.email, @@ -60,7 +61,7 @@ export class PersonWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconBrandLinkedin', }) @WorkspaceIsNullable() - linkedinLink: LinkMetadata; + linkedinLink: LinkMetadata | null; @WorkspaceField({ standardId: PERSON_STANDARD_FIELD_IDS.xLink, @@ -70,7 +71,7 @@ export class PersonWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconBrandX', }) @WorkspaceIsNullable() - xLink: LinkMetadata; + xLink: LinkMetadata | null; @WorkspaceField({ standardId: PERSON_STANDARD_FIELD_IDS.jobTitle, @@ -118,7 +119,7 @@ export class PersonWorkspaceEntity extends BaseWorkspaceEntity { }) @WorkspaceIsSystem() @WorkspaceIsNullable() - position: number; + position: number | null; // Relations @WorkspaceRelation({ @@ -127,12 +128,14 @@ export class PersonWorkspaceEntity extends BaseWorkspaceEntity { label: 'Company', description: 'Contact’s company', icon: 'IconBuildingSkyscraper', - joinColumn: 'companyId', inverseSideTarget: () => CompanyWorkspaceEntity, inverseSideFieldKey: 'people', }) @WorkspaceIsNullable() - company: Relation<CompanyWorkspaceEntity>; + company: Relation<CompanyWorkspaceEntity> | null; + + @WorkspaceJoinColumn('company') + companyId: string | null; @WorkspaceRelation({ standardId: PERSON_STANDARD_FIELD_IDS.pointOfContactForOpportunities, diff --git a/packages/twenty-server/src/modules/timeline/jobs/create-audit-log-from-internal-event.ts b/packages/twenty-server/src/modules/timeline/jobs/create-audit-log-from-internal-event.ts index 32d8b7637005..edf4c3d9e488 100644 --- a/packages/twenty-server/src/modules/timeline/jobs/create-audit-log-from-internal-event.ts +++ b/packages/twenty-server/src/modules/timeline/jobs/create-audit-log-from-internal-event.ts @@ -1,18 +1,15 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { ObjectRecordBaseEvent } from 'src/engine/integrations/event-emitter/types/object-record.base.event'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { AuditLogRepository } from 'src/modules/timeline/repositiories/audit-log.repository'; import { AuditLogWorkspaceEntity } from 'src/modules/timeline/standard-objects/audit-log.workspace-entity'; import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; -@Injectable() -export class CreateAuditLogFromInternalEvent - implements MessageQueueJob<ObjectRecordBaseEvent> -{ +@Processor(MessageQueue.entityEventsToDbQueue) +export class CreateAuditLogFromInternalEvent { constructor( @InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity) private readonly workspaceMemberService: WorkspaceMemberRepository, @@ -20,6 +17,7 @@ export class CreateAuditLogFromInternalEvent private readonly auditLogRepository: AuditLogRepository, ) {} + @Process(CreateAuditLogFromInternalEvent.name) async handle(data: ObjectRecordBaseEvent): Promise<void> { let workspaceMemberId: string | null = null; diff --git a/packages/twenty-server/src/modules/timeline/jobs/timeline-job.module.ts b/packages/twenty-server/src/modules/timeline/jobs/timeline-job.module.ts index 3fd03709ce2b..416e81aed53f 100644 --- a/packages/twenty-server/src/modules/timeline/jobs/timeline-job.module.ts +++ b/packages/twenty-server/src/modules/timeline/jobs/timeline-job.module.ts @@ -16,14 +16,8 @@ import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/sta TimelineActivityModule, ], providers: [ - { - provide: CreateAuditLogFromInternalEvent.name, - useClass: CreateAuditLogFromInternalEvent, - }, - { - provide: UpsertTimelineActivityFromInternalEvent.name, - useClass: UpsertTimelineActivityFromInternalEvent, - }, + CreateAuditLogFromInternalEvent, + UpsertTimelineActivityFromInternalEvent, ], }) export class TimelineJobModule {} diff --git a/packages/twenty-server/src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job.ts b/packages/twenty-server/src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job.ts index 2f8490764963..5abcb2e10da6 100644 --- a/packages/twenty-server/src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job.ts +++ b/packages/twenty-server/src/modules/timeline/jobs/upsert-timeline-activity-from-internal-event.job.ts @@ -1,23 +1,21 @@ -import { Injectable } from '@nestjs/common'; - -import { MessageQueueJob } from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { ObjectRecordBaseEvent } from 'src/engine/integrations/event-emitter/types/object-record.base.event'; import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; import { WorkspaceMemberRepository } from 'src/modules/workspace-member/repositories/workspace-member.repository'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; import { TimelineActivityService } from 'src/modules/timeline/services/timeline-activity.service'; +import { Processor } from 'src/engine/integrations/message-queue/decorators/processor.decorator'; +import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; +import { Process } from 'src/engine/integrations/message-queue/decorators/process.decorator'; -@Injectable() -export class UpsertTimelineActivityFromInternalEvent - implements MessageQueueJob<ObjectRecordBaseEvent> -{ +@Processor(MessageQueue.entityEventsToDbQueue) +export class UpsertTimelineActivityFromInternalEvent { constructor( @InjectObjectMetadataRepository(WorkspaceMemberWorkspaceEntity) private readonly workspaceMemberService: WorkspaceMemberRepository, private readonly timelineActivityService: TimelineActivityService, ) {} + @Process(UpsertTimelineActivityFromInternalEvent.name) async handle(data: ObjectRecordBaseEvent): Promise<void> { if (data.userId) { const workspaceMember = await this.workspaceMemberService.getByIdOrFail( diff --git a/packages/twenty-server/src/modules/timeline/repositiories/timeline-activity.repository.ts b/packages/twenty-server/src/modules/timeline/repositiories/timeline-activity.repository.ts index 8d9a99c7d98e..1ab962ea9cf1 100644 --- a/packages/twenty-server/src/modules/timeline/repositiories/timeline-activity.repository.ts +++ b/packages/twenty-server/src/modules/timeline/repositiories/timeline-activity.repository.ts @@ -2,6 +2,8 @@ import { Injectable } from '@nestjs/common'; import { EntityManager } from 'typeorm'; +import { Record } from 'src/engine/api/graphql/workspace-query-builder/interfaces/record.interface'; + import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { objectRecordDiffMerge } from 'src/engine/integrations/event-emitter/utils/object-record-diff-merge'; @@ -13,7 +15,7 @@ export class TimelineActivityRepository { async upsertOne( name: string, - properties: Record<string, any>, + properties: Partial<Record>, objectName: string, recordId: string, workspaceId: string, @@ -103,7 +105,7 @@ export class TimelineActivityRepository { private async updateTimelineActivity( dataSourceSchema: string, id: string, - properties: Record<string, any>, + properties: Partial<Record>, workspaceMemberId: string | undefined, workspaceId: string, ) { @@ -119,7 +121,7 @@ export class TimelineActivityRepository { private async insertTimelineActivity( dataSourceSchema: string, name: string, - properties: Record<string, any>, + properties: Partial<Record>, objectName: string, recordId: string, workspaceMemberId: string | undefined, @@ -149,11 +151,11 @@ export class TimelineActivityRepository { objectName: string, activities: { name: string; - properties: Record<string, any> | null; + properties: Partial<Record> | null; workspaceMemberId: string | undefined; - recordId: string; + recordId: string | null; linkedRecordCachedName: string; - linkedRecordId: string | undefined; + linkedRecordId: string | null | undefined; linkedObjectMetadataId: string | undefined; }[], workspaceId: string, diff --git a/packages/twenty-server/src/modules/timeline/standard-objects/audit-log.workspace-entity.ts b/packages/twenty-server/src/modules/timeline/standard-objects/audit-log.workspace-entity.ts index d7b42a75b089..e6cdc4a77f87 100644 --- a/packages/twenty-server/src/modules/timeline/standard-objects/audit-log.workspace-entity.ts +++ b/packages/twenty-server/src/modules/timeline/standard-objects/audit-log.workspace-entity.ts @@ -1,16 +1,17 @@ import { Relation } from 'src/engine/workspace-manager/workspace-sync-metadata/interfaces/relation.interface'; import { FieldMetadataType } from 'src/engine/metadata-modules/field-metadata/field-metadata.entity'; -import { AUDIT_LOGS_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; -import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; -import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; +import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; -import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; -import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { AUDIT_LOGS_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; +import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; +import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.auditLog, @@ -39,7 +40,7 @@ export class AuditLogWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconListDetails', }) @WorkspaceIsNullable() - properties: JSON; + properties: JSON | null; @WorkspaceField({ standardId: AUDIT_LOGS_STANDARD_FIELD_IDS.context, @@ -50,22 +51,22 @@ export class AuditLogWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconListDetails', }) @WorkspaceIsNullable() - context: JSON; + context: JSON | null; @WorkspaceField({ standardId: AUDIT_LOGS_STANDARD_FIELD_IDS.objectName, type: FieldMetadataType.TEXT, label: 'Object name', - description: 'If the event is related to a particular object', + description: 'Object name', icon: 'IconAbc', }) objectName: string; @WorkspaceField({ - standardId: AUDIT_LOGS_STANDARD_FIELD_IDS.objectName, + standardId: AUDIT_LOGS_STANDARD_FIELD_IDS.objectMetadataId, type: FieldMetadataType.TEXT, - label: 'Object name', - description: 'If the event is related to a particular object', + label: 'Object metadata id', + description: 'Object metadata id', icon: 'IconAbc', }) objectMetadataId: string; @@ -73,12 +74,12 @@ export class AuditLogWorkspaceEntity extends BaseWorkspaceEntity { @WorkspaceField({ standardId: AUDIT_LOGS_STANDARD_FIELD_IDS.recordId, type: FieldMetadataType.UUID, - label: 'Object id', - description: 'Event name/type', + label: 'Record id', + description: 'Record id', icon: 'IconAbc', }) @WorkspaceIsNullable() - recordId: string; + recordId: string | null; @WorkspaceRelation({ standardId: AUDIT_LOGS_STANDARD_FIELD_IDS.workspaceMember, @@ -86,10 +87,12 @@ export class AuditLogWorkspaceEntity extends BaseWorkspaceEntity { label: 'Workspace Member', description: 'Event workspace member', icon: 'IconCircleUser', - joinColumn: 'workspaceMemberId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'auditLogs', }) @WorkspaceIsNullable() - workspaceMember: Relation<WorkspaceMemberWorkspaceEntity>; + workspaceMember: Relation<WorkspaceMemberWorkspaceEntity> | null; + + @WorkspaceJoinColumn('workspaceMember') + workspaceMemberId: string | null; } diff --git a/packages/twenty-server/src/modules/timeline/standard-objects/behavioral-event.workspace-entity.ts b/packages/twenty-server/src/modules/timeline/standard-objects/behavioral-event.workspace-entity.ts index 3460768e4f81..a74069685bbc 100644 --- a/packages/twenty-server/src/modules/timeline/standard-objects/behavioral-event.workspace-entity.ts +++ b/packages/twenty-server/src/modules/timeline/standard-objects/behavioral-event.workspace-entity.ts @@ -56,7 +56,7 @@ export class BehavioralEventWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconListDetails', }) @WorkspaceIsNullable() - properties: JSON; + properties: JSON | null; @WorkspaceField({ standardId: BEHAVIORAL_EVENT_STANDARD_FIELD_IDS.context, @@ -67,7 +67,7 @@ export class BehavioralEventWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconListDetails', }) @WorkspaceIsNullable() - context: JSON; + context: JSON | null; @WorkspaceField({ standardId: BEHAVIORAL_EVENT_STANDARD_FIELD_IDS.objectName, @@ -86,5 +86,5 @@ export class BehavioralEventWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconAbc', }) @WorkspaceIsNullable() - recordId: string; + recordId: string | null; } diff --git a/packages/twenty-server/src/modules/timeline/standard-objects/timeline-activity.workspace-entity.ts b/packages/twenty-server/src/modules/timeline/standard-objects/timeline-activity.workspace-entity.ts index 40cd49307684..c104b5df8931 100644 --- a/packages/twenty-server/src/modules/timeline/standard-objects/timeline-activity.workspace-entity.ts +++ b/packages/twenty-server/src/modules/timeline/standard-objects/timeline-activity.workspace-entity.ts @@ -17,6 +17,7 @@ import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace- import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { WorkspaceDynamicRelation } from 'src/engine/twenty-orm/decorators/workspace-dynamic-relation.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.timelineActivity, @@ -56,7 +57,7 @@ export class TimelineActivityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconListDetails', }) @WorkspaceIsNullable() - properties: JSON; + properties: JSON | null; // Special objects that don't have their own timeline and are 'link' to the main object @WorkspaceField({ @@ -76,7 +77,7 @@ export class TimelineActivityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconAbc', }) @WorkspaceIsNullable() - linkedRecordId: string; + linkedRecordId: string | null; @WorkspaceField({ standardId: TIMELINE_ACTIVITY_STANDARD_FIELD_IDS.linkedObjectMetadataId, @@ -86,7 +87,7 @@ export class TimelineActivityWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconAbc', }) @WorkspaceIsNullable() - linkedObjectMetadataId: string; + linkedObjectMetadataId: string | null; // Who made the action @WorkspaceRelation({ @@ -95,12 +96,14 @@ export class TimelineActivityWorkspaceEntity extends BaseWorkspaceEntity { label: 'Workspace Member', description: 'Event workspace member', icon: 'IconCircleUser', - joinColumn: 'workspaceMemberId', inverseSideTarget: () => WorkspaceMemberWorkspaceEntity, inverseSideFieldKey: 'timelineActivities', }) @WorkspaceIsNullable() - workspaceMember: Relation<WorkspaceMemberWorkspaceEntity>; + workspaceMember: Relation<WorkspaceMemberWorkspaceEntity> | null; + + @WorkspaceJoinColumn('workspaceMember') + workspaceMemberId: string | null; @WorkspaceRelation({ standardId: TIMELINE_ACTIVITY_STANDARD_FIELD_IDS.person, @@ -108,12 +111,14 @@ export class TimelineActivityWorkspaceEntity extends BaseWorkspaceEntity { label: 'Person', description: 'Event person', icon: 'IconUser', - joinColumn: 'personId', inverseSideTarget: () => PersonWorkspaceEntity, inverseSideFieldKey: 'timelineActivities', }) @WorkspaceIsNullable() - person: Relation<PersonWorkspaceEntity>; + person: Relation<PersonWorkspaceEntity> | null; + + @WorkspaceJoinColumn('person') + personId: string | null; @WorkspaceRelation({ standardId: TIMELINE_ACTIVITY_STANDARD_FIELD_IDS.company, @@ -121,12 +126,14 @@ export class TimelineActivityWorkspaceEntity extends BaseWorkspaceEntity { label: 'Company', description: 'Event company', icon: 'IconBuildingSkyscraper', - joinColumn: 'companyId', inverseSideTarget: () => CompanyWorkspaceEntity, inverseSideFieldKey: 'timelineActivities', }) @WorkspaceIsNullable() - company: Relation<CompanyWorkspaceEntity>; + company: Relation<CompanyWorkspaceEntity> | null; + + @WorkspaceJoinColumn('company') + companyId: string | null; @WorkspaceRelation({ standardId: TIMELINE_ACTIVITY_STANDARD_FIELD_IDS.opportunity, @@ -134,12 +141,14 @@ export class TimelineActivityWorkspaceEntity extends BaseWorkspaceEntity { label: 'Opportunity', description: 'Event opportunity', icon: 'IconTargetArrow', - joinColumn: 'opportunityId', inverseSideTarget: () => OpportunityWorkspaceEntity, inverseSideFieldKey: 'timelineActivities', }) @WorkspaceIsNullable() - opportunity: Relation<OpportunityWorkspaceEntity>; + opportunity: Relation<OpportunityWorkspaceEntity> | null; + + @WorkspaceJoinColumn('opportunity') + opportunityId: string | null; @WorkspaceDynamicRelation({ type: RelationMetadataType.MANY_TO_ONE, diff --git a/packages/twenty-server/src/modules/view/standard-objects/view-field.workspace-entity.ts b/packages/twenty-server/src/modules/view/standard-objects/view-field.workspace-entity.ts index 424f4299ce5c..e46ee8c0effd 100644 --- a/packages/twenty-server/src/modules/view/standard-objects/view-field.workspace-entity.ts +++ b/packages/twenty-server/src/modules/view/standard-objects/view-field.workspace-entity.ts @@ -10,6 +10,7 @@ import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-re import { VIEW_FIELD_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; import { ViewWorkspaceEntity } from 'src/modules/view/standard-objects/view.workspace-entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.viewField, @@ -69,8 +70,10 @@ export class ViewFieldWorkspaceEntity extends BaseWorkspaceEntity { icon: 'IconLayoutCollage', inverseSideTarget: () => ViewWorkspaceEntity, inverseSideFieldKey: 'viewFields', - joinColumn: 'viewId', }) @WorkspaceIsNullable() - view?: ViewWorkspaceEntity; + view?: ViewWorkspaceEntity | null; + + @WorkspaceJoinColumn('view') + viewId: string | null; } diff --git a/packages/twenty-server/src/modules/view/standard-objects/view-filter.workspace-entity.ts b/packages/twenty-server/src/modules/view/standard-objects/view-filter.workspace-entity.ts index bce3e944aff4..149171e6eb7f 100644 --- a/packages/twenty-server/src/modules/view/standard-objects/view-filter.workspace-entity.ts +++ b/packages/twenty-server/src/modules/view/standard-objects/view-filter.workspace-entity.ts @@ -12,6 +12,7 @@ import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.viewFilter, @@ -63,10 +64,12 @@ export class ViewFilterWorkspaceEntity extends BaseWorkspaceEntity { label: 'View', description: 'View Filter related view', icon: 'IconLayoutCollage', - joinColumn: 'viewId', inverseSideTarget: () => ViewWorkspaceEntity, inverseSideFieldKey: 'viewFilters', }) @WorkspaceIsNullable() - view: Relation<ViewWorkspaceEntity>; + view: Relation<ViewWorkspaceEntity> | null; + + @WorkspaceJoinColumn('view') + viewId: string | null; } diff --git a/packages/twenty-server/src/modules/view/standard-objects/view-sort.workspace-entity.ts b/packages/twenty-server/src/modules/view/standard-objects/view-sort.workspace-entity.ts index 458d4d316881..5abc9848ed7d 100644 --- a/packages/twenty-server/src/modules/view/standard-objects/view-sort.workspace-entity.ts +++ b/packages/twenty-server/src/modules/view/standard-objects/view-sort.workspace-entity.ts @@ -12,6 +12,7 @@ import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { RelationMetadataType } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { WorkspaceJoinColumn } from 'src/engine/twenty-orm/decorators/workspace-join-column.decorator'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.viewSort, @@ -48,10 +49,12 @@ export class ViewSortWorkspaceEntity extends BaseWorkspaceEntity { label: 'View', description: 'View Sort related view', icon: 'IconLayoutCollage', - joinColumn: 'viewId', inverseSideTarget: () => ViewWorkspaceEntity, inverseSideFieldKey: 'viewSorts', }) @WorkspaceIsNullable() - view: Relation<ViewWorkspaceEntity>; + view: Relation<ViewWorkspaceEntity> | null; + + @WorkspaceJoinColumn('view') + viewId: string | null; } diff --git a/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-many.pre-query.hook.ts b/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-many.pre-query.hook.ts index e0650b7e2548..9effd33285b7 100644 --- a/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-many.pre-query.hook.ts +++ b/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-many.pre-query.hook.ts @@ -1,10 +1,12 @@ -import { Injectable, MethodNotAllowedException } from '@nestjs/common'; +import { MethodNotAllowedException } from '@nestjs/common'; -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; -@Injectable() +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; + +@WorkspaceQueryHook(`workspaceMember.deleteMany`) export class WorkspaceMemberDeleteManyPreQueryHook - implements WorkspacePreQueryHook + implements WorkspaceQueryHookInstance { constructor() {} diff --git a/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-one.pre-query.hook.ts b/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-one.pre-query.hook.ts index efa6cd863cc0..e068da436be3 100644 --- a/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-one.pre-query.hook.ts +++ b/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-delete-one.pre-query.hook.ts @@ -1,23 +1,21 @@ -import { Injectable } from '@nestjs/common'; - -import { WorkspacePreQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-pre-query-hook/interfaces/workspace-pre-query-hook.interface'; +import { WorkspaceQueryHookInstance } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/interfaces/workspace-query-hook.interface'; import { DeleteOneResolverArgs } from 'src/engine/api/graphql/workspace-resolver-builder/interfaces/workspace-resolvers-builder.interface'; -import { InjectObjectMetadataRepository } from 'src/engine/object-metadata-repository/object-metadata-repository.decorator'; -import { CommentRepository } from 'src/modules/activity/repositories/comment.repository'; +import { WorkspaceQueryHook } from 'src/engine/api/graphql/workspace-query-runner/workspace-query-hook/decorators/workspace-query-hook.decorator'; +import { InjectWorkspaceRepository } from 'src/engine/twenty-orm/decorators/inject-workspace-repository.decorator'; +import { WorkspaceRepository } from 'src/engine/twenty-orm/repository/workspace.repository'; import { CommentWorkspaceEntity } from 'src/modules/activity/standard-objects/comment.workspace-entity'; -import { AttachmentRepository } from 'src/modules/attachment/repositories/attachment.repository'; import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objects/attachment.workspace-entity'; -@Injectable() +@WorkspaceQueryHook(`workspaceMember.deleteOne`) export class WorkspaceMemberDeleteOnePreQueryHook - implements WorkspacePreQueryHook + implements WorkspaceQueryHookInstance { constructor( - @InjectObjectMetadataRepository(AttachmentWorkspaceEntity) - private readonly attachmentRepository: AttachmentRepository, - @InjectObjectMetadataRepository(CommentWorkspaceEntity) - private readonly commentRepository: CommentRepository, + @InjectWorkspaceRepository(AttachmentWorkspaceEntity) + private readonly attachmentRepository: WorkspaceRepository<AttachmentWorkspaceEntity>, + @InjectWorkspaceRepository(CommentWorkspaceEntity) + private readonly commentRepository: WorkspaceRepository<CommentWorkspaceEntity>, ) {} // There is no need to validate the user's access to the workspace member since we don't have permission yet. @@ -26,16 +24,14 @@ export class WorkspaceMemberDeleteOnePreQueryHook workspaceId: string, payload: DeleteOneResolverArgs, ): Promise<void> { - const workspaceMemberId = payload.id; + const authorId = payload.id; - await this.attachmentRepository.deleteByAuthorId( - workspaceMemberId, - workspaceId, - ); + await this.attachmentRepository.delete({ + authorId, + }); - await this.commentRepository.deleteByAuthorId( - workspaceMemberId, - workspaceId, - ); + await this.commentRepository.delete({ + authorId, + }); } } diff --git a/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-query-hook.module.ts b/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-query-hook.module.ts index 14c1ef5e5384..051354c6becb 100644 --- a/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-query-hook.module.ts +++ b/packages/twenty-server/src/modules/workspace-member/query-hooks/workspace-member-query-hook.module.ts @@ -1,6 +1,6 @@ import { Module } from '@nestjs/common'; -import { ObjectMetadataRepositoryModule } from 'src/engine/object-metadata-repository/object-metadata-repository.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; import { CommentWorkspaceEntity } from 'src/modules/activity/standard-objects/comment.workspace-entity'; import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objects/attachment.workspace-entity'; import { WorkspaceMemberDeleteManyPreQueryHook } from 'src/modules/workspace-member/query-hooks/workspace-member-delete-many.pre-query.hook'; @@ -8,20 +8,14 @@ import { WorkspaceMemberDeleteOnePreQueryHook } from 'src/modules/workspace-memb @Module({ imports: [ - ObjectMetadataRepositoryModule.forFeature([ + TwentyORMModule.forFeature([ AttachmentWorkspaceEntity, CommentWorkspaceEntity, ]), ], providers: [ - { - provide: WorkspaceMemberDeleteOnePreQueryHook.name, - useClass: WorkspaceMemberDeleteOnePreQueryHook, - }, - { - provide: WorkspaceMemberDeleteManyPreQueryHook.name, - useClass: WorkspaceMemberDeleteManyPreQueryHook, - }, + WorkspaceMemberDeleteOnePreQueryHook, + WorkspaceMemberDeleteManyPreQueryHook, ], }) export class WorkspaceMemberQueryHookModule {} diff --git a/packages/twenty-server/src/modules/workspace-member/repositories/workspace-member.repository.ts b/packages/twenty-server/src/modules/workspace-member/repositories/workspace-member.repository.ts index c67c5e66fcec..0d3b5acc3bad 100644 --- a/packages/twenty-server/src/modules/workspace-member/repositories/workspace-member.repository.ts +++ b/packages/twenty-server/src/modules/workspace-member/repositories/workspace-member.repository.ts @@ -4,7 +4,6 @@ import { EntityManager } from 'typeorm'; import { WorkspaceDataSourceService } from 'src/engine/workspace-datasource/workspace-datasource.service'; import { WorkspaceMemberWorkspaceEntity } from 'src/modules/workspace-member/standard-objects/workspace-member.workspace-entity'; -import { ObjectRecord } from 'src/engine/workspace-manager/workspace-sync-metadata/types/object-record'; @Injectable() export class WorkspaceMemberRepository { @@ -12,22 +11,6 @@ export class WorkspaceMemberRepository { private readonly workspaceDataSourceService: WorkspaceDataSourceService, ) {} - public async getByIds( - userIds: string[], - workspaceId: string, - ): Promise<ObjectRecord<WorkspaceMemberWorkspaceEntity>[]> { - const dataSourceSchema = - this.workspaceDataSourceService.getSchemaName(workspaceId); - - const result = await this.workspaceDataSourceService.executeRawQuery( - `SELECT * FROM ${dataSourceSchema}."workspaceMember" WHERE "userId" = ANY($1)`, - [userIds], - workspaceId, - ); - - return result; - } - public async find(workspaceMemberId: string, workspaceId: string) { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -45,7 +28,7 @@ export class WorkspaceMemberRepository { public async getByIdOrFail( userId: string, workspaceId: string, - ): Promise<ObjectRecord<WorkspaceMemberWorkspaceEntity>> { + ): Promise<WorkspaceMemberWorkspaceEntity> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); @@ -68,7 +51,7 @@ export class WorkspaceMemberRepository { public async getAllByWorkspaceId( workspaceId: string, transactionManager?: EntityManager, - ): Promise<ObjectRecord<WorkspaceMemberWorkspaceEntity>[]> { + ): Promise<WorkspaceMemberWorkspaceEntity[]> { const dataSourceSchema = this.workspaceDataSourceService.getSchemaName(workspaceId); diff --git a/packages/twenty-server/src/modules/workspace-member/standard-objects/workspace-member.workspace-entity.ts b/packages/twenty-server/src/modules/workspace-member/standard-objects/workspace-member.workspace-entity.ts index 17378a598f60..9094a62b0541 100644 --- a/packages/twenty-server/src/modules/workspace-member/standard-objects/workspace-member.workspace-entity.ts +++ b/packages/twenty-server/src/modules/workspace-member/standard-objects/workspace-member.workspace-entity.ts @@ -6,26 +6,26 @@ import { RelationMetadataType, RelationOnDeleteAction, } from 'src/engine/metadata-modules/relation-metadata/relation-metadata.entity'; +import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; +import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; +import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; +import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; +import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; +import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; +import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { WORKSPACE_MEMBER_STANDARD_FIELD_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-field-ids'; import { STANDARD_OBJECT_IDS } from 'src/engine/workspace-manager/workspace-sync-metadata/constants/standard-object-ids'; import { ActivityWorkspaceEntity } from 'src/modules/activity/standard-objects/activity.workspace-entity'; -import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objects/attachment.workspace-entity'; -import { BlocklistWorkspaceEntity } from 'src/modules/connected-account/standard-objects/blocklist.workspace-entity'; -import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/standard-objects/calendar-event-participant.workspace-entity'; import { CommentWorkspaceEntity } from 'src/modules/activity/standard-objects/comment.workspace-entity'; +import { AttachmentWorkspaceEntity } from 'src/modules/attachment/standard-objects/attachment.workspace-entity'; +import { BlocklistWorkspaceEntity } from 'src/modules/blocklist/standard-objects/blocklist.workspace-entity'; +import { CalendarEventParticipantWorkspaceEntity } from 'src/modules/calendar/common/standard-objects/calendar-event-participant.workspace-entity'; import { CompanyWorkspaceEntity } from 'src/modules/company/standard-objects/company.workspace-entity'; import { ConnectedAccountWorkspaceEntity } from 'src/modules/connected-account/standard-objects/connected-account.workspace-entity'; import { FavoriteWorkspaceEntity } from 'src/modules/favorite/standard-objects/favorite.workspace-entity'; -import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; -import { AuditLogWorkspaceEntity } from 'src/modules/timeline/standard-objects/audit-log.workspace-entity'; -import { BaseWorkspaceEntity } from 'src/engine/twenty-orm/base.workspace-entity'; -import { WorkspaceEntity } from 'src/engine/twenty-orm/decorators/workspace-entity.decorator'; -import { WorkspaceIsSystem } from 'src/engine/twenty-orm/decorators/workspace-is-system.decorator'; -import { WorkspaceIsNotAuditLogged } from 'src/engine/twenty-orm/decorators/workspace-is-not-audit-logged.decorator'; -import { WorkspaceField } from 'src/engine/twenty-orm/decorators/workspace-field.decorator'; -import { WorkspaceIsNullable } from 'src/engine/twenty-orm/decorators/workspace-is-nullable.decorator'; -import { WorkspaceRelation } from 'src/engine/twenty-orm/decorators/workspace-relation.decorator'; import { MessageParticipantWorkspaceEntity } from 'src/modules/messaging/common/standard-objects/message-participant.workspace-entity'; +import { AuditLogWorkspaceEntity } from 'src/modules/timeline/standard-objects/audit-log.workspace-entity'; +import { TimelineActivityWorkspaceEntity } from 'src/modules/timeline/standard-objects/timeline-activity.workspace-entity'; @WorkspaceEntity({ standardId: STANDARD_OBJECT_IDS.workspaceMember, diff --git a/packages/twenty-server/src/queue-worker/queue-worker.module.ts b/packages/twenty-server/src/queue-worker/queue-worker.module.ts index d6c87b882383..1cf307ffefee 100644 --- a/packages/twenty-server/src/queue-worker/queue-worker.module.ts +++ b/packages/twenty-server/src/queue-worker/queue-worker.module.ts @@ -2,8 +2,17 @@ import { Module } from '@nestjs/common'; import { JobsModule } from 'src/engine/integrations/message-queue/jobs.module'; import { IntegrationsModule } from 'src/engine/integrations/integrations.module'; +import { TwentyORMModule } from 'src/engine/twenty-orm/twenty-orm.module'; +import { MessageQueueModule } from 'src/engine/integrations/message-queue/message-queue.module'; @Module({ - imports: [IntegrationsModule, JobsModule], + imports: [ + TwentyORMModule.register({ + workspaceEntities: ['dist/src/**/*.workspace-entity{.ts,.js}'], + }), + IntegrationsModule, + MessageQueueModule.registerExplorer(), + JobsModule, + ], }) export class QueueWorkerModule {} diff --git a/packages/twenty-server/src/queue-worker/queue-worker.ts b/packages/twenty-server/src/queue-worker/queue-worker.ts index fd62872d47bd..0b2af24eccd6 100644 --- a/packages/twenty-server/src/queue-worker/queue-worker.ts +++ b/packages/twenty-server/src/queue-worker/queue-worker.ts @@ -1,17 +1,8 @@ import { NestFactory } from '@nestjs/core'; -import { - MessageQueueJob, - MessageQueueJobData, -} from 'src/engine/integrations/message-queue/interfaces/message-queue-job.interface'; - import { shouldFilterException } from 'src/engine/utils/global-exception-handler.util'; import { ExceptionHandlerService } from 'src/engine/integrations/exception-handler/exception-handler.service'; import { LoggerService } from 'src/engine/integrations/logger/logger.service'; -import { JobsModule } from 'src/engine/integrations/message-queue/jobs.module'; -import { MessageQueue } from 'src/engine/integrations/message-queue/message-queue.constants'; -import { MessageQueueService } from 'src/engine/integrations/message-queue/services/message-queue.service'; -import { getJobClassName } from 'src/engine/integrations/message-queue/utils/get-job-class-name.util'; import { QueueWorkerModule } from 'src/queue-worker/queue-worker.module'; async function bootstrap() { @@ -28,29 +19,6 @@ async function bootstrap() { // Inject our logger app.useLogger(loggerService!); - - for (const queueName of Object.values(MessageQueue)) { - const messageQueueService: MessageQueueService = app.get(queueName); - - await messageQueueService.work(async (jobData: MessageQueueJobData) => { - const jobClassName = getJobClassName(jobData.name); - const job: MessageQueueJob<MessageQueueJobData> = app - .select(JobsModule) - .get(jobClassName, { strict: false }); - - try { - await job.handle(jobData.data); - } catch (err) { - exceptionHandlerService?.captureExceptions([ - new Error( - `Error occurred while processing job ${jobClassName} #${jobData.id}`, - ), - err, - ]); - throw err; - } - }); - } } catch (err) { loggerService?.error(err?.message, err?.name); diff --git a/packages/twenty-server/src/utils/custom-exception.ts b/packages/twenty-server/src/utils/custom-exception.ts new file mode 100644 index 000000000000..e12d8840c7b3 --- /dev/null +++ b/packages/twenty-server/src/utils/custom-exception.ts @@ -0,0 +1,8 @@ +export class CustomException extends Error { + code: string; + + constructor(message: string, code: string) { + super(message); + this.code = code; + } +} diff --git a/packages/twenty-server/src/utils/is-defined.ts b/packages/twenty-server/src/utils/is-defined.ts new file mode 100644 index 000000000000..1be478d76dd0 --- /dev/null +++ b/packages/twenty-server/src/utils/is-defined.ts @@ -0,0 +1,3 @@ +export const isDefined = <T>(value: T | null | undefined): value is T => { + return value !== null && value !== undefined; +}; diff --git a/packages/twenty-server/src/utils/is-group-email.ts b/packages/twenty-server/src/utils/is-group-email.ts new file mode 100644 index 000000000000..748495138b3e --- /dev/null +++ b/packages/twenty-server/src/utils/is-group-email.ts @@ -0,0 +1,6 @@ +export const isGroupEmail = (email: string): boolean => { + const isGroupPattern = + /noreply|no-reply|do_not_reply|no\.reply|^(info@|contact@|hello@|support@|feedback@|service@|help@|invites@|invite@|welcome@|alerts@|team@|notifications@|notification@|news@)/; + + return isGroupPattern.test(email); +}; diff --git a/packages/twenty-server/src/utils/typed-reflect.ts b/packages/twenty-server/src/utils/typed-reflect.ts index 6e9fdb819385..5abec76b8401 100644 --- a/packages/twenty-server/src/utils/typed-reflect.ts +++ b/packages/twenty-server/src/utils/typed-reflect.ts @@ -8,6 +8,7 @@ export interface ReflectMetadataTypeMap { ['workspace:is-system-metadata-args']: true; ['workspace:is-audit-logged-metadata-args']: false; ['workspace:is-primary-field-metadata-args']: true; + ['workspace:is-deprecated-field-metadata-args']: true; } export class TypedReflect { diff --git a/packages/twenty-server/test/company.e2e-spec.ts b/packages/twenty-server/test/company.e2e-spec.ts index 40658d931e3f..1a52998601d8 100644 --- a/packages/twenty-server/test/company.e2e-spec.ts +++ b/packages/twenty-server/test/company.e2e-spec.ts @@ -31,7 +31,9 @@ describe('CompanyResolver (e2e)', () => { id name domainName - address + address { + addressCity + } } } `, @@ -39,7 +41,7 @@ describe('CompanyResolver (e2e)', () => { data: { name: 'New Company', domainName: 'new-company.com', - address: 'New Address', + address: { addressCity: 'Paris' }, }, }, }; @@ -57,7 +59,7 @@ describe('CompanyResolver (e2e)', () => { expect(data).toHaveProperty('id'); expect(data).toHaveProperty('name', 'New Company'); expect(data).toHaveProperty('domainName', 'new-company.com'); - expect(data).toHaveProperty('address', 'New Address'); + expect(data).toHaveProperty('address', { addressCity: 'Paris' }); }); }); @@ -69,7 +71,9 @@ describe('CompanyResolver (e2e)', () => { id name domainName - address + address { + addressCity + } } } `, @@ -92,7 +96,7 @@ describe('CompanyResolver (e2e)', () => { expect(company).toHaveProperty('id'); expect(company).toHaveProperty('name', 'New Company'); expect(company).toHaveProperty('domainName', 'new-company.com'); - expect(company).toHaveProperty('address', 'New Address'); + expect(company).toHaveProperty('address', { addressCity: 'Paris' }); // Check if we have access to ressources outside of our workspace const instagramCompany = data.find((c) => c.name === 'Instagram'); @@ -109,7 +113,9 @@ describe('CompanyResolver (e2e)', () => { id name domainName - address + address { + addressCity + } } } `, @@ -131,7 +137,7 @@ describe('CompanyResolver (e2e)', () => { expect(data).toHaveProperty('id'); expect(data).toHaveProperty('name', 'New Company'); expect(data).toHaveProperty('domainName', 'new-company.com'); - expect(data).toHaveProperty('address', 'New Address'); + expect(data).toHaveProperty('address', { addressCity: 'Paris' }); }); }); @@ -143,7 +149,9 @@ describe('CompanyResolver (e2e)', () => { id name domainName - address + address { + addressCity + } } } `, @@ -175,7 +183,9 @@ describe('CompanyResolver (e2e)', () => { id name domainName - address + address { + addressCity + } } } `, @@ -186,7 +196,7 @@ describe('CompanyResolver (e2e)', () => { data: { name: 'Updated Company', domainName: 'updated-company.com', - address: 'Updated Address', + address: { addressCity: 'Updated City' }, }, }, }; @@ -202,7 +212,7 @@ describe('CompanyResolver (e2e)', () => { expect(data).toHaveProperty('id'); expect(data).toHaveProperty('name', 'Updated Company'); expect(data).toHaveProperty('domainName', 'updated-company.com'); - expect(data).toHaveProperty('address', 'Updated Address'); + expect(data).toHaveProperty('address', { addressCity: 'Updated City' }); }); }); @@ -214,7 +224,9 @@ describe('CompanyResolver (e2e)', () => { id name domainName - address + address { + addressCity + } } } `, diff --git a/packages/twenty-ui/.storybook/main.ts b/packages/twenty-ui/.storybook/main.ts index a27e1bccf947..270259adfdd7 100644 --- a/packages/twenty-ui/.storybook/main.ts +++ b/packages/twenty-ui/.storybook/main.ts @@ -16,15 +16,6 @@ const config: StorybookConfig = { name: '@storybook/react-vite', options: {}, }, - build: { - test: { - disableMDXEntries: true, - disabledAddons: [ - '@storybook/addon-docs', - '@storybook/addon-essentials/docs', - ], - }, - }, }; export default config; diff --git a/packages/twenty-ui/package.json b/packages/twenty-ui/package.json index 06c014e3d64a..a0788e34553f 100644 --- a/packages/twenty-ui/package.json +++ b/packages/twenty-ui/package.json @@ -1,6 +1,6 @@ { "name": "twenty-ui", - "version": "0.20.0", + "version": "0.22.0", "type": "module", "main": "./src/index.ts", "exports": { diff --git a/packages/twenty-ui/project.json b/packages/twenty-ui/project.json index 641a15fae554..951e50cd78cb 100644 --- a/packages/twenty-ui/project.json +++ b/packages/twenty-ui/project.json @@ -43,10 +43,10 @@ "test": {} } }, - "storybook:dev": { + "storybook:serve:dev": { "options": { "port": 6007 } }, - "storybook:static": { + "storybook:serve:static": { "options": { "buildTarget": "twenty-ui:storybook:build", "port": 6007 @@ -59,7 +59,7 @@ "storybook:test": { "options": { "port": 6007 } }, - "storybook:static:test": { + "storybook:serve-and-test:static": { "options": { "port": 6007 } } } diff --git a/packages/twenty-ui/src/display/avatar/components/Avatar.module.css b/packages/twenty-ui/src/display/avatar/components/Avatar.module.css deleted file mode 100644 index ed560dead2b6..000000000000 --- a/packages/twenty-ui/src/display/avatar/components/Avatar.module.css +++ /dev/null @@ -1,23 +0,0 @@ -.avatar { - align-items: center; - border-radius: 2px; - display: flex; - flex-shrink: 0; - justify-content: center; - overflow: hidden; - user-select: none; -} - -.rounded { - border-radius: 50%; -} - -.avatar-on-click:hover { - box-shadow: 0 0 0 4px var(--twentycrm-background-transparent-light); -} - -.avatar-image { - object-fit: cover; - width: 100%; - height: 100%; -} \ No newline at end of file diff --git a/packages/twenty-ui/src/display/avatar/components/Avatar.tsx b/packages/twenty-ui/src/display/avatar/components/Avatar.tsx index d7024b347fd8..b0373b8cb8e5 100644 --- a/packages/twenty-ui/src/display/avatar/components/Avatar.tsx +++ b/packages/twenty-ui/src/display/avatar/components/Avatar.tsx @@ -1,104 +1,112 @@ -import { useState } from 'react'; +import { styled } from '@linaria/react'; import { isNonEmptyString, isUndefined } from '@sniptt/guards'; -import clsx from 'clsx'; +import { useContext } from 'react'; +import { useRecoilState } from 'recoil'; +import { invalidAvatarUrlsState } from '@ui/display/avatar/components/states/isInvalidAvatarUrlState'; +import { AVATAR_PROPERTIES_BY_SIZE } from '@ui/display/avatar/constants/AvatarPropertiesBySize'; +import { AvatarSize } from '@ui/display/avatar/types/AvatarSize'; +import { AvatarType } from '@ui/display/avatar/types/AvatarType'; +import { ThemeContext } from '@ui/theme'; import { Nullable, stringToHslColor } from '@ui/utilities'; -import styles from './Avatar.module.css'; +const StyledAvatar = styled.div<{ + size: AvatarSize; + rounded?: boolean; + clickable?: boolean; + color: string; + backgroundColor: string; + backgroundTransparentLight: string; +}>` + align-items: center; + flex-shrink: 0; + overflow: hidden; + user-select: none; -export type AvatarType = 'squared' | 'rounded'; + border-radius: ${({ rounded }) => (rounded ? '50%' : '2px')}; + display: flex; + font-size: ${({ size }) => AVATAR_PROPERTIES_BY_SIZE[size].fontSize}; + height: ${({ size }) => AVATAR_PROPERTIES_BY_SIZE[size].width}; + justify-content: center; -export type AvatarSize = 'xl' | 'lg' | 'md' | 'sm' | 'xs'; + width: ${({ size }) => AVATAR_PROPERTIES_BY_SIZE[size].width}; + + color: ${({ color }) => color}; + background: ${({ backgroundColor }) => backgroundColor}; + + &:hover { + box-shadow: ${({ clickable, backgroundTransparentLight }) => + clickable ? `0 0 0 4px ${backgroundTransparentLight}` : 'none'}; + } +`; +const StyledImage = styled.img` + height: 100%; + object-fit: cover; + width: 100%; +`; export type AvatarProps = { avatarUrl?: string | null; className?: string; size?: AvatarSize; placeholder: string | undefined; - entityId?: string; + placeholderColorSeed?: string; type?: Nullable<AvatarType>; color?: string; backgroundColor?: string; onClick?: () => void; }; -const propertiesBySize = { - xl: { - fontSize: '16px', - width: '40px', - }, - lg: { - fontSize: '13px', - width: '24px', - }, - md: { - fontSize: '12px', - width: '16px', - }, - sm: { - fontSize: '10px', - width: '14px', - }, - xs: { - fontSize: '8px', - width: '12px', - }, -}; - +// TODO: Remove recoil because we don't want it into twenty-ui and find a solution for invalid avatar urls export const Avatar = ({ avatarUrl, size = 'md', placeholder, - entityId = placeholder, + placeholderColorSeed = placeholder, onClick, type = 'squared', color, backgroundColor, }: AvatarProps) => { - const [isInvalidAvatarUrl, setIsInvalidAvatarUrl] = useState(false); + const { theme } = useContext(ThemeContext); + const [invalidAvatarUrls, setInvalidAvatarUrls] = useRecoilState( + invalidAvatarUrlsState, + ); const noAvatarUrl = !isNonEmptyString(avatarUrl); const placeholderChar = placeholder?.[0]?.toLocaleUpperCase(); - const showPlaceholder = noAvatarUrl || isInvalidAvatarUrl; + const showPlaceholder = noAvatarUrl || invalidAvatarUrls.includes(avatarUrl); const handleImageError = () => { - setIsInvalidAvatarUrl(true); + if (isNonEmptyString(avatarUrl)) { + setInvalidAvatarUrls((prev) => [...prev, avatarUrl]); + } }; - const fixedColor = color ?? stringToHslColor(entityId ?? '', 75, 25); + const fixedColor = + color ?? stringToHslColor(placeholderColorSeed ?? '', 75, 25); const fixedBackgroundColor = - backgroundColor ?? stringToHslColor(entityId ?? '', 75, 85); + backgroundColor ?? stringToHslColor(placeholderColorSeed ?? '', 75, 85); const showBackgroundColor = showPlaceholder; return ( - <div - className={clsx({ - [styles.avatar]: true, - [styles.rounded]: type === 'rounded', - [styles.avatarOnClick]: !isUndefined(onClick), - })} + <StyledAvatar + size={size} + backgroundColor={showBackgroundColor ? fixedBackgroundColor : 'none'} + color={fixedColor} + clickable={!isUndefined(onClick)} + rounded={type === 'rounded'} onClick={onClick} - style={{ - color: fixedColor, - backgroundColor: showBackgroundColor ? fixedBackgroundColor : 'none', - width: propertiesBySize[size].width, - height: propertiesBySize[size].width, - fontSize: propertiesBySize[size].fontSize, - }} + backgroundTransparentLight={theme.background.transparent.light} > {showPlaceholder ? ( placeholderChar ) : ( - <img - src={avatarUrl} - className={styles.avatarImage} - onError={handleImageError} - alt="" - /> + <StyledImage src={avatarUrl} onError={handleImageError} alt="" /> )} - </div> + </StyledAvatar> ); }; diff --git a/packages/twenty-ui/src/display/avatar/components/__stories__/AvatarGroup.stories.tsx b/packages/twenty-ui/src/display/avatar/components/__stories__/AvatarGroup.stories.tsx index 18ad3c3d4d58..e3792988dc5a 100644 --- a/packages/twenty-ui/src/display/avatar/components/__stories__/AvatarGroup.stories.tsx +++ b/packages/twenty-ui/src/display/avatar/components/__stories__/AvatarGroup.stories.tsx @@ -1,11 +1,8 @@ import { Meta, StoryObj } from '@storybook/react'; -import { - Avatar, - AvatarProps, - AvatarSize, - AvatarType, -} from '@ui/display/avatar/components/Avatar'; +import { Avatar, AvatarProps } from '@ui/display/avatar/components/Avatar'; +import { AvatarSize } from '@ui/display/avatar/types/AvatarSize'; +import { AvatarType } from '@ui/display/avatar/types/AvatarType'; import { AVATAR_URL_MOCK, CatalogDecorator, @@ -16,7 +13,7 @@ import { AvatarGroup, AvatarGroupProps } from '../AvatarGroup'; const makeAvatar = (userName: string, props: Partial<AvatarProps> = {}) => ( // eslint-disable-next-line react/jsx-props-no-spreading - <Avatar placeholder={userName} entityId={userName} {...props} /> + <Avatar placeholder={userName} placeholderColorSeed={userName} {...props} /> ); const getAvatars = (commonProps: Partial<AvatarProps> = {}) => [ diff --git a/packages/twenty-ui/src/display/avatar/components/states/isInvalidAvatarUrlState.ts b/packages/twenty-ui/src/display/avatar/components/states/isInvalidAvatarUrlState.ts new file mode 100644 index 000000000000..13d410a6fa4e --- /dev/null +++ b/packages/twenty-ui/src/display/avatar/components/states/isInvalidAvatarUrlState.ts @@ -0,0 +1,6 @@ +import { atom } from 'recoil'; + +export const invalidAvatarUrlsState = atom<string[]>({ + key: 'invalidAvatarUrlsState', + default: [], +}); diff --git a/packages/twenty-ui/src/display/avatar/constants/AvatarPropertiesBySize.ts b/packages/twenty-ui/src/display/avatar/constants/AvatarPropertiesBySize.ts new file mode 100644 index 000000000000..e44c170cc269 --- /dev/null +++ b/packages/twenty-ui/src/display/avatar/constants/AvatarPropertiesBySize.ts @@ -0,0 +1,22 @@ +export const AVATAR_PROPERTIES_BY_SIZE = { + xl: { + fontSize: '16px', + width: '40px', + }, + lg: { + fontSize: '13px', + width: '24px', + }, + md: { + fontSize: '12px', + width: '16px', + }, + sm: { + fontSize: '10px', + width: '14px', + }, + xs: { + fontSize: '8px', + width: '12px', + }, +}; diff --git a/packages/twenty-ui/src/display/avatar/types/AvatarSize.ts b/packages/twenty-ui/src/display/avatar/types/AvatarSize.ts new file mode 100644 index 000000000000..db1915e1718b --- /dev/null +++ b/packages/twenty-ui/src/display/avatar/types/AvatarSize.ts @@ -0,0 +1 @@ +export type AvatarSize = 'xl' | 'lg' | 'md' | 'sm' | 'xs'; diff --git a/packages/twenty-ui/src/display/avatar/types/AvatarType.ts b/packages/twenty-ui/src/display/avatar/types/AvatarType.ts new file mode 100644 index 000000000000..9e9b7dc9589e --- /dev/null +++ b/packages/twenty-ui/src/display/avatar/types/AvatarType.ts @@ -0,0 +1 @@ +export type AvatarType = 'squared' | 'rounded'; diff --git a/packages/twenty-ui/src/display/chip/components/EntityChip.tsx b/packages/twenty-ui/src/display/chip/components/AvatarChip.tsx similarity index 54% rename from packages/twenty-ui/src/display/chip/components/EntityChip.tsx rename to packages/twenty-ui/src/display/chip/components/AvatarChip.tsx index f17ee4108935..89c4bdabef11 100644 --- a/packages/twenty-ui/src/display/chip/components/EntityChip.tsx +++ b/packages/twenty-ui/src/display/chip/components/AvatarChip.tsx @@ -1,55 +1,47 @@ -import * as React from 'react'; -import { useNavigate } from 'react-router-dom'; import { useTheme } from '@emotion/react'; -import { isNonEmptyString } from '@sniptt/guards'; -import { Avatar, AvatarType } from '@ui/display/avatar/components/Avatar'; +import { Avatar } from '@ui/display/avatar/components/Avatar'; +import { AvatarType } from '@ui/display/avatar/types/AvatarType'; import { Chip, ChipVariant } from '@ui/display/chip/components/Chip'; import { IconComponent } from '@ui/display/icon/types/IconComponent'; +import { isDefined } from '@ui/utilities/isDefined'; import { Nullable } from '@ui/utilities/types/Nullable'; +import { MouseEvent } from 'react'; -export type EntityChipProps = { - linkToEntity?: string; - entityId: string; +export type AvatarChipProps = { name: string; avatarUrl?: string; avatarType?: Nullable<AvatarType>; - variant?: EntityChipVariant; + variant?: AvatarChipVariant; LeftIcon?: IconComponent; className?: string; + placeholderColorSeed?: string; + onClick?: (event: MouseEvent) => void; }; -export enum EntityChipVariant { +export enum AvatarChipVariant { Regular = 'regular', Transparent = 'transparent', } -export const EntityChip = ({ - linkToEntity, - entityId, +export const AvatarChip = ({ name, avatarUrl, avatarType = 'rounded', - variant = EntityChipVariant.Regular, + variant = AvatarChipVariant.Regular, LeftIcon, className, -}: EntityChipProps) => { - const navigate = useNavigate(); + placeholderColorSeed, + onClick, +}: AvatarChipProps) => { const theme = useTheme(); - const handleLinkClick = (event: React.MouseEvent<HTMLDivElement>) => { - if (isNonEmptyString(linkToEntity)) { - event.stopPropagation(); - navigate(linkToEntity); - } - }; - return ( <Chip label={name} variant={ - linkToEntity - ? variant === EntityChipVariant.Regular + isDefined(onClick) + ? variant === AvatarChipVariant.Regular ? ChipVariant.Highlighted : ChipVariant.Regular : ChipVariant.Transparent @@ -60,15 +52,15 @@ export const EntityChip = ({ ) : ( <Avatar avatarUrl={avatarUrl} - entityId={entityId} + placeholderColorSeed={placeholderColorSeed} placeholder={name} size="sm" type={avatarType} /> ) } - clickable={!!linkToEntity} - onClick={handleLinkClick} + clickable={isDefined(onClick)} + onClick={onClick} className={className} /> ); diff --git a/packages/twenty-ui/src/display/chip/components/__stories__/EntityChip.stories.tsx b/packages/twenty-ui/src/display/chip/components/__stories__/EntityChip.stories.tsx index 254fceef4018..2682eec0409e 100644 --- a/packages/twenty-ui/src/display/chip/components/__stories__/EntityChip.stories.tsx +++ b/packages/twenty-ui/src/display/chip/components/__stories__/EntityChip.stories.tsx @@ -1,21 +1,19 @@ import { Meta, StoryObj } from '@storybook/react'; +import { AvatarChip } from '@ui/display/chip/components/AvatarChip'; import { ComponentDecorator, RouterDecorator } from '@ui/testing'; -import { EntityChip } from '../EntityChip'; - -const meta: Meta<typeof EntityChip> = { - title: 'UI/Display/Chip/EntityChip', - component: EntityChip, +const meta: Meta<typeof AvatarChip> = { + title: 'UI/Display/Chip/AvatarChip', + component: AvatarChip, decorators: [RouterDecorator, ComponentDecorator], args: { name: 'Entity name', - linkToEntity: '/entity-link', avatarType: 'squared', }, }; export default meta; -type Story = StoryObj<typeof EntityChip>; +type Story = StoryObj<typeof AvatarChip>; export const Default: Story = {}; diff --git a/packages/twenty-ui/src/display/icon/components/IconTwentyStarFilled.tsx b/packages/twenty-ui/src/display/icon/components/IconTwentyStarFilled.tsx index 465483017156..c82ec61041c9 100644 --- a/packages/twenty-ui/src/display/icon/components/IconTwentyStarFilled.tsx +++ b/packages/twenty-ui/src/display/icon/components/IconTwentyStarFilled.tsx @@ -1,14 +1,14 @@ -import { useTheme } from '@emotion/react'; - import IconTwentyStarFilledRaw from '@ui/display/icon/assets/twenty-star-filled.svg?react'; import { IconComponentProps } from '@ui/display/icon/types/IconComponent'; +import { THEME_COMMON } from '@ui/theme'; type IconTwentyStarFilledProps = Pick<IconComponentProps, 'size' | 'stroke'>; +const iconStrokeMd = THEME_COMMON.icon.stroke.md; + export const IconTwentyStarFilled = (props: IconTwentyStarFilledProps) => { - const theme = useTheme(); const size = props.size ?? 24; - const stroke = props.stroke ?? theme.icon.stroke.md; + const stroke = props.stroke ?? iconStrokeMd; return ( <IconTwentyStarFilledRaw height={size} width={size} strokeWidth={stroke} /> diff --git a/packages/twenty-ui/src/display/icon/components/TablerIcons.ts b/packages/twenty-ui/src/display/icon/components/TablerIcons.ts index 2855731a344b..0ba89be2f7e2 100644 --- a/packages/twenty-ui/src/display/icon/components/TablerIcons.ts +++ b/packages/twenty-ui/src/display/icon/components/TablerIcons.ts @@ -33,6 +33,7 @@ export { IconCalendarEvent, IconCalendarTime, IconCalendarX, + IconChartCandle, IconCheck, IconCheckbox, IconChevronDown, @@ -52,6 +53,7 @@ export { IconMessageCircle as IconComment, IconCopy, IconCreditCard, + IconCurrencyBaht, IconCurrencyDirham, IconCurrencyDollar, IconCurrencyEuro, @@ -59,7 +61,9 @@ export { IconCurrencyKroneCzech, IconCurrencyKroneSwedish, IconCurrencyPound, + IconCurrencyReal, IconCurrencyRiyal, + IconCurrencyWon, IconCurrencyYen, IconCurrencyYuan, IconDatabase, @@ -134,10 +138,13 @@ export { IconReload, IconRepeat, IconRocket, + IconRotate, IconSearch, IconSend, IconSettings, IconSortDescending, + IconSparkles, + IconSql, IconSquareRoundedCheck, IconTable, IconTag, diff --git a/packages/twenty-ui/src/display/index.ts b/packages/twenty-ui/src/display/index.ts index 74833f5db945..fa961ff5abf3 100644 --- a/packages/twenty-ui/src/display/index.ts +++ b/packages/twenty-ui/src/display/index.ts @@ -1,9 +1,13 @@ export * from './avatar/components/Avatar'; export * from './avatar/components/AvatarGroup'; +export * from './avatar/components/states/isInvalidAvatarUrlState'; +export * from './avatar/constants/AvatarPropertiesBySize'; +export * from './avatar/types/AvatarSize'; +export * from './avatar/types/AvatarType'; export * from './checkmark/components/AnimatedCheckmark'; export * from './checkmark/components/Checkmark'; +export * from './chip/components/AvatarChip'; export * from './chip/components/Chip'; -export * from './chip/components/EntityChip'; export * from './color/components/ColorSample'; export * from './icon/components/IconAddressBook'; export * from './icon/components/IconGmail'; diff --git a/packages/twenty-ui/src/display/tag/components/Tag.tsx b/packages/twenty-ui/src/display/tag/components/Tag.tsx index 3b2cb67b7354..1ed0dc1dd07f 100644 --- a/packages/twenty-ui/src/display/tag/components/Tag.tsx +++ b/packages/twenty-ui/src/display/tag/components/Tag.tsx @@ -1,5 +1,5 @@ -import { useContext } from 'react'; import { styled } from '@linaria/react'; +import { useContext } from 'react'; import { IconComponent, OverflowingTextWithTooltip } from '@ui/display'; import { @@ -16,14 +16,17 @@ const spacing1 = THEME_COMMON.spacing(1); const StyledTag = styled.h3<{ theme: ThemeType; - color: ThemeColor; + color: TagColor; weight: TagWeight; + variant: TagVariant; preventShrink?: boolean; }>` align-items: center; - background: ${({ color, theme }) => theme.tag.background[color]}; + background: ${({ color, theme }) => + color === 'transparent' ? color : theme.tag.background[color]}; border-radius: ${BORDER_COMMON.radius.sm}; - color: ${({ color, theme }) => theme.tag.text[color]}; + color: ${({ color, theme }) => + color === 'transparent' ? theme.tag.text['gray'] : theme.tag.text[color]}; display: inline-flex; font-size: ${({ theme }) => theme.font.size.md}; font-style: normal; @@ -35,6 +38,8 @@ const StyledTag = styled.h3<{ margin: 0; overflow: hidden; padding: 0 ${spacing2}; + border: ${({ variant, theme }) => + variant === 'outline' ? `2px dashed ${theme.tag.background['gray']}` : ''}; gap: ${spacing1}; @@ -58,14 +63,17 @@ const StyledIconContainer = styled.div` `; type TagWeight = 'regular' | 'medium'; +type TagVariant = 'solid' | 'outline'; +type TagColor = ThemeColor | 'transparent'; type TagProps = { className?: string; - color: ThemeColor; + color: TagColor; text: string; Icon?: IconComponent; onClick?: () => void; weight?: TagWeight; + variant?: TagVariant; preventShrink?: boolean; }; @@ -77,6 +85,7 @@ export const Tag = ({ Icon, onClick, weight = 'regular', + variant = 'solid', preventShrink, }: TagProps) => { const { theme } = useContext(ThemeContext); @@ -88,6 +97,7 @@ export const Tag = ({ color={color} onClick={onClick} weight={weight} + variant={variant} preventShrink={preventShrink} > {!!Icon && ( diff --git a/packages/twenty-ui/src/display/tooltip/AppTooltip.tsx b/packages/twenty-ui/src/display/tooltip/AppTooltip.tsx index e3f126f5a882..e0a00d2750e2 100644 --- a/packages/twenty-ui/src/display/tooltip/AppTooltip.tsx +++ b/packages/twenty-ui/src/display/tooltip/AppTooltip.tsx @@ -10,6 +10,12 @@ export enum TooltipPosition { Bottom = 'bottom', } +export enum TooltipDelay { + noDelay = '0ms', + shortDelay = '300ms', + mediumDelay = '500ms', +} + const StyledAppTooltip = styled(Tooltip)` backdrop-filter: ${({ theme }) => theme.blur.strong}; background-color: ${({ theme }) => RGBA(theme.color.gray80, 0.8)}; @@ -36,38 +42,51 @@ export type AppTooltipProps = { anchorSelect?: string; content?: string; children?: React.ReactNode; - delayHide?: number; offset?: number; noArrow?: boolean; isOpen?: boolean; place?: PlacesType; + delay?: TooltipDelay; positionStrategy?: PositionStrategy; + clickable?: boolean; }; export const AppTooltip = ({ anchorSelect, className, content, - delayHide, isOpen, noArrow, offset, + delay = TooltipDelay.mediumDelay, place, positionStrategy, children, -}: AppTooltipProps) => ( - <StyledAppTooltip - {...{ - anchorSelect, - className, - content, - delayHide, - isOpen, - noArrow, - offset, - place, - positionStrategy, - children, - }} - /> -); + clickable, +}: AppTooltipProps) => { + const delayInMs = + delay === TooltipDelay.noDelay + ? 0 + : delay === TooltipDelay.shortDelay + ? 300 + : 500; + + return ( + <StyledAppTooltip + {...{ + anchorSelect, + className, + content, + delayShow: delayInMs, + delayHide: delayInMs, + isOpen, + noArrow, + offset, + place, + positionStrategy, + children, + clickable, + }} + /> + ); +}; diff --git a/packages/twenty-ui/src/display/tooltip/OverflowingTextWithTooltip.module.css b/packages/twenty-ui/src/display/tooltip/OverflowingTextWithTooltip.module.css deleted file mode 100644 index 2df2dca40bc7..000000000000 --- a/packages/twenty-ui/src/display/tooltip/OverflowingTextWithTooltip.module.css +++ /dev/null @@ -1,20 +0,0 @@ -.main { - font-family: inherit; - font-size: inherit; - - font-weight: inherit; - max-width: 100%; - overflow: hidden; - text-decoration: inherit; - - text-overflow: ellipsis; - white-space: nowrap; -} - -.cursor { - cursor: pointer; -} - -.large { - height: calc(var(--twentycrm-spacing-multiplicator) * 4px); -} \ No newline at end of file diff --git a/packages/twenty-ui/src/display/tooltip/OverflowingTextWithTooltip.tsx b/packages/twenty-ui/src/display/tooltip/OverflowingTextWithTooltip.tsx index 0fbd883f5331..ed0580e9fd65 100644 --- a/packages/twenty-ui/src/display/tooltip/OverflowingTextWithTooltip.tsx +++ b/packages/twenty-ui/src/display/tooltip/OverflowingTextWithTooltip.tsx @@ -4,7 +4,7 @@ import { styled } from '@linaria/react'; import { THEME_COMMON } from '@ui/theme'; -import { AppTooltip } from './AppTooltip'; +import { AppTooltip, TooltipDelay } from './AppTooltip'; const spacing4 = THEME_COMMON.spacing(4); @@ -87,12 +87,12 @@ export const OverflowingTextWithTooltip = ({ <AppTooltip anchorSelect={`#${textElementId}`} content={mutliline ? undefined : text ?? ''} - delayHide={1} offset={5} isOpen noArrow place="bottom" positionStrategy="absolute" + delay={TooltipDelay.mediumDelay} > {mutliline ? <pre>{text}</pre> : ''} </AppTooltip> diff --git a/packages/twenty-ui/src/display/tooltip/__stories__/Tooltip.stories.tsx b/packages/twenty-ui/src/display/tooltip/__stories__/Tooltip.stories.tsx index 7ce9997ab049..3b46a1fc005b 100644 --- a/packages/twenty-ui/src/display/tooltip/__stories__/Tooltip.stories.tsx +++ b/packages/twenty-ui/src/display/tooltip/__stories__/Tooltip.stories.tsx @@ -6,7 +6,11 @@ import { ComponentDecorator, } from '@ui/testing'; -import { AppTooltip as Tooltip, TooltipPosition } from '../AppTooltip'; +import { + AppTooltip as Tooltip, + TooltipDelay, + TooltipPosition, +} from '../AppTooltip'; const meta: Meta<typeof Tooltip> = { title: 'UI/Display/Tooltip', @@ -19,6 +23,7 @@ type Story = StoryObj<typeof Tooltip>; export const Default: Story = { args: { place: TooltipPosition.Bottom, + delay: TooltipDelay.mediumDelay, content: 'Tooltip Test', isOpen: true, anchorSelect: '#hover-text', @@ -28,12 +33,13 @@ export const Default: Story = { anchorSelect, className, content, - delayHide, + delay, isOpen, noArrow, offset, place, positionStrategy, + clickable, }) => ( <> <p id="hover-text" data-testid="tooltip"> @@ -44,12 +50,52 @@ export const Default: Story = { anchorSelect, className, content, - delayHide, + delay, isOpen, noArrow, offset, place, positionStrategy, + clickable, + }} + /> + </> + ), +}; + +export const Hoverable: Story = { + args: { + place: TooltipPosition.Bottom, + delay: TooltipDelay.mediumDelay, + content: 'Tooltip Test', + isOpen: true, + anchorSelect: '#hover-text', + }, + decorators: [ComponentDecorator], + render: ({ + anchorSelect, + className, + content, + delay, + noArrow, + offset, + place, + positionStrategy, + }) => ( + <> + <p id="hover-text" data-testid="tooltip"> + Hover me! + </p> + <Tooltip + {...{ + anchorSelect, + className, + content, + delay, + noArrow, + offset, + place, + positionStrategy, }} /> </> diff --git a/packages/twenty-ui/src/theme/constants/BorderCommon.ts b/packages/twenty-ui/src/theme/constants/BorderCommon.ts index ca10ce0bab5e..a68f017bc44c 100644 --- a/packages/twenty-ui/src/theme/constants/BorderCommon.ts +++ b/packages/twenty-ui/src/theme/constants/BorderCommon.ts @@ -1,7 +1,7 @@ export const BORDER_COMMON = { radius: { xs: '2px', - sm: 'var(--twentycrm-border-radius-sm)', + sm: '4px', md: '8px', xl: '20px', pill: '999px', diff --git a/packages/twenty-ui/src/theme/constants/FontCommon.ts b/packages/twenty-ui/src/theme/constants/FontCommon.ts index 6d14f8d1831d..65556c018d38 100644 --- a/packages/twenty-ui/src/theme/constants/FontCommon.ts +++ b/packages/twenty-ui/src/theme/constants/FontCommon.ts @@ -10,7 +10,7 @@ export const FONT_COMMON = { }, weight: { regular: 400, - medium: 'var(--twentycrm-font-weight-medium)', + medium: 500, semiBold: 600, }, family: 'Inter, sans-serif', diff --git a/packages/twenty-ui/src/theme/provider/ThemeProvider.tsx b/packages/twenty-ui/src/theme/provider/ThemeProvider.tsx index 1282213f8f45..3b513f34e1bf 100644 --- a/packages/twenty-ui/src/theme/provider/ThemeProvider.tsx +++ b/packages/twenty-ui/src/theme/provider/ThemeProvider.tsx @@ -1,23 +1,16 @@ -import { ReactNode, useEffect } from 'react'; +import { ReactNode } from 'react'; import { ThemeProvider as EmotionThemeProvider } from '@emotion/react'; import { ThemeContextProvider } from '@ui/theme/provider/ThemeContextProvider'; import { ThemeType } from '..'; -import './theme.css'; - type ThemeProviderProps = { theme: ThemeType; children: ReactNode; }; const ThemeProvider = ({ theme, children }: ThemeProviderProps) => { - useEffect(() => { - document.documentElement.className = - theme.name === 'dark' ? 'dark' : 'light'; - }, [theme]); - return ( <EmotionThemeProvider theme={theme}> <ThemeContextProvider theme={theme}>{children}</ThemeContextProvider> diff --git a/packages/twenty-ui/src/theme/provider/theme.css b/packages/twenty-ui/src/theme/provider/theme.css deleted file mode 100644 index 076090f6370f..000000000000 --- a/packages/twenty-ui/src/theme/provider/theme.css +++ /dev/null @@ -1,85 +0,0 @@ -:root { - --twentycrm-spacing-multiplicator: 4; - --twentycrm-border-radius-sm: 4px; - --twentycrm-font-weight-medium: 500; - - /* Grays */ - --twentycrm-gray-100: #000000; - --twentycrm-gray-100-4: #0000000A; - --twentycrm-gray-100-10: #00000019; - --twentycrm-gray-100-16: #00000029; - --twentycrm-gray-90: #141414; - --twentycrm-gray-85: #171717; - --twentycrm-gray-85-80: #171717CC; - --twentycrm-gray-80: #1b1b1b; - --twentycrm-gray-80-80: #1b1b1bCC; - --twentycrm-gray-75: #1d1d1d; - --twentycrm-gray-70: #222222; - --twentycrm-gray-65: #292929; - --twentycrm-gray-60: #333333; - --twentycrm-gray-55: #4c4c4c; - --twentycrm-gray-50: #666666; - --twentycrm-gray-45: #818181; - --twentycrm-gray-40: #999999; - --twentycrm-gray-35: #b3b3b3; - --twentycrm-gray-30: #cccccc; - --twentycrm-gray-25: #d6d6d6; - --twentycrm-gray-20: #ebebeb; - --twentycrm-gray-15: #f1f1f1; - --twentycrm-gray-10: #fcfcfc; - --twentycrm-gray-10-80: #fcfcfcCC; - --twentycrm-gray-0: #ffffff; - --twentycrm-gray-0-6: #ffffff0f; - --twentycrm-gray-0-10: #ffffff19; - --twentycrm-gray-0-14: #ffffff23; - - /* Blues */ - --twentycrm-blue-accent-90: #141a25; - --twentycrm-blue-accent-10: #f5f9fd; -} - -:root.dark { - /* Accent color */ - --twentycrm-accent-quaternary: var(--twentycrm-blue-accent-90); - - /* Font color */ - --twentycrm-font-color-secondary: var(--twentycrm-gray-35); - --twentycrm-font-color-primary: var(--twentycrm-gray-20); - --twentycrm-font-color-light: var(--twentycrm-gray-50); - --twentycrm-font-color-extra-light: var(--twentycrm-gray-55); - - /* Background color */ - --twentycrm-background-primary: var(--twentycrm-gray-85); - - /* Background transparent color */ - --twentycrm-background-transparent-secondary: var(--twentycrm-gray-80-80); - --twentycrm-background-transparent-light: var(--twentycrm-gray-0-6); - --twentycrm-background-transparent-medium: var(--twentycrm-gray-0-10); - --twentycrm-background-transparent-strong: var(--twentycrm-gray-0-14); - - /* Border color */ - --twentycrm-border-color-medium: var(--twentycrm-gray-65); -} - -:root.light { - /* Accent color */ - --twentycrm-accent-quaternary: var(--twentycrm-blue-accent-10); - - /* Colors */ - --twentycrm-font-color-primary: var(--twentycrm-gray-60); - --twentycrm-font-color-secondary: var(--twentycrm-gray-50); - --twentycrm-font-color-light: var(--twentycrm-gray-35); - --twentycrm-font-color-extra-light: var(--twentycrm-gray-30); - - /* Background color */ - --twentycrm-background-primary: var(--twentycrm-gray-0); - - /* Background transparent color */ - --twentycrm-background-transparent-secondary: var(--twentycrm-gray-10-80); - --twentycrm-background-transparent-light: var(--twentycrm-gray-100-4); - --twentycrm-background-transparent-medium: var(--twentycrm-gray-100-10); - --twentycrm-background-transparent-strong: var(--twentycrm-gray-100-16); - - /* Border color */ - --twentycrm-border-color-medium: var(--twentycrm-gray-20); -} diff --git a/packages/twenty-ui/src/utilities/index.ts b/packages/twenty-ui/src/utilities/index.ts index 1e20365c2bcf..3d99deafeead 100644 --- a/packages/twenty-ui/src/utilities/index.ts +++ b/packages/twenty-ui/src/utilities/index.ts @@ -1,3 +1,4 @@ export * from './color/utils/stringToHslColor'; +export * from './isDefined'; export * from './state/utils/createState'; export * from './types/Nullable'; diff --git a/packages/twenty-ui/src/utilities/isDefined.ts b/packages/twenty-ui/src/utilities/isDefined.ts new file mode 100644 index 000000000000..81eb67203a03 --- /dev/null +++ b/packages/twenty-ui/src/utilities/isDefined.ts @@ -0,0 +1,4 @@ +import { isNull, isUndefined } from '@sniptt/guards'; + +export const isDefined = <T>(value: T | null | undefined): value is T => + !isUndefined(value) && !isNull(value); diff --git a/packages/twenty-ui/vite.config.ts b/packages/twenty-ui/vite.config.ts index bac435e19c1f..46804e3bee7b 100644 --- a/packages/twenty-ui/vite.config.ts +++ b/packages/twenty-ui/vite.config.ts @@ -33,6 +33,7 @@ export default defineConfig({ '**/OverflowingTextWithTooltip.tsx', '**/Chip.tsx', '**/Tag.tsx', + '**/Avatar.tsx', ], babelOptions: { presets: ['@babel/preset-typescript', '@babel/preset-react'], diff --git a/packages/twenty-website/package.json b/packages/twenty-website/package.json index 7976b89b724c..fd47048b5a4f 100644 --- a/packages/twenty-website/package.json +++ b/packages/twenty-website/package.json @@ -1,6 +1,6 @@ { "name": "twenty-website", - "version": "0.20.0", + "version": "0.22.0", "private": true, "scripts": { "nx": "NX_DEFAULT_PROJECT=twenty-website node ../../node_modules/nx/bin/nx.js", @@ -14,6 +14,7 @@ "database:generate:pg": "npx drizzle-kit generate:pg --config=src/database/drizzle-posgres.config.ts" }, "dependencies": { + "next-runtime-env": "^3.2.2", "postgres": "^3.4.3" } } diff --git a/packages/twenty-website/public/images/releases/0.20/0.20-blocklist.png b/packages/twenty-website/public/images/releases/0.20/0.20-blocklist.png new file mode 100644 index 000000000000..92fbf686bafc Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.20/0.20-blocklist.png differ diff --git a/packages/twenty-website/public/images/releases/0.20/0.20-onboarding.png b/packages/twenty-website/public/images/releases/0.20/0.20-onboarding.png new file mode 100644 index 000000000000..b29133411cc1 Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.20/0.20-onboarding.png differ diff --git a/packages/twenty-website/public/images/releases/0.20/0.20-timeline.png b/packages/twenty-website/public/images/releases/0.20/0.20-timeline.png new file mode 100644 index 000000000000..20fbd8a773e4 Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.20/0.20-timeline.png differ diff --git a/packages/twenty-website/public/images/releases/0.21/0.21-advanced-email-settings.png b/packages/twenty-website/public/images/releases/0.21/0.21-advanced-email-settings.png new file mode 100644 index 000000000000..cc69c0cca7de Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.21/0.21-advanced-email-settings.png differ diff --git a/packages/twenty-website/public/images/releases/0.21/0.21-many-many.png b/packages/twenty-website/public/images/releases/0.21/0.21-many-many.png new file mode 100644 index 000000000000..c76dbdc5eccb Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.21/0.21-many-many.png differ diff --git a/packages/twenty-website/public/images/releases/0.22/0.22-kanban-improvements.png b/packages/twenty-website/public/images/releases/0.22/0.22-kanban-improvements.png new file mode 100644 index 000000000000..e9625df62206 Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.22/0.22-kanban-improvements.png differ diff --git a/packages/twenty-website/public/images/releases/0.22/0.22-mass-deletion.png b/packages/twenty-website/public/images/releases/0.22/0.22-mass-deletion.png new file mode 100644 index 000000000000..f09ab6514800 Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.22/0.22-mass-deletion.png differ diff --git a/packages/twenty-website/public/images/releases/0.22/0.22-navbar.png b/packages/twenty-website/public/images/releases/0.22/0.22-navbar.png new file mode 100644 index 000000000000..6b0bb59762e2 Binary files /dev/null and b/packages/twenty-website/public/images/releases/0.22/0.22-navbar.png differ diff --git a/packages/twenty-website/src/app/_components/contributors/PullRequestItem.tsx b/packages/twenty-website/src/app/_components/contributors/PullRequestItem.tsx index 617ed7c1778b..dd00e056219e 100644 --- a/packages/twenty-website/src/app/_components/contributors/PullRequestItem.tsx +++ b/packages/twenty-website/src/app/_components/contributors/PullRequestItem.tsx @@ -6,6 +6,7 @@ import { PullRequestIcon } from '@/app/_components/ui/icons/SvgIcons'; import { Theme } from '@/app/_components/ui/theme/theme'; import { formatIntoRelativeDate } from '@/shared-utils/formatIntoRelativeDate'; +// TODO: use twenty-ui Tooltip const StyledTooltip = styled(Tooltip)``; const Item = styled.div` diff --git a/packages/twenty-website/src/app/_components/docs/AlgoliaDocSearch.tsx b/packages/twenty-website/src/app/_components/docs/AlgoliaDocSearch.tsx index 17e2e1db6215..cbd4b5b7a1e0 100644 --- a/packages/twenty-website/src/app/_components/docs/AlgoliaDocSearch.tsx +++ b/packages/twenty-website/src/app/_components/docs/AlgoliaDocSearch.tsx @@ -1,5 +1,6 @@ import { DocSearch } from '@docsearch/react'; import { StoredDocSearchHit } from '@docsearch/react/dist/esm/types'; +import { env } from 'next-runtime-env'; interface AlgoliaHit extends StoredDocSearchHit { _snippetResult?: { @@ -47,8 +48,8 @@ export const AlgoliaDocSearch = ({ pathname }: AlgoliaDocSearchProps) => { </a> </section> )} - appId={process.env.NEXT_PUBLIC_ALGOLIA_APP_ID as string} - apiKey={process.env.NEXT_PUBLIC_ALGOLIA_API_KEY as string} + appId={env('NEXT_PUBLIC_ALGOLIA_APP_ID') ?? ''} + apiKey={env('NEXT_PUBLIC_ALGOLIA_API_KEY') ?? ''} indexName={`twenty-${indexName}`} /> ); diff --git a/packages/twenty-website/src/app/_components/docs/DocsCard.tsx b/packages/twenty-website/src/app/_components/docs/DocsCard.tsx index 3c3164e0cfc7..c5eeff85a284 100644 --- a/packages/twenty-website/src/app/_components/docs/DocsCard.tsx +++ b/packages/twenty-website/src/app/_components/docs/DocsCard.tsx @@ -1,12 +1,14 @@ 'use client'; import styled from '@emotion/styled'; -import { usePathname, useRouter } from 'next/navigation'; +import Link from 'next/link'; +import { usePathname } from 'next/navigation'; import { Theme } from '@/app/_components/ui/theme/theme'; import { DocsArticlesProps } from '@/content/user-guide/constants/getDocsArticles'; import { getCardPath } from '@/shared-utils/getCardPath'; -const StyledContainer = styled.div` +const StyledContainer = styled(Link)` + text-decoration: none; color: ${Theme.border.color.plain}; border: 2px solid ${Theme.border.color.plain}; border-radius: ${Theme.border.radius.md}; @@ -58,13 +60,12 @@ export default function DocsCard({ card: DocsArticlesProps; isSection?: boolean; }) { - const router = useRouter(); const pathname = usePathname(); const path = getCardPath(card, pathname, isSection); if (card.title) { return ( - <StyledContainer onClick={() => router.push(path)}> + <StyledContainer href={path}> <StyledImage src={card.image} alt={card.title} /> <StyledHeading>{card.title}</StyledHeading> <StyledSubHeading>{card.info}</StyledSubHeading> diff --git a/packages/twenty-website/src/app/_components/docs/DocsContent.tsx b/packages/twenty-website/src/app/_components/docs/DocsContent.tsx index eb46ee97a439..5d4c6c92603c 100644 --- a/packages/twenty-website/src/app/_components/docs/DocsContent.tsx +++ b/packages/twenty-website/src/app/_components/docs/DocsContent.tsx @@ -149,6 +149,7 @@ export default function DocsContent({ item }: { item: FileContent }) { style={{ objectFit: 'cover' }} onLoad={() => setImageLoaded(true)} loaded={imageLoaded.toString()} + unoptimized /> )} </StyledImageContainer> diff --git a/packages/twenty-website/src/app/_components/playground/rest-api-wrapper.tsx b/packages/twenty-website/src/app/_components/playground/rest-api-wrapper.tsx new file mode 100644 index 000000000000..cd2da8ea8e6d --- /dev/null +++ b/packages/twenty-website/src/app/_components/playground/rest-api-wrapper.tsx @@ -0,0 +1,29 @@ +import React, { useEffect } from 'react'; +// @ts-expect-error Migration loader as text not passing warnings +import { API } from '@stoplight/elements'; + +// @ts-expect-error Migration loader as text not passing warnings +import spotlightTheme from '!css-loader!@stoplight/elements/styles.min.css'; + +export const RestApiWrapper = ({ openApiJson }: { openApiJson: any }) => { + // We load spotlightTheme style using useEffect as it breaks remaining docs style + useEffect(() => { + const styleElement = document.createElement('style'); + styleElement.innerHTML = spotlightTheme.toString(); + document.head.append(styleElement); + + return () => styleElement.remove(); + }, []); + + return ( + <div + style={{ + height: 'calc(100vh - var(--ifm-navbar-height) - 45px)', + width: '100%', + overflow: 'auto', + }} + > + <API apiDescriptionDocument={JSON.stringify(openApiJson)} router="hash" /> + </div> + ); +}; diff --git a/packages/twenty-website/src/app/_components/playground/token-form.tsx b/packages/twenty-website/src/app/_components/playground/token-form.tsx index 398fbad7a525..1a0051c62edd 100644 --- a/packages/twenty-website/src/app/_components/playground/token-form.tsx +++ b/packages/twenty-website/src/app/_components/playground/token-form.tsx @@ -195,7 +195,9 @@ const TokenForm = ({ className="select" onChange={(event) => router.replace( - '/' + pathname.split('/').at(-2) + '/' + event.target.value, + pathname.split('/').slice(0, -1).join('/') + + '/' + + event.target.value, ) } value={pathname.split('/').at(-1)} diff --git a/packages/twenty-website/src/app/_components/ui/layout/PostImage.tsx b/packages/twenty-website/src/app/_components/ui/layout/PostImage.tsx index b5bc150d7cbf..00e215aa7754 100644 --- a/packages/twenty-website/src/app/_components/ui/layout/PostImage.tsx +++ b/packages/twenty-website/src/app/_components/ui/layout/PostImage.tsx @@ -1,5 +1,3 @@ -import Image from 'next/image'; - export const PostImage = ({ sources, style, @@ -7,5 +5,5 @@ export const PostImage = ({ sources: { light: string; dark: string }; style?: React.CSSProperties; }) => { - return <Image src={sources.light} style={style} alt={sources.light} />; + return <img src={sources.light} style={style} alt={sources.light} />; }; diff --git a/packages/twenty-website/src/app/developers/[slug]/page.tsx b/packages/twenty-website/src/app/developers/[slug]/page.tsx index 05cc57264ad3..47efae715865 100644 --- a/packages/twenty-website/src/app/developers/[slug]/page.tsx +++ b/packages/twenty-website/src/app/developers/[slug]/page.tsx @@ -5,8 +5,6 @@ import DocsContent from '@/app/_components/docs/DocsContent'; import { fetchArticleFromSlug } from '@/shared-utils/fetchArticleFromSlug'; import { formatSlug } from '@/shared-utils/formatSlug'; -export const dynamic = 'force-dynamic'; - export async function generateMetadata({ params, }: { diff --git a/packages/twenty-website/src/app/developers/graphql/core/page.tsx b/packages/twenty-website/src/app/developers/graphql/core/page.tsx index e6a0bc9481ed..f2c9e18ff56c 100644 --- a/packages/twenty-website/src/app/developers/graphql/core/page.tsx +++ b/packages/twenty-website/src/app/developers/graphql/core/page.tsx @@ -1,6 +1,11 @@ +'use client'; import React from 'react'; +import dynamic from 'next/dynamic'; -import GraphQlPlayground from '../../../_components/playground/graphql-playground'; +const GraphQlPlayground = dynamic( + () => import('../../../_components/playground/graphql-playground'), + { ssr: false }, +); const CoreGraphql = () => { return <GraphQlPlayground subDoc={'core'} />; diff --git a/packages/twenty-website/src/app/developers/graphql/metadata/page.tsx b/packages/twenty-website/src/app/developers/graphql/metadata/page.tsx index b215e2344458..ebc4490ae132 100644 --- a/packages/twenty-website/src/app/developers/graphql/metadata/page.tsx +++ b/packages/twenty-website/src/app/developers/graphql/metadata/page.tsx @@ -1,6 +1,11 @@ +'use client'; import React from 'react'; +import dynamic from 'next/dynamic'; -import GraphQlPlayground from '../../../_components/playground/graphql-playground'; +const GraphQlPlayground = dynamic( + () => import('../../../_components/playground/graphql-playground'), + { ssr: false }, +); const CoreGraphql = () => { return <GraphQlPlayground subDoc={'metadata'} />; diff --git a/packages/twenty-website/src/app/developers/page.tsx b/packages/twenty-website/src/app/developers/page.tsx index eb425f467bf1..c2de5a6f7adb 100644 --- a/packages/twenty-website/src/app/developers/page.tsx +++ b/packages/twenty-website/src/app/developers/page.tsx @@ -7,8 +7,6 @@ export const metadata = { icons: '/images/core/logo.svg', }; -export const dynamic = 'force-dynamic'; - export default async function DocsHome() { const filePath = 'src/content/developers/'; const docsArticleCards = getDocsArticles(filePath); diff --git a/packages/twenty-website/src/app/developers/rest-api/core/page.tsx b/packages/twenty-website/src/app/developers/rest-api/core/page.tsx index ae245a9959d0..4c211c3ec4d0 100644 --- a/packages/twenty-website/src/app/developers/rest-api/core/page.tsx +++ b/packages/twenty-website/src/app/developers/rest-api/core/page.tsx @@ -1,40 +1,23 @@ 'use client'; -import React, { useEffect, useState } from 'react'; -// @ts-expect-error Migration loader as text not passing warnings -import { API } from '@stoplight/elements'; + +import { useEffect, useState } from 'react'; import Playground from '@/app/_components/playground/playground'; +import { RestApiWrapper } from '@/app/_components/playground/rest-api-wrapper'; -// @ts-expect-error Migration loader as text not passing warnings -import spotlightTheme from '!css-loader!@stoplight/elements/styles.min.css'; +const RestApi = () => { + const [openApiJson, setOpenApiJson] = useState({}); + const [isClient, setIsClient] = useState(false); -const RestApiComponent = ({ openApiJson }: { openApiJson: any }) => { - // We load spotlightTheme style using useEffect as it breaks remaining docs style useEffect(() => { - const styleElement = document.createElement('style'); - styleElement.innerHTML = spotlightTheme.toString(); - document.head.append(styleElement); - - return () => styleElement.remove(); + setIsClient(true); }, []); - return ( - <div - style={{ - height: 'calc(100vh - var(--ifm-navbar-height) - 45px)', - width: '100%', - overflow: 'auto', - }} - > - <API apiDescriptionDocument={JSON.stringify(openApiJson)} router="hash" /> - </div> - ); -}; - -const restApi = () => { - const [openApiJson, setOpenApiJson] = useState({}); + if (!isClient) { + return null; + } - const children = <RestApiComponent openApiJson={openApiJson} />; + const children = <RestApiWrapper openApiJson={openApiJson} />; return ( <div style={{ width: '100vw' }}> @@ -47,4 +30,4 @@ const restApi = () => { ); }; -export default restApi; +export default RestApi; diff --git a/packages/twenty-website/src/app/developers/rest-api/metadata/page.tsx b/packages/twenty-website/src/app/developers/rest-api/metadata/page.tsx index c49726892b99..e76cadc93b77 100644 --- a/packages/twenty-website/src/app/developers/rest-api/metadata/page.tsx +++ b/packages/twenty-website/src/app/developers/rest-api/metadata/page.tsx @@ -1,39 +1,22 @@ 'use client'; import React, { useEffect, useState } from 'react'; -// @ts-expect-error Migration loader as text not passing warnings -import { API } from '@stoplight/elements'; import Playground from '@/app/_components/playground/playground'; +import { RestApiWrapper } from '@/app/_components/playground/rest-api-wrapper'; -// @ts-expect-error Migration loader as text not passing warnings -import spotlightTheme from '!css-loader!@stoplight/elements/styles.min.css'; +const restApi = () => { + const [openApiJson, setOpenApiJson] = useState({}); + const [isClient, setIsClient] = useState(false); -const RestApiComponent = ({ openApiJson }: { openApiJson: any }) => { - // We load spotlightTheme style using useEffect as it breaks remaining docs style useEffect(() => { - const styleElement = document.createElement('style'); - styleElement.innerHTML = spotlightTheme.toString(); - document.head.append(styleElement); - - return () => styleElement.remove(); + setIsClient(true); }, []); - return ( - <div - style={{ - height: 'calc(100vh - var(--ifm-navbar-height) - 45px)', - overflow: 'auto', - }} - > - <API apiDescriptionDocument={JSON.stringify(openApiJson)} router="hash" /> - </div> - ); -}; - -const restApi = () => { - const [openApiJson, setOpenApiJson] = useState({}); + if (!isClient) { + return null; + } - const children = <RestApiComponent openApiJson={openApiJson} />; + const children = <RestApiWrapper openApiJson={openApiJson} />; return ( <Playground diff --git a/packages/twenty-website/src/app/developers/section/[folder]/page.tsx b/packages/twenty-website/src/app/developers/section/[folder]/page.tsx index 5879c5277a29..6e57de3bdc61 100644 --- a/packages/twenty-website/src/app/developers/section/[folder]/page.tsx +++ b/packages/twenty-website/src/app/developers/section/[folder]/page.tsx @@ -6,8 +6,6 @@ import { getDocsArticles } from '@/content/user-guide/constants/getDocsArticles' import { fetchArticleFromSlug } from '@/shared-utils/fetchArticleFromSlug'; import { formatSlug } from '@/shared-utils/formatSlug'; -export const dynamic = 'force-dynamic'; - export async function generateMetadata({ params, }: { diff --git a/packages/twenty-website/src/app/layout.tsx b/packages/twenty-website/src/app/layout.tsx index d82f80c89c07..c302c9e3e51d 100644 --- a/packages/twenty-website/src/app/layout.tsx +++ b/packages/twenty-website/src/app/layout.tsx @@ -1,5 +1,6 @@ import { Metadata } from 'next'; import { Gabarito, Inter } from 'next/font/google'; +import { PublicEnvScript } from 'next-runtime-env'; import { AppHeader } from '@/app/_components/ui/layout/header'; @@ -40,6 +41,7 @@ export default function RootLayout({ return ( <html lang="en" className={`${gabarito.variable} ${inter.variable}`}> <body> + <PublicEnvScript /> <EmotionRootStyleRegistry> <AppHeader /> <div className="container">{children}</div> diff --git a/packages/twenty-website/src/app/user-guide/[slug]/page.tsx b/packages/twenty-website/src/app/user-guide/[slug]/page.tsx index 3c33f6025901..6c919639d5b4 100644 --- a/packages/twenty-website/src/app/user-guide/[slug]/page.tsx +++ b/packages/twenty-website/src/app/user-guide/[slug]/page.tsx @@ -5,8 +5,6 @@ import DocsContent from '@/app/_components/docs/DocsContent'; import { fetchArticleFromSlug } from '@/shared-utils/fetchArticleFromSlug'; import { formatSlug } from '@/shared-utils/formatSlug'; -export const dynamic = 'force-dynamic'; - export async function generateMetadata({ params, }: { diff --git a/packages/twenty-website/src/app/user-guide/page.tsx b/packages/twenty-website/src/app/user-guide/page.tsx index 308bd2fb2be9..e23f4eb87a43 100644 --- a/packages/twenty-website/src/app/user-guide/page.tsx +++ b/packages/twenty-website/src/app/user-guide/page.tsx @@ -8,8 +8,6 @@ export const metadata = { icons: '/images/core/logo.svg', }; -export const dynamic = 'force-dynamic'; - export default async function UserGuideHome() { const filePath = 'src/content/user-guide/'; const docsArticleCards = getDocsArticles(filePath); diff --git a/packages/twenty-website/src/app/user-guide/section/[folder]/[documentation]/page.tsx b/packages/twenty-website/src/app/user-guide/section/[folder]/[documentation]/page.tsx index c331f118e1e0..4804aa7248ca 100644 --- a/packages/twenty-website/src/app/user-guide/section/[folder]/[documentation]/page.tsx +++ b/packages/twenty-website/src/app/user-guide/section/[folder]/[documentation]/page.tsx @@ -5,8 +5,6 @@ import DocsContent from '@/app/_components/docs/DocsContent'; import { fetchArticleFromSlug } from '@/shared-utils/fetchArticleFromSlug'; import { formatSlug } from '@/shared-utils/formatSlug'; -export const dynamic = 'force-dynamic'; - export async function generateMetadata({ params, }: { diff --git a/packages/twenty-website/src/content/developers/frontend-development/frontend-commands.mdx b/packages/twenty-website/src/content/developers/frontend-development/frontend-commands.mdx index 3dfcdd5f6177..56dc4e718a37 100644 --- a/packages/twenty-website/src/content/developers/frontend-development/frontend-commands.mdx +++ b/packages/twenty-website/src/content/developers/frontend-development/frontend-commands.mdx @@ -28,9 +28,9 @@ nx lint twenty-front ```bash nx test twenty-front# run jest tests -nx storybook:dev twenty-front# run storybook -nx storybook:test twenty-front# run tests # (needs yarn storybook:dev to be running) -nx storybook:coverage twenty-front # (needs yarn storybook:dev to be running) +nx storybook:serve:dev twenty-front# run storybook +nx storybook:test twenty-front# run tests # (needs yarn storybook:serve:dev to be running) +nx storybook:coverage twenty-front # (needs yarn storybook:serve:dev to be running) ``` ## Tech Stack diff --git a/packages/twenty-website/src/content/developers/self-hosting/cloud-providers.mdx b/packages/twenty-website/src/content/developers/self-hosting/cloud-providers.mdx index b996568ed657..08e541a21a19 100644 --- a/packages/twenty-website/src/content/developers/self-hosting/cloud-providers.mdx +++ b/packages/twenty-website/src/content/developers/self-hosting/cloud-providers.mdx @@ -6,21 +6,31 @@ image: /images/user-guide/notes/notes_header.png <ArticleWarning> This document is maintained by the community. It might contain issues. -Feel free to join our discord if you need assistance. </ArticleWarning> +## Kubernetes via Terraform and Manifests + +Community-led documentation for Kubernetes deployment is available (here)[https://github.com/twentyhq/twenty/tree/main/packages/twenty-docker/k8s] + ## Render +Community-led, might not be up to date + [![Deploy to Render](https://render.com/images/deploy-to-render-button.svg)](https://render.com/deploy?repo=https://github.com/twentyhq/twenty) -## RepoCloud + +## RepoCloud + +Community-led, might not be up to date [![Deploy on RepoCloud](https://d16t0pc4846x52.cloudfront.net/deploy.png)](https://repocloud.io/details/?app_id=259) ## Azure Container Apps +Community-led, might not be up to date + ### About Hosts Twenty CRM using Azure Container Apps. diff --git a/packages/twenty-website/src/content/developers/self-hosting/self-hosting-var.mdx b/packages/twenty-website/src/content/developers/self-hosting/self-hosting-var.mdx index 9c4937fb3259..f133fde180a3 100644 --- a/packages/twenty-website/src/content/developers/self-hosting/self-hosting-var.mdx +++ b/packages/twenty-website/src/content/developers/self-hosting/self-hosting-var.mdx @@ -13,6 +13,7 @@ Twenty offers integrations with Gmail and Google Calendar. To enable these featu # from your worker container yarn command:prod cron:messaging:messages-import yarn command:prod cron:messaging:message-list-fetch +yarn command:prod cron:calendar:calendar-event-list-fetch ``` # Setup Environment Variables @@ -141,6 +142,8 @@ yarn command:prod cron:messaging:message-list-fetch ['STORAGE_S3_REGION', '', 'Storage Region'], ['STORAGE_S3_NAME', '', 'Bucket Name'], ['STORAGE_S3_ENDPOINT', '', 'Use if a different Endpoint is needed (for example Google)'], + ['STORAGE_S3_ACCESS_KEY_ID', '', 'Optional depending on the authentication method'], + ['STORAGE_S3_SECRET_ACCESS_KEY', '', 'Optional depending on the authentication method'], ['STORAGE_LOCAL_PATH', '.local-storage', 'data path (local storage)'], ]}></ArticleTable> @@ -153,7 +156,7 @@ yarn command:prod cron:messaging:message-list-fetch ### Logging <ArticleTable options={[ - ['LOGGER_DRIVER', 'console', "The logging driver can be: 'console' or 'sentry'"], + ['LOGGER_DRIVER', 'console', "Currently, only supports 'console'"], ['LOGGER_IS_BUFFER_ENABLED', 'true', 'Buffer the logs before sending them to the logging driver'], ['LOG_LEVELS', 'error,warn', "The loglevels which are logged to the logging driver. Can include: 'log', 'warn', 'error'"], ['EXCEPTION_HANDLER_DRIVER', 'sentry', "The exception handler driver can be: 'console' or 'sentry'"], @@ -167,8 +170,13 @@ yarn command:prod cron:messaging:message-list-fetch ### Data enrichment and AI <ArticleTable options={[ - ['OPENROUTER_API_KEY', '', "The API key for openrouter.ai, an abstraction layer over models from Mistral, OpenAI and more"] - ]}></ArticleTable> + ['OPENROUTER_API_KEY', '', "The API key for openrouter.ai, an abstraction layer over models from Mistral, OpenAI and more"], + ['OPENAI_API_KEY', 'sk-proj-abcdabcd...', "OpenAI API key"], + ['LLM_CHAT_MODEL_DRIVER', 'openai', "LLM provider"], + ['LLM_TRACING_DRIVER', 'langfuse', "Where to output LangChain logs. 'langfuse' or 'console'."], + ['LANGFUSE_SECRET_KEY', 'sk-lf-abcdabcd-abcd...', "Langfuse secret key"], + ['LANGFUSE_PUBLIC_KEY', 'pk-lf-abcdabcd-abcd...', "Langfuse public key"], +]}></ArticleTable> ### Support Chat @@ -208,4 +216,4 @@ yarn command:prod cron:messaging:message-list-fetch ['CAPTCHA_SECRET_KEY', '', 'The captcha secret key'], ]}></ArticleTable> -<ArticleEditContent></ArticleEditContent> \ No newline at end of file +<ArticleEditContent></ArticleEditContent> diff --git a/packages/twenty-website/src/content/releases/0.20.0.mdx b/packages/twenty-website/src/content/releases/0.20.0.mdx new file mode 100644 index 000000000000..b80da631407f --- /dev/null +++ b/packages/twenty-website/src/content/releases/0.20.0.mdx @@ -0,0 +1,27 @@ +--- +release: 0.20.0 +Date: June 14th 2024 +--- + +# Enhanced Timeline + +The timeline on every record page has been significantly improved. It now provides detailed updates for: + +- Record creations +- Field updates +- Received emails +- Created calendar events + +![](/images/releases/0.20/0.20-timeline.png) + +# Improved Onboarding Experience + +Our onboarding process has been streamlined to let you import your calendar and emails seamlessly. You can now also configure your privacy settings directly during onboarding, allowing you to choose between sharing content with your team or keeping it hidden. + +![](/images/releases/0.20/0.20-onboarding.png) + +# Email and calendar Blocklist + +To enhance privacy, you can now add specific email addresses to a blocklist within the "Accounts" settings. This feature prevents sensitive content from being synced to the CRM when corresponding with certain individuals. This can be particularly useful when managing sensitive deals. + +![](/images/releases/0.20/0.20-blocklist.png) \ No newline at end of file diff --git a/packages/twenty-website/src/content/releases/0.21.0.mdx b/packages/twenty-website/src/content/releases/0.21.0.mdx new file mode 100644 index 000000000000..2614c981f1a9 --- /dev/null +++ b/packages/twenty-website/src/content/releases/0.21.0.mdx @@ -0,0 +1,24 @@ +--- +release: 0.21.0 +Date: June 28th 2024 +--- + +# Enhanced One-to-Many Relations Editing + +You can now edit one-to-many relations directly from the "many side". This means you can assign people to a company directly from the company list view, instead of having to navigate to each individual person's profile to assign them a company. + +![](/images/releases/0.21/0.21-many-many.png) + + +# Advanced Email and Calendar Settings + +We've introduced advanced settings for email and calendar management: +- **Auto-Create Contact Options:** Choose when an email interaction should automatically create a contact. Options include: + - People I've sent emails to and received emails from + - People I've sent emails to + - Don't auto create contact + +- **Email Exclusions:** Ability to exclude non-professional emails (e.g., Gmail, Outlook) and team emails (e.g., support@, team@) from being synced to the CRM. +- **Calendar Events:** Auto-contact creation settings are now available for calendar events as well. + +![](/images/releases/0.21/0.21-advanced-email-settings.png) \ No newline at end of file diff --git a/packages/twenty-website/src/content/releases/0.22.0.mdx b/packages/twenty-website/src/content/releases/0.22.0.mdx new file mode 100644 index 000000000000..a4b7e856370f --- /dev/null +++ b/packages/twenty-website/src/content/releases/0.22.0.mdx @@ -0,0 +1,26 @@ +--- +release: 0.22.0 +Date: July 11th 2024 +--- + +# Enhanced Kanban Board + +- **Edit Kanban Stages:** You can now edit Kanban stages directly from the app, not just from the settings. This makes it easier to manage and customize your workflow on the fly. +- **"No Value" Column:** Cards that are not assigned to a specific value will now appear in a "No Value" column. This column can be shown or hidden as needed, ensuring no cards are overlooked. + +![](/images/releases/0.22/0.22-kanban-improvements.png) + +# Revamped Navigation Bar + +Navigate more quickly with our revamped record page navbar: +- Navigate directly from one record page to another. +- View the total number of records within a view. +- Easily return to the corresponding index view with a new "Close" button. + +![](/images/releases/0.22/0.22-navbar.png) + +# Bulk Deletion + +You can now delete up to 10,000 records at once. (For when you want to Marie Kondo your database! 🧹) + +![](/images/releases/0.22/0.22-mass-deletion.png) \ No newline at end of file diff --git a/packages/twenty-website/src/content/user-guide/functions/integrations.mdx b/packages/twenty-website/src/content/user-guide/functions/integrations.mdx index 34e4d8b492a8..1cd3713141d3 100644 --- a/packages/twenty-website/src/content/user-guide/functions/integrations.mdx +++ b/packages/twenty-website/src/content/user-guide/functions/integrations.mdx @@ -18,7 +18,8 @@ Sync Twenty with 3000+ apps using <ArticleLink href="https://zapier.com/apps/twe 2. Click on `+ Create Zap` in the left sidebar. 3. Choose the application you want to set as the trigger. A trigger refers to an event that starts the automation. 4. Select Twenty as the action. An action is the event performed whenever an application triggers an automation. <ArticleLink href="https://zapier.com/how-it-works">Learn more about triggers and actions in Zapier.</ArticleLink> -5. Once you choose the Twenty account that you want to use for your automation, you'll have to allow Zapier to access it by adding an API key. You can learn [how to generate your API key here.](/user-guide/api-webhooks) +5. Once you choose the Twenty account that you want to use for your automation, you'll have to allow Zapier to access it by adding an API key. You can learn [how to generate your API key here.](/user-guide/section/functions/api-webhooks) + 6. Enter your API key and click on 'Yes, Continue to Twenty.' @@ -43,4 +44,4 @@ Sync Twenty with 3000+ apps using <ArticleLink href="https://zapier.com/apps/twe You can now continue creating your automation! -<ArticleEditContent></ArticleEditContent> \ No newline at end of file +<ArticleEditContent></ArticleEditContent> diff --git a/packages/twenty-website/src/content/user-guide/getting-started/create-workspace.mdx b/packages/twenty-website/src/content/user-guide/getting-started/create-workspace.mdx index 2639b7b6ff62..c10b7496b7ef 100644 --- a/packages/twenty-website/src/content/user-guide/getting-started/create-workspace.mdx +++ b/packages/twenty-website/src/content/user-guide/getting-started/create-workspace.mdx @@ -7,7 +7,7 @@ sectionInfo: Discover Twenty, an open-source CRM. --- ## Step 1: Registration -1. Navigate to <ArticleLink href="https://app.twenty.com/sign-up">Twenty Sign Up</ArticleLink>. +1. Navigate to <ArticleLink href="https://app.twenty.com">Twenty Sign Up</ArticleLink>. 2. Select your preferred sign-up method: - **Continue with Google** for Google account registration. - Or, **Continue With Email** for email registration. @@ -50,4 +50,4 @@ Post payment approval via Stripe, you're directed to create your workspace and u ## Support For queries or help, connect with the dedicated support team at [contact@twenty.com](mailto:contact@twenty.com) or send a message on <ArticleLink href="https://discord.gg/cx5n4Jzs57">Discord</ArticleLink> -<ArticleEditContent></ArticleEditContent> \ No newline at end of file +<ArticleEditContent></ArticleEditContent> diff --git a/packages/twenty-website/src/content/user-guide/getting-started/what-is-twenty.mdx b/packages/twenty-website/src/content/user-guide/getting-started/what-is-twenty.mdx index 2c18a7fde891..a13746ffe694 100644 --- a/packages/twenty-website/src/content/user-guide/getting-started/what-is-twenty.mdx +++ b/packages/twenty-website/src/content/user-guide/getting-started/what-is-twenty.mdx @@ -12,23 +12,23 @@ Twenty is the leading open-source CRM, crafted by hundreds of contributors to su ### Main Features -**Contact Management:** Efficiently store and manage customer data. [Learn more](/user-guide/objects). +**Contact Management:** Efficiently store and manage customer data. [Learn more](/user-guide/section/objects/standard-objects). -**Custom Objects:** Create and customize objects to fit your business needs. [Details](/user-guide/objects). +**Custom Objects:** Create and customize objects to fit your business needs. [Details](/user-guide/section/objects/standard-objects). -**Custom Fields:** Tailor data fields to capture and organize information specific to your operations. [Understand more](/user-guide/fields). +**Custom Fields:** Tailor data fields to capture and organize information specific to your operations. [Understand more](/user-guide/section/objects/fields). -**Kanban & Table Views:** Optimize your workflow with flexible [Table Views](/user-guide/table-views) and [Kanban Views](/user-guide/kanban-views). +**Kanban & Table Views:** Optimize your workflow with flexible [Table Views](/user-guide/section/objects/table-views) and [Kanban Views](/user-guide/section/objects/kanban-views). -**Pipeline Visualization:** Get a clear view of your processes with customizable views. [Explore views](/user-guide/views-sort-and-filter). +**Pipeline Visualization:** Get a clear view of your processes with customizable views. [Explore views](/user-guide/section/objects/views-sort-filter). -**Email Integration:** View the emails of a specific customer or company within your workspace. [Integrate now](/user-guide/emails). +**Email Integration:** View the emails of a specific customer or company within your workspace. [Integrate now](/user-guide/section/functions/emails). -**Notes:** Create detailed notes for each record to share knowledge more effectively. [Add notes](/user-guide/notes). +**Notes:** Create detailed notes for each record to share knowledge more effectively. [Add notes](/user-guide/section/functions/notes). -**Tasks:** Schedule tasks to track customer interactions. [See how](/user-guide/tasks). +**Tasks:** Schedule tasks to track customer interactions. [See how](/user-guide/section/functions/tasks). -**API & Webhooks:** Connect to other apps and automate workflows with API and Webhooks. [Start integrating](/user-guide/integrations). +**API & Webhooks:** Connect to other apps and automate workflows with API and Webhooks. [Start integrating](/user-guide/section/functions/integrations). ### Benefits @@ -68,4 +68,4 @@ And Open-source is the bedrock of our approach, ensuring that Twenty evolves wit <ArticleLink href="https://app.twenty.com">Register here</ArticleLink> or <ArticleLink href="https://github.com/twentyhq/twenty">become a contributor on GitHub</ArticleLink>. -<ArticleEditContent></ArticleEditContent> \ No newline at end of file +<ArticleEditContent></ArticleEditContent> diff --git a/packages/twenty-website/src/shared-utils/getCardPath.tsx b/packages/twenty-website/src/shared-utils/getCardPath.tsx index f7cadc48ad2f..6bb3d599c894 100644 --- a/packages/twenty-website/src/shared-utils/getCardPath.tsx +++ b/packages/twenty-website/src/shared-utils/getCardPath.tsx @@ -16,7 +16,7 @@ export const getCardPath = ( if (isPlayground.includes(card.fileName)) { const apiType = card.fileName.includes('rest') ? 'rest-api' : 'graphql'; const apiName = card.fileName.includes('core') ? 'core' : 'metadata'; - return `${basePath}/${apiType}/${apiName}`; + return `/developers/${apiType}/${apiName}`; } else if (card.fileName.includes('storybook')) { return 'https://storybook.twenty.com'; } else if (card.fileName.includes('components')) { diff --git a/packages/twenty-zapier/src/test/creates/crud_record.test.ts b/packages/twenty-zapier/src/test/creates/crud_record.test.ts index f1c177233fcc..eb763338b82b 100644 --- a/packages/twenty-zapier/src/test/creates/crud_record.test.ts +++ b/packages/twenty-zapier/src/test/creates/crud_record.test.ts @@ -14,7 +14,7 @@ describe('creates.create_company', () => { nameSingular: 'Company', crudZapierOperation: Operation.create, name: 'Company Name', - address: 'Company Address', + address: { addressCity: 'Paris' }, domainName: 'Company Domain Name', linkedinLink: { url: '/linkedin_url', label: 'Test linkedinUrl' }, xLink: { url: '/x_url', label: 'Test xUrl' }, diff --git a/packages/twenty-zapier/src/test/triggers/trigger_record.test.ts b/packages/twenty-zapier/src/test/triggers/trigger_record.test.ts index 962fe91111e3..9139629da1c2 100644 --- a/packages/twenty-zapier/src/test/triggers/trigger_record.test.ts +++ b/packages/twenty-zapier/src/test/triggers/trigger_record.test.ts @@ -67,7 +67,7 @@ describe('triggers.trigger_record.created', () => { name: '', domainName: '', createdAt: '2023-10-19 10:10:12.490', - address: '', + address: { addressCity: null }, employees: null, linkedinUrl: null, xUrl: null, diff --git a/packages/twenty-zapier/src/test/utils/handleQueryParams.test.ts b/packages/twenty-zapier/src/test/utils/handleQueryParams.test.ts index c407c4a6c9ef..872879900166 100644 --- a/packages/twenty-zapier/src/test/utils/handleQueryParams.test.ts +++ b/packages/twenty-zapier/src/test/utils/handleQueryParams.test.ts @@ -10,7 +10,7 @@ describe('utils.handleQueryParams', () => { test('should format', () => { const inputData = { name: 'Company Name', - address: 'Company Address', + address: { addressCity: 'Paris' }, domainName: 'Company Domain Name', linkedinUrl__url: '/linkedin_url', linkedinUrl__label: 'Test linkedinUrl', @@ -23,7 +23,7 @@ describe('utils.handleQueryParams', () => { const result = handleQueryParams(inputData); const expectedResult = 'name: "Company Name", ' + - 'address: "Company Address", ' + + 'address: { addressCity: "Paris" }, ' + 'domainName: "Company Domain Name", ' + 'linkedinUrl: {url: "/linkedin_url", label: "Test linkedinUrl"}, ' + 'xUrl: {url: "/x_url", label: "Test xUrl"}, ' + diff --git a/packages/twenty-zapier/src/utils/computeInputFields.ts b/packages/twenty-zapier/src/utils/computeInputFields.ts index 7b8afbc58bbc..f2eabc331692 100644 --- a/packages/twenty-zapier/src/utils/computeInputFields.ts +++ b/packages/twenty-zapier/src/utils/computeInputFields.ts @@ -25,7 +25,6 @@ const getTypeFromFieldMetadataType = ( case FieldMetadataType.NUMBER: return 'integer'; case FieldMetadataType.NUMERIC: - case FieldMetadataType.PROBABILITY: return 'number'; default: return undefined; @@ -183,7 +182,6 @@ export const computeInputFields = ( case FieldMetadataType.BOOLEAN: case FieldMetadataType.NUMBER: case FieldMetadataType.NUMERIC: - case FieldMetadataType.PROBABILITY: case FieldMetadataType.RATING: { const nodeFieldType = getTypeFromFieldMetadataType(nodeField.type); if (!nodeFieldType) { diff --git a/packages/twenty-zapier/src/utils/data.types.ts b/packages/twenty-zapier/src/utils/data.types.ts index d7b38d4876c7..ba73ae18e09a 100644 --- a/packages/twenty-zapier/src/utils/data.types.ts +++ b/packages/twenty-zapier/src/utils/data.types.ts @@ -40,7 +40,6 @@ export enum FieldMetadataType { BOOLEAN = 'BOOLEAN', NUMBER = 'NUMBER', NUMERIC = 'NUMERIC', - PROBABILITY = 'PROBABILITY', LINK = 'LINK', CURRENCY = 'CURRENCY', FULL_NAME = 'FULL_NAME', diff --git a/render.yaml b/render.yaml index b0719ea46b63..580cd4a26cfd 100644 --- a/render.yaml +++ b/render.yaml @@ -1,29 +1,58 @@ services: - type: web - name: front + name: server runtime: image - plan: free image: - url: twentycrm/twenty-front:latest + url: twentycrm/twenty:latest + dockerCommand: "sh -c ./scripts/render-run.sh" autoDeploy: false + plan: standard envVars: - - key: REACT_APP_SERVER_BASE_URL + - key: FRONT_BASE_URL fromService: name: server type: web envVarKey: RENDER_EXTERNAL_URL -- type: web - name: server + - key: SERVER_URL + fromService: + name: server + type: web + envVarKey: RENDER_EXTERNAL_URL + - key: ACCESS_TOKEN_SECRET + generateValue: true + - key: LOGIN_TOKEN_SECRET + generateValue: true + - key: REFRESH_TOKEN_SECRET + generateValue: true + - key: FILE_TOKEN_SECRET + generateValue: true + - key: PG_DATABASE_HOST + fromService: + name: twenty_postgres + type: pserv + property: host + - key: PG_DATABASE_PORT + fromService: + name: twenty_postgres + type: pserv + property: port +- type: worker + name: worker runtime: image image: - url: twentycrm/twenty-server:latest - dockerCommand: "sh -c ./scripts/render-run.sh" + url: twentycrm/twenty:latest + dockerCommand: "sh -c ./scripts/render-worker.sh" autoDeploy: false plan: standard envVars: - key: FRONT_BASE_URL fromService: - name: front + name: server + type: web + envVarKey: RENDER_EXTERNAL_URL + - key: SERVER_URL + fromService: + name: server type: web envVarKey: RENDER_EXTERNAL_URL - key: ACCESS_TOKEN_SECRET diff --git a/yarn.lock b/yarn.lock index 7145165f5042..38f5ae31c335 100644 --- a/yarn.lock +++ b/yarn.lock @@ -7695,6 +7695,95 @@ __metadata: languageName: node linkType: hard +"@langchain/core@npm:>0.1.56 <0.3.0, @langchain/core@npm:>=0.2.5 <0.3.0": + version: 0.2.6 + resolution: "@langchain/core@npm:0.2.6" + dependencies: + ansi-styles: "npm:^5.0.0" + camelcase: "npm:6" + decamelize: "npm:1.2.0" + js-tiktoken: "npm:^1.0.12" + langsmith: "npm:~0.1.30" + ml-distance: "npm:^4.0.0" + mustache: "npm:^4.2.0" + p-queue: "npm:^6.6.2" + p-retry: "npm:4" + uuid: "npm:^9.0.0" + zod: "npm:^3.22.4" + zod-to-json-schema: "npm:^3.22.3" + checksum: 4ae46528854f9dfc9e6e91c9351275cecc5cb4d246c8a2ae1326fbe046bb8637e5e26743e1ebd86f06f2ef12dc1741ea3f4480701be1e89ceecc5146b8c873b1 + languageName: node + linkType: hard + +"@langchain/core@npm:>0.2.0 <0.3.0, @langchain/core@npm:>=0.2.8 <0.3.0, @langchain/core@npm:~0.2.0": + version: 0.2.9 + resolution: "@langchain/core@npm:0.2.9" + dependencies: + ansi-styles: "npm:^5.0.0" + camelcase: "npm:6" + decamelize: "npm:1.2.0" + js-tiktoken: "npm:^1.0.12" + langsmith: "npm:~0.1.30" + ml-distance: "npm:^4.0.0" + mustache: "npm:^4.2.0" + p-queue: "npm:^6.6.2" + p-retry: "npm:4" + uuid: "npm:^9.0.0" + zod: "npm:^3.22.4" + zod-to-json-schema: "npm:^3.22.3" + checksum: e336b2c90d4955cc522f3295b1f2b09e89b88d483108ca89af20c88ee41f815b91a307a193d0359e5012fb348022dcafa43a7ed28a195ae74ef4ec6a59678e4d + languageName: node + linkType: hard + +"@langchain/mistralai@npm:^0.0.24": + version: 0.0.24 + resolution: "@langchain/mistralai@npm:0.0.24" + dependencies: + "@langchain/core": "npm:>0.1.56 <0.3.0" + "@mistralai/mistralai": "npm:^0.4.0" + uuid: "npm:^9.0.0" + zod: "npm:^3.22.4" + zod-to-json-schema: "npm:^3.22.4" + checksum: abbc862685dfa48e9f4418ff94843b38d779514f46d5a971dc0c98b55a44d6fe91f3610432ccc0af39496ef075fbffd3e514e4104af5968938e0270e1a07716d + languageName: node + linkType: hard + +"@langchain/openai@npm:>=0.1.0 <0.3.0": + version: 0.2.0 + resolution: "@langchain/openai@npm:0.2.0" + dependencies: + "@langchain/core": "npm:>=0.2.8 <0.3.0" + js-tiktoken: "npm:^1.0.12" + openai: "npm:^4.49.1" + zod: "npm:^3.22.4" + zod-to-json-schema: "npm:^3.22.3" + checksum: a55cf7f42f4df901049b98e592bde9ee3e4a027635b07697110f511d2ccb4118ea562cd9c9d0de0e43efe291ed18425d6f6dea47447b27ba35e756c9b969fba0 + languageName: node + linkType: hard + +"@langchain/openai@npm:^0.1.3": + version: 0.1.3 + resolution: "@langchain/openai@npm:0.1.3" + dependencies: + "@langchain/core": "npm:>=0.2.5 <0.3.0" + js-tiktoken: "npm:^1.0.12" + openai: "npm:^4.49.1" + zod: "npm:^3.22.4" + zod-to-json-schema: "npm:^3.22.3" + checksum: b693bd9d5ec118136f99279c50b531865702d0fa41479cde4ced05bb20b487cc75656761e491b380e69ff191b38b8a41e4573a485b4da42f57061adb7aa692eb + languageName: node + linkType: hard + +"@langchain/textsplitters@npm:~0.0.0": + version: 0.0.3 + resolution: "@langchain/textsplitters@npm:0.0.3" + dependencies: + "@langchain/core": "npm:>0.2.0 <0.3.0" + js-tiktoken: "npm:^1.0.12" + checksum: 3297b48f636a8a6acbd65f1465624741e6d557512ea8020a208cc6b2aa6e8d752cd08511a92ef980a06ed95858b7750a1126a4e6acfbb75fd4733e050651f405 + languageName: node + linkType: hard + "@leichtgewicht/ip-codec@npm:^2.0.1": version: 2.0.4 resolution: "@leichtgewicht/ip-codec@npm:2.0.4" @@ -7991,6 +8080,15 @@ __metadata: languageName: node linkType: hard +"@mistralai/mistralai@npm:^0.4.0": + version: 0.4.0 + resolution: "@mistralai/mistralai@npm:0.4.0" + dependencies: + node-fetch: "npm:^2.6.7" + checksum: 1857ceb56f9119e8248ebb3947f8ee1da6e0731aeced38e1de44823448376ddc733182dd782e50a0e448ddd6b01789bd9dc622e0594e7a354af302a06ced77e7 + languageName: node + linkType: hard + "@mole-inc/bin-wrapper@npm:^8.0.1": version: 8.0.1 resolution: "@mole-inc/bin-wrapper@npm:8.0.1" @@ -8650,6 +8748,13 @@ __metadata: languageName: node linkType: hard +"@next/env@npm:14.2.4": + version: 14.2.4 + resolution: "@next/env@npm:14.2.4" + checksum: cc284e3dd0666df04d8321645d8409c10cb8e325884c226abbb2e7bea20f0a4232f988216aa506a9d0457b46f28b594a61179d1e978c0ca22497cd8cab8196c7 + languageName: node + linkType: hard + "@next/eslint-plugin-next@npm:14.0.4": version: 14.0.4 resolution: "@next/eslint-plugin-next@npm:14.0.4" @@ -8675,6 +8780,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-darwin-arm64@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-darwin-arm64@npm:14.2.4" + conditions: os=darwin & cpu=arm64 + languageName: node + linkType: hard + "@next/swc-darwin-x64@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-darwin-x64@npm:14.0.4" @@ -8682,6 +8794,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-darwin-x64@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-darwin-x64@npm:14.2.4" + conditions: os=darwin & cpu=x64 + languageName: node + linkType: hard + "@next/swc-linux-arm64-gnu@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-linux-arm64-gnu@npm:14.0.4" @@ -8689,6 +8808,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-linux-arm64-gnu@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-linux-arm64-gnu@npm:14.2.4" + conditions: os=linux & cpu=arm64 & libc=glibc + languageName: node + linkType: hard + "@next/swc-linux-arm64-musl@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-linux-arm64-musl@npm:14.0.4" @@ -8696,6 +8822,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-linux-arm64-musl@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-linux-arm64-musl@npm:14.2.4" + conditions: os=linux & cpu=arm64 & libc=musl + languageName: node + linkType: hard + "@next/swc-linux-x64-gnu@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-linux-x64-gnu@npm:14.0.4" @@ -8703,6 +8836,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-linux-x64-gnu@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-linux-x64-gnu@npm:14.2.4" + conditions: os=linux & cpu=x64 & libc=glibc + languageName: node + linkType: hard + "@next/swc-linux-x64-musl@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-linux-x64-musl@npm:14.0.4" @@ -8710,6 +8850,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-linux-x64-musl@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-linux-x64-musl@npm:14.2.4" + conditions: os=linux & cpu=x64 & libc=musl + languageName: node + linkType: hard + "@next/swc-win32-arm64-msvc@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-win32-arm64-msvc@npm:14.0.4" @@ -8717,6 +8864,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-win32-arm64-msvc@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-win32-arm64-msvc@npm:14.2.4" + conditions: os=win32 & cpu=arm64 + languageName: node + linkType: hard + "@next/swc-win32-ia32-msvc@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-win32-ia32-msvc@npm:14.0.4" @@ -8724,6 +8878,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-win32-ia32-msvc@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-win32-ia32-msvc@npm:14.2.4" + conditions: os=win32 & cpu=ia32 + languageName: node + linkType: hard + "@next/swc-win32-x64-msvc@npm:14.0.4": version: 14.0.4 resolution: "@next/swc-win32-x64-msvc@npm:14.0.4" @@ -8731,6 +8892,13 @@ __metadata: languageName: node linkType: hard +"@next/swc-win32-x64-msvc@npm:14.2.4": + version: 14.2.4 + resolution: "@next/swc-win32-x64-msvc@npm:14.2.4" + conditions: os=win32 & cpu=x64 + languageName: node + linkType: hard + "@nivo/calendar@npm:^0.84.0": version: 0.84.0 resolution: "@nivo/calendar@npm:0.84.0" @@ -15630,6 +15798,16 @@ __metadata: languageName: node linkType: hard +"@swc/helpers@npm:0.5.5": + version: 0.5.5 + resolution: "@swc/helpers@npm:0.5.5" + dependencies: + "@swc/counter": "npm:^0.1.3" + tslib: "npm:^2.4.0" + checksum: 21a9b9cfe7e00865f9c9f3eb4c1cc5b397143464f7abee76a2c5366e591e06b0155b5aac93fe8269ef8d548df253f6fd931e9ddfc0fd12efd405f90f45506e7d + languageName: node + linkType: hard + "@swc/helpers@npm:~0.5.0": version: 0.5.3 resolution: "@swc/helpers@npm:0.5.3" @@ -17265,6 +17443,15 @@ __metadata: languageName: node linkType: hard +"@types/lodash.chunk@npm:^4.2.9": + version: 4.2.9 + resolution: "@types/lodash.chunk@npm:4.2.9" + dependencies: + "@types/lodash": "npm:*" + checksum: 5759b3d969c5db4b0893b70261ae40d4b9a6466c984c16de6fa1d3945b3199cc09f948a444a3b4e6cfa0dd984044cf937cbc8dab5fe0ac8da67244ed74d9e4e4 + languageName: node + linkType: hard + "@types/lodash.compact@npm:^3.0.9": version: 3.0.9 resolution: "@types/lodash.compact@npm:3.0.9" @@ -17382,6 +17569,15 @@ __metadata: languageName: node linkType: hard +"@types/lodash.omitby@npm:^4.6.9": + version: 4.6.9 + resolution: "@types/lodash.omitby@npm:4.6.9" + dependencies: + "@types/lodash": "npm:*" + checksum: e8850219326634c5b531e3398d24701000328e4366504d9315c1660c2fe2a0d4fc9aa2983b8c652ee7239921cd16103b37b1e44efdf25658de2a36f64b76888a + languageName: node + linkType: hard + "@types/lodash.pick@npm:^4.3.7": version: 4.4.9 resolution: "@types/lodash.pick@npm:4.4.9" @@ -17614,6 +17810,15 @@ __metadata: languageName: node linkType: hard +"@types/node@npm:^18.11.18": + version: 18.19.34 + resolution: "@types/node@npm:18.19.34" + dependencies: + undici-types: "npm:~5.26.4" + checksum: e985f50684def801801069e236165ee511f9195fc04ad4a2af7642d86aeaeaf7bfe34c147f894a48618a5c71c15b388ca91341a244792149543a712e38351988 + languageName: node + linkType: hard + "@types/nodemailer@npm:^6.4.14": version: 6.4.14 resolution: "@types/nodemailer@npm:6.4.14" @@ -19171,6 +19376,15 @@ __metadata: languageName: node linkType: hard +"abort-controller@npm:^3.0.0": + version: 3.0.0 + resolution: "abort-controller@npm:3.0.0" + dependencies: + event-target-shim: "npm:^5.0.0" + checksum: 90ccc50f010250152509a344eb2e71977fbf8db0ab8f1061197e3275ddf6c61a41a6edfd7b9409c664513131dd96e962065415325ef23efa5db931b382d24ca5 + languageName: node + linkType: hard + "accepts@npm:^1.3.5, accepts@npm:~1.3.4, accepts@npm:~1.3.5, accepts@npm:~1.3.8": version: 1.3.8 resolution: "accepts@npm:1.3.8" @@ -21495,7 +21709,7 @@ __metadata: languageName: node linkType: hard -"base64-js@npm:^1.0.2, base64-js@npm:^1.3.0, base64-js@npm:^1.3.1": +"base64-js@npm:^1.0.2, base64-js@npm:^1.3.0, base64-js@npm:^1.3.1, base64-js@npm:^1.5.1": version: 1.5.1 resolution: "base64-js@npm:1.5.1" checksum: f23823513b63173a001030fae4f2dabe283b99a9d324ade3ad3d148e218134676f1ee8568c877cd79ec1c53158dcf2d2ba527a97c606618928ba99dd930102bf @@ -21644,6 +21858,20 @@ __metadata: languageName: node linkType: hard +"binary-extensions@npm:^2.2.0": + version: 2.3.0 + resolution: "binary-extensions@npm:2.3.0" + checksum: 75a59cafc10fb12a11d510e77110c6c7ae3f4ca22463d52487709ca7f18f69d886aa387557cc9864fbdb10153d0bdb4caacabf11541f55e89ed6e18d12ece2b5 + languageName: node + linkType: hard + +"binary-search@npm:^1.3.5": + version: 1.3.6 + resolution: "binary-search@npm:1.3.6" + checksum: 786a770e3411cf563c9c7829e2854d79583a207b8faaa5022f93352893e1d06035ae5d80de1b168dcbd9d346fdb0dd2e3d7fcdf309b3a63dc027e92624da32a0 + languageName: node + linkType: hard + "binaryextensions@npm:^4.15.0, binaryextensions@npm:^4.16.0": version: 4.19.0 resolution: "binaryextensions@npm:4.19.0" @@ -22476,6 +22704,13 @@ __metadata: languageName: node linkType: hard +"camelcase@npm:6, camelcase@npm:^6.2.0": + version: 6.3.0 + resolution: "camelcase@npm:6.3.0" + checksum: 0d701658219bd3116d12da3eab31acddb3f9440790c0792e0d398f0a520a6a4058018e546862b6fba89d7ae990efaeb97da71e1913e9ebf5a8b5621a3d55c710 + languageName: node + linkType: hard + "camelcase@npm:^5.0.0, camelcase@npm:^5.3.1": version: 5.3.1 resolution: "camelcase@npm:5.3.1" @@ -22483,13 +22718,6 @@ __metadata: languageName: node linkType: hard -"camelcase@npm:^6.2.0": - version: 6.3.0 - resolution: "camelcase@npm:6.3.0" - checksum: 0d701658219bd3116d12da3eab31acddb3f9440790c0792e0d398f0a520a6a4058018e546862b6fba89d7ae990efaeb97da71e1913e9ebf5a8b5621a3d55c710 - languageName: node - linkType: hard - "camelcase@npm:^7.0.1": version: 7.0.1 resolution: "camelcase@npm:7.0.1" @@ -22518,24 +22746,17 @@ __metadata: languageName: node linkType: hard -"caniuse-lite@npm:^1.0.0, caniuse-lite@npm:^1.0.30001538, caniuse-lite@npm:^1.0.30001565": - version: 1.0.30001568 - resolution: "caniuse-lite@npm:1.0.30001568" - checksum: 13f01e5a2481134bd61cf565ce9fecbd8e107902927a0dcf534230a92191a81f1715792170f5f39719c767c3a96aa6df9917a8d5601f15bbd5e4041a8cfecc99 - languageName: node - linkType: hard - -"caniuse-lite@npm:^1.0.30001406": - version: 1.0.30001571 - resolution: "caniuse-lite@npm:1.0.30001571" - checksum: 632f476e39febbfb5dc91c236981f3d518dc0cf55c42cc2bba431a6b6f4cceae3f9cd74d26312f30e9de65a3cc92ccf80d964ba8de061e25f37b7f0518303dad +"caniuse-lite@npm:^1.0.0, caniuse-lite@npm:^1.0.30001406, caniuse-lite@npm:^1.0.30001538, caniuse-lite@npm:^1.0.30001565, caniuse-lite@npm:^1.0.30001587": + version: 1.0.30001636 + resolution: "caniuse-lite@npm:1.0.30001636" + checksum: e5f965b4da7bae1531fd9f93477d015729ff9e3fa12670ead39a9e6cdc4c43e62c272d47857c5cc332e7b02d697cb3f2f965a1030870ac7476da60c2fc81ee94 languageName: node linkType: hard -"caniuse-lite@npm:^1.0.30001587": - version: 1.0.30001589 - resolution: "caniuse-lite@npm:1.0.30001589" - checksum: 20debfb949413f603011bc7dacaf050010778bc4f8632c86fafd1bd0c43180c95ae7c31f6c82348f6309e5e221934e327c3607a216e3f09640284acf78cd6d4d +"caniuse-lite@npm:^1.0.30001579": + version: 1.0.30001640 + resolution: "caniuse-lite@npm:1.0.30001640" + checksum: d87fce999e52c354029893a23887d2e48ac297e3af55bd14161fcafdd711f97bdb2649c79d2d3049e628603cb59bc4257ca2961644b0b8d206e7b7dd126d37ea languageName: node linkType: hard @@ -23573,7 +23794,7 @@ __metadata: languageName: node linkType: hard -"commander@npm:^10.0.0": +"commander@npm:^10.0.0, commander@npm:^10.0.1": version: 10.0.1 resolution: "commander@npm:10.0.1" checksum: 53f33d8927758a911094adadda4b2cbac111a5b377d8706700587650fd8f45b0bbe336de4b5c3fe47fd61f420a3d9bd452b6e0e6e5600a7e74d7bf0174f6efe3 @@ -25096,7 +25317,7 @@ __metadata: languageName: node linkType: hard -"decamelize@npm:^1.2.0": +"decamelize@npm:1.2.0, decamelize@npm:^1.2.0": version: 1.2.0 resolution: "decamelize@npm:1.2.0" checksum: 85c39fe8fbf0482d4a1e224ef0119db5c1897f8503bcef8b826adff7a1b11414972f6fef2d7dec2ee0b4be3863cf64ac1439137ae9e6af23a3d8dcbe26a5b4b2 @@ -27553,6 +27774,13 @@ __metadata: languageName: node linkType: hard +"event-target-shim@npm:^5.0.0": + version: 5.0.1 + resolution: "event-target-shim@npm:5.0.1" + checksum: 0255d9f936215fd206156fd4caa9e8d35e62075d720dc7d847e89b417e5e62cf1ce6c9b4e0a1633a9256de0efefaf9f8d26924b1f3c8620cffb9db78e7d3076b + languageName: node + linkType: hard + "eventemitter2@npm:6.4.9": version: 6.4.9 resolution: "eventemitter2@npm:6.4.9" @@ -28653,6 +28881,13 @@ __metadata: languageName: node linkType: hard +"form-data-encoder@npm:1.7.2": + version: 1.7.2 + resolution: "form-data-encoder@npm:1.7.2" + checksum: 56553768037b6d55d9de524f97fe70555f0e415e781cb56fc457a68263de3d40fadea2304d4beef2d40b1a851269bd7854e42c362107071892cb5238debe9464 + languageName: node + linkType: hard + "form-data-encoder@npm:^2.1.2": version: 2.1.4 resolution: "form-data-encoder@npm:2.1.4" @@ -28689,7 +28924,7 @@ __metadata: languageName: node linkType: hard -"formdata-node@npm:^4.4.1": +"formdata-node@npm:^4.3.2, formdata-node@npm:^4.4.1": version: 4.4.1 resolution: "formdata-node@npm:4.4.1" dependencies: @@ -29108,6 +29343,7 @@ __metadata: "@types/jest": "npm:^29.5.11" "@types/js-cookie": "npm:^3.0.3" "@types/lodash.camelcase": "npm:^4.3.7" + "@types/lodash.chunk": "npm:^4.2.9" "@types/lodash.compact": "npm:^3.0.9" "@types/lodash.debounce": "npm:^4.0.7" "@types/lodash.groupby": "npm:^4.6.9" @@ -29219,6 +29455,7 @@ __metadata: jsonwebtoken: "npm:^9.0.0" libphonenumber-js: "npm:^1.10.26" lodash.camelcase: "npm:^4.3.0" + lodash.chunk: "npm:^4.2.0" lodash.compact: "npm:^3.0.1" lodash.debounce: "npm:^4.0.8" lodash.groupby: "npm:^4.6.0" @@ -32135,6 +32372,13 @@ __metadata: languageName: node linkType: hard +"is-any-array@npm:^2.0.0": + version: 2.0.1 + resolution: "is-any-array@npm:2.0.1" + checksum: f9807458a51e63ca1ac27fd6f3a3ace8200f077094e00d9b05b24cfbc9d5594d586d6ecf3416271f26939d5cb93fc52ca869cb5744e77318c3f53ec70b08d61f + languageName: node + linkType: hard + "is-arguments@npm:^1.0.4, is-arguments@npm:^1.1.1": version: 1.1.1 resolution: "is-arguments@npm:1.1.1" @@ -33884,6 +34128,15 @@ __metadata: languageName: node linkType: hard +"js-tiktoken@npm:^1.0.12": + version: 1.0.12 + resolution: "js-tiktoken@npm:1.0.12" + dependencies: + base64-js: "npm:^1.5.1" + checksum: 7afb4826e21342386a1884754fbc1c1828f948c4dd0ab093bf778d1323e65343bd5343d15f7cda46af396f1fe4a0297739936149b7c40a0601eefe3fcaef8727 + languageName: node + linkType: hard + "js-tokens@npm:^3.0.0 || ^4.0.0, js-tokens@npm:^4.0.0": version: 4.0.0 resolution: "js-tokens@npm:4.0.0" @@ -34360,7 +34613,7 @@ __metadata: languageName: node linkType: hard -"jsonpointer@npm:^5.0.0": +"jsonpointer@npm:^5.0.0, jsonpointer@npm:^5.0.1": version: 5.0.1 resolution: "jsonpointer@npm:5.0.1" checksum: 89929e58b400fcb96928c0504fcf4fc3f919d81e9543ceb055df125538470ee25290bb4984251e172e6ef8fcc55761eb998c118da763a82051ad89d4cb073fe7 @@ -34560,6 +34813,242 @@ __metadata: languageName: node linkType: hard +"langchain@npm:^0.2.6": + version: 0.2.6 + resolution: "langchain@npm:0.2.6" + dependencies: + "@langchain/core": "npm:~0.2.0" + "@langchain/openai": "npm:>=0.1.0 <0.3.0" + "@langchain/textsplitters": "npm:~0.0.0" + binary-extensions: "npm:^2.2.0" + js-tiktoken: "npm:^1.0.12" + js-yaml: "npm:^4.1.0" + jsonpointer: "npm:^5.0.1" + langchainhub: "npm:~0.0.8" + langsmith: "npm:~0.1.30" + ml-distance: "npm:^4.0.0" + openapi-types: "npm:^12.1.3" + p-retry: "npm:4" + uuid: "npm:^9.0.0" + yaml: "npm:^2.2.1" + zod: "npm:^3.22.4" + zod-to-json-schema: "npm:^3.22.3" + peerDependencies: + "@aws-sdk/client-s3": ^3.310.0 + "@aws-sdk/client-sagemaker-runtime": ^3.310.0 + "@aws-sdk/client-sfn": ^3.310.0 + "@aws-sdk/credential-provider-node": ^3.388.0 + "@azure/storage-blob": ^12.15.0 + "@browserbasehq/sdk": "*" + "@gomomento/sdk": ^1.51.1 + "@gomomento/sdk-core": ^1.51.1 + "@gomomento/sdk-web": ^1.51.1 + "@mendable/firecrawl-js": ^0.0.13 + "@notionhq/client": ^2.2.10 + "@pinecone-database/pinecone": "*" + "@supabase/supabase-js": ^2.10.0 + "@vercel/kv": ^0.2.3 + "@xata.io/client": ^0.28.0 + apify-client: ^2.7.1 + assemblyai: ^4.0.0 + axios: "*" + cheerio: ^1.0.0-rc.12 + chromadb: "*" + convex: ^1.3.1 + couchbase: ^4.3.0 + d3-dsv: ^2.0.0 + epub2: ^3.0.1 + fast-xml-parser: "*" + handlebars: ^4.7.8 + html-to-text: ^9.0.5 + ignore: ^5.2.0 + ioredis: ^5.3.2 + jsdom: "*" + mammoth: ^1.6.0 + mongodb: ">=5.2.0" + node-llama-cpp: "*" + notion-to-md: ^3.1.0 + officeparser: ^4.0.4 + pdf-parse: 1.1.1 + peggy: ^3.0.2 + playwright: ^1.32.1 + puppeteer: ^19.7.2 + pyodide: ^0.24.1 + redis: ^4.6.4 + sonix-speech-recognition: ^2.1.1 + srt-parser-2: ^1.2.3 + typeorm: ^0.3.20 + weaviate-ts-client: "*" + web-auth-library: ^1.0.3 + ws: ^8.14.2 + youtube-transcript: ^1.0.6 + youtubei.js: ^9.1.0 + peerDependenciesMeta: + "@aws-sdk/client-s3": + optional: true + "@aws-sdk/client-sagemaker-runtime": + optional: true + "@aws-sdk/client-sfn": + optional: true + "@aws-sdk/credential-provider-node": + optional: true + "@azure/storage-blob": + optional: true + "@browserbasehq/sdk": + optional: true + "@gomomento/sdk": + optional: true + "@gomomento/sdk-core": + optional: true + "@gomomento/sdk-web": + optional: true + "@mendable/firecrawl-js": + optional: true + "@notionhq/client": + optional: true + "@pinecone-database/pinecone": + optional: true + "@supabase/supabase-js": + optional: true + "@vercel/kv": + optional: true + "@xata.io/client": + optional: true + apify-client: + optional: true + assemblyai: + optional: true + axios: + optional: true + cheerio: + optional: true + chromadb: + optional: true + convex: + optional: true + couchbase: + optional: true + d3-dsv: + optional: true + epub2: + optional: true + faiss-node: + optional: true + fast-xml-parser: + optional: true + handlebars: + optional: true + html-to-text: + optional: true + ignore: + optional: true + ioredis: + optional: true + jsdom: + optional: true + mammoth: + optional: true + mongodb: + optional: true + node-llama-cpp: + optional: true + notion-to-md: + optional: true + officeparser: + optional: true + pdf-parse: + optional: true + peggy: + optional: true + playwright: + optional: true + puppeteer: + optional: true + pyodide: + optional: true + redis: + optional: true + sonix-speech-recognition: + optional: true + srt-parser-2: + optional: true + typeorm: + optional: true + weaviate-ts-client: + optional: true + web-auth-library: + optional: true + ws: + optional: true + youtube-transcript: + optional: true + youtubei.js: + optional: true + checksum: c267d618f20b75eeba0c13b3ee9aaa8e7a57d87d64344c4360d7332bdda82c6976c80c705a81a0be02006f350b2f42142ca98a7b71148ee0aa934a592ddbc47a + languageName: node + linkType: hard + +"langchainhub@npm:~0.0.8": + version: 0.0.11 + resolution: "langchainhub@npm:0.0.11" + checksum: 6ed781b9e8165bfb5cedc822a25bc70df0f3fc02662061d19a5e2044243cfae797857a05d139de8f326539b1f3fe03f2662060eed82669e405181f1f0f435c47 + languageName: node + linkType: hard + +"langfuse-core@npm:^3.11.2": + version: 3.11.2 + resolution: "langfuse-core@npm:3.11.2" + dependencies: + mustache: "npm:^4.2.0" + checksum: 341cddedf16cf0c4b980989c4ee4daa6be0dad7a37349d86ff7b3fcf4e2e35e25318fb9ec6e4c9d27023031c7e780d50d75493f7cf9911938b6581ee2c1728c6 + languageName: node + linkType: hard + +"langfuse-langchain@npm:^3.11.2": + version: 3.11.2 + resolution: "langfuse-langchain@npm:3.11.2" + dependencies: + langfuse: "npm:^3.11.2" + langfuse-core: "npm:^3.11.2" + peerDependencies: + langchain: ">=0.0.157 <0.3.0" + checksum: ec57481128b4b738ee4e146b0f0c42e94d0f75d5a525a031407d7ecd8ffcd50dc9730f580901ed2432694555e544459e032e9181af9d853a156aef42c3ef9b41 + languageName: node + linkType: hard + +"langfuse@npm:^3.11.2": + version: 3.11.2 + resolution: "langfuse@npm:3.11.2" + dependencies: + langfuse-core: "npm:^3.11.2" + checksum: e74053aa5bb3e62a91d3c7a5f95c86f9808342ca9a0dcda5c5a33bf8abf8ace70357f84db7fae36e93d722e9aa383b55ced785e6d6cbf23d47d0252b1fef57b0 + languageName: node + linkType: hard + +"langsmith@npm:~0.1.30": + version: 0.1.30 + resolution: "langsmith@npm:0.1.30" + dependencies: + "@types/uuid": "npm:^9.0.1" + commander: "npm:^10.0.1" + p-queue: "npm:^6.6.2" + p-retry: "npm:4" + uuid: "npm:^9.0.0" + peerDependencies: + "@langchain/core": "*" + langchain: "*" + openai: "*" + peerDependenciesMeta: + "@langchain/core": + optional: true + langchain: + optional: true + openai: + optional: true + checksum: 181719d73bd89918f0ab60768f824449e2bfd5a691242c3540291114053ba6e49ce4670bfa0abd129dd7556f80b3025371ba9cd1884090dc1941fdf39850545c + languageName: node + linkType: hard + "language-subtag-registry@npm:^0.3.20": version: 0.3.22 resolution: "language-subtag-registry@npm:0.3.22" @@ -34867,6 +35356,13 @@ __metadata: languageName: node linkType: hard +"lodash.chunk@npm:^4.2.0": + version: 4.2.0 + resolution: "lodash.chunk@npm:4.2.0" + checksum: f9f99969561ad2f62af1f9a96c5bd0af776f000292b0d8db3126c28eb3b32e210d7c31b49c18d0d7901869bd769057046dc134b60cfa0c2c4ce017823a26bb23 + languageName: node + linkType: hard + "lodash.clonedeep@npm:^4.5.0": version: 4.5.0 resolution: "lodash.clonedeep@npm:4.5.0" @@ -35091,6 +35587,13 @@ __metadata: languageName: node linkType: hard +"lodash.omitby@npm:^4.6.0": + version: 4.6.0 + resolution: "lodash.omitby@npm:4.6.0" + checksum: 4608b1d8c4063b63349a3462852465fbe74781d737fbb26a0a7f00b0e65f6ccbc13fa490a38f9380103d93fc398e3873983038efadfafc67ccafbb25d9bc7bf4 + languageName: node + linkType: hard + "lodash.once@npm:^4.0.0": version: 4.1.1 resolution: "lodash.once@npm:4.1.1" @@ -37936,6 +38439,52 @@ __metadata: languageName: node linkType: hard +"ml-array-mean@npm:^1.1.6": + version: 1.1.6 + resolution: "ml-array-mean@npm:1.1.6" + dependencies: + ml-array-sum: "npm:^1.1.6" + checksum: 41ab68308e3472702f775a49c8ab9ee1e678e01cd59dbc59424c0f1017a37df1bb638e702831305f0e6366300eca48353f526773ab8f4d8d142a64d0461f9944 + languageName: node + linkType: hard + +"ml-array-sum@npm:^1.1.6": + version: 1.1.6 + resolution: "ml-array-sum@npm:1.1.6" + dependencies: + is-any-array: "npm:^2.0.0" + checksum: fb3973ce2bfa19ab4f5e657f722494425b57025547657d332bf5bafe3e4e7e4bd1e082dfb970bcc0bfa87efa345b80a20a596dbb204be7b4802b52c35fd6cf77 + languageName: node + linkType: hard + +"ml-distance-euclidean@npm:^2.0.0": + version: 2.0.0 + resolution: "ml-distance-euclidean@npm:2.0.0" + checksum: 877aef472e134f79be9540b02f889b2a27976ca45d77f5d4ef7d8dd24058a60cf4637365b40a5aba1ab5490348a0fb1b3803143b25af88cdc66137fbfd665442 + languageName: node + linkType: hard + +"ml-distance@npm:^4.0.0": + version: 4.0.1 + resolution: "ml-distance@npm:4.0.1" + dependencies: + ml-array-mean: "npm:^1.1.6" + ml-distance-euclidean: "npm:^2.0.0" + ml-tree-similarity: "npm:^1.0.0" + checksum: 8c2eb077d2ba61437f2414f3b9ca1091c43fcabe2282ecc31d8ebf9e083c1df068e43c67f59a4674e7c8f473201ace4f02779b446427d6169a5d669cae94c816 + languageName: node + linkType: hard + +"ml-tree-similarity@npm:^1.0.0": + version: 1.0.0 + resolution: "ml-tree-similarity@npm:1.0.0" + dependencies: + binary-search: "npm:^1.3.5" + num-sort: "npm:^2.0.0" + checksum: e3ecd07bead5d18bc7b6fed1dfefbe65aea4008d5556181b94b7d70550fba543d2501b224f12a9f5197c1d23d95faef2accc7fd265c5afd15ef55a38190ffc6e + languageName: node + linkType: hard + "mlly@npm:^1.2.0, mlly@npm:^1.4.2": version: 1.6.1 resolution: "mlly@npm:1.6.1" @@ -38162,6 +38711,15 @@ __metadata: languageName: node linkType: hard +"mustache@npm:^4.2.0": + version: 4.2.0 + resolution: "mustache@npm:4.2.0" + bin: + mustache: bin/mustache + checksum: 1f8197e8a19e63645a786581d58c41df7853da26702dbc005193e2437c98ca49b255345c173d50c08fe4b4dbb363e53cb655ecc570791f8deb09887248dd34a2 + languageName: node + linkType: hard + "mute-stream@npm:0.0.8": version: 0.0.8 resolution: "mute-stream@npm:0.0.8" @@ -38305,6 +38863,19 @@ __metadata: languageName: node linkType: hard +"next-runtime-env@npm:^3.2.2": + version: 3.2.2 + resolution: "next-runtime-env@npm:3.2.2" + dependencies: + next: "npm:^14" + react: "npm:^18" + peerDependencies: + next: ^14 + react: ^18 + checksum: 9ac2649fd765b82f340af5d77083f851a4e865acc9e32e9df092ba06cf5f4066b94ee7a3994677ef362d59802457f5becad8d49f3c9a75e77b68179e65c5ecee + languageName: node + linkType: hard + "next-tick@npm:1, next-tick@npm:^1.1.0": version: 1.1.0 resolution: "next-tick@npm:1.1.0" @@ -38368,6 +38939,64 @@ __metadata: languageName: node linkType: hard +"next@npm:^14": + version: 14.2.4 + resolution: "next@npm:14.2.4" + dependencies: + "@next/env": "npm:14.2.4" + "@next/swc-darwin-arm64": "npm:14.2.4" + "@next/swc-darwin-x64": "npm:14.2.4" + "@next/swc-linux-arm64-gnu": "npm:14.2.4" + "@next/swc-linux-arm64-musl": "npm:14.2.4" + "@next/swc-linux-x64-gnu": "npm:14.2.4" + "@next/swc-linux-x64-musl": "npm:14.2.4" + "@next/swc-win32-arm64-msvc": "npm:14.2.4" + "@next/swc-win32-ia32-msvc": "npm:14.2.4" + "@next/swc-win32-x64-msvc": "npm:14.2.4" + "@swc/helpers": "npm:0.5.5" + busboy: "npm:1.6.0" + caniuse-lite: "npm:^1.0.30001579" + graceful-fs: "npm:^4.2.11" + postcss: "npm:8.4.31" + styled-jsx: "npm:5.1.1" + peerDependencies: + "@opentelemetry/api": ^1.1.0 + "@playwright/test": ^1.41.2 + react: ^18.2.0 + react-dom: ^18.2.0 + sass: ^1.3.0 + dependenciesMeta: + "@next/swc-darwin-arm64": + optional: true + "@next/swc-darwin-x64": + optional: true + "@next/swc-linux-arm64-gnu": + optional: true + "@next/swc-linux-arm64-musl": + optional: true + "@next/swc-linux-x64-gnu": + optional: true + "@next/swc-linux-x64-musl": + optional: true + "@next/swc-win32-arm64-msvc": + optional: true + "@next/swc-win32-ia32-msvc": + optional: true + "@next/swc-win32-x64-msvc": + optional: true + peerDependenciesMeta: + "@opentelemetry/api": + optional: true + "@playwright/test": + optional: true + sass: + optional: true + bin: + next: dist/bin/next + checksum: 630c2a197b57c1f29caf4672a0f8fb74dbb048e77e4513f567279467332212f3eebcb68279885f1d525d7aaebbb452f522b02c0b5cd3ca66f385341e4b4eac67 + languageName: node + linkType: hard + "nice-napi@npm:^1.0.2": version: 1.0.2 resolution: "nice-napi@npm:1.0.2" @@ -39029,6 +39658,13 @@ __metadata: languageName: node linkType: hard +"num-sort@npm:^2.0.0": + version: 2.1.0 + resolution: "num-sort@npm:2.1.0" + checksum: cc1d43adbc9adfd5d208a8eb653827277376ff2e6eb75379f96e6a23d481040e317e63505e075b84ce49e19b9d960570646096428a715d12c5ef1381504d5135 + languageName: node + linkType: hard + "number-is-nan@npm:^1.0.0": version: 1.0.1 resolution: "number-is-nan@npm:1.0.1" @@ -39421,6 +40057,24 @@ __metadata: languageName: node linkType: hard +"openai@npm:^4.49.1": + version: 4.51.0 + resolution: "openai@npm:4.51.0" + dependencies: + "@types/node": "npm:^18.11.18" + "@types/node-fetch": "npm:^2.6.4" + abort-controller: "npm:^3.0.0" + agentkeepalive: "npm:^4.2.1" + form-data-encoder: "npm:1.7.2" + formdata-node: "npm:^4.3.2" + node-fetch: "npm:^2.6.7" + web-streams-polyfill: "npm:^3.2.1" + bin: + openai: bin/cli + checksum: c9adcca092aa528fe3556d9e91e022f67515e76c31ee6899d781b5bf17fbfd783e7aca15da4b6dca4bed53d12da021973789f5ae584d2769c5c560976939c45b + languageName: node + linkType: hard + "openapi-types@npm:^12.1.3": version: 12.1.3 resolution: "openapi-types@npm:12.1.3" @@ -39735,7 +40389,7 @@ __metadata: languageName: node linkType: hard -"p-retry@npm:^4.5.0": +"p-retry@npm:4, p-retry@npm:^4.5.0": version: 4.6.2 resolution: "p-retry@npm:4.6.2" dependencies: @@ -47497,6 +48151,8 @@ __metadata: resolution: "twenty-server@workspace:packages/twenty-server" dependencies: "@graphql-yoga/nestjs": "patch:@graphql-yoga/nestjs@2.1.0#./patches/@graphql-yoga-nestjs-npm-2.1.0-cb509e6047.patch" + "@langchain/mistralai": "npm:^0.0.24" + "@langchain/openai": "npm:^0.1.3" "@nestjs/cache-manager": "npm:^2.2.1" "@nestjs/cli": "npm:10.3.0" "@nestjs/devtools-integration": "npm:^0.1.6" @@ -47508,6 +48164,7 @@ __metadata: "@types/lodash.isequal": "npm:^4.5.8" "@types/lodash.isobject": "npm:^3.0.7" "@types/lodash.omit": "npm:^4.5.9" + "@types/lodash.omitby": "npm:^4.6.9" "@types/lodash.snakecase": "npm:^4.1.7" "@types/lodash.uniq": "npm:^4.5.9" "@types/lodash.uniqby": "npm:^4.7.9" @@ -47519,7 +48176,10 @@ __metadata: graphql-middleware: "npm:^6.1.35" jsdom: "npm:~22.1.0" jwt-decode: "npm:^4.0.0" + langchain: "npm:^0.2.6" + langfuse-langchain: "npm:^3.11.2" lodash.differencewith: "npm:^4.5.0" + lodash.omitby: "npm:^4.6.0" lodash.uniq: "npm:^4.5.0" lodash.uniqby: "npm:^4.7.0" passport: "npm:^0.7.0" @@ -47527,6 +48187,7 @@ __metadata: rimraf: "npm:^5.0.5" tsconfig-paths: "npm:^4.2.0" typescript: "npm:5.3.3" + zod-to-json-schema: "npm:^3.23.1" languageName: unknown linkType: soft @@ -47546,6 +48207,7 @@ __metadata: version: 0.0.0-use.local resolution: "twenty-website@workspace:packages/twenty-website" dependencies: + next-runtime-env: "npm:^3.2.2" postgres: "npm:^3.4.3" languageName: unknown linkType: soft @@ -50302,6 +50964,15 @@ __metadata: languageName: node linkType: hard +"yaml@npm:^2.2.1": + version: 2.4.5 + resolution: "yaml@npm:2.4.5" + bin: + yaml: bin.mjs + checksum: e1ee78b381e5c710f715cc4082fd10fc82f7f5c92bd6f075771d20559e175616f56abf1c411f545ea0e9e16e4f84a83a50b42764af5f16ec006328ba9476bb31 + languageName: node + linkType: hard + "yaml@npm:^2.2.2, yaml@npm:^2.3.4": version: 2.3.4 resolution: "yaml@npm:2.3.4" @@ -50634,7 +51305,25 @@ __metadata: languageName: node linkType: hard -"zod@npm:3.23.8": +"zod-to-json-schema@npm:^3.22.3, zod-to-json-schema@npm:^3.22.4": + version: 3.23.0 + resolution: "zod-to-json-schema@npm:3.23.0" + peerDependencies: + zod: ^3.23.3 + checksum: bcd966fa040765d7170a89c0c5f1717575e7d8823b84cbbb606689d494ae308c9eaadd4b71a74752e3170deef64c1f1bb2985f4663c44a0ed2e7854ff6fda724 + languageName: node + linkType: hard + +"zod-to-json-schema@npm:^3.23.1": + version: 3.23.1 + resolution: "zod-to-json-schema@npm:3.23.1" + peerDependencies: + zod: ^3.23.3 + checksum: d48d733f7cba9fdc631ebe3dada3f48b820a16e49f7ded9f363cccafa42461ff95cc7afcf974c27af7cd6d5fa5191212bb7ec15ec203bcb61f829a6d0d3e192f + languageName: node + linkType: hard + +"zod@npm:3.23.8, zod@npm:^3.22.4": version: 3.23.8 resolution: "zod@npm:3.23.8" checksum: 8f14c87d6b1b53c944c25ce7a28616896319d95bc46a9660fe441adc0ed0a81253b02b5abdaeffedbeb23bdd25a0bf1c29d2c12dd919aef6447652dd295e3e69