2020-09-28 18:36:59 +02:00
|
|
|
const LinkController = require("./LinkController")
|
2020-10-16 13:13:27 +02:00
|
|
|
const {
|
|
|
|
IncludeDocs,
|
|
|
|
getLinkDocuments,
|
|
|
|
createLinkView,
|
|
|
|
getUniqueByProp,
|
2021-02-19 14:21:38 +01:00
|
|
|
getRelatedTableForField,
|
|
|
|
getLinkedTableIDs,
|
|
|
|
getLinkedTable,
|
2020-10-16 13:13:27 +02:00
|
|
|
} = require("./linkUtils")
|
2021-02-15 18:47:14 +01:00
|
|
|
const { flatten } = require("lodash")
|
2021-02-17 18:09:59 +01:00
|
|
|
const CouchDB = require("../../db")
|
2021-04-29 20:06:58 +02:00
|
|
|
const { FieldTypes } = require("../../constants")
|
2021-06-08 17:11:46 +02:00
|
|
|
const { getMultiIDParams, USER_METDATA_PREFIX } = require("../../db/utils")
|
2021-06-08 17:06:30 +02:00
|
|
|
const { partition } = require("lodash")
|
2021-09-02 18:13:00 +02:00
|
|
|
const { getGlobalUsersFromMetadata } = require("../../utilities/global")
|
2021-09-02 19:33:41 +02:00
|
|
|
const { processFormulas } = require("../../utilities/rowProcessor/utils")
|
2020-09-28 18:36:59 +02:00
|
|
|
|
|
|
|
/**
|
2020-10-09 20:10:28 +02:00
|
|
|
* This functionality makes sure that when rows with links are created, updated or deleted they are processed
|
2020-09-28 18:36:59 +02:00
|
|
|
* correctly - making sure that no stale links are left around and that all links have been made successfully.
|
|
|
|
*/
|
|
|
|
|
|
|
|
const EventType = {
|
2020-10-09 20:10:28 +02:00
|
|
|
ROW_SAVE: "row:save",
|
|
|
|
ROW_UPDATE: "row:update",
|
|
|
|
ROW_DELETE: "row:delete",
|
2020-10-09 19:49:23 +02:00
|
|
|
TABLE_SAVE: "table:save",
|
|
|
|
TABLE_UPDATED: "table:updated",
|
|
|
|
TABLE_DELETE: "table:delete",
|
2020-09-28 18:36:59 +02:00
|
|
|
}
|
|
|
|
|
2020-09-29 17:40:59 +02:00
|
|
|
exports.EventType = EventType
|
2021-03-26 14:46:20 +01:00
|
|
|
// re-export search here for ease of use
|
2020-09-30 18:52:57 +02:00
|
|
|
exports.IncludeDocs = IncludeDocs
|
|
|
|
exports.getLinkDocuments = getLinkDocuments
|
|
|
|
exports.createLinkView = createLinkView
|
2020-09-29 12:55:18 +02:00
|
|
|
|
2021-09-13 17:28:52 +02:00
|
|
|
function clearRelationshipFields(table, rows) {
|
|
|
|
for (let [key, field] of Object.entries(table.schema)) {
|
|
|
|
if (field.type === FieldTypes.LINK) {
|
|
|
|
rows = rows.map(row => {
|
|
|
|
delete row[key]
|
|
|
|
return row
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return rows
|
|
|
|
}
|
|
|
|
|
2021-02-17 18:09:59 +01:00
|
|
|
async function getLinksForRows(appId, rows) {
|
2021-05-04 12:32:22 +02:00
|
|
|
const tableIds = [...new Set(rows.map(el => el.tableId))]
|
2021-02-17 18:09:59 +01:00
|
|
|
// start by getting all the link values for performance reasons
|
2021-02-17 19:04:21 +01:00
|
|
|
const responses = flatten(
|
2021-02-17 18:09:59 +01:00
|
|
|
await Promise.all(
|
2021-05-04 12:32:22 +02:00
|
|
|
tableIds.map(tableId =>
|
2021-02-17 18:09:59 +01:00
|
|
|
getLinkDocuments({
|
|
|
|
appId,
|
|
|
|
tableId: tableId,
|
|
|
|
includeDocs: IncludeDocs.EXCLUDE,
|
|
|
|
})
|
|
|
|
)
|
|
|
|
)
|
|
|
|
)
|
|
|
|
// have to get unique as the previous table query can
|
|
|
|
// return duplicates, could be querying for both tables in a relation
|
|
|
|
return getUniqueByProp(
|
|
|
|
responses
|
|
|
|
// create a unique ID which we can use for getting only unique ones
|
2021-05-04 12:32:22 +02:00
|
|
|
.map(el => ({ ...el, unique: el.id + el.thisId + el.fieldName })),
|
2021-02-17 18:09:59 +01:00
|
|
|
"unique"
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-08-05 10:59:08 +02:00
|
|
|
async function getFullLinkedDocs(ctx, appId, links) {
|
2021-06-08 17:06:30 +02:00
|
|
|
// create DBs
|
|
|
|
const db = new CouchDB(appId)
|
|
|
|
const linkedRowIds = links.map(link => link.id)
|
2021-09-21 16:59:50 +02:00
|
|
|
const uniqueRowIds = [...new Set(linkedRowIds)]
|
|
|
|
let dbRows = (await db.allDocs(getMultiIDParams(uniqueRowIds))).rows.map(
|
2021-06-08 17:06:30 +02:00
|
|
|
row => row.doc
|
|
|
|
)
|
2021-09-21 16:59:50 +02:00
|
|
|
// convert the unique db rows back to a full list of linked rows
|
2021-10-21 18:23:10 +02:00
|
|
|
const linked = linkedRowIds
|
|
|
|
.map(id => dbRows.find(row => row && row._id === id))
|
|
|
|
.filter(row => row != null)
|
2021-06-08 17:06:30 +02:00
|
|
|
// need to handle users as specific cases
|
2021-06-08 17:11:46 +02:00
|
|
|
let [users, other] = partition(linked, linkRow =>
|
|
|
|
linkRow._id.startsWith(USER_METDATA_PREFIX)
|
|
|
|
)
|
2021-09-02 18:13:00 +02:00
|
|
|
users = await getGlobalUsersFromMetadata(appId, users)
|
2021-06-08 17:11:46 +02:00
|
|
|
return [...other, ...users]
|
2021-06-08 17:06:30 +02:00
|
|
|
}
|
|
|
|
|
2020-09-29 12:55:18 +02:00
|
|
|
/**
|
2020-10-09 20:10:28 +02:00
|
|
|
* Update link documents for a row or table - this is to be called by the API controller when a change is occurring.
|
2021-06-25 18:14:23 +02:00
|
|
|
* @param {string} args.eventType states what type of change which is occurring, means this can be expanded upon in the
|
2020-09-29 12:55:18 +02:00
|
|
|
* future quite easily (all updates go through one function).
|
2021-06-25 18:14:23 +02:00
|
|
|
* @param {string} args.appId The ID of the instance in which the change is occurring.
|
|
|
|
* @param {string} args.tableId The ID of the of the table which is being changed.
|
|
|
|
* @param {object|null} args.row The row which is changing, e.g. created, updated or deleted.
|
|
|
|
* @param {object|null} args.table If the table has already been retrieved this can be used to reduce database gets.
|
|
|
|
* @param {object|null} args.oldTable If the table is being updated then the old table can be provided for differencing.
|
2020-10-09 20:10:28 +02:00
|
|
|
* @returns {Promise<object>} When the update is complete this will respond successfully. Returns the row for
|
|
|
|
* row operations and the table for table operations.
|
2020-09-29 12:55:18 +02:00
|
|
|
*/
|
2021-06-25 18:14:23 +02:00
|
|
|
exports.updateLinks = async function (args) {
|
2021-06-25 18:21:36 +02:00
|
|
|
const { eventType, appId, row, tableId, table, oldTable } = args
|
2020-10-12 18:45:11 +02:00
|
|
|
const baseReturnObj = row == null ? table : row
|
2020-10-29 11:28:27 +01:00
|
|
|
if (appId == null) {
|
2020-10-01 12:33:37 +02:00
|
|
|
throw "Cannot operate without an instance ID."
|
|
|
|
}
|
2020-10-09 19:49:23 +02:00
|
|
|
// make sure table ID is set
|
|
|
|
if (tableId == null && table != null) {
|
2021-06-25 18:14:23 +02:00
|
|
|
args.tableId = table._id
|
2020-09-28 18:36:59 +02:00
|
|
|
}
|
2021-06-25 18:14:23 +02:00
|
|
|
let linkController = new LinkController(args)
|
2020-10-12 18:45:11 +02:00
|
|
|
try {
|
|
|
|
if (
|
2021-09-21 16:59:50 +02:00
|
|
|
!(await linkController.doesTableHaveLinkedFields(table)) &&
|
2020-10-12 18:45:11 +02:00
|
|
|
(oldTable == null ||
|
|
|
|
!(await linkController.doesTableHaveLinkedFields(oldTable)))
|
|
|
|
) {
|
|
|
|
return baseReturnObj
|
|
|
|
}
|
|
|
|
} catch (err) {
|
|
|
|
return baseReturnObj
|
2020-09-29 12:55:18 +02:00
|
|
|
}
|
|
|
|
switch (eventType) {
|
2020-10-09 20:10:28 +02:00
|
|
|
case EventType.ROW_SAVE:
|
|
|
|
case EventType.ROW_UPDATE:
|
|
|
|
return await linkController.rowSaved()
|
|
|
|
case EventType.ROW_DELETE:
|
|
|
|
return await linkController.rowDeleted()
|
2020-10-09 19:49:23 +02:00
|
|
|
case EventType.TABLE_SAVE:
|
|
|
|
return await linkController.tableSaved()
|
|
|
|
case EventType.TABLE_UPDATED:
|
|
|
|
return await linkController.tableUpdated()
|
|
|
|
case EventType.TABLE_DELETE:
|
|
|
|
return await linkController.tableDeleted()
|
2020-09-29 12:55:18 +02:00
|
|
|
default:
|
2020-10-09 20:10:28 +02:00
|
|
|
throw "Type of event is not known, linked row handler requires update."
|
2020-09-29 12:55:18 +02:00
|
|
|
}
|
|
|
|
}
|
2020-09-28 18:36:59 +02:00
|
|
|
|
2021-02-17 19:04:21 +01:00
|
|
|
/**
|
2021-04-29 20:06:58 +02:00
|
|
|
* Given a table and a list of rows this will retrieve all of the attached docs and enrich them into the row.
|
|
|
|
* This is required for formula fields, this may only be utilised internally (for now).
|
2021-08-05 10:59:08 +02:00
|
|
|
* @param {object} ctx The request which is looking for rows.
|
2021-02-17 19:04:21 +01:00
|
|
|
* @param {object} table The table from which the rows originated.
|
2021-04-29 20:06:58 +02:00
|
|
|
* @param {array<object>} rows The rows which are to be enriched.
|
|
|
|
* @return {Promise<*>} returns the rows with all of the enriched relationships on it.
|
2021-02-17 19:04:21 +01:00
|
|
|
*/
|
2021-08-05 10:59:08 +02:00
|
|
|
exports.attachFullLinkedDocs = async (ctx, table, rows) => {
|
|
|
|
const appId = ctx.appId
|
2021-02-17 19:04:21 +01:00
|
|
|
const linkedTableIds = getLinkedTableIDs(table)
|
|
|
|
if (linkedTableIds.length === 0) {
|
|
|
|
return rows
|
|
|
|
}
|
2021-06-08 17:06:30 +02:00
|
|
|
// create DBs
|
2021-02-17 18:09:59 +01:00
|
|
|
const db = new CouchDB(appId)
|
2021-06-08 17:06:30 +02:00
|
|
|
// get all the links
|
2021-05-04 12:32:22 +02:00
|
|
|
const links = (await getLinksForRows(appId, rows)).filter(link =>
|
|
|
|
rows.some(row => row._id === link.thisId)
|
2021-02-17 18:09:59 +01:00
|
|
|
)
|
2021-09-13 17:28:52 +02:00
|
|
|
// clear any existing links that could be dupe'd
|
|
|
|
rows = clearRelationshipFields(table, rows)
|
|
|
|
// now get the docs and combine into the rows
|
2021-08-05 10:59:08 +02:00
|
|
|
let linked = await getFullLinkedDocs(ctx, appId, links)
|
2021-02-19 14:21:38 +01:00
|
|
|
const linkedTables = []
|
2021-02-17 18:09:59 +01:00
|
|
|
for (let row of rows) {
|
2021-05-04 12:32:22 +02:00
|
|
|
for (let link of links.filter(link => link.thisId === row._id)) {
|
2021-02-19 14:21:38 +01:00
|
|
|
if (row[link.fieldName] == null) {
|
|
|
|
row[link.fieldName] = []
|
|
|
|
}
|
2021-05-04 12:32:22 +02:00
|
|
|
const linkedRow = linked.find(row => row._id === link.id)
|
2021-10-21 18:23:10 +02:00
|
|
|
if (linkedRow) {
|
|
|
|
const linkedTableId =
|
|
|
|
linkedRow.tableId || getRelatedTableForField(table, link.fieldName)
|
|
|
|
const linkedTable = await getLinkedTable(
|
|
|
|
db,
|
|
|
|
linkedTableId,
|
|
|
|
linkedTables
|
|
|
|
)
|
|
|
|
if (linkedTable) {
|
|
|
|
row[link.fieldName].push(processFormulas(linkedTable, linkedRow))
|
|
|
|
}
|
2021-02-19 14:21:38 +01:00
|
|
|
}
|
|
|
|
}
|
2021-02-17 18:09:59 +01:00
|
|
|
}
|
2021-02-19 11:32:24 +01:00
|
|
|
return rows
|
2021-02-17 18:09:59 +01:00
|
|
|
}
|
2021-04-29 20:06:58 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* This function will take the given enriched rows and squash the links to only contain the primary display field.
|
|
|
|
* @param {string} appId The app in which the tables/rows/links exist.
|
|
|
|
* @param {object} table The table from which the rows originated.
|
|
|
|
* @param {array<object>} enriched The pre-enriched rows (full docs) which are to be squashed.
|
|
|
|
* @returns {Promise<Array>} The rows after having their links squashed to only contain the ID and primary display.
|
|
|
|
*/
|
|
|
|
exports.squashLinksToPrimaryDisplay = async (appId, table, enriched) => {
|
|
|
|
const db = new CouchDB(appId)
|
|
|
|
// will populate this as we find them
|
2021-09-03 15:49:56 +02:00
|
|
|
const linkedTables = [table]
|
|
|
|
for (let row of enriched) {
|
|
|
|
// this only fetches the table if its not already in array
|
|
|
|
const rowTable = await getLinkedTable(db, row.tableId, linkedTables)
|
|
|
|
for (let [column, schema] of Object.entries(rowTable.schema)) {
|
|
|
|
if (schema.type !== FieldTypes.LINK || !Array.isArray(row[column])) {
|
2021-04-29 20:06:58 +02:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
const newLinks = []
|
|
|
|
for (let link of row[column]) {
|
|
|
|
const linkTblId = link.tableId || getRelatedTableForField(table, column)
|
|
|
|
const linkedTable = await getLinkedTable(db, linkTblId, linkedTables)
|
|
|
|
const obj = { _id: link._id }
|
|
|
|
if (link[linkedTable.primaryDisplay]) {
|
|
|
|
obj.primaryDisplay = link[linkedTable.primaryDisplay]
|
|
|
|
}
|
|
|
|
newLinks.push(obj)
|
|
|
|
}
|
|
|
|
row[column] = newLinks
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return enriched
|
|
|
|
}
|