This commit is contained in:
Marcel Pociot
2020-04-29 22:05:03 +02:00
parent 6cf206e0a2
commit b515a55325
27 changed files with 215 additions and 253 deletions

View File

@@ -104,7 +104,7 @@ class Factory
$this->router->post('/logs', PushLogsToDashboardController::class); $this->router->post('/logs', PushLogsToDashboardController::class);
$this->router->get('/replay/{log}', ReplayLogController::class); $this->router->get('/replay/{log}', ReplayLogController::class);
$this->router->post('/logs/{request_id}/data', AttachDataToLogController::class); $this->router->post('/logs/{request_id}/data', AttachDataToLogController::class);
$this->router->post('/logs/clear', ClearLogsController::class); $this->router->get('/logs/clear', ClearLogsController::class);
$this->app->route('/socket', new WsServer(new Socket()), ['*']); $this->app->route('/socket', new WsServer(new Socket()), ['*']);

View File

@@ -2,14 +2,14 @@
namespace App\Client\Http\Controllers; namespace App\Client\Http\Controllers;
use App\Http\Controllers\PostController; use App\Http\Controllers\Controller;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use App\Logger\RequestLogger; use App\Logger\RequestLogger;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
class AttachDataToLogController extends PostController class AttachDataToLogController extends Controller
{ {
/** @var RequestLogger */ /** @var RequestLogger */
protected $requestLogger; protected $requestLogger;

View File

@@ -2,14 +2,10 @@
namespace App\Client\Http\Controllers; namespace App\Client\Http\Controllers;
use App\Client\Http\HttpClient;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Http\QueryParameters;
use App\Logger\RequestLogger; use App\Logger\RequestLogger;
use GuzzleHttp\Psr7\Response; use Illuminate\Http\Request;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
use function GuzzleHttp\Psr7\str;
use Psr\Http\Message\RequestInterface;
class ClearLogsController extends Controller class ClearLogsController extends Controller
{ {
@@ -21,18 +17,10 @@ class ClearLogsController extends Controller
$this->requestLogger = $requestLogger; $this->requestLogger = $requestLogger;
} }
public function onOpen(ConnectionInterface $connection, RequestInterface $request = null) public function handle(Request $request, ConnectionInterface $httpConnection)
{ {
$this->requestLogger->clear(); $this->requestLogger->clear();
$connection->send( $httpConnection->send(respond_json([], 200));
str(new Response(
200,
['Content-Type' => 'application/json'],
''
))
);
$connection->close();
} }
} }

View File

@@ -5,24 +5,18 @@ namespace App\Client\Http\Controllers;
use App\Client\Client; use App\Client\Client;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
use Psr\Http\Message\RequestInterface; use Psr\Http\Message\RequestInterface;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
class DashboardController extends Controller class DashboardController extends Controller
{ {
public function onOpen(ConnectionInterface $connection, RequestInterface $request = null)
{
$connection->send(
str(new Response(
200,
['Content-Type' => 'text/html'],
$this->getView('client.dashboard', [
'subdomains' => Client::$subdomains,
])
))
);
$connection->close(); public function handle(Request $request, ConnectionInterface $httpConnection)
{
$httpConnection->send(respond_html($this->getView('client.dashboard', [
'subdomains' => Client::$subdomains,
])));
} }
} }

View File

@@ -5,6 +5,7 @@ namespace App\Client\Http\Controllers;
use App\Http\Controllers\Controller; use App\Http\Controllers\Controller;
use App\Logger\RequestLogger; use App\Logger\RequestLogger;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
use Psr\Http\Message\RequestInterface; use Psr\Http\Message\RequestInterface;
@@ -19,16 +20,8 @@ class LogController extends Controller
$this->requestLogger = $requestLogger; $this->requestLogger = $requestLogger;
} }
public function onOpen(ConnectionInterface $connection, RequestInterface $request = null) public function handle(Request $request, ConnectionInterface $httpConnection)
{ {
$connection->send( $httpConnection->send(respond_json($this->requestLogger->getData()));
str(new Response(
200,
['Content-Type' => 'application/json'],
json_encode($this->requestLogger->getData(), JSON_INVALID_UTF8_IGNORE)
))
);
$connection->close();
} }
} }

