mirror of
https://github.com/bitinflow/expose.git
synced 2026-03-15 06:25:56 +00:00
Merge branch 'master' of github.com:beyondcode/expose into master
This commit is contained in:
@@ -70,11 +70,9 @@ class TunnelMessageController extends Controller
|
|||||||
|
|
||||||
protected function detectSubdomain(Request $request): ?string
|
protected function detectSubdomain(Request $request): ?string
|
||||||
{
|
{
|
||||||
if (substr_count($request->header('Host'), '.') === 1) {
|
$serverHost = $this->detectServerHost($request);
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$subdomain = Str::before($request->header('Host'), '.');
|
$subdomain = Str::before($request->header('Host'), '.'.$serverHost);
|
||||||
|
|
||||||
return $subdomain === $request->header('Host') ? null : $subdomain;
|
return $subdomain === $request->header('Host') ? null : $subdomain;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -135,7 +135,7 @@ class DatabaseSubdomainRepository implements SubdomainRepository
|
|||||||
{
|
{
|
||||||
$deferred = new Deferred();
|
$deferred = new Deferred();
|
||||||
|
|
||||||
$this->database->query('DELETE FROM subdomains WHERE id = :id AND user_id = :user_id', [
|
$this->database->query('DELETE FROM subdomains WHERE (id = :id OR subdomain = :id) AND user_id = :user_id', [
|
||||||
'id' => $subdomainId,
|
'id' => $subdomainId,
|
||||||
'user_id' => $userId,
|
'user_id' => $userId,
|
||||||
])
|
])
|
||||||
|
|||||||
@@ -150,15 +150,38 @@ class DatabaseUserRepository implements UserRepository
|
|||||||
{
|
{
|
||||||
$deferred = new Deferred();
|
$deferred = new Deferred();
|
||||||
|
|
||||||
$this->database->query("
|
$this->getUserByToken($data['auth_token'])
|
||||||
|
->then(function ($existingUser) use ($data, $deferred) {
|
||||||
|
if (is_null($existingUser)) {
|
||||||
|
$this->database->query("
|
||||||
INSERT INTO users (name, auth_token, can_specify_subdomains, can_specify_domains, can_share_tcp_ports, max_connections, created_at)
|
INSERT INTO users (name, auth_token, can_specify_subdomains, can_specify_domains, can_share_tcp_ports, max_connections, created_at)
|
||||||
VALUES (:name, :auth_token, :can_specify_subdomains, :can_specify_domains, :can_share_tcp_ports, :max_connections, DATETIME('now'))
|
VALUES (:name, :auth_token, :can_specify_subdomains, :can_specify_domains, :can_share_tcp_ports, :max_connections, DATETIME('now'))
|
||||||
", $data)
|
", $data)
|
||||||
->then(function (Result $result) use ($deferred) {
|
->then(function (Result $result) use ($deferred) {
|
||||||
$this->database->query('SELECT * FROM users WHERE id = :id', ['id' => $result->insertId])
|
$this->database->query('SELECT * FROM users WHERE id = :id', ['id' => $result->insertId])
|
||||||
->then(function (Result $result) use ($deferred) {
|
->then(function (Result $result) use ($deferred) {
|
||||||
$deferred->resolve($result->rows[0]);
|
$deferred->resolve($result->rows[0]);
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
$this->database->query('
|
||||||
|
UPDATE users
|
||||||
|
SET
|
||||||
|
name = :name,
|
||||||
|
can_specify_subdomains = :can_specify_subdomains,
|
||||||
|
can_specify_domains = :can_specify_domains,
|
||||||
|
can_share_tcp_ports = :can_share_tcp_ports,
|
||||||
|
max_connections = :max_connections
|
||||||
|
WHERE
|
||||||
|
auth_token = :auth_token
|
||||||
|
', $data)
|
||||||
|
->then(function (Result $result) use ($existingUser, $deferred) {
|
||||||
|
$this->database->query('SELECT * FROM users WHERE id = :id', ['id' => $existingUser['id']])
|
||||||
|
->then(function (Result $result) use ($deferred) {
|
||||||
|
$deferred->resolve($result->rows[0]);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
return $deferred->promise();
|
return $deferred->promise();
|
||||||
|
|||||||
@@ -65,6 +65,97 @@ class ApiTest extends TestCase
|
|||||||
$this->assertSame([], $users[0]->sites);
|
$this->assertSame([], $users[0]->sites);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @test */
|
||||||
|
public function it_can_specify_a_token_when_creating_a_user()
|
||||||
|
{
|
||||||
|
/** @var Response $response */
|
||||||
|
$this->await($this->browser->post('http://127.0.0.1:8080/api/users', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
], json_encode([
|
||||||
|
'name' => 'Marcel',
|
||||||
|
'token' => 'my-token',
|
||||||
|
])));
|
||||||
|
|
||||||
|
/** @var Response $response */
|
||||||
|
$response = $this->await($this->browser->get('http://127.0.0.1:8080/api/users', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
]));
|
||||||
|
|
||||||
|
$body = json_decode($response->getBody()->getContents());
|
||||||
|
$users = $body->paginated->users;
|
||||||
|
|
||||||
|
$this->assertCount(1, $users);
|
||||||
|
$this->assertSame('Marcel', $users[0]->name);
|
||||||
|
$this->assertSame('my-token', $users[0]->auth_token);
|
||||||
|
$this->assertSame([], $users[0]->sites);
|
||||||
|
}
|
||||||
|
|
||||||
|
/** @test */
|
||||||
|
public function it_updates_users_instead_of_creating_new_ones()
|
||||||
|
{
|
||||||
|
/** @var Response $response */
|
||||||
|
$this->await($this->browser->post('http://127.0.0.1:8080/api/users', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
], json_encode([
|
||||||
|
'name' => 'Marcel',
|
||||||
|
'token' => 'my-token',
|
||||||
|
])));
|
||||||
|
|
||||||
|
/** @var Response $response */
|
||||||
|
$response = $this->await($this->browser->get('http://127.0.0.1:8080/api/users', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
]));
|
||||||
|
|
||||||
|
$body = json_decode($response->getBody()->getContents());
|
||||||
|
$users = $body->paginated->users;
|
||||||
|
|
||||||
|
$this->assertCount(1, $users);
|
||||||
|
$this->assertSame('Marcel', $users[0]->name);
|
||||||
|
$this->assertSame('my-token', $users[0]->auth_token);
|
||||||
|
$this->assertSame(0, $users[0]->can_specify_subdomains);
|
||||||
|
$this->assertSame(0, $users[0]->can_specify_domains);
|
||||||
|
$this->assertSame(0, $users[0]->can_share_tcp_ports);
|
||||||
|
$this->assertSame([], $users[0]->sites);
|
||||||
|
|
||||||
|
$this->await($this->browser->post('http://127.0.0.1:8080/api/users', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
], json_encode([
|
||||||
|
'name' => 'Marcel Changed',
|
||||||
|
'token' => 'my-token',
|
||||||
|
'can_specify_subdomains' => 1,
|
||||||
|
'can_specify_domains' => 1,
|
||||||
|
'can_share_tcp_ports' => 1,
|
||||||
|
])));
|
||||||
|
|
||||||
|
/** @var Response $response */
|
||||||
|
$response = $this->await($this->browser->get('http://127.0.0.1:8080/api/users', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
]));
|
||||||
|
|
||||||
|
$body = json_decode($response->getBody()->getContents());
|
||||||
|
$users = $body->paginated->users;
|
||||||
|
|
||||||
|
$this->assertCount(1, $users);
|
||||||
|
$this->assertSame('Marcel Changed', $users[0]->name);
|
||||||
|
$this->assertSame('my-token', $users[0]->auth_token);
|
||||||
|
$this->assertSame(1, $users[0]->can_specify_subdomains);
|
||||||
|
$this->assertSame(1, $users[0]->can_specify_domains);
|
||||||
|
$this->assertSame(1, $users[0]->can_share_tcp_ports);
|
||||||
|
$this->assertSame([], $users[0]->sites);
|
||||||
|
}
|
||||||
|
|
||||||
/** @test */
|
/** @test */
|
||||||
public function it_can_specify_tokens_when_creating_a_user()
|
public function it_can_specify_tokens_when_creating_a_user()
|
||||||
{
|
{
|
||||||
@@ -288,6 +379,51 @@ class ApiTest extends TestCase
|
|||||||
$this->assertCount(0, $subdomains);
|
$this->assertCount(0, $subdomains);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @test */
|
||||||
|
public function it_can_delete_subdomains_by_name()
|
||||||
|
{
|
||||||
|
/** @var Response $response */
|
||||||
|
$response = $this->await($this->browser->post('http://127.0.0.1:8080/api/users', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
], json_encode([
|
||||||
|
'name' => 'Marcel',
|
||||||
|
'can_specify_subdomains' => 1,
|
||||||
|
])));
|
||||||
|
|
||||||
|
$user = json_decode($response->getBody()->getContents())->user;
|
||||||
|
|
||||||
|
$response = $this->await($this->browser->post('http://127.0.0.1:8080/api/subdomains', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
], json_encode([
|
||||||
|
'subdomain' => 'reserved',
|
||||||
|
'auth_token' => $user->auth_token,
|
||||||
|
])));
|
||||||
|
|
||||||
|
$this->await($this->browser->delete('http://127.0.0.1:8080/api/subdomains/reserved', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
], json_encode([
|
||||||
|
'auth_token' => $user->auth_token,
|
||||||
|
])));
|
||||||
|
|
||||||
|
/** @var Response $response */
|
||||||
|
$response = $this->await($this->browser->get('http://127.0.0.1:8080/api/users/1', [
|
||||||
|
'Host' => 'expose.localhost',
|
||||||
|
'Authorization' => base64_encode('username:secret'),
|
||||||
|
'Content-Type' => 'application/json',
|
||||||
|
]));
|
||||||
|
|
||||||
|
$body = json_decode($response->getBody()->getContents());
|
||||||
|
$subdomains = $body->subdomains;
|
||||||
|
|
||||||
|
$this->assertCount(0, $subdomains);
|
||||||
|
}
|
||||||
|
|
||||||
/** @test */
|
/** @test */
|
||||||
public function it_can_not_reserve_an_already_reserved_subdomain()
|
public function it_can_not_reserve_an_already_reserved_subdomain()
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -64,6 +64,17 @@ class TunnelTest extends TestCase
|
|||||||
]));
|
]));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @test */
|
||||||
|
public function it_returns_404_for_non_existing_clients_on_custom_hosts()
|
||||||
|
{
|
||||||
|
$this->expectException(ResponseException::class);
|
||||||
|
$this->expectExceptionMessage(404);
|
||||||
|
|
||||||
|
$this->await($this->browser->get('http://127.0.0.1:8080/', [
|
||||||
|
'Host' => 'tunnel.share.beyondco.de',
|
||||||
|
]));
|
||||||
|
}
|
||||||
|
|
||||||
/** @test */
|
/** @test */
|
||||||
public function it_sends_incoming_requests_to_the_connected_client()
|
public function it_sends_incoming_requests_to_the_connected_client()
|
||||||
{
|
{
|
||||||
@@ -91,6 +102,33 @@ class TunnelTest extends TestCase
|
|||||||
$this->assertSame('Hello World!', $response->getBody()->getContents());
|
$this->assertSame('Hello World!', $response->getBody()->getContents());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/** @test */
|
||||||
|
public function it_sends_incoming_requests_to_the_connected_client_on_custom_hosts()
|
||||||
|
{
|
||||||
|
$this->app['config']['expose.admin.validate_auth_tokens'] = false;
|
||||||
|
|
||||||
|
$this->createTestHttpServer();
|
||||||
|
|
||||||
|
$this->app['config']['expose.admin.validate_auth_tokens'] = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* We create an expose client that connects to our server and shares
|
||||||
|
* the created test HTTP server.
|
||||||
|
*/
|
||||||
|
$client = $this->createClient();
|
||||||
|
$this->await($client->connectToServer('127.0.0.1:8085', 'tunnel', 'share.beyondco.de'));
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Once the client is connected, we perform a GET request on the
|
||||||
|
* created tunnel.
|
||||||
|
*/
|
||||||
|
$response = $this->await($this->browser->get('http://127.0.0.1:8080/', [
|
||||||
|
'Host' => 'tunnel.share.beyondco.de',
|
||||||
|
]));
|
||||||
|
|
||||||
|
$this->assertSame('Hello World!', $response->getBody()->getContents());
|
||||||
|
}
|
||||||
|
|
||||||
/** @test */
|
/** @test */
|
||||||
public function it_sends_incoming_requests_to_the_connected_client_via_tcp()
|
public function it_sends_incoming_requests_to_the_connected_client_via_tcp()
|
||||||
{
|
{
|
||||||
|
|||||||
Reference in New Issue
Block a user