Passed
Push — dev ( 715dc9...eea5c3 )
by Marcin
10:55 queued 58s
created
src/lang/de/builder.php 1 patch
Indentation   +50 added lines, -50 removed lines patch added patch discarded remove patch
@@ -10,58 +10,58 @@
 block discarded – undo
10 10
  */
11 11
 return [
12 12
 
13
-    'ok'                       => 'OK',
14
-    'no_error_message'         => 'Fehler #:api_code',
13
+	'ok'                       => 'OK',
14
+	'no_error_message'         => 'Fehler #:api_code',
15 15
 
16
-    // Used by Exception Handler Helper (when used)
17
-    'uncaught_exception'       => 'Ungefangene Ausnahme: :message',
18
-    'http_exception'           => 'HTTP Ausnahme: :message',
16
+	// Used by Exception Handler Helper (when used)
17
+	'uncaught_exception'       => 'Ungefangene Ausnahme: :message',
18
+	'http_exception'           => 'HTTP Ausnahme: :message',
19 19
 
20
-    // HttpException handler (added in 6.4.0)
21
-    // Error messages for HttpException caught w/o custom messages
22
-    // https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
23
-    //
24
-    // German translation based on https://wiki.selfhtml.org/wiki/HTTP/Statuscodes
25
-    'http_400'                 => 'Ungültige Anfrage',
26
-    'http_401'                 => 'Unautorisiert',
27
-    'http_402'                 => 'Bezahlung benötigt',
28
-    'http_403'                 => 'Verboten',
29
-    'http_404'                 => 'Nicht gefunden',
30
-    'http_405'                 => 'Methode nicht erlaubt',
31
-    'http_406'                 => 'Nicht akzeptabel',
32
-    'http_407'                 => 'Proxy-Authentifizierung benötigt',
33
-    'http_408'                 => 'Anfrage-Zeitüberschreitung',
34
-    'http_409'                 => 'Konflikt',
35
-    'http_410'                 => 'Verschwunden',
36
-    'http_411'                 => 'Länge benötigt',
37
-    'http_412'                 => 'Vorbedingung missglückt',
38
-    'http_413'                 => 'Anfrage-Entität zu groß',
39
-    'http_414'                 => 'Anfrage-URI zu lang',
40
-    'http_415'                 => 'Nicht unterstützter Medientyp',
41
-    'http_416'                 => 'Anfrage-Bereich nicht erfüllbar',
42
-    'http_417'                 => 'Erwartung missglückt',
43
-    'http_421'                 => 'Fehlgeleitete Anforderung',
44
-    'http_422'                 => 'Kann nicht verarbeitet werden',
45
-    'http_423'                 => 'Gesperrt',
46
-    'http_424'                 => 'Vorhergehende Bedingung nicht erfüllt',
47
-    'http_425'                 => 'Too Early',  // FIXME
48
-    'http_426'                 => 'Update benötigt',
49
-    'http_428'                 => ' Vorbedingung benötigt',
50
-    'http_429'                 => 'Zu viele Anfragen',
51
-    'http_431'                 => 'Headerfelds zu groß',
52
-    'http_451'                 => 'Ressource aus rechtlichen Gründen nicht verfügbar',
20
+	// HttpException handler (added in 6.4.0)
21
+	// Error messages for HttpException caught w/o custom messages
22
+	// https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
23
+	//
24
+	// German translation based on https://wiki.selfhtml.org/wiki/HTTP/Statuscodes
25
+	'http_400'                 => 'Ungültige Anfrage',
26
+	'http_401'                 => 'Unautorisiert',
27
+	'http_402'                 => 'Bezahlung benötigt',
28
+	'http_403'                 => 'Verboten',
29
+	'http_404'                 => 'Nicht gefunden',
30
+	'http_405'                 => 'Methode nicht erlaubt',
31
+	'http_406'                 => 'Nicht akzeptabel',
32
+	'http_407'                 => 'Proxy-Authentifizierung benötigt',
33
+	'http_408'                 => 'Anfrage-Zeitüberschreitung',
34
+	'http_409'                 => 'Konflikt',
35
+	'http_410'                 => 'Verschwunden',
36
+	'http_411'                 => 'Länge benötigt',
37
+	'http_412'                 => 'Vorbedingung missglückt',
38
+	'http_413'                 => 'Anfrage-Entität zu groß',
39
+	'http_414'                 => 'Anfrage-URI zu lang',
40
+	'http_415'                 => 'Nicht unterstützter Medientyp',
41
+	'http_416'                 => 'Anfrage-Bereich nicht erfüllbar',
42
+	'http_417'                 => 'Erwartung missglückt',
43
+	'http_421'                 => 'Fehlgeleitete Anforderung',
44
+	'http_422'                 => 'Kann nicht verarbeitet werden',
45
+	'http_423'                 => 'Gesperrt',
46
+	'http_424'                 => 'Vorhergehende Bedingung nicht erfüllt',
47
+	'http_425'                 => 'Too Early',  // FIXME
48
+	'http_426'                 => 'Update benötigt',
49
+	'http_428'                 => ' Vorbedingung benötigt',
50
+	'http_429'                 => 'Zu viele Anfragen',
51
+	'http_431'                 => 'Headerfelds zu groß',
52
+	'http_451'                 => 'Ressource aus rechtlichen Gründen nicht verfügbar',
53 53
 
54
-    'http_500'                 => 'Interner Server-Fehler',
55
-    'http_501'                 => 'Nicht implementiert',
56
-    'http_502'                 => 'Schlechtes Portal',
57
-    'http_503'                 => 'Dienst nicht verfügbar',
58
-    'http_504'                 => 'Portal-Auszeit',
59
-    'http_505'                 => 'HTTP-Version nicht unterstützt',
60
-    'http_506'                 => 'Variant Also Negotiates',  // FIXME
61
-    'http_507'                 => 'Speicher des Servers reicht nicht aus',
62
-    'http_508'                 => 'Endlosschleife',
63
-    'http_509'                 => 'Unassigned',     // FIXME
64
-    'http_510'                 => 'Zu wenig Informationen',
65
-    'http_511'                 => 'Identizifierung benötigt',
54
+	'http_500'                 => 'Interner Server-Fehler',
55
+	'http_501'                 => 'Nicht implementiert',
56
+	'http_502'                 => 'Schlechtes Portal',
57
+	'http_503'                 => 'Dienst nicht verfügbar',
58
+	'http_504'                 => 'Portal-Auszeit',
59
+	'http_505'                 => 'HTTP-Version nicht unterstützt',
60
+	'http_506'                 => 'Variant Also Negotiates',  // FIXME
61
+	'http_507'                 => 'Speicher des Servers reicht nicht aus',
62
+	'http_508'                 => 'Endlosschleife',
63
+	'http_509'                 => 'Unassigned',     // FIXME
64
+	'http_510'                 => 'Zu wenig Informationen',
65
+	'http_511'                 => 'Identizifierung benötigt',
66 66
 ];
