Greasy Fork

Greasy Fork is available in English.

Trakt.tv | Scheduled E-Mail Data Exports

Automatic trakt.tv backups for free users. On every trakt.tv visit a background e-mail data export is triggered, if one is overdue based on the specified cron expression (defaults to weekly). See README for details.

// ==UserScript==
// @name         Trakt.tv | Scheduled E-Mail Data Exports
// @description  Automatic trakt.tv backups for free users. On every trakt.tv visit a background e-mail data export is triggered, if one is overdue based on the specified cron expression (defaults to weekly). See README for details.
// @version      1.0.1
// @namespace    https://github.com/Fenn3c401
// @author       Fenn3c401
// @license      GPL-3.0-or-later
// @homepageURL  https://github.com/Fenn3c401/Trakt.tv-Userscript-Collection#readme
// @supportURL   https://github.com/Fenn3c401/Trakt.tv-Userscript-Collection/issues
// @icon         
// @match        https://trakt.tv/*
// @run-at       document-body
// @require      https://cdn.jsdelivr.net/npm/[email protected]/dist/croner.umd.min.js#sha256-HN5mqfFU4KGwMqzXnKEjtnbQQpK8bGtloFZZQ93GADw=
// @grant        unsafeWindow
// @grant        GM_info
// @grant        GM.getValues
// @grant        GM.setValue
// ==/UserScript==

/* README
### General
- You might want to consider the use of an e-mail filter, so as to e.g. automatically move the data export e-mails to a dedicated trakt-tv-data-exports folder.
- If you don't like the success toasts, you can turn them off by setting showToastOnSuccess to false in the userscript storage tab *(note: only displayed after first run)*, there you can
    also specify your own [cron expression](https://crontab.guru/examples.html). E-Mail data exports have a cooldown period of 24 hours, there is no point in going below that with your cron expression.
*/


/* global Cron */

'use strict';

(async () => { // iife because esbuild doesn't like top-level-await
  const $ = unsafeWindow.jQuery,
        toastr = unsafeWindow.toastr,
        userslug = unsafeWindow.Cookies?.get('trakt_userslug');
  if (!$ || !toastr || !userslug) return;

  let { showToastOnSuccess, cronExpr, lastRun } = await GM.getValues(['showToastOnSuccess', 'cronExpr', 'lastRun']);
  if (showToastOnSuccess === undefined) GM.setValue('showToastOnRun', showToastOnSuccess = true);
  if (cronExpr === undefined) GM.setValue('cronExpr', cronExpr = '@weekly');
  if (lastRun === undefined) GM.setValue('lastRun', lastRun = {});

  const Logger = Object.freeze({
    _DEFAULT_PREFIX: GM_info.script.name.replace('Trakt.tv', 'Userscript') + ': ',
    _DEFAULT_TOAST: true,
    _printMsg(fnConsole, fnToastr, msg, { data, prefix = Logger._DEFAULT_PREFIX, toast = Logger._DEFAULT_TOAST } = {}) {
      msg = prefix + msg;
      console[fnConsole](msg, (data ? data : ''));
      if (toast) toastr[fnToastr](msg + (data ? ' See console for details.' : ''));
    },
    info: (msg, opt) => Logger._printMsg('info', 'info', msg, opt),
    success: (msg, opt) => Logger._printMsg('info', 'success', msg, { ...opt, toast: showToastOnSuccess }),
    warning: (msg, opt) => Logger._printMsg('warn', 'warning', msg, opt),
    error: (msg, opt) => Logger._printMsg('error', 'error', msg, opt),
  });


  let cron;
  try {
    cron = new Cron(cronExpr, {
      timezone: Intl.DateTimeFormat().resolvedOptions().timeZone,
    });
  } catch (err) {
    Logger.error('Invalid cron expression. Aborting..', { data: err });
    return;
  }

  window.addEventListener('turbo:load', () => {
    const dateNow = new Date();
    if (!lastRun[userslug] || cron.nextRun(lastRun[userslug]) <= dateNow && dateNow - new Date(lastRun[userslug]) >= 24 * 60 * 60 * 1000) {
      $.post('/settings/export_data')
        .done(() => {
          Logger.success('Success. Your data export is processing. You will receive an e-mail when it is ready.');
          lastRun[userslug] = dateNow.toISOString();
          GM.setValue('lastRun', lastRun);
        })
        .fail((xhr) => {
          if (xhr.status === 409) {
            Logger.warning('Failed. Cooldown from previous export is still active. Will retry on next scheduled data export.');
            lastRun[userslug] = dateNow.toISOString();
            GM.setValue('lastRun', lastRun);
          } else {
            Logger.error(`Failed with status: ${xhr.status}. Reload page to try again.`, { data: xhr });
          }
        });
    }
  });
})();