const API_URL: string = "https://api.slopfarmer.jack-case.pro" let access_token: string const login_form = document.getElementById("login-form") as HTMLFormElement if(login_form) { const login_status = document.getElementById("login-status") if (localStorage.getItem("accessToken")) { login_status.setAttribute("style", "visibility: visible;") } login_form.addEventListener("submit", (event) => {event.preventDefault(); submit_login_form()}) } function setup_storage_db() { /* create indexeddb object store to retain objects in the form of * {"domain": "domain.tld", * "paths": [ * "page/1", * "page/2" * ] * } */ let db const db_request = window.indexedDB.open("SlopDB", 1) db_request.onerror = (event) => { // handle error console.log(event) } db_request.onsuccess = (event) => { // create objectstore console.log(event) db = event.target.result } db_request.onupgradeneeded = (event) => { console.log(event) db = event.target.result const slop_store = db.createObjectStore("slop", {keyPath: "domain"}) } } function on_install_handler() { setup_storage_db() } async function get_slop_store(readwrite: boolean) { const slop_store_promise: Promise = new Promise((resolve, reject) => { const db_request = window.indexedDB.open("SlopDB", 1) db_request.onsuccess = (event) => { const db = event.target.result const transaction = db.transaction(["slop"], readwrite ? "readwrite" : undefined) const slop_store = transaction.objectStore("slop") resolve(slop_store) } db_request.onerror = (event) => { reject(event) } }) return await slop_store_promise } async function insert_slop(domain: string, path: string, report: boolean = true) { let db const db_request = window.indexedDB.open("SlopDB", 1) db_request.onsuccess = (event) => { db = event.target.result const transaction = db.transaction(["slop"], "readwrite") const slop_store = transaction.objectStore("slop") // is this domain already stored? const request = slop_store.get(domain) request.onsuccess = () => { let result = request.result if (result) { // domain exists, add this path result.paths.add(path) } else { // create a new domain object const paths_set = new Set() paths_set.add(path) result = {domain: domain, paths: paths_set} } // persist to indexeddb const store_request = slop_store.put(result) store_request.onsuccess = () => { console.log(domain, path, "stored") } } } if(report) { const report_url = new URL("/report", API_URL) const request = new Request(report_url, { method: "POST", headers: { "Content-Type": "application/json", "Bearer": get_access_token() }, body: JSON.stringify({slop_urls: [new URL(path, "http://"+domain).toString()]}) }) fetch(request) } } async function check_local_slop(url: string) { const slop_url = new URL(url) const slop_store = await get_slop_store(false) const known_slop: Promise = new Promise((resolve, reject) => { const request = slop_store.get(slop_url.hostname) request.onsuccess = (event) => { resolve(request.result) } request.onerror = (event) => { reject(event) } }) const slop_object = await known_slop let result = {slop_domain: false, slop_path: false} if (slop_object) { // domain was found result.slop_domain = true if (slop_object.paths.has(slop_url.pathname)) { // specific page was found result.slop_path = true } } return result } async function check_remote_slop(urls: string[]) { const check_url = new URL("/check", API_URL) const request = new Request(check_url, {method: "POST", headers: { "Content-Type": "application/json", "Bearer": get_access_token() }, body: JSON.stringify({slop_urls: urls})}) const response = await fetch(request) let domain_objects = await response.json() domain_objects.forEach((domain: any) => {insert_slop(domain.domain_name, "/", false)}) return domain_objects } async function on_button_clicked_handler(tab: any) { // insert the current tab's page into slop storage const tab_url = new URL(tab.url) const domain = tab_url.hostname const path = tab_url.pathname await insert_slop(domain, path) // @ts-ignore update_page_action_icon({frameId: 0, tabId: tab.id, url: tab.url}) } async function update_page_action_icon(details: browser.webNavigation._OnCommittedDetails) { if(details.frameId != 0) { return } const is_slop = await check_local_slop(details.url) if(is_slop.slop_path) { browser.pageAction.setIcon({ path: "icons/virus_red.png", tabId: details.tabId }) } else if(is_slop.slop_domain) { browser.pageAction.setIcon({ path: "icons/virus_yellow.png", tabId: details.tabId }) } else { browser.pageAction.setIcon({ path: "icons/virus-slash.png", tabId: details.tabId }) } console.log(is_slop) } async function message_listener(message: any, sender: any) { const tabid = sender.tab.id if (message.type === "check") { let check_promises = new Array() let not_found_local = new Array() message.urls.forEach((url: string) => { check_promises.push(check_local_slop(url).then(async (result) => { if (result.slop_domain) { browser.tabs.sendMessage(tabid, { type: "check_result", url: url, result: result }) } else { not_found_local.push(url) } })) }) await Promise.all(check_promises) let remote_slop = await check_remote_slop(not_found_local) remote_slop.forEach((result: any) => { browser.tabs.sendMessage(tabid, { type: "check_result", domain: result.domain_name, result: result }) }) } } function get_access_token() { access_token = localStorage.getItem("accessToken") if (!access_token) { // get an access token from the API } return access_token } async function submit_login_form() { const login_url = new URL("/login", API_URL) const request = new Request(login_url, { method: "POST", body: new FormData(login_form) }) const response = await fetch(request) if(response.ok) { const body = await response.json() const token = body.access_token localStorage.setItem("accessToken", token) const status_el = document.getElementById("login-status") status_el.setAttribute("style", "visibility: visible;") } } if(!login_form) { browser.runtime.onInstalled.addListener(on_install_handler) browser.runtime.onStartup.addListener(get_access_token) browser.pageAction.onClicked.addListener(on_button_clicked_handler) browser.webNavigation.onCommitted.addListener(update_page_action_icon) browser.runtime.onMessage.addListener(message_listener) }