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

265 lines
6.4 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,
isSelfHostAccount,
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 { Configs } from "../constants"
import * as hashing from "../hashing"
import * as installation from "../installation"
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()
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()
return {
id: formatDistinctId(installationId, identityType),
type: identityType,
installationId,
}
} else if (identityType === IdentityType.TENANT) {
const installationId = await getInstallationId()
const tenantId = await getCurrentTenantId()
return {
id: formatDistinctId(tenantId, identityType),
type: identityType,
installationId,
tenantId,
}
} else if (identityType === IdentityType.USER) {
const userContext = identityContext as UserContext
const tenantId = await getCurrentTenantId()
let installationId: string | undefined
2022-05-23 23:14:44 +02:00
// self host account users won't have installation
if (!userContext.account || !isSelfHostAccount(userContext.account)) {
installationId = await getInstallationId()
}
return {
id: userContext._id,
type: identityType,
installationId,
tenantId,
}
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 group: InstallationGroup = {
id,
type,
hosting,
version,
}
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> => {
const id = await getGlobalTenantId(tenantId)
const type = IdentityType.TENANT
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,
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
const tenantId = await getGlobalTenantId(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()
2022-05-23 23:14:44 +02:00
const identity: UserIdentity = {
2022-05-23 23:14:44 +02:00
id,
type,
installationId,
tenantId,
verified,
accountHolder,
providerType,
2022-05-23 23:14:44 +02:00
builder,
admin,
}
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
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,
tenantId,
2022-05-24 21:01:13 +02:00
providerType,
verified,
accountHolder,
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 getHostingFromEnv = () => {
return env.SELF_HOSTED ? Hosting.SELF : Hosting.CLOUD
}
export const getCurrentTenantId = () => getGlobalTenantId(context.getTenantId())
export const getInstallationId = async () => {
if (isAccountPortal()) {
return "account-portal"
}
const install = await installation.getInstall()
return install.installId
}
const getGlobalTenantId = async (tenantId: string): Promise<string> => {
if (env.SELF_HOSTED) {
return getGlobalId(tenantId)
} else {
// tenant id's in the cloud are already unique
return tenantId
}
}
// TODO: cache in redis
const getGlobalId = async (tenantId: string): Promise<string> => {
const db = context.getGlobalDB()
const config: SettingsConfig = await dbUtils.getScopedFullConfig(db, {
type: Configs.SETTINGS,
})
let globalId: string
if (config.config.globalId) {
return config.config.globalId
} else {
globalId = `${hashing.newid()}_${tenantId}`
config.config.globalId = globalId
await db.put(config)
return globalId
}
}
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
}
}