67 67
 
Please login to merge, or discard this patch.
src/lang/en/builder.php 1 patch
Indentation   +48 added lines, -48 removed lines patch added patch discarded remove patch
@@ -10,56 +10,56 @@
 block discarded – undo
10 10
  */
11 11
 return [
12 12
 
13
-    'ok'                       => 'OK',
14
-    'no_error_message'         => 'Error #:api_code',
13
+	'ok'                       => 'OK',
14
+	'no_error_message'         => 'Error #:api_code',
15 15
 
16
-    // Used by Exception Handler Helper (when used)
17
-    'uncaught_exception'       => 'Uncaught exception: :message',
18
-    'http_exception'           => 'HTTP exception: :message',
16
+	// Used by Exception Handler Helper (when used)
17
+	'uncaught_exception'       => 'Uncaught exception: :message',
18
+	'http_exception'           => 'HTTP exception: :message',
19 19
 
20
-    // HttpException handler (added in 6.4.0)
21
-    // Error messages for HttpException caught w/o custom messages
22
-    // https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
23
-    'http_400'                 => 'Bad Request',
24
-    'http_401'                 => 'Unauthorized',
25
-    'http_402'                 => 'Payment Required',
26
-    'http_403'                 => 'Forbidden',
27
-    'http_404'                 => 'Not Found',
28
-    'http_405'                 => 'Method Not Allowed',
29
-    'http_406'                 => 'Not Acceptable',
30
-    'http_407'                 => 'Proxy Authentication Required',
31
-    'http_408'                 => 'Request Timeout',
32
-    'http_409'                 => 'Conflict',
33
-    'http_410'                 => 'Gone',
34
-    'http_411'                 => 'Length Required',
35
-    'http_412'                 => 'Precondition Failed',
36
-    'http_413'                 => 'Payload Too Large',
37
-    'http_414'                 => 'URI Too Long',
38
-    'http_415'                 => 'Unsupported Media Type',
39
-    'http_416'                 => 'Range Not Satisfiable',
40
-    'http_417'                 => 'Expectation Failed',
41
-    'http_421'                 => 'Misdirected Request',
42
-    'http_422'                 => 'Unprocessable Entity',
43
-    'http_423'                 => 'Locked',
44
-    'http_424'                 => 'Failed Dependency',
45
-    'http_425'                 => 'Too Early',
46
-    'http_426'                 => 'Upgrade Required',
47
-    'http_428'                 => 'Precondition Required',
48
-    'http_429'                 => 'Too Many Requests',
49
-    'http_431'                 => 'Request Header Fields Too Large',
50
-    'http_451'                 => 'Unavailable For Legal Reasons',
20
+	// HttpException handler (added in 6.4.0)
21
+	// Error messages for HttpException caught w/o custom messages
22
+	// https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
23
+	'http_400'                 => 'Bad Request',
24
+	'http_401'                 => 'Unauthorized',
25
+	'http_402'                 => 'Payment Required',
26
+	'http_403'                 => 'Forbidden',
27
+	'http_404'                 => 'Not Found',
28
+	'http_405'                 => 'Method Not Allowed',
29
+	'http_406'                 => 'Not Acceptable',
30
+	'http_407'                 => 'Proxy Authentication Required',
31
+	'http_408'                 => 'Request Timeout',
32
+	'http_409'                 => 'Conflict',
33
+	'http_410'                 => 'Gone',
34
+	'http_411'                 => 'Length Required',
35
+	'http_412'                 => 'Precondition Failed',
36
+	'http_413'                 => 'Payload Too Large',
37
+	'http_414'                 => 'URI Too Long',
38
+	'http_415'                 => 'Unsupported Media Type',
39
+	'http_416'                 => 'Range Not Satisfiable',
40
+	'http_417'                 => 'Expectation Failed',
41
+	'http_421'                 => 'Misdirected Request',
42
+	'http_422'                 => 'Unprocessable Entity',
43
+	'http_423'                 => 'Locked',
44
+	'http_424'                 => 'Failed Dependency',
45
+	'http_425'                 => 'Too Early',
46
+	'http_426'                 => 'Upgrade Required',
47
+	'http_428'                 => 'Precondition Required',
48
+	'http_429'                 => 'Too Many Requests',
49
+	'http_431'                 => 'Request Header Fields Too Large',
50
+	'http_451'                 => 'Unavailable For Legal Reasons',
51 51
 
52
-    'http_500'                 => 'Internal Server Error',
53
-    'http_501'                 => 'Not Implemented',
54
-    'http_502'                 => 'Bad Gateway',
55
-    'http_503'                 => 'Service Unavailable',
56
-    'http_504'                 => 'Gateway Timeout',
57
-    'http_505'                 => 'HTTP Version Not Supported',
58
-    'http_506'                 => 'Variant Also Negotiates',
59
-    'http_507'                 => 'Insufficient Storage',
60
-    'http_508'                 => 'Loop Detected',
61
-    'http_509'                 => 'Unassigned',
62
-    'http_510'                 => 'Not Extended',
63
-    'http_511'                 => 'Network Authentication Required',
52
+	'http_500'                 => 'Internal Server Error',
53
+	'http_501'                 => 'Not Implemented',
54
+	'http_502'                 => 'Bad Gateway',
55
+	'http_503'                 => 'Service Unavailable',
56
+	'http_504'                 => 'Gateway Timeout',
57
+	'http_505'                 => 'HTTP Version Not Supported',
58
+	'http_506'                 => 'Variant Also Negotiates',
59
+	'http_507'                 => 'Insufficient Storage',
60
+	'http_508'                 => 'Loop Detected',
61
+	'http_509'                 => 'Unassigned',
62
+	'http_510'                 => 'Not Extended',
63
+	'http_511'                 => 'Network Authentication Required',
64 64
 ];
65 65
 
