mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
(core) Add PUT /records DocApi endpoint to AddOrUpdate records
Summary: As designed in https://grist.quip.com/fZSrAnJKgO5j/Add-or-Update-Records-API Current `POST /records` adds records, and `PATCH /records` updates them by row ID. This adds `PUT /records` to 'upsert' records, applying the AddOrUpdate user action. PUT was chosen because it's idempotent. Using a separate method (instead of inferring based on the request body) also cleanly separates validation, documentation, etc. The name `require` for the new property was suggested by Paul because `where` isn't very clear when adding records. Test Plan: New DocApi tests Reviewers: jarek Reviewed By: jarek Differential Revision: https://phab.getgrist.com/D3251
This commit is contained in:
parent
66eb0b91b8
commit
0de0cb0f4a
@ -357,21 +357,35 @@ function parseColValues<T extends ColValues | BulkColValues>(
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function parseUserAction(ua: UserAction, docData: DocData): UserAction {
|
export function parseUserAction(ua: UserAction, docData: DocData): UserAction {
|
||||||
const actionType = ua[0] as string;
|
switch (ua[0]) {
|
||||||
let parseBulk: boolean;
|
case 'AddRecord':
|
||||||
|
case 'UpdateRecord':
|
||||||
if (['AddRecord', 'UpdateRecord'].includes(actionType)) {
|
return _parseUserActionColValues(ua, docData, false);
|
||||||
parseBulk = false;
|
case 'BulkAddRecord':
|
||||||
} else if (['BulkAddRecord', 'BulkUpdateRecord', 'ReplaceTableData'].includes(actionType)) {
|
case 'BulkUpdateRecord':
|
||||||
parseBulk = true;
|
case 'ReplaceTableData':
|
||||||
} else {
|
return _parseUserActionColValues(ua, docData, true);
|
||||||
return ua;
|
case 'AddOrUpdateRecord':
|
||||||
|
// Parse `require` (2) and `col_values` (3). The action looks like:
|
||||||
|
// ['AddOrUpdateRecord', table_id, require, col_values, options]
|
||||||
|
// (`col_values` is called `fields` in the API)
|
||||||
|
ua = _parseUserActionColValues(ua, docData, false, 2);
|
||||||
|
ua = _parseUserActionColValues(ua, docData, false, 3);
|
||||||
|
return ua;
|
||||||
|
default:
|
||||||
|
return ua;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns a copy of the user action with one element parsed, by default the last one
|
||||||
|
function _parseUserActionColValues(ua: UserAction, docData: DocData, parseBulk: boolean, index?: number
|
||||||
|
): UserAction {
|
||||||
ua = ua.slice();
|
ua = ua.slice();
|
||||||
const tableId = ua[1] as string;
|
const tableId = ua[1] as string;
|
||||||
const lastIndex = ua.length - 1;
|
if (index === undefined) {
|
||||||
const colValues = ua[lastIndex] as ColValues | BulkColValues;
|
index = ua.length - 1;
|
||||||
ua[lastIndex] = parseColValues(tableId, colValues, docData, parseBulk);
|
}
|
||||||
|
const colValues = ua[index] as ColValues | BulkColValues;
|
||||||
|
ua[index] = parseColValues(tableId, colValues, docData, parseBulk);
|
||||||
return ua;
|
return ua;
|
||||||
}
|
}
|
||||||
|
@ -93,7 +93,7 @@ export class DocApiForwarder {
|
|||||||
method: req.method,
|
method: req.method,
|
||||||
headers,
|
headers,
|
||||||
};
|
};
|
||||||
if (['POST', 'PATCH'].includes(req.method)) {
|
if (['POST', 'PATCH', 'PUT'].includes(req.method)) {
|
||||||
// uses `req` as a stream
|
// uses `req` as a stream
|
||||||
options.body = req;
|
options.body = req;
|
||||||
}
|
}
|
||||||
|
@ -56,9 +56,10 @@ const MAX_PARALLEL_REQUESTS_PER_DOC = 10;
|
|||||||
type WithDocHandler = (activeDoc: ActiveDoc, req: RequestWithLogin, resp: Response) => Promise<void>;
|
type WithDocHandler = (activeDoc: ActiveDoc, req: RequestWithLogin, resp: Response) => Promise<void>;
|
||||||
|
|
||||||
// Schema validators for api endpoints that creates or updates records.
|
// Schema validators for api endpoints that creates or updates records.
|
||||||
const {RecordsPatch, RecordsPost} = t.createCheckers(DocApiTypesTI, GristDataTI);
|
const {RecordsPatch, RecordsPost, RecordsPut} = t.createCheckers(DocApiTypesTI, GristDataTI);
|
||||||
RecordsPatch.setReportedPath("body");
|
RecordsPatch.setReportedPath("body");
|
||||||
RecordsPost.setReportedPath("body");
|
RecordsPost.setReportedPath("body");
|
||||||
|
RecordsPut.setReportedPath("body");
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Middleware for validating request's body with a Checker instance.
|
* Middleware for validating request's body with a Checker instance.
|
||||||
@ -265,13 +266,16 @@ export class DocWorkerApi {
|
|||||||
return allSame;
|
return allSame;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function fieldNames(records: any[]) {
|
||||||
|
return new Set<string>(_.flatMap(records, r => Object.keys({...r.fields, ...r.require})));
|
||||||
|
}
|
||||||
|
|
||||||
function convertToBulkColValues(records: Array<Types.Record | Types.NewRecord>): BulkColValues {
|
function convertToBulkColValues(records: Array<Types.Record | Types.NewRecord>): BulkColValues {
|
||||||
// User might want to create empty records, without providing a field name, for example for requests:
|
// User might want to create empty records, without providing a field name, for example for requests:
|
||||||
// { records: [{}] }; { records: [{fields:{}}] }
|
// { records: [{}] }; { records: [{fields:{}}] }
|
||||||
// Retrieve all field names from fields property.
|
// Retrieve all field names from fields property.
|
||||||
const fieldNames = new Set<string>(_.flatMap(records, r => Object.keys(r.fields ?? {})));
|
|
||||||
const result: BulkColValues = {};
|
const result: BulkColValues = {};
|
||||||
for (const fieldName of fieldNames) {
|
for (const fieldName of fieldNames(records)) {
|
||||||
result[fieldName] = records.map(record => record.fields?.[fieldName] ?? null);
|
result[fieldName] = records.map(record => record.fields?.[fieldName] ?? null);
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
@ -414,6 +418,31 @@ export class DocWorkerApi {
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Add or update records given in records format
|
||||||
|
this._app.put('/api/docs/:docId/tables/:tableId/records', canEdit, validate(RecordsPut),
|
||||||
|
withDoc(async (activeDoc, req, res) => {
|
||||||
|
const {records} = req.body as Types.RecordsPut;
|
||||||
|
const {tableId} = req.params;
|
||||||
|
const {noadd, noupdate, noparse, allow_empty_require} = req.query;
|
||||||
|
const onmany = stringParam(req.query.onmany || "first", "onmany", ["first", "none", "all"]);
|
||||||
|
const options = {
|
||||||
|
add: !isAffirmative(noadd),
|
||||||
|
update: !isAffirmative(noupdate),
|
||||||
|
on_many: onmany,
|
||||||
|
allow_empty_require: isAffirmative(allow_empty_require),
|
||||||
|
};
|
||||||
|
const actions = records.map(rec =>
|
||||||
|
["AddOrUpdateRecord", tableId, rec.require, rec.fields || {}, options]
|
||||||
|
);
|
||||||
|
await handleSandboxError(tableId, [...fieldNames(records)], activeDoc.applyUserActions(
|
||||||
|
docSessionFromRequest(req),
|
||||||
|
actions,
|
||||||
|
{parseStrings: !isAffirmative(noparse)},
|
||||||
|
));
|
||||||
|
res.json(null);
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
// Add a new webhook and trigger
|
// Add a new webhook and trigger
|
||||||
this._app.post('/api/docs/:docId/tables/:tableId/_subscribe', isOwner,
|
this._app.post('/api/docs/:docId/tables/:tableId/_subscribe', isOwner,
|
||||||
withDoc(async (activeDoc, req, res) => {
|
withDoc(async (activeDoc, req, res) => {
|
||||||
@ -937,7 +966,7 @@ async function handleSandboxError<T>(tableId: string, colNames: string[], p: Pro
|
|||||||
if (match) {
|
if (match) {
|
||||||
throw new ApiError(`Invalid row id ${match[1]}`, 400);
|
throw new ApiError(`Invalid row id ${match[1]}`, 400);
|
||||||
}
|
}
|
||||||
match = e.message.match(/\[Sandbox\] KeyError u?'(.*?)'/);
|
match = e.message.match(/\[Sandbox] KeyError u?'(?:Table \w+ has no column )?(\w+)'/);
|
||||||
if (match) {
|
if (match) {
|
||||||
if (match[1] === tableId) {
|
if (match[1] === tableId) {
|
||||||
throw new ApiError(`Table not found "${tableId}"`, 404);
|
throw new ApiError(`Table not found "${tableId}"`, 404);
|
||||||
|
@ -17,6 +17,17 @@ export const Record = t.iface([], {
|
|||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const AddOrUpdateRecord = t.iface([], {
|
||||||
|
"require": t.intersection(t.iface([], {
|
||||||
|
[t.indexKey]: "CellValue",
|
||||||
|
}), t.iface([], {
|
||||||
|
"id": t.opt("number"),
|
||||||
|
})),
|
||||||
|
"fields": t.opt(t.iface([], {
|
||||||
|
[t.indexKey]: "CellValue",
|
||||||
|
})),
|
||||||
|
});
|
||||||
|
|
||||||
export const RecordsPatch = t.iface([], {
|
export const RecordsPatch = t.iface([], {
|
||||||
"records": t.tuple("Record", t.rest(t.array("Record"))),
|
"records": t.tuple("Record", t.rest(t.array("Record"))),
|
||||||
});
|
});
|
||||||
@ -25,10 +36,16 @@ export const RecordsPost = t.iface([], {
|
|||||||
"records": t.tuple("NewRecord", t.rest(t.array("NewRecord"))),
|
"records": t.tuple("NewRecord", t.rest(t.array("NewRecord"))),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const RecordsPut = t.iface([], {
|
||||||
|
"records": t.tuple("AddOrUpdateRecord", t.rest(t.array("AddOrUpdateRecord"))),
|
||||||
|
});
|
||||||
|
|
||||||
const exportedTypeSuite: t.ITypeSuite = {
|
const exportedTypeSuite: t.ITypeSuite = {
|
||||||
NewRecord,
|
NewRecord,
|
||||||
Record,
|
Record,
|
||||||
|
AddOrUpdateRecord,
|
||||||
RecordsPatch,
|
RecordsPatch,
|
||||||
RecordsPost,
|
RecordsPost,
|
||||||
|
RecordsPut,
|
||||||
};
|
};
|
||||||
export default exportedTypeSuite;
|
export default exportedTypeSuite;
|
||||||
|
@ -15,6 +15,14 @@ export interface Record {
|
|||||||
fields: { [coldId: string]: CellValue };
|
fields: { [coldId: string]: CellValue };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON schema for api /record endpoint. Used in PUT method for adding or updating records.
|
||||||
|
*/
|
||||||
|
export interface AddOrUpdateRecord {
|
||||||
|
require: { [coldId: string]: CellValue } & { id?: number };
|
||||||
|
fields?: { [coldId: string]: CellValue };
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* JSON schema for the body of api /record PATCH endpoint
|
* JSON schema for the body of api /record PATCH endpoint
|
||||||
*/
|
*/
|
||||||
@ -28,3 +36,10 @@ export interface RecordsPatch {
|
|||||||
export interface RecordsPost {
|
export interface RecordsPost {
|
||||||
records: [NewRecord, ...NewRecord[]]; // at least one record is required
|
records: [NewRecord, ...NewRecord[]]; // at least one record is required
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON schema for the body of api /record PUT endpoint
|
||||||
|
*/
|
||||||
|
export interface RecordsPut {
|
||||||
|
records: [AddOrUpdateRecord, ...AddOrUpdateRecord[]]; // at least one record is required
|
||||||
|
}
|
||||||
|
@ -1671,7 +1671,7 @@ function allowTestLogin() {
|
|||||||
function trustOriginHandler(req: express.Request, res: express.Response, next: express.NextFunction) {
|
function trustOriginHandler(req: express.Request, res: express.Response, next: express.NextFunction) {
|
||||||
if (trustOrigin(req, res)) {
|
if (trustOrigin(req, res)) {
|
||||||
res.header("Access-Control-Allow-Credentials", "true");
|
res.header("Access-Control-Allow-Credentials", "true");
|
||||||
res.header("Access-Control-Allow-Methods", "GET, PATCH, POST, DELETE, OPTIONS");
|
res.header("Access-Control-Allow-Methods", "GET, PATCH, PUT, POST, DELETE, OPTIONS");
|
||||||
res.header("Access-Control-Allow-Headers", "Authorization, Content-Type, X-Requested-With");
|
res.header("Access-Control-Allow-Headers", "Authorization, Content-Type, X-Requested-With");
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Unrecognized origin');
|
throw new Error('Unrecognized origin');
|
||||||
|
@ -218,8 +218,12 @@ export function optStringParam(p: any): string|undefined {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export function stringParam(p: any, name: string, allowed?: string[]): string {
|
export function stringParam(p: any, name: string, allowed?: string[]): string {
|
||||||
if (typeof p !== 'string') { throw new Error(`${name} parameter should be a string: ${p}`); }
|
if (typeof p !== 'string') {
|
||||||
if (allowed && !allowed.includes(p)) { throw new Error(`${name} parameter ${p} should be one of ${allowed}`); }
|
throw new ApiError(`${name} parameter should be a string: ${p}`, 400);
|
||||||
|
}
|
||||||
|
if (allowed && !allowed.includes(p)) {
|
||||||
|
throw new ApiError(`${name} parameter ${p} should be one of ${allowed}`, 400);
|
||||||
|
}
|
||||||
return p;
|
return p;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -473,23 +473,8 @@ class BaseReferenceColumn(BaseColumn):
|
|||||||
.get_column_rec(self.table_id, self.col_id).visibleCol.colId
|
.get_column_rec(self.table_id, self.col_id).visibleCol.colId
|
||||||
or "id"
|
or "id"
|
||||||
)
|
)
|
||||||
column = self._target_table.get_column(col_id)
|
value = objtypes.decode_object(value)
|
||||||
# `value` is an object encoded for transmission from JS to Python,
|
return self._target_table.lookup_one_record(**{col_id: value})
|
||||||
# which is decoded to `decoded_value`.
|
|
||||||
# `raw_value` is the kind of value that would be stored in `column`.
|
|
||||||
# `rich_value` is the type of value used in formulas, especially with `lookupRecords`.
|
|
||||||
# For example, for a Date column, `raw_value` is a numerical timestamp
|
|
||||||
# and `rich_value` is a `datetime.date` object,
|
|
||||||
# assuming `value` isn't of an invalid type.
|
|
||||||
# However `value` could either be just a number
|
|
||||||
# (in which case `decoded_value` would be a number as well)
|
|
||||||
# or an encoded date (or even datetime) object like ['d', number]
|
|
||||||
# (in which case `decoded_value` would be a `datetime.date` object,
|
|
||||||
# which would get converted back to a number and then back to a date object again!)
|
|
||||||
decoded_value = objtypes.decode_object(value)
|
|
||||||
raw_value = column.convert(decoded_value)
|
|
||||||
rich_value = column._convert_raw_value(raw_value)
|
|
||||||
return self._target_table.lookup_one_record(**{col_id: rich_value})
|
|
||||||
|
|
||||||
|
|
||||||
class ReferenceColumn(BaseReferenceColumn):
|
class ReferenceColumn(BaseReferenceColumn):
|
||||||
|
@ -443,6 +443,10 @@ class Table(object):
|
|||||||
# the marker is moved to col_id so that the LookupMapColumn knows how to
|
# the marker is moved to col_id so that the LookupMapColumn knows how to
|
||||||
# update its index correctly for that column.
|
# update its index correctly for that column.
|
||||||
col_id = lookup._Contains(col_id)
|
col_id = lookup._Contains(col_id)
|
||||||
|
else:
|
||||||
|
col = self.get_column(col_id)
|
||||||
|
# Convert `value` to the correct type of rich value for that column
|
||||||
|
value = col._convert_raw_value(col.convert(value))
|
||||||
key.append(value)
|
key.append(value)
|
||||||
col_ids.append(col_id)
|
col_ids.append(col_id)
|
||||||
col_ids = tuple(col_ids)
|
col_ids = tuple(col_ids)
|
||||||
|
@ -773,3 +773,31 @@ return ",".join(str(r.id) for r in Students.lookupRecords(firstName=fn, lastName
|
|||||||
["id", "lookup"],
|
["id", "lookup"],
|
||||||
[1, [None, 0, 1, 2, 3, 'foo']],
|
[1, [None, 0, 1, 2, 3, 'foo']],
|
||||||
])
|
])
|
||||||
|
|
||||||
|
def test_conversion(self):
|
||||||
|
# Test that values are converted to the type of the column when looking up
|
||||||
|
# i.e. '123' is converted to 123
|
||||||
|
# and 'foo' is converted to AltText('foo')
|
||||||
|
self.load_sample(testutil.parse_test_sample({
|
||||||
|
"SCHEMA": [
|
||||||
|
[1, "Table1", [
|
||||||
|
[1, "num", "Numeric", False, "", "", ""],
|
||||||
|
[2, "lookup1", "RefList:Table1", True, "Table1.lookupRecords(num='123')", "", ""],
|
||||||
|
[3, "lookup2", "RefList:Table1", True, "Table1.lookupRecords(num='foo')", "", ""],
|
||||||
|
]]
|
||||||
|
],
|
||||||
|
"DATA": {
|
||||||
|
"Table1": [
|
||||||
|
["id", "num"],
|
||||||
|
[1, 123],
|
||||||
|
[2, 'foo'],
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
|
||||||
|
self.assertTableData(
|
||||||
|
"Table1", data=[
|
||||||
|
["id", "num", "lookup1", "lookup2"],
|
||||||
|
[1, 123, [1], [2]],
|
||||||
|
[2, 'foo', [1], [2]],
|
||||||
|
])
|
||||||
|
@ -905,6 +905,7 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
[3, "pet", "Text", False, "", "pet", ""],
|
[3, "pet", "Text", False, "", "pet", ""],
|
||||||
[4, "color", "Text", False, "", "color", ""],
|
[4, "color", "Text", False, "", "color", ""],
|
||||||
[5, "formula", "Text", True, "''", "formula", ""],
|
[5, "formula", "Text", True, "''", "formula", ""],
|
||||||
|
[6, "date", "Date", False, None, "date", ""],
|
||||||
]],
|
]],
|
||||||
],
|
],
|
||||||
"DATA": {
|
"DATA": {
|
||||||
@ -917,9 +918,9 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
})
|
})
|
||||||
self.load_sample(sample)
|
self.load_sample(sample)
|
||||||
|
|
||||||
def check(where, values, options, stored):
|
def check(require, values, options, stored):
|
||||||
self.assertPartialOutActions(
|
self.assertPartialOutActions(
|
||||||
self.apply_user_action(["AddOrUpdateRecord", "Table1", where, values, options]),
|
self.apply_user_action(["AddOrUpdateRecord", "Table1", require, values, options]),
|
||||||
{"stored": stored},
|
{"stored": stored},
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -958,6 +959,26 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# Update all records with empty require and allow_empty_require
|
||||||
|
check(
|
||||||
|
{},
|
||||||
|
{"color": "greener"},
|
||||||
|
{"on_many": "all", "allow_empty_require": True},
|
||||||
|
[
|
||||||
|
["UpdateRecord", "Table1", 1, {"color": "greener"}],
|
||||||
|
["UpdateRecord", "Table1", 2, {"color": "greener"}],
|
||||||
|
],
|
||||||
|
)
|
||||||
|
|
||||||
|
# Missing allow_empty_require
|
||||||
|
with self.assertRaises(ValueError):
|
||||||
|
check(
|
||||||
|
{},
|
||||||
|
{"color": "greenest"},
|
||||||
|
{},
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
|
||||||
# Don't update any records when there's several matches
|
# Don't update any records when there's several matches
|
||||||
check(
|
check(
|
||||||
{"first_name": "John"},
|
{"first_name": "John"},
|
||||||
@ -992,7 +1013,7 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
)
|
)
|
||||||
|
|
||||||
# No matching record, make a new one.
|
# No matching record, make a new one.
|
||||||
# first_name=Jack in `values` overrides first_name=John in `where`
|
# first_name=Jack in `values` overrides first_name=John in `require`
|
||||||
check(
|
check(
|
||||||
{"first_name": "John", "last_name": "Johnson"},
|
{"first_name": "John", "last_name": "Johnson"},
|
||||||
{"first_name": "Jack", "color": "yellow"},
|
{"first_name": "Jack", "color": "yellow"},
|
||||||
@ -1003,7 +1024,7 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
],
|
],
|
||||||
)
|
)
|
||||||
|
|
||||||
# Specifying a row ID in `where` is allowed
|
# Specifying a row ID in `require` is allowed
|
||||||
check(
|
check(
|
||||||
{"first_name": "Bob", "id": 100},
|
{"first_name": "Bob", "id": 100},
|
||||||
{"pet": "fish"},
|
{"pet": "fish"},
|
||||||
@ -1019,7 +1040,7 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
[],
|
[],
|
||||||
)
|
)
|
||||||
|
|
||||||
# Nothing matches this `where`, but the row ID already exists
|
# Nothing matches this `require`, but the row ID already exists
|
||||||
with self.assertRaises(AssertionError):
|
with self.assertRaises(AssertionError):
|
||||||
check(
|
check(
|
||||||
{"first_name": "Alice", "id": 100},
|
{"first_name": "Alice", "id": 100},
|
||||||
@ -1028,7 +1049,7 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
[],
|
[],
|
||||||
)
|
)
|
||||||
|
|
||||||
# Formula columns in `where` can't be used as values when creating records
|
# Formula columns in `require` can't be used as values when creating records
|
||||||
check(
|
check(
|
||||||
{"formula": "anything"},
|
{"formula": "anything"},
|
||||||
{"first_name": "Alice"},
|
{"first_name": "Alice"},
|
||||||
@ -1036,6 +1057,29 @@ class TestUserActions(test_engine.EngineTestCase):
|
|||||||
[["AddRecord", "Table1", 101, {"first_name": "Alice"}]],
|
[["AddRecord", "Table1", 101, {"first_name": "Alice"}]],
|
||||||
)
|
)
|
||||||
|
|
||||||
|
with self.assertRaises(ValueError):
|
||||||
|
# Row ID too high
|
||||||
|
check(
|
||||||
|
{"first_name": "Alice", "id": 2000000},
|
||||||
|
{"pet": "fish"},
|
||||||
|
{},
|
||||||
|
[],
|
||||||
|
)
|
||||||
|
|
||||||
|
# Check that encoded objects are decoded correctly
|
||||||
|
check(
|
||||||
|
{"date": ['d', 950400]},
|
||||||
|
{},
|
||||||
|
{},
|
||||||
|
[["AddRecord", "Table1", 102, {"date": 950400}]],
|
||||||
|
)
|
||||||
|
check(
|
||||||
|
{"date": ['d', 950400]},
|
||||||
|
{"date": ['d', 1900800]},
|
||||||
|
{},
|
||||||
|
[["UpdateRecord", "Table1", 102, {"date": 1900800}]],
|
||||||
|
)
|
||||||
|
|
||||||
def test_reference_lookup(self):
|
def test_reference_lookup(self):
|
||||||
sample = testutil.parse_test_sample({
|
sample = testutil.parse_test_sample({
|
||||||
"SCHEMA": [
|
"SCHEMA": [
|
||||||
|
@ -13,7 +13,7 @@ import actions
|
|||||||
import column
|
import column
|
||||||
import sort_specs
|
import sort_specs
|
||||||
import identifiers
|
import identifiers
|
||||||
from objtypes import strict_equal, encode_object
|
from objtypes import strict_equal, encode_object, decode_object
|
||||||
import schema
|
import schema
|
||||||
from schema import RecalcWhen
|
from schema import RecalcWhen
|
||||||
import summary
|
import summary
|
||||||
@ -319,6 +319,8 @@ class UserActions(object):
|
|||||||
for i, row_id in enumerate(filled_row_ids):
|
for i, row_id in enumerate(filled_row_ids):
|
||||||
if row_id is None or row_id < 0:
|
if row_id is None or row_id < 0:
|
||||||
filled_row_ids[i] = row_id = next_row_id
|
filled_row_ids[i] = row_id = next_row_id
|
||||||
|
elif row_id > 1000000:
|
||||||
|
raise ValueError("Row ID too high")
|
||||||
next_row_id = max(next_row_id, row_id) + 1
|
next_row_id = max(next_row_id, row_id) + 1
|
||||||
|
|
||||||
# Whenever we add new rows, remember the mapping from any negative row_ids to their final
|
# Whenever we add new rows, remember the mapping from any negative row_ids to their final
|
||||||
@ -793,9 +795,35 @@ class UserActions(object):
|
|||||||
raise ValueError("Can't save value to formula column %s" % col_id)
|
raise ValueError("Can't save value to formula column %s" % col_id)
|
||||||
|
|
||||||
@useraction
|
@useraction
|
||||||
def AddOrUpdateRecord(self, table_id, where, col_values, options):
|
def AddOrUpdateRecord(self, table_id, require, col_values, options):
|
||||||
|
"""
|
||||||
|
Add or Update ('upsert') a single record depending on `options`
|
||||||
|
and on whether a record matching `require` already exists.
|
||||||
|
|
||||||
|
`require` and `col_values` are dictionaries mapping column IDs to single cell values.
|
||||||
|
|
||||||
|
By default, if `table.lookupRecords(**require)` returns any records,
|
||||||
|
update the first one with the values in `col_values`.
|
||||||
|
Otherwise create a new record with values `{**require, **col_values}`.
|
||||||
|
|
||||||
|
`options` is a dictionary with optional settings to choose other behaviours:
|
||||||
|
- Set "on_many" to "all" or "none" to change which records are updated when several match.
|
||||||
|
- Set "update" or "add" to False to disable updating or adding records respectively,
|
||||||
|
i.e. if you only want to add records that don't already exist
|
||||||
|
or if you only want to update records that do already exist.
|
||||||
|
- Set "allow_empty_require" to True to allow `require` to be an empty dictionary,
|
||||||
|
which would mean that every record in the table is matched.
|
||||||
|
Otherwise this will raise an error to prevent mistakes like updating an entire column.
|
||||||
|
"""
|
||||||
table = self._engine.tables[table_id]
|
table = self._engine.tables[table_id]
|
||||||
records = list(table.lookup_records(**where))
|
|
||||||
|
if not require and not options.get("allow_empty_require", False):
|
||||||
|
raise ValueError("require is empty but allow_empty_require isn't set")
|
||||||
|
|
||||||
|
# Decode `require` before looking up, but let AddRecord/UpdateRecord decode the final
|
||||||
|
# values when adding/updating
|
||||||
|
decoded_require = {k: decode_object(v) for k, v in six.iteritems(require)}
|
||||||
|
records = list(table.lookup_records(**decoded_require))
|
||||||
|
|
||||||
if records and options.get("update", True):
|
if records and options.get("update", True):
|
||||||
if len(records) > 1:
|
if len(records) > 1:
|
||||||
@ -813,8 +841,12 @@ class UserActions(object):
|
|||||||
if not records and options.get("add", True):
|
if not records and options.get("add", True):
|
||||||
values = {
|
values = {
|
||||||
key: value
|
key: value
|
||||||
for key, value in six.iteritems(where)
|
for key, value in six.iteritems(require)
|
||||||
if not table.get_column(key).is_formula()
|
if not (
|
||||||
|
table.get_column(key).is_formula() and
|
||||||
|
# Check that there actually is a formula and this isn't just an empty column
|
||||||
|
self._engine.docmodel.get_column_rec(table_id, key).formula
|
||||||
|
)
|
||||||
}
|
}
|
||||||
values.update(col_values)
|
values.update(col_values)
|
||||||
self.AddRecord(table_id, values.pop("id", None), values)
|
self.AddRecord(table_id, values.pop("id", None), values)
|
||||||
|
Loading…
Reference in New Issue
Block a user