2022-09-13 13:22:03 +02:00
|
|
|
import { DocumentType, ViewName, DeprecatedViews, SEPARATOR } from "./utils"
|
|
|
|
import { getGlobalDB } from "../tenancy"
|
|
|
|
import { StaticDatabases } from "./constants"
|
|
|
|
import { doWithDB } from "./"
|
2021-04-20 18:17:44 +02:00
|
|
|
|
2022-07-13 22:50:19 +02:00
|
|
|
const DESIGN_DB = "_design/database"
|
|
|
|
|
2021-04-20 18:17:44 +02:00
|
|
|
function DesignDoc() {
|
|
|
|
return {
|
2022-07-13 22:50:19 +02:00
|
|
|
_id: DESIGN_DB,
|
2021-04-20 18:17:44 +02:00
|
|
|
// view collation information, read before writing any complex views:
|
|
|
|
// https://docs.couchdb.org/en/master/ddocs/views/collation.html#collation-specification
|
|
|
|
views: {},
|
|
|
|
}
|
|
|
|
}
|
2021-04-19 18:31:47 +02:00
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
interface DesignDocument {
|
|
|
|
views: any
|
|
|
|
}
|
|
|
|
|
|
|
|
async function removeDeprecated(db: PouchDB.Database, viewName: ViewName) {
|
|
|
|
// @ts-ignore
|
2022-07-13 22:50:19 +02:00
|
|
|
if (!DeprecatedViews[viewName]) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
try {
|
2022-09-13 13:22:03 +02:00
|
|
|
const designDoc = await db.get<DesignDocument>(DESIGN_DB)
|
|
|
|
// @ts-ignore
|
2022-07-13 22:50:19 +02:00
|
|
|
for (let deprecatedNames of DeprecatedViews[viewName]) {
|
|
|
|
delete designDoc.views[deprecatedNames]
|
|
|
|
}
|
|
|
|
await db.put(designDoc)
|
|
|
|
} catch (err) {
|
|
|
|
// doesn't exist, ignore
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const createNewUserEmailView = async () => {
|
2022-02-11 23:24:48 +01:00
|
|
|
const db = getGlobalDB()
|
2021-04-20 18:17:44 +02:00
|
|
|
let designDoc
|
|
|
|
try {
|
2022-07-13 22:50:19 +02:00
|
|
|
designDoc = await db.get(DESIGN_DB)
|
2021-04-20 18:17:44 +02:00
|
|
|
} catch (err) {
|
|
|
|
// no design doc, make one
|
|
|
|
designDoc = DesignDoc()
|
|
|
|
}
|
2021-04-19 18:31:47 +02:00
|
|
|
const view = {
|
|
|
|
// if using variables in a map function need to inject them before use
|
|
|
|
map: `function(doc) {
|
2022-08-11 14:50:05 +02:00
|
|
|
if (doc._id.startsWith("${DocumentType.USER}${SEPARATOR}")) {
|
2021-10-26 13:40:30 +02:00
|
|
|
emit(doc.email.toLowerCase(), doc._id)
|
2021-04-19 18:31:47 +02:00
|
|
|
}
|
|
|
|
}`,
|
|
|
|
}
|
|
|
|
designDoc.views = {
|
|
|
|
...designDoc.views,
|
2022-08-11 14:50:05 +02:00
|
|
|
[ViewName.USER_BY_EMAIL]: view,
|
2021-04-19 18:31:47 +02:00
|
|
|
}
|
|
|
|
await db.put(designDoc)
|
|
|
|
}
|
2022-03-04 14:42:50 +01:00
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const createAccountEmailView = async () => {
|
|
|
|
await doWithDB(
|
|
|
|
StaticDatabases.PLATFORM_INFO.name,
|
|
|
|
async (db: PouchDB.Database) => {
|
|
|
|
let designDoc
|
|
|
|
try {
|
|
|
|
designDoc = await db.get<DesignDocument>(DESIGN_DB)
|
|
|
|
} catch (err) {
|
|
|
|
// no design doc, make one
|
|
|
|
designDoc = DesignDoc()
|
|
|
|
}
|
|
|
|
const view = {
|
|
|
|
// if using variables in a map function need to inject them before use
|
|
|
|
map: `function(doc) {
|
2022-08-25 20:41:47 +02:00
|
|
|
if (doc._id.startsWith("${DocumentType.ACCOUNT_METADATA}${SEPARATOR}")) {
|
|
|
|
emit(doc.email.toLowerCase(), doc._id)
|
2022-08-23 10:37:13 +02:00
|
|
|
}
|
|
|
|
}`,
|
2022-09-13 13:22:03 +02:00
|
|
|
}
|
|
|
|
designDoc.views = {
|
|
|
|
...designDoc.views,
|
|
|
|
[ViewName.ACCOUNT_BY_EMAIL]: view,
|
|
|
|
}
|
|
|
|
await db.put(designDoc)
|
2022-08-23 10:37:13 +02:00
|
|
|
}
|
2022-09-13 13:22:03 +02:00
|
|
|
)
|
2022-08-23 10:37:13 +02:00
|
|
|
}
|
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const createUserAppView = async () => {
|
|
|
|
const db = getGlobalDB() as PouchDB.Database
|
2022-07-06 17:09:05 +02:00
|
|
|
let designDoc
|
|
|
|
try {
|
2022-09-13 13:22:03 +02:00
|
|
|
designDoc = await db.get<DesignDocument>("_design/database")
|
2022-07-06 17:09:05 +02:00
|
|
|
} catch (err) {
|
|
|
|
// no design doc, make one
|
|
|
|
designDoc = DesignDoc()
|
|
|
|
}
|
|
|
|
const view = {
|
|
|
|
// if using variables in a map function need to inject them before use
|
|
|
|
map: `function(doc) {
|
2022-08-11 14:50:05 +02:00
|
|
|
if (doc._id.startsWith("${DocumentType.USER}${SEPARATOR}") && doc.roles) {
|
2022-07-06 17:09:05 +02:00
|
|
|
for (let prodAppId of Object.keys(doc.roles)) {
|
|
|
|
let emitted = prodAppId + "${SEPARATOR}" + doc._id
|
|
|
|
emit(emitted, null)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
}
|
|
|
|
designDoc.views = {
|
|
|
|
...designDoc.views,
|
2022-08-11 14:50:05 +02:00
|
|
|
[ViewName.USER_BY_APP]: view,
|
2022-07-06 17:09:05 +02:00
|
|
|
}
|
|
|
|
await db.put(designDoc)
|
|
|
|
}
|
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const createApiKeyView = async () => {
|
2022-02-11 23:24:48 +01:00
|
|
|
const db = getGlobalDB()
|
|
|
|
let designDoc
|
|
|
|
try {
|
|
|
|
designDoc = await db.get("_design/database")
|
|
|
|
} catch (err) {
|
|
|
|
designDoc = DesignDoc()
|
|
|
|
}
|
|
|
|
const view = {
|
|
|
|
map: `function(doc) {
|
2022-08-11 14:50:05 +02:00
|
|
|
if (doc._id.startsWith("${DocumentType.DEV_INFO}") && doc.apiKey) {
|
2022-02-11 23:24:48 +01:00
|
|
|
emit(doc.apiKey, doc.userId)
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
}
|
|
|
|
designDoc.views = {
|
|
|
|
...designDoc.views,
|
2022-08-11 14:50:05 +02:00
|
|
|
[ViewName.BY_API_KEY]: view,
|
2022-02-11 23:24:48 +01:00
|
|
|
}
|
|
|
|
await db.put(designDoc)
|
|
|
|
}
|
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const createUserBuildersView = async () => {
|
2022-03-22 01:23:22 +01:00
|
|
|
const db = getGlobalDB()
|
2022-03-04 14:42:50 +01:00
|
|
|
let designDoc
|
|
|
|
try {
|
|
|
|
designDoc = await db.get("_design/database")
|
|
|
|
} catch (err) {
|
|
|
|
// no design doc, make one
|
|
|
|
designDoc = DesignDoc()
|
|
|
|
}
|
|
|
|
const view = {
|
|
|
|
map: `function(doc) {
|
|
|
|
if (doc.builder && doc.builder.global === true) {
|
|
|
|
emit(doc._id, doc._id)
|
|
|
|
}
|
|
|
|
}`,
|
|
|
|
}
|
|
|
|
designDoc.views = {
|
|
|
|
...designDoc.views,
|
2022-08-11 14:50:05 +02:00
|
|
|
[ViewName.USER_BY_BUILDERS]: view,
|
2022-03-04 14:42:50 +01:00
|
|
|
}
|
|
|
|
await db.put(designDoc)
|
|
|
|
}
|
2022-03-21 18:13:16 +01:00
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const createPlatformUserView = async () => {
|
|
|
|
await doWithDB(
|
|
|
|
StaticDatabases.PLATFORM_INFO.name,
|
|
|
|
async (db: PouchDB.Database) => {
|
|
|
|
let designDoc
|
|
|
|
try {
|
|
|
|
designDoc = await db.get<DesignDocument>(DESIGN_DB)
|
|
|
|
} catch (err) {
|
|
|
|
// no design doc, make one
|
|
|
|
designDoc = DesignDoc()
|
|
|
|
}
|
|
|
|
const view = {
|
|
|
|
// if using variables in a map function need to inject them before use
|
|
|
|
map: `function(doc) {
|
2022-09-02 17:07:30 +02:00
|
|
|
if (doc.tenantId) {
|
|
|
|
emit(doc._id.toLowerCase(), doc._id)
|
|
|
|
}
|
2022-09-02 16:42:20 +02:00
|
|
|
}`,
|
2022-09-13 13:22:03 +02:00
|
|
|
}
|
|
|
|
designDoc.views = {
|
|
|
|
...designDoc.views,
|
|
|
|
[ViewName.PLATFORM_USERS_LOWERCASE]: view,
|
|
|
|
}
|
|
|
|
await db.put(designDoc)
|
2022-09-02 16:42:20 +02:00
|
|
|
}
|
2022-09-13 13:22:03 +02:00
|
|
|
)
|
2022-09-02 10:12:03 +02:00
|
|
|
}
|
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export interface QueryViewOptions {
|
|
|
|
arrayResponse?: boolean
|
|
|
|
}
|
|
|
|
|
|
|
|
export const queryView = async <T>(
|
|
|
|
viewName: ViewName,
|
|
|
|
params: PouchDB.Query.Options<T, T>,
|
|
|
|
db: PouchDB.Database,
|
|
|
|
CreateFuncByName: any,
|
|
|
|
opts?: QueryViewOptions
|
|
|
|
): Promise<T[] | T | undefined> => {
|
2022-02-11 23:24:48 +01:00
|
|
|
try {
|
2022-09-13 13:22:03 +02:00
|
|
|
let response = await db.query<T, T>(`database/${viewName}`, params)
|
|
|
|
const rows = response.rows
|
|
|
|
const docs = rows.map((resp: any) =>
|
2022-02-11 23:24:48 +01:00
|
|
|
params.include_docs ? resp.doc : resp.value
|
|
|
|
)
|
2022-09-13 13:22:03 +02:00
|
|
|
|
|
|
|
if (opts?.arrayResponse) {
|
|
|
|
return docs
|
2022-08-23 10:37:13 +02:00
|
|
|
} else {
|
2022-09-13 13:22:03 +02:00
|
|
|
return docs.length <= 1 ? docs[0] : docs
|
2022-08-23 10:37:13 +02:00
|
|
|
}
|
2022-09-13 13:22:03 +02:00
|
|
|
} catch (err: any) {
|
2022-02-11 23:24:48 +01:00
|
|
|
if (err != null && err.name === "not_found") {
|
|
|
|
const createFunc = CreateFuncByName[viewName]
|
2022-07-13 22:50:19 +02:00
|
|
|
await removeDeprecated(db, viewName)
|
2022-02-11 23:24:48 +01:00
|
|
|
await createFunc()
|
2022-08-25 20:41:47 +02:00
|
|
|
return exports.queryView(viewName, params, db, CreateFuncByName)
|
2022-02-11 23:24:48 +01:00
|
|
|
} else {
|
|
|
|
throw err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-08-23 10:37:13 +02:00
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const queryPlatformView = async <T>(
|
|
|
|
viewName: ViewName,
|
|
|
|
params: PouchDB.Query.Options<T, T>,
|
|
|
|
opts?: QueryViewOptions
|
|
|
|
): Promise<T[] | T | undefined> => {
|
2022-08-23 10:37:13 +02:00
|
|
|
const CreateFuncByName = {
|
|
|
|
[ViewName.ACCOUNT_BY_EMAIL]: exports.createAccountEmailView,
|
2022-09-02 16:42:20 +02:00
|
|
|
[ViewName.PLATFORM_USERS_LOWERCASE]: exports.createPlatformUserView,
|
2022-08-23 10:37:13 +02:00
|
|
|
}
|
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
return doWithDB(
|
|
|
|
StaticDatabases.PLATFORM_INFO.name,
|
|
|
|
async (db: PouchDB.Database) => {
|
|
|
|
return exports.queryView(viewName, params, db, CreateFuncByName, opts)
|
|
|
|
}
|
|
|
|
)
|
2022-08-23 10:37:13 +02:00
|
|
|
}
|
|
|
|
|
2022-09-13 13:22:03 +02:00
|
|
|
export const queryGlobalView = async <T>(
|
|
|
|
viewName: ViewName,
|
|
|
|
params: PouchDB.Query.Options<T, T>,
|
|
|
|
db?: PouchDB.Database,
|
|
|
|
opts?: QueryViewOptions
|
|
|
|
): Promise<T[] | T | undefined> => {
|
2022-08-23 10:37:13 +02:00
|
|
|
const CreateFuncByName = {
|
|
|
|
[ViewName.USER_BY_EMAIL]: exports.createNewUserEmailView,
|
|
|
|
[ViewName.BY_API_KEY]: exports.createApiKeyView,
|
|
|
|
[ViewName.USER_BY_BUILDERS]: exports.createUserBuildersView,
|
|
|
|
[ViewName.USER_BY_APP]: exports.createUserAppView,
|
|
|
|
}
|
|
|
|
// can pass DB in if working with something specific
|
|
|
|
if (!db) {
|
|
|
|
db = getGlobalDB()
|
|
|
|
}
|
|
|
|
return exports.queryView(viewName, params, db, CreateFuncByName)
|
|
|
|
}
|