Please login to merge, or discard this patch.
src/lang/pl/builder.php 1 patch
Indentation   +48 added lines, -48 removed lines patch added patch discarded remove patch
@@ -10,55 +10,55 @@
 block discarded – undo
10 10
  */
11 11
 return [
12 12
 
13
-    'ok'                       => 'OK',
14
-    'no_error_message'         => 'Błąd #:api_code',
13
+	'ok'                       => 'OK',
14
+	'no_error_message'         => 'Błąd #:api_code',
15 15
 
16
-    // Used by Exception Handler Helper (when used)
17
-    'uncaught_exception'       => 'Nieprzechwycony wyjątek: :message',
18
-    'http_exception'           => 'Wyjątek HTTP: :message',
16
+	// Used by Exception Handler Helper (when used)
17
+	'uncaught_exception'       => 'Nieprzechwycony wyjątek: :message',
18
+	'http_exception'           => 'Wyjątek HTTP: :message',
19 19
 
20
-    // HttpException handler (added in 6.4.0)
21
-    // Error messages for HttpException caught w/o custom messages
22
-    // https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
23
-    'http_400'                 => 'Bad Request',
24
-    'http_401'                 => 'Unauthorized',
25
-    'http_402'                 => 'Payment Required',
26
-    'http_403'                 => 'Forbidden',
27
-    'http_404'                 => 'Not Found',
28
-    'http_405'                 => 'Method Not Allowed',
29
-    'http_406'                 => 'Not Acceptable',
30
-    'http_407'                 => 'Proxy Authentication Required',
31
-    'http_408'                 => 'Request Timeout',
32
-    'http_409'                 => 'Conflict',
33
-    'http_410'                 => 'Gone',
34
-    'http_411'                 => 'Length Required',
35
-    'http_412'                 => 'Precondition Failed',
36
-    'http_413'                 => 'Payload Too Large',
37
-    'http_414'                 => 'URI Too Long',
38
-    'http_415'                 => 'Unsupported Media Type',
39
-    'http_416'                 => 'Range Not Satisfiable',
40
-    'http_417'                 => 'Expectation Failed',
41
-    'http_421'                 => 'Misdirected Request',
42
-    'http_422'                 => 'Unprocessable Entity',
43
-    'http_423'                 => 'Locked',
44
-    'http_424'                 => 'Failed Dependency',
45
-    'http_425'                 => 'Too Early',
46
-    'http_426'                 => 'Upgrade Required',
47
-    'http_428'                 => 'Precondition Required',
48
-    'http_429'                 => 'Too Many Requests',
49
-    'http_431'                 => 'Request Header Fields Too Large',
50
-    'http_451'                 => 'Unavailable For Legal Reasons',
20
+	// HttpException handler (added in 6.4.0)
21
+	// Error messages for HttpException caught w/o custom messages
22
+	// https://www.iana.org/assignments/http-status-codes/http-status-codes.xhtml
23
+	'http_400'                 => 'Bad Request',
24
+	'http_401'                 => 'Unauthorized',
25
+	'http_402'                 => 'Payment Required',
26
+	'http_403'                 => 'Forbidden',
27
+	'http_404'                 => 'Not Found',
28
+	'http_405'                 => 'Method Not Allowed',
29
+	'http_406'                 => 'Not Acceptable',
30
+	'http_407'                 => 'Proxy Authentication Required',
31
+	'http_408'                 => 'Request Timeout',
32
+	'http_409'                 => 'Conflict',
33
+	'http_410'                 => 'Gone',
34
+	'http_411'                 => 'Length Required',
35
+	'http_412'                 => 'Precondition Failed',
36
+	'http_413'                 => 'Payload Too Large',
37
+	'http_414'                 => 'URI Too Long',
38
+	'http_415'                 => 'Unsupported Media Type',
39
+	'http_416'                 => 'Range Not Satisfiable',
40
+	'http_417'                 => 'Expectation Failed',
41
+	'http_421'                 => 'Misdirected Request',
42
+	'http_422'                 => 'Unprocessable Entity',
43
+	'http_423'                 => 'Locked',
44
+	'http_424'                 => 'Failed Dependency',
45
+	'http_425'                 => 'Too Early',
46
+	'http_426'                 => 'Upgrade Required',
47
+	'http_428'                 => 'Precondition Required',
48
+	'http_429'                 => 'Too Many Requests',
49
+	'http_431'                 => 'Request Header Fields Too Large',
50
+	'http_451'                 => 'Unavailable For Legal Reasons',
51 51
 
52
-    'http_500'                 => 'Internal Server Error',
53
-    'http_501'                 => 'Not Implemented',
54
-    'http_502'                 => 'Bad Gateway',
55
-    'http_503'                 => 'Service Unavailable',
56
-    'http_504'                 => 'Gateway Timeout',
57
-    'http_505'                 => 'HTTP Version Not Supported',
58
-    'http_506'                 => 'Variant Also Negotiates',
59
-    'http_507'                 => 'Insufficient Storage',
60
-    'http_508'                 => 'Loop Detected',
61
-    'http_509'                 => 'Unassigned',
62
-    'http_510'                 => 'Not Extended',
63
-    'http_511'                 => 'Network Authentication Required',
52
+	'http_500'                 => 'Internal Server Error',
53
+	'http_501'                 => 'Not Implemented',
54
+	'http_502'                 => 'Bad Gateway',
55
+	'http_503'                 => 'Service Unavailable',
56
+	'http_504'                 => 'Gateway Timeout',
57
+	'http_505'                 => 'HTTP Version Not Supported',
58
+	'http_506'                 => 'Variant Also Negotiates',
59
+	'http_507'                 => 'Insufficient Storage',
60
+	'http_508'                 => 'Loop Detected',
61
+	'http_509'                 => 'Unassigned',
62
+	'http_510'                 => 'Not Extended',
63
+	'http_511'                 => 'Network Authentication Required',
64 64
 ];
Please login to merge, or discard this patch.
src/ApiCodesHelpers.php 1 patch
Indentation   +119 added lines, -119 removed lines patch added patch discarded remove patch
@@ -21,124 +21,124 @@
 block discarded – undo
21 21
  */
22 22
 trait ApiCodesHelpers
