2020-06-07 19:04:43 +03:00
|
|
|
|
<?php declare(strict_types=1);
|
|
|
|
|
namespace openvk\Web\Models\Entities;
|
|
|
|
|
use openvk\Web\Util\Shell\Shell;
|
2021-10-15 12:34:59 +03:00
|
|
|
|
use openvk\Web\Util\Shell\Shell\Exceptions\{ShellUnavailableException, UnknownCommandException};
|
2020-06-07 19:04:43 +03:00
|
|
|
|
use openvk\Web\Models\VideoDrivers\VideoDriver;
|
|
|
|
|
use Nette\InvalidStateException as ISE;
|
|
|
|
|
|
|
|
|
|
define("VIDEOS_FRIENDLY_ERROR", "Uploads are disabled on this instance :<", false);
|
|
|
|
|
|
|
|
|
|
class Video extends Media
|
|
|
|
|
{
|
|
|
|
|
const TYPE_DIRECT = 0;
|
|
|
|
|
const TYPE_EMBED = 1;
|
|
|
|
|
|
|
|
|
|
protected $tableName = "videos";
|
2023-01-28 23:25:00 +03:00
|
|
|
|
protected $fileExtension = "mp4";
|
2022-04-12 22:01:12 +03:00
|
|
|
|
|
|
|
|
|
protected $processingPlaceholder = "video/rendering";
|
2020-06-07 19:04:43 +03:00
|
|
|
|
|
|
|
|
|
protected function saveFile(string $filename, string $hash): bool
|
|
|
|
|
{
|
2021-10-15 12:34:59 +03:00
|
|
|
|
if(!Shell::commandAvailable("ffmpeg") || !Shell::commandAvailable("ffprobe"))
|
|
|
|
|
exit(VIDEOS_FRIENDLY_ERROR);
|
|
|
|
|
|
|
|
|
|
$error = NULL;
|
|
|
|
|
$streams = Shell::ffprobe("-i", $filename, "-show_streams", "-select_streams v", "-loglevel error")->execute($error);
|
|
|
|
|
if($error !== 0)
|
|
|
|
|
throw new \DomainException("$filename is not a valid video file");
|
|
|
|
|
else if(empty($streams) || ctype_space($streams))
|
|
|
|
|
throw new \DomainException("$filename does not contain any video streams");
|
|
|
|
|
|
|
|
|
|
$durations = [];
|
2023-01-28 01:27:16 +03:00
|
|
|
|
preg_match_all('%duration=([0-9\.]++)%', $streams, $durations);
|
2021-10-15 12:34:59 +03:00
|
|
|
|
if(sizeof($durations[1]) === 0)
|
|
|
|
|
throw new \DomainException("$filename does not contain any meaningful video streams");
|
|
|
|
|
|
|
|
|
|
foreach($durations[1] as $duration)
|
|
|
|
|
if(floatval($duration) < 1.0)
|
|
|
|
|
throw new \DomainException("$filename does not contain any meaningful video streams");
|
2020-06-07 19:04:43 +03:00
|
|
|
|
|
|
|
|
|
try {
|
2022-04-12 22:01:12 +03:00
|
|
|
|
if(!is_dir($dirId = dirname($this->pathFromHash($hash))))
|
2020-06-07 19:04:43 +03:00
|
|
|
|
mkdir($dirId);
|
|
|
|
|
|
2020-06-23 23:04:17 +03:00
|
|
|
|
$dir = $this->getBaseDir();
|
2022-03-20 19:01:48 +03:00
|
|
|
|
$ext = Shell::isPowershell() ? "ps1" : "sh";
|
|
|
|
|
$cmd = Shell::isPowershell() ? "powershell" : "bash";
|
|
|
|
|
Shell::$cmd(__DIR__ . "/../shell/processVideo.$ext", OPENVK_ROOT, $filename, $dir, $hash)->start(); #async :DDD
|
2020-06-07 19:04:43 +03:00
|
|
|
|
} catch(ShellUnavailableException $suex) {
|
|
|
|
|
exit(OPENVK_ROOT_CONF["openvk"]["debug"] ? "Shell is unavailable" : VIDEOS_FRIENDLY_ERROR);
|
|
|
|
|
} catch(UnknownCommandException $ucex) {
|
|
|
|
|
exit(OPENVK_ROOT_CONF["openvk"]["debug"] ? "bash is not installed" : VIDEOS_FRIENDLY_ERROR);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
usleep(200100);
|
|
|
|
|
return true;
|
|
|
|
|
}
|
2022-04-12 22:01:12 +03:00
|
|
|
|
|
|
|
|
|
protected function checkIfFileIsProcessed(): bool
|
|
|
|
|
{
|
|
|
|
|
if($this->getType() != Video::TYPE_DIRECT)
|
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
if(!file_exists($this->getFileName())) {
|
|
|
|
|
if((time() - $this->getRecord()->last_checked) > 3600) {
|
2022-05-08 13:06:26 +03:00
|
|
|
|
# TODO notify that video processor is probably dead
|
2022-04-12 22:01:12 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-07 19:04:43 +03:00
|
|
|
|
function getName(): string
|
|
|
|
|
{
|
|
|
|
|
return $this->getRecord()->name;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getType(): int
|
|
|
|
|
{
|
|
|
|
|
if(!is_null($this->getRecord()->hash))
|
|
|
|
|
return Video::TYPE_DIRECT;
|
|
|
|
|
else if(!is_null($this->getRecord()->link))
|
|
|
|
|
return Video::TYPE_EMBED;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getVideoDriver(): ?VideoDriver
|
|
|
|
|
{
|
|
|
|
|
if($this->getType() !== Video::TYPE_EMBED)
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
[$videoDriver, $pointer] = explode(":", $this->getRecord()->link);
|
|
|
|
|
$videoDriver = "openvk\\Web\\Models\\VideoDrivers\\$videoDriver" . "VideoDriver";
|
|
|
|
|
if(!class_exists($videoDriver))
|
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
|
|
return new $videoDriver($pointer);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function getThumbnailURL(): string
|
|
|
|
|
{
|
|
|
|
|
if($this->getType() === Video::TYPE_DIRECT) {
|
2022-04-12 22:01:12 +03:00
|
|
|
|
if(!$this->isProcessed())
|
|
|
|
|
return "/assets/packages/static/openvk/video/rendering.apng";
|
|
|
|
|
|
2023-01-29 00:09:05 +03:00
|
|
|
|
return preg_replace("%\.[A-z0-9]++$%", ".gif", $this->getURL());
|
2020-06-07 19:04:43 +03:00
|
|
|
|
} else {
|
|
|
|
|
return $this->getVideoDriver()->getThumbnailURL();
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-06-17 21:58:25 +03:00
|
|
|
|
|
|
|
|
|
function getOwnerVideo(): int
|
|
|
|
|
{
|
|
|
|
|
return $this->getRecord()->owner;
|
|
|
|
|
}
|
2023-02-21 00:35:10 +03:00
|
|
|
|
|
|
|
|
|
function getApiStructure(): object
|
|
|
|
|
{
|
2023-06-22 14:22:07 +03:00
|
|
|
|
$fromYoutube = $this->getType() == Video::TYPE_EMBED;
|
2023-02-21 00:35:10 +03:00
|
|
|
|
return (object)[
|
|
|
|
|
"type" => "video",
|
|
|
|
|
"video" => [
|
|
|
|
|
"can_comment" => 1,
|
|
|
|
|
"can_like" => 0, // we don't h-have wikes in videos
|
|
|
|
|
"can_repost" => 0,
|
|
|
|
|
"can_subscribe" => 1,
|
|
|
|
|
"can_add_to_faves" => 0,
|
|
|
|
|
"can_add" => 0,
|
|
|
|
|
"comments" => $this->getCommentsCount(),
|
|
|
|
|
"date" => $this->getPublicationTime()->timestamp(),
|
|
|
|
|
"description" => $this->getDescription(),
|
|
|
|
|
"duration" => 0, // я хуй знает как получить длину видео
|
|
|
|
|
"image" => [
|
|
|
|
|
[
|
|
|
|
|
"url" => $this->getThumbnailURL(),
|
|
|
|
|
"width" => 320,
|
|
|
|
|
"height" => 240,
|
|
|
|
|
"with_padding" => 1
|
|
|
|
|
]
|
|
|
|
|
],
|
|
|
|
|
"width" => 640,
|
|
|
|
|
"height" => 480,
|
|
|
|
|
"id" => $this->getVirtualId(),
|
|
|
|
|
"owner_id" => $this->getOwner()->getId(),
|
|
|
|
|
"user_id" => $this->getOwner()->getId(),
|
|
|
|
|
"title" => $this->getName(),
|
|
|
|
|
"is_favorite" => false,
|
2023-06-22 14:22:07 +03:00
|
|
|
|
"player" => !$fromYoutube ? $this->getURL() : $this->getVideoDriver()->getURL(),
|
|
|
|
|
"files" => !$fromYoutube ? [
|
2023-07-05 18:06:15 +03:00
|
|
|
|
"mp4_480" => $this->getURL()
|
2023-06-22 14:22:07 +03:00
|
|
|
|
] : NULL,
|
|
|
|
|
"platform" => $fromYoutube ? "youtube" : NULL,
|
2023-02-21 00:35:10 +03:00
|
|
|
|
"added" => 0,
|
|
|
|
|
"repeat" => 0,
|
|
|
|
|
"type" => "video",
|
|
|
|
|
"views" => 0,
|
|
|
|
|
"likes" => [
|
|
|
|
|
"count" => 0,
|
|
|
|
|
"user_likes" => 0
|
|
|
|
|
],
|
|
|
|
|
"reposts" => [
|
|
|
|
|
"count" => 0,
|
|
|
|
|
"user_reposted" => 0
|
|
|
|
|
]
|
|
|
|
|
]
|
|
|
|
|
];
|
|
|
|
|
}
|
2020-06-07 19:04:43 +03:00
|
|
|
|
|
VKAPI: методы для подарков, заметок, статусов, обсуждений и немного для фоток и групп (#876)
* API methods for gifts, notes, statuses
* Some fixes
Строки локализации у gifts.send теперь не костыльные и можно прикрепить до 10 аттачей к посту
* Small imp
Пофиксил пагинацию у заметков и подарок
Перенёс структуру заметок
Добавил аттачи к комментариям
Добавил проверку на удалённость аттача
Ну и пофиксил сортировку заметок
* VKAPI: Some methods for topics and photos
Добавлены методы для обсуждений (addTopic, closeTopic(), createComment(), deleteComment(), deleteTopic(), editTopic(), fixTopic(), getComments(), getTopics(), openTopic(), unfixTopic())
и для фотографий (createAlbum(), editAlbum(), getAlbums(), getAlbumsCount(), getById(), get(), deleteAlbum(), edit(), delete(), deleteComment(), createComment(), getAll(), getComments())
* fixsex
2023-06-13 21:03:43 +03:00
|
|
|
|
function toVkApiStruct(): object
|
|
|
|
|
{
|
|
|
|
|
return $this->getApiStructure();
|
|
|
|
|
}
|
|
|
|
|
|
2020-06-07 19:04:43 +03:00
|
|
|
|
function setLink(string $link): string
|
|
|
|
|
{
|
|
|
|
|
if(preg_match(file_get_contents(__DIR__ . "/../VideoDrivers/regex/youtube.txt"), $link, $matches)) {
|
|
|
|
|
$pointer = "YouTube:$matches[1]";
|
|
|
|
|
} else if(preg_match(file_get_contents(__DIR__ . "/../VideoDrivers/regex/vimeo.txt"), $link, $matches)) {
|
|
|
|
|
$pointer = "Vimeo:$matches[1]";
|
|
|
|
|
} else {
|
|
|
|
|
throw new ISE("Invalid link");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$this->stateChanges("link", $pointer);
|
|
|
|
|
|
|
|
|
|
return $pointer;
|
|
|
|
|
}
|
2020-06-17 21:58:25 +03:00
|
|
|
|
|
|
|
|
|
function isDeleted(): bool
|
|
|
|
|
{
|
2021-10-13 22:51:28 +03:00
|
|
|
|
if ($this->getRecord()->deleted == 1)
|
|
|
|
|
return TRUE;
|
|
|
|
|
else
|
|
|
|
|
return FALSE;
|
2020-06-17 21:58:25 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function deleteVideo(): void
|
|
|
|
|
{
|
|
|
|
|
$this->setDeleted(1);
|
|
|
|
|
$this->unwire();
|
|
|
|
|
$this->save();
|
|
|
|
|
}
|
2021-10-13 22:51:28 +03:00
|
|
|
|
|
2023-05-23 15:28:21 +03:00
|
|
|
|
static function fastMake(int $owner, string $name = "Unnamed Video.ogv", string $description = "", array $file, bool $unlisted = true, bool $anon = false): Video
|
2021-10-13 22:51:28 +03:00
|
|
|
|
{
|
2023-05-27 09:31:20 +03:00
|
|
|
|
if(OPENVK_ROOT_CONF['openvk']['preferences']['videos']['disableUploading'])
|
|
|
|
|
exit(VIDEOS_FRIENDLY_ERROR);
|
|
|
|
|
|
2021-10-13 22:51:28 +03:00
|
|
|
|
$video = new Video;
|
|
|
|
|
$video->setOwner($owner);
|
2023-05-23 15:28:21 +03:00
|
|
|
|
$video->setName(ovk_proc_strtr($name, 61));
|
2021-10-13 22:51:28 +03:00
|
|
|
|
$video->setDescription(ovk_proc_strtr($description, 300));
|
2021-11-15 22:45:48 +03:00
|
|
|
|
$video->setAnonymous($anon);
|
2021-10-13 22:51:28 +03:00
|
|
|
|
$video->setCreated(time());
|
|
|
|
|
$video->setFile($file);
|
|
|
|
|
$video->setUnlisted($unlisted);
|
|
|
|
|
$video->save();
|
|
|
|
|
|
|
|
|
|
return $video;
|
|
|
|
|
}
|
2023-07-05 18:06:15 +03:00
|
|
|
|
|
|
|
|
|
function canBeViewedBy(?User $user = NULL): bool
|
|
|
|
|
{
|
|
|
|
|
if($this->isDeleted() || $this->getOwner()->isDeleted()) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if(get_class($this->getOwner()) == "openvk\\Web\\Models\\Entities\\User") {
|
|
|
|
|
return $this->getOwner()->canBeViewedBy($user) && $this->getOwner()->getPrivacyPermission('videos.read', $user);
|
|
|
|
|
} else {
|
|
|
|
|
# когда у видосов появятся группы
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-06-07 19:04:43 +03:00
|
|
|
|
}
|