cuenca-validations 2.1.36.dev3__py3-none-any.whl → 2.1.38__py3-none-any.whl

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.
@@ -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
@@ -1,7 +1,7 @@
1
1
  import datetime as dt
2
2
  from typing import Annotated, Any, Optional, Union
3
3
 
4
- from clabe import Clabe
4
+ from clabe import BANK_NAMES, Clabe
5
5
  from pydantic import (
6
6
  BaseModel,
7
7
  ConfigDict,
@@ -326,10 +326,24 @@ 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
330
331
  amount: Optional[StrictPositiveInt] = None
331
332
  concepto: Optional[NonEmptyStr] = None
332
333
 
334
+ @field_validator('bank_code')
335
+ @classmethod
336
+ def validate_bank_code(cls, bank_code: Optional[str]) -> Optional[str]:
337
+ if bank_code is not None and bank_code not in BANK_NAMES:
338
+ raise ValueError('Not a valid bank code')
339
+ return bank_code
340
+
341
+ @model_validator(mode='after')
342
+ def validate_destination(self) -> 'FraudFundsTransferRequest':
343
+ if self.clabe is None and self.bank_code is None:
344
+ raise ValueError('clabe or bank_code required')
345
+ return self
346
+
333
347
 
334
348
  class FraudValidationRequest(BaseModel):
335
349
  amount: StrictPositiveInt
@@ -1 +1 @@
1
- __version__ = '2.1.36.dev3'
1
+ __version__ = '2.1.38'
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: cuenca_validations
3
- Version: 2.1.36.dev3
3
+ Version: 2.1.38
4
4
  Summary: Cuenca common validations
5
5
  Home-page: https://github.com/cuenca-mx/cuenca-validations
6
6
  Author: Cuenca
@@ -4,7 +4,7 @@ cuenca_validations/errors.py,sha256=OtM8EgiKqYdz9Hn66AbBO96orL1or7efkyt0vh0Zxbs,
4
4
  cuenca_validations/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
5
5
  cuenca_validations/typing.py,sha256=1QCu81IbVZZpyInjyeAuO-nF36gpT5Gi4o6V9PozuOU,204
6
6
  cuenca_validations/validators.py,sha256=zXSnU5-EMbQZBD-PfFXgP4Z6G7cm7KFDWXB7Nie6WUk,2682
7
- cuenca_validations/version.py,sha256=Hr0593SCkGfAV0-XJN8kmgux2qLlqiXP9Mm-RL3Lqy0,28
7
+ cuenca_validations/version.py,sha256=WQ6ee1JenmJovO-9nOpQ6FB-YQ8tWxhJhMmu1_crDFQ,23
8
8
  cuenca_validations/types/__init__.py,sha256=BZNEJKcQ6dlIVtOWrU-7hAFs9j_-Qi5V_VuG9FLjz3w,5639
9
9
  cuenca_validations/types/card.py,sha256=UGzz8NTFAverUmdUKAK1oGHnOnjSNTpIRUm93vKSSGY,1295
10
10
  cuenca_validations/types/enums.py,sha256=f-qMUdxLZsAqRvEwh3EcUsB2K-loufvPzdcK0JFQ1JY,20586
@@ -13,18 +13,18 @@ cuenca_validations/types/general.py,sha256=eYFYwyx_a4_J480GYpqW3DFbZabDFcUjvLRMQ
13
13
  cuenca_validations/types/helpers.py,sha256=4veeLZbugHHqZk0ezSim978VhH6Vq8XTrEhe1eE_r3A,1531
14
14
  cuenca_validations/types/identities.py,sha256=j2xxh3UYHJe6IbAwr9yNXJkebTth_-g3SUmHeiPez8M,5513
15
15
  cuenca_validations/types/morals.py,sha256=davabh5hAnFVQyM7-yCyDaT5ewXnm0cr1BtqDIwzkX8,1833
16
- cuenca_validations/types/queries.py,sha256=iVr6Z8ahXon0rlqQLu7aqRY6WtRxkN-1C7A2zeVt2-4,5314
17
- cuenca_validations/types/requests.py,sha256=rKfxJrCKVKAdYtXfy_sV8ZozuHC9eCSHrReSmVmPcXo,24638
18
- cuenca_validations-2.1.36.dev3.dist-info/licenses/LICENSE,sha256=wR76FmxBbfnQpwELkkE5iMF8sFIafEMgXLTE4N4WPTc,1063
16
+ cuenca_validations/types/queries.py,sha256=OonJ6CBDR6OF_kTv49AsE3zLEzQPCF4xpSqzSBpQgEs,5795
17
+ cuenca_validations/types/requests.py,sha256=b9MEYOf2nMNdpnyoavEGhkipWLzUe2FzVwDMBoJEt0w,25217
18
+ cuenca_validations-2.1.38.dist-info/licenses/LICENSE,sha256=wR76FmxBbfnQpwELkkE5iMF8sFIafEMgXLTE4N4WPTc,1063
19
19
  tests/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
20
20
  tests/test_card.py,sha256=QAfRz7e11gWICPnFJZ2tiYgUsFV3C9TwzJXrDnDNXFw,1202
21
21
  tests/test_errors.py,sha256=ixiIgEuBuzfsL5p4uCFdF32XqFRtTPF6EVhGJ0keOrI,930
22
22
  tests/test_helpers.py,sha256=fx2R7y6oQO8XeQzevJe826NuHkBgvtznZicomchB3Gg,899
23
23
  tests/test_requests.py,sha256=3XMx2A-BK21g1Exm4RTVm6eKTOzeZH-8FzKJXQmmMmA,4141
24
24
  tests/test_statement.py,sha256=IOE0rRRBgBZSJv_FLaETEyn5NzzXKMNTqgjv99GX-68,1436
25
- tests/test_types.py,sha256=gHfT5b8Xz2n4EGK5o5KiaC4Mm8BelwKafmqGAvj6Y58,22158
25
+ tests/test_types.py,sha256=rPjqPGfPU73MSdbeJpoW_ai5TAO25hWSnQAt_H7ZIeI,23055
26
26
  tests/test_validators.py,sha256=Jjr9gWTT4cRntGiKvQK4fncqx3JkEuTWkKm1VqpRHTs,1829
27
- cuenca_validations-2.1.36.dev3.dist-info/METADATA,sha256=jRLL9UGbM6SJM7TI5BPVtgglCUpXurgEv8nBcZF8ry0,1600
28
- cuenca_validations-2.1.36.dev3.dist-info/WHEEL,sha256=aeYiig01lYGDzBgS8HxWXOg3uV61G9ijOsup-k9o1sk,91
29
- cuenca_validations-2.1.36.dev3.dist-info/top_level.txt,sha256=4233xdOs2HtuT-GFRjcDcwK0IwdwvWdczOtk0fPB6Gw,25
30
- cuenca_validations-2.1.36.dev3.dist-info/RECORD,,
27
+ cuenca_validations-2.1.38.dist-info/METADATA,sha256=tqPfwvoClfYQkWSa7Dwp12VdBKZ8vLp8M242UDCdNos,1595
28
+ cuenca_validations-2.1.38.dist-info/WHEEL,sha256=aeYiig01lYGDzBgS8HxWXOg3uV61G9ijOsup-k9o1sk,91
29
+ cuenca_validations-2.1.38.dist-info/top_level.txt,sha256=4233xdOs2HtuT-GFRjcDcwK0IwdwvWdczOtk0fPB6Gw,25
30
+ cuenca_validations-2.1.38.dist-info/RECORD,,
tests/test_types.py CHANGED
@@ -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')
@@ -638,30 +676,18 @@ def test_bank_account_validation_clabe_request():
638
676
 
639
677
 
640
678
  def test_fraud_funds_transfer_request():
641
- request = FraudFundsTransferRequest(
679
+ assert FraudFundsTransferRequest(
642
680
  user_id='US123',
643
- clabe='646180157098510917',
644
- amount=10000,
645
- concepto=' Devolución fraude ',
646
- )
681
+ bank_code='40012',
682
+ ).model_dump() == {'user_id': 'US123', 'bank_code': '40012'}
647
683
 
648
- assert request.concepto == 'Devolución fraude'
649
- assert request.model_dump() == {
650
- 'user_id': 'US123',
651
- 'clabe': '646180157098510917',
652
- 'amount': 10000,
653
- 'concepto': 'Devolución fraude',
654
- }
655
-
656
- request_full_balance = FraudFundsTransferRequest(
657
- user_id='US123',
658
- clabe='646180157098510917',
659
- )
684
+ with pytest.raises(ValidationError) as exc:
685
+ FraudFundsTransferRequest(user_id='US123')
686
+ assert 'clabe or bank_code required' in str(exc.value)
660
687
 
661
- assert request_full_balance.model_dump() == {
662
- 'user_id': 'US123',
663
- 'clabe': '646180157098510917',
664
- }
688
+ with pytest.raises(ValidationError) as exc:
689
+ FraudFundsTransferRequest(user_id='US123', bank_code='99999')
690
+ assert 'Not a valid bank code' in str(exc.value)
665
691
 
666
692
 
667
693
  @pytest.mark.parametrize(