View File

@@ -6,6 +6,7 @@ use App\Http\Controllers\Controller;
use Exception; use Exception;
use App\WebSockets\Socket; use App\WebSockets\Socket;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use Illuminate\Support\Collection; use Illuminate\Support\Collection;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
@@ -14,50 +15,20 @@ use Psr\Http\Message\RequestInterface;
class PushLogsToDashboardController extends Controller class PushLogsToDashboardController extends Controller
{ {
public function onOpen(ConnectionInterface $connection, RequestInterface $request = null) public function handle(Request $request, ConnectionInterface $httpConnection)
{ {
$connection->contentLength = $this->findContentLength($request->getHeaders()); try {
/*
$connection->requestBuffer = (string) $request->getBody(); * This is the post payload from our PHPUnit tests.
* Send it to the connected connections.
$this->checkContentLength($connection); */
} foreach (Socket::$connections as $webSocketConnection) {
$webSocketConnection->send($request->getContent());
public function onMessage(ConnectionInterface $from, $msg)
{
$from->requestBuffer .= $msg;
$this->checkContentLength($from);
}
protected function findContentLength(array $headers): int
{
return Collection::make($headers)->first(function ($values, $header) {
return strtolower($header) === 'content-length';
})[0] ?? 0;
}
protected function checkContentLength(ConnectionInterface $connection)
{
if (strlen($connection->requestBuffer) === $connection->contentLength) {
try {
/*
* This is the post payload from our PHPUnit tests.
* Send it to the connected connections.
*/
foreach (Socket::$connections as $webSocketConnection) {
$webSocketConnection->send($connection->requestBuffer);
}
$connection->send(str(new Response(200)));
} catch (Exception $e) {
$connection->send(str(new Response(500, [], $e->getMessage())));
} }
$connection->close(); $httpConnection->send(str(new Response(200)));
} catch (Exception $e) {
unset($connection->requestBuffer); $httpConnection->send(str(new Response(500, [], $e->getMessage())));
unset($connection->contentLength);
} }
} }
} }

View File

@@ -7,6 +7,7 @@ use App\Http\Controllers\Controller;
use App\Http\QueryParameters; use App\Http\QueryParameters;
use App\Logger\RequestLogger; use App\Logger\RequestLogger;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
use Psr\Http\Message\RequestInterface; use Psr\Http\Message\RequestInterface;
@@ -25,35 +26,17 @@ class ReplayLogController extends Controller
$this->httpClient = $httpClient; $this->httpClient = $httpClient;
} }
public function onOpen(ConnectionInterface $connection, RequestInterface $request = null) public function handle(Request $request, ConnectionInterface $httpConnection)
{ {
$loggedRequest = $this->requestLogger->findLoggedRequest(QueryParameters::create($request)->get('log')); $loggedRequest = $this->requestLogger->findLoggedRequest($request->get('log'));
if (is_null($loggedRequest)) { if (is_null($loggedRequest)) {
$connection->send( $httpConnection->send(str(new Response(404)));
str(new Response(
404,
['Content-Type' => 'application/json'],
))
);
$connection->close();
return; return;
} }
$requestData = $loggedRequest->getRequestData(); $this->httpClient->performRequest($loggedRequest->getRequestData());
/** @var HttpClient $tunnel */ $httpConnection->send(str(new Response(200)));
$this->httpClient->performRequest($requestData);
$connection->send(
str(new Response(
200,
['Content-Type' => 'application/json'],
''
))
);
$connection->close();
} }
} }

View File