23 23
 {
24
-    /**
25
-     * Returns lowest allowed error code for this module
26
-     *
27
-     * @return integer
28
-     *
29
-     * @throws \RuntimeException Throws exception if no min_code set up
30
-     */
31
-    public static function getMinCode(): int
32
-    {
33
-        $key = ResponseBuilder::CONF_KEY_MIN_CODE;
34
-        $min_code = Config::get($key, null);
35
-
36
-        if ($min_code === null) {
37
-            throw new \RuntimeException(sprintf('CONFIG: Missing "%s" key', $key));
38
-        }
39
-
40
-        return $min_code;
41
-    }
42
-
43
-    /**
44
-     * Returns highest allowed error code for this module
45
-     *
46
-     * @return integer
47
-     *
48
-     * @throws \RuntimeException Throws exception if no max_code set up
49
-     */
50
-    public static function getMaxCode(): int
51
-    {
52
-        $key = ResponseBuilder::CONF_KEY_MAX_CODE;
53
-        $max_code = Config::get($key, null);
54
-
55
-        if ($max_code === null) {
56
-            throw new \RuntimeException(sprintf('CONFIG: Missing "%s" key', $key));
57
-        }
58
-
59
-        return $max_code;
60
-    }
61
-
62
-    /**
63
-     * Returns array of error code constants defined in this class. Used mainly for debugging/tests
64
-     *
65
-     * @return array
66
-     */
67
-    public static function getApiCodeConstants(): array
68
-    {
69
-        /** @noinspection PhpUnhandledExceptionInspection */
70
-        return (new \ReflectionClass(static::class))->getConstants();
71
-    }
72
-
73
-    /**
74
-     * Returns complete error code to locale string mapping array
75
-     *
76
-     * @return array
77
-     *
78
-     * @throws \RuntimeException Thrown when builder map is not configured.
79
-     */
80
-    public static function getMap(): array
81
-    {
82
-        $user_map = Config::get(ResponseBuilder::CONF_KEY_MAP, null);
83
-        if ($user_map === null) {
84
-            throw new \RuntimeException(sprintf('CONFIG: Missing "%s" key', ResponseBuilder::CONF_KEY_MAP));
85
-        }
86
-        if (!\is_array($user_map)) {
87
-            throw new \RuntimeException(sprintf('CONFIG: "%s" must be an array', ResponseBuilder::CONF_KEY_MAP));
88
-        }
89
-        return Util::mergeConfig(BaseApiCodes::getBaseMap(), $user_map);
90
-    }
91
-
92
-    /**
93
-     * Returns locale mappings key for given api code or @null if there's no mapping
94
-     *
95
-     * @param integer $api_code Api code to look for mapped message for.
96
-     *
97
-     * @return string|null
98
-     *
99
-     * @throws \InvalidArgumentException If $code is not in allowed range.
100
-     */
101
-    public static function getCodeMessageKey(int $api_code): ?string
102
-    {
103
-        if (!static::isCodeValid($api_code)) {
104
-            $min = static::getMinCode();
105
-            $max = static::getMaxCode();
106
-            throw new \InvalidArgumentException("API code value ({$api_code}) is out of allowed range {$min}-{$max}");
107
-        }
108
-
109
-        $map = static::getMap();
110
-
111
-        return $map[ $api_code ] ?? null;
112
-    }
113
-
114
-    /**
115
-     * Checks if given API $code can be used in current configuration.
116
-     *
117
-     * @param int $code API code to validate
118
-     *
119
-     * @return bool
120
-     */
121
-    public static function isCodeValid(int $code): bool
122
-    {
123
-        return ($code === 0) || (($code >= static::getMinCode()) && ($code <= static::getMaxCode()));
124
-    }
125
-
126
-    /**
127
-     * Returns final API code for internal code, remapped to configured code range
128
-     *
129
-     * @param int $internal_code
130
-     *
131
-     * @return int
132
-     *
133
-     * @throws \InvalidArgumentException
134
-     */
135
-    public static function getCodeForInternalOffset(int $internal_code): int
136
-    {
137
-        $min = static::RESERVED_MIN_API_CODE_OFFSET;
138
-        $max = static::RESERVED_MAX_API_CODE_OFFSET;
139
-        Validator::assertIsIntRange('internal_code', $internal_code, $min, $max);
140
-
141
-        return ($internal_code === 0) ? 0 : $internal_code + static::getMinCode();
142
-    }
24
+	/**
25
+	 * Returns lowest allowed error code for this module
26
+	 *
27
+	 * @return integer
28
+	 *
29
+	 * @throws \RuntimeException Throws exception if no min_code set up
30
+	 */
31
+	public static function getMinCode(): int
32
+	{
33
+		$key = ResponseBuilder::CONF_KEY_MIN_CODE;
34
+		$min_code = Config::get($key, null);
35
+
36
+		if ($min_code === null) {
37
+			throw new \RuntimeException(sprintf('CONFIG: Missing "%s" key', $key));
38
+		}
39
+
40
+		return $min_code;
41
+	}
42
+
43
+	/**
44
+	 * Returns highest allowed error code for this module
45
+	 *
46
+	 * @return integer
47
+	 *
48
+	 * @throws \RuntimeException Throws exception if no max_code set up
49
+	 */
50
+	public static function getMaxCode(): int
51
+	{
52
+		$key = ResponseBuilder::CONF_KEY_MAX_CODE;
53
+		$max_code = Config::get($key, null);
54
+
55
+		if ($max_code === null) {
56
+			throw new \RuntimeException(sprintf('CONFIG: Missing "%s" key', $key));
57
+		}
58
+
59
+		return $max_code;
60
+	}
61
+
62
+	/**
63
+	 * Returns array of error code constants defined in this class. Used mainly for debugging/tests
64
+	 *
65
+	 * @return array
66
+	 */
67
+	public static function getApiCodeConstants(): array
68
+	{
69
+		/** @noinspection PhpUnhandledExceptionInspection */
70
+		return (new \ReflectionClass(static::class))->getConstants();
71
+	}
72
+
73
+	/**
74
+	 * Returns complete error code to locale string mapping array
75
+	 *
76
+	 * @return array
77
+	 *
78
+	 * @throws \RuntimeException Thrown when builder map is not configured.
79
+	 */
80
+	public static function getMap(): array
81
+	{
82
+		$user_map = Config::get(ResponseBuilder::CONF_KEY_MAP, null);
83
+		if ($user_map === null) {
84
+			throw new \RuntimeException(sprintf('CONFIG: Missing "%s" key', ResponseBuilder::CONF_KEY_MAP));
85
+		}
86
+		if (!\is_array($user_map)) {
87
+			throw new \RuntimeException(sprintf('CONFIG: "%s" must be an array', ResponseBuilder::CONF_KEY_MAP));
88
+		}
89
+		return Util::mergeConfig(BaseApiCodes::getBaseMap(), $user_map);
90
+	}
91
+
92
+	/**
93
+	 * Returns locale mappings key for given api code or @null if there's no mapping
94
+	 *
95
+	 * @param integer $api_code Api code to look for mapped message for.
96
+	 *
97
+	 * @return string|null
98
+	 *
99
+	 * @throws \InvalidArgumentException If $code is not in allowed range.
100
+	 */
101
+	public static function getCodeMessageKey(int $api_code): ?string
102
+	{
103
+		if (!static::isCodeValid($api_code)) {
104
+			$min = static::getMinCode();
105
+			$max = static::getMaxCode();
106
+			throw new \InvalidArgumentException("API code value ({$api_code}) is out of allowed range {$min}-{$max}");
107
+		}
108
+
109
+		$map = static::getMap();
110
+
111
+		return $map[ $api_code ] ?? null;
112
+	}
113
+
114
+	/**
115
+	 * Checks if given API $code can be used in current configuration.
116
+	 *
117
+	 * @param int $code API code to validate
118
+	 *
119
+	 * @return bool
120
+	 */
121
+	public static function isCodeValid(int $code): bool
122
+	{
123
+		return ($code === 0) || (($code >= static::getMinCode()) && ($code <= static::getMaxCode()));
124
+	}
125
+
126
+	/**
127
+	 * Returns final API code for internal code, remapped to configured code range
128
+	 *
129
+	 * @param int $internal_code
130
+	 *
131
+	 * @return int
132
+	 *
133
+	 * @throws \InvalidArgumentException
134
+	 */
135
+	public static function getCodeForInternalOffset(int $internal_code): int
136
+	{
137
+		$min = static::RESERVED_MIN_API_CODE_OFFSET;
138
+		$max = static::RESERVED_MAX_API_CODE_OFFSET;
139
+		Validator::assertIsIntRange('internal_code', $internal_code, $min, $max);
140
+
141
+		return ($internal_code === 0) ? 0 : $internal_code + static::getMinCode();
142
+	}
143 143
 
144 144
 }
