Я использую API YouTube для загрузки фрагментов видео (см. код ниже). Тем не менее, загрузка иногда прерывается с большими файлами (+ 1 ГБ), но не всегда. Загрузка будет завершена, но можно сыграть всего пару минут, а остальная часть будет усечена. Я сделал некоторые исследования, но без видимых успехов. Мой вопрос сейчас:
- Есть ли возможность напрямую связаться с YouTube (просмотрев журналы о том, что происходит)?
- Это проблема с кодировкой?
- Может ли ошибка быть обнаружена/обнаружена через API (на данный момент не генерируется исключение)
- Возможно ли это, если вы загружаете разные видеоролики сразу (параллельно, то есть)?
- Кто-нибудь еще столкнулся с этой проблемой?
Любая помощь/руководство в правильном направлении очень ценится. Я даже назвал бы щедрость в 500 очков, так как это сводит меня с ума (просто сделал это...)
Приложение. script запускается в командной строке через сервер Gearman с параметром set_time_limit(0);
. Код/функция - это просто экстракт (и отлично работает с меньшими файлами, иногда даже до 10 ГБ).
РЕДАКТИРОВАТЬ. В соответствии с вышеприведенными комментариями от aergistal и GeorgeQ я изменил цикл while, чтобы читать фрагменты напрямую (нет feof()
) и сохранить статус в базе данных.
/*
Uploads one file to youtube chunk by chunk
*/
function uploadFile($dbfile) {
$client = $this->client;
$youtube = new Google_Service_YouTube($client);
$htmlBody = "";
try {
// Create a snippet with title, description, tags and category ID
// Create an asset resource and set its snippet metadata and type.
// This example sets the video title, description, keyword tags, and
// video category.
$snippet = new Google_Service_YouTube_VideoSnippet();
$snippet->setTitle("SO Example");
// Numeric video category. See
// https://developers.google.com/youtube/v3/docs/videoCategories/list
$snippet->setCategoryId("22");
// Set the video status to "public". Valid statuses are "public",
// "private" and "unlisted".
$status = new Google_Service_YouTube_VideoStatus();
$status->privacyStatus = "private";
// Associate the snippet and status objects with a new video resource.
$video = new Google_Service_YouTube_Video();
$video->setSnippet($snippet);
$video->setStatus($status);
// Specify the size of each chunk of data, in bytes. Set a higher value for
// reliable connection as fewer chunks lead to faster uploads. Set a lower
// value for better recovery on less reliable connections.
$chunkSizeBytes = 1 * 1024 * 1024;
// Setting the defer flag to true tells the client to return a request which can be called
// with ->execute(); instead of making the API call immediately.
$client->setDefer(true);
// Create a request for the API videos.insert method to create and upload the video.
$insertRequest = $youtube->videos->insert("status,snippet", $video);
// Create a MediaFileUpload object for resumable uploads.
$media = new Google_Http_MediaFileUpload(
$client,
$insertRequest,
'video/*',
null,
true,
$chunkSizeBytes);
$media->setFileSize(filesize($dbfile->localfile));
// Read the media file and upload it chunk by chunk.
$status = false;
$handle = fopen($dbfile->localfile, "rb");
while (!$status && ($chunk = (fread($handle, $chunkSizeBytes))) !== FALSE) {
$status = $media->nextChunk($chunk);
$data = array("filename" => $dbfile->localfile, "status" => print_r($status, true));
$db->saveLog($data);
}
/* the old code
while (!$status && !feof($handle)) {
$chunk = fread($handle, $chunkSizeBytes);
$status = $media->nextChunk($chunk);
}
*/
fclose($handle);
// If you want to make other calls after the file upload, set setDefer back to false
$client->setDefer(false);
$log = array("success" => true, "snippet_id" => $status["id"]);
} catch (Google_ServiceException $e) {
$log = array("success" => false, "errormsg" => $e->getMessage());
} catch (Google_Exception $e) {
$log = array("success" => false, "errormsg" => $e->getMessage());
}
return $log;
}