08e941f8af
* Implemented workspace general page * Added notifications to user input * Allowed non-admins to view but not edit * Added redirect to homeroute * Fixed validation * Squashed commit of the following: commit 7bf14ab8af0f76b4c9d0aa87fc08085af7c34959 Author: Chuck Driesler <chuck@speckle.systems> Date: Tue Aug 6 19:40:50 2024 +0200 mob next [ci-skip] [ci skip] [skip ci] lastFile:packages/server/modules/workspacesCore/migrations/20240806160740_workspace_domains.ts commit 8aa3fb0cb052c10eeeb83bf9874ae0d1c065e480 Author: Alessandro Magionami <alessandro.magionami@gmail.com> Date: Tue Aug 6 18:54:15 2024 +0200 mob next [ci-skip] [ci skip] [skip ci] lastFile:packages/server/modules/core/domain/userEmails/operations.ts commit 66dfd0cf6c15a789c8f96a65a3168323e83a7b9e Author: Chuck Driesler <chuck@speckle.systems> Date: Tue Aug 6 18:30:22 2024 +0200 mob next [ci-skip] [ci skip] [skip ci] lastFile:packages/server/modules/workspacesCore/domain/types.ts Co-authored-by: Alessandro Magionami <alessandro.magionami@gmail.com> * Move General to workspaces folder * feat(workspaces): inputs on security section * feat(workspaces): add domain to workspace mutation * chore(workspaces): add blocked domains list * fix(workspaces): modals with buttons * feat(workspaceDomains): delete domain * fix(workspaces): use mutation * fix(workspaces): present user verified domains as options * Moved sidebar menu to a composable * Added coming soon tag back * feat(workspaces): create domains resolver for workspace * chore(workspaces): fix tests * chore(workspaces): fix types * chore(workspaces): fix linter * fix(workspaces): do some delete I think * chore(workspaces): add domainBasedMembershipProtectionEnabled field to workspace * chore(workspaces): improve validation for email domain * fix(workspace): query and do the thing * chore(workspaces): add graphql schema for domainBasedMembershipProtection * chore(workspaces): lint and test failures * fix(workspaces): test issues w new field * feat(workspaces): add discoverability flag * chore(workspaces): they made me do it * feat(workspaces): enable toggling domain protection * feat(workspaces): add discoverability toggle to workspace settings * feat(workspace): auto enable discoverability on first domain registration * feat(workspace): discoverability toggle fixes * fix(eventBus): fix tests * feat(workspaces): user discoverable workspaces (#2620) * feat(workspaces): it works just trust me * fix(workspaces): don't worry about it * fix(workspaces); happy path success * fix(workspaces): almost there * fix(workspaces): successful tests! * fix(workspaces): we have DISCOVERED (#2621) * Fixed linting issue * Updated query * Updated validation rules * Updated validation rules * Fix unsaved file with type export * Addressed PR comments * Updated cache * Updated item classes, add fragment back * Gergo/web 1574 join workspaces via discovery (#2623) * chore(useremails): add find verified emails by user function * chore(workspace): table helper for workspace domains * chore(workspace): get workspace with domains function * chore(workspace): test get workspace with domains function * feat(workspace): restrict workspace membership when updating workspace role * chore(workspaces): fix types * feat(workspaces): WIP join * feat(workspaces): join button makes u join * chore(useremails): fix type for find verified emails function * feat(workspaces): join * feat(workspace): prevent inviting user without email matching domain * chore(workspaces): fix linter * fix(workspaces): invoke join (gergo wrote this) * fuck * fix(workspaces): properly get discoverable workspaces * fix(workspaces): test --------- Co-authored-by: Gergő Jedlicska <gergo@jedlicska.com> Co-authored-by: Chuck Driesler <chuck@speckle.systems> * fix(workspaces): some query stuff * fix(workspaces): mutate cache instead of refetch * fix(workspaces): more adjustments to gql query and fragment structure * fix(workspaces): queries, style, structure * fix(workspaces): match discoverability with current styles * chore(workspaces): lint lint lint * fix(workspaces): got it twisted * chore(workspaces): fix test * fix(workspaces): route to joined workspace on join --------- Co-authored-by: Mike Tasset <mike.tasset@gmail.com> Co-authored-by: Chuck Driesler <chuck@speckle.systems> Co-authored-by: Alessandro Magionami <alessandro.magionami@gmail.com>
134 lines
4.1 KiB
TypeScript
134 lines
4.1 KiB
TypeScript
import {
|
|
WorkspaceEventsPayloads,
|
|
workspaceEventNamespace
|
|
} from '@/modules/workspacesCore/domain/events'
|
|
import { MaybeAsync } from '@speckle/shared'
|
|
import { UnionToIntersection } from 'type-fest'
|
|
|
|
import EventEmitter from 'eventemitter2'
|
|
import {
|
|
serverinvitesEventNamespace,
|
|
ServerInvitesEventsPayloads
|
|
} from '@/modules/serverinvites/domain/events'
|
|
|
|
type EventWildcard = '*'
|
|
|
|
type TestEvents = {
|
|
['test.string']: string
|
|
['test.number']: number
|
|
}
|
|
|
|
// we should only ever extend this type, other helper types will be derived from this
|
|
type EventsByNamespace = {
|
|
test: TestEvents
|
|
[workspaceEventNamespace]: WorkspaceEventsPayloads
|
|
[serverinvitesEventNamespace]: ServerInvitesEventsPayloads
|
|
}
|
|
|
|
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 =
|
|
| EventWildcard
|
|
| `${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]
|
|
}
|
|
}
|
|
}
|
|
|
|
type EventPayload<T extends EventSubscriptionKey> = T extends EventWildcard
|
|
? // 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 })
|
|
|
|
return {
|
|
/**
|
|
* 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<unknown[]> => {
|
|
// curate the proper payload here and eventName object here, before emitting
|
|
return 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()
|
|
}
|
|
}
|
|
}
|
|
|
|
export type EventBus = ReturnType<typeof initializeEventBus>
|
|
export type EventBusPayloads = EventTypes
|
|
export type EventBusEmit = EventBus['emit']
|
|
|
|
let eventBus: EventBus
|
|
|
|
export function getEventBus(): EventBus {
|
|
if (!eventBus) eventBus = initializeEventBus()
|
|
return eventBus
|
|
}
|