Files
speckle-server/packages/server/modules/shared/services/eventBus.ts
T
Oğuzhan Koral 05e00d2c5c feat(acc): revamp (#5501)
* chore(acc): put permission gql in correct place

* feat(acc): swap to new rvt import

* fix(acc): add oda secrets

* feat(acc): auth cookies

* feat(acc): introduce integrations as workspace setting

* feat(acc): create sync item from models

* fix(acc): bump

* fix(acc): naming lost in merge

* feat(acc): no acc tab - table under settings

* chore(acc): new sync but will disapper

* feat(acc): see statuses over model list

* chore(acc): fix return type

* chore(acc): type saga

* chore(acc): status badge

* chore(acc): refactor acc gql (#5556)

* checkpoint

* fix(acc): refactor gql items

* feat(acc): double button

* chore(acc): gqlgen

* fix(acc): model ids are not project ids

* chore(acc): bump function version

* chore(acc): split up clients

* feat(acc): more-optimised gql folder fetching schema

* feat(acc): acc folder contents gql impl

* feat(acc): apollo cache optimisations

* chore(acc): gqlgen

* fix(acc): return something for

* fix(acc): handle null values correctly

* chore(acc): specify prod functions

---------

Co-authored-by: Chuck Driesler <chuck@speckle.systems>
2025-10-03 13:54:17 +01:00

261 lines
7.9 KiB
TypeScript

/* eslint-disable @typescript-eslint/no-explicit-any */
import type {
WorkspaceEventsPayloads,
workspaceEventNamespace
} from '@/modules/workspacesCore/domain/events'
import type {
gatekeeperEventNamespace,
GatekeeperEventPayloads
} from '@/modules/gatekeeperCore/domain/events'
import type { MaybeAsync } from '@speckle/shared'
import type { UnionToIntersection } from 'type-fest'
import EventEmitter from 'eventemitter2'
import type {
serverinvitesEventNamespace,
ServerInvitesEventsPayloads
} from '@/modules/serverinvites/domain/events'
import type {
modelEventsNamespace,
ModelEventsPayloads
} from '@/modules/core/domain/branches/events'
import type {
projectEventsNamespace,
ProjectEventsPayloads
} from '@/modules/core/domain/projects/events'
import type {
userEventsNamespace,
UserEventsPayloads
} from '@/modules/core/domain/users/events'
import type {
versionEventsNamespace,
VersionEventsPayloads
} from '@/modules/core/domain/commits/events'
import type {
accessRequestEventsNamespace,
AccessRequestEventsPayloads
} from '@/modules/accessrequests/domain/events'
import type {
commentEventsNamespace,
CommentEventsPayloads
} from '@/modules/comments/domain/events'
import type {
automationEventsNamespace,
AutomationEventsPayloads,
automationRunEventsNamespace,
AutomationRunEventsPayloads
} from '@/modules/automate/domain/events'
import type {
multiregionEventNamespace,
MultiregionEventsPayloads
} from '@/modules/multiregion/domain/events'
import type {
fileuploadEventNamespace,
FileuploadEventsPayloads
} from '@/modules/fileuploads/domain/events'
import type {
accSyncItemEventsNamespace,
AccSyncItemEventsPayloads
} from '@/modules/acc/domain/acc/events'
import type {
emailsEventNamespace,
EmailsEventsPayloads
} from '@/modules/emails/domain/events'
import type {
notificationsEventNamespace,
NotificationsEventsPayloads
} from '@/modules/notifications/domain/events'
import type {
savedViewsEventNamespace,
SavedViewsEventsPayloads
} from '@/modules/viewer/domain/events/savedViews'
type AllEventsWildcard = '**'
type EventWildcard = '*'
export const TestEvents = {
String: 'test.string',
Number: 'test.number'
} as const
type TestEventsPayloads = {
[TestEvents.String]: string
[TestEvents.Number]: number
}
// we should only ever extend this type, other helper types will be derived from this
type EventsByNamespace = {
test: TestEventsPayloads
[accSyncItemEventsNamespace]: AccSyncItemEventsPayloads
[workspaceEventNamespace]: WorkspaceEventsPayloads
[gatekeeperEventNamespace]: GatekeeperEventPayloads
[serverinvitesEventNamespace]: ServerInvitesEventsPayloads
[modelEventsNamespace]: ModelEventsPayloads
[projectEventsNamespace]: ProjectEventsPayloads
[userEventsNamespace]: UserEventsPayloads
[versionEventsNamespace]: VersionEventsPayloads
[accessRequestEventsNamespace]: AccessRequestEventsPayloads
[commentEventsNamespace]: CommentEventsPayloads
[automationEventsNamespace]: AutomationEventsPayloads
[automationRunEventsNamespace]: AutomationRunEventsPayloads
[multiregionEventNamespace]: MultiregionEventsPayloads
[fileuploadEventNamespace]: FileuploadEventsPayloads
[emailsEventNamespace]: EmailsEventsPayloads
[notificationsEventNamespace]: NotificationsEventsPayloads
[savedViewsEventNamespace]: SavedViewsEventsPayloads
}
type EventTypes = UnionToIntersection<EventsByNamespace[keyof EventsByNamespace]>
// generated union to collect all event
type EventNamesByNamespace = {
[Namespace in keyof EventsByNamespace]: keyof EventsByNamespace[Namespace]
}
// generated type for a top level wildcard one level nested wildcards per namespace and each possible event
type EventSubscriptionKey =
| AllEventsWildcard
| `${keyof EventNamesByNamespace}.${EventWildcard}`
| {
[Namespace in keyof EventNamesByNamespace]: EventNamesByNamespace[Namespace]
}[keyof EventNamesByNamespace]
// generated flatten of each specific event name with the emitted event type
type EventPayloadsMap = UnionToIntersection<
EventPayloadsByNamespaceMap[keyof EventPayloadsByNamespaceMap]
>
export type EventNames = keyof EventPayloadsMap
type EventPayloadsByNamespaceMap = {
// for each event namespace
[Key in keyof EventsByNamespace]: {
// for each event
[EventName in keyof EventsByNamespace[Key]]: {
// create a type with they original event as the payload, and the eventName
eventName: EventName
payload: EventsByNamespace[Key][EventName]
}
}
}
export type EventPayload<T extends EventSubscriptionKey> = T extends AllEventsWildcard
? // if event key is "**", get all events from the flat object
EventPayloadsMap[keyof EventPayloadsMap]
: // else if, the key is a "namespace.*" wildcard
T extends `${infer Namespace}.${EventWildcard}`
? // the Namespace needs to extend the keys of the type, otherwise we never
Namespace extends keyof EventPayloadsByNamespaceMap
? // get the union type of all possible events in a namespace
EventPayloadsByNamespaceMap[Namespace][keyof EventPayloadsByNamespaceMap[Namespace]]
: never
: // else if, the key is a "namespace.event" concrete key
T extends keyof EventPayloadsMap
? EventPayloadsMap[T]
: never
export function initializeEventBus() {
const emitter = new EventEmitter({ wildcard: true })
const core = {
/**
* Emit a module event. This function must be awaited to ensure all listeners
* execute. Any errors thrown in the listeners will bubble up and throw from
* the part of code that triggers this emit() call.
*/
emit: async <EventName extends EventNames>(args: {
eventName: EventName
payload: EventTypes[EventName]
}): Promise<void> => {
// curate the proper payload here and eventName object here, before emitting
await emitter.emitAsync(args.eventName, args)
},
/**
* Listen for module events. Any errors thrown here will bubble out of where
* emit() was invoked.
*
* @returns Callback for stopping listening
*/
listen: <K extends EventSubscriptionKey>(
eventName: K,
// we should add some error type object here with a type discriminator
handler: (event: EventPayload<K>) => MaybeAsync<unknown>
) => {
emitter.on(eventName, handler, {
async: true,
promisify: true
})
return () => {
emitter.removeListener(eventName, handler)
}
},
/**
* Destroy event emitter
*/
destroy() {
emitter.removeAllListeners()
}
}
// Extra utils
const listenOnce = <K extends EventSubscriptionKey>(
eventName: K,
handler: (event: EventPayload<K>) => MaybeAsync<unknown>,
options?: Partial<{
/**
* Timeout in milliseconds after which the listener will be removed even if it never fires
* (useful in tests for cleanup)
*/
timeout: number
}>
) => {
const removeListener = core.listen(eventName, async (event) => {
try {
await handler(event)
} finally {
removeListener()
}
})
if (options?.timeout) {
setTimeout(removeListener, options.timeout)
}
return removeListener
}
return {
...core,
/**
* Listen for module events only once. Any errors thrown here will bubble out of where
* emit() was invoked.
*
* @returns Callback for stopping listening
*/
listenOnce
}
}
export type EventBus = ReturnType<typeof initializeEventBus>
export type EventBusPayloads = EventTypes
export type EventBusEmit = EventBus['emit']
export type EventBusListen = EventBus['listen']
export type EmitArg = Parameters<EventBusEmit>[0]
let eventBus: EventBus
export function getEventBus(): EventBus {
if (!eventBus) eventBus = initializeEventBus()
return eventBus
}
export const isSpecificEventPayload = <EventName extends EventNames>(
payload: EventPayload<any>,
eventKey: EventName
): payload is EventPayload<EventName> => {
return payload.eventName === eventKey
}