cuenca-validations 2.1.36.dev3__tar.gz → 2.1.38.dev0__tar.gz
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/PKG-INFO +1 -1
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/queries.py +14 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/requests.py +11 -1
- cuenca_validations-2.1.38.dev0/cuenca_validations/version.py +1 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations.egg-info/PKG-INFO +1 -1
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/test_types.py +92 -23
- cuenca_validations-2.1.36.dev3/cuenca_validations/version.py +0 -1
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/LICENSE +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/README.md +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/__init__.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/card_bins.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/errors.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/py.typed +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/__init__.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/card.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/enums.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/files.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/general.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/helpers.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/identities.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/morals.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/typing.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/validators.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations.egg-info/SOURCES.txt +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations.egg-info/dependency_links.txt +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations.egg-info/requires.txt +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations.egg-info/top_level.txt +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/setup.cfg +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/setup.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/__init__.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/test_card.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/test_errors.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/test_helpers.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/test_requests.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/test_statement.py +0 -0
- {cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/tests/test_validators.py +0 -0
|
@@ -40,6 +40,7 @@ class QueryParams(BaseModel):
|
|
|
40
40
|
]
|
|
41
41
|
limit: Optional[PositiveInt] = None
|
|
42
42
|
user_id: Optional[str] = None
|
|
43
|
+
ids: Optional[str] = None
|
|
43
44
|
created_before: Optional[dt.datetime] = None
|
|
44
45
|
created_after: Optional[dt.datetime] = None
|
|
45
46
|
related_transaction: Optional[str] = None
|
|
@@ -64,6 +65,18 @@ class QueryParams(BaseModel):
|
|
|
64
65
|
},
|
|
65
66
|
)
|
|
66
67
|
|
|
68
|
+
@field_validator('ids')
|
|
69
|
+
@classmethod
|
|
70
|
+
def validate_ids_count(cls, value: Optional[str]) -> Optional[str]:
|
|
71
|
+
if value is None:
|
|
72
|
+
return None
|
|
73
|
+
id_count = len([part for part in value.split(',') if part.strip()])
|
|
74
|
+
if id_count > MAX_PAGE_SIZE:
|
|
75
|
+
raise ValueError(
|
|
76
|
+
f'ids must contain at most {MAX_PAGE_SIZE} values'
|
|
77
|
+
)
|
|
78
|
+
return value
|
|
79
|
+
|
|
67
80
|
def model_dump(self, *args, **kwargs) -> DictStrAny:
|
|
68
81
|
kwargs.setdefault('exclude_none', True)
|
|
69
82
|
kwargs.setdefault('exclude_unset', True)
|
|
@@ -157,6 +170,7 @@ class UserQuery(QueryParams):
|
|
|
157
170
|
phone_number: Optional[str] = None
|
|
158
171
|
email_address: Optional[EmailStr] = None
|
|
159
172
|
status: Optional[UserStatus] = None
|
|
173
|
+
is_blocked: Optional[bool] = None
|
|
160
174
|
identity_uri: Optional[str] = None
|
|
161
175
|
has_curp_document: Optional[bool] = None
|
|
162
176
|
clabe: Optional[Clabe] = None
|
|
@@ -326,10 +326,20 @@ class WalletTransactionRequest(BaseRequest):
|
|
|
326
326
|
|
|
327
327
|
class FraudFundsTransferRequest(BaseRequest):
|
|
328
328
|
user_id: NonEmptyStr
|
|
329
|
-
clabe: Clabe
|
|
329
|
+
clabe: Optional[Clabe] = None
|
|
330
|
+
bank_code: Optional[str] = None
|
|
331
|
+
tipo_pago: Optional[int] = None
|
|
330
332
|
amount: Optional[StrictPositiveInt] = None
|
|
331
333
|
concepto: Optional[NonEmptyStr] = None
|
|
332
334
|
|
|
335
|
+
@model_validator(mode='after')
|
|
336
|
+
def validate_destination(self) -> 'FraudFundsTransferRequest':
|
|
337
|
+
if self.clabe is None and self.bank_code is None:
|
|
338
|
+
raise ValueError('clabe or bank_code required')
|
|
339
|
+
if self.bank_code is not None and self.tipo_pago is None:
|
|
340
|
+
raise ValueError('tipo_pago required when using bank_code')
|
|
341
|
+
return self
|
|
342
|
+
|
|
333
343
|
|
|
334
344
|
class FraudValidationRequest(BaseModel):
|
|
335
345
|
amount: StrictPositiveInt
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
__version__ = '2.1.38.dev0'
|
|
@@ -248,6 +248,44 @@ def test_user_query_rejects_invalid(field, value):
|
|
|
248
248
|
UserQuery(**{field: value})
|
|
249
249
|
|
|
250
250
|
|
|
251
|
+
def test_user_query_accepts_ids_and_is_blocked():
|
|
252
|
+
query = UserQuery(ids='US1,US2', is_blocked=True)
|
|
253
|
+
assert query.ids == 'US1,US2'
|
|
254
|
+
assert query.is_blocked is True
|
|
255
|
+
dumped = query.model_dump(exclude_none=True)
|
|
256
|
+
assert dumped['ids'] == 'US1,US2'
|
|
257
|
+
assert dumped['is_blocked'] is True
|
|
258
|
+
assert isinstance(dumped['ids'], str)
|
|
259
|
+
|
|
260
|
+
|
|
261
|
+
def test_query_params_accepts_ids():
|
|
262
|
+
query = QueryParams(ids='US1,US2')
|
|
263
|
+
assert query.ids == 'US1,US2'
|
|
264
|
+
|
|
265
|
+
|
|
266
|
+
def test_query_params_ids_none():
|
|
267
|
+
query = QueryParams(ids=None)
|
|
268
|
+
assert query.ids is None
|
|
269
|
+
|
|
270
|
+
|
|
271
|
+
def test_user_query_rejects_ids_over_limit():
|
|
272
|
+
with pytest.raises(ValidationError):
|
|
273
|
+
UserQuery(ids=','.join(f'US{i}' for i in range(101)))
|
|
274
|
+
|
|
275
|
+
|
|
276
|
+
def test_user_query_ids_empty_string_ok():
|
|
277
|
+
query = UserQuery(ids='')
|
|
278
|
+
assert query.ids == ''
|
|
279
|
+
assert 'ids' in query.model_dump(exclude_none=True)
|
|
280
|
+
|
|
281
|
+
|
|
282
|
+
def test_user_query_ids_none_excluded_from_dump():
|
|
283
|
+
query = UserQuery()
|
|
284
|
+
dumped = query.model_dump(exclude_none=True)
|
|
285
|
+
assert 'ids' not in dumped
|
|
286
|
+
assert 'is_blocked' not in dumped
|
|
287
|
+
|
|
288
|
+
|
|
251
289
|
def test_exclude_none_in_dict():
|
|
252
290
|
request = ApiKeyUpdateRequest(user_id='US123')
|
|
253
291
|
assert request.model_dump() == dict(user_id='US123')
|
|
@@ -637,31 +675,62 @@ def test_bank_account_validation_clabe_request():
|
|
|
637
675
|
assert BankAccountValidationRequest(account_number='646180157098510917')
|
|
638
676
|
|
|
639
677
|
|
|
640
|
-
|
|
641
|
-
|
|
642
|
-
|
|
643
|
-
|
|
644
|
-
|
|
645
|
-
|
|
646
|
-
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
|
|
651
|
-
|
|
652
|
-
|
|
653
|
-
|
|
654
|
-
|
|
678
|
+
@pytest.mark.parametrize(
|
|
679
|
+
'data, expected_dump',
|
|
680
|
+
[
|
|
681
|
+
(
|
|
682
|
+
dict(
|
|
683
|
+
user_id='US123',
|
|
684
|
+
clabe='646180157098510917',
|
|
685
|
+
amount=10000,
|
|
686
|
+
concepto=' Devolución fraude ',
|
|
687
|
+
),
|
|
688
|
+
dict(
|
|
689
|
+
user_id='US123',
|
|
690
|
+
clabe='646180157098510917',
|
|
691
|
+
amount=10000,
|
|
692
|
+
concepto='Devolución fraude',
|
|
693
|
+
),
|
|
694
|
+
),
|
|
695
|
+
(
|
|
696
|
+
dict(user_id='US123', clabe='646180157098510917'),
|
|
697
|
+
dict(user_id='US123', clabe='646180157098510917'),
|
|
698
|
+
),
|
|
699
|
+
(
|
|
700
|
+
dict(
|
|
701
|
+
user_id='US123',
|
|
702
|
+
bank_code='40012',
|
|
703
|
+
tipo_pago=1,
|
|
704
|
+
amount=5000,
|
|
705
|
+
),
|
|
706
|
+
dict(
|
|
707
|
+
user_id='US123',
|
|
708
|
+
bank_code='40012',
|
|
709
|
+
tipo_pago=1,
|
|
710
|
+
amount=5000,
|
|
711
|
+
),
|
|
712
|
+
),
|
|
713
|
+
],
|
|
714
|
+
)
|
|
715
|
+
def test_fraud_funds_transfer_request(data, expected_dump):
|
|
716
|
+
request = FraudFundsTransferRequest(**data)
|
|
717
|
+
assert request.model_dump() == expected_dump
|
|
655
718
|
|
|
656
|
-
request_full_balance = FraudFundsTransferRequest(
|
|
657
|
-
user_id='US123',
|
|
658
|
-
clabe='646180157098510917',
|
|
659
|
-
)
|
|
660
719
|
|
|
661
|
-
|
|
662
|
-
|
|
663
|
-
|
|
664
|
-
|
|
720
|
+
@pytest.mark.parametrize(
|
|
721
|
+
'data, expected_error',
|
|
722
|
+
[
|
|
723
|
+
(dict(user_id='US123'), 'clabe or bank_code required'),
|
|
724
|
+
(
|
|
725
|
+
dict(user_id='US123', bank_code='40012'),
|
|
726
|
+
'tipo_pago required when using bank_code',
|
|
727
|
+
),
|
|
728
|
+
],
|
|
729
|
+
)
|
|
730
|
+
def test_fraud_funds_transfer_request_invalid(data, expected_error):
|
|
731
|
+
with pytest.raises(ValidationError) as exc:
|
|
732
|
+
FraudFundsTransferRequest(**data)
|
|
733
|
+
assert expected_error in str(exc.value)
|
|
665
734
|
|
|
666
735
|
|
|
667
736
|
@pytest.mark.parametrize(
|
|
@@ -1 +0,0 @@
|
|
|
1
|
-
__version__ = '2.1.36.dev3'
|
|
File without changes
|
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/__init__.py
RENAMED
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/card_bins.py
RENAMED
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/errors.py
RENAMED
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/py.typed
RENAMED
|
File without changes
|
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/card.py
RENAMED
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/enums.py
RENAMED
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/files.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/types/morals.py
RENAMED
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/typing.py
RENAMED
|
File without changes
|
{cuenca_validations-2.1.36.dev3 → cuenca_validations-2.1.38.dev0}/cuenca_validations/validators.py
RENAMED
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|