/* ========================================================================= *\ |* === Basic utils useful only in browser window === *| \* ========================================================================= */ // create an object to hold everything that needs to be held globally var libresilient = { info: {}, status: false, contentUnavailable: false, cacheStale: false, clientId: null } // some basic method stats libresilient.methodStats = {} // UI elements displaying the status for each local resource URL libresilient.resourceDisplays = {} /** * creating a safe CSS class name from a string */ libresilient.safeClassName = (name) => { return encodeURIComponent(name.toLowerCase()).replace(/%[0-9A-F]{2}/gi,'-') } /** * creating the standalone LibResilient UI */ libresilient.addUI = () => { var uiTemplate = document.createElement('template') uiTemplate.innerHTML = `

LibResilient is a tool that helps keep websites up without centralized CDNs.
If you are seeing this it means some content is unavailable.
LibResilient will attempt to get it for you anyway.

` var uiStyle = document.createElement('style') uiStyle.innerHTML = `#libresilient-ui { display:flex; align-items: flex-end; flex-direction:column-reverse; flex-wrap:nowrap; position:fixed; top:0px; right:0px; visibility:hidden; z-index: 1000; } #libresilient-ui.content-unavailable, #libresilient-ui:target { visibility:visible; } #libresilient-ui .libresilient-message-container { } #libresilient-ui .libresilient-message { font-size:90%; text-align:center; background:#dfd; border-radius:1em; box-shadow:0px 0px 3px #dfd; padding:0.5em 2em 0.5em 1em; transition: ease-in 0.5s opacity; opacity: 1; position: relative; top:16px; right:5px; color: #060; text-shadow: 0px 0px 2px white; font-family: sans; } #libresilient-ui .libresilient-message::after { display: block; content: "x"; position: absolute; right: 0.5em; top: 0.7em; font-size:90%; border-radius: 100%; width: 1em; height: 1em; line-height: 0.8em; padding-left: 0.01em; box-shadow: inset 0px 0px 2px #080; transition: ease-in 0.5s color, ease-in 0.5s background-color, ease-in 0.5s box-shadow-color; color: #080; background:white; } #libresilient-ui .libresilient-message:hover::after { background: #080; color: white; box-shadow: inset 0px 0px 2px black; } #libresilient-ui .libresilient-message:first-child::before { display:block; content:" "; width:1em; height:1em; position:absolute; right:1em; top:-0.5em; background:#dfd; box-shadow:0px 0px 3px #dfd; transform: rotate(45deg); z-index:-1; } #libresilient-ui-container { background:#ddd; box-shadow:0px 0px 3px black; border-bottom-left-radius:30px; padding: 4px 4px 8px 8px; display:flex; flex-wrap:nowrap; } #libresilient-ui-container .libresilient-toggle { width:32px; height:32px; background:url('') center center no-repeat; display: block; background-size:contain; } #libresilient-ui-container .libresilient-toggle > div { width:100%; height:100%; background:url('') center center no-repeat; display: block; background-size:50% 50%; } #libresilient-ui-container.active .libresilient-toggle { animation-name: libresilient-ball-rolling; animation-duration:10s; animation-iteration-count: infinite; animation-timing-function: linear; } #libresilient-ui-container #libresilient-ui-toggle { display:none; } #libresilient-ui-container > div { display:none; } #libresilient-ui-container > #libresilient-ui-toggle:checked ~ div { display:block; } #libresilient-ui-container .libresilient-description > p { font-size:80%; margin-top: 0.5em; margin-bottom: 0.5em; margin-right: 1em; text-align: right; text-shadow: -1px -1px 0px #ccc, 1px 1px 0px #eee; color: #666; font-family: sans-serif; } #libresilient-ui-container .libresilient-description > p a { color: #d70; } #libresilient-ui-container .libresilient-status-display { justify-content: right; display: flex; padding-right: 0.5em; } /* * these will be useful also outside the #libresilient-ui * for example, if there is a .libresilient-status-display in the page's HTML */ .libresilient-status-display > li { display:inline-block; font-size:80%; font-family: Monospace; } .libresilient-status-element { font-weight: bold; display: inline-block; text-align: center; text-decoration:none; background:#bbb; padding:0.4em 1em; border-radius:0.6em; color:#777; box-shadow: inset 0px 0px 3px #777; margin: 0.5em; transition: background-color 1s ease, color 1s ease, box-shadow 1s ease; } .libresilient-status-element.active { box-shadow: 0px 0px 3px #f80, 0px 0px 3px #a60; color: #fff; background: #e70; } @keyframes libresilient-ball-rolling { from {transform:rotate(0deg)} to {transform:rotate(359deg)} }` document.head.insertAdjacentElement('afterbegin', uiStyle) document.body.insertAdjacentElement('afterbegin', uiTemplate.content.firstChild) } /** * internal logging facility * * component - name of the component being logged about * items - the rest of arguments will be passed to console.debug() */ self.log = function(component, ...items) { console.debug(`LibResilient [COMMIT_UNKNOWN, ${component}] ::`, ...items) } /** * fetched resource display element */ libresilient.addFetchedResourceElements = (url, fetchedResourcesDisplays) => { // make sure we have the container element to work with if (typeof fetchedResourcesDisplays !== 'object') { fetchedResourcesDisplays = document.getElementsByClassName("libresilient-fetched-resources-list") } var itemHTML = `
  • `; var item = document.createElement('template') item.innerHTML = itemHTML; libresilient.resourceDisplays[url] = new Array() for (let frd of fetchedResourcesDisplays) { libresilient.resourceDisplays[url].push( frd.insertAdjacentElement('beforeend', item.content.firstChild.cloneNode(true)) ) } } /** * creating/updating fetched resources data */ libresilient.updateFetchedResources = () => { // getting these elements once instead of once per URL... var fetchedResourcesDisplays = document.getElementsByClassName("libresilient-fetched-resources-list") Object.keys(libresilient.info).forEach((url)=>{ // simplify var si = libresilient.info[url] // if there are no status display elements for this URL... if (typeof libresilient.resourceDisplays[url] === 'undefined') { // ...create the elements libresilient.addFetchedResourceElements(url, fetchedResourcesDisplays) // otherwise, if si.method evaluates to true (i.e. is not an empty string nor null in this case) } else { // libresilient.methodStats has the most comprehensive list of methods used Object.keys(libresilient.methodStats).forEach((method)=>{ var pclass = libresilient.safeClassName(method); var foundSuccess = false // handle per-resource displays // TODO: this needs to be done in a more efficient and elegant way for (let rdisplay of libresilient.resourceDisplays[url]) { // if we don't seem to have a display for this method in this resource displa... if (rdisplay.getElementsByClassName(pclass).length == 0) { var method_class = pclass if (typeof libresilient.info[url] !== "undefined" && typeof libresilient.info[url][method] !== "undefined") { method_class += ' ' + libresilient.info[url][method].state; } var method_item = document.createElement('template') method_item.innerHTML = `${method}` rdisplay.childNodes[0].insertAdjacentElement('beforeend', method_item.content.firstChild.cloneNode(true)) } } // do we have the method even? if (typeof si[method] === "object") { // is this a success? if (si[method].state === "success") { for (let rdisplay of libresilient.resourceDisplays[url]) { if (! rdisplay.getElementsByClassName(pclass)[0].classList.contains('success')) { // make sure the right classes are on rdisplay.getElementsByClassName(pclass)[0].classList.remove('running') rdisplay.getElementsByClassName(pclass)[0].classList.add('success') // make sure the checkbox is checked rdisplay.getElementsByTagName('input')[0].checked = true rdisplay.getElementsByTagName('input')[0].disabled = false } } // is this a running thing? } else if (si[method].state === "running") { for (let rdisplay of libresilient.resourceDisplays[url]) { if (! rdisplay.getElementsByClassName(pclass)[0].classList.contains('running')) { // make sure the right classes are on rdisplay.getElementsByClassName(pclass)[0].classList.remove('success') rdisplay.getElementsByClassName(pclass)[0].classList.add('running') } } // nope, an error presumably } else { for (let rdisplay of libresilient.resourceDisplays[url]) { // make sure the right classes are on rdisplay.getElementsByClassName(pclass)[0].classList.remove('success') rdisplay.getElementsByClassName(pclass)[0].classList.remove('running') } } // clarly this method has not even been used for the resource } else { for (let rdisplay of libresilient.resourceDisplays[url]) { // make sure the right classes are on rdisplay.getElementsByClassName(pclass)[0].classList.remove('success') rdisplay.getElementsByClassName(pclass)[0].classList.remove('running') } } }) } }) } /** * adding status display per plugin * * plugin - plugin name * description - plugin description (optional; default: empty string) * status - status text (optional; default: number of resources fetched * using this plugin, based on methodStats) */ libresilient.addPluginStatus = (plugin, description='', status=null) => { self.log('browser-side', 'addPluginStatus(' + plugin + ')') var statusDisplays = document.getElementsByClassName("libresilient-status-display"); var pclass = libresilient.safeClassName(plugin); var pcount = 0; if (typeof libresilient.methodStats[plugin] !== 'undefined') { pcount = libresilient.methodStats[plugin]; } // handle the status displays for (let sd of statusDisplays) { sd.insertAdjacentHTML('beforeend', `
  • ${plugin}: ${status ? status : pcount}
  • `) } } /** * updating status display per plugin * * expects an object that contains at least `name` attribute */ libresilient.updatePluginStatus = (plugin) => { //self.log('browser-side', 'updatePluginStatus :: ' + plugin) var pclass = libresilient.safeClassName(plugin); //self.log('browser-side', 'updatePluginStatus :: pclass: ' + pclass) var statusDisplay = document.querySelectorAll(".libresilient-status-" + pclass + " > .status"); //self.log('browser-side', 'updatePluginStatus :: statusDisplay: ' + typeof statusDisplay) var pcount = 0; if (typeof libresilient.methodStats[plugin] !== 'undefined') { pcount = libresilient.methodStats[plugin] } for (let statusDisplay of document.querySelectorAll(".libresilient-status-" + pclass + " > .status")) { statusDisplay.innerText = pcount if ( (pcount === 0) && statusDisplay.parentElement.classList.contains('active')) { statusDisplay.parentElement.classList.remove('active') } else if ( (pcount > 0) && ! statusDisplay.parentElement.classList.contains('active')) { statusDisplay.parentElement.classList.add('active') } } } /** * toggling resource checkboxes (only if not disabled) */ libresilient.toggleResourceCheckboxes = () => { document.querySelectorAll('.libresilient-fetched-resources-item input') .forEach((el)=>{ el.checked = ! el.disabled && ! el.checked }) } /** * stashing and unstashing resources * * stash param means "stash" if set to true (the default), "unstash" otherwise */ libresilient.stashOrUnstashResources = (stash=true) => { // what are we doing? var operation = { clientId: libresilient.clientId } // get the resources var resources = [] document .querySelectorAll('.libresilient-fetched-resources-item input:checked') .forEach((el)=>{ resources.push(el.parentElement.querySelector('.libresilient-fetched-resource-url').innerText) }) if (stash) { operation.stash = [resources] self.log('browser-side', 'Calling `stash()` on the service worker to stash the resources...') } else { operation.unstash = [resources] self.log('browser-side', 'Calling `unstash()` on the service worker to unstash the resources...') } // RPC call on the service worker return navigator .serviceWorker .controller .postMessage(operation) } /** * publishing certain resources to Gun and IPFS */ libresilient.publishResourcesToGunAndIPFS = () => { var user = document.getElementById('libresilient-gun-user').value var pass = document.getElementById('libresilient-gun-password').value if (! user || ! pass) { throw new Error("Gun user/password required!") } var resources = [] document.querySelectorAll('.libresilient-fetched-resources-item input:checked') .forEach((el)=>{ resources.push(el.parentElement.querySelector('.libresilient-fetched-resource-url').innerText) }) // call it! self.log('browser-side', 'Calling `publish()` on the service worker to publish the resources...') return navigator .serviceWorker .controller .postMessage({ clientId: libresilient.clientId, publish: [resources, user, pass] }) } /** * display a LibResilient message */ libresilient.displayMessage = (msg) => { // prepare the template var messageBox = document.createElement('template') messageBox.innerHTML = `
    ${msg}
    ` // attach it to all libresilient-message-containers out there for (let smc of document.getElementsByClassName('libresilient-message-container')) { var msg_clone = messageBox.content.firstChild.cloneNode(true) msg_clone.onclick = (e) => { e.target.style.opacity=0 setTimeout(()=>{e.target.remove()}, 1000) } smc.insertAdjacentElement('beforeend', msg_clone) setTimeout(()=>{ msg_clone.style.opacity=0 setTimeout(()=>{msg_clone.remove()}, 1000) }, 5000) } self.log('browser-side', ' +-- message shown!') } /** * onload handler just to mark stuff as loaded * for purposes of informing the user all is loaded * when service worker messages us about it */ window.addEventListener('load', function() { libresilient.status = "loaded"; // was any content unavailable so far? if (libresilient.contentUnavailable) { libresilient.displayMessage('Some content seems unavailable. Attempting to retrieve it via LibResilient.') } }) self.log('browser-side', 'DOMContentLoaded!') // add the generic service worker "badge" libresilient.addUI() libresilient.addPluginStatus('service worker', 'A service worker is an event-driven worker that intercepts fetch events.', 'no') /* ========================================================================= *\ |* === Service worker setup === *| \* ========================================================================= */ if ('serviceWorker' in navigator) { if (navigator.serviceWorker.controller) { // Service worker already registered. self.log('browser-side', 'Service Worker already registered.') } else { var scriptPath = document.currentScript.src var scriptFolder = scriptPath.substr(0, scriptPath.lastIndexOf( '/' )+1 ) var serviceWorkerPath = scriptFolder + 'service-worker.js' self.log('browser-side', 'Service Worker script at: ' + serviceWorkerPath) // TODO: is there a way to provide config params for the Service Worker here? // TODO: it would be good if the config.json file could reside outside of the libresilient directory // TODO: https://developer.mozilla.org/en-US/docs/Web/API/ServiceWorkerContainer/register navigator.serviceWorker.register(serviceWorkerPath, { // TODO: what is the scope relative to? is it the HTML file that included it, or this script? // TODO: "It is relative to the base URL of the application." ¯\_(ツ)_/¯ // TODO: "There is frequent confusion surrounding the meaning and use of scope. // TODO: Since a service worker can't have a scope broader than its own location, // TODO: only use the scope option when you need a scope that is narrower than the default." scope: './' }).then(function(reg) { // Success. self.log('browser-side', 'Service Worker registered.') }).catch(error => { self.log('browser-side', "Error while registering a service worker: ", error) }) } // handling the messages from ServiceWorker navigator.serviceWorker.addEventListener('message', event => { self.log('browser-side', 'LibResilientInfo received!') if (event.data.url) { self.log('browser-side', '+-- for:', event.data.url) if (event.data.method) { self.log('browser-side', ' +-- method:', event.data.method) self.log('browser-side', ' +-- state :', event.data.state) libresilient.info[event.data.url] = libresilient.info[event.data.url] || {} libresilient.info[event.data.url][event.data.method] = event.data // update method stats if (typeof libresilient.methodStats[event.data.method] === 'undefined') { // setup the stats libresilient.methodStats[event.data.method] = 0 // but also we now know this method has not been seen before // so set-up the plugin status display libresilient.addPluginStatus(event.data.method) } if (event.data.state === "success") { libresilient.methodStats[event.data.method]++ self.log('browser-side', ' +-- methodStats incremented to:', libresilient.methodStats[event.data.method]) libresilient.updatePluginStatus(event.data.method) // if the method was `fetch`, and that was the first method, and the outcome is `error`, we *might* be down } else if ( event.data.state === "error" && event.data.method === "fetch" && Object.keys(libresilient.info[event.data.url]).length === 1 && Object.keys(libresilient.info[event.data.url])[0] === "fetch" ) { // we seem to be down document.getElementById('libresilient-ui').classList.add('content-unavailable') // if contentUnavailable is false, that means this is the first time we hit a problem fetching if (!libresilient.contentUnavailable) { // mark it properly libresilient.contentUnavailable = true // if loaded, show the message to the user. // if not, the message will be shown on `load` event anyway if (libresilient.status === "loaded") { libresilient.displayMessage('Some content seems unavailable. Attempting to retrieve it via LibResilient.') } } } // update the fetched resources display // TODO: this updates *all* resources on each received message, // TODO: and so is rather wasteful libresilient.updateFetchedResources() } // we only want to mark that new content is available, and handle the message // at allFetched event if (event.data.fetchedDiffers) { self.log('browser-side', ' +-- fetched version apparently differs from cached for:', event.data.url) // record fo the URL libresilient.info[event.data.url].cacheStale = true // record gloally libresilient.cacheStale = true } } if (event.data.allFetched) { if (libresilient.status === "loaded") { // set the status so that we don't get the message doubled libresilient.status = "complete" // inform the user if (libresilient.cacheStale) { libresilient.displayMessage('Newer version of this page is available; please reload to see it.') } else { self.log('browser-side', '+-- all fetched!..') libresilient.displayMessage('Fetching via LibResilient finished; no new content found.') } } } if (event.data.clientId) { self.log('browser-side', '+-- got our clientId:', event.data.clientId) // if libresilient.clientId is null, this is the first time // we got wind that the service worker is running // service worker info if (libresilient.clientId === null) { for (let libresilient_sw of document.querySelectorAll(".libresilient-status-service-worker")) { libresilient_sw.className += " active"; try { libresilient_sw.querySelector('.status').innerHTML = "yes"; } catch(e) { // nothing to do here, move along } } } // set the clientId internally, we will need it libresilient.clientId = event.data.clientId } if (event.data.plugins) { var msg = '+-- got the plugin list:' event.data.plugins.forEach((p)=>{ msg += '\n +-- ' + p // initialize methodStats if (typeof libresilient.methodStats[p] === 'undefined') { libresilient.methodStats[p] = 0 // set-up the plugin status display libresilient.addPluginStatus(p) } }) self.log('browser-side', msg) } if (event.data.serviceWorker) { self.log('browser-side', '+-- got the serviceWorker version:', event.data.serviceWorker) var libresilient_sws = document.getElementsByClassName("libresilient-commit-service-worker"); for (let element of libresilient_sws) { element.innerHTML = event.data.serviceWorker; } } }); }