Skip to content

Commit 3dba977

Browse files
shmaxbighappyface
authored andcommitted
centralize errors (#364)
* centralize errors * isolate 'more' info * throw exception for missing error message * swap args
1 parent 72b94c1 commit 3dba977

15 files changed

+268
-70
lines changed

composer.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,8 @@
3636
}
3737
}],
3838
"require": {
39-
"php": ">=5.3.3"
39+
"php": ">=5.3.3",
40+
"marc-mabe/php-enum":"2.3.1"
4041
},
4142
"require-dev": {
4243
"json-schema/JSON-Schema-Test-Suite": "1.2.0",

src/JsonSchema/ConstraintError.php

+104
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
<?php
2+
3+
namespace JsonSchema;
4+
5+
class ConstraintError extends \MabeEnum\Enum
6+
{
7+
const ADDITIONAL_ITEMS = 'additionalItems';
8+
const ADDITIONAL_PROPERTIES = 'additionalProp';
9+
const ALL_OF = 'allOf';
10+
const ANY_OF = 'anyOf';
11+
const DEPENDENCIES = 'dependencies';
12+
const DISALLOW = 'disallow';
13+
const DIVISIBLE_BY = 'divisibleBy';
14+
const ENUM = 'enum';
15+
const EXCLUSIVE_MINIMUM = 'exclusiveMinimum';
16+
const EXCLUSIVE_MAXIMUM = 'exclusiveMaximum';
17+
const FORMAT_COLOR = 'colorFormat';
18+
const FORMAT_DATE = 'dateFormat';
19+
const FORMAT_DATE_TIME = 'dateTimeFormat';
20+
const FORMAT_DATE_UTC = 'dateUtcFormat';
21+
const FORMAT_EMAIL = 'emailFormat';
22+
const FORMAT_HOSTNAME = 'styleHostName';
23+
const FORMAT_IP = 'ipFormat';
24+
const FORMAT_PHONE = 'phoneFormat';
25+
const FORMAT_REGEX= 'regexFormat';
26+
const FORMAT_STYLE = 'styleFormat';
27+
const FORMAT_TIME = 'timeFormat';
28+
const FORMAT_URL = 'urlFormat';
29+
const LENGTH_MAX = 'maxLength';
30+
const LENGTH_MIN = 'minLength';
31+
const MAXIMUM = 'maximum';
32+
const MIN_ITEMS = 'minItems';
33+
const MINIMUM = 'minimum';
34+
const MISSING_MAXIMUM = 'missingMaximum';
35+
const MISSING_MINIMUM = 'missingMinimum';
36+
const MAX_ITEMS = 'maxItems';
37+
const MULTIPLE_OF = 'multipleOf';
38+
const NOT = 'not';
39+
const ONE_OF = 'oneOf';
40+
const REQUIRED = 'required';
41+
const REQUIRED_D3 = 'selfRequired';
42+
const REQUIRES = 'requires';
43+
const PATTERN = 'pattern';
44+
const PREGEX_INVALID = 'pregrex';
45+
const PROPERTIES_MIN = 'minProperties';
46+
const PROPERTIES_MAX = 'maxProperties';
47+
const TYPE = 'type';
48+
const UNIQUE_ITEMS = 'uniqueItems';
49+
50+
public function getMessage()
51+
{
52+
$name = $this->getValue();
53+
static $messages = array(
54+
self::ADDITIONAL_ITEMS => 'The item %s[%s] is not defined and the definition does not allow additional items',
55+
self::ADDITIONAL_PROPERTIES => 'The property %s is not defined and the definition does not allow additional properties',
56+
self::ALL_OF => 'Failed to match all schemas',
57+
self::ANY_OF => 'Failed to match at least one schema',
58+
self::DEPENDENCIES => '%s depends on %s, which is missing',
59+
self::DISALLOW => 'Disallowed value was matched',
60+
self::DIVISIBLE_BY => 'Is not divisible by %d',
61+
self::ENUM => 'Does not have a value in the enumeration %s',
62+
self::EXCLUSIVE_MINIMUM => 'Must have a minimum value greater than %d',
63+
self::EXCLUSIVE_MAXIMUM => 'Must have a maximum value less than %d',
64+
self::FORMAT_COLOR => 'Invalid color',
65+
self::FORMAT_DATE => 'Invalid date %s, expected format YYYY-MM-DD',
66+
self::FORMAT_DATE_TIME => 'Invalid date-time %s, expected format YYYY-MM-DDThh:mm:ssZ or YYYY-MM-DDThh:mm:ss+hh:mm',
67+
self::FORMAT_DATE_UTC => 'Invalid time %s, expected integer of milliseconds since Epoch',
68+
self::FORMAT_EMAIL => 'Invalid email',
69+
self::FORMAT_HOSTNAME => 'Invalid hostname',
70+
self::FORMAT_IP => 'Invalid IP address',
71+
self::FORMAT_PHONE => 'Invalid phone number',
72+
self::FORMAT_REGEX=> 'Invalid regex format %s',
73+
self::FORMAT_STYLE => 'Invalid style',
74+
self::FORMAT_TIME => 'Invalid time %s, expected format hh:mm:ss',
75+
self::FORMAT_URL => 'Invalid URL format',
76+
self::LENGTH_MAX => 'Must be at most %d characters long',
77+
self::LENGTH_MIN => 'Must be at least %d characters long',
78+
self::MAX_ITEMS => 'There must be a maximum of %d items in the array',
79+
self::MAXIMUM => 'Must have a maximum value less than or equal to %d',
80+
self::MIN_ITEMS => 'There must be a minimum of %d items in the array',
81+
self::MINIMUM => 'Must have a minimum value greater than or equal to %d',
82+
self::MISSING_MAXIMUM => 'Use of exclusiveMaximum requires presence of maximum',
83+
self::MISSING_MINIMUM => 'Use of exclusiveMinimum requires presence of minimum',
84+
self::MULTIPLE_OF => 'Must be a multiple of %d',
85+
self::NOT => 'Matched a schema which it should not',
86+
self::ONE_OF => 'Failed to match exactly one schema',
87+
self::REQUIRED => 'The property %s is required',
88+
self::REQUIRED_D3 => 'Is missing and it is required',
89+
self::REQUIRES => 'The presence of the property %s requires that %s also be present',
90+
self::PATTERN => 'Does not match the regex pattern %s',
91+
self::PREGEX_INVALID => 'The pattern %s is invalid',
92+
self::PROPERTIES_MIN => 'Must contain a minimum of %d properties',
93+
self::PROPERTIES_MAX => 'Must contain no more than %d properties',
94+
self::TYPE => '%s value found, but %s is required',
95+
self::UNIQUE_ITEMS => 'There are no duplicates allowed in the array'
96+
);
97+
98+
if (!isset($messages[$name])) {
99+
throw new InvalidArgumentException('Missing error message for ' . $name);
100+
}
101+
102+
return $messages[$name];
103+
}
104+
}

src/JsonSchema/Constraints/BaseConstraint.php

+15-7
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99

1010
namespace JsonSchema\Constraints;
1111

12+
use JsonSchema\ConstraintError;
1213
use JsonSchema\Entity\JsonPointer;
1314
use JsonSchema\Exception\ValidationException;
1415

@@ -36,23 +37,30 @@ public function __construct(Factory $factory = null)
3637
$this->factory = $factory ?: new Factory();
3738
}
3839

