Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Capture a report of all errors and warnings issued from validators #1421

Open
redruin1 opened this issue Mar 26, 2025 · 0 comments
Open

Capture a report of all errors and warnings issued from validators #1421

redruin1 opened this issue Mar 26, 2025 · 0 comments

Comments

@redruin1
Copy link

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.define
class Example:
    name: str = attrs.field()

    @name.validator
    def always_fails(self, attr, value):
        raise ValueError()
    
    @name.validator
    def would_never_run(self, attr, value): 
        # In pure attrs this function will never even run because of the previous validator
        raise TypeError() 

The simplest solution I can think of is to simply change the signature of validate() and validators to handle kwargs:

def attrs.validate(inst, **kwargs):
    ...

def validator(self, attr, value, **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.define
class Example:
    name: str = attrs.field()

    @name.validator
    def only_when_strict(self, attr, value, strict: Optional[bool] = False):
        if strict:
            raise ValueError()
        
e = Example(name="passes, because 'strict' defaults to 'False'")
e.name = "same as above"
attrs.validate(e)               # Same as above
attrs.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:

@attrs.define
class Example:
    name: str = attrs.field()

    @name.validator
    def always_fails(self, attr, value, error_list: Optional[list] = None):
        if error_list:
            error_list.append(ValueError())
        else:
            raise ValueError()
    
    @name.validator
    def would_never_run(self, attr, value, error_list: Optional[list] = None): 
        if error_list:
            error_list.append(TypeError())
        else:
            raise TypeError() 

with attrs.validators.disabled():
    e = Example(name="something") # Normally raises because `error_list` defaults to `None`

error_list = []
attrs.validate(e, error_list=error_list) # Collect errors in `error_list`
print(error_list) # -> [ValueError(), TypeError()]

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 way
e.name = "whatever" # would only raise ValueError, instead of ValueError + TypeError

# But suppose I could overwrite the default validation in a manner similar to:
def my_hook(validators):
    es = []
    for validator in validators:
        try:
            validator()
        except Exception as e:
            es.append(e)
    raise ExceptionGroup("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.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

No branches or pull requests

1 participant