Skip to content

Allow conditionally disabling fields #1953

Open
@ThiefMaster

Description

@ThiefMaster

I have this schema:

class RequestAccessSchema(mm.Schema):
    request_cern_access = fields.Bool(load_default=False, data_key='cern_access_request_cern_access')
    birth_date = fields.Date(load_default=None, data_key='cern_access_birth_date',
                             validate=validate_with_message(lambda x: x <= date.today(),
                                                            'The specified date is in the future'))
    nationality = fields.String(load_default='', data_key='cern_access_nationality')
    birth_place = fields.String(load_default='', data_key='cern_access_birth_place')
    by_car = fields.Bool(load_default=False, data_key='cern_access_by_car')
    license_plate = fields.String(data_key='cern_access_license_plate', load_default='')

    @validates_schema
    def validate_everything(self, data, **kwargs):
        # This ugly mess is needed since we can't skip fields conditionally...
        if not data['request_cern_access']:
            return
        required_fields = {'birth_date', 'nationality', 'birth_place'}
        if data['by_car']:
            required_fields.add('license_plate')
        errors = {}
        for field in required_fields:
            if not data[field]:
                errors[self.fields[field].data_key] = ['This field is required.']
        if data['by_car'] and data['license_plate']:
            try:
                validate.And(
                    validate.Length(min=3),
                    validate.Regexp(r'^[0-9A-Za-z]+([- ][ ]*[0-9A-Za-z]+)*$')
                )(data['license_plate'])
            except ValidationError as exc:
                errors.setdefault(self.fields['license_plate'].data_key, []).extend(exc.messages)
        if errors:
            raise ValidationError(errors)

Depending on the bool fields in there, the rest of the data is either completely ignored (I set it to falsy values in a post_load function) or required (with some validation needed). It would be really nice if there was some way to disable fields during parse time; that way I could simply make them required and specify the validators declaratively, instead of having to fall back to this abomination up there.

If you have any other ideas on how this could be done without the mess above, maybe even using features already present in marshmallow, please let me know! :)

Metadata

Metadata

Assignees

No one assigned

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions