(core) Add CellError to explain when a formula refers to a cell with an error value

Summary:
get_cell_value wraps RaisedException with CellError to expand the error message for the user.

This is still pretty conceptual, the comments explain some things to think about, but it works and is an improvement.

Test Plan: Updated Python unit tests

Reviewers: dsagal, paulfitz

Reviewed By: dsagal

Differential Revision: https://phab.getgrist.com/D2928
This commit is contained in:
Alex Hall 2021-07-30 20:51:56 +02:00
parent 1605e18f66
commit 0e5c2bee59
4 changed files with 51 additions and 8 deletions

View File

@ -147,7 +147,12 @@ class BaseColumn(object):
""" """
raw = self.raw_get(row_id) raw = self.raw_get(row_id)
if isinstance(raw, objtypes.RaisedException): if isinstance(raw, objtypes.RaisedException):
if isinstance(raw.error, depend.CircularRefError):
# Wrapping a CircularRefError in a CellError is often redundant, but
# TODO a CellError is still useful if the calling cell is not involved in the cycle
raise raw.error raise raw.error
else:
raise objtypes.CellError(self.table_id, self.col_id, row_id, raw.error)
if self.type_obj.is_right_type(raw): if self.type_obj.is_right_type(raw):
return self._make_rich_value(raw) return self._make_rich_value(raw)
return usertypes.AltText(str(raw), self.type_obj.typename()) return usertypes.AltText(str(raw), self.type_obj.typename())

View File

@ -12,6 +12,7 @@ import six
import column import column
from functions import date # pylint: disable=import-error from functions import date # pylint: disable=import-error
from functions.unimplemented import unimplemented from functions.unimplemented import unimplemented
from objtypes import CellError
from usertypes import AltText # pylint: disable=import-error from usertypes import AltText # pylint: disable=import-error
from records import Record, RecordSet from records import Record, RecordSet
@ -597,6 +598,9 @@ def _prepare_record_dict(record, dates_as_iso=False, expand_refs=0):
result[col_id] = val result[col_id] = val
except Exception as e: except Exception as e:
result[col_id] = None result[col_id] = None
while isinstance(e, CellError):
# The extra information from CellError is redundant here
e = e.error # pylint: disable=no-member
errors[col_id] = "%s: %s" % (type(e).__name__, str(e)) errors[col_id] = "%s: %s" % (type(e).__name__, str(e))
if errors: if errors:

View File

@ -263,11 +263,18 @@ class RaisedException(object):
# and to DB (for when we store formula results). There are two concerns: one is that it's # and to DB (for when we store formula results). There are two concerns: one is that it's
# potentially quite verbose; the other is that it's makes the tests more annoying (again b/c # potentially quite verbose; the other is that it's makes the tests more annoying (again b/c
# verbose). # verbose).
error = self.error
location = ""
while isinstance(error, CellError):
if not location:
location = "\n(in referenced cell {error.location})".format(error=error)
error = error.error
name = type(error).__name__
if self.details: if self.details:
return [type(self.error).__name__, str(self.error), self.details] return [name, str(error) + location, self.details]
if isinstance(self.error, InvalidTypedValue): if isinstance(error, InvalidTypedValue):
return [type(self.error).__name__, self.error.typename, self.error.value] return [name, error.typename, error.value]
return [type(self.error).__name__] return [name]
@classmethod @classmethod
def decode_args(cls, *args): def decode_args(cls, *args):
@ -275,6 +282,24 @@ class RaisedException(object):
return cls(None, encoded_error=list(args)) return cls(None, encoded_error=list(args))
class CellError(Exception):
def __init__(self, table_id, col_id, row_id, error):
super(CellError, self).__init__(table_id, col_id, row_id, error)
self.table_id = table_id
self.col_id = col_id
self.row_id = row_id
self.error = error
def __str__(self):
return (
"{self.error.__class__.__name__} in referenced cell {self.location}"
).format(self=self)
@property
def location(self):
return "{self.table_id}[{self.row_id}].{self.col_id}".format(self=self)
class RecordList(list): class RecordList(list):
""" """
Just like list but allows setting custom attributes, which we use for remembering _group_by and Just like list but allows setting custom attributes, which we use for remembering _group_by and

View File

@ -330,9 +330,18 @@ else:
self.assertFormulaError(self.engine.get_formula_error('AttrTest', 'B', 1), self.assertFormulaError(self.engine.get_formula_error('AttrTest', 'B', 1),
AttributeError, "Table 'AttrTest' has no column 'AA'", AttributeError, "Table 'AttrTest' has no column 'AA'",
r"AttributeError: Table 'AttrTest' has no column 'AA'") r"AttributeError: Table 'AttrTest' has no column 'AA'")
self.assertFormulaError(self.engine.get_formula_error('AttrTest', 'C', 1), cell_error = self.engine.get_formula_error('AttrTest', 'C', 1)
AttributeError, "Table 'AttrTest' has no column 'AA'", self.assertFormulaError(cell_error,
r"AttributeError: Table 'AttrTest' has no column 'AA'") objtypes.CellError, "AttributeError in referenced cell AttrTest[1].B",
r"CellError: AttributeError in referenced cell AttrTest\[1\].B")
self.assertEqual(
objtypes.encode_object(cell_error),
['E',
'AttributeError',
"Table 'AttrTest' has no column 'AA'\n"
"(in referenced cell AttrTest[1].B)",
cell_error.details]
)
def test_cumulative_formula(self): def test_cumulative_formula(self):
formula = ("Table1.lookupOne(A=$A-1).Principal + Table1.lookupOne(A=$A-1).Interest " + formula = ("Table1.lookupOne(A=$A-1).Principal + Table1.lookupOne(A=$A-1).Interest " +