mirror of
https://github.com/ajnart/homarr.git
synced 2025-11-10 07:25:48 +01:00
✨ Add NZBGet download client
This commit is contained in:
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Tillader dig at se din Sabnzbd kø og historie, pause og genoptage downloads"
|
"description": "Giver dig mulighed for at se din usenet (Sabnzbd eller NZBGet) kø og historik, pause og genoptage downloads"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Ermöglicht es Ihnen, Ihre Sabnzbd-Warteschlange und den Verlauf zu sehen, Downloads anzuhalten und fortzusetzen"
|
"description": "Ermöglicht es Ihnen, Ihre Usenet-Warteschlange (Sabnzbd oder NZBGet) und den Verlauf anzuzeigen, Downloads anzuhalten und fortzusetzen"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -90,6 +90,22 @@
|
|||||||
"invalidPassword": "Invalid password"
|
"invalidPassword": "Invalid password"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"nzbget": {
|
||||||
|
"username": {
|
||||||
|
"label": "Username",
|
||||||
|
"placeholder": "admin",
|
||||||
|
"validation": {
|
||||||
|
"invalidUsername": "Invalid username"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"password": {
|
||||||
|
"label": "Password",
|
||||||
|
"placeholder": "password",
|
||||||
|
"validation": {
|
||||||
|
"invalidPassword": "Invalid password"
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Allows you to see your Sabnzbd queue and history, pause and resume downloads"
|
"description": "Allows you to see your usenet (Sabnzbd or NZBGet) queue and history, pause and resume downloads"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Te permite ver tu cola e historial de Sabnzbd, pausar y resumir descargas"
|
"description": "Le permite ver la cola y el historial de usenet (Sabnzbd o NZBGet), pausar y reanudar las descargas"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Permet de voir votre file d'attente et votre historique Sabnzbd, de mettre en pause et de reprendre les téléchargements"
|
"description": "Vous permet de voir votre file d'attente et votre historique Usenet (Sabnzbd ou NZBGet), de mettre en pause et de reprendre les téléchargements"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Sabnzbd",
|
||||||
"description": "מאפשר צפייה בהיסטוריה ותור ההורדות, עצירה וחידוש הורדות"
|
"description": "מאמאפשר לך לראות את התור וההיסטוריה של usenet (Sabnzbd או NZBGet), להשהות ולחדש הורדות"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Permette di vedere la coda e la cronologia di Sabnzbd, di mettere in pausa e di riprendere i download."
|
"description": "Ti consente di vedere la coda e la cronologia di usenet (Sabnzbd o NZBGet), mettere in pausa e riprendere i download."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "ユースネット",
|
||||||
"description": "Sabnzbdのキューと履歴の確認、ダウンロードの一時停止と再開が可能です。"
|
"description": "Usenet (Sabnzbd または NZBGet) のキューと履歴を表示し、ダウンロードを一時停止および再開できます。"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "ユースネット",
|
||||||
"description": "Sabnzbd의 대기열 및 기록을 확인하고 다운로드를 일시 중지 및 재개할 수 있습니다"
|
"description": "유즈넷(Sabnzbd 또는 NZBGet) 대기열 및 기록을 보고 다운로드를 일시 중지 및 재개할 수 있습니다."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Allowz u 2 c ur Sabnzbd queue an histowee, paues an resuem downloadz"
|
"description": "Allowz u 2 c ur usenet (Sabnzbd & NZBGet) queue an histowee, paues an resuem downloadz"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Hiermee kunt u uw Sabnzbd wachtrij en geschiedenis weergeven, downloads pauzeren en hervatten"
|
"description": "Hiermee kunt u uw usenet (Sabnzbd of NZBGet) wachtrij en geschiedenis bekijken, downloads pauzeren en hervatten."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Pozwala zobaczyć kolejkę i historię Sabnzbd, wstrzymywać i wznawiać pobieranie"
|
"description": "Pozwala zobaczyć kolejkę i historię usenetu (Sabnzbd lub NZBGet), wstrzymywać i wznawiać pobieranie."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "",
|
"name": "Usenet",
|
||||||
"description": ""
|
"description": "Permite que você veja sua fila e histórico de usenet (Sabnzbd ou NZBGet), pausar e retomar downloads."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "",
|
"name": "Юнет",
|
||||||
"description": ""
|
"description": "Позволяет просматривать очередь и историю Usenet (Sabnzbd или NZBGet), приостанавливать и возобновлять загрузки."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Omogoča ogled čakalne vrste in zgodovine v Sabnzbd ter prekinitev in nadaljevanje prenosov"
|
"description": "Omogoča ogled vaše čakalne vrste in zgodovine usenet (Sabnzbd ali NZBGet), zaustavitev in nadaljevanje prenosov."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Usenet",
|
||||||
"description": "Låter dig se din Sabnzbd kö och historik, pausa och återuppta nedladdningar"
|
"description": "Låter dig se din usenet (Sabnzbd eller NZBGet) kö och historik, pausa och återuppta nedladdningar."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "Sabnzbd",
|
||||||
"description": "Дозволяє вам бачити вашу чергу в Sabnzbd і історію, призупинення і відновлення завантажень"
|
"description": "Дозволяє переглядати чергу та історію usenet (Sabnzbd або NZBGet), призупиняти та відновлювати завантаження."
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"descriptor": {
|
"descriptor": {
|
||||||
"name": "Sabnzbd",
|
"name": "用户网",
|
||||||
"description": "允许您查看您的Sabnzbd队列和历史,进行暂停和恢复下载操作"
|
"description": "允许您查看您的用户网(Sabnzbd 或 NZBGet)队列和历史记录,暂停和恢复下载。"
|
||||||
},
|
},
|
||||||
"card": {
|
"card": {
|
||||||
"errors": {
|
"errors": {
|
||||||
|
|||||||
@@ -407,6 +407,42 @@ export function AddAppShelfItemForm(props: AddAppShelfItemFormProps) {
|
|||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
{form.values.type === 'NZBGet' && (
|
||||||
|
<>
|
||||||
|
<TextInput
|
||||||
|
label={t('modal.tabs.options.form.integrations.nzbget.username.label')}
|
||||||
|
placeholder={t(
|
||||||
|
'modal.tabs.options.form.integrations.nzbget.username.placeholder'
|
||||||
|
)}
|
||||||
|
value={form.values.username}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('username', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={
|
||||||
|
form.errors.username &&
|
||||||
|
t(
|
||||||
|
'modal.tabs.options.form.integrations.nzbget.username.validation.invalidUsername'
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
<PasswordInput
|
||||||
|
label={t('modal.tabs.options.form.integrations.nzbget.password.label')}
|
||||||
|
placeholder={t(
|
||||||
|
'modal.tabs.options.form.integrations.nzbget.password.placeholder'
|
||||||
|
)}
|
||||||
|
value={form.values.password}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('password', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={
|
||||||
|
form.errors.password &&
|
||||||
|
t(
|
||||||
|
'modal.tabs.options.form.integrations.nzbget.password.validation.invalidPassword'
|
||||||
|
)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</Stack>
|
</Stack>
|
||||||
</Tabs.Panel>
|
</Tabs.Panel>
|
||||||
<Tabs.Panel value="Advanced Options">
|
<Tabs.Panel value="Advanced Options">
|
||||||
|
|||||||
@@ -16,7 +16,7 @@ import { AddItemShelfButton } from '../../components/AppShelf/AddAppShelfItem';
|
|||||||
dayjs.extend(duration);
|
dayjs.extend(duration);
|
||||||
|
|
||||||
export const UsenetComponent: FunctionComponent = () => {
|
export const UsenetComponent: FunctionComponent = () => {
|
||||||
const downloadServices = useGetServiceByType('Sabnzbd');
|
const downloadServices = useGetServiceByType('Sabnzbd', 'NZBGet');
|
||||||
|
|
||||||
const { t } = useTranslation('modules/usenet');
|
const { t } = useTranslation('modules/usenet');
|
||||||
|
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ import { UsenetHistoryItem } from '../../../../modules';
|
|||||||
import { getConfig } from '../../../../tools/getConfig';
|
import { getConfig } from '../../../../tools/getConfig';
|
||||||
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
||||||
import { Config } from '../../../../tools/types';
|
import { Config } from '../../../../tools/types';
|
||||||
|
import { NzbgetHistoryItem } from './nzbget/types';
|
||||||
|
import { NzbgetClient } from './nzbget/nzbget-client';
|
||||||
|
|
||||||
dayjs.extend(duration);
|
dayjs.extend(duration);
|
||||||
|
|
||||||
@@ -33,23 +35,72 @@ async function Get(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!service.apiKey) {
|
let response: UsenetHistoryResponse;
|
||||||
throw new Error(`API Key for service "${service.name}" is missing`);
|
switch (service.type) {
|
||||||
|
case 'NZBGet': {
|
||||||
|
const url = new URL(service.url);
|
||||||
|
const options = {
|
||||||
|
host: url.hostname,
|
||||||
|
port: url.port,
|
||||||
|
login: service.username,
|
||||||
|
hash: service.password,
|
||||||
|
};
|
||||||
|
|
||||||
|
const nzbGet = NzbgetClient(options);
|
||||||
|
|
||||||
|
const nzbgetHistory:NzbgetHistoryItem[] = await new Promise((resolve, reject) => {
|
||||||
|
nzbGet.history(false, (err: any, result: NzbgetHistoryItem[]) => {
|
||||||
|
if (!err) {
|
||||||
|
resolve(result);
|
||||||
|
} else {
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!nzbgetHistory) {
|
||||||
|
throw new Error('Error while getting NZBGet history');
|
||||||
|
}
|
||||||
|
|
||||||
|
const nzbgetItems: UsenetHistoryItem[] = nzbgetHistory.map((item: NzbgetHistoryItem) => ({
|
||||||
|
id: item.NZBID.toString(),
|
||||||
|
name: item.Name,
|
||||||
|
// Convert from MB to bytes
|
||||||
|
size: item.DownloadedSizeMB * 1000000,
|
||||||
|
time: item.DownloadTimeSec,
|
||||||
|
}));
|
||||||
|
|
||||||
|
response = {
|
||||||
|
items: nzbgetItems,
|
||||||
|
total: nzbgetItems.length,
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'Sabnzbd': {
|
||||||
|
const { origin } = new URL(service.url);
|
||||||
|
|
||||||
|
if (!service.apiKey) {
|
||||||
|
throw new Error(`API Key for service "${service.name}" is missing`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const history = await new Client(origin, service.apiKey).history(offset, limit);
|
||||||
|
|
||||||
|
const items: UsenetHistoryItem[] = history.slots.map((slot) => ({
|
||||||
|
id: slot.nzo_id,
|
||||||
|
name: slot.name,
|
||||||
|
size: slot.bytes,
|
||||||
|
time: slot.download_time,
|
||||||
|
}));
|
||||||
|
|
||||||
|
response = {
|
||||||
|
items,
|
||||||
|
total: history.noofslots,
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw new Error(`Service type "${service.type}" unrecognized.`);
|
||||||
}
|
}
|
||||||
const { origin } = new URL(service.url);
|
|
||||||
|
|
||||||
const history = await new Client(origin, service.apiKey).history(offset, limit);
|
|
||||||
|
|
||||||
const items: UsenetHistoryItem[] = history.slots.map((slot) => ({
|
|
||||||
id: slot.nzo_id,
|
|
||||||
name: slot.name,
|
|
||||||
size: slot.bytes,
|
|
||||||
time: slot.download_time,
|
|
||||||
}));
|
|
||||||
const response: UsenetHistoryResponse = {
|
|
||||||
items,
|
|
||||||
total: history.noofslots,
|
|
||||||
};
|
|
||||||
|
|
||||||
return res.status(200).json(response);
|
return res.status(200).json(response);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import { Client } from 'sabnzbd-api';
|
|||||||
import { getConfig } from '../../../../tools/getConfig';
|
import { getConfig } from '../../../../tools/getConfig';
|
||||||
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
||||||
import { Config } from '../../../../tools/types';
|
import { Config } from '../../../../tools/types';
|
||||||
|
import { NzbgetStatus } from './nzbget/types';
|
||||||
|
import { NzbgetClient } from './nzbget/nzbget-client';
|
||||||
|
|
||||||
dayjs.extend(duration);
|
dayjs.extend(duration);
|
||||||
|
|
||||||
@@ -32,28 +34,71 @@ async function Get(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!service.apiKey) {
|
let response: UsenetInfoResponse;
|
||||||
throw new Error(`API Key for service "${service.name}" is missing`);
|
switch (service.type) {
|
||||||
|
case 'NZBGet': {
|
||||||
|
const url = new URL(service.url);
|
||||||
|
const options = {
|
||||||
|
host: url.hostname,
|
||||||
|
port: url.port,
|
||||||
|
login: service.username,
|
||||||
|
hash: service.password,
|
||||||
|
};
|
||||||
|
|
||||||
|
const nzbGet = NzbgetClient(options);
|
||||||
|
|
||||||
|
const nzbgetStatus:NzbgetStatus = await new Promise((resolve, reject) => {
|
||||||
|
nzbGet.status((err: any, result: NzbgetStatus) => {
|
||||||
|
if (!err) {
|
||||||
|
resolve(result);
|
||||||
|
} else {
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!nzbgetStatus) {
|
||||||
|
throw new Error('Error while getting NZBGet status');
|
||||||
|
}
|
||||||
|
|
||||||
|
const bytesRemaining = nzbgetStatus.RemainingSizeMB * 1000000;
|
||||||
|
const eta = bytesRemaining / nzbgetStatus.DownloadRate;
|
||||||
|
response = {
|
||||||
|
paused: nzbgetStatus.DownloadPaused,
|
||||||
|
sizeLeft: bytesRemaining,
|
||||||
|
speed: nzbgetStatus.DownloadRate,
|
||||||
|
eta,
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'Sabnzbd': {
|
||||||
|
if (!service.apiKey) {
|
||||||
|
throw new Error(`API Key for service "${service.name}" is missing`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { origin } = new URL(service.url);
|
||||||
|
|
||||||
|
const queue = await new Client(origin, service.apiKey).queue(0, -1);
|
||||||
|
|
||||||
|
const [hours, minutes, seconds] = queue.timeleft.split(':');
|
||||||
|
const eta = dayjs.duration({
|
||||||
|
hour: parseInt(hours, 10),
|
||||||
|
minutes: parseInt(minutes, 10),
|
||||||
|
seconds: parseInt(seconds, 10),
|
||||||
|
} as any);
|
||||||
|
|
||||||
|
response = {
|
||||||
|
paused: queue.paused,
|
||||||
|
sizeLeft: parseFloat(queue.mbleft) * 1024 * 1024,
|
||||||
|
speed: parseFloat(queue.kbpersec) * 1000,
|
||||||
|
eta: eta.asSeconds(),
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw new Error(`Service type "${service.type}" unrecognized.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const { origin } = new URL(service.url);
|
|
||||||
|
|
||||||
const queue = await new Client(origin, service.apiKey).queue(0, -1);
|
|
||||||
|
|
||||||
const [hours, minutes, seconds] = queue.timeleft.split(':');
|
|
||||||
const eta = dayjs.duration({
|
|
||||||
hour: parseInt(hours, 10),
|
|
||||||
minutes: parseInt(minutes, 10),
|
|
||||||
seconds: parseInt(seconds, 10),
|
|
||||||
} as any);
|
|
||||||
|
|
||||||
const response: UsenetInfoResponse = {
|
|
||||||
paused: queue.paused,
|
|
||||||
sizeLeft: parseFloat(queue.mbleft) * 1024 * 1024,
|
|
||||||
speed: parseFloat(queue.kbpersec) * 1000,
|
|
||||||
eta: eta.asSeconds(),
|
|
||||||
};
|
|
||||||
|
|
||||||
return res.status(200).json(response);
|
return res.status(200).json(response);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return res.status(500).send((err as any).message);
|
return res.status(500).send((err as any).message);
|
||||||
|
|||||||
1
src/pages/api/modules/usenet/nzbget/nzbget-api.d.ts
vendored
Normal file
1
src/pages/api/modules/usenet/nzbget/nzbget-api.d.ts
vendored
Normal file
@@ -0,0 +1 @@
|
|||||||
|
declare module 'nzbget-api';
|
||||||
22
src/pages/api/modules/usenet/nzbget/nzbget-client.ts
Normal file
22
src/pages/api/modules/usenet/nzbget/nzbget-client.ts
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import NZBGet from 'nzbget-api';
|
||||||
|
import { NzbgetClientOptions } from './types';
|
||||||
|
|
||||||
|
export function NzbgetClient(options: NzbgetClientOptions) {
|
||||||
|
if (!options?.host) {
|
||||||
|
throw new Error('Cannot connect to NZBGet. Missing host in service config.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!options?.port) {
|
||||||
|
throw new Error('Cannot connect to NZBGet. Missing port in service config.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!options?.login) {
|
||||||
|
throw new Error('Cannot connect to NZBGet. Missing username in service config.');
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!options?.hash) {
|
||||||
|
throw new Error('Cannot connect to NZBGet. Missing password in service config.');
|
||||||
|
}
|
||||||
|
|
||||||
|
return new NZBGet(options);
|
||||||
|
}
|
||||||
135
src/pages/api/modules/usenet/nzbget/types.ts
Normal file
135
src/pages/api/modules/usenet/nzbget/types.ts
Normal file
@@ -0,0 +1,135 @@
|
|||||||
|
export interface NzbgetHistoryItem {
|
||||||
|
NZBID: number;
|
||||||
|
Kind: 'NZB' | 'URL' | 'DUP';
|
||||||
|
NZBFilename: string;
|
||||||
|
Name: string;
|
||||||
|
URL: string;
|
||||||
|
HistoryTime: number;
|
||||||
|
DestDir: string;
|
||||||
|
FinalDir: string;
|
||||||
|
Category: string;
|
||||||
|
FileSizeLo: number;
|
||||||
|
FileSizeHi: number;
|
||||||
|
FileSizeMB: number;
|
||||||
|
FileCount: number;
|
||||||
|
RemainingFileCount: number;
|
||||||
|
MinPostTime: number;
|
||||||
|
MaxPostTime: number;
|
||||||
|
TotalArticles: number;
|
||||||
|
SuccessArticles: number;
|
||||||
|
FailedArticles: number;
|
||||||
|
Health: number;
|
||||||
|
DownloadedSizeLo: number;
|
||||||
|
DownloadedSizeHi: number;
|
||||||
|
DownloadedSizeMB: number;
|
||||||
|
DownloadTimeSec: number;
|
||||||
|
PostTotalTimeSec: number;
|
||||||
|
ParTimeSec: number;
|
||||||
|
RepairTimeSec: number
|
||||||
|
UnpackTimeSec: number;
|
||||||
|
MessageCount:number;
|
||||||
|
DupeKey: string;
|
||||||
|
DupeScore: number;
|
||||||
|
DupeMode: 'SCORE' | 'ALL' | 'FORCE';
|
||||||
|
Status: string;
|
||||||
|
ParStatus: 'NONE' | 'FAILURE' | 'REPAIR_POSSIBLE' | 'SUCCESS' | 'MANUAL';
|
||||||
|
ExParStatus: 'RECIPIENT' | 'DONOR';
|
||||||
|
UnpackStatus: 'NONE' | 'FAILURE' | 'SPACE' | 'PASSWORD' | 'SUCCESS';
|
||||||
|
UrlStatus: 'NONE' | 'SUCCESS' | 'FAILURE' | 'SCAN_SKIPPED' | 'SCAN_FAILURE';
|
||||||
|
ScriptStatus: 'NONE' | 'FAILURE' | 'SUCCESS';
|
||||||
|
ScriptStatuses: [];
|
||||||
|
MoveStatus: 'NONE' | 'SUCCESS' | 'FAILURE';
|
||||||
|
DeleteStatus: 'NONE' | 'MANUAL' | 'HEALTH' | 'DUPE' | 'BAD' | 'SCAN' | 'COPY';
|
||||||
|
MarkStatus: 'NONE' | 'GOOD' | 'BAD';
|
||||||
|
ExtraParBlocks: number;
|
||||||
|
Parameters: [];
|
||||||
|
ServerStats: [];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface NzbgetQueueItem {
|
||||||
|
NZBID: number;
|
||||||
|
NZBFilename: string;
|
||||||
|
NZBName: string;
|
||||||
|
Kind: 'NZB' | 'URL';
|
||||||
|
URL: string;
|
||||||
|
DestDir: string;
|
||||||
|
FinalDir: string;
|
||||||
|
Category: string;
|
||||||
|
FileSizeLo: number;
|
||||||
|
FileSizeHi: number;
|
||||||
|
FileSizeMB: number;
|
||||||
|
RemainingSizeLo: number;
|
||||||
|
RemainingSizeHi: number;
|
||||||
|
RemainingSizeMB: number;
|
||||||
|
PausedSizeLo: number;
|
||||||
|
PausedSizeHi: number;
|
||||||
|
PausedSizeMB: number;
|
||||||
|
FileCount: number;
|
||||||
|
RemainingFileCount: number;
|
||||||
|
RemainingParCount: number;
|
||||||
|
MinPostTime: number;
|
||||||
|
MaxPostTime: number;
|
||||||
|
MaxPriority: number;
|
||||||
|
ActiveDownloads: number;
|
||||||
|
Status: 'QUEUED' | 'PAUSED' | 'DOWNLOADING' | 'FETCHING' | 'PP_QUEUED' | 'LOADING_PARS' | 'VERIFYING_SOURCES' | 'REPAIRING' | 'VERIFYING_REPAIRED' | 'RENAMING' | 'UNPACKING' | 'MOVING' | 'EXECUTING_SCRIPT' | 'PP_FINISHED';
|
||||||
|
TotalArticles: number;
|
||||||
|
SuccessArticles: number;
|
||||||
|
FailedArticles: number;
|
||||||
|
Health: number;
|
||||||
|
CriticalHealth: number;
|
||||||
|
DownloadedSizeLo: number;
|
||||||
|
DownloadedSizeHi: number;
|
||||||
|
DownloadedSizeMB: number;
|
||||||
|
DownloadTimeSec: number;
|
||||||
|
MessageCount: number;
|
||||||
|
DupeKey: string;
|
||||||
|
DupeScore: number;
|
||||||
|
DupeMode: string;
|
||||||
|
Parameters: [];
|
||||||
|
ServerStats: [];
|
||||||
|
PostInfoText: string;
|
||||||
|
PostStageProgress: number;
|
||||||
|
PostTotalTimeSec: number;
|
||||||
|
PostStageTimeSec: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface NzbgetStatus {
|
||||||
|
RemainingSizeLo: number;
|
||||||
|
RemainingSizeHi: number;
|
||||||
|
RemainingSizeMB: number;
|
||||||
|
ForcedSizeLo: number;
|
||||||
|
ForcedSizeHi: number;
|
||||||
|
ForcedSizeMB: number;
|
||||||
|
DownloadedSizeLo: number;
|
||||||
|
DownloadedSizeHi: number;
|
||||||
|
DownloadedSizeMB: number;
|
||||||
|
ArticleCacheLo: number;
|
||||||
|
ArticleCacheHi: number;
|
||||||
|
ArticleCacheMB: number;
|
||||||
|
DownloadRate: number;
|
||||||
|
AverageDownloadRate: number;
|
||||||
|
DownloadLimit: number;
|
||||||
|
ThreadCount: number;
|
||||||
|
PostJobCount: number;
|
||||||
|
UrlCount: number;
|
||||||
|
UpTimeSec: number;
|
||||||
|
DownloadTimeSec: number;
|
||||||
|
ServerStandBy: boolean;
|
||||||
|
DownloadPaused: boolean;
|
||||||
|
PostPaused: boolean;
|
||||||
|
ScanPaused: boolean;
|
||||||
|
ServerTime: number;
|
||||||
|
ResumeTime: number;
|
||||||
|
FeedActive: boolean;
|
||||||
|
FreeDiskSpaceLo: number;
|
||||||
|
FreeDiskSpaceHi: number;
|
||||||
|
FreeDiskSpaceMB: number;
|
||||||
|
NewsServers: [];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface NzbgetClientOptions {
|
||||||
|
host: string,
|
||||||
|
port: string,
|
||||||
|
login: string | undefined,
|
||||||
|
hash: string | undefined,
|
||||||
|
}
|
||||||
@@ -6,6 +6,7 @@ import { Client } from 'sabnzbd-api';
|
|||||||
import { getConfig } from '../../../../tools/getConfig';
|
import { getConfig } from '../../../../tools/getConfig';
|
||||||
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
||||||
import { Config } from '../../../../tools/types';
|
import { Config } from '../../../../tools/types';
|
||||||
|
import { NzbgetClient } from './nzbget/nzbget-client';
|
||||||
|
|
||||||
dayjs.extend(duration);
|
dayjs.extend(duration);
|
||||||
|
|
||||||
@@ -25,14 +26,44 @@ async function Post(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!service.apiKey) {
|
let result;
|
||||||
throw new Error(`API Key for service "${service.name}" is missing`);
|
switch (service.type) {
|
||||||
|
case 'NZBGet': {
|
||||||
|
const url = new URL(service.url);
|
||||||
|
const options = {
|
||||||
|
host: url.hostname,
|
||||||
|
port: url.port,
|
||||||
|
login: service.username,
|
||||||
|
hash: service.password,
|
||||||
|
};
|
||||||
|
|
||||||
|
const nzbGet = NzbgetClient(options);
|
||||||
|
|
||||||
|
result = await new Promise((resolve, reject) => {
|
||||||
|
nzbGet.pauseDownload(false, (err: any, result: any) => {
|
||||||
|
if (!err) {
|
||||||
|
resolve(result);
|
||||||
|
} else {
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'Sabnzbd': {
|
||||||
|
if (!service.apiKey) {
|
||||||
|
throw new Error(`API Key for service "${service.name}" is missing`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { origin } = new URL(service.url);
|
||||||
|
|
||||||
|
result = await new Client(origin, service.apiKey).queuePause();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw new Error(`Service type "${service.type}" unrecognized.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const { origin } = new URL(service.url);
|
|
||||||
|
|
||||||
const result = await new Client(origin, service.apiKey).queuePause();
|
|
||||||
|
|
||||||
return res.status(200).json(result);
|
return res.status(200).json(result);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return res.status(500).send((err as any).message);
|
return res.status(500).send((err as any).message);
|
||||||
|
|||||||
@@ -7,6 +7,8 @@ import { UsenetQueueItem } from '../../../../modules';
|
|||||||
import { getConfig } from '../../../../tools/getConfig';
|
import { getConfig } from '../../../../tools/getConfig';
|
||||||
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
||||||
import { Config } from '../../../../tools/types';
|
import { Config } from '../../../../tools/types';
|
||||||
|
import { NzbgetClient } from './nzbget/nzbget-client';
|
||||||
|
import { NzbgetQueueItem, NzbgetStatus } from './nzbget/types';
|
||||||
|
|
||||||
dayjs.extend(duration);
|
dayjs.extend(duration);
|
||||||
|
|
||||||
@@ -33,42 +35,116 @@ async function Get(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!service.apiKey) {
|
let response: UsenetQueueResponse;
|
||||||
throw new Error(`API Key for service "${service.name}" is missing`);
|
switch (service.type) {
|
||||||
|
case 'NZBGet': {
|
||||||
|
const url = new URL(service.url);
|
||||||
|
const options = {
|
||||||
|
host: url.hostname,
|
||||||
|
port: url.port,
|
||||||
|
login: service.username,
|
||||||
|
hash: service.password,
|
||||||
|
};
|
||||||
|
|
||||||
|
const nzbGet = NzbgetClient(options);
|
||||||
|
|
||||||
|
const nzbgetQueue:NzbgetQueueItem[] = await new Promise((resolve, reject) => {
|
||||||
|
nzbGet.listGroups((err: any, result: NzbgetQueueItem[]) => {
|
||||||
|
if (!err) {
|
||||||
|
resolve(result);
|
||||||
|
} else {
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!nzbgetQueue) {
|
||||||
|
throw new Error('Error while getting NZBGet queue');
|
||||||
|
}
|
||||||
|
|
||||||
|
const nzbgetStatus:NzbgetStatus = await new Promise((resolve, reject) => {
|
||||||
|
nzbGet.status((err: any, result: NzbgetStatus) => {
|
||||||
|
if (!err) {
|
||||||
|
resolve(result);
|
||||||
|
} else {
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!nzbgetStatus) {
|
||||||
|
throw new Error('Error while getting NZBGet status');
|
||||||
|
}
|
||||||
|
|
||||||
|
const nzbgetItems: UsenetQueueItem[] = nzbgetQueue.map((item: NzbgetQueueItem) => ({
|
||||||
|
id: item.NZBID.toString(),
|
||||||
|
name: item.NZBName,
|
||||||
|
progress: (item.DownloadedSizeMB / item.FileSizeMB) * 100,
|
||||||
|
eta: (item.RemainingSizeMB * 1000000) / nzbgetStatus.DownloadRate,
|
||||||
|
// Multiple MB to get bytes
|
||||||
|
size: item.FileSizeMB * 1000 * 1000,
|
||||||
|
state: getNzbgetState(item.Status),
|
||||||
|
}));
|
||||||
|
|
||||||
|
response = {
|
||||||
|
items: nzbgetItems,
|
||||||
|
total: nzbgetItems.length,
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'Sabnzbd': {
|
||||||
|
if (!service.apiKey) {
|
||||||
|
throw new Error(`API Key for service "${service.name}" is missing`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { origin } = new URL(service.url);
|
||||||
|
const queue = await new Client(origin, service.apiKey).queue(offset, limit);
|
||||||
|
|
||||||
|
const items: UsenetQueueItem[] = queue.slots.map((slot) => {
|
||||||
|
const [hours, minutes, seconds] = slot.timeleft.split(':');
|
||||||
|
const eta = dayjs.duration({
|
||||||
|
hour: parseInt(hours, 10),
|
||||||
|
minutes: parseInt(minutes, 10),
|
||||||
|
seconds: parseInt(seconds, 10),
|
||||||
|
} as any);
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: slot.nzo_id,
|
||||||
|
eta: eta.asSeconds(),
|
||||||
|
name: slot.filename,
|
||||||
|
progress: parseFloat(slot.percentage),
|
||||||
|
size: parseFloat(slot.mb) * 1000 * 1000,
|
||||||
|
state: slot.status.toLowerCase() as any,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
response = {
|
||||||
|
items,
|
||||||
|
total: queue.noofslots,
|
||||||
|
};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw new Error(`Service type "${service.type}" unrecognized.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const { origin } = new URL(service.url);
|
|
||||||
const queue = await new Client(origin, service.apiKey).queue(offset, limit);
|
|
||||||
|
|
||||||
const items: UsenetQueueItem[] = queue.slots.map((slot) => {
|
|
||||||
const [hours, minutes, seconds] = slot.timeleft.split(':');
|
|
||||||
const eta = dayjs.duration({
|
|
||||||
hour: parseInt(hours, 10),
|
|
||||||
minutes: parseInt(minutes, 10),
|
|
||||||
seconds: parseInt(seconds, 10),
|
|
||||||
} as any);
|
|
||||||
|
|
||||||
return {
|
|
||||||
id: slot.nzo_id,
|
|
||||||
eta: eta.asSeconds(),
|
|
||||||
name: slot.filename,
|
|
||||||
progress: parseFloat(slot.percentage),
|
|
||||||
size: parseFloat(slot.mb) * 1000 * 1000,
|
|
||||||
state: slot.status.toLowerCase() as any,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
const response: UsenetQueueResponse = {
|
|
||||||
items,
|
|
||||||
total: queue.noofslots,
|
|
||||||
};
|
|
||||||
|
|
||||||
return res.status(200).json(response);
|
return res.status(200).json(response);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return res.status(500).send((err as any).message);
|
return res.status(500).send((err as any).message);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getNzbgetState(status: string) {
|
||||||
|
switch (status) {
|
||||||
|
case 'QUEUED':
|
||||||
|
return 'queued';
|
||||||
|
case 'PAUSED ':
|
||||||
|
return 'paused';
|
||||||
|
default:
|
||||||
|
return 'downloading';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
||||||
// Filter out if the reuqest is a POST or a GET
|
// Filter out if the reuqest is a POST or a GET
|
||||||
if (req.method === 'GET') {
|
if (req.method === 'GET') {
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import { Client } from 'sabnzbd-api';
|
|||||||
import { getConfig } from '../../../../tools/getConfig';
|
import { getConfig } from '../../../../tools/getConfig';
|
||||||
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
import { getServiceById } from '../../../../tools/hooks/useGetServiceByType';
|
||||||
import { Config } from '../../../../tools/types';
|
import { Config } from '../../../../tools/types';
|
||||||
|
import { NzbgetClient } from './nzbget/nzbget-client';
|
||||||
|
|
||||||
dayjs.extend(duration);
|
dayjs.extend(duration);
|
||||||
|
|
||||||
@@ -26,14 +27,44 @@ async function Post(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
throw new Error(`Service with ID "${req.query.serviceId}" could not be found.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!service.apiKey) {
|
let result;
|
||||||
throw new Error(`API Key for service "${service.name}" is missing`);
|
switch (service.type) {
|
||||||
|
case 'NZBGet': {
|
||||||
|
const url = new URL(service.url);
|
||||||
|
const options = {
|
||||||
|
host: url.hostname,
|
||||||
|
port: url.port,
|
||||||
|
login: service.username,
|
||||||
|
hash: service.password,
|
||||||
|
};
|
||||||
|
|
||||||
|
const nzbGet = NzbgetClient(options);
|
||||||
|
|
||||||
|
result = await new Promise((resolve, reject) => {
|
||||||
|
nzbGet.resumeDownload(false, (err: any, result: any) => {
|
||||||
|
if (!err) {
|
||||||
|
resolve(result);
|
||||||
|
} else {
|
||||||
|
reject(err);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'Sabnzbd': {
|
||||||
|
if (!service.apiKey) {
|
||||||
|
throw new Error(`API Key for service "${service.name}" is missing`);
|
||||||
|
}
|
||||||
|
|
||||||
|
const { origin } = new URL(service.url);
|
||||||
|
|
||||||
|
result = await new Client(origin, service.apiKey).queueResume();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default:
|
||||||
|
throw new Error(`Service type "${service.type}" unrecognized.`);
|
||||||
}
|
}
|
||||||
|
|
||||||
const { origin } = new URL(service.url);
|
|
||||||
|
|
||||||
const result = await new Client(origin, service.apiKey).queueResume();
|
|
||||||
|
|
||||||
return res.status(200).json(result);
|
return res.status(200).json(result);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
return res.status(500).send((err as any).message);
|
return res.status(500).send((err as any).message);
|
||||||
|
|||||||
@@ -76,6 +76,7 @@ export const ServiceTypeList = [
|
|||||||
'Overseerr',
|
'Overseerr',
|
||||||
'Jellyseerr',
|
'Jellyseerr',
|
||||||
'Sabnzbd',
|
'Sabnzbd',
|
||||||
|
'NZBGet'
|
||||||
];
|
];
|
||||||
export type ServiceType =
|
export type ServiceType =
|
||||||
| 'Other'
|
| 'Other'
|
||||||
@@ -91,7 +92,8 @@ export type ServiceType =
|
|||||||
| 'Overseerr'
|
| 'Overseerr'
|
||||||
| 'Jellyseerr'
|
| 'Jellyseerr'
|
||||||
| 'Transmission'
|
| 'Transmission'
|
||||||
| 'Sabnzbd';
|
| 'Sabnzbd'
|
||||||
|
| 'NZBGet';
|
||||||
|
|
||||||
export function tryMatchPort(name: string | undefined, form?: any) {
|
export function tryMatchPort(name: string | undefined, form?: any) {
|
||||||
if (!name) {
|
if (!name) {
|
||||||
@@ -118,6 +120,7 @@ export const portmap = [
|
|||||||
{ name: 'overseerr', value: '5055' },
|
{ name: 'overseerr', value: '5055' },
|
||||||
{ name: 'dash.', value: '3001' },
|
{ name: 'dash.', value: '3001' },
|
||||||
{ name: 'sabnzbd', value: '8080' },
|
{ name: 'sabnzbd', value: '8080' },
|
||||||
|
{ name: 'nzbget', value: '6789' }
|
||||||
];
|
];
|
||||||
|
|
||||||
export const MatchingImages: {
|
export const MatchingImages: {
|
||||||
|
|||||||
Reference in New Issue
Block a user