cerberus_extend.py 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152
  1. '''
  2. @author: olivier.massot, 2018
  3. '''
  4. from datetime import datetime
  5. import locale
  6. import re
  7. import cerberus
  8. from core import gis
  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 ValueError:
  26. error(field, 'Doit être un nombre entier: {}'.format(value))
  27. def is_float(field, value, error):
  28. try:
  29. value = locale.atof(str(value))
  30. if value != float(value):
  31. error(field, 'Doit être un nombre décimal ({})'.format(value))
  32. except ValueError:
  33. error(field, 'Doit être un nombre décimal ({})'.format(value))
  34. # Ref: http://docs.python-cerberus.org/en/stable/api.html#error-codes
  35. class ExtendedValidator(cerberus.validator.Validator):
  36. def __init__(self, *args, **kwargs):
  37. super(ExtendedValidator, self).__init__(*args, **kwargs)
  38. # Rends tous les champs requis par défaut, à moins que 'required' ait été défini dans le schéma
  39. for field in self.schema:
  40. if not 'required' in self.schema[field]:
  41. self.schema[field]['required'] = True
  42. def _validate_contains_any_of(self, iterable, field, value):
  43. """ Controle que la chaine contient l'une des substring contenue dans l'iterable
  44. The rule's arguments are validated against this schema:
  45. {'type': 'list'}
  46. """
  47. if not re.search(r".*{}.*".format("|".join(iterable)), value):
  48. self._error(field, "Doit contenir un de ces éléments: {} ('{}')".format(", ".join(iterable), value))
  49. def _validate_multiallowed(self, allowed, field, value):
  50. """ Comme 'allowed', mais autorise plusieurs valeurs séparées par un '-'
  51. The rule's arguments are validated against this schema:
  52. {'type': 'list'}
  53. """
  54. for item in re.split("\s?-\s?", value):
  55. if not item in allowed:
  56. self._error(field, "Valeur non-autorisée: {}".format(item))
  57. class GeoValidator(ExtendedValidator):
  58. def _validate_geometry(self, geom, field, value):
  59. """ Contrôle la géométrie d'un objet
  60. The rule's arguments are validated against this schema:
  61. {'type': 'list', 'items': [{'type':'integer'},{'type': 'list'}]}
  62. """
  63. shapeType, bbox = geom
  64. if value.shapeType != shapeType:
  65. self._error(field, "Le type de géométrie est {} (attendu: {})".format(gis.SHAPE_NAMES[value.shapeType], gis.SHAPE_NAMES[shapeType]))
  66. xmin, ymin, xmax, ymax = bbox
  67. try:
  68. x1, y1, x2, y2 = value.bbox
  69. if any(x < xmin or x > xmax for x in (x1, x2)) or \
  70. any(y < ymin or y > ymax for y in (y1, y2)):
  71. self._error(field, "Certaines coordonnées hors de l'emprise autorisée")
  72. except AttributeError:
  73. pass
  74. def _translate_messages(message):
  75. message = message.replace("string", "texte")
  76. message = message.replace("integer", "nombre entier")
  77. message = message.replace("float", "nombre décimal")
  78. message = message.replace("boolean", "booléen")
  79. return message
  80. class CerberusErrorHandler(cerberus.errors.BasicErrorHandler):
  81. messages = {0x00: "{0}",
  82. 0x01: "Le document est manquant",
  83. 0x02: "Le champs est obligatoire",
  84. 0x03: "Champs inconnu",
  85. 0x04: "Le champs '{0}' est obligatoire",
  86. 0x05: "Dépends de ces valeurs: {constraint}",
  87. 0x06: "{0} ne doit pas être présent avec '{field}'",
  88. 0x21: "'{0}' is not a document, must be a dict",
  89. 0x22: "Le champs doit être renseigné",
  90. 0x23: "Les valeurs NULL ne sont pas autorisées",
  91. 0x24: "Doit être du type {constraint}: {value}",
  92. 0x25: "Doit être de type 'dictionnaire': {value}",
  93. 0x26: "La longueur de la liste doit être de {constraint}, elle est de {0}",
  94. 0x27: "La longueur minimum du champs est de {constraint}: {value}",
  95. 0x28: "Trop long, la longueur max. du champs est de {constraint}: {value}",
  96. 0x41: "La valeur n'est pas au bon format ('{constraint}')",
  97. 0x42: "La valeur minimum autorisée est {constraint}",
  98. 0x43: "La valeur maximum autorisée est {constraint}",
  99. 0x44: "Valeur non-autorisée: {value}",
  100. 0x45: "Valeurs non-autorisées: {0}",
  101. 0x46: "Valeur non-autorisée: {value}",
  102. 0x47: "Valeurs non-autorisées: {0}",
  103. 0x61: "Le champs '{field}' ne peut pas être converti: {0}",
  104. 0x62: "Le champs '{field}' ne peut pas être renommé: {0}",
  105. 0x63: "Le champs est en lecture seule",
  106. 0x64: "La valeur par défaut du champs '{field}' ne peut pas être appliquée: {0}",
  107. 0x81: "mapping doesn't validate subschema: {0}",
  108. 0x82: "one or more sequence-items don't validate: {0}",
  109. 0x83: "one or more keys of a mapping don't validate: {0}",
  110. 0x84: "one or more values in a mapping don't validate: {0}",
  111. 0x85: "one or more sequence-items don't validate: {0}",
  112. 0x91: "one or more definitions validate",
  113. 0x92: "none or more than one rule validate",
  114. 0x93: "no definitions validate",
  115. 0x94: "one or more definitions don't validate"
  116. }