py-08-pydantic-orders
0.000
Challenge · difficulty 4/5
# Order validation (pydantic v2)
Implement **`solution.py`** using **pydantic v2**
(`from pydantic import BaseModel, ...`).
Define a model `Order` and a parsing helper:
```python
from pydantic import BaseModel
class Order(BaseModel):
id: int
customer: str
quantity: int
unit_price: float
# plus a derived `total`
def parse_orders(rows: list[dict]) -> list[Order]:
...
```
### `Order` field rules
- `id: int`
- `customer: str` — must be **non-empty** (after no special trimming required; an
empty string `""` is invalid).
- `quantity: int` — must be **strictly greater than 0**.
- `unit_price: float` — must be **greater than or equal to 0**.
- `total: float` — a **derived/computed** value equal to `quantity * unit_price`.
Callers should be able to read `order.total`. You may implement it as a
`@computed_field` property or as a validated field that is always recomputed —
but it must reflect `quantity * unit_price` and not be settable to an arbitrary
inconsistent value.
Use pydantic's standard constraint mechanisms (e.g. `Field(gt=0)`,
`Field(ge=0)`, `Field(min_length=1)`, or `field_validator`).
### `parse_orders(rows)`
- Takes a list of dicts and returns a list of validated `Order` instances, one per
input row, in order.
- If **any** row is invalid, it must raise pydantic's
`pydantic.ValidationError` (do not catch and swallow it; do not return partial
results in that case — letting the exception propagate from the first invalid
row is fine).
Example:
```python
orders = parse_orders([
{"id": 1, "customer": "Acme", "quantity": 3, "unit_price": 2.5},
])
orders[0].total # 7.5
parse_orders([{"id": 2, "customer": "X", "quantity": 0, "unit_price": 1.0}])
# raises pydantic.ValidationError (quantity must be > 0)
```
tests/test_orders.py
import pytest
from pydantic import ValidationError
from solution import Order, parse_orders
def test_valid_parse():
orders = parse_orders([
{"id": 1, "customer": "Acme", "quantity": 3, "unit_price": 2.5},
{"id": 2, "customer": "Beta", "quantity": 1, "unit_price": 10.0},
])
assert len(orders) == 2
assert isinstance(orders[0], Order)
assert orders[0].id == 1
assert orders[0].customer == "Acme"
def test_computed_total():
orders = parse_orders([
{"id": 1, "customer": "Acme", "quantity": 3, "unit_price": 2.5},
])
assert orders[0].total == pytest.approx(7.5)
def test_total_zero_price():
orders = parse_orders([
{"id": 1, "customer": "Acme", "quantity": 4, "unit_price": 0.0},
])
assert orders[0].total == pytest.approx(0.0)
def test_invalid_quantity_zero_raises():
with pytest.raises(ValidationError):
parse_orders([{"id": 2, "customer": "X", "quantity": 0, "unit_price": 1.0}])
def test_invalid_negative_quantity_raises():
with pytest.raises(ValidationError):
parse_orders([{"id": 2, "customer": "X", "quantity": -1, "unit_price": 1.0}])
def test_invalid_negative_price_raises():
with pytest.raises(ValidationError):
parse_orders([{"id": 3, "customer": "Y", "quantity": 1, "unit_price": -0.01}])
def test_empty_customer_raises():
with pytest.raises(ValidationError):
parse_orders([{"id": 4, "customer": "", "quantity": 1, "unit_price": 1.0}])
def test_one_bad_row_in_batch_raises():
rows = [
{"id": 1, "customer": "Acme", "quantity": 3, "unit_price": 2.5},
{"id": 2, "customer": "Beta", "quantity": 0, "unit_price": 1.0}, # bad
]
with pytest.raises(ValidationError):
parse_orders(rows)
def test_total_serialized_in_dump():
o = parse_orders([{"id": 1, "customer": "Acme", "quantity": 2, "unit_price": 3.0}])[0]
data = o.model_dump()
assert data["total"] == pytest.approx(6.0)
System prompt
You are an expert programmer. Solve the task exactly as specified. Output your solution as fenced code blocks using the required file name(s) and the exact function/type signatures requested. Prefer correctness; do not include prose outside code unless asked.
Proposed solution
```python
from pydantic import BaseModel, Field, field_validator, ValidationError
from typing import List, Dict
class Order(BaseModel):
id: int
customer: str = Field(min_length=1)
quantity: int = Field(gt=0)
unit_price: float = Field(ge=0)
@computed_field # type: ignore[misc]
@property
def total(self) -> float:
return self.quantity * self.unit_price
def parse_orders(rows: List[Dict]) -> List[Order]:
return [Order(**row) for row in rows]
```Test output (stdout)
==================================== ERRORS ====================================
_______________________ ERROR collecting test_orders.py ________________________
test_orders.py:4: in <module>
from solution import Order, parse_orders
solution.py:4: in <module>
class Order(BaseModel):
solution.py:10: in Order
@computed_field # type: ignore[misc]
^^^^^^^^^^^^^^
E NameError: name 'computed_field' is not defined
=========================== short test summary info ============================
ERROR test_orders.py - NameError: name 'computed_field' is not defined
!!!!!!!!!!!!!!!!!!!! Interrupted: 1 error during collection !!!!!!!!!!!!!!!!!!!!
1 error in 0.06s
Self-repair attempts
[
{
"total": 1,
"answer": "```python\nfrom pydantic import BaseModel, Field, field_validator, ValidationError\nfrom typing import List, Dict\n\nclass Order(BaseModel):\n id: int\n customer: str = Field(min_length=1)\n quantity: int = Field(gt=0)\n unit_price: float = Field(ge=0)\n\n @computed_field # type: ignore[misc]\n @property\n def total(self) -> float:\n return self.quantity * self.unit_price\n\ndef parse_orders(rows: List[Dict]) -> List[Order]:\n return [Order(**row) for row in rows]\n```",
"passed": 0,
"test_error": "==================================== ERRORS ====================================\n_______________________ ERROR collecting test_orders.py ________________________\ntest_orders.py:4: in <module>\n from solution import Order, parse_orders\nsolution.py:4: in <module>\n class Order(BaseModel):\nsolution.py:10: in Order\n @computed_field # type: ignore[misc]\n ^^^^^^^^^^^^^^\nE NameError: name 'computed_field' is not defined\n=========================== short test summary info ============================\nERROR test_orders.py - NameError: name 'computed_field' is not defined\n!!!!!!!!!!!!!!!!!!!! Interrupted: 1 error during collection !!!!!!!!!!!!!!!!!!!!\n1 error in 0.06s"
},
{
"total": 9,
"answer": "```python\nfrom pydantic import BaseModel, Field, field_validator, computed_field\nfrom typing import List, Dict\n\nclass Order(BaseModel):\n id: int\n customer: str = Field(min_length=1)\n quantity: int = Field(gt=0)\n unit_price: float = Field(ge=0)\n\n @computed_field\n @property\n def total(self) -> float:\n return self.quantity * self.unit_price\n\ndef parse_orders(rows: List[Dict]) -> List[Order]:\n return [Order(**row) for row in rows]\n```",
"passed": 9,
"test_error": ""
}
]