budibase/packages/backend-core/src/events/identification.ts

303 lines
7.6 KiB
TypeScript
Raw Normal View History

2022-05-23 23:14:44 +02:00
import * as context from "../context"
import * as identityCtx from "../context/identity"
2022-05-23 23:14:44 +02:00
import env from "../environment"
import {
Hosting,
User,
Identity,
IdentityType,
Account,
2022-05-24 10:54:36 +02:00
isCloudAccount,
isSSOAccount,
TenantGroup,
2022-05-24 21:01:13 +02:00
SettingsConfig,
CloudAccount,
UserIdentity,
InstallationGroup,
UserContext,
Group,
2022-05-23 23:14:44 +02:00
} from "@budibase/types"
import { processors } from "./processors"
2022-05-24 21:01:13 +02:00
import * as dbUtils from "../db/utils"
import { Config } from "../constants"
2022-05-24 21:01:13 +02:00
import * as hashing from "../hashing"
import * as installation from "../installation"
2022-05-30 22:46:08 +02:00
import { withCache, TTL, CacheKeys } from "../cache/generic"
2022-05-23 23:14:44 +02:00
const pkg = require("../../package.json")
/**
* An identity can be:
* - account user (Self host)
* - budibase user
* - tenant
* - installation
*/
2022-05-24 21:01:13 +02:00
export const getCurrentIdentity = async (): Promise<Identity> => {
let identityContext = identityCtx.getIdentity()
const environment = getDeploymentEnvironment()
let identityType
2022-05-23 23:14:44 +02:00
if (!identityContext) {
identityType = IdentityType.TENANT
2022-05-23 23:14:44 +02:00
} else {
identityType = identityContext.type
2022-05-23 23:14:44 +02:00
}
if (identityType === IdentityType.INSTALLATION) {
const installationId = await getInstallationId()
const hosting = getHostingFromEnv()
return {
id: formatDistinctId(installationId, identityType),
hosting,
type: identityType,
installationId,
environment,
}
} else if (identityType === IdentityType.TENANT) {
const installationId = await getInstallationId()
2022-05-30 22:46:08 +02:00
const tenantId = await getEventTenantId(context.getTenantId())
const hosting = getHostingFromEnv()
return {
id: formatDistinctId(tenantId, identityType),
type: identityType,
hosting,
installationId,
tenantId,
environment,
}
} else if (identityType === IdentityType.USER) {
const userContext = identityContext as UserContext
2022-05-30 22:46:08 +02:00
const tenantId = await getEventTenantId(context.getTenantId())
const installationId = await getInstallationId()
2022-05-23 23:14:44 +02:00
const account = userContext.account
let hosting
if (account) {
hosting = account.hosting
} else {
hosting = getHostingFromEnv()
}
return {
id: userContext._id,
type: identityType,
hosting,
installationId,
tenantId,
environment,
}
2022-05-24 21:01:13 +02:00
} else {
throw new Error("Unknown identity type")
2022-05-24 21:01:13 +02:00
}
}
export const identifyInstallationGroup = async (
installId: string,
timestamp?: string | number
): Promise<void> => {
const id = installId
const type = IdentityType.INSTALLATION
const hosting = getHostingFromEnv()
const version = pkg.version
const environment = getDeploymentEnvironment()
const group: InstallationGroup = {
id,
type,
hosting,
version,
environment,
}
await identifyGroup(group, timestamp)
// need to create a normal identity for the group to be able to query it globally
// match the posthog syntax to link this identity to the empty auto generated one
await identify({ ...group, id: `$${type}_${id}` }, timestamp)
}
export const identifyTenantGroup = async (
2022-05-25 22:32:08 +02:00
tenantId: string,
account: Account | undefined,
2022-05-25 22:32:08 +02:00
timestamp?: string | number
): Promise<void> => {
2022-05-30 22:46:08 +02:00
const id = await getEventTenantId(tenantId)
const type = IdentityType.TENANT
const installationId = await getInstallationId()
const environment = getDeploymentEnvironment()
2022-05-24 21:01:13 +02:00
let hosting: Hosting
let profession: string | undefined
let companySize: string | undefined
if (account) {
profession = account.profession
companySize = account.size
hosting = account.hosting
} else {
hosting = getHostingFromEnv()
}
const group: TenantGroup = {
id,
type,
hosting,
environment,
installationId,
profession,
companySize,
2022-05-24 21:01:13 +02:00
}
await identifyGroup(group, timestamp)
// need to create a normal identity for the group to be able to query it globally
// match the posthog syntax to link this identity to the auto generated one
await identify({ ...group, id: `$${type}_${id}` }, timestamp)
2022-05-24 21:01:13 +02:00
}
export const identifyUser = async (
user: User,
account: CloudAccount | undefined,
timestamp?: string | number
) => {
2022-05-23 23:14:44 +02:00
const id = user._id as string
2022-05-30 22:46:08 +02:00
const tenantId = await getEventTenantId(user.tenantId)
2022-05-23 23:14:44 +02:00
const type = IdentityType.USER
let builder = user.builder?.global || false
let admin = user.admin?.global || false
2022-05-24 21:01:13 +02:00
let providerType = user.providerType
const accountHolder = account?.budibaseUserId === user._id || false
const verified =
account && account?.budibaseUserId === user._id ? account.verified : false
const installationId = await getInstallationId()
const hosting = account ? account.hosting : getHostingFromEnv()
const environment = getDeploymentEnvironment()
2022-05-23 23:14:44 +02:00
const identity: UserIdentity = {
2022-05-23 23:14:44 +02:00
id,
type,
hosting,
installationId,
tenantId,
verified,
accountHolder,
providerType,
2022-05-23 23:14:44 +02:00
builder,
admin,
environment,
2022-05-23 23:14:44 +02:00
}
2022-05-25 22:32:08 +02:00
await identify(identity, timestamp)
2022-05-23 23:14:44 +02:00
}
export const identifyAccount = async (account: Account) => {
let id = account.accountId
const tenantId = account.tenantId
let type = IdentityType.USER
2022-05-24 21:01:13 +02:00
let providerType = isSSOAccount(account) ? account.providerType : undefined
const verified = account.verified
const accountHolder = true
const hosting = account.hosting
const installationId = await getInstallationId()
const environment = getDeploymentEnvironment()
2022-05-23 23:14:44 +02:00
if (isCloudAccount(account)) {
if (account.budibaseUserId) {
// use the budibase user as the id if set
id = account.budibaseUserId
}
}
const identity: UserIdentity = {
2022-05-23 23:14:44 +02:00
id,
type,
hosting,
installationId,
tenantId,
2022-05-24 21:01:13 +02:00
providerType,
verified,
accountHolder,
environment,
2022-05-23 23:14:44 +02:00
}
await identify(identity)
}
export const identify = async (
identity: Identity,
timestamp?: string | number
) => {
await processors.identify(identity, timestamp)
2022-05-23 23:14:44 +02:00
}
export const identifyGroup = async (
group: Group,
timestamp?: string | number
) => {
await processors.identifyGroup(group, timestamp)
}
const getDeploymentEnvironment = () => {
if (env.isDev()) {
return "development"
} else {
return env.DEPLOYMENT_ENVIRONMENT
}
}
const getHostingFromEnv = () => {
return env.SELF_HOSTED ? Hosting.SELF : Hosting.CLOUD
}
export const getInstallationId = async () => {
if (isAccountPortal()) {
return "account-portal"
}
const install = await installation.getInstall()
return install.installId
}
2022-05-30 22:46:08 +02:00
const getEventTenantId = async (tenantId: string): Promise<string> => {
if (env.SELF_HOSTED) {
2022-05-30 22:46:08 +02:00
return getUniqueTenantId(tenantId)
} else {
// tenant id's in the cloud are already unique
return tenantId
}
}
2022-05-30 22:46:08 +02:00
const getUniqueTenantId = async (tenantId: string): Promise<string> => {
// make sure this tenantId always matches the tenantId in context
return context.doInTenant(tenantId, () => {
return withCache(CacheKeys.UNIQUE_TENANT_ID, TTL.ONE_DAY, async () => {
const db = context.getGlobalDB()
const config: SettingsConfig = await dbUtils.getScopedFullConfig(db, {
type: Config.SETTINGS,
2022-05-30 22:46:08 +02:00
})
let uniqueTenantId: string
if (config.config.uniqueTenantId) {
return config.config.uniqueTenantId
} else {
uniqueTenantId = `${hashing.newid()}_${tenantId}`
config.config.uniqueTenantId = uniqueTenantId
await db.put(config)
return uniqueTenantId
}
})
})
}
const isAccountPortal = () => {
return env.SERVICE === "account-portal"
}
const formatDistinctId = (id: string, type: IdentityType) => {
if (type === IdentityType.INSTALLATION || type === IdentityType.TENANT) {
return `$${type}_${id}`
} else {
return id
}
}