Please login to merge, or discard this patch.
config/response_builder.php 1 patch
Indentation   +78 added lines, -78 removed lines patch added patch discarded remove patch
@@ -12,78 +12,78 @@  discard block
 block discarded – undo
12 12
  */
13 13
 
14 14
 return [
15
-    /*
15
+	/*
16 16
     |-----------------------------------------------------------------------------------------------------------
17 17
     | Code range settings
18 18
     |-----------------------------------------------------------------------------------------------------------
19 19
     */
20
-    'min_code'          => 100,
21
-    'max_code'          => 1024,
20
+	'min_code'          => 100,
21
+	'max_code'          => 1024,
22 22
 
23
-    /*
23
+	/*
24 24
     |-----------------------------------------------------------------------------------------------------------
25 25
     | Error code to message mapping
26 26
     |-----------------------------------------------------------------------------------------------------------
27 27
     |
28 28
     */
29
-    'map'               => [
30
-        // YOUR_API_CODE => '<MESSAGE_KEY>',
31
-    ],
29
+	'map'               => [
30
+		// YOUR_API_CODE => '<MESSAGE_KEY>',
31
+	],
32 32
 
33
-    /*
33
+	/*
34 34
     |-----------------------------------------------------------------------------------------------------------
35 35
     | Response Builder data converter
36 36
     |-----------------------------------------------------------------------------------------------------------
37 37
     |
38 38
     */
39
-    'converter'         => [
40
-        \Illuminate\Database\Eloquent\Model::class          => [
41
-            'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
42
-            // 'key'     => 'item',
43
-            'pri'     => 0,
44
-        ],
45
-        \Illuminate\Support\Collection::class               => [
46
-            'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
47
-            // 'key'     => 'item',
48
-            'pri'     => 0,
49
-        ],
50
-        \Illuminate\Database\Eloquent\Collection::class     => [
51
-            'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
52
-            // 'key'     => 'item',
53
-            'pri'     => 0,
54
-        ],
55
-        \Illuminate\Http\Resources\Json\JsonResource::class => [
56
-            'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
57
-            // 'key'     => 'item',
58
-            'pri'     => 0,
59
-        ],
39
+	'converter'         => [
40
+		\Illuminate\Database\Eloquent\Model::class          => [
41
+			'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
42
+			// 'key'     => 'item',
43
+			'pri'     => 0,
44
+		],
45
+		\Illuminate\Support\Collection::class               => [
46
+			'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
47
+			// 'key'     => 'item',
48
+			'pri'     => 0,
49
+		],
50
+		\Illuminate\Database\Eloquent\Collection::class     => [
51
+			'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
52
+			// 'key'     => 'item',
53
+			'pri'     => 0,
54
+		],
55
+		\Illuminate\Http\Resources\Json\JsonResource::class => [
56
+			'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ToArrayConverter::class,
57
+			// 'key'     => 'item',
58
+			'pri'     => 0,
59
+		],
60 60
 
61
-        /*
61
+		/*
62 62
         |-----------------------------------------------------------------------------------------------------------
63 63
         | Generic converters should have lower pri to allow dedicated ones to kick in first when class matches
64 64
         |-----------------------------------------------------------------------------------------------------------
65 65
         */
66
-        \JsonSerializable::class                            => [
67
-            'handler' => \MarcinOrlowski\ResponseBuilder\Converters\JsonSerializableConverter::class,
68
-            // 'key'     => 'item',
69
-            'pri'     => -10,
70
-        ],
71
-        \Illuminate\Contracts\Support\Arrayable::class      => [
72
-            'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ArrayableConverter::class,
73
-            // 'key'     => 'item',
74
-            'pri'     => -10,
75
-        ],
66
+		\JsonSerializable::class                            => [
67
+			'handler' => \MarcinOrlowski\ResponseBuilder\Converters\JsonSerializableConverter::class,
68
+			// 'key'     => 'item',
69
+			'pri'     => -10,
70
+		],
71
+		\Illuminate\Contracts\Support\Arrayable::class      => [
72
+			'handler' => \MarcinOrlowski\ResponseBuilder\Converters\ArrayableConverter::class,
73
+			// 'key'     => 'item',
74
+			'pri'     => -10,
75
+		],
76 76
 
77
-    ],
77
+	],
78 78
 
79
-    /*
79
+	/*
80 80
     |-----------------------------------------------------------------------------------------------------------
81 81
     | Exception handler error codes
82 82
     |-----------------------------------------------------------------------------------------------------------
83 83
     |
84 84
     */
85
-    'exception_handler' => [
86
-	    /*
85
+	'exception_handler' => [
86
+		/*
87 87
 	     * The following keys are supported for each handler specified.
88 88
 	     *   `handler`
89 89
 	     *   `pri`
@@ -103,19 +103,19 @@  discard block
 block discarded – undo
103 103
 		 *                  message ($ex->getMessage()).
104 104
 		 */
105 105
 
106
-    	\Illuminate\Validation\ValidationException::class => [
107
-		    'handler' => \MarcinOrlowski\ResponseBuilder\ExceptionHandlers\ValidationExceptionHandler::class,
108
-		    'pri'     => -100,
109
-		    'config' => [
106
+		\Illuminate\Validation\ValidationException::class => [
107
+			'handler' => \MarcinOrlowski\ResponseBuilder\ExceptionHandlers\ValidationExceptionHandler::class,
108
+			'pri'     => -100,
109
+			'config' => [
110 110
 //		        'api_code'  => ApiCodes::YOUR_API_CODE_FOR_VALIDATION_EXCEPTION,
111 111
 //		        'http_code' => HttpResponse::HTTP_UNPROCESSABLE_ENTITY,
112
-		    	],
113
-	    ],
112
+				],
113
+		],
114 114
 
115 115
 		\Symfony\Component\HttpKernel\Exception\HttpException::class => [
116
-	        'handler' => \MarcinOrlowski\ResponseBuilder\ExceptionHandlers\HttpExceptionHandler::class,
117
-	        'pri'     => -100,
118
-	        'config'  => [
116
+			'handler' => \MarcinOrlowski\ResponseBuilder\ExceptionHandlers\HttpExceptionHandler::class,
117
+			'pri'     => -100,
118
+			'config'  => [
119 119
 //		        HttpException::class => [
120 120
 //			        // used by unauthenticated() to obtain api and http code for the exception
121 121
 //			        HttpResponse::HTTP_UNAUTHORIZED         => [
@@ -131,47 +131,47 @@  discard block
 block discarded – undo
131 131
 //				        'http_code' => HttpResponse::HTTP_BAD_REQUEST,
132 132
 //			        ],
133 133
 //		        ],
134
-	        ],
134
+			],
135 135
 //	        // This is final exception handler. If ex is not dealt with yet this is its last stop.
136
-	        // default handler is mandatory and MUST have both `api_code` and `http_code` set.
136
+			// default handler is mandatory and MUST have both `api_code` and `http_code` set.
137 137
 
138
-	        'default' => [
139
-		        'handler' => \MarcinOrlowski\ResponseBuilder\ExceptionHandlers\HttpExceptionHandler::class,
140
-		        'pri'     => -127,
141
-		        'config'  => [
138
+			'default' => [
139
+				'handler' => \MarcinOrlowski\ResponseBuilder\ExceptionHandlers\HttpExceptionHandler::class,
140
+				'pri'     => -127,
141
+				'config'  => [
142 142
 //			        'api_code'  => ApiCodes::YOUR_API_CODE_FOR_UNHANDLED_EXCEPTION,
143 143
 //			        'http_code' => HttpResponse::HTTP_INTERNAL_SERVER_ERROR,
144
-		        ],
145
-	        ],
146
-	    ],
147
-    ],
144
+				],
145
+			],
146
+		],
147
+	],
148 148
 
