Passed
Push — develop ( 28ee2c...8aa457 )
by Jens
10:01
created

Formatter   A

Complexity

Total Complexity 32

Size/Duplication

Total Lines 149
Duplicated Lines 0 %

Coupling/Cohesion

Components 1
Dependencies 0

Test Coverage

Coverage 0%

Importance

Changes 0
Metric Value
wmc 32
lcom 1
cbo 0
dl 0
loc 149
ccs 0
cts 109
cp 0
rs 9.6
c 0
b 0
f 0

3 Methods

Rating   Name   Duplication   Size   Complexity  
A __construct() 0 4 2
D format() 0 102 28
A headers() 0 9 2
1
<?php
2
/**
3
 * Origin: https://github.com/guzzle/log-subscriber
4
 * Copyright (c) 2014 Michael Dowling, https://github.com/mtdowling <[email protected]>
5
 *
6
 * Permission is hereby granted, free of charge, to any person obtaining a copy
7
 * of this software and associated documentation files (the "Software"), to deal
8
 * in the Software without restriction, including without limitation the rights
9
 * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10
 * copies of the Software, and to permit persons to whom the Software is
11
 * furnished to do so, subject to the following conditions:
12
 *
13
 * The above copyright notice and this permission notice shall be included in
14
 * all copies or substantial portions of the Software.
15
 *
16
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18
 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19
 * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20
 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21
 * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22
 * THE SOFTWARE.
23
 */
24
25
namespace Commercetools\Core\Helper\Subscriber\Log;
26
27
use GuzzleHttp\Message\MessageInterface;
28
use GuzzleHttp\Message\RequestInterface;
29
use GuzzleHttp\Message\ResponseInterface;
30
31
/**
32
 * Formats log messages using variable substitutions for requests, responses,
33
 * and other transactional data.
34
 *
35
 * The following variable substitutions are supported:
36
 *
37
 * - {request}:      Full HTTP request message
38
 * - {response}:     Full HTTP response message
39
 * - {ts}:           Timestamp
40
 * - {host}:         Host of the request
41
 * - {method}:       Method of the request
42
 * - {url}:          URL of the request
43
 * - {host}:         Host of the request
44
 * - {protocol}:     Request protocol
45
 * - {version}:      Protocol version
46
 * - {resource}:     Resource of the request (path + query + fragment)
47
 * - {hostname}:     Hostname of the machine that sent the request
48
 * - {code}:         Status code of the response (if available)
49
 * - {phrase}:       Reason phrase of the response  (if available)
50
 * - {error}:        Any error messages (if available)
51
 * - {req_header_*}: Replace `*` with the lowercased name of a request header to add to the message
52
 * - {res_header_*}: Replace `*` with the lowercased name of a response header to add to the message
53
 * - {req_headers}:  Request headers
54
 * - {res_headers}:  Response headers
55
 * - {req_body}:     Request body
56
 * - {res_body}:     Response body
57
 */