39-
public function addError(JsonPointer $path = null, $message, $constraint = '', array $more = null)
40+
public function addError(ConstraintError $constraint, JsonPointer $path = null, array $more = array())
4041
{
42+
$message = $constraint ? $constraint->getMessage() : '';
43+
$name = $constraint ? $constraint->getValue() : '';
4144
$error = array(
4245
'property' => $this->convertJsonPointerIntoPropertyPath($path ?: new JsonPointer('')),
4346
'pointer' => ltrim(strval($path ?: new JsonPointer('')), '#'),
44-
'message' => $message,
45-
'constraint' => $constraint,
47+
'message' => ucfirst(vsprintf($message, array_map(function ($val) {
48+
if (is_scalar($val)) {
49+
return $val;
50+
}
51+
52+
return json_encode($val);
53+
}, array_values($more)))),
54+
'constraint' => array(
55+
'name' => $name,
56+
'params' => $more
57+
)
4658
);
4759

4860
if ($this->factory->getConfig(Constraint::CHECK_MODE_EXCEPTIONS)) {
4961
throw new ValidationException(sprintf('Error validating %s: %s', $error['pointer'], $error['message']));
5062
}
5163

52-
if (is_array($more) && count($more) > 0) {
53-
$error += $more;
54-
}
55-
5664
$this->errors[] = $error;
5765
}
5866

src/JsonSchema/Constraints/CollectionConstraint.php

+12-4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99

1010
namespace JsonSchema\Constraints;
1111

12+
use JsonSchema\ConstraintError;
1213
use JsonSchema\Entity\JsonPointer;
1314

