|
@ -87,6 +87,9 @@ multipart_incorrect_install_error = ( |
|
|
"pip install python-multipart\n" |
|
|
"pip install python-multipart\n" |
|
|
) |
|
|
) |
|
|
|
|
|
|
|
|
|
|
|
# Sentinel value for unspecified fields |
|
|
|
|
|
_NOT_SPECIFIED = object() |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
def ensure_multipart_is_installed() -> None: |
|
|
def ensure_multipart_is_installed() -> None: |
|
|
try: |
|
|
try: |
|
@ -690,7 +693,7 @@ async def solve_dependencies( |
|
|
def _validate_value_with_model_field( |
|
|
def _validate_value_with_model_field( |
|
|
*, field: ModelField, value: Any, values: Dict[str, Any], loc: Tuple[str, ...] |
|
|
*, field: ModelField, value: Any, values: Dict[str, Any], loc: Tuple[str, ...] |
|
|
) -> Tuple[Any, List[Any]]: |
|
|
) -> Tuple[Any, List[Any]]: |
|
|
if value is None: |
|
|
if value is _NOT_SPECIFIED: |
|
|
if field.required: |
|
|
if field.required: |
|
|
return None, [get_missing_field_error(loc=loc)] |
|
|
return None, [get_missing_field_error(loc=loc)] |
|
|
else: |
|
|
else: |
|
@ -712,20 +715,24 @@ def _get_multidict_value( |
|
|
if is_sequence_field(field) and isinstance(values, (ImmutableMultiDict, Headers)): |
|
|
if is_sequence_field(field) and isinstance(values, (ImmutableMultiDict, Headers)): |
|
|
value = values.getlist(alias) |
|
|
value = values.getlist(alias) |
|
|
else: |
|
|
else: |
|
|
value = values.get(alias, None) |
|
|
value = values.get(alias, _NOT_SPECIFIED) |
|
|
if ( |
|
|
if ( |
|
|
value is None |
|
|
isinstance(field.field_info, params.Form) |
|
|
or ( |
|
|
and isinstance(value, str) # For type checks |
|
|
isinstance(field.field_info, params.Form) |
|
|
and value == "" |
|
|
and isinstance(value, str) # For type checks |
|
|
|
|
|
and value == "" |
|
|
|
|
|
) |
|
|
|
|
|
or (is_sequence_field(field) and len(value) == 0) |
|
|
|
|
|
): |
|
|
): |
|
|
|
|
|
# Empty strings in a form can be a representation of None values |
|
|
|
|
|
_, error = field.validate(value=None) |
|
|
|
|
|
# If None is an accepted value for this field, use that |
|
|
|
|
|
if error is None: |
|
|
|
|
|
value = None |
|
|
|
|
|
|
|
|
|
|
|
if value == _NOT_SPECIFIED or (is_sequence_field(field) and len(value) == 0): |
|
|
if field.required: |
|
|
if field.required: |
|
|
return |
|
|
return _NOT_SPECIFIED |
|
|
else: |
|
|
else: |
|
|
return deepcopy(field.default) |
|
|
return deepcopy(field.default) |
|
|
|
|
|
|
|
|
return value |
|
|
return value |
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -763,7 +770,7 @@ def request_params_to_args( |
|
|
else field.name.replace("_", "-") |
|
|
else field.name.replace("_", "-") |
|
|
) |
|
|
) |
|
|
value = _get_multidict_value(field, received_params, alias=alias) |
|
|
value = _get_multidict_value(field, received_params, alias=alias) |
|
|
if value is not None: |
|
|
if value != _NOT_SPECIFIED: |
|
|
params_to_process[field.name] = value |
|
|
params_to_process[field.name] = value |
|
|
processed_keys.add(alias or field.alias) |
|
|
processed_keys.add(alias or field.alias) |
|
|
processed_keys.add(field.name) |
|
|
processed_keys.add(field.name) |
|
@ -830,6 +837,8 @@ async def _extract_form_body( |
|
|
first_field = body_fields[0] |
|
|
first_field = body_fields[0] |
|
|
first_field_info = first_field.field_info |
|
|
first_field_info = first_field.field_info |
|
|
|
|
|
|
|
|
|
|
|
processed_keys = set() |
|
|
|
|
|
|
|
|
for field in body_fields: |
|
|
for field in body_fields: |
|
|
value = _get_multidict_value(field, received_body) |
|
|
value = _get_multidict_value(field, received_body) |
|
|
if ( |
|
|
if ( |
|
@ -857,10 +866,13 @@ async def _extract_form_body( |
|
|
for sub_value in value: |
|
|
for sub_value in value: |
|
|
tg.start_soon(process_fn, sub_value.read) |
|
|
tg.start_soon(process_fn, sub_value.read) |
|
|
value = serialize_sequence_value(field=field, value=results) |
|
|
value = serialize_sequence_value(field=field, value=results) |
|
|
if value is not None: |
|
|
if value != _NOT_SPECIFIED: |
|
|
values[field.alias] = value |
|
|
values[field.alias] = value |
|
|
|
|
|
processed_keys.add(field.alias) |
|
|
|
|
|
processed_keys.add(field.name) |
|
|
|
|
|
|
|
|
for key, value in received_body.items(): |
|
|
for key, value in received_body.items(): |
|
|
if key not in values: |
|
|
if key not in processed_keys: |
|
|
values[key] = value |
|
|
values[key] = value |
|
|
return values |
|
|
return values |
|
|
|
|
|
|
|
@ -888,15 +900,18 @@ async def request_body_to_args( |
|
|
if single_not_embedded_field: |
|
|
if single_not_embedded_field: |
|
|
loc: Tuple[str, ...] = ("body",) |
|
|
loc: Tuple[str, ...] = ("body",) |
|
|
v_, errors_ = _validate_value_with_model_field( |
|
|
v_, errors_ = _validate_value_with_model_field( |
|
|
field=first_field, value=body_to_process, values=values, loc=loc |
|
|
field=first_field, |
|
|
|
|
|
value=body_to_process if body_to_process is not None else _NOT_SPECIFIED, |
|
|
|
|
|
values=values, |
|
|
|
|
|
loc=loc, |
|
|
) |
|
|
) |
|
|
return {first_field.name: v_}, errors_ |
|
|
return {first_field.name: v_}, errors_ |
|
|
for field in body_fields: |
|
|
for field in body_fields: |
|
|
loc = ("body", field.alias) |
|
|
loc = ("body", field.alias) |
|
|
value: Optional[Any] = None |
|
|
value: Optional[Any] = _NOT_SPECIFIED |
|
|
if body_to_process is not None: |
|
|
if body_to_process is not None: |
|
|
try: |
|
|
try: |
|
|
value = body_to_process.get(field.alias) |
|
|
value = _get_multidict_value(field, values=body_to_process) |
|
|
# If the received body is a list, not a dict |
|
|
# If the received body is a list, not a dict |
|
|
except AttributeError: |
|
|
except AttributeError: |
|
|
errors.append(get_missing_field_error(loc)) |
|
|
errors.append(get_missing_field_error(loc)) |
|
|