1 | <?php declare(strict_types=1); |
||
2 | |||
3 | /** |
||
4 | * ownCloud - Music app |
||
5 | * |
||
6 | * This file is licensed under the Affero General Public License version 3 or |
||
7 | * later. See the COPYING file. |
||
8 | * |
||
9 | * @author Pauli Järvinen <[email protected]> |
||
10 | * @copyright Pauli Järvinen 2021 - 2025 |
||
11 | */ |
||
12 | |||
13 | namespace OCA\Music\Db; |
||
14 | |||
15 | use OCA\Music\Utility\Util; |
||
16 | use OCP\IURLGenerator; |
||
17 | |||
18 | /** |
||
19 | * @method int getChannelId() |
||
20 | * @method void setChannelId(int $id) |
||
21 | * @method ?string getStreamUrl() |
||
22 | * @method void setStreamUrl(?string $url) |
||
23 | * @method ?string getMimetype() |
||
24 | * @method void setMimetype(?string $mime) |
||
25 | * @method ?int getSize() |
||
26 | * @method void setSize(?int $size) |
||
27 | * @method ?int getDuration() |
||
28 | * @method void setDuration(?int $duration) |
||
29 | * @method string getGuid() |
||
30 | * @method void setGuid(string $guid) |
||
31 | * @method string getGuidHash() |
||
32 | * @method void setGuidHash(string $guidHash) |
||
33 | * @method ?string getTitle() |
||
34 | * @method void setTitle(?string $title) |
||
35 | * @method ?int getEpisode() |
||
36 | * @method void setEpisode(?int $episode) |
||
37 | * @method ?int getSeason() |
||
38 | * @method void setSeason(?int $season) |
||
39 | * @method ?string getLinkUrl() |
||
40 | * @method void setLinkUrl(?string $url) |
||
41 | * @method ?string getPublished() |
||
42 | * @method void setPublished(?string $timestamp) |
||
43 | * @method ?string getKeywords() |
||
44 | * @method void setKeywords(?string $keywords) |
||
45 | * @method ?string getCopyright() |
||
46 | * @method void setCopyright(?string $copyright) |
||
47 | * @method ?string getAuthor() |
||
48 | * @method void setAuthor(?string $author) |
||
49 | * @method ?string getDescription() |
||
50 | * @method void setDescription(?string $description) |
||
51 | * @method ?string getStarred() |
||
52 | * @method void setStarred(?string $timestamp) |
||
53 | * @method int getRating() |
||
54 | * @method void setRating(int $rating) |
||
55 | */ |
||
56 | class PodcastEpisode extends Entity { |
||
57 | public int $channelId = 0; |
||
58 | public ?string $streamUrl = null; |
||
59 | public ?string $mimetype = null; |
||
60 | public ?int $size = null; |
||
61 | public ?int $duration = null; |
||
62 | public string $guid = ''; |
||
63 | public string $guidHash = ''; |
||
64 | public ?string $title = null; |
||
65 | public ?int $episode = null; |
||
66 | public ?int $season = null; |
||
67 | public ?string $linkUrl = null; |
||
68 | public ?string $published = null; |
||
69 | public ?string $keywords = null; |
||
70 | public ?string $copyright = null; |
||
71 | public ?string $author = null; |
||
72 | public ?string $description = null; |
||
73 | public ?string $starred = null; |
||
74 | public int $rating = 0; |
||
75 | |||
76 | public function __construct() { |
||
77 | $this->addType('channelId', 'int'); |
||
78 | $this->addType('size', 'int'); |
||
79 | $this->addType('duration', 'int'); |
||
80 | $this->addType('episode', 'int'); |
||
81 | $this->addType('season', 'int'); |
||
82 | $this->addType('rating', 'int'); |
||
83 | } |
||
84 | |||
85 | public function toApi(IURLGenerator $urlGenerator) : array { |
||
86 | return [ |
||
87 | 'id' => $this->getId(), |
||
88 | 'title' => $this->getTitle(), |
||
89 | 'ordinal' => $this->getEpisodeWithSeason(), |
||
90 | 'stream_url' => $urlGenerator->linkToRoute('music.podcastApi.episodeStream', ['id' => $this->id]), |
||
91 | 'mimetype' => $this->getMimetype() |
||
92 | ]; |
||
93 | } |
||
94 | |||
95 | public function detailsToApi() : array { |
||
96 | return [ |
||
97 | 'id' => $this->getId(), |
||
98 | 'title' => $this->getTitle(), |
||
99 | 'episode' => $this->getEpisode(), |
||
100 | 'season' => $this->getSeason(), |
||
101 | 'description' => $this->getDescription(), |
||
102 | 'channel_id' => $this->getChannelId(), |
||
103 | 'link_url' => $this->getLinkUrl(), |
||
104 | 'stream_url' => $this->getStreamUrl(), |
||
105 | 'mimetype' => $this->getMimetype(), |
||
106 | 'author' => $this->getAuthor(), |
||
107 | 'copyright' => $this->getCopyright(), |
||
108 | 'duration' => $this->getDuration(), |
||
109 | 'size' => $this->getSize(), |
||
110 | 'bit_rate' => $this->getBitrate(), |
||
111 | 'guid' => $this->getGuid(), |
||
112 | 'keywords' => $this->getKeywords(), |
||
113 | 'published' => $this->getPublished(), |
||
114 | ]; |
||
115 | } |
||
116 | |||
117 | public function toAmpacheApi(callable $createImageUrl, ?callable $createStreamUrl) : array { |
||
118 | $imageUrl = $createImageUrl($this); |
||
119 | return [ |
||
120 | 'id' => (string)$this->getId(), |
||
121 | 'name' => $this->getTitle(), |
||
122 | 'title' => $this->getTitle(), |
||
123 | 'description' => $this->getDescription(), |
||
124 | 'author' => $this->getAuthor(), |
||
125 | 'author_full' => $this->getAuthor(), |
||
126 | 'website' => $this->getLinkUrl(), |
||
127 | 'pubdate' => Util::formatDateTimeUtcOffset($this->getPublished()), |
||
128 | 'state' => 'Completed', |
||
129 | 'filelength' => Util::formatTime($this->getDuration()), |
||
130 | 'filesize' => Util::formatFileSize($this->getSize(), 2) . 'B', |
||
131 | 'bitrate' => $this->getBitrate(), |
||
132 | 'stream_bitrate' => $this->getBitrate(), |
||
133 | 'time' => $this->getDuration(), |
||
134 | 'size' => $this->getSize(), |
||
135 | 'mime' => $this->getMimetype(), |
||
136 | 'url' => $createStreamUrl ? $createStreamUrl($this) : $this->getStreamUrl(), |
||
137 | 'art' => $imageUrl, |
||
138 | 'has_art' => !empty($imageUrl), |
||
139 | 'flag' => !empty($this->getStarred()), |
||
140 | 'rating' => $this->getRating(), |
||
141 | 'preciserating' => $this->getRating(), |
||
142 | ]; |
||
143 | } |
||
144 | |||
145 | public function toSubsonicApi() : array { |
||
146 | return [ |
||
147 | 'id' => 'podcast_episode-' . $this->getId(), |
||
148 | 'streamId' => 'podcast_episode-' . $this->getId(), |
||
149 | 'channelId' => 'podcast_channel-' . $this->getChannelId(), |
||
150 | 'title' => $this->getTitle(), |
||
151 | 'artist' => $this->getAuthor(), |
||
152 | 'track' => $this->getEpisode(), |
||
153 | 'description' => $this->getDescription(), |
||
154 | 'publishDate' => Util::formatZuluDateTime($this->getPublished()), |
||
155 | 'status' => 'completed', |
||
156 | 'parent' => 'podcast_channel-' . $this->getChannelId(), |
||
157 | 'isDir' => false, |
||
158 | 'year' => $this->getYear(), |
||
159 | 'genre' => 'Podcast', |
||
160 | 'coverArt' => 'podcast_channel-' . $this->getChannelId(), |
||
161 | 'size' => $this->getSize(), |
||
162 | 'contentType' => $this->getMimetype(), |
||
163 | 'suffix' => $this->getSuffix(), |
||
164 | 'duration' => $this->getDuration(), |
||
165 | 'bitRate' => empty($this->getBitrate()) ? 0 : (int)\round($this->getBitrate()/1000), // convert bps to kbps |
||
166 | 'type' => 'podcast', |
||
167 | 'created' => Util::formatZuluDateTime($this->getCreated()), |
||
168 | 'starred' => Util::formatZuluDateTime($this->getStarred()), |
||
169 | 'userRating' => $this->getRating() ?: null, |
||
170 | 'averageRating' => $this->getRating() ?: null, |
||
171 | ]; |
||
172 | } |
||
173 | |||
174 | public function getEpisodeWithSeason() : ?string { |
||
175 | $result = (string)$this->getEpisode(); |
||
176 | // the season is considered only if there actually is an episode |
||
177 | $season = $this->getSeason(); |
||
178 | if ($season !== null) { |
||
179 | $result = "$season-$result"; |
||
180 | } |
||
181 | return $result; |
||
182 | } |
||
183 | |||
184 | public function getYear() : ?int { |
||
185 | $matches = null; |
||
186 | if (\is_string($this->published) && \preg_match('/^(\d\d\d\d)-\d\d-\d\d.*/', $this->published, $matches) === 1) { |
||
187 | return (int)$matches[1]; |
||
188 | } else { |
||
189 | return null; |
||
190 | } |
||
191 | } |
||
192 | |||
193 | /** @return ?float bits per second (bps) */ |
||
194 | public function getBitrate() : ?float { |
||
195 | if (empty($this->size) || empty($this->duration)) { |
||
196 | return null; |
||
197 | } else { |
||
198 | return $this->size / $this->duration * 8; |
||
199 | } |
||
200 | } |
||
201 | |||
202 | public function getSuffix() : ?string { |
||
203 | return self::mimeToSuffix($this->mimetype) ?? self::extractSuffixFromUrl($this->streamUrl); |
||
204 | } |
||
205 | |||
206 | private static function mimeToSuffix(?string $mime) : ?string { |
||
207 | // a relevant subset from https://stackoverflow.com/a/53662733/4348850 wit a few additions |
||
208 | $mime_map = [ |
||
209 | 'audio/x-acc' => 'aac', |
||
210 | 'audio/ac3' => 'ac3', |
||
211 | 'audio/x-aiff' => 'aif', |
||
212 | 'audio/aiff' => 'aif', |
||
213 | 'audio/x-au' => 'au', |
||
214 | 'audio/x-flac' => 'flac', |
||
215 | 'audio/x-m4a' => 'm4a', |
||
216 | 'audio/mp4' => 'm4a', |
||
217 | 'audio/midi' => 'mid', |
||
218 | 'audio/mpeg' => 'mp3', |
||
219 | 'audio/mpg' => 'mp3', |
||
220 | 'audio/mpeg3' => 'mp3', |
||
221 | 'audio/mp3' => 'mp3', |
||
222 | 'audio/ogg' => 'ogg', |
||
223 | 'application/ogg' => 'ogg', |
||
224 | 'audio/x-realaudio' => 'ra', |
||
225 | 'audio/x-pn-realaudio' => 'ram', |
||
226 | 'audio/x-wav' => 'wav', |
||
227 | 'audio/wave' => 'wav', |
||
228 | 'audio/wav' => 'wav', |
||
229 | 'audio/x-ms-wma' => 'wma', |
||
230 | 'audio/m4b' => 'm4b', |
||
231 | 'application/vnd.apple.mpegurl' => 'm3u', |
||
232 | 'audio/mpegurl' => 'm3u', |
||
233 | ]; |
||
234 | |||
235 | return $mime_map[$mime] ?? null; |
||
236 | } |
||
237 | |||
238 | private static function extractSuffixFromUrl(?string $url) : ?string { |
||
239 | if ($url === null) { |
||
240 | return null; |
||
241 | } else { |
||
242 | $path = \parse_url($url, PHP_URL_PATH); |
||
243 | if (\is_string($path)) { |
||
0 ignored issues
–
show
introduced
by
![]() |
|||
244 | $ext = (string)\pathinfo($path, PATHINFO_EXTENSION); |
||
245 | return !empty($ext) ? $ext : null; |
||
246 | } else { |
||
247 | return null; |
||
248 | } |
||
249 | } |
||
250 | } |
||
251 | } |
||
252 |