58
class Formatter
59
{
60
    /**
61
     * Apache Common Log Format.
62
     * @link http://httpd.apache.org/docs/1.3/logs.html#common
63
     * @var string
64
     * @codingStandardsIgnoreStart
65
     */
66
    const CLF = "{hostname} {req_header_User-Agent} - [{ts}] \"{method} {resource} {protocol}/{version}\" {code} {res_header_Content-Length}";
67
    /**
68
     * codingStandardsIgnoreEnd
69
     */
70
    const DEBUG = ">>>>>>>>\n{request}\n<<<<<<<<\n{response}\n--------\n{error}";
71
    const SHORT = '[{ts}] "{method} {resource} {protocol}/{version}" {code}';
72
73
    /** @var string Template used to format log messages */
74
    private $template;
75
76
    /**
77
     * @param string $template Log message template
78
     */
79
    public function __construct($template = self::CLF)
80
    {
81
        $this->template = $template ?: self::CLF;
82
    }
83
84
    /**
85
     * Returns a formatted message
86
     *
87
     * @param RequestInterface  $request    Request that was sent
88
     * @param ResponseInterface $response   Response that was received
89
     * @param \Exception        $error      Exception that was received
90
     * @param array             $customData Associative array of custom template data
91
     *
92
     * @return string
93
     */
94
    public function format(
95
        RequestInterface $request,
96
        ResponseInterface $response = null,
97
        \Exception $error = null,
98
        array $customData = []
99
    ) {
100
        $cache = $customData;
101
102
        return preg_replace_callback(
103
            '/{\s*([A-Za-z_\-\.0-9]+)\s*}/',
104
            function (array $matches) use ($request, $response, $error, &$cache) {
105
106
                if (isset($cache[$matches[1]])) {
107
                    return $cache[$matches[1]];
108
                }
109
110
                $result = '';
111
                switch ($matches[1]) {
112
                    case 'request':
113
                        $result = $request;
114
                        break;
115
                    case 'response':
116
                        $result = $response;
117
                        break;
118
                    case 'req_headers':
119
                        $result = trim($request->getMethod() . ' '
120
                            . $request->getResource()) . ' HTTP/'
121
                            . $request->getProtocolVersion() . "\r\n"
122
                            . $this->headers($request);
123
                        break;
124
                    case 'res_headers':
125
                        $result = $response ?
126
                            sprintf(
127
                                'HTTP/%s %d %s',
128
                                $response->getProtocolVersion(),
129
                                $response->getStatusCode(),
130
                                $response->getReasonPhrase()
131
                            ) . "\r\n" . $this->headers($response)
132
                            : 'NULL';
133
                        break;
134
                    case 'req_body':
135
                        $result = $request->getBody();
136
                        break;
137
                    case 'res_body':
138
                        $result = $response ? $response->getBody() : 'NULL';
139
                        break;
140
                    case 'ts':
141
                        $result = gmdate('c');
142
                        break;
143
                    case 'method':
144
                        $result = $request->getMethod();
145
                        break;
146
                    case 'url':
147
                        $result = $request->getUrl();
148
                        break;
149
                    case 'resource':
150
                        $result = $request->getResource();
151
                        break;
152
                    case 'req_version':
153
                        $result = $request->getProtocolVersion();
154
                        break;
155
                    case 'res_version':
156
                        $result = $response
157
                            ? $response->getProtocolVersion()
158
                            : 'NULL';
159
                        break;
160
                    case 'host':
161
                        $result = $request->getHost();
162
                        break;
163
                    case 'hostname':
164
                        $result = gethostname();
165
                        break;
166
                    case 'code':
167
                        $result = $response
168
                            ? $response->getStatusCode()
169
                            : 'NULL';
170
                        break;
171
                    case 'phrase':
172
                        $result = $response
173
                            ? $response->getReasonPhrase()
174
                            : 'NULL';
175
                        break;
176
                    case 'error':
177
                        $result = $error ? $error->getMessage() : 'NULL';
178
                        break;
179
                    default:
180
                        // handle prefixed dynamic headers
181
                        if (strpos($matches[1], 'req_header_') === 0) {
182
                            $result = $request->getHeader(substr($matches[1], 11));
183
                        } elseif (strpos($matches[1], 'res_header_') === 0) {
184
                            $result = $response
185
                                ? $response->getHeader(substr($matches[1], 11))
186
                                : 'NULL';
187
                        }
188
                }
189
190
                $cache[$matches[1]] = $result;
191
                return $result;
192
            },
193
            $this->template
194
        );
195
    }
196
197
    private function headers(MessageInterface $message)
198
    {
199
        $result = '';
200
        foreach ($message->getHeaders() as $name => $values) {
201
            $result .= $name . ': ' . implode(', ', $values) . "\r\n";
202
        }
203
204
        return trim($result);
205
    }
206
}
207