loop = $loop; $this->configuration = $configuration; $this->logger = $logger; } public function share(string $sharedUrl, array $subdomains = []) { $sharedUrl = $this->prepareSharedUrl($sharedUrl); foreach ($subdomains as $subdomain) { $this->connectToServer($sharedUrl, $subdomain, $this->configuration->auth()); } } public function sharePort(int $port) { $this->connectToServerAndShareTcp($port, $this->configuration->auth()); } protected function prepareSharedUrl(string $sharedUrl): string { if (! $parsedUrl = parse_url($sharedUrl)) { return $sharedUrl; } $url = Arr::get($parsedUrl, 'host', Arr::get($parsedUrl, 'path')); $scheme = Arr::get($parsedUrl, 'scheme'); $port = Arr::get($parsedUrl, 'port', $scheme === 'https' ? 443 : 80); return sprintf('%s:%s:%s', $scheme, $url, $port); } public function connectToServer(string $sharedUrl, $subdomain, $authToken = ''): PromiseInterface { $deferred = new Deferred(); $promise = $deferred->promise(); $wsProtocol = $this->configuration->port() === 443 ? 'wss' : 'ws'; connect($wsProtocol."://{$this->configuration->host()}:{$this->configuration->port()}/expose/control?authToken={$authToken}", [], [ 'X-Expose-Control' => 'enabled', ], $this->loop) ->then(function (WebSocket $clientConnection) use ($sharedUrl, $subdomain, $deferred, $authToken) { $this->connectionRetries = 0; $connection = ControlConnection::create($clientConnection); $connection->authenticate($sharedUrl, $subdomain); $clientConnection->on('close', function () use ($sharedUrl, $subdomain, $authToken) { $this->logger->error('Connection to server closed.'); $this->retryConnectionOrExit(function () use ($sharedUrl, $subdomain, $authToken) { $this->connectToServer($sharedUrl, $subdomain, $authToken); }); }); $this->attachCommonConnectionListeners($connection, $deferred); $connection->on('subdomainTaken', function ($data) use ($deferred) { $this->logger->error($data->message); $this->exit($deferred); }); $connection->on('authenticated', function ($data) use ($deferred, $sharedUrl) { $httpProtocol = $this->configuration->port() === 443 ? 'https' : 'http'; $host = $this->configuration->host(); if ($httpProtocol !== 'https') { $host .= ":{$this->configuration->port()}"; } $this->logger->info($data->message); $this->logger->info("Local-URL:\t\t{$sharedUrl}"); $this->logger->info("Dashboard-URL:\t\thttp://127.0.0.1:".config()->get('expose.dashboard_port')); $this->logger->info("Expose-URL:\t\t{$httpProtocol}://{$data->subdomain}.{$host}"); $this->logger->line(''); static::$subdomains[] = "{$httpProtocol}://{$data->subdomain}.{$host}"; $deferred->resolve($data); }); }, function (\Exception $e) use ($deferred, $sharedUrl, $subdomain, $authToken) { if ($this->connectionRetries > 0) { $this->retryConnectionOrExit(function () use ($sharedUrl, $subdomain, $authToken) { $this->connectToServer($sharedUrl, $subdomain, $authToken); }); return; } $this->logger->error('Could not connect to the server.'); $this->logger->error($e->getMessage()); $this->exit($deferred); }); return $promise; } public function connectToServerAndShareTcp(int $port, $authToken = ''): PromiseInterface { $deferred = new Deferred(); $promise = $deferred->promise(); $wsProtocol = $this->configuration->port() === 443 ? 'wss' : 'ws'; connect($wsProtocol."://{$this->configuration->host()}:{$this->configuration->port()}/expose/control?authToken={$authToken}", [], [ 'X-Expose-Control' => 'enabled', ], $this->loop) ->then(function (WebSocket $clientConnection) use ($port, $deferred, $authToken) { $this->connectionRetries = 0; $connection = ControlConnection::create($clientConnection); $connection->authenticateTcp($port); $this->attachCommonConnectionListeners($connection, $deferred); $clientConnection->on('close', function () use ($port, $authToken) { $this->logger->error('Connection to server closed.'); $this->retryConnectionOrExit(function () use ($port, $authToken) { $this->connectToServerAndShareTcp($port, $authToken); }); }); $connection->on('authenticated', function ($data) use ($deferred, $port) { $host = $this->configuration->host(); $this->logger->info($data->message); $this->logger->info("Local-Port:\t\t{$port}"); $this->logger->info("Shared-Port:\t\t{$data->shared_port}"); $this->logger->info("Expose-URL:\t\ttcp://{$host}:{$data->shared_port}."); $this->logger->line(''); $deferred->resolve($data); }); }, function (\Exception $e) use ($deferred, $port, $authToken) { if ($this->connectionRetries > 0) { $this->retryConnectionOrExit(function () use ($port, $authToken) { $this->connectToServerAndShareTcp($port, $authToken); }); return; } $this->logger->error('Could not connect to the server.'); $this->logger->error($e->getMessage()); $this->exit($deferred); }); return $promise; } protected function attachCommonConnectionListeners(ControlConnection $connection, Deferred $deferred) { $connection->on('info', function ($data) { $this->logger->info($data->message); }); $connection->on('error', function ($data) { $this->logger->error($data->message); }); $connection->on('authenticationFailed', function ($data) use ($deferred) { $this->logger->error($data->message); $this->exit($deferred); }); $connection->on('setMaximumConnectionLength', function ($data) { $timeoutSection = $this->logger->getOutput()->section(); $this->loop->addPeriodicTimer(1, function () use ($data, $timeoutSection) { $this->timeConnected++; $secondsRemaining = $data->length * 60 - $this->timeConnected; $remaining = Carbon::now()->diff(Carbon::now()->addSeconds($secondsRemaining)); $timeoutSection->clear(); $timeoutSection->writeln('Remaining time: '.$remaining->format('%H:%I:%S')); }); }); } protected function exit(Deferred $deferred) { $deferred->reject(); $this->loop->futureTick(function () { exit(1); }); } protected function retryConnectionOrExit(callable $retry) { $this->connectionRetries++; if ($this->connectionRetries <= static::MAX_CONNECTION_RETRIES) { $this->loop->addTimer($this->connectionRetries, function () use ($retry) { $this->logger->info("Retrying connection ({$this->connectionRetries}/".static::MAX_CONNECTION_RETRIES.')'); $retry(); }); } else { exit(1); } } }