You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
Part of the reason why attrs is so appealing is that all validations are essentially just plain functions that are run at certain times. However, I find myself pining for a little bit more control on how those functions are run. Notably, I'm looking for the ability to collect a report of all exceptions/warnings issued from validators in some object returned to users, which can then be manipulated and optionally reissued (somewhat similar to how cattrs uses ExceptionGroups, but on the attrs instance itself). In a trivial case, consider a toy example where two validators always raise an exception:
@attrs.defineclassExample:
name: str=attrs.field()
@name.validatordefalways_fails(self, attr, value):
raiseValueError()
@name.validatordefwould_never_run(self, attr, value):
# In pure attrs this function will never even run because of the previous validatorraiseTypeError()
The simplest solution I can think of is to simply change the signature of validate() and validators to handle kwargs:
This would allow you pass whatever values you wanted to attrs.validate() and they would be passed straight through to every evaluated function. In cases where no extra args are specified (such as during attribute setting) then they would either error (appropriately) or resolve to defaults:
@attrs.defineclassExample:
name: str=attrs.field()
@name.validatordefonly_when_strict(self, attr, value, strict: Optional[bool] =False):
ifstrict:
raiseValueError()
e=Example(name="passes, because 'strict' defaults to 'False'")
e.name="same as above"attrs.validate(e) # Same as aboveattrs.validate(e, strict=True) # Raises ValueError
To achieve simple error collection, I could simply pass in an optional list into kwargs and have errors be collected there if a validator detects said list:
In essence, this works in the same way as Pydantic's context argument. Writing a decorator like @collect_errors to remove the boilerplate of the if statement would be trivial, while also preserving the notion that validators are just "regular" functions. This (to me) seems a very simple and unobtrusive way to expand the functionality of existing validators while also maintaining backwards compatibility, though it unfortunately is only limited to cases where you explicitly call attrs.validate().
Alternatively, some well defined way to control or change the function that calls the validators would also be sufficient for my needs, and would allow for proper error collection in all cases. For example, suppose I want to rewrite the validation logic to utilize the aforementioned ExceptionGroups:
# Under the kwargs solution outlined above, the following woudn't collect both issues in the same waye.name="whatever"# would only raise ValueError, instead of ValueError + TypeError# But suppose I could overwrite the default validation in a manner similar to:defmy_hook(validators):
es= []
forvalidatorinvalidators:
try:
validator()
exceptExceptionase:
es.append(e)
raiseExceptionGroup("blah", es)
Example.__attrs_validator_func__=my_hook# or whatever isn't horrifying# Then:e.name="whatever"""" + Exception Group Traceback (most recent call last): | File ".\example.py", line 78, in <module> | raise ExceptionGroup("blah", es) | exceptiongroup.ExceptionGroup: blah (2 sub-exceptions) +-+---------------- 1 ---------------- | ValueError +---------------- 2 ---------------- | TypeError +------------------------------------"""
I don't believe cattrs itself is a solution here, because from what I understand cattrs is only concerned with validation of the object as it is being converted to/from a "raw" format (such as a dictionary to an attrs instance). In my case, I want the validators to run on manipulation of the attrs class itself, which only attrs itself is in charge of.
The text was updated successfully, but these errors were encountered:
Part of the reason why attrs is so appealing is that all validations are essentially just plain functions that are run at certain times. However, I find myself pining for a little bit more control on how those functions are run. Notably, I'm looking for the ability to collect a report of all exceptions/warnings issued from validators in some object returned to users, which can then be manipulated and optionally reissued (somewhat similar to how cattrs uses
ExceptionGroup
s, but on the attrs instance itself). In a trivial case, consider a toy example where two validators always raise an exception:The simplest solution I can think of is to simply change the signature of
validate()
and validators to handlekwargs
:This would allow you pass whatever values you wanted to
attrs.validate()
and they would be passed straight through to every evaluated function. In cases where no extra args are specified (such as during attribute setting) then they would either error (appropriately) or resolve to defaults:To achieve simple error collection, I could simply pass in an optional list into kwargs and have errors be collected there if a validator detects said list:
In essence, this works in the same way as Pydantic's
context
argument. Writing a decorator like@collect_errors
to remove the boilerplate of the if statement would be trivial, while also preserving the notion that validators are just "regular" functions. This (to me) seems a very simple and unobtrusive way to expand the functionality of existing validators while also maintaining backwards compatibility, though it unfortunately is only limited to cases where you explicitly callattrs.validate()
.Alternatively, some well defined way to control or change the function that calls the validators would also be sufficient for my needs, and would allow for proper error collection in all cases. For example, suppose I want to rewrite the validation logic to utilize the aforementioned
ExceptionGroups
:I don't believe cattrs itself is a solution here, because from what I understand cattrs is only concerned with validation of the object as it is being converted to/from a "raw" format (such as a dictionary to an attrs instance). In my case, I want the validators to run on manipulation of the attrs class itself, which only attrs itself is in charge of.
The text was updated successfully, but these errors were encountered: