Tsinghua E-Reserves Lib Downloader

Download PDF from Tsinghua University Electronic Course Reserves Service Platform

// ==UserScript==
// @name         Tsinghua E-Reserves Lib Downloader
// @namespace    anyi.fan
// @version      0.2.2
// @license      GPL-3.0 License
// @description  Download PDF from Tsinghua University Electronic Course Reserves Service Platform
// @author       A1phaN
// @match        https://ereserves.lib.tsinghua.edu.cn/readkernel/ReadJPG/JPGJsNetPage/*
// @grant        none
// ==/UserScript==

const MAX_RETRY = 10;
const QUERY_INTERVAL = 100;

const sleep = time => new Promise(res => setTimeout(res, time));
const getImage = async (url, retry = MAX_RETRY) => {
  const img = new Image();
  img.src = url;
  img.style.display = 'none';
  const data = new Promise((res, rej) => {
    img.onload = () => {
      const canvas = document.createElement('canvas');
      canvas.width = img.width;
      canvas.height = img.height;
      canvas.getContext('2d')?.drawImage(img, 0, 0, img.width, img.height);
      res([img, canvas.toDataURL('image/jpeg')]);
    };
    img.onerror = err => {
      retry > 0 ? res(getImage(url, retry - 1)) : rej(err);
    };
  });
  document.body.appendChild(img);
  return data;
};

(async () => {
  const scanId = document.querySelector('#scanid')?.value;
  const bookId = location.href.split('/').at(-1);
  const bookNameElement = document.querySelector('#p_bookname');
  const bookName = bookNameElement.innerText;
  const BotuReadKernel = document.cookie.split(';')
    .map(c => c.trim())
    .find(c => c.startsWith('BotuReadKernel='))
    ?.split('=')[1];
  if (!scanId || !BotuReadKernel || !bookName) return;

  await new Promise(res => {
    if (window.jspdf) return res();
    const script = document.createElement('script');
    script.src = 'https://cdnjs.cloudflare.com/ajax/libs/jspdf/2.5.1/jspdf.umd.min.js';
    script.onload = res;
    document.body.appendChild(script);
  });
  if (!window.jspdf) return;

  const button = document.createElement('span');
  button.className = 'fucBtn icon iconfont';
  button.style = 'margin-left: 8px; position: relative; top: 2px;';
  button.innerHTML =
  `<svg
    t="1703917701009"
    viewBox="0 0 1024 1024"
    version="1.1"
    xmlns="http://www.w3.org/2000/svg"
    p-id="5061"
    height="2rem"
  >
    <path
      d="M896 672c-17.066667 0-32 14.933333-32 32v128c0 6.4-4.266667 10.666667-10.666667 10.666667H170.666667c-6.4 0-10.666667-4.266667-10.666667-10.666667v-128c0-17.066667-14.933333-32-32-32s-32 14.933333-32 32v128c0 40.533333 34.133333 74.666667 74.666667 74.666667h682.666666c40.533333 0 74.666667-34.133333 74.666667-74.666667v-128c0-17.066667-14.933333-32-32-32z"
      fill="#ddd"
      p-id="5062"
    ></path>
    <path
      d="M488.533333 727.466667c6.4 6.4 14.933333 8.533333 23.466667 8.533333s17.066667-2.133333 23.466667-8.533333l213.333333-213.333334c12.8-12.8 12.8-32 0-44.8-12.8-12.8-32-12.8-44.8 0l-157.866667 157.866667V170.666667c0-17.066667-14.933333-32-32-32s-34.133333 14.933333-34.133333 32v456.533333L322.133333 469.333333c-12.8-12.8-32-12.8-44.8 0-12.8 12.8-12.8 32 0 44.8l211.2 213.333334z"
      fill="#ddd"
      p-id="5063"
    ></path>
  </svg>`;
  document.querySelector('.option-list').appendChild(button);

  const downloadPDF = async () => {
    try {
      button.onclick = null;
      let doc = null;
      let page = 1;
      const chapters = await (
        await fetch(
          '/readkernel/KernelAPI/BookInfo/selectJgpBookChapters',
          {
            body: `SCANID=${scanId}`,
            headers: {
              Botureadkernel: BotuReadKernel,
              'Content-Type': 'application/x-www-form-urlencoded;charset=UTF-8',
            },
            method: 'POST',
          },
        )
      ).json();
      if (chapters.code !== 1 || !Array.isArray(chapters.data)) {
        alert('Get chapters data failed!');
        return;
      }

      for (let chap = 0; chap < chapters.data.length; ++chap) {
        bookNameElement.innerText = `${bookName}(正在获取第 ${chap + 1} 章...)`;
        const chapter = chapters.data[chap];
        const chapterData = await (
          await fetch(
            '/readkernel/KernelAPI/BookInfo/selectJgpBookChapter',
            {
              body: `EMID=${chapter.EMID}&BOOKID=${bookId}`,
              headers: {
                Botureadkernel: BotuReadKernel,
                'Content-Type': 'application/x-www-form-urlencoded;charset=UTF-8',
              },
              method: 'POST',
            },
          )
        ).json();
        if (chapterData.code !== 1 || !Array.isArray(chapterData.data.JGPS)) {
          alert(`Get chapter ${chap + 1} ${chapter.EFRAGMENTNAME} data failed!`);
          return;
        }

        for (let i = 0; i < chapterData.data.JGPS.length; ++i) {
          const jpg = chapterData.data.JGPS[i];
          try {
            const [img, dataURL] = await getImage(`/readkernel/JPGFile/DownJPGJsNetPage?filePath=${jpg.hfsKey}`);
            if (!doc) {
              doc = new jspdf.jsPDF({ format: [img.width, img.height], unit: 'px' });
            } else {
              doc.addPage([img.width, img.height]);
            }
            doc.addImage(dataURL, 'JPEG', 0, 0, img.width, img.height);
            bookNameElement.innerText = `${bookName}(正在获取第 ${chap + 1} 章,已完成: ${i + 1} / ${chapterData.data.JGPS.length})`;
            await sleep(QUERY_INTERVAL);
          } catch(e) {
            alert(`Get page ${i + 1} of chapter ${chap + 1} ${chapter.EFRAGMENTNAME} failed!`);
            return;
          }
        }
        doc.outline.add(null, chapter.EFRAGMENTNAME, { pageNumber: page });
        page += chapterData.data.JGPS.length;
      }
      if (doc) {
        const filename = `${bookName}.pdf`;
        try {
          doc.save(filename);
        } catch(e) {
          if (e instanceof RangeError && e.message === 'Invalid string length') {
            // jsPDF exceeds the maximum string length
            doc.__private__.resetCustomOutputDestination();
            const content = doc.__private__.out('');
            content.pop();
            const blob = new Blob(
              content.map((line, idx) => {
                const str = idx === content.length - 1 ? line : line + '\n';
                const arrayBuffer = new ArrayBuffer(str.length);
                const uint8Array = new Uint8Array(arrayBuffer);
                for (let i = 0; i < str.length; ++i) {
                  uint8Array[i] = str.charCodeAt(i);
                }
                return arrayBuffer;
              }),
              { type: 'application/pdf' },
            );
            const a = document.createElement('a');
            a.download = filename;
            a.rel = 'noopener';
            a.href = URL.createObjectURL(blob);
            a.click();
          } else {
            alert(`Unexpected Error when saving pdf: ${e?.message ?? e}`);
          }
        }
      }
      bookNameElement.innerText = `${bookName}(下载完成)`;
    } finally {
      button.onclick = downloadPDF;
    }
  };
  button.onclick = downloadPDF;
})();