149
-    /*
149
+	/*
150 150
     |-----------------------------------------------------------------------------------------------------------
151 151
     | data-to-json encoding options
152 152
     |-----------------------------------------------------------------------------------------------------------
153 153
     |
154 154
     */
155
-    'encoding_options'  => JSON_HEX_TAG | JSON_HEX_APOS | JSON_HEX_AMP | JSON_HEX_QUOT | JSON_UNESCAPED_UNICODE,
155
+	'encoding_options'  => JSON_HEX_TAG | JSON_HEX_APOS | JSON_HEX_AMP | JSON_HEX_QUOT | JSON_UNESCAPED_UNICODE,
156 156
 
157
-    /*
157
+	/*
158 158
     |-----------------------------------------------------------------------------------------------------------
159 159
     | Debug config
160 160
     |-----------------------------------------------------------------------------------------------------------
161 161
     |
162 162
     */
163
-    'debug'             => [
164
-        'debug_key'         => 'debug',
165
-        'exception_handler' => [
166
-            'trace_key'     => 'trace',
167
-            'trace_enabled' => env('APP_DEBUG', false),
168
-        ],
163
+	'debug'             => [
164
+		'debug_key'         => 'debug',
165
+		'exception_handler' => [
166
+			'trace_key'     => 'trace',
167
+			'trace_enabled' => env('APP_DEBUG', false),
168
+		],
169 169
 
170
-        // Controls debugging features of payload converter class.
171
-        'converter' => [
172
-        	// Set to true to figure out what converter is used for given data payload and why.
173
-        	'debug_enabled' => env('RB_CONVERTER_DEBUG', false),
174
-        ],
175
-    ],
170
+		// Controls debugging features of payload converter class.
171
+		'converter' => [
172
+			// Set to true to figure out what converter is used for given data payload and why.
173
+			'debug_enabled' => env('RB_CONVERTER_DEBUG', false),
174
+		],
175
+	],
176 176
 
177 177
 ];
Please login to merge, or discard this patch.
src/Converter.php 1 patch
Indentation   +182 added lines, -182 removed lines patch added patch discarded remove patch
@@ -23,187 +23,187 @@
 block discarded – undo
23 23
  */
24 24
 class Converter
