Files
speckle-server/packages/server/modules/shared/utils/subscriptions.ts
T
2023-12-04 10:55:40 +02:00

268 lines
8.1 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 {
AutomationRun,
AutomationsStatus,
ProjectAutomationsStatusUpdatedMessage,
ProjectCommentsUpdatedMessage,
ProjectFileImportUpdatedMessage,
ProjectModelsUpdatedMessage,
ProjectPendingModelsUpdatedMessage,
ProjectPendingVersionsUpdatedMessage,
ProjectUpdatedMessage,
ProjectVersionsPreviewGeneratedMessage,
ProjectVersionsUpdatedMessage,
SubscriptionProjectAutomationsStatusUpdatedArgs,
SubscriptionProjectCommentsUpdatedArgs,
SubscriptionProjectFileImportUpdatedArgs,
SubscriptionProjectModelsUpdatedArgs,
SubscriptionProjectPendingModelsUpdatedArgs,
SubscriptionProjectPendingVersionsUpdatedArgs,
SubscriptionProjectUpdatedArgs,
SubscriptionProjectVersionsPreviewGeneratedArgs,
SubscriptionProjectVersionsUpdatedArgs,
SubscriptionSubscribeFn,
SubscriptionViewerUserActivityBroadcastedArgs,
UserProjectsUpdatedMessage,
ViewerResourceItem,
ViewerUserActivityMessage
} 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 { AutomationFunctionRunGraphQLReturn } from '@/modules/automations/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',
ProjectAutomationStatusUpdated = 'PROJECT_AUTOMATION_STATUS_UPDATED'
}
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.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.ProjectAutomationStatusUpdated]: {
payload: {
projectAutomationsStatusUpdated: Merge<
ProjectAutomationsStatusUpdatedMessage,
{
version: VersionGraphQLReturn
model: ModelGraphQLReturn
project: ProjectGraphQLReturn
status: Merge<
AutomationsStatus,
{
automationRuns: Array<
Merge<
AutomationRun,
{ functionRuns: AutomationFunctionRunGraphQLReturn[] }
>
>
}
>
}
>
projectId: string
}
variables: SubscriptionProjectAutomationsStatusUpdatedArgs
}
} & { [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>
}