Files
speckle-server/packages/server/modules/activitystream/index.ts
T
Kristaps Fabians Geikins bde148f286 chore(server): migrating fully to ESM (#5042)
* wip

* some extra fixes

* stuff kinda works?

* need to figure out mocks

* need to figure out mocks

* fix db listener

* gqlgen fix

* minor gqlgen watch adjustment

* lint fixes

* delete old codegen file

* converting migrations to ESM

* getModuleDIrectory

* vitest sort of works

* added back ts-vitest

* resolve gql double load

* fixing test timeout configs

* TSC lint fix

* fix automate tests

* moar debugging

* debugging

* more debugging

* codegen update

* server works

* yargs migrated

* chore(server): getting rid of global mocks for Server ESM (#5046)

* got rid of email mock

* got rid of comment mocks

* got rid of multi region mocks

* got rid of stripe mock

* admin override mock updated

* removed final mock

* fixing import.meta.resolve calls

* another import.meta.resolve fix

* added requested test

* nyc ESM fix

* removed unneeded deps + linting

* yarn lock forgot to commit

* tryna fix flakyness

* email capture util fix

* sendEmail fix

* fix TSX check

* sender transporter fix + CR comments

* merge main fix

* test fixx

* circleci fix

* gqlgen bigint fix

* error formatter fix

* more error formatting improvements

* esmloader added to Dockerfile

* more dockerfile fixes

* bg jobs fix
2025-07-14 10:26:19 +03:00

160 lines
5.6 KiB
TypeScript

import { Optional, SpeckleModule } from '@/modules/shared/helpers/typeHelper'
import { publishNotification } from '@/modules/notifications/services/publication'
import { moduleLogger } from '@/observability/logging'
import { weeklyEmailDigestEnabled } from '@/modules/shared/helpers/envHelper'
import { EventBus, getEventBus } from '@/modules/shared/services/eventBus'
import { sendActivityNotificationsFactory } from '@/modules/activitystream/services/summary'
import {
getActiveUserStreamsFactory,
saveActivityFactory,
saveStreamActivityFactory
} from '@/modules/activitystream/repositories'
import { db } from '@/db/knex'
import { getStreamFactory } from '@/modules/core/repositories/streams'
import { ScheduleExecution } from '@/modules/core/domain/scheduledTasks/operations'
import { scheduleExecutionFactory } from '@/modules/core/services/taskScheduler'
import {
acquireTaskLockFactory,
releaseTaskLockFactory
} from '@/modules/core/repositories/scheduledTasks'
import { Knex } from 'knex'
import { reportUserActivityFactory } from '@/modules/activitystream/events/userListeners'
import { reportAccessRequestActivityFactory } from '@/modules/activitystream/events/accessRequestListeners'
import { reportBranchActivityFactory } from '@/modules/activitystream/events/branchListeners'
import { reportCommitActivityFactory } from '@/modules/activitystream/events/commitListeners'
import { reportCommentActivityFactory } from '@/modules/activitystream/events/commentListeners'
import { reportStreamInviteActivityFactory } from '@/modules/activitystream/events/streamInviteListeners'
import { getProjectInviteProjectFactory } from '@/modules/serverinvites/services/projectInviteManagement'
import { reportStreamActivityFactory } from '@/modules/activitystream/events/streamListeners'
import { TIME_MS } from '@speckle/shared'
import { reportGatekeeperActivityFactory } from '@/modules/activitystream/events/gatekeeperListeners'
import { reportWorkspaceActivityFactory } from '@/modules/activitystream/events/workspaceListeners'
let scheduledTask: ReturnType<ScheduleExecution> | null = null
let quitEventListeners: Optional<() => void> = undefined
/**
* Initialize event listener for tracking various Speckle events and responding
* to them by creating activitystream entries
*/
const initializeEventListeners = ({
eventBus,
db
}: {
eventBus: EventBus
db: Knex
}) => {
const saveActivity = saveActivityFactory({ db })
const saveStreamActivity = saveStreamActivityFactory({ db })
const reportUserActivity = reportUserActivityFactory({
eventListen: eventBus.listen,
saveStreamActivity
})
const reportAccessRequestActivity = reportAccessRequestActivityFactory({
eventListen: eventBus.listen,
saveStreamActivity
})
const reportBranchActivity = reportBranchActivityFactory({
eventListen: eventBus.listen,
saveStreamActivity
})
const reportCommitActivity = reportCommitActivityFactory({
eventListen: eventBus.listen,
saveStreamActivity
})
const reportCommentActivity = reportCommentActivityFactory({
eventListen: eventBus.listen,
saveStreamActivity
})
const reportStreamInviteActivity = reportStreamInviteActivityFactory({
eventListen: eventBus.listen,
saveStreamActivity,
getProjectInviteProject: getProjectInviteProjectFactory({
getStream: getStreamFactory({ db })
})
})
const reportStreamActivity = reportStreamActivityFactory({
eventListen: eventBus.listen,
saveActivity,
saveStreamActivity
})
const reportGatekeeperActivity = reportGatekeeperActivityFactory({
eventListen: eventBus.listen,
saveActivity
})
const reportWorkspaceActivity = reportWorkspaceActivityFactory({
eventListen: eventBus.listen,
saveActivity
})
const quitCbs = [
reportUserActivity(),
reportAccessRequestActivity(),
reportBranchActivity(),
reportCommitActivity(),
reportCommentActivity(),
reportStreamInviteActivity(),
reportStreamActivity(),
reportGatekeeperActivity(),
reportWorkspaceActivity()
]
return () => quitCbs.forEach((quit) => quit())
}
const scheduleWeeklyActivityNotifications = () => {
const scheduleExecution = scheduleExecutionFactory({
acquireTaskLock: acquireTaskLockFactory({ db }),
releaseTaskLock: releaseTaskLockFactory({ db })
})
// just to test stuff
// every 1000 seconds
// const cronExpression = '*/1000 * * * * *'
// at 00 minutest, 10 (am) hours, every month, every year,
// every 1st day of the week (monday)
// cheat sheet https://crontab.guru
const cronExpression = '00 10 * * 1'
// configure the number of days, the activities are scraped for
const numberOfDays = 7
return scheduleExecution(
cronExpression,
'weeklyActivityNotification',
//task should be locked for 10 minutes
async (now: Date, { logger }) => {
logger.info('Sending weekly activity digests notifications.')
const end = now
const start = new Date(end.getTime())
start.setDate(start.getDate() - numberOfDays)
await sendActivityNotificationsFactory({
publishNotification,
getActiveUserStreams: getActiveUserStreamsFactory({ db })
})(start, end)
},
10 * TIME_MS.minute
)
}
const activityModule: SpeckleModule = {
init: async ({ isInitial }) => {
moduleLogger.info('🤺 Init activity module')
if (isInitial) {
quitEventListeners = initializeEventListeners({
db,
eventBus: getEventBus()
})
if (weeklyEmailDigestEnabled())
scheduledTask = scheduleWeeklyActivityNotifications()
}
},
shutdown: () => {
scheduledTask?.stop()
quitEventListeners?.()
}
}
export default {
...activityModule
}