1415
/**
@@ -26,12 +27,12 @@ public function check(&$value, $schema = null, JsonPointer $path = null, $i = nu
2627
{
2728
// Verify minItems
2829
if (isset($schema->minItems) && count($value) < $schema->minItems) {
29-
$this->addError($path, 'There must be a minimum of ' . $schema->minItems . ' items in the array', 'minItems', array('minItems' => $schema->minItems));
30+
$this->addError(ConstraintError::MIN_ITEMS(), $path, array('minItems' => $schema->minItems));
3031
}
3132

3233
// Verify maxItems
3334
if (isset($schema->maxItems) && count($value) > $schema->maxItems) {
34-
$this->addError($path, 'There must be a maximum of ' . $schema->maxItems . ' items in the array', 'maxItems', array('maxItems' => $schema->maxItems));
35+
$this->addError(ConstraintError::MAX_ITEMS(), $path, array('maxItems' => $schema->maxItems));
3536
}
3637

3738
// Verify uniqueItems
@@ -43,7 +44,7 @@ public function check(&$value, $schema = null, JsonPointer $path = null, $i = nu
4344
}, $value);
4445
}
4546
if (count(array_unique($unique)) != count($value)) {
46-
$this->addError($path, 'There are no duplicates allowed in the array', 'uniqueItems');
47+
$this->addError(ConstraintError::UNIQUE_ITEMS(), $path);
4748
}
4849
}
4950

@@ -124,7 +125,14 @@ protected function validateItems(&$value, $schema = null, JsonPointer $path = nu
124125
$this->checkUndefined($v, $schema->additionalItems, $path, $k);
125126
} else {
126127
$this->addError(
127-
$path, 'The item ' . $i . '[' . $k . '] is not defined and the definition does not allow additional items', 'additionalItems', array('additionalItems' => $schema->additionalItems));
128+
ConstraintError::ADDITIONAL_ITEMS(),
129+
$path,
130+
array(
131+
'item' => $i,
132+
'property' => $k,
133+
'additionalItems' => $schema->additionalItems
134+
)
135+
);
128136
}
129137
} else {
130138
// Should be valid against an empty schema

src/JsonSchema/Constraints/ConstraintInterface.php

+5-5
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99

1010
namespace JsonSchema\Constraints;
1111

12+
use JsonSchema\ConstraintError;
1213
use JsonSchema\Entity\JsonPointer;
1314

1415
/**
@@ -35,12 +36,11 @@ public function addErrors(array $errors);
3536
/**
3637
* adds an error
3738
*
38-
* @param JsonPointer|null $path
39-
* @param string $message
40-
* @param string $constraint the constraint/rule that is broken, e.g.: 'minLength'
41-
* @param array $more more array elements to add to the error
39+
* @param ConstraintError $constraint the constraint/rule that is broken, e.g.: ConstraintErrors::LENGTH_MIN()
40+
* @param JsonPointer |null $path
41+
* @param array $more more array elements to add to the error
4242
*/
43-
public function addError(JsonPointer $path = null, $message, $constraint='', array $more = null);
43+
public function addError(ConstraintError $constraint, JsonPointer $path = null, array $more = array());
4444