25 25
 {
26
-    /**
27
-     * @var array
28
-     */
29
-    protected $classes = [];
30
-
31
-    /** @var bool */
32
-    protected $debug_enabled = false;
33
-
34
-    /**
35
-     * Converter constructor.
36
-     *
37
-     * @throws \RuntimeException
38
-     */
39
-    public function __construct()
40
-    {
41
-        $this->classes = static::getClassesMapping() ?? [];
42
-
43
-	    $this->debug_enabled = Config::get(ResponseBuilder::CONF_KEY_CONVERTER_DEBUG_KEY, false);
44
-    }
45
-
46
-    /**
47
-     * Returns local copy of configuration mapping for the classes.
48
-     *
49
-     * @return array
50
-     */
51
-    public function getClasses(): array
52
-    {
53
-        return $this->classes;
54
-    }
55
-
56
-    /**
57
-     * Checks if we have "classes" mapping configured for $data object class.
58
-     * Returns @true if there's valid config for this class.
59
-     * Throws \RuntimeException if there's no config "classes" mapping entry for this object configured.
60
-     * Throws \InvalidArgumentException if No data conversion mapping configured for given class.
61
-     *
62
-     * @param object $data Object to check mapping for.
63
-     *
64
-     * @return array
65
-     *
66
-     * @throws \InvalidArgumentException
67
-     */
68
-    protected function getClassMappingConfigOrThrow(object $data): array
69
-    {
70
-        $result = null;
71
-        $debug_result = '';
72
-
73
-        // check for exact class name match...
74
-        $cls = \get_class($data);
75
-        if (\is_string($cls)) {
76
-	        if (\array_key_exists($cls, $this->classes)) {
77
-		        $result = $this->classes[ $cls ];
78
-		        $debug_result = 'exact config match';
79
-	        } else {
80
-		        // no exact match, then lets try with `instanceof`
81
-		        foreach (\array_keys($this->getClasses()) as $class_name) {
82
-			        if ($data instanceof $class_name) {
83
-				        $result = $this->classes[ $class_name ];
84
-				        $debug_result = "subclass of {$class_name}";
85
-				        break;
86
-			        }
87
-		        }
88
-	        }
89
-        }
90
-
91
-        if ($result === null) {
92
-            throw new \InvalidArgumentException(sprintf('No data conversion mapping configured for "%s" class.', $cls));
93
-        }
94
-
95
-        if ($this->debug_enabled) {
26
+	/**
27
+	 * @var array
28
+	 */
29
+	protected $classes = [];
30
+
31
+	/** @var bool */
32
+	protected $debug_enabled = false;
33
+
34
+	/**
35
+	 * Converter constructor.
36
+	 *
37
+	 * @throws \RuntimeException
38
+	 */
39
+	public function __construct()
40
+	{
41
+		$this->classes = static::getClassesMapping() ?? [];
42
+
43
+		$this->debug_enabled = Config::get(ResponseBuilder::CONF_KEY_CONVERTER_DEBUG_KEY, false);
44
+	}
45
+
46
+	/**
47
+	 * Returns local copy of configuration mapping for the classes.
48
+	 *
49
+	 * @return array
50
+	 */
51
+	public function getClasses(): array
52
+	{
53
+		return $this->classes;
54
+	}
55
+
56
+	/**
57
+	 * Checks if we have "classes" mapping configured for $data object class.
58
+	 * Returns @true if there's valid config for this class.
59
+	 * Throws \RuntimeException if there's no config "classes" mapping entry for this object configured.
60
+	 * Throws \InvalidArgumentException if No data conversion mapping configured for given class.
61
+	 *
62
+	 * @param object $data Object to check mapping for.
63
+	 *
64
+	 * @return array
65
+	 *
66
+	 * @throws \InvalidArgumentException
67
+	 */
68
+	protected function getClassMappingConfigOrThrow(object $data): array
69
+	{
70
+		$result = null;
71
+		$debug_result = '';
72
+
73
+		// check for exact class name match...
74
+		$cls = \get_class($data);
75
+		if (\is_string($cls)) {
76
+			if (\array_key_exists($cls, $this->classes)) {
77
+				$result = $this->classes[ $cls ];
78
+				$debug_result = 'exact config match';
79
+			} else {
80
+				// no exact match, then lets try with `instanceof`
81
+				foreach (\array_keys($this->getClasses()) as $class_name) {
82
+					if ($data instanceof $class_name) {
83
+						$result = $this->classes[ $class_name ];
84
+						$debug_result = "subclass of {$class_name}";
85
+						break;
86
+					}
87
+				}
88
+			}
89
+		}
90
+
91
+		if ($result === null) {
92
+			throw new \InvalidArgumentException(sprintf('No data conversion mapping configured for "%s" class.', $cls));
93
+		}
94
+
95
+		if ($this->debug_enabled) {
96 96
 			Log::debug(__CLASS__ . ": Converting {$cls} using {$result['handler']} because: {$debug_result}.");
97
-        }
98
-
99
-	    return $result;
100
-    }
101
-
102
-    /**
103
-     * We need to prepare source data
104
-     *
105
-     * @param object|array|null $data
106
-     *
107
-     * @return array|null
108
-     *
109
-     * @throws \InvalidArgumentException
110
-     */
111
-    public function convert($data = null): ?array
112
-    {
113
-        if ($data === null) {
114
-            return null;
115
-        }
116
-
117
-        Validator::assertIsType('data', $data, [Validator::TYPE_ARRAY,
118
-                                                Validator::TYPE_OBJECT]);
119
-
120
-        if (\is_object($data)) {
121
-            $cfg = $this->getClassMappingConfigOrThrow($data);
122
-            $worker = new $cfg[ ResponseBuilder::KEY_HANDLER ]();
123
-            $data = $worker->convert($data, $cfg);
124
-        } else {
125
-            $data = $this->convertArray($data);
126
-        }
127
-
128
-        return $data;
129
-    }
130
-
131
-    /**
132
-     * Recursively walks $data array and converts all known objects if found. Note
133
-     * $data array is passed by reference so source $data array may be modified.
134
-     *
135
-     * @param array $data array to recursively convert known elements of
136
-     *
137
-     * @return array
138
-     *
139
-     * @throws \RuntimeException
140
-     */
141
-    protected function convertArray(array $data): array
142
-    {
143
-        // This is to ensure that we either have array with user provided keys i.e. ['foo'=>'bar'], which will then
144
-        // be turned into JSON object or array without user specified keys (['bar']) which we would return as JSON
145
-        // array. But you can't mix these two as the final JSON would not produce predictable results.
146
-        $string_keys_cnt = 0;
147
-        $int_keys_cnt = 0;
148
-        foreach ($data as $key => $val) {
149
-            if (\is_int($key)) {
150
-                $int_keys_cnt++;
151
-            } else {
152
-                $string_keys_cnt++;
153
-            }
154
-
155
-            if (($string_keys_cnt > 0) && ($int_keys_cnt > 0)) {
156
-                throw new \RuntimeException(
157
-                    'Invalid data array. Either set own keys for all the items or do not specify any keys at all. ' .
158
-                    'Arrays with mixed keys are not supported by design.');
159
-            }
160
-        }
161
-
162
-        foreach ($data as $key => $val) {
163
-            if (\is_array($val)) {
164
-                $data[ $key ] = $this->convertArray($val);
165
-            } elseif (\is_object($val)) {
166
-                $cfg = $this->getClassMappingConfigOrThrow($val);
167
-                $worker = new $cfg[ ResponseBuilder::KEY_HANDLER ]();
168
-                $converted_data = $worker->convert($val, $cfg);
169
-                $data[ $key ] = $converted_data;
170
-            }
171
-        }
172
-
173
-        return $data;
174
-    }
175
-
176
-    /**
177
-     * Reads and validates "classes" config mapping
178
-     *
179
-     * @return array Classes mapping as specified in configuration or empty array if configuration found
180
-     *
181
-     * @throws \RuntimeException if "classes" mapping is technically invalid (i.e. not array etc).
182
-     */
183
-    protected static function getClassesMapping(): array
184
-    {
185
-        $classes = Config::get(ResponseBuilder::CONF_KEY_CONVERTER);
186
-
187
-        if ($classes !== null) {
188
-            if (!\is_array($classes)) {
189
-                throw new \RuntimeException(
190
-                    \sprintf('CONFIG: "classes" mapping must be an array (%s given)', \gettype($classes)));
191
-            }
192
-
193
-            $mandatory_keys = [
194
-                ResponseBuilder::KEY_HANDLER,
195
-            ];
196
-            foreach ($classes as $class_name => $class_config) {
197
-                foreach ($mandatory_keys as $key_name) {
198
-                    if (!\array_key_exists($key_name, $class_config)) {
199
-                        throw new \RuntimeException("CONFIG: Missing '{$key_name}' for '{$class_name}' class mapping");
200
-                    }
201
-                }
202
-            }
203
-        } else {
204
-            $classes = [];
205
-        }
206
-
207
-        return $classes;
208
-    }
97
+		}
98
+
99
+		return $result;
100
+	}
101
+
102
+	/**
103
+	 * We need to prepare source data
104
+	 *
105
+	 * @param object|array|null $data
106
+	 *
107
+	 * @return array|null
108
+	 *
109
+	 * @throws \InvalidArgumentException
110
+	 */
111
+	public function convert($data = null): ?array
112
+	{
113
+		if ($data === null) {
114
+			return null;
115
+		}
116
+
117
+		Validator::assertIsType('data', $data, [Validator::TYPE_ARRAY,
118
+												Validator::TYPE_OBJECT]);
119
+
120
+		if (\is_object($data)) {
121
+			$cfg = $this->getClassMappingConfigOrThrow($data);
122
+			$worker = new $cfg[ ResponseBuilder::KEY_HANDLER ]();
123
+			$data = $worker->convert($data, $cfg);
124
+		} else {
125
+			$data = $this->convertArray($data);
126
+		}
127
+
128
+		return $data;
129
+	}
130
+
131
+	/**
132
+	 * Recursively walks $data array and converts all known objects if found. Note
133
+	 * $data array is passed by reference so source $data array may be modified.
134
+	 *
135
+	 * @param array $data array to recursively convert known elements of
136
+	 *
137
+	 * @return array
138
+	 *
139
+	 * @throws \RuntimeException
140
+	 */
141
+	protected function convertArray(array $data): array
142
+	{
143
+		// This is to ensure that we either have array with user provided keys i.e. ['foo'=>'bar'], which will then
144
+		// be turned into JSON object or array without user specified keys (['bar']) which we would return as JSON
145
+		// array. But you can't mix these two as the final JSON would not produce predictable results.
146
+		$string_keys_cnt = 0;
147
+		$int_keys_cnt = 0;
148
+		foreach ($data as $key => $val) {
149
+			if (\is_int($key)) {
150
+				$int_keys_cnt++;
151
+			} else {
152
+				$string_keys_cnt++;
153
+			}
154
+
155
+			if (($string_keys_cnt > 0) && ($int_keys_cnt > 0)) {
156
+				throw new \RuntimeException(
157
+					'Invalid data array. Either set own keys for all the items or do not specify any keys at all. ' .
158
+					'Arrays with mixed keys are not supported by design.');
159
+			}
160
+		}
161
+
162
+		foreach ($data as $key => $val) {
163
+			if (\is_array($val)) {
164
+				$data[ $key ] = $this->convertArray($val);
165
+			} elseif (\is_object($val)) {
166
+				$cfg = $this->getClassMappingConfigOrThrow($val);
167
+				$worker = new $cfg[ ResponseBuilder::KEY_HANDLER ]();
168
+				$converted_data = $worker->convert($val, $cfg);
169
+				$data[ $key ] = $converted_data;
170
+			}
171
+		}
172
+
173
+		return $data;
174
+	}
175
+
176
+	/**
177
+	 * Reads and validates "classes" config mapping
178
+	 *
179
+	 * @return array Classes mapping as specified in configuration or empty array if configuration found
180
+	 *
181
+	 * @throws \RuntimeException if "classes" mapping is technically invalid (i.e. not array etc).
182
+	 */
183
+	protected static function getClassesMapping(): array
184
+	{
185
+		$classes = Config::get(ResponseBuilder::CONF_KEY_CONVERTER);
186
+
187
+		if ($classes !== null) {
188
+			if (!\is_array($classes)) {
189
+				throw new \RuntimeException(
190
+					\sprintf('CONFIG: "classes" mapping must be an array (%s given)', \gettype($classes)));
191
+			}
192
+
193
+			$mandatory_keys = [
194
+				ResponseBuilder::KEY_HANDLER,
195
+			];
196
+			foreach ($classes as $class_name => $class_config) {
197
+				foreach ($mandatory_keys as $key_name) {
198
+					if (!\array_key_exists($key_name, $class_config)) {
199
+						throw new \RuntimeException("CONFIG: Missing '{$key_name}' for '{$class_name}' class mapping");
200
+					}
201
+				}
202
+			}
203
+		} else {
204
+			$classes = [];
205
+		}
206
+
207
+		return $classes;
208
+	}
209 209
 }
Please login to merge, or discard this patch.