app and account properties, add account details to all user and tenant identities
This commit is contained in:
parent
163b667f95
commit
cbc3e72757
|
@ -1,39 +0,0 @@
|
||||||
const API = require("./api")
|
|
||||||
const env = require("../environment")
|
|
||||||
const { Headers } = require("../constants")
|
|
||||||
|
|
||||||
const api = new API(env.ACCOUNT_PORTAL_URL)
|
|
||||||
|
|
||||||
exports.getAccount = async email => {
|
|
||||||
const payload = {
|
|
||||||
email,
|
|
||||||
}
|
|
||||||
const response = await api.post(`/api/accounts/search`, {
|
|
||||||
body: payload,
|
|
||||||
headers: {
|
|
||||||
[Headers.API_KEY]: env.ACCOUNT_PORTAL_API_KEY,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
const json = await response.json()
|
|
||||||
|
|
||||||
if (response.status !== 200) {
|
|
||||||
throw new Error(`Error getting account by email ${email}`, json)
|
|
||||||
}
|
|
||||||
|
|
||||||
return json[0]
|
|
||||||
}
|
|
||||||
|
|
||||||
exports.getStatus = async () => {
|
|
||||||
const response = await api.get(`/api/status`, {
|
|
||||||
headers: {
|
|
||||||
[Headers.API_KEY]: env.ACCOUNT_PORTAL_API_KEY,
|
|
||||||
},
|
|
||||||
})
|
|
||||||
const json = await response.json()
|
|
||||||
|
|
||||||
if (response.status !== 200) {
|
|
||||||
throw new Error(`Error getting status`)
|
|
||||||
}
|
|
||||||
|
|
||||||
return json
|
|
||||||
}
|
|
|
@ -0,0 +1,63 @@
|
||||||
|
import API from "./api"
|
||||||
|
import env from "../environment"
|
||||||
|
import { Headers } from "../constants"
|
||||||
|
import { CloudAccount } from "@budibase/types"
|
||||||
|
|
||||||
|
const api = new API(env.ACCOUNT_PORTAL_URL)
|
||||||
|
|
||||||
|
export const getAccount = async (
|
||||||
|
email: string
|
||||||
|
): Promise<CloudAccount | undefined> => {
|
||||||
|
const payload = {
|
||||||
|
email,
|
||||||
|
}
|
||||||
|
const response = await api.post(`/api/accounts/search`, {
|
||||||
|
body: payload,
|
||||||
|
headers: {
|
||||||
|
[Headers.API_KEY]: env.ACCOUNT_PORTAL_API_KEY,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
if (response.status !== 200) {
|
||||||
|
throw new Error(`Error getting account by email ${email}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
const json: CloudAccount[] = await response.json()
|
||||||
|
return json[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getAccountByTenantId = async (
|
||||||
|
tenantId: string
|
||||||
|
): Promise<CloudAccount | undefined> => {
|
||||||
|
const payload = {
|
||||||
|
tenantId,
|
||||||
|
}
|
||||||
|
const response = await api.post(`/api/accounts/search`, {
|
||||||
|
body: payload,
|
||||||
|
headers: {
|
||||||
|
[Headers.API_KEY]: env.ACCOUNT_PORTAL_API_KEY,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
if (response.status !== 200) {
|
||||||
|
throw new Error(`Error getting account by tenantId ${tenantId}`)
|
||||||
|
}
|
||||||
|
|
||||||
|
const json: CloudAccount[] = await response.json()
|
||||||
|
return json[0]
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getStatus = async () => {
|
||||||
|
const response = await api.get(`/api/status`, {
|
||||||
|
headers: {
|
||||||
|
[Headers.API_KEY]: env.ACCOUNT_PORTAL_API_KEY,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
const json = await response.json()
|
||||||
|
|
||||||
|
if (response.status !== 200) {
|
||||||
|
throw new Error(`Error getting status`)
|
||||||
|
}
|
||||||
|
|
||||||
|
return json
|
||||||
|
}
|
|
@ -7,12 +7,13 @@ import {
|
||||||
Identity,
|
Identity,
|
||||||
IdentityType,
|
IdentityType,
|
||||||
Account,
|
Account,
|
||||||
AccountIdentity,
|
|
||||||
BudibaseIdentity,
|
BudibaseIdentity,
|
||||||
isCloudAccount,
|
isCloudAccount,
|
||||||
isSSOAccount,
|
isSSOAccount,
|
||||||
TenantIdentity,
|
TenantIdentity,
|
||||||
SettingsConfig,
|
SettingsConfig,
|
||||||
|
CloudAccount,
|
||||||
|
UserIdentity,
|
||||||
} from "@budibase/types"
|
} from "@budibase/types"
|
||||||
import { analyticsProcessor } from "./processors"
|
import { analyticsProcessor } from "./processors"
|
||||||
import * as dbUtils from "../db/utils"
|
import * as dbUtils from "../db/utils"
|
||||||
|
@ -81,20 +82,32 @@ const getHostingFromEnv = () => {
|
||||||
|
|
||||||
export const identifyTenant = async (
|
export const identifyTenant = async (
|
||||||
tenantId: string,
|
tenantId: string,
|
||||||
|
account: CloudAccount | undefined,
|
||||||
timestamp?: string | number
|
timestamp?: string | number
|
||||||
) => {
|
) => {
|
||||||
const global = await getGlobalIdentifiers(tenantId)
|
const global = await getGlobalIdentifiers(tenantId)
|
||||||
|
const id = global.id
|
||||||
|
const hosting = getHostingFromEnv()
|
||||||
|
const type = IdentityType.TENANT
|
||||||
|
const profession = account?.profession
|
||||||
|
const companySize = account?.size
|
||||||
|
|
||||||
const identity: TenantIdentity = {
|
const identity: TenantIdentity = {
|
||||||
id: global.id,
|
id,
|
||||||
tenantId: global.tenantId,
|
tenantId: global.tenantId,
|
||||||
hosting: getHostingFromEnv(),
|
hosting,
|
||||||
type: IdentityType.TENANT,
|
type,
|
||||||
|
profession,
|
||||||
|
companySize,
|
||||||
}
|
}
|
||||||
await identify(identity, timestamp)
|
await identify(identity, timestamp)
|
||||||
}
|
}
|
||||||
|
|
||||||
export const identifyUser = async (user: User, timestamp?: string | number) => {
|
export const identifyUser = async (
|
||||||
|
user: User,
|
||||||
|
account: CloudAccount | undefined,
|
||||||
|
timestamp?: string | number
|
||||||
|
) => {
|
||||||
const id = user._id as string
|
const id = user._id as string
|
||||||
const tenantId = user.tenantId
|
const tenantId = user.tenantId
|
||||||
const hosting = env.SELF_HOSTED ? Hosting.SELF : Hosting.CLOUD
|
const hosting = env.SELF_HOSTED ? Hosting.SELF : Hosting.CLOUD
|
||||||
|
@ -102,6 +115,10 @@ export const identifyUser = async (user: User, timestamp?: string | number) => {
|
||||||
let builder = user.builder?.global
|
let builder = user.builder?.global
|
||||||
let admin = user.admin?.global
|
let admin = user.admin?.global
|
||||||
let providerType = user.providerType
|
let providerType = user.providerType
|
||||||
|
const accountHolder = account?.budibaseUserId === user._id
|
||||||
|
const verified = account ? account.verified : false
|
||||||
|
const profession = account?.profession
|
||||||
|
const companySize = account?.size
|
||||||
|
|
||||||
const identity: BudibaseIdentity = {
|
const identity: BudibaseIdentity = {
|
||||||
id,
|
id,
|
||||||
|
@ -111,6 +128,10 @@ export const identifyUser = async (user: User, timestamp?: string | number) => {
|
||||||
builder,
|
builder,
|
||||||
admin,
|
admin,
|
||||||
providerType,
|
providerType,
|
||||||
|
accountHolder,
|
||||||
|
verified,
|
||||||
|
profession,
|
||||||
|
companySize,
|
||||||
}
|
}
|
||||||
|
|
||||||
await identify(identity, timestamp)
|
await identify(identity, timestamp)
|
||||||
|
@ -122,6 +143,10 @@ export const identifyAccount = async (account: Account) => {
|
||||||
const hosting = account.hosting
|
const hosting = account.hosting
|
||||||
let type = IdentityType.USER
|
let type = IdentityType.USER
|
||||||
let providerType = isSSOAccount(account) ? account.providerType : undefined
|
let providerType = isSSOAccount(account) ? account.providerType : undefined
|
||||||
|
const verified = account.verified
|
||||||
|
const profession = account.profession
|
||||||
|
const companySize = account.size
|
||||||
|
const accountHolder = true
|
||||||
|
|
||||||
if (isCloudAccount(account)) {
|
if (isCloudAccount(account)) {
|
||||||
if (account.budibaseUserId) {
|
if (account.budibaseUserId) {
|
||||||
|
@ -130,15 +155,16 @@ export const identifyAccount = async (account: Account) => {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const identity: AccountIdentity = {
|
const identity: UserIdentity = {
|
||||||
id,
|
id,
|
||||||
tenantId,
|
tenantId,
|
||||||
hosting,
|
hosting,
|
||||||
type,
|
type,
|
||||||
providerType,
|
providerType,
|
||||||
verified: account.verified,
|
verified,
|
||||||
profession: account.profession,
|
profession,
|
||||||
companySize: account.size,
|
companySize,
|
||||||
|
accountHolder,
|
||||||
}
|
}
|
||||||
|
|
||||||
await identify(identity)
|
await identify(identity)
|
||||||
|
|
|
@ -6,13 +6,19 @@ export default class LoggingProcessor implements EventProcessor {
|
||||||
async processEvent(
|
async processEvent(
|
||||||
event: Event,
|
event: Event,
|
||||||
identity: Identity,
|
identity: Identity,
|
||||||
properties: any
|
properties: any,
|
||||||
|
timestamp?: string
|
||||||
): Promise<void> {
|
): Promise<void> {
|
||||||
if (env.SELF_HOSTED && !env.isDev()) {
|
if (env.SELF_HOSTED && !env.isDev()) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
let timestampString = ""
|
||||||
|
if (timestamp) {
|
||||||
|
timestampString = `[timestamp=${new Date(timestamp).toISOString()}]`
|
||||||
|
}
|
||||||
|
|
||||||
console.log(
|
console.log(
|
||||||
`[audit] [tenant=${identity.tenantId}] [identityType=${identity.type}] [identity=${identity.id}] ${event}`
|
`[audit] [tenant=${identity.tenantId}] [identityType=${identity.type}] [identity=${identity.id}] ${timestampString} ${event} `
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,17 +1,29 @@
|
||||||
import { publishEvent } from "../events"
|
import { publishEvent } from "../events"
|
||||||
import { Event, Account } from "@budibase/types"
|
import {
|
||||||
|
Event,
|
||||||
|
Account,
|
||||||
|
AccountCreatedEvent,
|
||||||
|
AccountDeletedEvent,
|
||||||
|
AccountVerifiedEvent,
|
||||||
|
} from "@budibase/types"
|
||||||
|
|
||||||
export async function created(account: Account) {
|
export async function created(account: Account) {
|
||||||
const properties = {}
|
const properties: AccountCreatedEvent = {
|
||||||
|
tenantId: account.tenantId,
|
||||||
|
}
|
||||||
await publishEvent(Event.ACCOUNT_CREATED, properties)
|
await publishEvent(Event.ACCOUNT_CREATED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleted(account: Account) {
|
export async function deleted(account: Account) {
|
||||||
const properties = {}
|
const properties: AccountDeletedEvent = {
|
||||||
|
tenantId: account.tenantId,
|
||||||
|
}
|
||||||
await publishEvent(Event.ACCOUNT_DELETED, properties)
|
await publishEvent(Event.ACCOUNT_DELETED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function verified(account: Account) {
|
export async function verified(account: Account) {
|
||||||
const properties = {}
|
const properties: AccountVerifiedEvent = {
|
||||||
|
tenantId: account.tenantId,
|
||||||
|
}
|
||||||
await publishEvent(Event.ACCOUNT_VERIFIED, properties)
|
await publishEvent(Event.ACCOUNT_VERIFIED, properties)
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,58 +16,93 @@ import {
|
||||||
} from "@budibase/types"
|
} from "@budibase/types"
|
||||||
|
|
||||||
export const created = async (app: App, timestamp?: string | number) => {
|
export const created = async (app: App, timestamp?: string | number) => {
|
||||||
const properties: AppCreatedEvent = {}
|
const properties: AppCreatedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
version: app.version,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_CREATED, properties, timestamp)
|
await publishEvent(Event.APP_CREATED, properties, timestamp)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function updated(app: App) {
|
export async function updated(app: App) {
|
||||||
const properties: AppUpdatedEvent = {}
|
const properties: AppUpdatedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
version: app.version,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_UPDATED, properties)
|
await publishEvent(Event.APP_UPDATED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleted(app: App) {
|
export async function deleted(app: App) {
|
||||||
const properties: AppDeletedEvent = {}
|
const properties: AppDeletedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_DELETED, properties)
|
await publishEvent(Event.APP_DELETED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function published(app: App, timestamp?: string | number) {
|
export async function published(app: App, timestamp?: string | number) {
|
||||||
const properties: AppPublishedEvent = {}
|
const properties: AppPublishedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_PUBLISHED, properties, timestamp)
|
await publishEvent(Event.APP_PUBLISHED, properties, timestamp)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function unpublished(app: App) {
|
export async function unpublished(app: App) {
|
||||||
const properties: AppUnpublishedEvent = {}
|
const properties: AppUnpublishedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_UNPUBLISHED, properties)
|
await publishEvent(Event.APP_UNPUBLISHED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function fileImported(app: App) {
|
export async function fileImported(app: App) {
|
||||||
const properties: AppFileImportedEvent = {}
|
const properties: AppFileImportedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_FILE_IMPORTED, properties)
|
await publishEvent(Event.APP_FILE_IMPORTED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function templateImported(templateKey: string) {
|
export async function templateImported(app: App, templateKey: string) {
|
||||||
const properties: AppTemplateImportedEvent = {
|
const properties: AppTemplateImportedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
templateKey,
|
templateKey,
|
||||||
}
|
}
|
||||||
await publishEvent(Event.APP_TEMPLATE_IMPORTED, properties)
|
await publishEvent(Event.APP_TEMPLATE_IMPORTED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function versionUpdated(app: App) {
|
export async function versionUpdated(
|
||||||
const properties: AppVersionUpdatedEvent = {}
|
app: App,
|
||||||
|
currentVersion: string,
|
||||||
|
updatedToVersion: string
|
||||||
|
) {
|
||||||
|
const properties: AppVersionUpdatedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
currentVersion,
|
||||||
|
updatedToVersion,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_VERSION_UPDATED, properties)
|
await publishEvent(Event.APP_VERSION_UPDATED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function versionReverted(app: App) {
|
export async function versionReverted(
|
||||||
const properties: AppVersionRevertedEvent = {}
|
app: App,
|
||||||
|
currentVersion: string,
|
||||||
|
revertedToVersion: string
|
||||||
|
) {
|
||||||
|
const properties: AppVersionRevertedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
currentVersion,
|
||||||
|
revertedToVersion,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_VERSION_REVERTED, properties)
|
await publishEvent(Event.APP_VERSION_REVERTED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function reverted(app: App) {
|
export async function reverted(app: App) {
|
||||||
const properties: AppRevertedEvent = {}
|
const properties: AppRevertedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_REVERTED, properties)
|
await publishEvent(Event.APP_REVERTED, properties)
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function exported(app: App) {
|
export async function exported(app: App) {
|
||||||
const properties: AppExportedEvent = {}
|
const properties: AppExportedEvent = {
|
||||||
|
appId: app.appId,
|
||||||
|
}
|
||||||
await publishEvent(Event.APP_EXPORTED, properties)
|
await publishEvent(Event.APP_EXPORTED, properties)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,8 +3,8 @@ import errors from "./errors"
|
||||||
import * as events from "./events"
|
import * as events from "./events"
|
||||||
import * as migrations from "./migrations"
|
import * as migrations from "./migrations"
|
||||||
import * as users from "./users"
|
import * as users from "./users"
|
||||||
|
import * as accounts from "./cloud/accounts"
|
||||||
import env from "./environment"
|
import env from "./environment"
|
||||||
import accounts from "./cloud/accounts"
|
|
||||||
import tenancy from "./tenancy"
|
import tenancy from "./tenancy"
|
||||||
import featureFlags from "./featureFlags"
|
import featureFlags from "./featureFlags"
|
||||||
import sessions from "./security/sessions"
|
import sessions from "./security/sessions"
|
||||||
|
|
|
@ -298,19 +298,19 @@ const creationEvents = async (request: any, app: App) => {
|
||||||
const body = request.body
|
const body = request.body
|
||||||
if (body.useTemplate === "true") {
|
if (body.useTemplate === "true") {
|
||||||
// from template
|
// from template
|
||||||
if (body.templateKey) {
|
if (body.templateKey && body.templateKey !== "undefined") {
|
||||||
creationFns.push(app => events.app.templateImported(body.templateKey))
|
creationFns.push(a => events.app.templateImported(a, body.templateKey))
|
||||||
}
|
}
|
||||||
// from file
|
// from file
|
||||||
else if (request.files?.templateFile) {
|
else if (request.files?.templateFile) {
|
||||||
creationFns.push(events.app.fileImported)
|
creationFns.push(a => events.app.fileImported(a))
|
||||||
}
|
}
|
||||||
// unknown
|
// unknown
|
||||||
else {
|
else {
|
||||||
console.error("Could not determine template creation event")
|
console.error("Could not determine template creation event")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
creationFns.push(events.app.created)
|
creationFns.push(a => events.app.created(a))
|
||||||
|
|
||||||
for (let fn of creationFns) {
|
for (let fn of creationFns) {
|
||||||
await fn(app)
|
await fn(app)
|
||||||
|
@ -381,12 +381,13 @@ export const updateClient = async (ctx: any) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update versions in app package
|
// Update versions in app package
|
||||||
|
const updatedToVersion = packageJson.version
|
||||||
const appPackageUpdates = {
|
const appPackageUpdates = {
|
||||||
version: packageJson.version,
|
version: updatedToVersion,
|
||||||
revertableVersion: currentVersion,
|
revertableVersion: currentVersion,
|
||||||
}
|
}
|
||||||
const app = await updateAppPackage(appPackageUpdates, ctx.params.appId)
|
const app = await updateAppPackage(appPackageUpdates, ctx.params.appId)
|
||||||
await events.app.versionUpdated(app)
|
await events.app.versionUpdated(app, currentVersion, updatedToVersion)
|
||||||
ctx.status = 200
|
ctx.status = 200
|
||||||
ctx.body = app
|
ctx.body = app
|
||||||
}
|
}
|
||||||
|
@ -405,12 +406,14 @@ export const revertClient = async (ctx: any) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update versions in app package
|
// Update versions in app package
|
||||||
|
const currentVersion = application.version
|
||||||
|
const revertedToVersion = application.revertableVersion
|
||||||
const appPackageUpdates = {
|
const appPackageUpdates = {
|
||||||
version: application.revertableVersion,
|
version: revertedToVersion,
|
||||||
revertableVersion: null,
|
revertableVersion: null,
|
||||||
}
|
}
|
||||||
const app = await updateAppPackage(appPackageUpdates, ctx.params.appId)
|
const app = await updateAppPackage(appPackageUpdates, ctx.params.appId)
|
||||||
await events.app.versionReverted(app)
|
await events.app.versionReverted(app, currentVersion, revertedToVersion)
|
||||||
ctx.status = 200
|
ctx.status = 200
|
||||||
ctx.body = app
|
ctx.body = app
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,6 @@
|
||||||
const { streamBackup } = require("../../utilities/fileSystem")
|
const { streamBackup } = require("../../utilities/fileSystem")
|
||||||
const { events } = require("@budibase/backend-core")
|
const { events, context } = require("@budibase/backend-core")
|
||||||
|
const { DocumentTypes } = require("../../db/utils")
|
||||||
|
|
||||||
exports.exportAppDump = async function (ctx) {
|
exports.exportAppDump = async function (ctx) {
|
||||||
const { appId } = ctx.query
|
const { appId } = ctx.query
|
||||||
|
@ -7,5 +8,10 @@ exports.exportAppDump = async function (ctx) {
|
||||||
const backupIdentifier = `${appName}-export-${new Date().getTime()}.txt`
|
const backupIdentifier = `${appName}-export-${new Date().getTime()}.txt`
|
||||||
ctx.attachment(backupIdentifier)
|
ctx.attachment(backupIdentifier)
|
||||||
ctx.body = await streamBackup(appId)
|
ctx.body = await streamBackup(appId)
|
||||||
await events.app.exported()
|
|
||||||
|
await context.doInAppContext(appId, async () => {
|
||||||
|
const appDb = context.getAppDB()
|
||||||
|
const app = await appDb.get(DocumentTypes.APP_METADATA)
|
||||||
|
await events.app.exported(app)
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
import * as users from "./global/users"
|
import * as users from "./global/users"
|
||||||
import * as rows from "./global/rows"
|
import * as rows from "./global/rows"
|
||||||
import * as configs from "./global/configs"
|
import * as configs from "./global/configs"
|
||||||
import { tenancy, events, migrations } from "@budibase/backend-core"
|
import { tenancy, events, migrations, accounts } from "@budibase/backend-core"
|
||||||
|
import { CloudAccount } from "@budibase/types"
|
||||||
|
import env from "../../../environment"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Date:
|
* Date:
|
||||||
|
@ -14,12 +16,20 @@ import { tenancy, events, migrations } from "@budibase/backend-core"
|
||||||
export const run = async (db: any) => {
|
export const run = async (db: any) => {
|
||||||
const tenantId = tenancy.getTenantId()
|
const tenantId = tenancy.getTenantId()
|
||||||
const installTimestamp = (await getInstallTimestamp(db)) as number
|
const installTimestamp = (await getInstallTimestamp(db)) as number
|
||||||
|
let account: CloudAccount | undefined
|
||||||
|
if (!env.SELF_HOSTED && !env.DISABLE_ACCOUNT_PORTAL) {
|
||||||
|
account = await accounts.getAccountByTenantId(tenantId)
|
||||||
|
}
|
||||||
|
|
||||||
await events.identification.identifyTenant(tenantId, installTimestamp)
|
await events.identification.identifyTenant(
|
||||||
|
tenantId,
|
||||||
|
account,
|
||||||
|
installTimestamp
|
||||||
|
)
|
||||||
await configs.backfill(db, installTimestamp)
|
await configs.backfill(db, installTimestamp)
|
||||||
|
|
||||||
// users and rows provide their own timestamp
|
// users and rows provide their own timestamp
|
||||||
await users.backfill(db)
|
await users.backfill(db, account)
|
||||||
await rows.backfill()
|
await rows.backfill()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { events, db as dbUtils } from "@budibase/backend-core"
|
import { events, db as dbUtils } from "@budibase/backend-core"
|
||||||
import { User } from "@budibase/types"
|
import { User, CloudAccount } from "@budibase/types"
|
||||||
|
|
||||||
// manually define user doc params - normally server doesn't read users from the db
|
// manually define user doc params - normally server doesn't read users from the db
|
||||||
const getUserParams = (props: any) => {
|
const getUserParams = (props: any) => {
|
||||||
|
@ -15,12 +15,15 @@ export const getUsers = async (globalDb: any): Promise<User[]> => {
|
||||||
return response.rows.map((row: any) => row.doc)
|
return response.rows.map((row: any) => row.doc)
|
||||||
}
|
}
|
||||||
|
|
||||||
export const backfill = async (globalDb: any) => {
|
export const backfill = async (
|
||||||
|
globalDb: any,
|
||||||
|
account: CloudAccount | undefined
|
||||||
|
) => {
|
||||||
const users = await getUsers(globalDb)
|
const users = await getUsers(globalDb)
|
||||||
|
|
||||||
for (const user of users) {
|
for (const user of users) {
|
||||||
const timestamp = user.createdAt as number
|
const timestamp = user.createdAt as number
|
||||||
await events.identification.identifyUser(user, timestamp)
|
await events.identification.identifyUser(user, account, timestamp)
|
||||||
await events.user.created(user, timestamp)
|
await events.user.created(user, timestamp)
|
||||||
|
|
||||||
if (user.admin?.global) {
|
if (user.admin?.global) {
|
||||||
|
|
|
@ -11,6 +11,7 @@ export interface App extends Document {
|
||||||
instance: AppInstance
|
instance: AppInstance
|
||||||
tenantId: string
|
tenantId: string
|
||||||
status: string
|
status: string
|
||||||
|
revertableVersion?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AppInstance {
|
export interface AppInstance {
|
||||||
|
|
|
@ -0,0 +1,13 @@
|
||||||
|
export interface AccountCreatedEvent {
|
||||||
|
tenantId: string
|
||||||
|
registrationStep?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AccountDeletedEvent {
|
||||||
|
tenantId: string
|
||||||
|
registrationStep?: string
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface AccountVerifiedEvent {
|
||||||
|
tenantId: string
|
||||||
|
}
|
|
@ -1,21 +1,50 @@
|
||||||
export interface AppCreatedEvent {}
|
export interface AppCreatedEvent {
|
||||||
|
appId: string
|
||||||
|
version: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppUpdatedEvent {}
|
export interface AppUpdatedEvent {
|
||||||
|
appId: string
|
||||||
|
version: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppDeletedEvent {}
|
export interface AppDeletedEvent {
|
||||||
|
appId: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppPublishedEvent {}
|
export interface AppPublishedEvent {
|
||||||
|
appId: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppUnpublishedEvent {}
|
export interface AppUnpublishedEvent {
|
||||||
|
appId: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppFileImportedEvent {}
|
export interface AppFileImportedEvent {
|
||||||
|
appId: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppTemplateImportedEvent {}
|
export interface AppTemplateImportedEvent {
|
||||||
|
appId: string
|
||||||
|
templateKey: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppVersionUpdatedEvent {}
|
export interface AppVersionUpdatedEvent {
|
||||||
|
appId: string
|
||||||
|
currentVersion: string
|
||||||
|
updatedToVersion: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppVersionRevertedEvent {}
|
export interface AppVersionRevertedEvent {
|
||||||
|
appId: string
|
||||||
|
currentVersion: string
|
||||||
|
revertedToVersion: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppRevertedEvent {}
|
export interface AppRevertedEvent {
|
||||||
|
appId: string
|
||||||
|
}
|
||||||
|
|
||||||
export interface AppExportedEvent {}
|
export interface AppExportedEvent {
|
||||||
|
appId: string
|
||||||
|
}
|
||||||
|
|
|
@ -13,11 +13,15 @@ export interface Identity {
|
||||||
|
|
||||||
export interface TenantIdentity extends Identity {
|
export interface TenantIdentity extends Identity {
|
||||||
hosting: Hosting
|
hosting: Hosting
|
||||||
|
profession?: string
|
||||||
|
companySize?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface UserIdentity extends TenantIdentity {
|
export interface UserIdentity extends TenantIdentity {
|
||||||
hosting: Hosting
|
hosting: Hosting
|
||||||
type: IdentityType
|
type: IdentityType
|
||||||
|
verified: boolean
|
||||||
|
accountHolder: boolean
|
||||||
providerType?: string
|
providerType?: string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -25,9 +29,3 @@ export interface BudibaseIdentity extends UserIdentity {
|
||||||
builder?: boolean
|
builder?: boolean
|
||||||
admin?: boolean
|
admin?: boolean
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AccountIdentity extends UserIdentity {
|
|
||||||
verified: boolean
|
|
||||||
profession: string | undefined
|
|
||||||
companySize: string | undefined
|
|
||||||
}
|
|
||||||
|
|
|
@ -16,3 +16,4 @@ export * from "./table"
|
||||||
export * from "./user"
|
export * from "./user"
|
||||||
export * from "./view"
|
export * from "./view"
|
||||||
export * from "./identification"
|
export * from "./identification"
|
||||||
|
export * from "./account"
|
||||||
|
|
|
@ -2,14 +2,13 @@ import { EmailTemplatePurpose } from "../../../constants"
|
||||||
import { checkInviteCode } from "../../../utilities/redis"
|
import { checkInviteCode } from "../../../utilities/redis"
|
||||||
import { sendEmail } from "../../../utilities/email"
|
import { sendEmail } from "../../../utilities/email"
|
||||||
import { users } from "../../../sdk"
|
import { users } from "../../../sdk"
|
||||||
import { User } from "@budibase/types"
|
import env from "../../../environment"
|
||||||
import { events } from "@budibase/backend-core"
|
import { User, CloudAccount } from "@budibase/types"
|
||||||
import { getGlobalDB } from "@budibase/backend-core/dist/src/context"
|
import { events, accounts, tenancy } from "@budibase/backend-core"
|
||||||
|
|
||||||
const {
|
const {
|
||||||
errors,
|
errors,
|
||||||
users: usersCore,
|
users: usersCore,
|
||||||
tenancy,
|
|
||||||
db: dbUtils,
|
db: dbUtils,
|
||||||
} = require("@budibase/backend-core")
|
} = require("@budibase/backend-core")
|
||||||
|
|
||||||
|
@ -66,7 +65,11 @@ export const adminUser = async (ctx: any) => {
|
||||||
ctx.body = await tenancy.doInTenant(tenantId, async () => {
|
ctx.body = await tenancy.doInTenant(tenantId, async () => {
|
||||||
return users.save(user, hashPassword, requirePassword)
|
return users.save(user, hashPassword, requirePassword)
|
||||||
})
|
})
|
||||||
await events.identification.identifyTenant(tenantId)
|
let account: CloudAccount | undefined
|
||||||
|
if (!env.SELF_HOSTED && !env.DISABLE_ACCOUNT_PORTAL) {
|
||||||
|
account = await accounts.getAccountByTenantId(tenantId)
|
||||||
|
}
|
||||||
|
await events.identification.identifyTenant(tenantId, account)
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
ctx.throw(err.status || 400, err)
|
ctx.throw(err.status || 400, err)
|
||||||
}
|
}
|
||||||
|
@ -141,7 +144,7 @@ export const inviteAccept = async (ctx: any) => {
|
||||||
email,
|
email,
|
||||||
...info,
|
...info,
|
||||||
})
|
})
|
||||||
const db = getGlobalDB()
|
const db = tenancy.getGlobalDB()
|
||||||
const user = await db.get(saved._id)
|
const user = await db.get(saved._id)
|
||||||
await events.user.inviteAccepted(user)
|
await events.user.inviteAccepted(user)
|
||||||
return saved
|
return saved
|
||||||
|
|
|
@ -2,11 +2,10 @@ import env from "../../environment"
|
||||||
import { quotas } from "@budibase/pro"
|
import { quotas } from "@budibase/pro"
|
||||||
import * as apps from "../../utilities/appService"
|
import * as apps from "../../utilities/appService"
|
||||||
import * as eventHelpers from "./events"
|
import * as eventHelpers from "./events"
|
||||||
import { User } from "@budibase/types"
|
import { User, CloudAccount } from "@budibase/types"
|
||||||
|
|
||||||
const {
|
const {
|
||||||
tenancy,
|
tenancy,
|
||||||
accounts,
|
|
||||||
utils,
|
utils,
|
||||||
db: dbUtils,
|
db: dbUtils,
|
||||||
constants,
|
constants,
|
||||||
|
@ -17,7 +16,7 @@ const {
|
||||||
HTTPError,
|
HTTPError,
|
||||||
} = require("@budibase/backend-core")
|
} = require("@budibase/backend-core")
|
||||||
|
|
||||||
import { events } from "@budibase/backend-core"
|
import { events, accounts } from "@budibase/backend-core"
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Retrieves all users from the current tenancy.
|
* Retrieves all users from the current tenancy.
|
||||||
|
@ -133,7 +132,14 @@ export const save = async (
|
||||||
user._rev = response.rev
|
user._rev = response.rev
|
||||||
|
|
||||||
await eventHelpers.handleSaveEvents(user, dbUser)
|
await eventHelpers.handleSaveEvents(user, dbUser)
|
||||||
await events.identification.identifyUser(user)
|
|
||||||
|
// identify
|
||||||
|
let tenantAccount: CloudAccount | undefined
|
||||||
|
if (!env.SELF_HOSTED && !env.DISABLE_ACCOUNT_PORTAL) {
|
||||||
|
tenantAccount = await accounts.getAccountByTenantId(tenantId)
|
||||||
|
}
|
||||||
|
await events.identification.identifyUser(user, tenantAccount)
|
||||||
|
|
||||||
await tenancy.tryAddTenant(tenantId, _id, email)
|
await tenancy.tryAddTenant(tenantId, _id, email)
|
||||||
await cache.user.invalidateUser(response.id)
|
await cache.user.invalidateUser(response.id)
|
||||||
// let server know to sync user
|
// let server know to sync user
|
||||||
|
|
Loading…
Reference in New Issue