da7f0dda0e
* feat(gatekeeper): add gatekeeper module feature flag * feat(gatekeeper): add workspace pricing table domain * feat(gatekeeper): add checkout session creation * feat(gatekeeper): verify stripe signature * wip(gatekeeper): checkout callbacks * feat(gatekeeper): add unlimited and academia plan types * refactor(envHelper): getStringFromEnv helper * chore(gatekeeper): add future todos * feat(gatekeeper): add productId to the subscription domain * feat(gatekeeper): add in memory repositories * feat(gatekeeper): add more errors * feat(gatekeeper): complete checkout session service * feat(gatekeeper): add stripe client implementation * feat(gatekeeper): add checkout session completion webhook callback path * feat(gendo): fix not needing env vars if gendo module is not enabled * feat(gatekeeper): require a license for billing * chore(gatekeeper): cleanup before testing * feat(gatekeeper): subscriptionData parsing model * ci: add billing integration and gatekeeper modules to test config * test(gatekeeper): add checkout service tests * feat(gatekeeper): make completeCheckout callback idempotent properly * feat(gatekeeper): move to knex based repositories * test(gatekeeper): billing repository tests * feat(gatekeeper): add yearly billing cycle toggle * feat(ci): add stripe integration context to test job * feat(billingPage): conditionally render the checkout CTAs * fix(gatekeeper): remove flaky test condition * feat(helm): add billing integration feature flag * WIP billing gql api * feat(gatekeeper): cancel checkout session api * feat(gatekeeper): handle existing checkout sessions, when trying to create a new one * feat(gatekeeper): add workspace plans gql api * feat(gatekeeper): handle cancelation and subscription updates * fix(gatekeeper): scope initialization * fix(gatekeeper): eliminate stripe client import sideeffect * fix(gatekeeper): eliminate stripe client import sideeffect 2 * feat(gatekeeper): upsize subscription on workspace role change * feat(shared): add command pattern implementation * refactor(eventBus): remove return capabilities from the event bus * refactor(workspaces): use new commandFactory in workspace resolver * feat(core): facelift taskLock * feat(gatekeeper): shedule subscription downscale * feat(gatekeeper): manage subscription downscale * feat(gatekeeper): get workspace subscriptions, that are about to expire * feat(gatekeeper): manage subscription downscale * fix(gatekeeper): do not update subscription to canceled subs * ci: bump postgres and max connections * feat(workspaces): fix command factory event bugs
117 lines
4.3 KiB
TypeScript
117 lines
4.3 KiB
TypeScript
import cron from 'node-cron'
|
|
import { logger, moduleLogger } from '@/logging/logging'
|
|
import { SpeckleModule } from '@/modules/shared/helpers/typeHelper'
|
|
import { getFeatureFlags } from '@/modules/shared/helpers/envHelper'
|
|
import { validateModuleLicense } from '@/modules/gatekeeper/services/validateLicense'
|
|
import { getBillingRouter } from '@/modules/gatekeeper/rest/billing'
|
|
import { registerOrUpdateScopeFactory } from '@/modules/shared/repositories/scopes'
|
|
import { db } from '@/db/knex'
|
|
import { gatekeeperScopes } from '@/modules/gatekeeper/scopes'
|
|
import { initializeEventListenersFactory } from '@/modules/gatekeeper/events/eventListener'
|
|
import { getStripeClient, getWorkspacePlanProductId } from '@/modules/gatekeeper/stripe'
|
|
import { scheduleExecutionFactory } from '@/modules/core/services/taskScheduler'
|
|
import {
|
|
acquireTaskLockFactory,
|
|
releaseTaskLockFactory
|
|
} from '@/modules/core/repositories/scheduledTasks'
|
|
import {
|
|
downscaleWorkspaceSubscriptionFactory,
|
|
manageSubscriptionDownscaleFactory
|
|
} from '@/modules/gatekeeper/services/subscriptions'
|
|
import {
|
|
getWorkspacePlanFactory,
|
|
getWorkspaceSubscriptionsPastBillingCycleEndFactory,
|
|
upsertWorkspaceSubscriptionFactory
|
|
} from '@/modules/gatekeeper/repositories/billing'
|
|
import { countWorkspaceRoleWithOptionalProjectRoleFactory } from '@/modules/workspaces/repositories/workspaces'
|
|
import { reconcileWorkspaceSubscriptionFactory } from '@/modules/gatekeeper/clients/stripe'
|
|
|
|
const { FF_GATEKEEPER_MODULE_ENABLED, FF_BILLING_INTEGRATION_ENABLED } =
|
|
getFeatureFlags()
|
|
|
|
const initScopes = async () => {
|
|
const registerFunc = registerOrUpdateScopeFactory({ db })
|
|
await Promise.all(gatekeeperScopes.map((scope) => registerFunc({ scope })))
|
|
}
|
|
|
|
const scheduleWorkspaceSubscriptionDownscale = () => {
|
|
const scheduleExecution = scheduleExecutionFactory({
|
|
acquireTaskLock: acquireTaskLockFactory({ db }),
|
|
releaseTaskLock: releaseTaskLockFactory({ db })
|
|
})
|
|
|
|
const stripe = getStripeClient()
|
|
|
|
const manageSubscriptionDownscale = manageSubscriptionDownscaleFactory({
|
|
logger,
|
|
downscaleWorkspaceSubscription: downscaleWorkspaceSubscriptionFactory({
|
|
countWorkspaceRole: countWorkspaceRoleWithOptionalProjectRoleFactory({ db }),
|
|
getWorkspacePlan: getWorkspacePlanFactory({ db }),
|
|
reconcileSubscriptionData: reconcileWorkspaceSubscriptionFactory({ stripe }),
|
|
getWorkspacePlanProductId
|
|
}),
|
|
getWorkspaceSubscriptions: getWorkspaceSubscriptionsPastBillingCycleEndFactory({
|
|
db
|
|
}),
|
|
updateWorkspaceSubscription: upsertWorkspaceSubscriptionFactory({ db })
|
|
})
|
|
|
|
const cronExpression = '*/10 * * * * *'
|
|
return scheduleExecution(
|
|
cronExpression,
|
|
'WorkspaceSubscriptionDownscale',
|
|
async () => {
|
|
await manageSubscriptionDownscale()
|
|
// await cleanOrphanedWebhookConfigs()
|
|
}
|
|
)
|
|
}
|
|
|
|
let scheduledTask: cron.ScheduledTask | undefined = undefined
|
|
let quitListeners: (() => void) | undefined = undefined
|
|
|
|
const gatekeeperModule: SpeckleModule = {
|
|
async init(app, isInitial) {
|
|
await initScopes()
|
|
if (!FF_GATEKEEPER_MODULE_ENABLED) return
|
|
|
|
const isLicenseValid = await validateModuleLicense({
|
|
requiredModules: ['gatekeeper']
|
|
})
|
|
if (!isLicenseValid)
|
|
throw new Error(
|
|
'The gatekeeper module needs a valid license to run, contact Speckle to get one.'
|
|
)
|
|
|
|
moduleLogger.info('🗝️ Init gatekeeper module')
|
|
|
|
if (isInitial) {
|
|
// TODO: need to subscribe to the workspaceCreated event and store the workspacePlan as a trial if billing enabled, else store as unlimited
|
|
if (FF_BILLING_INTEGRATION_ENABLED) {
|
|
app.use(getBillingRouter())
|
|
|
|
scheduledTask = scheduleWorkspaceSubscriptionDownscale()
|
|
|
|
quitListeners = initializeEventListenersFactory({
|
|
db,
|
|
stripe: getStripeClient()
|
|
})()
|
|
|
|
const isLicenseValid = await validateModuleLicense({
|
|
requiredModules: ['billing']
|
|
})
|
|
if (!isLicenseValid)
|
|
throw new Error(
|
|
'The the billing module needs a valid license to run, contact Speckle to get one.'
|
|
)
|
|
// TODO: create a cron job, that removes unused seats from the subscription at the beginning of each workspace plan's billing cycle
|
|
}
|
|
}
|
|
},
|
|
async shutdown() {
|
|
if (quitListeners) quitListeners()
|
|
if (scheduledTask) scheduledTask.stop()
|
|
}
|
|
}
|
|
export = gatekeeperModule
|