mirror of
https://github.com/bitinflow/expose.git
synced 2026-03-16 23:15:55 +00:00
wip
This commit is contained in:
@@ -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()), ['*']);
|
||||||
|
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,
|
||||||
|
])));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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'))
|
||||||
);
|
);
|
||||||
|
|||||||
24
app/Http/Controllers/Concerns/LoadsViews.php
Normal file
24
app/Http/Controllers/Concerns/LoadsViews.php
Normal 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));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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 {
|
||||||
@@ -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);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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);
|
||||||
|
|||||||
37
app/Server/Http/Controllers/Admin/AdminController.php
Normal file
37
app/Server/Http/Controllers/Admin/AdminController.php
Normal 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;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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;
|
||||||
|
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|
||||||
|
|||||||
@@ -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'))
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -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'
|
||||||
|
])));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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;
|
||||||
|
|
||||||
|
|||||||
@@ -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();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -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' => [
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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)
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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'
|
||||||
|
]
|
||||||
|
]
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
8
tests/Feature/Server/AdminTest.php
Normal file
8
tests/Feature/Server/AdminTest.php
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Tests\Feature\Server;
|
||||||
|
|
||||||
|
class AdminTest
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
@@ -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;
|
||||||
Reference in New Issue
Block a user