Notion Invoices Bulk Downloader

Adds bulk download for Notion invoices and receipts.

이 스크립트를 설치하려면 Tampermonkey, Greasemonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

You will need to install an extension such as Tampermonkey to install this script.

이 스크립트를 설치하려면 Tampermonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey 또는 Userscripts와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 유저 스크립트 관리자 확장 프로그램이 필요합니다.

(이미 유저 스크립트 관리자가 설치되어 있습니다. 설치를 진행합니다!)

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

(이미 유저 스타일 관리자가 설치되어 있습니다. 설치를 진행합니다!)

// ==UserScript==
// @name         Notion Invoices Bulk Downloader
// @name:ru      Notion Invoices Bulk Downloader
// @namespace    http://tampermonkey.net/
// @version      1.0.1
// @description  Adds bulk download for Notion invoices and receipts.
// @description:ru Массовое скачивание инвойсов и квитанций в Notion.
// @author       DayDve
// @license      MIT
// @icon         
// @match        https://www.notion.so/*
// @connect      stripe.com
// @connect      amazonaws.com
// @grant        GM_xmlhttpRequest
// @grant        GM_addStyle
// @run-at       document-start
// @keywords     notion, invoices, download, pdf
// ==/UserScript==

(function() {
    'use strict';

    // --- STYLES ---
    GM_addStyle(`
        /* Download Selected Button */
        #nir-mass-download-btn {
            user-select: none; transition: background 20ms ease-in, opacity 200ms;
            display: flex; align-items: center; justify-content: center; height: 32px;
            padding-inline: 12px; border-radius: 6px; white-space: nowrap;
            font-size: 14px; font-weight: 500; line-height: 1.2;
            margin-inline-start: 12px;
            cursor: not-allowed;

            opacity: 0.6;
            color: #888;
            border: 1px solid #555;
            background: #333;
            flex-shrink: 0;
        }

        #nir-mass-download-btn.active {
            opacity: 1;
            cursor: pointer;
            color: #2383e2;
            border: 1px solid rgba(35, 131, 226, 0.5);
            background: rgba(35, 131, 226, 0.1);
        }
        #nir-mass-download-btn.active:hover {
            background: rgba(35, 131, 226, 0.25);
        }

        #nir-mass-download-btn.loading {
            opacity: 0.7;
            cursor: wait !important;
            pointer-events: none;
        }

        /* Invoice Header Wrapper: Makes original header and new button inline */
        .nir-invoices-header-flex {
            display: flex;
            align-items: center;
            justify-content: flex-start;
        }

        /* Original Invoices Title (hidden to use modified copy in the flex container) */
        .nir-original-hidden {
            display: none !important;
        }

        /* Checkbox Container (inserted into row) */
        .nir-checkbox-container {
            display: flex;
            align-items: center;
            justify-content: center;
            width: 20px;
            margin-right: 8px;
            flex-shrink: 0;
            flex-grow: 0;
        }

        /* Modified Invoice Row: Enables checkbox placement */
        .nir-invoice-row-modified {
            display: flex !important;
            align-items: center !important;
            transition: background-color 0.3s ease;
        }

        /* Status Colors */
        .nir-row-downloading {
            background-color: rgba(255, 255, 0, 0.1) !important;
        }
        .nir-row-success {
            background-color: rgba(0, 128, 0, 0.1) !important;
        }
        .nir-row-error {
            background-color: rgba(255, 0, 0, 0.1) !important;
        }
    `);

    // --- GLOBAL STATE ---
    const STATE = {
        headers: null,
        invoiceMap: {}, // "Month D, YYYY" -> "in_XXXXXX"
        rawEvents: [],
        selectedInvoices: new Set(),
        processing: false
    };

    // --- 1. INTERCEPTOR (Capture Billing Data) ---
    function injectInterceptor() {
        const script = document.createElement('script');
        script.textContent = `
            (() => {
                const originalFetch = window.fetch;
                window.fetch = function(...args) {
                    const [resource, config] = args;
                    if (typeof resource === 'string' && resource.includes('getBillingHistory')) {
                        if (config && config.body) {
                            const p = originalFetch.apply(this, args);
                            p.then(res => res.clone().json()).then(json => {
                                window.dispatchEvent(new CustomEvent('NotionInvoiceCaptured', {
                                    detail: { headers: config.headers, data: json }
                                }));
                            });
                            return p;
                        }
                    }
                    return originalFetch.apply(this, args);
                };
            })();
        `;
        (document.head || document.documentElement).appendChild(script);
        script.remove();
    }
    injectInterceptor();

    // --- 2. DATA LISTENER ---
    window.addEventListener('NotionInvoiceCaptured', (e) => {
        const events = e.detail.data.events || [];
        const invoices = events.filter(x => x.type === 'invoice');

        STATE.headers = e.detail.headers;
        STATE.rawEvents = invoices;
        STATE.selectedInvoices.clear();

        // Build map for DOM injection lookup
        const formatter = new Intl.DateTimeFormat('en-US', { month: 'long', day: 'numeric', year: 'numeric' });

        invoices.forEach(inv => {
            const m = inv.url.match(/(in_[a-zA-Z0-9]+)/);
            if (m) {
                const id = m[1];
                const dateText = formatter.format(new Date(inv.timestamp));
                STATE.invoiceMap[dateText] = id;
            }
        });

        startObserver();
    });

    // --- 3. DOM OBSERVER (Injection Logic) ---
    let observerTimer = null;
    function startObserver() {
        if (observerTimer) clearInterval(observerTimer);

        observerTimer = setInterval(() => {
            injectMassDownloadButton();

            // Find invoice rows based on unique styling
            const invoiceRows = document.querySelectorAll('div[style*="display: flex;"][style*="gap: 40px;"][style*="border-bottom: 1px solid var(--ca-regDivCol);"][style*="padding: 11px 0px;"]');

            invoiceRows.forEach(row => {
                if (row.classList.contains('nir-invoice-row-modified')) return;

                const infoCol = row.children[0];
                if (!infoCol) return;

                // Look for date text in the info column
                const dateMatch = infoCol.innerText.match(/([A-Z][a-z]+ \d{1,2}, \d{4})/);

                if (dateMatch) {
                    const dateString = dateMatch[0];
                    const invoiceId = STATE.invoiceMap[dateString];

                    if (invoiceId) {
                        row.classList.add('nir-invoice-row-modified');
                        injectCheckbox(row, invoiceId);
                    }
                }
            });
        }, 500);
    }

    // --- 4. UI INJECTION FUNCTIONS ---

    function injectCheckbox(rowElement, invoiceId) {
        const checkContainer = document.createElement('div');
        checkContainer.className = 'nir-checkbox-container';

        const checkbox = document.createElement('input');
        checkbox.type = 'checkbox';
        checkbox.style.cursor = 'pointer';
        checkbox.id = `nir-check-${invoiceId}`;

        checkbox.onchange = () => {
            if (checkbox.checked) {
                STATE.selectedInvoices.add(invoiceId);
            } else {
                STATE.selectedInvoices.delete(invoiceId);
            }
            updateMassDownloadButtonState();
        };
        checkContainer.appendChild(checkbox);

        rowElement.insertBefore(checkContainer, rowElement.firstChild);

        // Status indicator insertion (empty div for applying color classes)
        const actionsCol = rowElement.children[2];
        if (actionsCol) {
            const statusIndicator = document.createElement('div');
            statusIndicator.id = `nir-status-${invoiceId}`;
            // Preserve the original margin-end of Notion's default button column
            statusIndicator.style.cssText = 'width: 0px; margin-inline-end: 10px;';

            const viewButton = actionsCol.querySelector('[role="button"]');
            if (viewButton) {
                actionsCol.insertBefore(statusIndicator, viewButton);
            }
        }
    }

    function injectMassDownloadButton() {
        const xpathResult = document.evaluate(
            '//div[text()="Invoices" and @style]',
            document, null, XPathResult.FIRST_ORDERED_NODE_TYPE, null
        );
        const invoicesTitle = xpathResult.singleNodeValue;

        if (!invoicesTitle) return;
        if (invoicesTitle.classList.contains('nir-processed')) return;
        invoicesTitle.classList.add('nir-processed');

        const titleCopy = invoicesTitle.cloneNode(true);

        let massBtn = document.createElement('div');
        massBtn.id = 'nir-mass-download-btn';
        massBtn.innerHTML = 'Download Selected (0)';
        massBtn.onclick = runMassDownload;

        const newContainer = document.createElement('div');
        newContainer.className = 'nir-invoices-header-flex';

        // Transfer styles (border-bottom, padding-bottom, margin-bottom) to the new container
        const originalStyle = invoicesTitle.getAttribute('style');
        newContainer.setAttribute('style', originalStyle);

        // Clear conflicting styles from the title copy
        titleCopy.style.borderBottom = 'none';
        titleCopy.style.paddingBottom = '0px';
        titleCopy.style.marginBottom = '0px';
        titleCopy.style.flexShrink = '0';

        newContainer.appendChild(titleCopy);
        newContainer.appendChild(massBtn);

        // Insert the new container before the original title
        const originalParent = invoicesTitle.parentElement;
        originalParent.insertBefore(newContainer, invoicesTitle);

        // Hide the original title element
        invoicesTitle.classList.add('nir-original-hidden');

        updateMassDownloadButtonState(massBtn);
    }

    function updateMassDownloadButtonState(btn = document.getElementById('nir-mass-download-btn')) {
        if (!btn || STATE.processing) return;

        const count = STATE.selectedInvoices.size;
        btn.innerHTML = `Download Selected (${count})`;

        if (count > 0) {
            btn.classList.add('active');
        } else {
            btn.classList.remove('active');
        }
    }

    // --- 5. DOWNLOAD LOGIC ---

    async function runMassDownload() {
        const btn = document.getElementById('nir-mass-download-btn');
        if (!btn || !btn.classList.contains('active')) return;

        STATE.processing = true;
        btn.classList.add('loading');

        const headers = { ...STATE.headers, 'Cookie': document.cookie };
        delete headers['content-length'];

        const filesCollection = [];
        const queue = Array.from(STATE.selectedInvoices);

        try {
            for (let i = 0; i < queue.length; i++) {
                const invoiceId = queue[i];
                btn.innerHTML = `⏳ ${i + 1}/${queue.length}`;

                const rowEl = getRowElement(invoiceId);
                if (rowEl) {
                    rowEl.classList.add('nir-row-downloading');
                    rowEl.classList.remove('nir-row-success', 'nir-row-error');
                }

                try {
                    const result = await processViaNotionAPI(invoiceId, headers);
                    if (result.invoice.data) filesCollection.push(result.invoice);
                    if (result.receipt.data) filesCollection.push(result.receipt);

                    if (rowEl) {
                        rowEl.classList.remove('nir-row-downloading');
                        rowEl.classList.add('nir-row-success');
                    }
                } catch(e) {
                    if (rowEl) {
                        rowEl.classList.remove('nir-row-downloading');
                        rowEl.classList.add('nir-row-error');
                    }
                    console.warn(`Error processing ${invoiceId}:`, e);
                }

                await sleep(200);
            }

            btn.innerHTML = "Packing ZIP...";
            const zipBlob = createZip(filesCollection);
            saveBlob(zipBlob, `Notion_Invoices.zip`);

            STATE.selectedInvoices.clear();

            btn.innerHTML = "✅ Done!";
        } catch (e) {
            btn.innerHTML = "❌ Error";
            console.error(e);
        } finally {
            STATE.processing = false;
            // Reset status and checkboxes
            setTimeout(() => {
                btn.classList.remove('loading');
                updateMassDownloadButtonState();
                queue.forEach(id => {
                    const rowEl = getRowElement(id);
                    if (rowEl) {
                        rowEl.classList.remove('nir-row-success', 'nir-row-error');
                    }
                    const checkbox = document.getElementById(`nir-check-${id}`);
                    if (checkbox) checkbox.checked = false;
                });
            }, 3000);
        }
    }

    // --- CORE API WORKER (Native) ---
    async function processViaNotionAPI(invoiceId, authHeaders) {
        const notionRes = await fetch('https://www.notion.so/api/v3/getInvoiceData', {
            method: 'POST', headers: authHeaders, body: JSON.stringify({ type: 'invoice', invoiceId: invoiceId })
        });
        const nJson = await notionRes.json();
        const data = nJson.invoiceData;

        const dateStr = new Date(data.date).toISOString().split('T')[0];
        const num = data.invoiceNumber || invoiceId;

        const stripeUrl = data.hostedInvoiceUrl;
        const m = stripeUrl.match(/invoice\.stripe\.com\/i\/([^\/]+)\/([^\?]+)/);
        if (!m) throw new Error("Bad Stripe URL");

        const acct = m[1];
        const id = m[2];

        const [invData, recData] = await Promise.all([
            fetchDataViaJson(`https://invoicedata.stripe.com/invoice_pdf_file_url/${acct}/${id}`),
            fetchDataViaJson(`https://invoicedata.stripe.com/invoice_receipt_file_url/${acct}/${id}`)
        ]);

        return {
            invoice: { name: `${dateStr}__Invoice-${num}.pdf`, data: invData },
            receipt: { name: `${dateStr}__Receipt-${num}.pdf`, data: recData }
        };
    }

    async function fetchDataViaJson(apiUrl) {
        try {
            const json = await gmGetJSON(apiUrl);
            if (!json.file_url) return null;
            return await gmGetBytes(json.file_url);
        } catch (e) { return null; }
    }

    // --- ZIP ENGINE (No Lib) ---
    function createZip(files) {
        const parts = [];
        let offset = 0;
        const centralDirectory = [];
        const crcTable = new Int32Array(256);
        for (let i = 0; i < 256; i++) {
            let c = i;
            for (let k = 0; k < 8; k++) c = (c & 1) ? (0xEDB88320 ^ (c >>> 1)) : (c >>> 1);
            crcTable[i] = c;
        }
        function crc32(u8arr) {
            let crc = -1;
            for (let i = 0; i < u8arr.length; i++) crc = (crc >>> 8) ^ crcTable[(crc ^ u8arr[i]) & 0xFF];
            return (crc ^ -1) >>> 0;
        }
        const encoder = new TextEncoder();

        for (const file of files) {
            if(!file.data) continue;
            const nameBytes = encoder.encode(file.name);
            const data = file.data;
            const crc = crc32(data);
            const header = new Uint8Array(30 + nameBytes.length);
            const view = new DataView(header.buffer);
            view.setUint32(0, 0x04034b50, true);
            view.setUint16(4, 0x000a, true);
            view.setUint16(6, 0x0000, true);
            view.setUint16(8, 0x0000, true);
            view.setUint32(14, crc, true);
            view.setUint32(18, data.length, true);
            view.setUint32(22, data.length, true);
            view.setUint16(26, nameBytes.length, true);
            view.setUint16(28, 0, true);
            header.set(nameBytes, 30);
            parts.push(header);
            parts.push(data);
            centralDirectory.push({ nameBytes, crc, size: data.length, offset });
            offset += header.length + data.length;
        }
        const cdStartOffset = offset;
        for (const cd of centralDirectory) {
            const header = new Uint8Array(46 + cd.nameBytes.length);
            const view = new DataView(header.buffer);
            view.setUint32(0, 0x02014b50, true);
            view.setUint16(4, 0x000a, true);
            view.setUint16(6, 0x000a, true);
            view.setUint16(8, 0x0000, true);
            view.setUint16(10, 0x0000, true);
            view.setUint32(16, cd.crc, true);
            view.setUint32(20, cd.size, true);
            view.setUint32(24, cd.size, true);
            view.setUint16(28, cd.nameBytes.length, true);
            view.setUint16(30, 0, true);
            view.setUint16(32, 0, true);
            view.setUint16(34, 0, true);
            view.setUint16(36, 0, true);
            view.setUint32(38, 0, true);
            view.setUint32(42, cd.offset, true);
            header.set(cd.nameBytes, 46);
            parts.push(header);
            offset += header.length;
        }
        const eocd = new Uint8Array(22);
        const view = new DataView(eocd.buffer);
        view.setUint32(0, 0x06054b50, true);
        view.setUint16(4, 0, true);
        view.setUint16(6, 0, true);
        view.setUint16(8, centralDirectory.length, true);
        view.setUint16(10, centralDirectory.length, true);
        view.setUint32(12, offset - cdStartOffset, true);
        view.setUint32(16, cdStartOffset, true);
        parts.push(eocd);
        return new Blob(parts, { type: 'application/zip' });
    }

    // --- HELPERS ---
    function getRowElement(invoiceId) {
        const checkbox = document.getElementById(`nir-check-${invoiceId}`);
        return checkbox ? checkbox.closest('.nir-invoice-row-modified') : null;
    }
    function gmGetJSON(url) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "GET", url,
                onload: r => { try { resolve(JSON.parse(r.responseText)) } catch { reject() } },
                onerror: reject
            });
        });
    }
    function gmGetBytes(url) {
        return new Promise((resolve, reject) => {
            GM_xmlhttpRequest({
                method: "GET", url, responseType: 'arraybuffer',
                onload: r => { if(r.status===200) resolve(new Uint8Array(r.response)); else reject(); },
                onerror: reject
            });
        });
    }
    function saveBlob(blob, name) {
        const a = document.createElement("a");
        a.href = URL.createObjectURL(blob);
        a.download = name;
        a.style.display = 'none';
        document.body.appendChild(a);
        a.click();
        setTimeout(() => { URL.revokeObjectURL(a.href); document.body.removeChild(a); }, 1000);
    }
    function sleep(ms) { return new Promise(r => setTimeout(r, ms)); }
    // No-op log function for clean code
    function log() {}

})();