Files
speckle-server/packages/server/modules/shared/utils/subscriptions.ts
T
2024-06-20 15:35:39 +03:00

277 lines
8.8 KiB
TypeScript

import { MaybeAsync, Nullable } from '@speckle/shared'
import { getRedisUrl } from '@/modules/shared/helpers/envHelper'
import { RedisPubSub } from 'graphql-redis-subscriptions'
import Redis from 'ioredis'
import { withFilter } from 'graphql-subscriptions'
import { GraphQLContext } from '@/modules/shared/helpers/typeHelper'
import {
ProjectCommentsUpdatedMessage,
ProjectFileImportUpdatedMessage,
ProjectModelsUpdatedMessage,
ProjectPendingModelsUpdatedMessage,
ProjectPendingVersionsUpdatedMessage,
ProjectUpdatedMessage,
ProjectVersionsPreviewGeneratedMessage,
ProjectVersionsUpdatedMessage,
SubscriptionProjectAutomationsUpdatedArgs,
SubscriptionProjectCommentsUpdatedArgs,
SubscriptionProjectFileImportUpdatedArgs,
SubscriptionProjectModelsUpdatedArgs,
SubscriptionProjectPendingModelsUpdatedArgs,
SubscriptionProjectPendingVersionsUpdatedArgs,
SubscriptionProjectTriggeredAutomationsStatusUpdatedArgs,
SubscriptionProjectUpdatedArgs,
SubscriptionProjectVersionsPreviewGeneratedArgs,
SubscriptionProjectVersionsUpdatedArgs,
SubscriptionSubscribeFn,
SubscriptionViewerUserActivityBroadcastedArgs,
UserProjectsUpdatedMessage,
ViewerResourceItem,
ViewerUserActivityMessage,
GendoAiRender,
SubscriptionProjectVersionGendoAiRenderUpdatedArgs,
SubscriptionProjectVersionGendoAiRenderCreatedArgs
} from '@/modules/core/graph/generated/graphql'
import { Merge } from 'type-fest'
import {
ModelGraphQLReturn,
ProjectGraphQLReturn,
VersionGraphQLReturn
} from '@/modules/core/helpers/graphTypes'
import { CommentGraphQLReturn } from '@/modules/comments/helpers/graphTypes'
import { FileUploadGraphQLReturn } from '@/modules/fileuploads/helpers/types'
import {
ProjectTriggeredAutomationsStatusUpdatedMessageGraphQLReturn,
ProjectAutomationsUpdatedMessageGraphQLReturn
} from '@/modules/automate/helpers/graphTypes'
/**
* GraphQL Subscription PubSub instance
*/
export const pubsub = new RedisPubSub({
publisher: new Redis(getRedisUrl()),
subscriber: new Redis(getRedisUrl())
})
/**
* Subscription event keys
*/
// OLD:
export enum StreamSubscriptions {
UserStreamAdded = 'USER_STREAM_ADDED',
UserStreamRemoved = 'USER_STREAM_REMOVED',
StreamUpdated = 'STREAM_UPDATED',
StreamDeleted = 'STREAM_DELETED'
}
export enum CommitSubscriptions {
CommitCreated = 'COMMIT_CREATED',
CommitUpdated = 'COMMIT_UPDATED',
CommitDeleted = 'COMMIT_DELETED'
}
export enum BranchSubscriptions {
BranchCreated = 'BRANCH_CREATED',
BranchUpdated = 'BRANCH_UPDATED',
BranchDeleted = 'BRANCH_DELETED'
}
export enum CommentSubscriptions {
ViewerActivity = 'VIEWER_ACTIVITY',
CommentActivity = 'COMMENT_ACTIVITY',
CommentThreadActivity = 'COMMENT_THREAD_ACTIVITY'
}
// NEW:
export enum UserSubscriptions {
UserProjectsUpdated = 'USER_PROJECTS_UPDATED'
}
export enum ProjectSubscriptions {
ProjectUpdated = 'PROJECT_UPDATED',
ProjectModelsUpdated = 'PROJECT_MODELS_UPDATED',
ProjectVersionsUpdated = 'PROJECT_VERSIONS_UPDATED',
ProjectVersionsPreviewGenerated = 'PROJECT_VERSIONS_PREVIEW_GENERATED',
ProjectCommentsUpdated = 'PROJECT_COMMENTS_UPDATED',
// old beta subscription:
ProjectTriggeredAutomationsStatusUpdated = 'PROJECT_TRIGGERED_AUTOMATION_STATUS_UPDATED',
ProjectAutomationsUpdated = 'PROJECT_AUTOMATIONS_UPDATED',
ProjectVersionGendoAIRenderUpdated = 'PROJECT_VERSION_GENDO_AI_RENDER_UPDATED',
ProjectVersionGendoAIRenderCreated = 'PROJECT_VERSION_GENDO_AI_RENDER_CREATED'
}
export enum ViewerSubscriptions {
UserActivityBroadcasted = 'VIEWER_USER_ACTIVITY_BROADCASTED'
}
export enum FileImportSubscriptions {
ProjectPendingModelsUpdated = 'PROJECT_PENDING_MODELS_UPDATED',
ProjectPendingVersionsUpdated = 'PROJECT_PENDING_VERSIONS_UPDATED',
ProjectFileImportUpdated = 'PROJECT_FILE_IMPORT_UPDATED'
}
type NoVariables = Record<string, never>
// Add mappings between expected event constant, its payload and variables
type SubscriptionTypeMap = {
[UserSubscriptions.UserProjectsUpdated]: {
payload: {
userProjectsUpdated: Merge<
UserProjectsUpdatedMessage,
{ project: Nullable<ProjectGraphQLReturn> }
>
ownerId: string
}
variables: NoVariables
}
[ProjectSubscriptions.ProjectUpdated]: {
payload: {
projectUpdated: Merge<
ProjectUpdatedMessage,
{ project: Nullable<ProjectGraphQLReturn> }
>
}
variables: SubscriptionProjectUpdatedArgs
}
[ProjectSubscriptions.ProjectVersionGendoAIRenderUpdated]: {
payload: {
projectVersionGendoAIRenderUpdated: GendoAiRender
}
variables: SubscriptionProjectVersionGendoAiRenderUpdatedArgs
}
[ProjectSubscriptions.ProjectVersionGendoAIRenderCreated]: {
payload: {
projectVersionGendoAIRenderCreated: GendoAiRender
}
variables: SubscriptionProjectVersionGendoAiRenderCreatedArgs
}
[ProjectSubscriptions.ProjectModelsUpdated]: {
payload: {
projectModelsUpdated: Merge<
ProjectModelsUpdatedMessage,
{ model: Nullable<ModelGraphQLReturn> }
>
projectId: string
}
variables: SubscriptionProjectModelsUpdatedArgs
}
[ProjectSubscriptions.ProjectVersionsUpdated]: {
payload: {
projectVersionsUpdated: Merge<
ProjectVersionsUpdatedMessage,
{ version: Nullable<VersionGraphQLReturn> }
>
projectId: string
}
variables: SubscriptionProjectVersionsUpdatedArgs
}
[ProjectSubscriptions.ProjectVersionsPreviewGenerated]: {
payload: {
projectVersionsPreviewGenerated: ProjectVersionsPreviewGeneratedMessage
}
variables: SubscriptionProjectVersionsPreviewGeneratedArgs
}
[ViewerSubscriptions.UserActivityBroadcasted]: {
payload: {
viewerUserActivityBroadcasted: ViewerUserActivityMessage
projectId: string
resourceItems: ViewerResourceItem[]
userId: Nullable<string>
}
variables: SubscriptionViewerUserActivityBroadcastedArgs
}
[ProjectSubscriptions.ProjectCommentsUpdated]: {
payload: {
projectCommentsUpdated: Merge<
ProjectCommentsUpdatedMessage,
{ comment: Nullable<CommentGraphQLReturn> }
>
projectId: string
resourceItems: ViewerResourceItem[]
}
variables: SubscriptionProjectCommentsUpdatedArgs
}
[FileImportSubscriptions.ProjectPendingModelsUpdated]: {
payload: {
projectPendingModelsUpdated: Merge<
ProjectPendingModelsUpdatedMessage,
{ model: FileUploadGraphQLReturn }
>
projectId: string
}
variables: SubscriptionProjectPendingModelsUpdatedArgs
}
[FileImportSubscriptions.ProjectPendingVersionsUpdated]: {
payload: {
projectPendingVersionsUpdated: Merge<
ProjectPendingVersionsUpdatedMessage,
{ version: FileUploadGraphQLReturn }
>
projectId: string
branchName: string
}
variables: SubscriptionProjectPendingVersionsUpdatedArgs
}
[FileImportSubscriptions.ProjectFileImportUpdated]: {
payload: {
projectFileImportUpdated: Merge<
ProjectFileImportUpdatedMessage,
{ upload: FileUploadGraphQLReturn }
>
projectId: string
}
variables: SubscriptionProjectFileImportUpdatedArgs
}
[ProjectSubscriptions.ProjectTriggeredAutomationsStatusUpdated]: {
payload: {
projectTriggeredAutomationsStatusUpdated: ProjectTriggeredAutomationsStatusUpdatedMessageGraphQLReturn
projectId: string
}
variables: SubscriptionProjectTriggeredAutomationsStatusUpdatedArgs
}
[ProjectSubscriptions.ProjectAutomationsUpdated]: {
payload: {
projectAutomationsUpdated: ProjectAutomationsUpdatedMessageGraphQLReturn
projectId: string
}
variables: SubscriptionProjectAutomationsUpdatedArgs
}
} & { [k in SubscriptionEvent]: { payload: unknown; variables: unknown } }
type SubscriptionEvent =
| UserSubscriptions
| ProjectSubscriptions
| ViewerSubscriptions
| FileImportSubscriptions
/**
* Publish a GQL subscription event
*/
export const publish = <T extends SubscriptionEvent>(
event: T,
payload: SubscriptionTypeMap[T]['payload']
) => pubsub.publish(event, payload)
/**
* Subscribe to a GQL subscription and use the filter function to filter subscribers
* depending on the payload, variables and/or GQL context
*/
export const filteredSubscribe = <T extends SubscriptionEvent>(
event: T,
filterFn: (
payload: SubscriptionTypeMap[T]['payload'],
variables: SubscriptionTypeMap[T]['variables'],
context: GraphQLContext
) => MaybeAsync<boolean>
) => {
// we need to cast to graphql codegen types cause they're not fully compatible
// with our version of graphql-subscriptions
// https://github.com/dotansimha/graphql-code-generator/issues/7197#issuecomment-1098014584
return withFilter(
() => pubsub.asyncIterator([event]),
filterFn
// eslint-disable-next-line @typescript-eslint/no-explicit-any
) as unknown as SubscriptionSubscribeFn<any, any, any, any>
}