4545
/**
4646
* checks if the validator has not raised errors

src/JsonSchema/Constraints/EnumConstraint.php

+2-1
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99

1010
namespace JsonSchema\Constraints;
1111

12+
use JsonSchema\ConstraintError;
1213
use JsonSchema\Entity\JsonPointer;
1314

1415
/**
@@ -49,6 +50,6 @@ public function check(&$element, $schema = null, JsonPointer $path = null, $i =
4950
}
5051
}
5152

52-
$this->addError($path, 'Does not have a value in the enumeration ' . json_encode($schema->enum), 'enum', array('enum' => $schema->enum));
53+
$this->addError(ConstraintError::ENUM(), $path, array('enum' => $schema->enum));
5354
}
5455
}

src/JsonSchema/Constraints/FormatConstraint.php

+32-13
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@
99

1010
namespace JsonSchema\Constraints;
1111

12+
use JsonSchema\ConstraintError;
1213
use JsonSchema\Entity\JsonPointer;
1314
use JsonSchema\Rfc3339;
1415

@@ -33,49 +34,67 @@ public function check(&$element, $schema = null, JsonPointer $path = null, $i =
3334
switch ($schema->format) {
3435
case 'date':
3536
if (!$date = $this->validateDateTime($element, 'Y-m-d')) {
36-
$this->addError($path, sprintf('Invalid date %s, expected format YYYY-MM-DD', json_encode($element)), 'format', array('format' => $schema->format));
37+
$this->addError(ConstraintError::FORMAT_DATE(), $path, array(
38+
'date' => $element,
39+
'format' => $schema->format
40+
)
41+
);
3742
}
3843
break;
3944

4045
case 'time':
4146
if (!$this->validateDateTime($element, 'H:i:s')) {
42-
$this->addError($path, sprintf('Invalid time %s, expected format hh:mm:ss', json_encode($element)), 'format', array('format' => $schema->format));
47+
$this->addError(ConstraintError::FORMAT_TIME(), $path, array(
48+
'time' => json_encode($element),
49+
'format' => $schema->format,
50+
)
51+
);
4352
}
4453
break;
4554

4655
case 'date-time':
4756
if (null === Rfc3339::createFromString($element)) {
48-
$this->addError($path, sprintf('Invalid date-time %s, expected format YYYY-MM-DDThh:mm:ssZ or YYYY-MM-DDThh:mm:ss+hh:mm', json_encode($element)), 'format', array('format' => $schema->format));
57+
$this->addError(ConstraintError::FORMAT_DATE_TIME(), $path, array(
58+
'dateTime' => json_encode($element),
59+
'format' => $schema->format
60+
)
61+
);
4962
}
5063
break;
5164

5265
case 'utc-millisec':
5366
if (!$this->validateDateTime($element, 'U')) {
54-
$this->addError($path, sprintf('Invalid time %s, expected integer of milliseconds since Epoch', json_encode($element)), 'format', array('format' => $schema->format));
67+
$this->addError(ConstraintError::FORMAT_DATE_UTC(), $path, array(
68+
'value' => $element,
69+
'format' => $schema->format));
5570
}
5671
break;
5772

5873
case 'regex':
5974
if (!$this->validateRegex($element)) {
60-
$this->addError($path, 'Invalid regex format ' . $element, 'format', array('format' => $schema->format));
75+
$this->addError(ConstraintError::FORMAT_REGEX(), $path, array(
76+
'value' => $element,
77+
'format' => $schema->format
78+
)
79+
);
6180
}
6281
break;
6382

6483
case 'color':
6584
if (!$this->validateColor($element)) {
66-
$this->addError($path, 'Invalid color', 'format', array('format' => $schema->format));
85+
$this->addError(ConstraintError::FORMAT_COLOR(), $path, array('format' => $schema->format));
6786
}
6887
break;
6988

7089
case 'style':
7190
if (!$this->validateStyle($element)) {
72-
$this->addError($path, 'Invalid style', 'format', array('format' => $schema->format));
91+
$this->addError(ConstraintError::FORMAT_STYLE(), $path, array('format' => $schema->format));
7392
}
7493
break;
7594

7695
case 'phone':
7796
if (!$this->validatePhone($element)) {
78-
$this->addError($path, 'Invalid phone number', 'format', array('format' => $schema->format));
97+
$this->addError(ConstraintError::FORMAT_PHONE(), $path, array('format' => $schema->format));
7998
}
8099
break;
81100

@@ -99,34 +118,34 @@ public function check(&$element, $schema = null, JsonPointer $path = null, $i =
99118
$validURL = null;
100119
}
101120
if ($validURL === null) {
102-
$this->addError($path, 'Invalid URL format', 'format', array('format' => $schema->format));
121+
$this->addError(ConstraintError::FORMAT_URL(), $path, array('format' => $schema->format));
103122
}
104123
}
105124
break;
106125

107126
case 'email':
108127
if (null === filter_var($element, FILTER_VALIDATE_EMAIL, FILTER_NULL_ON_FAILURE)) {
109-
$this->addError($path, 'Invalid email', 'format', array('format' => $schema->format));
128+
$this->addError(ConstraintError::FORMAT_EMAIL(), $path, array('format' => $schema->format));
110129
}
111130
break;
112131

113132
case 'ip-address':
114133
case 'ipv4':
115134
if (null === filter_var($element, FILTER_VALIDATE_IP, FILTER_NULL_ON_FAILURE | FILTER_FLAG_IPV4)) {
116-
$this->addError($path, 'Invalid IP address', 'format', array('format' => $schema->format));
135+
$this->addError(ConstraintError::FORMAT_IP(), $path, array('format' => $schema->format));
117136
}
118137
break;
119138

120139
case 'ipv6':
121140
if (null === filter_var($element, FILTER_VALIDATE_IP, FILTER_NULL_ON_FAILURE | FILTER_FLAG_IPV6)) {
122-
$this->addError($path, 'Invalid IP address', 'format', array('format' => $schema->format));
141+
$this->addError(ConstraintError::FORMAT_IP(), $path, array('format' => $schema->format));
123142
}
124143
break;
125144

126145
case 'host-name':
127146
case 'hostname':
128147
if (!$this->validateHostname($element)) {
129-
$this->addError($path, 'Invalid hostname', 'format', array('format' => $schema->format));
148+
$this->addError(ConstraintError::FORMAT_HOSTNAME(), $path, array('format' => $schema->format));
130149
}
131150
break;
132151

0 commit comments

Comments
 (0)