@@ -3,6 +3,7 @@
namespace App\Client\Http\Modifiers; namespace App\Client\Http\Modifiers;
use App\Client\Configuration; use App\Client\Configuration;
use GuzzleHttp\Psr7\Response;
use Illuminate\Support\Arr; use Illuminate\Support\Arr;
use Psr\Http\Message\RequestInterface; use Psr\Http\Message\RequestInterface;
use Ratchet\Client\WebSocket; use Ratchet\Client\WebSocket;
@@ -28,7 +29,7 @@ class CheckBasicAuthentication
if (is_null($username)) { if (is_null($username)) {
$proxyConnection->send( $proxyConnection->send(
str(new \GuzzleHttp\Psr7\Response(401, [ str(new Response(401, [
'WWW-Authenticate' => 'Basic realm=Expose' 'WWW-Authenticate' => 'Basic realm=Expose'
], 'Unauthorized')) ], 'Unauthorized'))
); );

View File

@@ -0,0 +1,24 @@
<?php
namespace App\Http\Controllers\Concerns;
use Twig\Environment;
use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\stream_for;
trait LoadsViews
{
protected function getView(string $view, array $data = [])
{
$templatePath = implode(DIRECTORY_SEPARATOR, explode('.', $view));
$twig = new Environment(
new ArrayLoader([
'app' => file_get_contents(base_path('resources/views/server/layouts/app.twig')),
'template' => file_get_contents(base_path('resources/views/'.$templatePath.'.twig')),
])
);
return stream_for($twig->render('template', $data));
}
}

View File

@@ -1,45 +1,16 @@
<?php <?php
namespace App\Http\Controllers; namespace App\Http\Controllers\Concerns;
use App\Http\QueryParameters; use App\Http\QueryParameters;
use GuzzleHttp\Psr7\ServerRequest; use GuzzleHttp\Psr7\ServerRequest;
use Illuminate\Http\Request; use Illuminate\Http\Request;
use Illuminate\Support\Collection; use Illuminate\Support\Collection;
use Psr\Http\Message\RequestInterface;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
use Symfony\Bridge\PsrHttpMessage\Factory\HttpFoundationFactory; use Symfony\Bridge\PsrHttpMessage\Factory\HttpFoundationFactory;
use function GuzzleHttp\Psr7\parse_request;
abstract class PostController extends Controller trait ParsesIncomingRequest
{ {
protected $keepConnectionOpen = false;
public function onOpen(ConnectionInterface $connection, RequestInterface $request = null)
{
$connection->contentLength = $this->findContentLength($request->getHeaders());
$connection->requestBuffer = (string) $request->getBody();
$connection->request = $request;
$this->checkContentLength($connection);
}
public function onMessage(ConnectionInterface $from, $msg)
{
if (! isset($from->requestBuffer)) {
$request = parse_request($msg);
$from->contentLength = $this->findContentLength($request->getHeaders());
$from->request = $request;
$from->requestBuffer = (string) $request->getBody();
} else {
$from->requestBuffer .= $msg;
}
$this->checkContentLength($from);
}
protected function findContentLength(array $headers): int protected function findContentLength(array $headers): int
{ {
return Collection::make($headers)->first(function ($values, $header) { return Collection::make($headers)->first(function ($values, $header) {
@@ -47,14 +18,21 @@ abstract class PostController extends Controller
})[0] ?? 0; })[0] ?? 0;
} }
protected function shouldHandleRequest(Request $request, ConnectionInterface $httpConnection): bool
{
return true;
}
protected function checkContentLength(ConnectionInterface $connection) protected function checkContentLength(ConnectionInterface $connection)
{ {
if (strlen($connection->requestBuffer) === $connection->contentLength) { if (strlen($connection->requestBuffer) === $connection->contentLength) {
$laravelRequest = $this->createLaravelRequest($connection); $laravelRequest = $this->createLaravelRequest($connection);
$this->handle($laravelRequest, $connection); if ($this->shouldHandleRequest($laravelRequest, $connection)) {
$this->handle($laravelRequest, $connection);
}
if (! $this->keepConnectionOpen) { if (!$this->keepConnectionOpen) {
$connection->close(); $connection->close();
} }
@@ -64,8 +42,6 @@ abstract class PostController extends Controller
} }
} }
abstract public function handle(Request $request, ConnectionInterface $httpConnection);
protected function createLaravelRequest(ConnectionInterface $connection): Request protected function createLaravelRequest(ConnectionInterface $connection): Request
{ {
try { try {

View File

@@ -2,15 +2,30 @@
namespace App\Http\Controllers; namespace App\Http\Controllers;
use Exception; use App\Http\Controllers\Concerns\LoadsViews;
use App\Http\Controllers\Concerns\ParsesIncomingRequest;
use Illuminate\Http\Request;
use Psr\Http\Message\RequestInterface;
use Ratchet\ConnectionInterface; use Ratchet\ConnectionInterface;
use Ratchet\Http\HttpServerInterface; use Ratchet\Http\HttpServerInterface;
use Twig\Environment;
use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\stream_for;
abstract class Controller implements HttpServerInterface abstract class Controller implements HttpServerInterface
{ {
use LoadsViews, ParsesIncomingRequest;
protected $keepConnectionOpen = false;
public function onOpen(ConnectionInterface $connection, RequestInterface $request = null)
{
$connection->contentLength = $this->findContentLength($request->getHeaders());
$connection->requestBuffer = (string) $request->getBody();
$connection->request = $request;
$this->checkContentLength($connection);
}
public function onClose(ConnectionInterface $connection) public function onClose(ConnectionInterface $connection)
{ {
unset($connection->requestBuffer); unset($connection->requestBuffer);
@@ -18,25 +33,24 @@ abstract class Controller implements HttpServerInterface
unset($connection->request); unset($connection->request);
} }
public function onError(ConnectionInterface $connection, Exception $e)
{
}
public function onMessage(ConnectionInterface $from, $msg) public function onMessage(ConnectionInterface $from, $msg)
{ {
if (! isset($from->requestBuffer)) {
$request = parse_request($msg);
$from->contentLength = $this->findContentLength($request->getHeaders());
$from->request = $request;
$from->requestBuffer = (string) $request->getBody();
} else {
$from->requestBuffer .= $msg;
}
$this->checkContentLength($from);
} }
protected function getView(string $view, array $data = []) function onError(ConnectionInterface $conn, \Exception $e)
{ {
$templatePath = implode(DIRECTORY_SEPARATOR, explode('.', $view)); //
$twig = new Environment(
new ArrayLoader([
'app' => file_get_contents(base_path('resources/views/server/layouts/app.twig')),
'template' => file_get_contents(base_path('resources/views/'.$templatePath.'.twig')),
])
);
return stream_for($twig->render('template', $data));
} }
abstract public function handle(Request $request, ConnectionInterface $httpConnection);
} }

View File

@@ -10,6 +10,7 @@ use App\Server\Http\Controllers\Admin\DeleteUsersController;
use App\Server\Http\Controllers\Admin\ListSitesController; use App\Server\Http\Controllers\Admin\ListSitesController;
use App\Server\Http\Controllers\Admin\ListUsersController; use App\Server\Http\Controllers\Admin\ListUsersController;
use App\Server\Http\Controllers\Admin\LoginController; use App\Server\Http\Controllers\Admin\LoginController;
use App\Server\Http\Controllers\Admin\RedirectToUsersController;
use App\Server\Http\Controllers\Admin\StoreUsersController; use App\Server\Http\Controllers\Admin\StoreUsersController;
use App\Server\Http\Controllers\Admin\VerifyLoginController; use App\Server\Http\Controllers\Admin\VerifyLoginController;
use App\Server\Http\Controllers\ControlMessageController; use App\Server\Http\Controllers\ControlMessageController;
@@ -106,10 +107,9 @@ class Factory
protected function addAdminRoutes() protected function addAdminRoutes()
{ {
$adminCondition = 'request.headers.get("Host") matches "/'.config('expose.dashboard_subdomain').'\./i"'; $adminCondition = 'request.headers.get("Host") matches "/'.config('expose.admin.subdomain').'\./i"';
$this->router->get('/', LoginController::class, $adminCondition); $this->router->get('/', RedirectToUsersController::class, $adminCondition);
$this->router->post('/', VerifyLoginController::class, $adminCondition);
$this->router->get('/users', ListUsersController::class, $adminCondition); $this->router->get('/users', ListUsersController::class, $adminCondition);
$this->router->post('/users', StoreUsersController::class, $adminCondition); $this->router->post('/users', StoreUsersController::class, $adminCondition);
$this->router->delete('/users/delete/{id}', DeleteUsersController::class, $adminCondition); $this->router->delete('/users/delete/{id}', DeleteUsersController::class, $adminCondition);

View File

@@ -0,0 +1,37 @@
<?php
namespace App\Server\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use Illuminate\Support\Str;
use Ratchet\ConnectionInterface;
use function GuzzleHttp\Psr7\str;
abstract class AdminController extends Controller
{
protected function shouldHandleRequest(Request $request, ConnectionInterface $httpConnection): bool
{
try {
$authorization = Str::after($request->header('Authorization'), 'Basic ');
$authParts = explode(':', base64_decode($authorization), 2);
list($user, $password) = $authParts;
if (! $this->credentialsAreAllowed($user, $password)) {
throw new \InvalidArgumentException('Invalid Login');
}
return true;
} catch (\Throwable $e) {
$httpConnection->send(str(new Response(401, [
'WWW-Authenticate' => 'Basic realm="Expose"'
], 'foo')));
}
return false;
}
protected function credentialsAreAllowed(string $user, string $password)
{
return config('expose.admin.users.'.$user) === $password;
}
}

View File

@@ -2,7 +2,7 @@
namespace App\Server\Http\Controllers\Admin; namespace App\Server\Http\Controllers\Admin;
use App\Http\Controllers\PostController; use App\Http\Controllers\Controller;
use Clue\React\SQLite\DatabaseInterface; use Clue\React\SQLite\DatabaseInterface;
use Clue\React\SQLite\Result; use Clue\React\SQLite\Result;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
@@ -15,7 +15,7 @@ use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
use function GuzzleHttp\Psr7\stream_for; use function GuzzleHttp\Psr7\stream_for;
class DeleteUsersController extends PostController class DeleteUsersController extends AdminController
{ {
protected $keepConnectionOpen = true; protected $keepConnectionOpen = true;

View File

@@ -3,7 +3,7 @@
namespace App\Server\Http\Controllers\Admin; namespace App\Server\Http\Controllers\Admin;
use App\Contracts\ConnectionManager; use App\Contracts\ConnectionManager;
use App\Http\Controllers\PostController; use App\Http\Controllers\Controller;
use App\Server\Configuration; use App\Server\Configuration;
use Clue\React\SQLite\DatabaseInterface; use Clue\React\SQLite\DatabaseInterface;
use Clue\React\SQLite\Result; use Clue\React\SQLite\Result;
@@ -15,7 +15,7 @@ use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
use function GuzzleHttp\Psr7\stream_for; use function GuzzleHttp\Psr7\stream_for;
class ListSitesController extends PostController class ListSitesController extends AdminController
{ {
/** @var ConnectionManager */ /** @var ConnectionManager */
protected $connectionManager; protected $connectionManager;

View File

@@ -2,7 +2,7 @@
namespace App\Server\Http\Controllers\Admin; namespace App\Server\Http\Controllers\Admin;
use App\Http\Controllers\PostController; use App\Http\Controllers\Controller;
use Clue\React\SQLite\DatabaseInterface; use Clue\React\SQLite\DatabaseInterface;
use Clue\React\SQLite\Result; use Clue\React\SQLite\Result;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
@@ -13,7 +13,7 @@ use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
use function GuzzleHttp\Psr7\stream_for; use function GuzzleHttp\Psr7\stream_for;
class ListUsersController extends PostController class ListUsersController extends AdminController
{ {
protected $keepConnectionOpen = true; protected $keepConnectionOpen = true;

View File

@@ -1,25 +0,0 @@
<?php
namespace App\Server\Http\Controllers\Admin;
use App\Contracts\ConnectionManager;
use App\Http\Controllers\PostController;
use Clue\React\SQLite\DatabaseInterface;
use Clue\React\SQLite\Result;
use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use Ratchet\ConnectionInterface;
use Twig\Environment;
use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\str;
use function GuzzleHttp\Psr7\stream_for;
class LoginController extends PostController
{
public function handle(Request $request, ConnectionInterface $httpConnection)
{
$httpConnection->send(
respond_html($this->getView('server.login'))
);
}
}

View File

@@ -0,0 +1,20 @@
<?php
namespace App\Server\Http\Controllers\Admin;
use App\Http\Controllers\Controller;
use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use Illuminate\Support\Str;
use Ratchet\ConnectionInterface;
use function GuzzleHttp\Psr7\str;
class RedirectToUsersController extends AdminController
{
public function handle(Request $request, ConnectionInterface $httpConnection)
{
$httpConnection->send(str(new Response(301, [
'Location' => '/sites'
])));
}
}

View File

@@ -2,7 +2,7 @@
namespace App\Server\Http\Controllers\Admin; namespace App\Server\Http\Controllers\Admin;
use App\Http\Controllers\PostController; use App\Http\Controllers\Controller;
use Clue\React\SQLite\DatabaseInterface; use Clue\React\SQLite\DatabaseInterface;
use Clue\React\SQLite\Result; use Clue\React\SQLite\Result;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
@@ -15,7 +15,7 @@ use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
use function GuzzleHttp\Psr7\stream_for; use function GuzzleHttp\Psr7\stream_for;
class StoreUsersController extends PostController class StoreUsersController extends AdminController
{ {
protected $keepConnectionOpen = true; protected $keepConnectionOpen = true;

View File

@@ -1,51 +0,0 @@
<?php
namespace App\Server\Http\Controllers\Admin;
use App\Contracts\ConnectionManager;
use App\Http\Controllers\PostController;
use Clue\React\SQLite\DatabaseInterface;
use Clue\React\SQLite\Result;
use GuzzleHttp\Psr7\Response;
use Illuminate\Http\Request;
use Ratchet\ConnectionInterface;
use Twig\Environment;
use Twig\Loader\ArrayLoader;
use function GuzzleHttp\Psr7\str;
use function GuzzleHttp\Psr7\stream_for;
class VerifyLoginController extends PostController
{
protected $keepConnectionOpen = true;
/** @var DatabaseInterface */
protected $database;
public function __construct(DatabaseInterface $database)
{
$this->database = $database;
}
public function handle(Request $request, ConnectionInterface $httpConnection)
{
$this->database->query("SELECT * FROM users WHERE email = :email", ['email' => $request->email])
->then(function (Result $result) use ($httpConnection) {
if (!is_null($result->rows)) {
$httpConnection->send(
str(new Response(
301,
['Location' => '/users']
))
);
} else {
$httpConnection->send(
str(new Response(
301,
['Location' => '/users']
))
);
}
$httpConnection->close();
});
}
}

View File

@@ -131,7 +131,7 @@ class ControlMessageController implements MessageComponentInterface
{ {
if (! is_null($subdomain)) { if (! is_null($subdomain)) {
$controlConnection = $this->connectionManager->findControlConnectionForSubdomain($subdomain); $controlConnection = $this->connectionManager->findControlConnectionForSubdomain($subdomain);
if (! is_null($controlConnection) || $subdomain === config('expose.dashboard_subdomain')) { if (! is_null($controlConnection) || $subdomain === config('expose.admin.subdomain')) {
$connection->send(json_encode([ $connection->send(json_encode([
'event' => 'subdomainTaken', 'event' => 'subdomainTaken',
'data' => [ 'data' => [

View File

@@ -3,7 +3,7 @@
namespace App\Server\Http\Controllers; namespace App\Server\Http\Controllers;
use App\Contracts\ConnectionManager; use App\Contracts\ConnectionManager;
use App\Http\Controllers\PostController; use App\Http\Controllers\Controller;
use App\Server\Configuration; use App\Server\Configuration;
use App\Server\Connections\ControlConnection; use App\Server\Connections\ControlConnection;
use GuzzleHttp\Psr7\Response; use GuzzleHttp\Psr7\Response;
@@ -16,7 +16,7 @@ use Ratchet\RFC6455\Messaging\Frame;
use Symfony\Bridge\PsrHttpMessage\Factory\PsrHttpFactory; use Symfony\Bridge\PsrHttpMessage\Factory\PsrHttpFactory;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
class TunnelMessageController extends PostController class TunnelMessageController extends Controller
{ {
/** @var ConnectionManager */ /** @var ConnectionManager */
protected $connectionManager; protected $connectionManager;

View File

@@ -8,7 +8,7 @@ function respond_json($responseData, int $statusCode = 200)
return str(new Response( return str(new Response(
$statusCode, $statusCode,
['Content-Type' => 'application/json'], ['Content-Type' => 'application/json'],
json_encode($responseData) json_encode($responseData, JSON_INVALID_UTF8_IGNORE)
)); ));
} }

View File

@@ -4,5 +4,33 @@ return [
'host' => 'expose.dev', 'host' => 'expose.dev',
'port' => 8080, 'port' => 8080,
'auth_token' => '', 'auth_token' => '',
'dashboard_subdomain' => 'expose',
'admin' => [
/*
|--------------------------------------------------------------------------
| Subdomain
|--------------------------------------------------------------------------
|
| This is the subdomain that your expose admin dashboard will be available at.
| The given subdomain will be reserved, so no other tunnel connection can
| request this subdomain for their own connection.
|
*/
'subdomain' => 'expose',
/*
|--------------------------------------------------------------------------
| Users
|--------------------------------------------------------------------------
|
| The admin dashboard of expose is protected via HTTP basic authentication
| Here you may add the user/password combinations that you want to
| accept as valid logins for the dashboard.
|
*/
'users' => [
'username' => 'password'
]
]
]; ];

View File

@@ -15,6 +15,7 @@ use Psr\Http\Message\ResponseInterface;
use React\EventLoop\LoopInterface; use React\EventLoop\LoopInterface;
use Symfony\Component\Console\Output\ConsoleOutput; use Symfony\Component\Console\Output\ConsoleOutput;
use Symfony\Component\Console\Output\ConsoleOutputInterface; use Symfony\Component\Console\Output\ConsoleOutputInterface;
use Tests\Feature\TestCase;
use function GuzzleHttp\Psr7\str; use function GuzzleHttp\Psr7\str;
class DashboardTest extends TestCase class DashboardTest extends TestCase

View File

@@ -0,0 +1,8 @@
<?php
namespace Tests\Feature\Server;
class AdminTest
{
}

View File

@@ -1,6 +1,6 @@
<?php <?php
namespace Tests\Feature\Client; namespace Tests\Feature;
use React\EventLoop\Factory; use React\EventLoop\Factory;
use React\EventLoop\LoopInterface; use React\EventLoop\LoopInterface;
@@ -12,7 +12,7 @@ use function Clue\React\Block\await;
abstract class TestCase extends \Tests\TestCase abstract class TestCase extends \Tests\TestCase
{ {
const AWAIT_TIMEOUT = 1.0; const AWAIT_TIMEOUT = 5.0;
/** @var LoopInterface */ /** @var LoopInterface */
protected $loop; protected $loop;