// ==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/croner@9.0.0/dist/croner.umd.min.js#sha256-HN5mqfFU4KGwMqzXnKEjtnbQQpK8bGtloFZZQ93GADw= // @grant unsafeWindow // @grant GM_info // @grant GM.getValues // @grant GM.setValue // @downloadURL https://update.greasyfork.icu/scripts/550078/Trakttv%20%7C%20Scheduled%20E-Mail%20Data%20Exports.user.js // @updateURL https://update.greasyfork.icu/scripts/550078/Trakttv%20%7C%20Scheduled%20E-Mail%20Data%20Exports.meta.js // ==/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 }); } }); } }); })();