'Sonarr', 'enabled' => strpos('personal', $this->config['license']) !== false, 'image' => 'plugins/images/tabs/sonarr.png', 'category' => 'PVR', 'settingsArray' => __FUNCTION__ ]; if ($infoOnly) { return $homepageInformation; } $homepageSettings = [ 'docs' => $this->docs('features/homepage/sonarr-homepage-item'), 'debug' => true, 'settings' => [ 'About' => [ $this->settingsOption('about', 'Sonarr', ['about' => 'This item allows access to Sonarr\'s calendar data and aggregates it to Organizr\'s calendar. Along with that you also have the Downloader function that allow access to Sonarr\'s queue. The last item that is included is the API SOCKS function which acts as a middleman between API\'s which is useful if you are not port forwarding or reverse proxying Sonarr.']), ], 'Enable' => [ $this->settingsOption('enable', 'homepageSonarrEnabled'), $this->settingsOption('auth', 'homepageSonarrAuth'), ], 'Connection' => [ $this->settingsOption('multiple-url', 'sonarrURL'), $this->settingsOption('multiple-token', 'sonarrToken'), $this->settingsOption('disable-cert-check', 'sonarrDisableCertCheck'), $this->settingsOption('use-custom-certificate', 'sonarrUseCustomCertificate'), ], 'API SOCKS' => [ $this->settingsOption('socks', 'sonarr'), $this->settingsOption('blank'), $this->settingsOption('enable', 'sonarrSocksEnabled'), $this->settingsOption('auth', 'sonarrSocksAuth'), ], 'Queue' => [ $this->settingsOption('enable', 'homepageSonarrQueueEnabled'), $this->settingsOption('auth', 'homepageSonarrQueueAuth'), $this->settingsOption('combine', 'homepageSonarrQueueCombine'), $this->settingsOption('refresh', 'homepageSonarrQueueRefresh'), ], 'Calendar' => [ $this->settingsOption('calendar-start', 'calendarStart'), $this->settingsOption('calendar-end', 'calendarEnd'), $this->settingsOption('calendar-starting-day', 'calendarFirstDay'), $this->settingsOption('calendar-default-view', 'calendarDefault'), $this->settingsOption('calendar-time-format', 'calendarTimeFormat'), $this->settingsOption('calendar-locale', 'calendarLocale'), $this->settingsOption('calendar-limit', 'calendarLimit'), $this->settingsOption('refresh', 'calendarRefresh'), $this->settingsOption('blank'), $this->settingsOption('switch', 'sonarrUnmonitored', ['label' => 'Show Unmonitored']), $this->settingsOption('blank', '', ['type' => 'html', 'html' => '
']), $this->settingsOption('blank', '', ['type' => 'html', 'html' => '
']), $this->settingsOption('enable', 'sonarrIcon', ['label' => 'Show Sonarr Icon']), $this->settingsOption('calendar-link-url', 'sonarrCalendarLink'), $this->settingsOption('blank'), $this->settingsOption('calendar-frame-target', 'sonarrFrameTarget') ], 'Test Connection' => [ $this->settingsOption('blank', null, ['label' => 'Please Save before Testing']), $this->settingsOption('test', 'sonarr'), ] ] ]; return array_merge($homepageInformation, $homepageSettings); } public function testConnectionSonarr() { if (empty($this->config['sonarrURL'])) { $this->setAPIResponse('error', 'Sonarr URL is not defined', 422); return false; } if (empty($this->config['sonarrToken'])) { $this->setAPIResponse('error', 'Sonarr Token is not defined', 422); return false; } $failed = false; $errors = ''; $list = $this->csvHomepageUrlToken($this->config['sonarrURL'], $this->config['sonarrToken']); foreach ($list as $key => $value) { try { $options = $this->requestOptions($value['url'], null, $this->config['sonarrDisableCertCheck'], $this->config['sonarrUseCustomCertificate']); $downloader = new Kryptonit3\Sonarr\Sonarr($value['url'], $value['token'], 'sonarr', null, null, $options); $results = $downloader->getRootFolder(); $downloadList = json_decode($results, true); if (is_array($downloadList) || is_object($downloadList)) { $queue = (array_key_exists('error', $downloadList)) ? $downloadList['error']['msg'] : $downloadList; if (!is_array($queue)) { $ip = $value['url']; $errors .= $ip . ': ' . $queue; $failed = true; } } else { $ip = $value['url']; $errors .= $ip . ': Response was not JSON'; $failed = true; } } catch (Exception $e) { $failed = true; $ip = $value['url']; $errors .= $ip . ': ' . $e->getMessage(); $this->setLoggerChannel('Sonarr')->error($e); } } if ($failed) { $this->setAPIResponse('error', $errors, 500); return false; } else { $this->setAPIResponse('success', null, 200); return true; } } public function sonarrHomepagePermissions($key = null) { $permissions = [ 'calendar' => [ 'enabled' => [ 'homepageSonarrEnabled' ], 'auth' => [ 'homepageSonarrAuth' ], 'not_empty' => [ 'sonarrURL', 'sonarrToken' ] ], 'queue' => [ 'enabled' => [ 'homepageSonarrEnabled', 'homepageSonarrQueueEnabled' ], 'auth' => [ 'homepageSonarrAuth', 'homepageSonarrQueueAuth' ], 'not_empty' => [ 'sonarrURL', 'sonarrToken' ] ] ]; return $this->homepageCheckKeyPermissions($key, $permissions); } public function homepageOrderSonarrQueue() { if ($this->homepageItemPermissions($this->sonarrHomepagePermissions('queue'))) { $loadingBox = ($this->config['homepageSonarrQueueCombine']) ? '' : '

Loading Download Queue...

'; $builder = ($this->config['homepageSonarrQueueCombine']) ? 'buildDownloaderCombined(\'sonarr\');' : '$("#' . __FUNCTION__ . '").html(buildDownloader("sonarr"));'; return '
' . $loadingBox . '
'; } } public function getSonarrQueue() { if (!$this->homepageItemPermissions($this->sonarrHomepagePermissions('queue'), true)) { return false; } $queueItems = array(); $list = $this->csvHomepageUrlToken($this->config['sonarrURL'], $this->config['sonarrToken']); foreach ($list as $key => $value) { try { $options = $this->requestOptions($value['url'], $this->config['homepageSonarrQueueRefresh'], $this->config['sonarrDisableCertCheck'], $this->config['sonarrUseCustomCertificate']); $downloader = new Kryptonit3\Sonarr\Sonarr($value['url'], $value['token'], 'sonarr', null, null, $options); $results = $downloader->getQueue(); $downloadList = json_decode($results, true); if (is_array($downloadList) || is_object($downloadList)) { $queue = (array_key_exists('error', $downloadList)) ? [] : $downloadList; $queue = $queue['records'] ?? $queue; } else { $queue = []; } if (!empty($queue)) { $queueItems = array_merge($queueItems, $queue); } } catch (Exception $e) { $this->setLoggerChannel('Sonarr')->error($e); } } $api['content']['queueItems'] = $queueItems; $api['content']['historyItems'] = false; $api['content'] = $api['content'] ?? false; $this->setAPIResponse('success', null, 200, $api); return $api; } public function getSonarrCalendar($startDate = null, $endDate = null) { $startDate = ($startDate) ?? $_GET['start'] ?? date('Y-m-d', strtotime('-' . $this->config['calendarStart'] . ' days')); $endDate = ($endDate) ?? $_GET['end'] ?? date('Y-m-d', strtotime('+' . $this->config['calendarEnd'] . ' days')); if (!$this->homepageItemPermissions($this->sonarrHomepagePermissions('calendar'), true)) { return false; } if ($this->demo) { return $this->demoData('sonarr/calendar.json'); } $calendarItems = array(); $list = $this->csvHomepageUrlToken($this->config['sonarrURL'], $this->config['sonarrToken']); foreach ($list as $key => $value) { try { $options = $this->requestOptions($value['url'], null, $this->config['sonarrDisableCertCheck'], $this->config['sonarrUseCustomCertificate']); $sonarr = new Kryptonit3\Sonarr\Sonarr($value['url'], $value['token'], 'sonarr', null, null, $options); $sonarr = $sonarr->getCalendar($startDate, $endDate, $this->config['sonarrUnmonitored']); $result = json_decode($sonarr, true); if (is_array($result) || is_object($result)) { $sonarrCalendar = (array_key_exists('error', $result)) ? '' : $this->formatSonarrCalendar($sonarr, $key); } else { $sonarrCalendar = ''; } } catch (Exception $e) { $this->setLoggerChannel('Sonarr')->error($e); } if (!empty($sonarrCalendar)) { $calendarItems = array_merge($calendarItems, $sonarrCalendar); } } $this->setAPIResponse('success', null, 200, $calendarItems); return $calendarItems; } public function formatSonarrCalendar($array, $number) { $array = json_decode($array, true); $gotCalendar = []; $i = 0; foreach ($array as $child) { $i++; $seriesName = $child['series']['title']; $seriesID = $child['series']['tvdbId']; $episodeID = $child['series']['tvdbId']; $monitored = $child['monitored']; if (!isset($episodeID)) { $episodeID = ""; } $episodeAirDate = $child['airDateUtc']; $episodeAirDate = strtotime($episodeAirDate); $episodeAirDate = date("Y-m-d H:i:s", $episodeAirDate); if (new DateTime() < new DateTime($episodeAirDate)) { $unAired = true; } if ($child['episodeNumber'] == "1") { $episodePremier = "true"; } else { $episodePremier = "false"; $date = new DateTime($episodeAirDate); $date->add(new DateInterval("PT1S")); $date->format(DateTime::ATOM); $child['airDateUtc'] = gmdate('Y-m-d\TH:i:s\Z', strtotime($date->format(DateTime::ATOM))); } $downloaded = $child['hasFile']; if ($downloaded == "0" && isset($unAired) && $episodePremier == "true") { $downloaded = "text-primary animated flash"; } elseif ($downloaded == "0" && isset($unAired) && $monitored == "0") { $downloaded = "text-dark"; } elseif ($downloaded == "0" && isset($unAired)) { $downloaded = "text-info"; } elseif ($downloaded == "1") { $downloaded = "text-success"; } else { $downloaded = "text-danger"; } $fanArt = "/plugins/images/homepage/no-np.png"; foreach ($child['series']['images'] as $image) { if ($image['coverType'] == "fanart" && (isset($image['url']) && $image['url'] !== '')) { $fanArt = $image['url']; } if ($image['coverType'] == 'fanart' && (isset($image['remoteUrl']) && $image['remoteUrl'] !== '')) { $fanArt = $image['remoteUrl']; } } if ($fanArt !== "/plugins/images/homepage/no-np.png" || (strpos($fanArt, '://') === false)) { $cacheDirectory = dirname(__DIR__, 2) . DIRECTORY_SEPARATOR . 'data' . DIRECTORY_SEPARATOR . 'cache' . DIRECTORY_SEPARATOR; $imageURL = $fanArt; $fanArt = 'data/cache/' . $seriesID . '.jpg'; if (!file_exists($cacheDirectory . $seriesID . '.jpg')) { $this->cacheImage($imageURL, $seriesID); unset($imageURL); } } $bottomTitle = 'S' . sprintf("%02d", $child['seasonNumber']) . 'E' . sprintf("%02d", $child['episodeNumber']) . ' - ' . $child['title']; $href = $this->config['sonarrCalendarLink'] ?? ''; if (empty($href) && !empty($this->config['sonarrURL'])) { $href_arr = explode(',', $this->config['sonarrURL']); $href = reset($href_arr); } if (!empty($href)) { $href = $href . '/series/' . preg_replace('/[^A-Za-z0-9 -]/', '', str_replace('&', 'and', preg_replace('/[[:space:]]+/', '-', $seriesName))); $href = str_replace("//series/", "/series/", $href); } $details = [ "seasonCount" => $child['series']['seasonCount'] ?? isset($child['series']['seasons']) ? count($child['series']['seasons']) : 0, "status" => $child['series']['status'], "topTitle" => $seriesName, "bottomTitle" => $bottomTitle, "overview" => $child['overview'] ?? '', "runtime" => $child['series']['runtime'], "image" => $fanArt, "ratings" => $child['series']['ratings']['value'], "videoQuality" => $child["hasFile"] && isset($child['episodeFile']['quality']['quality']['name']) ? $child['episodeFile']['quality']['quality']['name'] : "unknown", "audioChannels" => $child["hasFile"] && isset($child['episodeFile']['mediaInfo']) ? $child['episodeFile']['mediaInfo']['audioChannels'] : "unknown", "audioCodec" => $child["hasFile"] && isset($child['episodeFile']['mediaInfo']) ? $child['episodeFile']['mediaInfo']['audioCodec'] : "unknown", "videoCodec" => $child["hasFile"] && isset($child['episodeFile']['mediaInfo']) ? $child['episodeFile']['mediaInfo']['videoCodec'] : "unknown", "size" => $child["hasFile"] && isset($child['episodeFile']['size']) ? $child['episodeFile']['size'] : "unknown", "genres" => $child['series']['genres'], "href" => strtolower($href), "icon" => "/plugins/images/tabs/sonarr.png", "frame" => $this->config['sonarrFrameTarget'], "showLink" => $this->config['sonarrIcon'] ]; $gotCalendar[] = [ "id" => "Sonarr-" . $number . "-" . $i, "title" => $seriesName, "start" => $child['airDateUtc'], "className" => "inline-popups bg-calendar calendar-item tvID--" . $episodeID, "imagetype" => "tv " . $downloaded, "imagetypeFilter" => "tv", "downloadFilter" => $downloaded, "bgColor" => str_replace('text', 'bg', $downloaded), "details" => $details ]; } if ($i != 0) { return $gotCalendar; } return false; } }