Skip to content

Commit c109042

Browse files
authoredNov 8, 2017
Merge pull request #16 from Vectorface/dev
v0.2.0 Release Changes
2 parents 224bd10 + b1013d3 commit c109042

File tree

10 files changed

+186
-42
lines changed

10 files changed

+186
-42
lines changed
 

‎.travis.yml

+6-3
Original file line numberDiff line numberDiff line change
@@ -3,18 +3,21 @@ sudo: false
33
cache: bundler
44

55
php:
6+
- 7.1
67
- 7.0
78
- 5.6
89
- 5.5
910
- 5.4
10-
- 5.3
11-
- 7.0
1211
- hhvm
1312

1413
matrix:
1514
allow_failures:
16-
- php: hhvm, 7.0
15+
- php: hhvm
1716
fast_finish: true
17+
include:
18+
- php: 5.3
19+
dist: precise
20+
1821

1922
before_script:
2023
- composer self-update

‎composer.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -31,7 +31,8 @@
3131
"php": ">=5.3.0",
3232
"vectorface/whip": "~0.2",
3333
"twig/twig": "~1.0",
34-
"danbruce/fast-route":"~0.6"
34+
"danbruce/fast-route":"~0.6",
35+
"psr/log": "~1.0"
3536
},
3637
"require-dev": {
3738
"phpunit/phpunit": "4.2.*",

‎docs/getting_started.md

+4-5
Original file line numberDiff line numberDiff line change
@@ -65,7 +65,7 @@ Create the file `tutorial/composer.json` with the following contents:
6565
"name": "vectorface/snappy-tutorial",
6666
"autoload": {
6767
"psr-4": {
68-
"Vectorface\\SnappyTutorial": "./app"
68+
"Vectorface\\SnappyTutorial\\": "./app"
6969
}
7070
},
7171
"require": {
@@ -219,11 +219,10 @@ Note that there are many ways to pass variables to the view.
219219
3. Directly rendering the view with `$this->renderView`. More details for this
220220
method can be found [here](handlers/controller_handler/#integration-with-twig).
221221

222-
We will divide our view into two files. The first file `layout.twig` will
222+
We will divide our view into two files. The first file `app/Views/layout.twig` will
223223
provide common boilerplate that we could reuse across multiple pages.
224224

225225
```html
226-
<!-- app/Views/layout.twig -->
227226
<!DOCTYPE html>
228227
<html lang="en">
229228
<head>
@@ -242,10 +241,9 @@ provide common boilerplate that we could reuse across multiple pages.
242241
</html>
243242
```
244243

245-
And a very simple view for our `indexAction`.
244+
And a very simple view for our `indexAction` in `app/Views/index/index.twig`:
246245

247246
```html
248-
<!-- app/Views/index/index.twig -->
249247
{% extends 'layout.twig' %}
250248

251249
{% block content %}
@@ -254,5 +252,6 @@ And a very simple view for our `indexAction`.
254252
</div>
255253
{% endblock %}
256254
```
255+
257256
Once you add the `tutorial` folder to your standard web root, you should have
258257
a working application at `http://localhost/tutorial/`.

‎src/Vectorface/SnappyRouter/Handler/ControllerHandler.php

+3-1
Original file line numberDiff line numberDiff line change
@@ -94,8 +94,10 @@ public function isAppropriate($path, $query, $post, $verb)
9494
$this->request = new HttpRequest(
9595
ucfirst($controller).'Controller',
9696
$action.'Action',
97-
$verb
97+
$verb,
98+
'php://input'
9899
);
100+
99101
$this->request->setQuery($query);
100102
$this->request->setPost($post);
101103

‎src/Vectorface/SnappyRouter/Handler/RestHandler.php

+4-9
Original file line numberDiff line numberDiff line change
@@ -45,16 +45,11 @@ public function isAppropriate($path, $query, $post, $verb)
4545
if ($routeInfo[1] & self::MATCHES_ID) {
4646
$this->routeParams[] = intval($routeInfo[2]['objectId']);
4747
}
48-
return true;
49-
}
5048

51-
/**
52-
* Returns the active response encoder.
53-
* @return EncoderInterface Returns the response encoder.
54-
*/
55-
public function getEncoder()
56-
{
57-
return new JsonEncoder();
49+
// use JSON encoder by default
50+
$this->encoder = new JsonEncoder();
51+
52+
return true;
5853
}
5954

6055
/**

‎src/Vectorface/SnappyRouter/Request/HttpRequest.php

+60-2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,8 @@
22

33
namespace Vectorface\SnappyRouter\Request;
44

5+
use Vectorface\SnappyRouter\Exception\InternalErrorException;
6+
57
/**
68
* A class representing an controller-modelled web request.
79
* @copyright Copyright (c) 2014, VectorFace, Inc.
@@ -15,10 +17,15 @@ class HttpRequest extends AbstractRequest implements HttpRequestInterface
1517
/** Holds the contents of the various inputs ($_GET, $_POST, etc) */
1618
private $input;
1719

20+
/** The input stream (stream or location string) */
21+
private $stream;
22+
1823
/** Array key for query parameters */
1924
const INPUT_METHOD_QUERY = 'QUERY';
2025
/** Array key for post parameters */
2126
const INPUT_METHOD_POST = 'POST';
27+
/** Array key for input stream body */
28+
const INPUT_METHOD_BODY = 'BODY';
2229

2330
// mappings between magic filter strings and the filter functions
2431
private static $filterCallbacks = array(
@@ -35,14 +42,18 @@ class HttpRequest extends AbstractRequest implements HttpRequestInterface
3542
* @param string $controller The controller being requested.
3643
* @param string $action The action being invoked.
3744
* @param string $verb The HTTP verb used in the request.
45+
* @param mixed $stream A stream or a string describing a stream location.
3846
*/
39-
public function __construct($controller, $action, $verb)
47+
public function __construct($controller, $action, $verb, $stream = 'php://input')
4048
{
4149
parent::__construct($controller, $action);
4250
$this->setVerb($verb);
51+
$this->setStream($stream);
52+
4353
$this->input = array(
4454
self::INPUT_METHOD_QUERY => array(),
45-
self::INPUT_METHOD_POST => array()
55+
self::INPUT_METHOD_POST => array(),
56+
self::INPUT_METHOD_BODY => null
4657
);
4758
}
4859

@@ -66,6 +77,17 @@ public function setVerb($verb)
6677
return $this;
6778
}
6879

80+
/**
81+
* Sets the stream used in the request.
82+
* @param mixed $stream The stream used in the request.
83+
* @return RequestInterface Returns $this.
84+
*/
85+
public function setStream($stream)
86+
{
87+
$this->stream = $stream;
88+
return $this;
89+
}
90+
6991
/**
7092
* Returns true if the request is a GET request and false otherwise.
7193
* @return bool Returns true if the request is a GET request and false
@@ -144,6 +166,42 @@ public function setPost($postData)
144166
return $this;
145167
}
146168

169+
/**
170+
* Returns the input stream data for the current request
171+
* @return string The input stream data
172+
*/
173+
public function getBody()
174+
{
175+
// If this value has been read from the stream, retrieve it from storage
176+
if (null !== $this->input[self::INPUT_METHOD_BODY]) {
177+
return $this->input[self::INPUT_METHOD_BODY];
178+
}
179+
180+
if (is_resource($this->stream) && 'stream' === get_resource_type($this->stream)) {
181+
$streamData = stream_get_contents($this->stream);
182+
} elseif (is_string($this->stream)) {
183+
$stream = @fopen($this->stream, "r");
184+
185+
if (false === $stream) {
186+
throw new InternalErrorException('Unable to open request input stream.');
187+
}
188+
189+
$streamData = stream_get_contents($stream);
190+
191+
fclose($stream);
192+
} else {
193+
$streamData = false;
194+
}
195+
196+
if (false === $streamData) {
197+
throw new InternalErrorException('Unable to open request input stream.');
198+
}
199+
200+
$this->input[self::INPUT_METHOD_BODY] = $streamData;
201+
202+
return $this->input[self::INPUT_METHOD_BODY];
203+
}
204+
147205
/**
148206
* Returns an array of all the input parameters from the query and post data.
149207
* @return array An array of all input.

‎src/Vectorface/SnappyRouter/Request/JsonRpcRequest.php

+1-1
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ public function __construct($controller, $payload, $verb = 'POST')
3333
if (is_object($payload) && isset($payload->method)) {
3434
$action = $payload->method;
3535
}
36-
parent::__construct($controller, $action, $verb);
36+
parent::__construct($controller, $action, $verb, 'php://input');
3737
$this->payload = $payload;
3838
}
3939

‎src/Vectorface/SnappyRouter/SnappyRouter.php

+60-18
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,8 @@
33
namespace Vectorface\SnappyRouter;
44

55
use \Exception;
6+
use Psr\Log\LoggerInterface;
7+
use Psr\Log\NullLogger;
68
use Vectorface\SnappyRouter\Config\Config;
79
use Vectorface\SnappyRouter\Config\ConfigInterface;
810
use Vectorface\SnappyRouter\Di\Di;
@@ -26,6 +28,11 @@ class SnappyRouter
2628
private $config; // the configuration
2729
private $handlers; // array of registered handlers
2830

31+
/**
32+
* @var LoggerInterface
33+
*/
34+
private $logger;
35+
2936
/**
3037
* The constructor for the service router.
3138
* @param array $config The configuration array.
@@ -34,6 +41,17 @@ public function __construct(ConfigInterface $config)
3441
{
3542
$this->config = $config;
3643
$this->parseConfig();
44+
$this->logger = new NullLogger();
45+
}
46+
47+
/**
48+
* Configure SnappyRouter to log its actions.
49+
*
50+
* @param LoggerInterface $logger
51+
*/
52+
public function setLogger(LoggerInterface $logger)
53+
{
54+
$this->logger = $logger;
3755
}
3856

3957
/**
@@ -85,6 +103,7 @@ public function handleRoute($environment = null)
85103
*/
86104
public function handleHttpRoute($path, $query, $post, $verb)
87105
{
106+
$this->logger->debug("SnappyRouter: Handling HTTP route: $path");
88107
return $this->invokeHandler(false, array($path, $query, $post, $verb));
89108
}
90109

@@ -95,6 +114,7 @@ public function handleHttpRoute($path, $query, $post, $verb)
95114
*/
96115
public function handleCliRoute($pathComponents)
97116
{
117+
$this->logger->debug("SnappyRouter: Handling CLI route: " . implode("/", $pathComponents));
98118
return $this->invokeHandler(true, array($pathComponents));
99119
}
100120

@@ -111,39 +131,61 @@ private function invokeHandler($isCli, $handlerParams)
111131
try {
112132
// determine which handler should handle this path
113133
$activeHandler = $this->determineHandler($isCli, $handlerParams);
134+
$this->logger->debug("SnappyRouter: Selected handler: " . get_class($activeHandler));
114135
// invoke the initial plugin hook
115136
$activeHandler->invokePluginsHook(
116137
'afterHandlerSelected',
117138
array($activeHandler)
118139
);
140+
$this->logger->debug("SnappyRouter: routing");
119141
$response = $activeHandler->performRoute();
120142
$activeHandler->invokePluginsHook(
121143
'afterFullRouteInvoked',
122144
array($activeHandler)
123145
);
124146
return $response;
125147
} catch (Exception $e) {
126-
// if we have a valid handler give it a chance to handle the error
127-
if (null !== $activeHandler) {
128-
$activeHandler->invokePluginsHook(
129-
'errorOccurred',
130-
array($activeHandler, $e)
131-
);
132-
return $activeHandler->getEncoder()->encode(
133-
new Response($activeHandler->handleException($e))
134-
);
135-
}
148+
return $this->handleInvocationException($e, $activeHandler, $isCli);
149+
}
150+
}
151+
152+
/**
153+
* Attempts to mop up after an exception during handler invocation.
154+
*
155+
* @param \Exception $exception The exception that occurred during invocation.
156+
* @param HandlerInterface $activeHandler The active handler, or null.
157+
* @param bool $isCli True for CLI handlers, false otherwise.
158+
* @return mixed Returns a handler-dependent response type, usually a string.
159+
*/
160+
private function handleInvocationException($exception, $activeHandler, $isCli)
161+
{
162+
$this->logger->debug(sprintf(
163+
"SnappyRouter: caught exception while invoking handler: %s (%d)",
164+
$exception->getMessage(),
165+
$exception->getCode()
166+
));
136167

137-
// if not on the command line, set an HTTP response code
138-
if (!$isCli) {
139-
$responseCode = AbstractResponse::RESPONSE_SERVER_ERROR;
140-
if ($e instanceof RouterExceptionInterface) {
141-
$responseCode = $e->getAssociatedStatusCode();
142-
}
143-
\Vectorface\SnappyRouter\http_response_code($responseCode);
168+
// if we have a valid handler give it a chance to handle the error
169+
if (null !== $activeHandler) {
170+
$activeHandler->invokePluginsHook(
171+
'errorOccurred',
172+
array($activeHandler, $exception)
173+
);
174+
return $activeHandler->getEncoder()->encode(
175+
new Response($activeHandler->handleException($exception))
176+
);
177+
}
178+
179+
// if not on the command line, set an HTTP response code
180+
if (!$isCli) {
181+
$responseCode = AbstractResponse::RESPONSE_SERVER_ERROR;
182+
if ($exception instanceof RouterExceptionInterface) {
183+
$responseCode = $exception->getAssociatedStatusCode();
144184
}
145-
return $e->getMessage();
185+
\Vectorface\SnappyRouter\http_response_code($responseCode);
146186
}
187+
return $exception->getMessage();
188+
147189
}
148190

149191
/**

0 commit comments

Comments
 (0)
Please sign in to comment.