Где-то в октябре Google решил внести недокументированные изменения в свой сервис прямой трансляции, который оставил меня озадаченным на месяцы.После того, как я наконец нашел проблему, я начал искать API YouTube V3 для решения, однако начинаю думать, что это невозможно, поэтому я был бы очень признателен за отзывы о том, чего я пытаюсь достичь.
В настоящее времяпри прямой трансляции на YouTube вам предоставляется 3 варианта конфиденциальности:
- Публичный - вы можете встраивать с помощью идентификатора канала, что означает его постоянство.
- Не указано - Вы можете встраивать только через идентификатор видео, который изменяется всякий раз, когда вы начинаете новый поток.
- Приватный
При поиске способа автоматизацииПроцесс получения переменной (идентификатора видео) из не перечисленного видео Я наткнулся на приведенный ниже код, предоставленный YouTube (https://github.com/youtube/api-samples);
<?php
/**
* Library Requirements
*
* 1. Install composer (https://getcomposer.org)
* 2. On the command line, change to this directory (api-samples/php)
* 3. Require the google/apiclient library
* $ composer require google/apiclient:~2.0
*/
if (!file_exists(__DIR__ . '/vendor/autoload.php')) {
throw new \Exception('please run "composer require google/apiclient:~2.0" in "' . __DIR__ .'"');
}
require_once __DIR__ . '/vendor/autoload.php';
session_start();
/*
* You can acquire an OAuth 2.0 client ID and client secret from the
* {{ Google Cloud Console }} <{{ https://cloud.google.com/console }}>
* For more information about using OAuth 2.0 to access Google APIs, please see:
* <https://developers.google.com/youtube/v3/guides/authentication>
* Please ensure that you have enabled the YouTube Data API for your project.
*/
$OAUTH2_CLIENT_ID = 'REPLACE_ME';
$OAUTH2_CLIENT_SECRET = 'REPLACE_ME';
$client = new Google_Client();
$client->setClientId($OAUTH2_CLIENT_ID);
$client->setClientSecret($OAUTH2_CLIENT_SECRET);
$client->setScopes('https://www.googleapis.com/auth/youtube');
$redirect = filter_var('http://' . $_SERVER['HTTP_HOST'] . $_SERVER['PHP_SELF'],
FILTER_SANITIZE_URL);
$client->setRedirectUri($redirect);
// Define an object that will be used to make all API requests.
$youtube = new Google_Service_YouTube($client);
// Check if an auth token exists for the required scopes
$tokenSessionKey = 'token-' . $client->prepareScopes();
if (isset($_GET['code'])) {
if (strval($_SESSION['state']) !== strval($_GET['state'])) {
die('The session state did not match.');
}
$client->authenticate($_GET['code']);
$_SESSION[$tokenSessionKey] = $client->getAccessToken();
header('Location: ' . $redirect);
}
if (isset($_SESSION[$tokenSessionKey])) {
$client->setAccessToken($_SESSION[$tokenSessionKey]);
}
// Check to ensure that the access token was successfully acquired.
if ($client->getAccessToken()) {
try {
// Call the channels.list method to retrieve information about the
// currently authenticated user's channel.
$channelsResponse = $youtube->channels->listChannels('contentDetails', array(
'mine' => 'true',
));
$htmlBody = '';
foreach ($channelsResponse['items'] as $channel) {
// Extract the unique playlist ID that identifies the list of videos
// uploaded to the channel, and then call the playlistItems.list method
// to retrieve that list.
$uploadsListId = $channel['contentDetails']['relatedPlaylists']['uploads'];
$playlistItemsResponse = $youtube->playlistItems->listPlaylistItems('snippet', array(
'playlistId' => $uploadsListId,
'maxResults' => 50
));
$htmlBody .= "<h3>Videos in list $uploadsListId</h3><ul>";
foreach ($playlistItemsResponse['items'] as $playlistItem) {
$htmlBody .= sprintf('<li>%s (%s)</li>', $playlistItem['snippet']['title'],
$playlistItem['snippet']['resourceId']['videoId']);
}
$htmlBody .= '</ul>';
}
} catch (Google_Service_Exception $e) {
$htmlBody = sprintf('<p>A service error occurred: <code>%s</code></p>',
htmlspecialchars($e->getMessage()));
} catch (Google_Exception $e) {
$htmlBody = sprintf('<p>An client error occurred: <code>%s</code></p>',
htmlspecialchars($e->getMessage()));
}
$_SESSION[$tokenSessionKey] = $client->getAccessToken();
} elseif ($OAUTH2_CLIENT_ID == 'REPLACE_ME') {
$htmlBody = <<<END
<h3>Client Credentials Required</h3>
<p>
You need to set <code>\$OAUTH2_CLIENT_ID</code> and
<code>\$OAUTH2_CLIENT_ID</code> before proceeding.
<p>
END;
} else {
$state = mt_rand();
$client->setState($state);
$_SESSION['state'] = $state;
$authUrl = $client->createAuthUrl();
$htmlBody = <<<END
<h3>Authorization Required</h3>
<p>You need to <a href="$authUrl">authorize access</a> before proceeding.<p>
END;
}
?>
<!doctype html>
<html>
<head>
<title>My Uploads</title>
</head>
<body>
<?=$htmlBody?>
</body>
</html>
. Приведенный выше пример фантастический, он работает так, как мне нужно.Когда я перехожу к /my_upload.php (источник выше) через веб-браузер, я могу войти через OAuth в свою учетную запись YouTube и распечатать список идентификаторов видео, которые совпадают с видеоЯ загрузил.Как ни странно, когда я изменяю опцию 'maxResults' на 1, в ней будет отображаться самый последний идентификатор видео, и если я в данный момент транслирую в режиме реального времени, этот идентификатор видео будет самым последним.Насколько я знаю, после того как вы прошли аутентификацию, когда вам не нужно повторно проходить аутентификацию, у меня возникают проблемы с аутентификацией на моем бэкэнд-сервере.
Система, которую я хотел бы использовать, включает использование cron.например, bash-скрипт выполняет следующие действия:
#!/bin/bash
ffmpeg -whatever -launch -parameters -I -need ### Launch the livestream
timeout 300 seconds ### Pause for 5 Minutes to allow YouTube to go live
curl/php path/to/my_uploads.php ### Run the php script to obtain video ID
Когда бы ни планировался cron, он запускал прямую трансляцию, а затем запускал скрипт my_uploads.php, чтобы получить идентификатор видео, который будет сохранен в документе .txt.готов к выводу в веб-плеер.
Так в чем же проблема?
Мне успешно удалось пройти проверку подлинности, перейдя в файл /my_uploads.php в браузереоднако это не аутентифицирует внутренний сервер, я использую VNC для подключения к внутреннему серверу для аутентификации.Я подумал, что это будет так же просто, как сделать что-то вроде:
curl http://backendserverip/my_uploads.php -> Follow auth steps
Но, конечно, нет, Google не позволяет использовать IP-адреса, поэтому в целях тестирования я отредактировал файл / etc / hosts, чтобы отразитьследующий;
myserverip localhost.mydomain.com
, который работает нормально в соответствии с тем, что принимает Google.Отлично, теперь это должно сработать ... Неверно.Когда я делаю следующее:
curl http://localhost.mydomain.com/my_uploads.php (ignore index.php)
Посетите ссылку, напечатанную в терминале;
Я получаюошибка «Состояние сеанса не соответствует», которую я предполагаю, заключается в том, что URL-адрес аутентификации был сгенерирован в терминале, а когда я захожу по ссылке в браузере, заголовки и все остальное отличается, так что в целях безопасности это не позволяет мне аутентифицироваться.
Конечно, должен быть способ, которым мой сервер может проходить аутентификацию, чтобы получить доступ к API, конечно, я делаю что-то не так, так как нет абсолютно никакого смысла в том, почему это вообще существует.Если бы это было только для использования в Интернете, который в этом мире получал бы доступ к стороннему сайту, войдите через OAuth, чтобы их видео данные были отражены.Почему бы им просто не использовать YouTube напрямую?
Меня смущает то, что я использовал VNC для подключения к своему серверу.Открыл firefox, перешел к файлу my_uploads.php, прошел проверку подлинности и он работает нормально;
Однако при выполнении команды через терминал / замазку;
curl http://localhost.mydomain.com/my_uploads.php (ignore index.php)
Я вынужден пройти повторную аутентификацию, однако я не могу сделать это через терминал, насколько я знаю, это невозможно.
Я идиот?Это даже невозможно, или я делаю совершенно неправильную вещь, я потратил последние 2 дня, пытаясь понять это.Я многому научился на этом пути, но я абсолютно потерян, и мне действительно нужна некоторая помощь здесь, спасибо за ваше время и извините за длинный пост.