mirror of
https://github.com/gristlabs/grist-core.git
synced 2026-03-02 04:09:24 +00:00
(core) AddOrUpdateRecord user action
Summary: New user action as described in https://grist.quip.com/fZSrAnJKgO5j/Add-or-Update-Records-API, with options to allow most of the mentioned possible behaviours. The Python code is due to Alex (as should be obvious from the u in behaviours). Test Plan: Added a unit test. Reviewers: alexmojaki Reviewed By: alexmojaki Differential Revision: https://phab.getgrist.com/D3239
This commit is contained in:
@@ -51,7 +51,7 @@ def parse_acl_formula_json(acl_formula):
|
||||
|
||||
|
||||
# Entities encountered in ACL formulas, which may get renamed.
|
||||
# type: 'recCol'|'userAttr'|'userAttrCol',
|
||||
# type : 'recCol'|'userAttr'|'userAttrCol',
|
||||
# start_pos: number, # start position of the token in the code.
|
||||
# name: string, # the name that may be updated by a rename.
|
||||
# extra: string|None, # name of userAttr in case of userAttrCol; otherwise None.
|
||||
|
||||
@@ -49,7 +49,7 @@ ReplaceTableData = namedtuple_eq('ReplaceTableData', BulkAddRecord._fields)
|
||||
# table_id: string name of the table.
|
||||
# col_id: string name of column
|
||||
# col_info: dictionary with particular keys
|
||||
# type: string type of the column
|
||||
# type : string type of the column
|
||||
# isFormula: bool, whether it is a formula column
|
||||
# formula: string text of the formula, or empty string
|
||||
# Other keys may be set in col_info (e.g. widgetOptions, label) but are not currently used in
|
||||
|
||||
@@ -896,6 +896,145 @@ class TestUserActions(test_engine.EngineTestCase):
|
||||
[2, 2, json.dumps({"included": ["b", "c"]}), None, 1]
|
||||
])
|
||||
|
||||
def test_add_or_update(self):
|
||||
sample = testutil.parse_test_sample({
|
||||
"SCHEMA": [
|
||||
[1, "Table1", [
|
||||
[1, "first_name", "Text", False, "", "first_name", ""],
|
||||
[2, "last_name", "Text", False, "", "last_name", ""],
|
||||
[3, "pet", "Text", False, "", "pet", ""],
|
||||
[4, "color", "Text", False, "", "color", ""],
|
||||
[5, "formula", "Text", True, "''", "formula", ""],
|
||||
]],
|
||||
],
|
||||
"DATA": {
|
||||
"Table1": [
|
||||
["id", "first_name", "last_name"],
|
||||
[1, "John", "Doe"],
|
||||
[2, "John", "Smith"],
|
||||
],
|
||||
}
|
||||
})
|
||||
self.load_sample(sample)
|
||||
|
||||
def check(where, values, options, stored):
|
||||
self.assertPartialOutActions(
|
||||
self.apply_user_action(["AddOrUpdateRecord", "Table1", where, values, options]),
|
||||
{"stored": stored},
|
||||
)
|
||||
|
||||
# Exactly one match, so on_many=none has no effect
|
||||
check(
|
||||
{"first_name": "John", "last_name": "Smith"},
|
||||
{"pet": "dog", "color": "red"},
|
||||
{"on_many": "none"},
|
||||
[["UpdateRecord", "Table1", 2, {"color": "red", "pet": "dog"}]],
|
||||
)
|
||||
|
||||
# Look for a record with pet=dog and change it to pet=cat
|
||||
check(
|
||||
{"first_name": "John", "pet": "dog"},
|
||||
{"pet": "cat"},
|
||||
{},
|
||||
[["UpdateRecord", "Table1", 2, {"pet": "cat"}]],
|
||||
)
|
||||
|
||||
# Two records match first_name=John, by default we only update the first
|
||||
check(
|
||||
{"first_name": "John"},
|
||||
{"color": "blue"},
|
||||
{},
|
||||
[["UpdateRecord", "Table1", 1, {"color": "blue"}]],
|
||||
)
|
||||
|
||||
# Update all matching records
|
||||
check(
|
||||
{"first_name": "John"},
|
||||
{"color": "green"},
|
||||
{"on_many": "all"},
|
||||
[
|
||||
["UpdateRecord", "Table1", 1, {"color": "green"}],
|
||||
["UpdateRecord", "Table1", 2, {"color": "green"}],
|
||||
],
|
||||
)
|
||||
|
||||
# Don't update any records when there's several matches
|
||||
check(
|
||||
{"first_name": "John"},
|
||||
{"color": "yellow"},
|
||||
{"on_many": "none"},
|
||||
[],
|
||||
)
|
||||
|
||||
# Invalid value of on_many
|
||||
with self.assertRaises(ValueError):
|
||||
check(
|
||||
{"first_name": "John"},
|
||||
{"color": "yellow"},
|
||||
{"on_many": "other"},
|
||||
[],
|
||||
)
|
||||
|
||||
# Since there's at least one matching record and update=False, do nothing
|
||||
check(
|
||||
{"first_name": "John"},
|
||||
{"color": "yellow"},
|
||||
{"update": False},
|
||||
[],
|
||||
)
|
||||
|
||||
# Since there's no matching records and add=False, do nothing
|
||||
check(
|
||||
{"first_name": "John", "last_name": "Johnson"},
|
||||
{"first_name": "Jack", "color": "yellow"},
|
||||
{"add": False},
|
||||
[],
|
||||
)
|
||||
|
||||
# No matching record, make a new one.
|
||||
# first_name=Jack in `values` overrides first_name=John in `where`
|
||||
check(
|
||||
{"first_name": "John", "last_name": "Johnson"},
|
||||
{"first_name": "Jack", "color": "yellow"},
|
||||
{},
|
||||
[
|
||||
["AddRecord", "Table1", 3,
|
||||
{"color": "yellow", "first_name": "Jack", "last_name": "Johnson"}]
|
||||
],
|
||||
)
|
||||
|
||||
# Specifying a row ID in `where` is allowed
|
||||
check(
|
||||
{"first_name": "Bob", "id": 100},
|
||||
{"pet": "fish"},
|
||||
{},
|
||||
[["AddRecord", "Table1", 100, {"first_name": "Bob", "pet": "fish"}]],
|
||||
)
|
||||
|
||||
# Now the row already exists
|
||||
check(
|
||||
{"first_name": "Bob", "id": 100},
|
||||
{"pet": "fish"},
|
||||
{},
|
||||
[],
|
||||
)
|
||||
|
||||
# Nothing matches this `where`, but the row ID already exists
|
||||
with self.assertRaises(AssertionError):
|
||||
check(
|
||||
{"first_name": "Alice", "id": 100},
|
||||
{"pet": "fish"},
|
||||
{},
|
||||
[],
|
||||
)
|
||||
|
||||
# Formula columns in `where` can't be used as values when creating records
|
||||
check(
|
||||
{"formula": "anything"},
|
||||
{"first_name": "Alice"},
|
||||
{},
|
||||
[["AddRecord", "Table1", 101, {"first_name": "Alice"}]],
|
||||
)
|
||||
|
||||
def test_reference_lookup(self):
|
||||
sample = testutil.parse_test_sample({
|
||||
|
||||
@@ -792,6 +792,33 @@ class UserActions(object):
|
||||
# Otherwise, this is an error. We can't save individual values to formula columns.
|
||||
raise ValueError("Can't save value to formula column %s" % col_id)
|
||||
|
||||
@useraction
|
||||
def AddOrUpdateRecord(self, table_id, where, col_values, options):
|
||||
table = self._engine.tables[table_id]
|
||||
records = list(table.lookup_records(**where))
|
||||
|
||||
if records and options.get("update", True):
|
||||
if len(records) > 1:
|
||||
on_many = options.get("on_many", "first")
|
||||
if on_many == "first":
|
||||
records = records[:1]
|
||||
elif on_many == "none":
|
||||
return
|
||||
elif on_many != "all":
|
||||
raise ValueError("on_many should be 'first', 'none', or 'all', not %r" % on_many)
|
||||
|
||||
for record in records:
|
||||
self.UpdateRecord(table_id, record.id, col_values)
|
||||
|
||||
if not records and options.get("add", True):
|
||||
values = {
|
||||
key: value
|
||||
for key, value in six.iteritems(where)
|
||||
if not table.get_column(key).is_formula()
|
||||
}
|
||||
values.update(col_values)
|
||||
self.AddRecord(table_id, values.pop("id", None), values)
|
||||
|
||||
#----------------------------------------
|
||||
# RemoveRecords & co.
|
||||
#----------------------------------------
|
||||
|
||||
Reference in New Issue
Block a user