lycan/js/hush-hush.js

190 lines
7.1 KiB
JavaScript

/*
Lycan: anonymous message board using the onionr network
Copyright (C) 2020 Kevin Froman
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU Affero General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Affero General Public License for more details.
You should have received a copy of the GNU Affero General Public License
along with this program. If not, see <https://www.gnu.org/licenses/>.
*/
var findMessageIntervalTime = 5000
var publicNodes = [
"ueawiiskhaxdhkqjvgz6drrlf7srvaifrewnb6rxf6tro3welajvlgyd",
"csb2thc5yzv2gbhoozbqrzv747irs5z2lbpd7eiyh6eivvltok76qrqd"
]
var initialNodes = JSON.parse(JSON.stringify(publicNodes))
publicNodes = []
var messageHashes = []
var blocks = []
var basicTextEncoder = new TextEncoder()
var difficulty = "00000"
var maxBlockAge = 2678400
var postTopic = document.getElementById('boardID').value // we use block types as the topic with 'kic' as the prefix
var lastLookup = Math.floor((Date.now() / 1000)) - maxBlockAge
if (document.location.protocol === "file:"){
alert("Sorry, this does not support file: access. Use a localhost HTTP server or Electron.\n\n$ python3 -m http.server --bind 127.0.0.1 8080")
}
function shuffleArray(array) {
if (document.hidden){return}
for (let i = array.length - 1; i > 0; i--) {
const j = Math.floor(Math.random() * (i + 1));
[array[i], array[j]] = [array[j], array[i]];
}
}
shuffleArray(initialNodes)
//https://stackoverflow.com/q/10420352
function getReadableFileSizeString(fileSizeInBytes) {
var i = -1
var byteUnits = [' kB', ' MB', ' GB', ' TB', 'PB', 'EB', 'ZB', 'YB']
do {
fileSizeInBytes = fileSizeInBytes / 1024
i++;
} while (fileSizeInBytes > 1024)
return Math.max(fileSizeInBytes, 0.1).toFixed(1) + byteUnits[i]
}
setInterval(function(){shuffleArray(publicNodes)}, 5000)
function getCurrentNode(){
// Very basic round-robin use of nodes
let current = publicNodes.shift()
publicNodes.push(current)
return current
}
function addMessage(message, timestamp){
function sortEntries() {
var entries = document.getElementsByClassName('entry')
if (entries.length > 1) {
const sortBy = 'data-epoch'
const parent = entries[0].parentNode
const sorted = Array.from(entries).sort((a, b) => b.getAttribute(sortBy) - a.getAttribute(sortBy))
sorted.forEach(element => parent.appendChild(element))
}
}
message = DOMPurify.sanitize(message,
{FORBID_ATTR: ['style'],
ALLOWED_TAGS: ['b', 'p', 'em', 'i', 'a', 'strong', 'sub', 'small', 'ul', 'li', 'ol', 'strike',
'tr', 'td', 'th', 'table', 'thead', 'tfoot', 'colgroup', 'col', 'caption', 'marquee',
'h1', 'h2', 'h3', 'h4', 'h5', 'h6', 'pre', 'center', 'br', 'hr']})
let childEl = document.createElement('div')
childEl.classList.add('content')
childEl.innerHTML = message
var tmpl = document.getElementById("cMsgTemplate")
timestamp = timestamp.toString()
let newEl = tmpl.content.cloneNode(true)
newEl.children[0].setAttribute('data-epoch', timestamp)
newEl.children[0].classList.add("entry")
newEl.children[0].children[0].children[0].innerText = ""
newEl.children[0].children[0].children[0].append(childEl)
newEl.children[0].children[0].children[2].innerText = new Date(timestamp * 1000).toString().split('GMT')[0]
document.getElementsByClassName("messageFeed")[0].prepend(newEl)
sortEntries()
}
async function apiGET(path, queryString, raw=false){
let nodeToUse = getCurrentNode()
if (! nodeToUse){
console.debug("no node yet")
return
}
let requestTimeout = setTimeout(function(){
console.debug(nodeToUse + " timed out")
publicNodes = publicNodes.filter(item => item !== nodeToUse)
}, 10000)
let response = await fetch("http://" + nodeToUse + ".onion/" + encodeURIComponent(path) + queryString)
if (response.ok) { // if HTTP-status is 200-299
// get the response body (the method explained below)
if (raw){
connectedToOnionr = true
return await response.blob()
}
clearTimeout(requestTimeout)
document.getElementById("lastReached").classList.replace("has-text-warning", "has-text-success")
document.getElementById("lastReached").innerText = "Onionr network reached"
return await response.text()
} else {
console.debug("HTTP-Error: " + response.status)
}
}
async function findMessages(){
findMessageIntervalTime = 3000
if (document.hidden){
findMessageIntervalTime = 1000
}
try{
var messages = (await apiGET("getblocklist", "?type=" + postTopic + "&date=" + lastLookup)).split('\n')
}
catch(e){
setTimeout(function(){findMessages()}, findMessageIntervalTime)
return
}
lastLookup = Math.floor((Date.now() / 1000)) - (findMessageIntervalTime / 1000) - 1200
messages.forEach(block => {
if (!block) { return}
block = reconstructHash(block)
if (!block.startsWith(difficulty)){console.debug("not difficulty reached:" + block); return}
if (blocks.includes(block)){return}
apiGET("getdata", "/" + block, raw=false).then(function(d){
let updateMemoryUsage = function(data, block){
let current = parseInt(document.getElementById('memUsage').innerText)
// Size is size of data (not metadata) and block hash
document.getElementById('memUsage').innerText = getReadableFileSizeString(current + ((basicTextEncoder.encode(data)).length + block.length))
}
try{
if (typeof d === "undefined"){
return
}
var metadata = JSON.parse(d.split("\n")[0])
// Make sure the block is an actual post so nodes can't send us stuff unrelated to dapp
if (JSON.parse(metadata['meta'])['type'] !== postTopic){
throw new Error("Not correct block type: " + block)
}
}
catch(e){
console.debug(e)
return
}
let data = d.substring(d.indexOf('\n') + 1);
try{
verifyBlock(d, block)
verifyTime(metadata['time'])
}
catch(e){
console.debug(block + ":" + e)
return
}
blocks.push(block)
addMessage(data, metadata['time'])
updateMemoryUsage(data, block)
})
})
setTimeout(function(){findMessages()}, findMessageIntervalTime)
}
setTimeout(function(){findMessages()}, findMessageIntervalTime)