Download on QNAP NAS

Download links or magnet links on QNAP NAS via Download Station, you can input url manually or right-click on the link and then hit on "Download from Last Link"

您需要先安裝使用者腳本管理器擴展,如 TampermonkeyGreasemonkeyViolentmonkey 之後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyViolentmonkey 後才能安裝該腳本。

您需要先安裝使用者腳本管理器擴充功能,如 TampermonkeyUserscripts 後才能安裝該腳本。

你需要先安裝一款使用者腳本管理器擴展,比如 Tampermonkey,才能安裝此腳本

您需要先安裝使用者腳本管理器擴充功能後才能安裝該腳本。

(我已經安裝了使用者腳本管理器,讓我安裝!)

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展,比如 Stylus,才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

你需要先安裝一款使用者樣式管理器擴展後才能安裝此樣式

(我已經安裝了使用者樣式管理器,讓我安裝!)

// ==UserScript==
// @name            Download on QNAP NAS
// @description     Download links or magnet links on QNAP NAS via Download Station, you can input url manually or right-click on the link and then hit on "Download from Last Link"
// @version         1.03
// @match           *://*/*
// @run-at          document-end
// @grant           none
// @copyright       2024, MSerj
// @license         MIT
//
// @grant           GM_registerMenuCommand
// @grant           GM_setValue
// @grant           GM_getValue
// @grant           GM_xmlhttpRequest

// @icon 

// @namespace       https://greasyfork.org/en/users/1321619-mserj
// ==/UserScript==

// Set up the NAS configuration prompts
async function setUpNasSettings() {
  const nasIP = prompt("Enter your NAS IP address:", GM_getValue("nasIP", ""));
  const nasPort = prompt("Enter your NAS port (default 8080):", GM_getValue("nasPort", "8080"));
  const nasProtocol = prompt("Connection protocol (default https):", GM_getValue("nasProtocol", "https"));
  const username = prompt("Enter your NAS username:", GM_getValue("username", ""));
  const password = prompt("Enter your NAS password:", GM_getValue("password", ""));
  const tempLocation = prompt("Location of Temporary Files:", GM_getValue("tempLocation", ""));
  const moveLocation = prompt("Move the completed downloads to:", GM_getValue("moveLocation", ""));
  const moveLocationAlternative = prompt("Move the completed downloads to (alternative):", GM_getValue("moveLocationAlternative", ""));
  
  GM_setValue("nasIP", nasIP);
  GM_setValue("nasPort", nasPort);
  GM_setValue("nasProtocol", nasProtocol);
  GM_setValue("username", username);
  GM_setValue("password", password);
  GM_setValue("tempLocation", tempLocation);
  GM_setValue("moveLocation", moveLocation);
  GM_setValue("moveLocationAlternative", moveLocationAlternative);
  alert("NAS settings saved.");
}

// Function to send download link to QNAP Download Station
function sendToNas(downloadUrl, useAlternativeLocation = false) {
  const nasIP = GM_getValue("nasIP");
  const nasPort = GM_getValue("nasPort");
  const nasProtocol = GM_getValue("nasProtocol");
  const username = GM_getValue("username");
  const password = GM_getValue("password");
  const tempLocation = GM_getValue("tempLocation");
  const moveLocation = GM_getValue("moveLocation");
  const moveLocationAlternative = GM_getValue("moveLocationAlternative");
  
  if (!nasIP || !nasPort || !username || !password) {
    alert("Please configure your NAS settings first.");
    setUpNasSettings();
    return;
  }
  
  // Authenticate with QNAP to get a session token
  GM_xmlhttpRequest({
    method: "POST",
    url: `${nasProtocol}://${nasIP}:${nasPort}/downloadstation/V5/Misc/Login`,
    headers: { "Content-Type": "application/x-www-form-urlencoded" },
    data: `user=${username}&pass=${btoa(password)}`,
    onload: function({response}) {
      const sid = JSON.parse(response || {})?.sid
      if (sid) {
        // Start the download task using the acquired session token (sid)
        GM_xmlhttpRequest({
          method: "GET",
          url: `${nasProtocol}://${nasIP}:${nasPort}/downloadstation/V5/Task/AddUrl?sid=${sid}&url=${encodeURIComponent(downloadUrl)}&temp=${tempLocation}&move=${useAlternativeLocation ? moveLocationAlternative : moveLocation}`,
          onload: function({response}) {
            const taskData = JSON.parse(response || {});
            if (taskData && taskData.error === 0) {
              alert("Download started on NAS!");
            } else {
              alert("Failed to start download. Please check the link and NAS settings.");
            }
          },
          onerror: function() {
            alert("Error adding the download task to NAS.");
          }
        });
      } else {
        alert("Failed to authenticate with NAS. Please check your credentials.");
      }
    },
    onerror: function() {
      alert("Error connecting to NAS for authentication.");
    }
  });
}

// Add event listener to detect right-click on links
document.addEventListener("contextmenu", (event) => {
  const target = event.target.closest("a");
  if (target && target.href) {
    // If right-clicked target is a link, create the menu command for "Download on NAS"
    GM_registerMenuCommand("Download from Last Link", () => {
      sendToNas(target.href);
    });
    GM_registerMenuCommand("Download from Last Link (alternative)", () => {
      sendToNas(target.href, true);
    });
  }
});

// Menu command to configure NAS settings
GM_registerMenuCommand("Configure NAS Settings", setUpNasSettings);

// Menu command to manually enter a link for download
GM_registerMenuCommand("Download from Manual Link", () => {
  const manualLink = prompt("Enter the URL or magnet link to download:");
  const useAlternativeLocation = prompt("Use alternative location? (y/n)");
  if (manualLink) {
    sendToNas(manualLink, !(['n', 'no', ''].some(answer => useAlternativeLocation ? answer === useAlternativeLocation?.toLowerCase() : true)));
  } else {
    alert("No link provided.");
  }
});