talked-client/lib/Command/Record.php

270 lines
7.6 KiB
PHP

<?php
declare(strict_types=1);
/**
* Talked - Call recording for Nextcloud Talk
*
* @copyright Copyright (C) 2021 Magnus Walbeck <mw@mwalbeck.org>
*
* @author Magnus Walbeck <mw@mwalbeck.org>
*
* @license GNU AGPL version 3 or any later version
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU Affero General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*
*/
namespace OCA\Talked\Command;
use OCP\IConfig;
use OCP\Util;
use Symfony\Component\Console\Command\Command;
use Symfony\Component\Console\Input\InputArgument;
use Symfony\Component\Console\Input\InputInterface;
use Symfony\Component\Console\Output\OutputInterface;
use Psr\Log\LoggerInterface;
class Record extends Command {
/** @var IConfig */
private $config;
/** @var LoggerInterface */
private $logger;
public function __construct(IConfig $config, LoggerInterface $logger) {
parent::__construct();
$this->config = $config;
$this->logger = $logger;
}
protected function configure(): void {
$this
->setName('talked:record')
->setDescription('Call recording for Nextcloud Talk')
->addArgument(
'token',
InputArgument::REQUIRED,
'A Talk room token.'
)
->addArgument(
'cmd',
InputArgument::OPTIONAL,
'The command to run, the following are valid commands: info, status, start, stop and help.',
'help'
)
;
}
protected function execute(InputInterface $input, OutputInterface $output): int {
$token = $input->getArgument('token');
$arguments = explode(" ", $input->getArgument('cmd'));
$serverUrl = $this->config->getAppValue('talked', 'server_url', '');
if ($serverUrl === '') {
$output->writeln('A recording server hasn\'t been configured yet.');
return 0;
}
if ($arguments[0] === 'help' or $arguments[0] === '') {
$message = 'Talked - Call recording for Nextcloud Talk
You have the following options available:
/recording start - Starts a call recording
/recording stop - Stops the call recording
/recording status - Checks if there is an active call recording
/recording help - Shows this help message
';
$output->writeln($message);
return 0;
}
if ($arguments[0] === 'info') {
$result = $this->sendGetRequest($serverUrl, '');
$output->writeln($result);
return 0;
}
if ($arguments[0] === 'status') {
$payload = [
'token' => $token
];
$result = $this->sendPostRequest($serverUrl, 'status', $payload);
$output->writeln($result);
return 0;
}
if ($arguments[0] === 'start') {
$payload = [
'token' => $token,
'nextcloud_version' => Util::getVersion()[0]
];
if (count($arguments) > 1) {
$parsedArguments = $this->parseArguments($arguments);
$payload = array_merge($parsedArguments, $payload);
}
$result = $this->sendPostRequest($serverUrl, 'start', $payload);
$output->writeln($result);
return 0;
}
if ($arguments[0] === 'stop') {
$payload = [
'token' => $token
];
$result = $this->sendPostRequest($serverUrl, 'stop', $payload);
$output->writeln($result);
return 0;
}
$output->writeln('The specified command doesn\'t exist.');
return 0;
}
protected function sendGetRequest(string $serverUrl, string $endpoint, array $headers = []): string {
if ($this->config->getAppValue('talked', 'server_url', '0')) {
$headers = $this->addBasicAuthHeaders($headers);
}
$curlHandle = curl_init();
$curlHandle = $this->configureServerUri($curlHandle, $serverUrl, $endpoint);
curl_setopt($curlHandle, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curlHandle, CURLOPT_HTTPHEADER, $headers);
curl_setopt($curlHandle, CURLOPT_TIMEOUT, 60);
curl_setopt($curlHandle, CURLOPT_CONNECTTIMEOUT, 10);
$result = curl_exec($curlHandle);
$curlErrorCode = curl_errno($curlHandle);
$curlError = curl_error($curlHandle);
curl_close($curlHandle);
if ($curlErrorCode > 0) {
$this->logger->error('cURL Error (' . $curlErrorCode . '): ' . $curlError);
$message = 'An error occured while running the command. Please try again or contact an administrator.';
} else {
$message = json_decode($result)->message;
}
return $message;
}
protected function sendPostRequest(string $serverUrl, string $endpoint, array $payload, array $headers = []): string {
if ($this->config->getAppValue('talked', 'server_url', '0')) {
$headers = $this->addBasicAuthHeaders($headers);
}
$headers[] = 'Content-Type: application/json';
$curlHandle = curl_init();
$curlHandle = $this->configureServerUri($curlHandle, $serverUrl, $endpoint);
curl_setopt($curlHandle, CURLOPT_RETURNTRANSFER, true);
curl_setopt($curlHandle, CURLOPT_POST, true);
curl_setopt($curlHandle, CURLOPT_POSTFIELDS, json_encode($payload));
curl_setopt($curlHandle, CURLOPT_HTTPHEADER, $headers);
curl_setopt($curlHandle, CURLOPT_TIMEOUT, 60);
curl_setopt($curlHandle, CURLOPT_CONNECTTIMEOUT, 10);
$result = curl_exec($curlHandle);
$curlErrorCode = curl_errno($curlHandle);
$curlError = curl_error($curlHandle);
curl_close($curlHandle);
if ($curlErrorCode > 0) {
$this->logger->error('cURL Error (' . $curlErrorCode . '): ' . $curlError);
$message = 'An error occured while running the command. Please try again or contact an administrator.';
} else {
$message = json_decode($result)->message;
}
return $message;
}
protected function addBasicAuthHeaders(): array {
$username = $this->config->getAppValue('talked', 'http_basic_auth_username');
$password = $this->config->getAppValue('talked', 'http_basic_auth_password');
$base64EncodedAuth = base64_encode($username . ':' . $password);
$headers[] = 'Authorization: Basic ' . $base64EncodedAuth;
return $headers;
}
protected function configureServerUri($curlHandle, string $serverUrl, string $endpoint) {
# Check if the URI is pointing to a unix socket
if (substr($serverUrl, 0, 5) === 'unix:') {
curl_setopt($curlHandle, CURLOPT_UNIX_SOCKET_PATH, substr($serverUrl, 5));
# Configure the URL the requests should go to. In later versions
# curl requires a dummy hostname, so here we just specify localhost.
curl_setopt($curlHandle, CURLOPT_URL, 'http://localhost' . '/' . $endpoint);
} else {
# If the URI isn't pointing to a unix socket, assume we are connecting over TCP
curl_setopt($curlHandle, CURLOPT_URL, $serverUrl . '/' . $endpoint);
}
return $curlHandle;
}
protected function parseArguments($arguments) {
// Remove the initial command
array_shift($arguments);
$parsedArguments = [];
foreach($arguments as $argument) {
if (!str_contains($argument, '=')) {
continue;
}
$parts = explode('=', $argument);
switch ($parts[0]) {
case 'audio_only':
if (strtolower($parts[1]) === 'true') {
$parsedArguments['audio_only'] = true;
} elseif (strtolower($parts[1]) === 'false') {
$parsedArguments['audio_only'] = false;
}
continue 2;
case 'grid_view':
if (strtolower($parts[1]) === 'true') {
$parsedArguments['grid_view'] = true;
} elseif (strtolower($parts[1]) === 'false') {
$parsedArguments['grid_view'] = false;
}
continue 2;
default:
continue 2;
}
}
return $parsedArguments;
}
}