diff --git a/.env.example b/.env.example index 7f9770aa3..e85be9c51 100644 --- a/.env.example +++ b/.env.example @@ -60,3 +60,6 @@ FILE_STORAGE= AWS_S3_BUCKET_NAME= AWS_S3_DOMAIN= + +// no of days delay the follow up for interaction +INTERACTION_FOLLOWUP_DELAY_IN_DAYS= diff --git a/.forestadmin-schema.json b/.forestadmin-schema.json index 633b047a2..6b66ef74c 100644 --- a/.forestadmin-schema.json +++ b/.forestadmin-schema.json @@ -1674,6 +1674,74 @@ "type": "String", "validations": [] }, + { + "defaultValue": null, + "enums": null, + "field": "MemberFeedbacks", + "integration": null, + "inverseOf": "Member", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "MemberFeedback.uid", + "relationship": "HasMany", + "type": ["String"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "MemberFollowUps", + "integration": null, + "inverseOf": "Member", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "MemberFollowUp.uid", + "relationship": "HasMany", + "type": ["String"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "MemberInteractions_through_Member_sourceMemberUid", + "integration": null, + "inverseOf": "Member_through_sourceMemberUid", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "MemberInteraction.uid", + "relationship": "HasMany", + "type": ["String"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "MemberInteractions_through_Member_targetMemberUid", + "integration": null, + "inverseOf": "Member_through_targetMemberUid", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "MemberInteraction.uid", + "relationship": "HasMany", + "type": ["String"], + "validations": [] + }, { "defaultValue": null, "enums": null, @@ -1745,91 +1813,587 @@ { "defaultValue": null, "enums": null, - "field": "TeamMemberRoles", + "field": "TeamMemberRoles", + "integration": null, + "inverseOf": "Member", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "TeamMemberRole.uid", + "relationship": "HasMany", + "type": ["String"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "Teams", + "integration": null, + "inverseOf": "Member", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "Team.uid", + "relationship": "HasMany", + "type": ["String"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "_MemberToMemberRoles", + "integration": null, + "inverseOf": "Member", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "_MemberToMemberRole.id", + "relationship": "HasMany", + "type": ["Number"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "_MemberToSkills", + "integration": null, + "inverseOf": "Member", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "_MemberToSkill.id", + "relationship": "HasMany", + "type": ["Number"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "airtableRecId", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "approvedAt", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Date", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "createdAt", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Date", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "discordHandler", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "email", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "externalId", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "githubHandler", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "id", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": true, + "isReadOnly": true, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Number", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "linkedinHandler", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "moreDetails", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "name", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": true, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] + }, + { + "defaultValue": null, + "enums": null, + "field": "officeHours", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": false, + "enums": null, + "field": "openToWork", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Boolean", + "validations": [] + }, + { + "defaultValue": false, + "enums": null, + "field": "plnFriend", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": true, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Boolean", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] + }, + { + "defaultValue": null, + "enums": null, + "field": "plnStartDate", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Date", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "preferences", + "integration": null, + "inverseOf": null, + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Json", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "telegramHandler", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "twitterHandler", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "uid", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": true, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] + }, + { + "defaultValue": null, + "enums": null, + "field": "updatedAt", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Date", + "validations": [] + } + ], + "icon": null, + "integration": null, + "isReadOnly": false, + "isSearchable": true, + "isVirtual": false, + "name": "Member", + "onlyForRelationships": false, + "paginationType": "page", + "segments": [] + }, + { + "actions": [], + "fields": [ + { + "defaultValue": null, + "enums": null, + "field": "Member", + "integration": null, + "inverseOf": "MemberFeedbacks", + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": true, + "isSortable": true, + "isVirtual": false, + "reference": "Member.uid", + "relationship": "BelongsTo", + "type": "String", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] + }, + { + "defaultValue": null, + "enums": null, + "field": "MemberFollowUp", + "integration": null, + "inverseOf": "MemberFeedback", + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": true, + "isSortable": true, + "isVirtual": false, + "reference": "MemberFollowUp.uid", + "relationship": "BelongsTo", + "type": "String", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] + }, + { + "defaultValue": null, + "enums": null, + "field": "comments", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": ["String"], + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "createdAt", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Date", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "data", + "integration": null, + "inverseOf": null, + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "Json", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "id", "integration": null, - "inverseOf": "Member", - "isFilterable": false, - "isPrimaryKey": false, - "isReadOnly": false, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": true, + "isReadOnly": true, "isRequired": false, - "isSortable": false, + "isSortable": true, "isVirtual": false, - "reference": "TeamMemberRole.uid", - "relationship": "HasMany", - "type": ["String"], + "reference": null, + "type": "Number", "validations": [] }, { "defaultValue": null, "enums": null, - "field": "Teams", + "field": "rating", "integration": null, - "inverseOf": "Member", - "isFilterable": false, + "inverseOf": null, + "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, "isRequired": false, - "isSortable": false, + "isSortable": true, "isVirtual": false, - "reference": "Team.uid", - "relationship": "HasMany", - "type": ["String"], + "reference": null, + "type": "Number", "validations": [] }, { "defaultValue": null, - "enums": null, - "field": "_MemberToMemberRoles", + "enums": ["POSITIVE", "NEGATIVE", "NEUTRAL"], + "field": "response", "integration": null, - "inverseOf": "Member", - "isFilterable": false, + "inverseOf": null, + "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, - "isRequired": false, - "isSortable": false, + "isRequired": true, + "isSortable": true, "isVirtual": false, - "reference": "_MemberToMemberRole.id", - "relationship": "HasMany", - "type": ["Number"], - "validations": [] + "reference": null, + "type": "Enum", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { "defaultValue": null, "enums": null, - "field": "_MemberToSkills", + "field": "type", "integration": null, - "inverseOf": "Member", - "isFilterable": false, + "inverseOf": null, + "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, - "isRequired": false, - "isSortable": false, + "isRequired": true, + "isSortable": true, "isVirtual": false, - "reference": "_MemberToSkill.id", - "relationship": "HasMany", - "type": ["Number"], - "validations": [] + "reference": null, + "type": "String", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { "defaultValue": null, "enums": null, - "field": "airtableRecId", + "field": "uid", "integration": null, "inverseOf": null, "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, - "isRequired": false, + "isRequired": true, "isSortable": true, "isVirtual": false, "reference": null, "type": "String", - "validations": [] + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { "defaultValue": null, "enums": null, - "field": "createdAt", + "field": "updatedAt", "integration": null, "inverseOf": null, "isFilterable": true, @@ -1841,43 +2405,78 @@ "reference": null, "type": "Date", "validations": [] + } + ], + "icon": null, + "integration": null, + "isReadOnly": false, + "isSearchable": true, + "isVirtual": false, + "name": "MemberFeedback", + "onlyForRelationships": false, + "paginationType": "page", + "segments": [] + }, + { + "actions": [], + "fields": [ + { + "defaultValue": null, + "enums": null, + "field": "Member", + "integration": null, + "inverseOf": "MemberFollowUps", + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": true, + "isSortable": true, + "isVirtual": false, + "reference": "Member.uid", + "relationship": "BelongsTo", + "type": "String", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { "defaultValue": null, "enums": null, - "field": "discordHandler", + "field": "MemberFeedback", "integration": null, - "inverseOf": null, + "inverseOf": "MemberFollowUp", "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, "isRequired": false, "isSortable": true, "isVirtual": false, - "reference": null, + "reference": "MemberFeedback.uid", + "relationship": "HasOne", "type": "String", "validations": [] }, { "defaultValue": null, "enums": null, - "field": "email", + "field": "MemberInteraction", "integration": null, - "inverseOf": null, + "inverseOf": "MemberFollowUps", "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, "isRequired": false, "isSortable": true, "isVirtual": false, - "reference": null, + "reference": "MemberInteraction.uid", + "relationship": "BelongsTo", "type": "String", "validations": [] }, { "defaultValue": null, "enums": null, - "field": "externalId", + "field": "createdAt", "integration": null, "inverseOf": null, "isFilterable": true, @@ -1887,23 +2486,23 @@ "isSortable": true, "isVirtual": false, "reference": null, - "type": "String", + "type": "Date", "validations": [] }, { "defaultValue": null, "enums": null, - "field": "githubHandler", + "field": "data", "integration": null, "inverseOf": null, - "isFilterable": true, + "isFilterable": false, "isPrimaryKey": false, "isReadOnly": false, "isRequired": false, "isSortable": true, "isVirtual": false, "reference": null, - "type": "String", + "type": "Json", "validations": [] }, { @@ -1923,41 +2522,45 @@ "validations": [] }, { - "defaultValue": null, + "defaultValue": false, "enums": null, - "field": "linkedinHandler", + "field": "isDelayed", "integration": null, "inverseOf": null, "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, - "isRequired": false, + "isRequired": true, "isSortable": true, "isVirtual": false, "reference": null, - "type": "String", - "validations": [] + "type": "Boolean", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { "defaultValue": null, - "enums": null, - "field": "moreDetails", + "enums": ["PENDING", "COMPLETED"], + "field": "status", "integration": null, "inverseOf": null, "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, - "isRequired": false, + "isRequired": true, "isSortable": true, "isVirtual": false, "reference": null, - "type": "String", - "validations": [] + "type": "Enum", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { "defaultValue": null, "enums": null, - "field": "name", + "field": "type", "integration": null, "inverseOf": null, "isFilterable": true, @@ -1975,23 +2578,25 @@ { "defaultValue": null, "enums": null, - "field": "officeHours", + "field": "uid", "integration": null, "inverseOf": null, "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, - "isRequired": false, + "isRequired": true, "isSortable": true, "isVirtual": false, "reference": null, "type": "String", - "validations": [] + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { - "defaultValue": false, + "defaultValue": null, "enums": null, - "field": "openToWork", + "field": "updatedAt", "integration": null, "inverseOf": null, "isFilterable": true, @@ -2001,23 +2606,55 @@ "isSortable": true, "isVirtual": false, "reference": null, - "type": "Boolean", + "type": "Date", + "validations": [] + } + ], + "icon": null, + "integration": null, + "isReadOnly": false, + "isSearchable": true, + "isVirtual": false, + "name": "MemberFollowUp", + "onlyForRelationships": false, + "paginationType": "page", + "segments": [] + }, + { + "actions": [], + "fields": [ + { + "defaultValue": null, + "enums": null, + "field": "MemberFollowUps", + "integration": null, + "inverseOf": "MemberInteraction", + "isFilterable": false, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": false, + "isVirtual": false, + "reference": "MemberFollowUp.uid", + "relationship": "HasMany", + "type": ["String"], "validations": [] }, { - "defaultValue": false, + "defaultValue": null, "enums": null, - "field": "plnFriend", + "field": "Member_through_sourceMemberUid", "integration": null, - "inverseOf": null, + "inverseOf": "MemberInteractions_through_Member_sourceMemberUid", "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, "isRequired": true, "isSortable": true, "isVirtual": false, - "reference": null, - "type": "Boolean", + "reference": "Member.uid", + "relationship": "BelongsTo", + "type": "String", "validations": [ {"type": "is present", "message": "Failed validation rule: 'Present'"} ] @@ -2025,7 +2662,24 @@ { "defaultValue": null, "enums": null, - "field": "plnStartDate", + "field": "Member_through_targetMemberUid", + "integration": null, + "inverseOf": "MemberInteractions_through_Member_targetMemberUid", + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": "Member.uid", + "relationship": "BelongsTo", + "type": "String", + "validations": [] + }, + { + "defaultValue": null, + "enums": null, + "field": "createdAt", "integration": null, "inverseOf": null, "isFilterable": true, @@ -2041,7 +2695,7 @@ { "defaultValue": null, "enums": null, - "field": "preferences", + "field": "data", "integration": null, "inverseOf": null, "isFilterable": false, @@ -2055,35 +2709,37 @@ "validations": [] }, { - "defaultValue": null, + "defaultValue": false, "enums": null, - "field": "telegramHandler", + "field": "hasFollowUp", "integration": null, "inverseOf": null, "isFilterable": true, "isPrimaryKey": false, "isReadOnly": false, - "isRequired": false, + "isRequired": true, "isSortable": true, "isVirtual": false, "reference": null, - "type": "String", - "validations": [] + "type": "Boolean", + "validations": [ + {"type": "is present", "message": "Failed validation rule: 'Present'"} + ] }, { "defaultValue": null, "enums": null, - "field": "twitterHandler", + "field": "id", "integration": null, "inverseOf": null, "isFilterable": true, - "isPrimaryKey": false, - "isReadOnly": false, + "isPrimaryKey": true, + "isReadOnly": true, "isRequired": false, "isSortable": true, "isVirtual": false, "reference": null, - "type": "String", + "type": "Number", "validations": [] }, { @@ -2126,7 +2782,7 @@ "isReadOnly": false, "isSearchable": true, "isVirtual": false, - "name": "Member", + "name": "MemberInteraction", "onlyForRelationships": false, "paginationType": "page", "segments": [] @@ -2834,6 +3490,22 @@ "type": "Number", "validations": [] }, + { + "defaultValue": null, + "enums": null, + "field": "officeHours", + "integration": null, + "inverseOf": null, + "isFilterable": true, + "isPrimaryKey": false, + "isReadOnly": false, + "isRequired": false, + "isSortable": true, + "isVirtual": false, + "reference": null, + "type": "String", + "validations": [] + }, { "defaultValue": null, "enums": null, diff --git a/apps/web-api/prisma/schema.prisma b/apps/web-api/prisma/schema.prisma index 785b108a1..941545a4e 100644 --- a/apps/web-api/prisma/schema.prisma +++ b/apps/web-api/prisma/schema.prisma @@ -82,10 +82,10 @@ model Member { eventGuests PLEventGuest[] teamFocusAreasVersionHistory TeamFocusAreaVersionHistory[] modifiedTeams Team[] @relation("LastModification") - events MemberEvent[] @relation("SourceMemberEvents") - targetEvents MemberEvent[] @relation("TargetMemberEvents") - actions MemberEventAction[] - feedbacks MemberEventFeedback[] + interactions MemberInteraction[] @relation("SourceMemberInteractions") + targetInteractions MemberInteraction[] @relation("TargetMemberInteractions") + followUps MemberFollowUp[] + feedbacks MemberFeedback[] } model MemberRole { @@ -113,18 +113,15 @@ enum PLEventType { INVITE_ONLY } -enum MemberEventType { - BOOK_OFFICE_HOURS -} - -enum MemberActionStatus { +enum MemberFollowUpStatus { PENDING COMPLETED } -enum MemberEventFeedbackType { - COMPLETED - CLOSED +enum MemberFeedbackResponseType { + POSITIVE + NEGATIVE + NEUTRAL } enum ImageSize { @@ -451,43 +448,48 @@ model TeamFocusAreaVersionHistory { @@unique([focusAreaUid, teamUid, version]) } -model MemberEvent { - id Int @id @default(autoincrement()) - uid String @unique @default(cuid()) - type MemberEventType - data Json - createdAt DateTime @default(now()) - sourceMemberUid String - sourceMember Member @relation("SourceMemberEvents", fields: [sourceMemberUid], references: [uid]) - targetMemberUid String? - targetMember Member? @relation("TargetMemberEvents", fields: [targetMemberUid], references: [uid]) - actions MemberEventAction[] -} - -model MemberEventAction { - id Int @id @default(autoincrement()) - uid String @unique @default(cuid()) - status MemberActionStatus - data Json - createdAt DateTime @default(now()) - eventUid String - event MemberEvent @relation(fields: [eventUid], references: [uid]) - createdBy String - creator Member @relation(fields: [createdBy], references: [uid]) - feedbacks MemberEventFeedback? - prevActionUid String? @unique - prevAction MemberEventAction? @relation("PreviousAction", fields: [prevActionUid], references: [uid]) - nextActions MemberEventAction? @relation("PreviousAction") -} - -model MemberEventFeedback { - id Int @id @default(autoincrement()) - uid String @unique @default(cuid()) - type String - data Json - createdAt DateTime @default(now()) - actionUid String @unique - action MemberEventAction @relation(fields: [actionUid], references: [uid]) - createdBy String - creator Member @relation(fields: [createdBy], references: [uid]) +model MemberInteraction { + id Int @id @default(autoincrement()) + uid String @unique @default(cuid()) + data Json? + hasFollowUp Boolean @default(false) + sourceMemberUid String + sourceMember Member @relation("SourceMemberInteractions", fields: [sourceMemberUid], references: [uid]) + targetMemberUid String? + targetMember Member? @relation("TargetMemberInteractions", fields: [targetMemberUid], references: [uid]) + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt + interactionFollowUps MemberFollowUp[] +} + +model MemberFollowUp { + id Int @id @default(autoincrement()) + uid String @unique @default(cuid()) + status MemberFollowUpStatus + type String + data Json? + isDelayed Boolean @default(false) + interactionUid String? + interaction MemberInteraction? @relation(fields: [interactionUid], references: [uid]) + createdBy String + creator Member @relation(fields: [createdBy], references: [uid]) + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt + feedbacks MemberFeedback? +} + +model MemberFeedback { + id Int @id @default(autoincrement()) + uid String @unique @default(cuid()) + type String + data Json? + rating Int? + comments String[] + response MemberFeedbackResponseType + followUpUid String @unique + followUp MemberFollowUp @relation(fields: [followUpUid], references: [uid]) + createdBy String + creator Member @relation(fields: [createdBy], references: [uid]) + createdAt DateTime @default(now()) + updatedAt DateTime @updatedAt } diff --git a/apps/web-api/src/app.module.ts b/apps/web-api/src/app.module.ts index aebea6b07..4bae06eaf 100644 --- a/apps/web-api/src/app.module.ts +++ b/apps/web-api/src/app.module.ts @@ -34,6 +34,9 @@ import { ProjectsModule } from './projects/projects.module'; import { JoinRequestsModule } from './join-requests/join-requests.module'; import { FocusAreasModule } from './focus-areas/focus-areas.module'; import { PLEventsModule } from './pl-events/pl-events.module'; +import { OfficeHoursModule } from './office-hours/office-hours.module'; +import { MemberFollowUpsModule } from './member-follow-ups/member-follow-ups.module'; +import { MemberFeedbacksModule } from './member-feedbacks/member-feedbacks.module'; @Module({ controllers: [AppController], @@ -83,7 +86,10 @@ import { PLEventsModule } from './pl-events/pl-events.module'; ProjectsModule, JoinRequestsModule, FocusAreasModule, - PLEventsModule + PLEventsModule, + OfficeHoursModule, + MemberFollowUpsModule, + MemberFeedbacksModule ], providers: [ { diff --git a/apps/web-api/src/member-feedbacks/member-feedbacks.module.ts b/apps/web-api/src/member-feedbacks/member-feedbacks.module.ts new file mode 100644 index 000000000..1197b46a6 --- /dev/null +++ b/apps/web-api/src/member-feedbacks/member-feedbacks.module.ts @@ -0,0 +1,17 @@ +import { Module } from '@nestjs/common'; +import { MemberFeedbacksService } from './member-feedbacks.service' +import { MemberFollowUpsModule } from '../member-follow-ups/member-follow-ups.module'; + +@Module({ + imports: [ + MemberFollowUpsModule + ], + controllers: [], + providers: [ + MemberFeedbacksService + ], + exports: [ + MemberFeedbacksService + ] +}) +export class MemberFeedbacksModule {} \ No newline at end of file diff --git a/apps/web-api/src/member-feedbacks/member-feedbacks.service.ts b/apps/web-api/src/member-feedbacks/member-feedbacks.service.ts new file mode 100644 index 000000000..5120926e3 --- /dev/null +++ b/apps/web-api/src/member-feedbacks/member-feedbacks.service.ts @@ -0,0 +1,57 @@ +import { + Injectable, + BadRequestException, + ConflictException, + NotFoundException +} from '@nestjs/common'; +import { LogService } from '../shared/log.service'; +import { PrismaService } from '../shared/prisma.service'; +import { Prisma } from '@prisma/client'; +import { MemberFollowUpsService } from '../member-follow-ups/member-follow-ups.service'; +import { + MemberFollowUpStatus +} from 'libs/contracts/src/schema'; + +@Injectable() +export class MemberFeedbacksService { + constructor( + private prisma: PrismaService, + private logger: LogService, + private followUpService: MemberFollowUpsService + ) {} + + async createFeedback(feedback: Prisma.MemberFeedbackUncheckedCreateInput, loggedInMember, followUp) { + try { + const result = await this.prisma.memberFeedback.create({ + data: { + ...feedback, + createdBy: loggedInMember.uid, + followUpUid: followUp.uid + } + }); + await this.followUpService.updateFollowUpStatusByUid(followUp.uid, MemberFollowUpStatus.Enum.COMPLETED) + return result; + } catch(error) { + this.handleErrors(error); + } + }; + + private handleErrors(error, message?) { + this.logger.error(error); + if (error instanceof Prisma.PrismaClientKnownRequestError) { + switch (error?.code) { + case 'P2002': + throw new ConflictException('Unique key constraint error on interaction feed back:', error.message); + case 'P2003': + throw new BadRequestException('Foreign key constraint error on interaction feed back', error.message); + case 'P2025': + throw new NotFoundException('Interaction Feed back is not found with uid:' + message); + default: + throw error; + } + } else if (error instanceof Prisma.PrismaClientValidationError) { + throw new BadRequestException('Database field validation error on interaction feed back', error.message); + } + throw error; + }; +} \ No newline at end of file diff --git a/apps/web-api/src/member-follow-ups/member-follow-ups.module.ts b/apps/web-api/src/member-follow-ups/member-follow-ups.module.ts new file mode 100644 index 000000000..15cca6daf --- /dev/null +++ b/apps/web-api/src/member-follow-ups/member-follow-ups.module.ts @@ -0,0 +1,14 @@ +import { Module } from '@nestjs/common' +import { MemberFollowUpsService } from './member-follow-ups.service'; + +@Module({ + imports: [], + controllers: [], + providers: [ + MemberFollowUpsService + ], + exports: [ + MemberFollowUpsService + ] +}) +export class MemberFollowUpsModule {} \ No newline at end of file diff --git a/apps/web-api/src/member-follow-ups/member-follow-ups.service.ts b/apps/web-api/src/member-follow-ups/member-follow-ups.service.ts new file mode 100644 index 000000000..8f2a0399e --- /dev/null +++ b/apps/web-api/src/member-follow-ups/member-follow-ups.service.ts @@ -0,0 +1,96 @@ +import { + Injectable, + BadRequestException, + ConflictException, + NotFoundException +} from '@nestjs/common'; +import { LogService } from '../shared/log.service'; +import { PrismaService } from '../shared/prisma.service'; +import { Prisma } from '@prisma/client'; + +@Injectable() +export class MemberFollowUpsService { + constructor( + private prisma: PrismaService, + private logger: LogService + ) {} + + async createFollowUp(followUp: Prisma.MemberFollowUpUncheckedCreateInput, interaction) { + try { + await this.prisma.memberFollowUp.create({ + data: { + ...followUp + } + }); + } catch(error) { + this.handleErrors(error); + } + } + + async getFollowUps(query: Prisma.MemberFollowUpFindManyArgs) { + try { + const followUps = await this.prisma.memberFollowUp.findMany({ + ...query, + include: { + interaction: true + } + }); + return followUps; + } catch(error) { + this.handleErrors(error); + } + } + + async updateFollowUpStatusByUid(uid: string, status) { + try { + return await this.prisma.memberFollowUp.update({ + where: { + uid + }, + data: { + status + } + }); + } catch(error) { + this.handleErrors(error); + } + } + + buildDelayedFollowUpQuery() { + const daysAgo = parseInt(process.env.INTERACTION_FOLLOWUP_DELAY_IN_DAYS || "7") + const dateOfNthWeekAgo = new Date(); + dateOfNthWeekAgo.setDate(dateOfNthWeekAgo.getDate() - daysAgo); + return { + OR: [ + { + isDelayed: false + }, + { + isDelayed: true, + createdAt: { + lte: dateOfNthWeekAgo, + } + } + ] + } + }; + + private handleErrors(error, message?) { + this.logger.error(error); + if (error instanceof Prisma.PrismaClientKnownRequestError) { + switch (error?.code) { + case 'P2002': + throw new ConflictException('Unique key constraint error on follow ups:', error.message); + case 'P2003': + throw new BadRequestException('Foreign key constraint error on follow ups', error.message); + case 'P2025': + throw new NotFoundException('Follow up is not found with uid:' + message); + default: + throw error; + } + } else if (error instanceof Prisma.PrismaClientValidationError) { + throw new BadRequestException('Database field validation error on follow ups', error.message); + } + throw error; + }; +} \ No newline at end of file diff --git a/apps/web-api/src/office-hours/office-hours.controller.ts b/apps/web-api/src/office-hours/office-hours.controller.ts new file mode 100644 index 000000000..295fdc4d4 --- /dev/null +++ b/apps/web-api/src/office-hours/office-hours.controller.ts @@ -0,0 +1,96 @@ +import { Body, Controller, NotFoundException, Req, UseGuards, UsePipes, Param } from '@nestjs/common'; +import { Api, initNestServer } from '@ts-rest/nest'; +import { Request } from 'express'; +import { NoCache } from '../decorators/no-cache.decorator'; +import { apiMemberInteractions } from '../../../../libs/contracts/src/lib/contract-member-interaction'; +import { MembersService } from '../members/members.service'; +import { ZodValidationPipe } from 'nestjs-zod'; +import { PrismaQueryBuilder } from '../utils/prisma-query-builder'; +import { prismaQueryableFieldsFromZod } from '../utils/prisma-queryable-fields-from-zod'; +import { UserTokenValidation } from '../guards/user-token-validation.guard'; +import { + CreateMemberInteractionSchemaDto, + CreateMemberFeedbackSchemaDto, + MemberFollowUpQueryParams, + ResponseMemberFollowUpWithRelationsSchema, + MemberFollowUpStatus +} from 'libs/contracts/src/schema'; +import { ApiQueryFromZod } from '../decorators/api-query-from-zod'; +import { ApiOkResponseFromZod } from '../decorators/api-response-from-zod'; +import { OfficeHoursService } from './office-hours.service'; +import { MemberFollowUpsService } from '../member-follow-ups/member-follow-ups.service'; +import { MemberFeedbacksService } from '../member-feedbacks/member-feedbacks.service'; + + +const server = initNestServer(apiMemberInteractions); + +@Controller() +@NoCache() +export class OfficeHoursController { + constructor( + private readonly memberService: MembersService, + private readonly interactionService: OfficeHoursService, + private readonly followUpService: MemberFollowUpsService + ) {} + + @Api(server.route.createMemberInteraction) + @UsePipes(ZodValidationPipe) + @UseGuards(UserTokenValidation) + async createMemberInteraction( + @Body() body: CreateMemberInteractionSchemaDto, + @Req() request: Request + ): Promise { + const member: any = await this.memberService.findMemberByEmail(request["userEmail"]); + return await this.interactionService.createInteraction(body as any, member); + } + + @Api(server.route.getMemberInteractionFollowUps) + @ApiQueryFromZod(MemberFollowUpQueryParams) + @ApiOkResponseFromZod(ResponseMemberFollowUpWithRelationsSchema.array()) + @UseGuards(UserTokenValidation) + @NoCache() + async findAll( + @Param('uid') interactionUid: string, + @Req() request: Request + ) { + const queryableFields = prismaQueryableFieldsFromZod( + ResponseMemberFollowUpWithRelationsSchema + ); + const builder = new PrismaQueryBuilder(queryableFields); + const builtQuery = builder.build(request.query); + const member: any = await this.memberService.findMemberByEmail(request["userEmail"]); + builtQuery.where = { + AND: [ + builtQuery.where, + { + createdBy: member?.uid, + status: MemberFollowUpStatus.Enum.PENDING + }, + this.followUpService.buildDelayedFollowUpQuery() + ] + } + return this.followUpService.getFollowUps(builtQuery); + } + + @Api(server.route.createMemberInteractionFeedback) + @UsePipes(ZodValidationPipe) + @UseGuards(UserTokenValidation) + async createMemberInteractionFeedback( + @Param('uid') interactionFollowUpUid: string, + @Body() body: CreateMemberFeedbackSchemaDto, + @Req() request: Request + ): Promise { + const member: any = await this.memberService.findMemberByEmail(request["userEmail"]); + const followUps = await this.followUpService.getFollowUps({ + where: { + uid : interactionFollowUpUid, + createdBy: member?.uid, + status: MemberFollowUpStatus.Enum.PENDING + } + }); + if (followUps && followUps.length === 0) { + throw new NotFoundException(`There is no follow-up associated with the given ID: ${interactionFollowUpUid}`); + } + return await this.interactionService.createInteractionFeedback(body as any, member, followUps?.[0]); + } +} \ No newline at end of file diff --git a/apps/web-api/src/office-hours/office-hours.module.ts b/apps/web-api/src/office-hours/office-hours.module.ts new file mode 100644 index 000000000..44f4ff2ec --- /dev/null +++ b/apps/web-api/src/office-hours/office-hours.module.ts @@ -0,0 +1,22 @@ +import { Module } from '@nestjs/common'; +import { MembersModule } from '../members/members.module'; +import { OfficeHoursService } from './office-hours.service'; +import { OfficeHoursController } from './office-hours.controller'; +import { MemberFollowUpsModule } from '../member-follow-ups/member-follow-ups.module'; +import { MemberFeedbacksModule } from '../member-feedbacks/member-feedbacks.module'; + +@Module({ + imports: [ + MembersModule, + MemberFollowUpsModule, + MemberFeedbacksModule + ], + controllers: [OfficeHoursController], + providers: [ + OfficeHoursService + ], + exports: [ + OfficeHoursService + ] +}) +export class OfficeHoursModule {} \ No newline at end of file diff --git a/apps/web-api/src/office-hours/office-hours.service.ts b/apps/web-api/src/office-hours/office-hours.service.ts new file mode 100644 index 000000000..ef486ced9 --- /dev/null +++ b/apps/web-api/src/office-hours/office-hours.service.ts @@ -0,0 +1,116 @@ +import { + Injectable, + BadRequestException, + ConflictException, + NotFoundException +} from '@nestjs/common'; +import { LogService } from '../shared/log.service'; +import { PrismaService } from '../shared/prisma.service'; +import { MemberFollowUpsService } from '../member-follow-ups/member-follow-ups.service'; +import { MemberFeedbacksService } from '../member-feedbacks/member-feedbacks.service'; +import { Prisma } from '@prisma/client'; +import { + MemberFollowUpStatus, + MemberFollowUpType, + MemberFeedbackResponseType +} from 'libs/contracts/src/schema'; +import { InteractionFailureReasons } from '../utils/constants'; + +@Injectable() +export class OfficeHoursService { + private delayedFollowUps = [ + MemberFollowUpType.Enum.MEETING_SCHEDULED, + MemberFollowUpType.Enum.MEETING_YET_TO_HAPPEN + ]; + + constructor( + private readonly prisma: PrismaService, + private readonly logger: LogService, + private readonly followUpService: MemberFollowUpsService, + private readonly feedbackService: MemberFeedbacksService + ) {} + + async createInteraction(interaction: Prisma.MemberInteractionUncheckedCreateInput, loggedInMember) { + try { + const result = await this.prisma.memberInteraction.create({ + data:{ + ...interaction, + sourceMemberUid: loggedInMember?.uid + } + }); + if (result?.hasFollowUp) { + await this.createInteractionFollowUp(result, loggedInMember, MemberFollowUpType.Enum.MEETING_INITIATED); + await this.createInteractionFollowUp(result, loggedInMember, MemberFollowUpType.Enum.MEETING_SCHEDULED); + }; + return result; + } catch(exception) { + this.handleErrors(exception); + } + } + + async createInteractionFollowUp(interaction, loggedInMember, type, scheduledAt?) { + const followUp: any = { + status: MemberFollowUpStatus.Enum.PENDING, + interactionUid: interaction?.uid, + createdBy: loggedInMember?.uid, + type, + data: { + ...interaction.data + }, + isDelayed: this.delayedFollowUps.includes(type) + }; + if (scheduledAt != null) { + followUp.createdAt = scheduledAt; + } + return await this.followUpService.createFollowUp(followUp, interaction); + } + + async createInteractionFeedback(feedback, member, followUp) { + feedback.comments = feedback.comments?.map(comment => InteractionFailureReasons[comment]) || []; + if ( + followUp.type === MemberFollowUpType.Enum.MEETING_INITIATED && + feedback.response === MemberFeedbackResponseType.Enum.NEGATIVE + ) {} + if ( + feedback.response === MemberFeedbackResponseType.Enum.NEGATIVE && + feedback.comments?.includes('IFR0004') + ) { + await this.createInteractionFollowUp( + followUp.interaction, + member, + MemberFollowUpType.Enum.MEETING_YET_TO_HAPPEN + ); + } + if ( + feedback.response === MemberFeedbackResponseType.Enum.NEGATIVE && + feedback.comments?.includes('IFR0005') + ) { + await this.createInteractionFollowUp( + followUp.interaction, + member, + MemberFollowUpType.Enum.MEETING_RESCHEDULED, + feedback?.data?.scheduledAt + ); + } + return await this.feedbackService.createFeedback(feedback, member, followUp); + } + + private handleErrors(error, message?) { + this.logger.error(error); + if (error instanceof Prisma.PrismaClientKnownRequestError) { + switch (error?.code) { + case 'P2002': + throw new ConflictException('Unique key constraint error on interactions:', error.message); + case 'P2003': + throw new BadRequestException('Foreign key constraint error on interactions', error.message); + case 'P2025': + throw new NotFoundException('Interactions is not found with uid:' + message); + default: + throw error; + } + } else if (error instanceof Prisma.PrismaClientValidationError) { + throw new BadRequestException('Database field validation error on Interactions', error.message); + } + throw error; + }; +} \ No newline at end of file diff --git a/apps/web-api/src/utils/constants.ts b/apps/web-api/src/utils/constants.ts index f88aef435..46c446aa4 100644 --- a/apps/web-api/src/utils/constants.ts +++ b/apps/web-api/src/utils/constants.ts @@ -99,3 +99,15 @@ export const DEFAULT_MEMBER_ROLES = { export const PROJECT = 'Project'; export const TEAM = 'Team'; + +export const InteractionFailureReasons: { [key: string]: string } = { + "Broken Link": "IFR0001", + "I plan to schedule soon": "IFR0002", + "Preferred slot not available": "IFR0003", + "Meeting yet to happen": "IFR0004", + "Got Rescheduled": "IFR0005", + "Got Cancelled" : "IFR0006", + "Member didn’t show up": "IFR0007", + "I did not show up":"IFR0008", + "Other": "IFR0009" +}; diff --git a/libs/contracts/src/lib/contract-member-interaction.ts b/libs/contracts/src/lib/contract-member-interaction.ts new file mode 100644 index 000000000..b65950294 --- /dev/null +++ b/libs/contracts/src/lib/contract-member-interaction.ts @@ -0,0 +1,38 @@ +import { initContract } from '@ts-rest/core'; +import { + ResponseMemberFollowUpWithRelationsSchema, + MemberFollowUpQueryParams +} from '../schema'; +import { getAPIVersionAsPath } from '../utils/versioned-path'; + +const contract = initContract(); + +export const apiMemberInteractions = contract.router({ + createMemberInteraction: { + method: 'POST', + path: `${getAPIVersionAsPath('1')}/members/:uid/interactions`, + body: contract.body(), + responses: { + 200: contract.response(), + }, + summary: 'create a new member interactions' + }, + getMemberInteractionFollowUps: { + method: 'GET', + path: `${getAPIVersionAsPath('1')}/members/:memberUid/interactions/:uid/follow-ups`, + query: MemberFollowUpQueryParams, + responses: { + 200: ResponseMemberFollowUpWithRelationsSchema.array(), + }, + summary: 'Get member interaction follow ups', + }, + createMemberInteractionFeedback: { + method: 'POST', + path: `${getAPIVersionAsPath('1')}/members/:uid/follow-ups/:uid/feedbacks`, + body: contract.body(), + responses: { + 200: contract.response(), + }, + summary: 'create a member interaction feedback', + } +}); diff --git a/libs/contracts/src/schema/index.ts b/libs/contracts/src/schema/index.ts index 90de3df89..8ec41610d 100644 --- a/libs/contracts/src/schema/index.ts +++ b/libs/contracts/src/schema/index.ts @@ -18,4 +18,7 @@ export * from './pl-event'; export * from './pl-event-guest'; export * from './focus-areas'; export * from './team-focus-areas'; -export * from './project-focus-areas'; \ No newline at end of file +export * from './project-focus-areas'; +export * from './member-interaction'; +export * from './member-follow-up'; +export * from './member-feedback'; \ No newline at end of file diff --git a/libs/contracts/src/schema/member-feedback.ts b/libs/contracts/src/schema/member-feedback.ts new file mode 100644 index 000000000..cc217873f --- /dev/null +++ b/libs/contracts/src/schema/member-feedback.ts @@ -0,0 +1,47 @@ +import { z, } from "zod"; +import { createZodDto } from '@abitia/zod-dto'; +import { ResponseMemberSchema } from './member'; +import { ResponseMemberFollowUpSchema } from './member-follow-up'; + +export const MemberFeedbackResponseType = z.enum([ + "POSITIVE", + "NEGATIVE", + "NEUTRAL" +]); + +const MemberFeedbackSchema = z.object({ + id: z.number().int(), + uid: z.string(), + type: z.string(), + data: z.any().optional(), + rating: z.number().int().optional(), + comments: z.array(z.string()).optional(), + response: MemberFeedbackResponseType, + createdAt: z.string(), + updatedAt: z.string(), + followUpUid: z.string(), + createdBy: z.string() +}); + +export const CreateMemberFeedbackSchema = MemberFeedbackSchema.pick({ + type: true, + data: true, + followUpUid: true, + rating: true, + comments: true, + response: true +}); + +export const ResponseMemberFeedbackSchema = MemberFeedbackSchema.omit({ id: true }).strict(); + +export const ResponseMemberFeedbackWithRelationsSchema = ResponseMemberFeedbackSchema.extend({ + creator: ResponseMemberSchema, + followUp: ResponseMemberFollowUpSchema +}); + +export const MemberFeedbackRelationalFields = ResponseMemberFeedbackWithRelationsSchema.pick({ + creator: true, + followUp: true +}).strip(); + +export class CreateMemberFeedbackSchemaDto extends createZodDto(CreateMemberFeedbackSchema) {} diff --git a/libs/contracts/src/schema/member-follow-up.ts b/libs/contracts/src/schema/member-follow-up.ts new file mode 100644 index 000000000..27fb213a5 --- /dev/null +++ b/libs/contracts/src/schema/member-follow-up.ts @@ -0,0 +1,47 @@ +import { z, } from "zod"; +import { ResponseMemberSchema } from './member'; +import { ResponseMemberInteractionSchema } from './member-interaction'; +import { QueryParams } from './query-params'; + +export const MemberFollowUpType = z.enum([ + "MEETING_INITIATED", + "MEETING_SCHEDULED", + "MEETING_YET_TO_HAPPEN", + "MEETING_RESCHEDULED" +]); + +export const MemberFollowUpStatus = z.enum(["PENDING", "COMPLETED"]); + +const MemberFollowUpSchema = z.object({ + id: z.number().int(), + uid: z.string(), + type: MemberFollowUpType, + status: MemberFollowUpStatus, + data: z.any().optional(), + isDelayed: z.boolean(), + createdBy: z.string(), + createdAt: z.string(), + updatedAt: z.string(), + interactionUid: z.string() +}); + +export const ResponseMemberFollowUpSchema = MemberFollowUpSchema.omit({ id: true }).strict(); + +export const ResponseMemberFollowUpWithRelationsSchema = ResponseMemberFollowUpSchema.extend({ + creator: ResponseMemberSchema, + interaction: ResponseMemberInteractionSchema +}); + +export const MemberFollowUpRelationalFields = ResponseMemberFollowUpWithRelationsSchema.pick({ + creator: true, + interaction: true +}).strip(); + +export const MemberFollowUpQueryableFields = ResponseMemberFollowUpSchema.keyof(); + +export const MemberFollowUpQueryParams = QueryParams({ + queryableFields: MemberFollowUpQueryableFields, + relationalFields: MemberFollowUpRelationalFields +}); + + diff --git a/libs/contracts/src/schema/member-interaction.ts b/libs/contracts/src/schema/member-interaction.ts new file mode 100644 index 000000000..1404363c4 --- /dev/null +++ b/libs/contracts/src/schema/member-interaction.ts @@ -0,0 +1,23 @@ +import { z, } from "zod"; +import { createZodDto } from '@abitia/zod-dto'; + +const MemberInteractionSchema = z.object({ + id: z.number().int(), + uid: z.string(), + data: z.any().optional(), + hasFollowUp: z.boolean().optional(), + createdAt: z.string(), + updatedAt: z.string(), + sourceMemberUid: z.string(), + targetMemberUid: z.string().optional(), +}); + +export const CreateMemberInteractionSchema = MemberInteractionSchema.pick({ + type: true, + data: true, + targetMemberUid: true +}); + +export const ResponseMemberInteractionSchema = MemberInteractionSchema.omit({ id: true }).strict(); + +export class CreateMemberInteractionSchemaDto extends createZodDto(CreateMemberInteractionSchema) {}