Attempt to fix race condition when creating multiple auto ID columns in parallel.

This commit is contained in:
Sam Rose 2024-05-03 14:14:10 +01:00
parent f783cb12a5
commit af516427fc
No known key found for this signature in database
5 changed files with 54 additions and 41 deletions

View File

@ -1,20 +1,11 @@
import { getRowParams } from "../../../db/utils" import { getRowParams } from "../../../db/utils"
import { import {
outputProcessing, outputProcessing,
processAutoColumn,
processFormulas, processFormulas,
} from "../../../utilities/rowProcessor" } from "../../../utilities/rowProcessor"
import { context, locks } from "@budibase/backend-core" import { context } from "@budibase/backend-core"
import { import { Table, Row, FormulaType, FieldType } from "@budibase/types"
Table,
Row,
LockType,
LockName,
FormulaType,
FieldType,
} from "@budibase/types"
import * as linkRows from "../../../db/linkedRows" import * as linkRows from "../../../db/linkedRows"
import sdk from "../../../sdk"
import isEqual from "lodash/isEqual" import isEqual from "lodash/isEqual"
import { cloneDeep } from "lodash/fp" import { cloneDeep } from "lodash/fp"
@ -151,30 +142,7 @@ export async function finaliseRow(
// if another row has been written since processing this will // if another row has been written since processing this will
// handle the auto ID clash // handle the auto ID clash
if (oldTable && !isEqual(oldTable, table)) { if (oldTable && !isEqual(oldTable, table)) {
try { await db.put(table)
await db.put(table)
} catch (err: any) {
if (err.status === 409) {
// Some conflicts with the autocolumns occurred, we need to refetch the table and recalculate
await locks.doWithLock(
{
type: LockType.AUTO_EXTEND,
name: LockName.PROCESS_AUTO_COLUMNS,
resource: table._id,
},
async () => {
const latestTable = await sdk.tables.getTable(table._id!)
let response = processAutoColumn(null, latestTable, row, {
reprocessing: true,
})
await db.put(response.table)
row = response.row
}
)
} else {
throw err
}
}
} }
const response = await db.put(row) const response = await db.put(row)
// for response, calculate the formulas for the enriched row // for response, calculate the formulas for the enriched row

View File

@ -145,6 +145,7 @@ describe("sdk >> rows >> internal", () => {
lastID: 1, lastID: 1,
}, },
}, },
_rev: expect.stringMatching("2-.*"),
}, },
row: { row: {
...row, ...row,

View File

@ -1,6 +1,6 @@
import * as linkRows from "../../db/linkedRows" import * as linkRows from "../../db/linkedRows"
import { processFormulas, fixAutoColumnSubType } from "./utils" import { processFormulas, fixAutoColumnSubType } from "./utils"
import { objectStore, utils } from "@budibase/backend-core" import { context, objectStore, utils } from "@budibase/backend-core"
import { InternalTables } from "../../db/utils" import { InternalTables } from "../../db/utils"
import { TYPE_TRANSFORM_MAP } from "./map" import { TYPE_TRANSFORM_MAP } from "./map"
import { import {
@ -9,6 +9,7 @@ import {
Row, Row,
RowAttachment, RowAttachment,
Table, Table,
isAutoColumnField,
} from "@budibase/types" } from "@budibase/types"
import { cloneDeep } from "lodash/fp" import { cloneDeep } from "lodash/fp"
import { import {
@ -25,7 +26,44 @@ type AutoColumnProcessingOpts = {
noAutoRelationships?: boolean noAutoRelationships?: boolean
} }
const BASE_AUTO_ID = 1 // Returns the next auto ID for a column in a table. On success, the table will
// be updated which is why it gets returned. The nextID returned is guaranteed
// to be given only to you, and if you don't use it it's gone forever (a gap
// will be left in the auto ID sequence).
//
// This function can throw if it fails to generate an auto ID after so many
// attempts.
async function getNextAutoId(
table: Table,
column: string
): Promise<{ table: Table; nextID: number }> {
const db = context.getAppDB()
for (let attempt = 0; attempt < 5; attempt++) {
const schema = table.schema[column]
if (!isAutoColumnField(schema)) {
throw new Error(`Column ${column} is not an auto column`)
}
schema.lastID = (schema.lastID || 0) + 1
try {
const resp = await db.put(table)
table._rev = resp.rev
return { table, nextID: schema.lastID }
} catch (e: any) {
if (e.status !== 409) {
throw e
}
// We wait for a random amount of time before retrying. The randomness
// makes it less likely for multiple requests modifying this table to
// collide.
await new Promise(resolve =>
setTimeout(resolve, Math.random() * 1.2 ** attempt * 1000)
)
table = await db.get(table._id)
}
}
throw new Error("Failed to generate an auto ID")
}
/** /**
* This will update any auto columns that are found on the row/table with the correct information based on * This will update any auto columns that are found on the row/table with the correct information based on
@ -37,7 +75,7 @@ const BASE_AUTO_ID = 1
* @returns The updated row and table, the table may need to be updated * @returns The updated row and table, the table may need to be updated
* for automatic ID purposes. * for automatic ID purposes.
*/ */
export function processAutoColumn( export async function processAutoColumn(
userId: string | null | undefined, userId: string | null | undefined,
table: Table, table: Table,
row: Row, row: Row,
@ -79,8 +117,9 @@ export function processAutoColumn(
break break
case AutoFieldSubType.AUTO_ID: case AutoFieldSubType.AUTO_ID:
if (creating) { if (creating) {
schema.lastID = !schema.lastID ? BASE_AUTO_ID : schema.lastID + 1 const { table: newTable, nextID } = await getNextAutoId(table, key)
row[key] = schema.lastID table = newTable
row[key] = nextID
} }
break break
} }

View File

@ -219,3 +219,9 @@ export function isAttachmentField(
): field is AttachmentFieldMetadata { ): field is AttachmentFieldMetadata {
return field.type === FieldType.ATTACHMENTS return field.type === FieldType.ATTACHMENTS
} }
export function isAutoColumnField(
field: FieldSchema
): field is AutoColumnFieldMetadata {
return field.type === FieldType.AUTO
}

View File

@ -21,7 +21,6 @@ export enum LockName {
PERSIST_WRITETHROUGH = "persist_writethrough", PERSIST_WRITETHROUGH = "persist_writethrough",
QUOTA_USAGE_EVENT = "quota_usage_event", QUOTA_USAGE_EVENT = "quota_usage_event",
APP_MIGRATION = "app_migrations", APP_MIGRATION = "app_migrations",
PROCESS_AUTO_COLUMNS = "process_auto_columns",
PROCESS_USER_INVITE = "process_user_invite", PROCESS_USER_INVITE = "process_user_invite",
} }