cerberus_.py 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163
  1. '''
  2. Extend the cerberus library with more validation methods and french translations
  3. @author: olivier.massot, 2018
  4. '''
  5. from datetime import datetime
  6. import locale
  7. import re
  8. import cerberus
  9. def is_french_date(field, value, error):
  10. try:
  11. datetime.strptime(value, '%d/%m/%Y')
  12. except:
  13. error(field, 'Doit être une date au format jj/mm/aaaa: {}'.format(value))
  14. def is_modern_french_date(field, value, error):
  15. try:
  16. d = datetime.strptime(value, '%d/%m/%Y')
  17. if not d.year >= 2000:
  18. error(field, "La date ne peut pas être antérieure à l'an 2000: {}".format(value))
  19. except:
  20. error(field, 'Doit être une date au format jj/mm/aaaa: {}'.format(value))
  21. def is_int(field, value, error):
  22. try:
  23. if float(value) != int(value):
  24. error(field, 'Doit être un nombre entier: {}'.format(value))
  25. except (TypeError, ValueError):
  26. error(field, 'Doit être un nombre entier: {}'.format(value))
  27. def is_positive_int(field, value, error):
  28. try:
  29. if float(value) != int(value) or int(value) < 0:
  30. error(field, 'Doit être un nombre entier positif: {}'.format(value))
  31. except (TypeError, ValueError):
  32. error(field, 'Doit être un nombre entier positif: {}'.format(value))
  33. def is_strictly_positive_int(field, value, error):
  34. try:
  35. if float(value) != int(value) or int(value) <= 0:
  36. error(field, 'Doit être un nombre entier positif: {}'.format(value))
  37. except (TypeError, ValueError):
  38. error(field, 'Doit être un nombre entier positif: {}'.format(value))
  39. def is_multi_int(field, value, error):
  40. if not re.match(r"\d(\s?-\s?\d)*", str(value)):
  41. error(field, 'Doit être un nombre entier, ou une liste de nombres séparés par des tirets: {}'.format(value))
  42. def _tofloat(val):
  43. return float(str(val).replace(",", "."))
  44. def is_float(field, value, error):
  45. try:
  46. _ = _tofloat(value)
  47. except ValueError:
  48. error(field, 'Doit être un nombre décimal ({})'.format(value))
  49. def is_positive_float(field, value, error):
  50. try:
  51. value = _tofloat(value)
  52. if float(value) < 0:
  53. error(field, 'Doit être un nombre décimal positif ({})'.format(value))
  54. except ValueError:
  55. error(field, 'Doit être un nombre décimal positif ({})'.format(value))
  56. def is_strictly_positive_float(field, value, error):
  57. try:
  58. value = _tofloat(value)
  59. if float(value) <= 0:
  60. error(field, 'Doit être un nombre décimal positif ({})'.format(value))
  61. except ValueError:
  62. error(field, 'Doit être un nombre décimal positif ({})'.format(value))
  63. # Ref: http://docs.python-cerberus.org/en/stable/api.html#error-codes
  64. class CerberusValidator(cerberus.validator.Validator):
  65. def __init__(self, *args, **kwargs):
  66. super().__init__(*args, **kwargs)
  67. # Rends tous les champs requis par défaut, à moins que 'required' ait été défini dans le schéma
  68. for field in self.schema:
  69. if not 'required' in self.schema[field]:
  70. self.schema[field]['required'] = True
  71. def _validate_contains_any_of(self, iterable, field, value):
  72. """ Controle que la chaine contient l'une des substring contenue dans l'iterable
  73. The rule's arguments are validated against this schema:
  74. {'type': 'list'}
  75. """
  76. if not re.search(r".*{}.*".format("|".join(iterable)), value):
  77. self._error(field, "Doit contenir un de ces éléments: {} ('{}')".format(", ".join(iterable), value))
  78. def _validate_multiallowed(self, allowed, field, value):
  79. """ Comme 'allowed', mais autorise plusieurs valeurs séparées par un '-'
  80. The rule's arguments are validated against this schema:
  81. {'type': 'list'}
  82. """
  83. for item in re.split("\s?-\s?", value or ""):
  84. if not item in allowed:
  85. if item == '':
  86. self._error(field, "Le champs doit être renseigné")
  87. else:
  88. self._error(field, "Valeur non-autorisée: {}".format(item))
  89. def _translate_messages(message):
  90. message = message.replace("string", "texte")
  91. message = message.replace("integer", "nombre entier")
  92. message = message.replace("float", "nombre décimal")
  93. message = message.replace("boolean", "booléen")
  94. return message
  95. class CerberusErrorHandler(cerberus.errors.BasicErrorHandler):
  96. messages = {0x00: "{0}",
  97. 0x01: "Le document est manquant",
  98. 0x02: "Le champs est obligatoire",
  99. 0x03: "Champs inconnu",
  100. 0x04: "Le champs '{0}' est obligatoire",
  101. 0x05: "Dépends de ces valeurs: {constraint}",
  102. 0x06: "{0} ne doit pas être présent avec '{field}'",
  103. 0x21: "'{0}' is not a document, must be a dict",
  104. 0x22: "Le champs doit être renseigné",
  105. 0x23: "Les valeurs NULL ne sont pas autorisées",
  106. 0x24: "Doit être du type {constraint}: {value}",
  107. 0x25: "Doit être de type 'dictionnaire': {value}",
  108. 0x26: "La longueur de la liste doit être de {constraint}, elle est de {0}",
  109. 0x27: "La longueur minimum du champs est de {constraint}: {value}",
  110. 0x28: "Trop long, la longueur max. du champs est de {constraint}: {value}",
  111. 0x41: "La valeur n'est pas au bon format ('{constraint}')",
  112. 0x42: "La valeur minimum autorisée est {constraint}",
  113. 0x43: "La valeur maximum autorisée est {constraint}",
  114. 0x44: "Valeur non-autorisée: {value}",
  115. 0x45: "Valeurs non-autorisées: {0}",
  116. 0x46: "Valeur non-autorisée: {value}",
  117. 0x47: "Valeurs non-autorisées: {0}",
  118. 0x61: "Le champs '{field}' ne peut pas être converti: {0}",
  119. 0x62: "Le champs '{field}' ne peut pas être renommé: {0}",
  120. 0x63: "Le champs est en lecture seule",
  121. 0x64: "La valeur par défaut du champs '{field}' ne peut pas être appliquée: {0}",
  122. 0x81: "mapping doesn't validate subschema: {0}",
  123. 0x82: "one or more sequence-items don't validate: {0}",
  124. 0x83: "one or more keys of a mapping don't validate: {0}",
  125. 0x84: "one or more values in a mapping don't validate: {0}",
  126. 0x85: "one or more sequence-items don't validate: {0}",
  127. 0x91: "one or more definitions validate",
  128. 0x92: "none or more than one rule validate",
  129. 0x93: "no definitions validate",
  130. 0x94: "one or more definitions don't validate"
  131. }