2022-05-23 23:14:44 +02:00
|
|
|
import * as context from "../context"
|
2022-05-28 22:38:22 +02:00
|
|
|
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,
|
2022-05-28 22:38:22 +02:00
|
|
|
TenantGroup,
|
2022-05-26 11:13:26 +02:00
|
|
|
CloudAccount,
|
|
|
|
UserIdentity,
|
2022-05-28 22:38:22 +02:00
|
|
|
InstallationGroup,
|
|
|
|
UserContext,
|
|
|
|
Group,
|
2023-02-21 09:23:53 +01:00
|
|
|
isSSOUser,
|
2022-05-23 23:14:44 +02:00
|
|
|
} from "@budibase/types"
|
2022-05-27 00:57:14 +02:00
|
|
|
import { processors } from "./processors"
|
2022-11-24 19:48:51 +01:00
|
|
|
import { newid } from "../utils"
|
2022-05-28 22:38:22 +02:00
|
|
|
import * as installation from "../installation"
|
2023-02-23 14:41:35 +01:00
|
|
|
import * as configs from "../configs"
|
2023-07-18 17:57:48 +02:00
|
|
|
import * as users from "../users"
|
2022-11-24 19:48:51 +01:00
|
|
|
import { withCache, TTL, CacheKey } from "../cache/generic"
|
2022-05-23 23:14:44 +02:00
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
/**
|
|
|
|
* An identity can be:
|
|
|
|
* - account user (Self host)
|
|
|
|
* - budibase user
|
|
|
|
* - tenant
|
|
|
|
* - installation
|
|
|
|
*/
|
2023-01-11 10:37:37 +01:00
|
|
|
const getCurrentIdentity = async (): Promise<Identity> => {
|
2022-05-28 22:38:22 +02:00
|
|
|
let identityContext = identityCtx.getIdentity()
|
2022-06-15 12:46:03 +02:00
|
|
|
const environment = getDeploymentEnvironment()
|
2022-05-27 00:57:14 +02:00
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
let identityType
|
2022-05-23 23:14:44 +02:00
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
if (!identityContext) {
|
|
|
|
identityType = IdentityType.TENANT
|
2022-05-23 23:14:44 +02:00
|
|
|
} else {
|
2022-05-28 22:38:22 +02:00
|
|
|
identityType = identityContext.type
|
2022-05-23 23:14:44 +02:00
|
|
|
}
|
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
if (identityType === IdentityType.INSTALLATION) {
|
|
|
|
const installationId = await getInstallationId()
|
2022-06-14 18:34:32 +02:00
|
|
|
const hosting = getHostingFromEnv()
|
2022-05-28 22:38:22 +02:00
|
|
|
return {
|
|
|
|
id: formatDistinctId(installationId, identityType),
|
2022-06-14 18:34:32 +02:00
|
|
|
hosting,
|
2022-05-28 22:38:22 +02:00
|
|
|
type: identityType,
|
|
|
|
installationId,
|
2022-06-15 12:46:03 +02:00
|
|
|
environment,
|
2022-05-28 22:38:22 +02:00
|
|
|
}
|
|
|
|
} else if (identityType === IdentityType.TENANT) {
|
|
|
|
const installationId = await getInstallationId()
|
2022-05-30 22:46:08 +02:00
|
|
|
const tenantId = await getEventTenantId(context.getTenantId())
|
2022-06-14 18:34:32 +02:00
|
|
|
const hosting = getHostingFromEnv()
|
2022-05-27 00:57:14 +02:00
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
return {
|
|
|
|
id: formatDistinctId(tenantId, identityType),
|
|
|
|
type: identityType,
|
2022-06-14 18:34:32 +02:00
|
|
|
hosting,
|
2022-05-28 22:38:22 +02:00
|
|
|
installationId,
|
|
|
|
tenantId,
|
2023-04-17 16:59:59 +02:00
|
|
|
realTenantId: context.getTenantId(),
|
2022-06-15 12:46:03 +02:00
|
|
|
environment,
|
2022-05-28 22:38:22 +02:00
|
|
|
}
|
|
|
|
} else if (identityType === IdentityType.USER) {
|
|
|
|
const userContext = identityContext as UserContext
|
2022-05-30 22:46:08 +02:00
|
|
|
const tenantId = await getEventTenantId(context.getTenantId())
|
2022-06-14 18:34:32 +02:00
|
|
|
const installationId = await getInstallationId()
|
2022-05-23 23:14:44 +02:00
|
|
|
|
2022-06-14 18:34:32 +02:00
|
|
|
const account = userContext.account
|
|
|
|
let hosting
|
|
|
|
if (account) {
|
|
|
|
hosting = account.hosting
|
|
|
|
} else {
|
|
|
|
hosting = getHostingFromEnv()
|
2022-05-28 22:38:22 +02:00
|
|
|
}
|
2022-05-27 00:57:14 +02:00
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
return {
|
|
|
|
id: userContext._id,
|
|
|
|
type: identityType,
|
2022-06-14 18:34:32 +02:00
|
|
|
hosting,
|
2022-05-28 22:38:22 +02:00
|
|
|
installationId,
|
|
|
|
tenantId,
|
2022-06-15 12:46:03 +02:00
|
|
|
environment,
|
2023-05-17 20:21:32 +02:00
|
|
|
realTenantId: context.getTenantId(),
|
2023-02-22 17:10:54 +01:00
|
|
|
hostInfo: userContext.hostInfo,
|
2022-05-28 22:38:22 +02:00
|
|
|
}
|
2022-05-24 21:01:13 +02:00
|
|
|
} else {
|
2022-05-28 22:38:22 +02:00
|
|
|
throw new Error("Unknown identity type")
|
2022-05-24 21:01:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-11 10:37:37 +01:00
|
|
|
const identifyInstallationGroup = async (
|
2022-05-28 22:38:22 +02:00
|
|
|
installId: string,
|
|
|
|
timestamp?: string | number
|
|
|
|
): Promise<void> => {
|
|
|
|
const id = installId
|
2022-05-27 00:57:14 +02:00
|
|
|
const type = IdentityType.INSTALLATION
|
|
|
|
const hosting = getHostingFromEnv()
|
2023-04-17 14:53:00 +02:00
|
|
|
const version = env.VERSION
|
2022-06-15 12:46:03 +02:00
|
|
|
const environment = getDeploymentEnvironment()
|
2022-05-27 00:57:14 +02:00
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
const group: InstallationGroup = {
|
2022-05-27 00:57:14 +02:00
|
|
|
id,
|
|
|
|
type,
|
|
|
|
hosting,
|
2022-05-28 22:38:22 +02:00
|
|
|
version,
|
2022-06-15 12:46:03 +02:00
|
|
|
environment,
|
2022-05-27 00:57:14 +02:00
|
|
|
}
|
2022-05-28 22:38:22 +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 empty auto generated one
|
|
|
|
await identify({ ...group, id: `$${type}_${id}` }, timestamp)
|
2022-05-27 00:57:14 +02:00
|
|
|
}
|
|
|
|
|
2023-01-11 10:37:37 +01:00
|
|
|
const identifyTenantGroup = async (
|
2022-05-25 22:32:08 +02:00
|
|
|
tenantId: string,
|
2022-05-28 22:38:22 +02:00
|
|
|
account: Account | undefined,
|
2022-05-25 22:32:08 +02:00
|
|
|
timestamp?: string | number
|
2022-05-28 22:38:22 +02:00
|
|
|
): Promise<void> => {
|
2022-05-30 22:46:08 +02:00
|
|
|
const id = await getEventTenantId(tenantId)
|
2022-05-26 11:13:26 +02:00
|
|
|
const type = IdentityType.TENANT
|
2022-06-14 18:34:32 +02:00
|
|
|
const installationId = await getInstallationId()
|
2022-06-15 12:46:03 +02:00
|
|
|
const environment = getDeploymentEnvironment()
|
2022-05-24 21:01:13 +02:00
|
|
|
|
2022-05-28 22:38:22 +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 = {
|
2022-05-26 11:13:26 +02:00
|
|
|
id,
|
|
|
|
type,
|
2022-05-28 22:38:22 +02:00
|
|
|
hosting,
|
2022-06-15 12:46:03 +02:00
|
|
|
environment,
|
2022-06-14 18:34:32 +02:00
|
|
|
installationId,
|
2022-05-26 11:13:26 +02:00
|
|
|
profession,
|
|
|
|
companySize,
|
2022-05-24 21:01:13 +02:00
|
|
|
}
|
2022-05-28 22:38:22 +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
|
|
|
}
|
|
|
|
|
2023-01-11 10:37:37 +01:00
|
|
|
const identifyUser = async (
|
2022-05-26 11:13:26 +02:00
|
|
|
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
|
2023-07-18 17:57:48 +02:00
|
|
|
let builder = users.hasBuilderPermissions(user)
|
|
|
|
let admin = users.hasAdminPermissions(user)
|
2023-02-21 09:23:53 +01:00
|
|
|
let providerType
|
|
|
|
if (isSSOUser(user)) {
|
|
|
|
providerType = user.providerType
|
|
|
|
}
|
2024-10-28 15:46:42 +01:00
|
|
|
const accountHolder = await users.getExistingAccounts([user.email])
|
|
|
|
const isAccountHolder = accountHolder.length > 0
|
|
|
|
const verified = !!account && isAccountHolder && account.verified
|
2022-05-28 22:38:22 +02:00
|
|
|
const installationId = await getInstallationId()
|
2022-06-14 18:34:32 +02:00
|
|
|
const hosting = account ? account.hosting : getHostingFromEnv()
|
2022-06-15 12:46:03 +02:00
|
|
|
const environment = getDeploymentEnvironment()
|
2022-05-23 23:14:44 +02:00
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
const identity: UserIdentity = {
|
2022-05-23 23:14:44 +02:00
|
|
|
id,
|
|
|
|
type,
|
2022-06-14 18:34:32 +02:00
|
|
|
hosting,
|
2022-05-28 22:38:22 +02:00
|
|
|
installationId,
|
|
|
|
tenantId,
|
|
|
|
verified,
|
2024-10-28 15:46:42 +01:00
|
|
|
accountHolder: isAccountHolder,
|
2022-05-28 22:38:22 +02:00
|
|
|
providerType,
|
2022-05-23 23:14:44 +02:00
|
|
|
builder,
|
|
|
|
admin,
|
2022-06-15 12:46:03 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2023-01-11 10:37:37 +01:00
|
|
|
const identifyAccount = async (account: Account) => {
|
2022-05-23 23:14:44 +02:00
|
|
|
let id = account.accountId
|
|
|
|
const tenantId = account.tenantId
|
2022-05-25 01:15:52 +02:00
|
|
|
let type = IdentityType.USER
|
2022-05-24 21:01:13 +02:00
|
|
|
let providerType = isSSOAccount(account) ? account.providerType : undefined
|
2022-05-26 11:13:26 +02:00
|
|
|
const verified = account.verified
|
|
|
|
const accountHolder = true
|
2022-06-14 18:34:32 +02:00
|
|
|
const hosting = account.hosting
|
|
|
|
const installationId = await getInstallationId()
|
2022-06-15 12:46:03 +02:00
|
|
|
const environment = getDeploymentEnvironment()
|
2022-05-23 23:14:44 +02:00
|
|
|
|
|
|
|
if (isCloudAccount(account)) {
|
2024-10-28 15:46:42 +01:00
|
|
|
const user = await users.getGlobalUserByEmail(account.email)
|
|
|
|
if (user?._id) {
|
2022-05-23 23:14:44 +02:00
|
|
|
// use the budibase user as the id if set
|
2024-10-28 15:46:42 +01:00
|
|
|
id = user._id
|
2022-05-23 23:14:44 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-26 11:13:26 +02:00
|
|
|
const identity: UserIdentity = {
|
2022-05-23 23:14:44 +02:00
|
|
|
id,
|
|
|
|
type,
|
2022-06-14 18:34:32 +02:00
|
|
|
hosting,
|
|
|
|
installationId,
|
2022-05-28 22:38:22 +02:00
|
|
|
tenantId,
|
2022-05-24 21:01:13 +02:00
|
|
|
providerType,
|
2022-05-26 11:13:26 +02:00
|
|
|
verified,
|
|
|
|
accountHolder,
|
2022-06-15 12:46:03 +02:00
|
|
|
environment,
|
2022-05-23 23:14:44 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
await identify(identity)
|
|
|
|
}
|
|
|
|
|
2023-01-11 10:37:37 +01:00
|
|
|
const identify = async (identity: Identity, timestamp?: string | number) => {
|
2022-05-27 00:57:14 +02:00
|
|
|
await processors.identify(identity, timestamp)
|
2022-05-23 23:14:44 +02:00
|
|
|
}
|
2022-05-28 22:38:22 +02:00
|
|
|
|
2023-01-11 10:37:37 +01:00
|
|
|
const identifyGroup = async (group: Group, timestamp?: string | number) => {
|
2022-05-28 22:38:22 +02:00
|
|
|
await processors.identifyGroup(group, timestamp)
|
|
|
|
}
|
|
|
|
|
2022-06-15 12:46:03 +02:00
|
|
|
const getDeploymentEnvironment = () => {
|
|
|
|
if (env.isDev()) {
|
|
|
|
return "development"
|
|
|
|
} else {
|
|
|
|
return env.DEPLOYMENT_ENVIRONMENT
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-28 22:38:22 +02:00
|
|
|
const getHostingFromEnv = () => {
|
|
|
|
return env.SELF_HOSTED ? Hosting.SELF : Hosting.CLOUD
|
|
|
|
}
|
|
|
|
|
2023-01-11 10:37:37 +01:00
|
|
|
const getInstallationId = async () => {
|
2022-05-28 22:38:22 +02:00
|
|
|
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> => {
|
2022-05-28 22:38:22 +02:00
|
|
|
if (env.SELF_HOSTED) {
|
2022-05-30 22:46:08 +02:00
|
|
|
return getUniqueTenantId(tenantId)
|
2022-05-28 22:38:22 +02:00
|
|
|
} else {
|
|
|
|
// tenant id's in the cloud are already unique
|
|
|
|
return tenantId
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-07-07 21:07:15 +02:00
|
|
|
export const getUniqueTenantId = async (tenantId: string): Promise<string> => {
|
2022-05-30 22:46:08 +02:00
|
|
|
// make sure this tenantId always matches the tenantId in context
|
|
|
|
return context.doInTenant(tenantId, () => {
|
2022-11-24 19:48:51 +01:00
|
|
|
return withCache(CacheKey.UNIQUE_TENANT_ID, TTL.ONE_DAY, async () => {
|
2022-05-30 22:46:08 +02:00
|
|
|
const db = context.getGlobalDB()
|
2023-02-23 14:41:35 +01:00
|
|
|
const config = await configs.getSettingsConfigDoc()
|
2022-05-30 22:46:08 +02:00
|
|
|
|
|
|
|
let uniqueTenantId: string
|
|
|
|
if (config.config.uniqueTenantId) {
|
|
|
|
return config.config.uniqueTenantId
|
|
|
|
} else {
|
2022-11-24 19:48:51 +01:00
|
|
|
uniqueTenantId = `${newid()}_${tenantId}`
|
2022-05-30 22:46:08 +02:00
|
|
|
config.config.uniqueTenantId = uniqueTenantId
|
|
|
|
await db.put(config)
|
|
|
|
return uniqueTenantId
|
|
|
|
}
|
|
|
|
})
|
2022-05-28 22:38:22 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
|
}
|
|
|
|
}
|
2023-01-11 10:37:37 +01:00
|
|
|
|
|
|
|
export default {
|
|
|
|
getCurrentIdentity,
|
|
|
|
identifyInstallationGroup,
|
|
|
|
identifyTenantGroup,
|
|
|
|
identifyUser,
|
|
|
|
identifyAccount,
|
|
|
|
identify,
|
|
|
|
identifyGroup,
|
|
|
|
getInstallationId,
|
2023-04-24 10:31:48 +02:00
|
|
|
getUniqueTenantId,
|
2023-01-11 10:37:37 +01:00
|
|
|
}
|