2019-02-04 00:31:03 +00:00
|
|
|
/*
|
|
|
|
Onionr - P2P Anonymous Storage Network
|
|
|
|
|
|
|
|
This file handles the mail interface
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU 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 General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2019-02-01 06:38:12 +00:00
|
|
|
pms = ''
|
2019-02-08 18:53:28 +00:00
|
|
|
sentbox = ''
|
2019-02-01 06:38:12 +00:00
|
|
|
threadPart = document.getElementById('threads')
|
2019-02-03 18:19:50 +00:00
|
|
|
threadPlaceholder = document.getElementById('threadPlaceholder')
|
2019-02-04 00:31:03 +00:00
|
|
|
tabBtns = document.getElementById('tabBtns')
|
2019-02-04 23:48:21 +00:00
|
|
|
threadContent = {}
|
2019-02-04 00:31:03 +00:00
|
|
|
myPub = httpGet('/getActivePubkey')
|
2019-02-28 03:02:44 +00:00
|
|
|
replyBtn = document.getElementById('replyBtn')
|
|
|
|
|
2019-04-09 17:30:54 +00:00
|
|
|
function openReply(bHash, quote, subject){
|
2019-02-28 03:02:44 +00:00
|
|
|
var inbox = document.getElementsByClassName('threadEntry')
|
|
|
|
var entry = ''
|
|
|
|
var friendName = ''
|
|
|
|
var key = ''
|
|
|
|
for(var i = 0; i < inbox.length; i++) {
|
|
|
|
if (inbox[i].getAttribute('data-hash') === bHash){
|
|
|
|
entry = inbox[i]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (entry.getAttribute('data-nameSet') == 'true'){
|
|
|
|
document.getElementById('friendSelect').value = entry.getElementsByTagName('input')[0].value
|
|
|
|
}
|
|
|
|
key = entry.getAttribute('data-pubkey')
|
|
|
|
document.getElementById('draftID').value = key
|
2019-04-09 17:30:54 +00:00
|
|
|
document.getElementById('draftSubject').value = 'RE: ' + subject
|
|
|
|
|
|
|
|
// Add quoted reply
|
|
|
|
var splitQuotes = quote.split('\n')
|
|
|
|
for (var x = 0; x < splitQuotes.length; x++){
|
|
|
|
splitQuotes[x] = '>' + splitQuotes[x]
|
|
|
|
}
|
|
|
|
quote = splitQuotes.join('\n')
|
|
|
|
document.getElementById('draftText').value = quote
|
2019-02-28 03:02:44 +00:00
|
|
|
setActiveTab('send message')
|
|
|
|
}
|
2019-02-04 00:31:03 +00:00
|
|
|
|
2019-04-09 17:30:54 +00:00
|
|
|
function openThread(bHash, sender, date, sigBool, pubkey, subjectLine){
|
2019-02-04 23:48:21 +00:00
|
|
|
var messageDisplay = document.getElementById('threadDisplay')
|
2019-02-07 01:03:31 +00:00
|
|
|
var blockContent = httpGet('/getblockbody/' + bHash)
|
2019-02-05 06:29:06 +00:00
|
|
|
document.getElementById('fromUser').value = sender
|
2019-04-09 17:30:54 +00:00
|
|
|
document.getElementById('subjectView').innerText = subjectLine
|
2019-02-05 06:29:06 +00:00
|
|
|
messageDisplay.innerText = blockContent
|
2019-02-07 01:03:31 +00:00
|
|
|
var sigEl = document.getElementById('sigValid')
|
|
|
|
var sigMsg = 'signature'
|
|
|
|
|
|
|
|
if (sigBool){
|
|
|
|
sigMsg = 'Good ' + sigMsg
|
|
|
|
sigEl.classList.remove('danger')
|
|
|
|
}
|
|
|
|
else{
|
2019-03-04 19:03:35 +00:00
|
|
|
sigMsg = 'Bad/no ' + sigMsg + ' (message could be impersonating someone)'
|
2019-02-07 01:03:31 +00:00
|
|
|
sigEl.classList.add('danger')
|
2019-03-04 19:03:35 +00:00
|
|
|
replyBtn.style.display = 'none'
|
2019-02-07 01:03:31 +00:00
|
|
|
}
|
|
|
|
sigEl.innerText = sigMsg
|
2019-02-04 23:48:21 +00:00
|
|
|
overlay('messageDisplay')
|
2019-02-28 03:02:44 +00:00
|
|
|
replyBtn.onclick = function(){
|
2019-04-09 17:30:54 +00:00
|
|
|
openReply(bHash, messageDisplay.innerText, subjectLine)
|
2019-02-28 03:02:44 +00:00
|
|
|
}
|
2019-02-04 23:48:21 +00:00
|
|
|
}
|
|
|
|
|
2019-02-04 00:31:03 +00:00
|
|
|
function setActiveTab(tabName){
|
|
|
|
threadPart.innerHTML = ""
|
|
|
|
switch(tabName){
|
|
|
|
case 'inbox':
|
2019-03-03 06:26:55 +00:00
|
|
|
refreshPms()
|
2019-02-04 00:31:03 +00:00
|
|
|
break
|
|
|
|
case 'sentbox':
|
2019-02-08 18:53:28 +00:00
|
|
|
getSentbox()
|
2019-02-04 00:31:03 +00:00
|
|
|
break
|
|
|
|
case 'send message':
|
2019-02-09 06:32:11 +00:00
|
|
|
overlay('sendMessage')
|
2019-02-04 00:31:03 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-03-03 06:26:55 +00:00
|
|
|
function deleteMessage(bHash){
|
|
|
|
fetch('/mail/deletemsg/' + bHash, {
|
|
|
|
"method": "post",
|
|
|
|
headers: {
|
|
|
|
"token": webpass
|
|
|
|
}})
|
|
|
|
.then((resp) => resp.text()) // Transform the data into json
|
|
|
|
.then(function(resp) {
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-03-11 05:10:37 +00:00
|
|
|
function mailPing(){
|
|
|
|
fetch('/mail/ping', {
|
|
|
|
"method": "get",
|
|
|
|
headers: {
|
|
|
|
"token": webpass
|
|
|
|
}})
|
|
|
|
.then(function(resp) {
|
|
|
|
var pings = document.getElementsByClassName('mailPing')
|
|
|
|
if (resp.ok){
|
|
|
|
for (var i=0; i < pings.length; i++){
|
|
|
|
pings[i].style.display = 'none';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
for (var i=0; i < pings.length; i++){
|
|
|
|
pings[i].style.display = 'block';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-02-28 03:02:44 +00:00
|
|
|
function loadInboxEntries(bHash){
|
2019-02-04 23:48:21 +00:00
|
|
|
fetch('/getblockheader/' + bHash, {
|
|
|
|
headers: {
|
|
|
|
"token": webpass
|
|
|
|
}})
|
|
|
|
.then((resp) => resp.json()) // Transform the data into json
|
|
|
|
.then(function(resp) {
|
2019-02-09 06:32:11 +00:00
|
|
|
//console.log(resp)
|
2019-02-04 23:48:21 +00:00
|
|
|
var entry = document.createElement('div')
|
|
|
|
var bHashDisplay = document.createElement('span')
|
|
|
|
var senderInput = document.createElement('input')
|
|
|
|
var subjectLine = document.createElement('span')
|
|
|
|
var dateStr = document.createElement('span')
|
2019-02-05 23:20:36 +00:00
|
|
|
var validSig = document.createElement('span')
|
2019-03-03 06:26:55 +00:00
|
|
|
var deleteBtn = document.createElement('button')
|
2019-02-04 23:48:21 +00:00
|
|
|
var humanDate = new Date(0)
|
|
|
|
var metadata = resp['metadata']
|
|
|
|
humanDate.setUTCSeconds(resp['meta']['time'])
|
2019-03-04 19:03:35 +00:00
|
|
|
validSig.style.display = 'none'
|
2019-02-10 02:21:36 +00:00
|
|
|
if (resp['meta']['signer'] != ''){
|
2019-02-26 04:19:37 +00:00
|
|
|
senderInput.value = httpGet('/friends/getinfo/' + resp['meta']['signer'] + '/name')
|
2019-02-10 02:21:36 +00:00
|
|
|
}
|
2019-03-04 19:03:35 +00:00
|
|
|
if (! resp['meta']['validSig']){
|
|
|
|
validSig.style.display = 'inline'
|
2019-02-05 23:20:36 +00:00
|
|
|
validSig.innerText = 'Signature Validity: Bad'
|
2019-02-09 06:32:11 +00:00
|
|
|
validSig.style.color = 'red'
|
2019-02-05 23:20:36 +00:00
|
|
|
}
|
2019-02-28 03:02:44 +00:00
|
|
|
entry.setAttribute('data-nameSet', true)
|
2019-02-05 23:20:36 +00:00
|
|
|
if (senderInput.value == ''){
|
2019-02-26 04:19:37 +00:00
|
|
|
senderInput.value = resp['meta']['signer']
|
2019-02-28 03:02:44 +00:00
|
|
|
entry.setAttribute('data-nameSet', false)
|
2019-02-05 23:20:36 +00:00
|
|
|
}
|
2019-02-04 23:48:21 +00:00
|
|
|
bHashDisplay.innerText = bHash.substring(0, 10)
|
2019-02-28 03:02:44 +00:00
|
|
|
entry.setAttribute('data-hash', bHash)
|
|
|
|
entry.setAttribute('data-pubkey', resp['meta']['signer'])
|
2019-02-04 23:48:21 +00:00
|
|
|
senderInput.readOnly = true
|
|
|
|
dateStr.innerText = humanDate.toString()
|
2019-03-03 06:26:55 +00:00
|
|
|
deleteBtn.innerText = 'X'
|
|
|
|
deleteBtn.classList.add('dangerBtn', 'deleteBtn')
|
2019-02-04 23:48:21 +00:00
|
|
|
if (metadata['subject'] === undefined || metadata['subject'] === null) {
|
|
|
|
subjectLine.innerText = '()'
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
subjectLine.innerText = '(' + metadata['subject'] + ')'
|
|
|
|
}
|
|
|
|
//entry.innerHTML = 'sender ' + resp['meta']['signer'] + ' - ' + resp['meta']['time']
|
|
|
|
threadPart.appendChild(entry)
|
2019-03-03 06:26:55 +00:00
|
|
|
entry.appendChild(deleteBtn)
|
2019-02-04 23:48:21 +00:00
|
|
|
entry.appendChild(bHashDisplay)
|
|
|
|
entry.appendChild(senderInput)
|
|
|
|
entry.appendChild(subjectLine)
|
|
|
|
entry.appendChild(dateStr)
|
2019-03-04 19:03:35 +00:00
|
|
|
entry.appendChild(validSig)
|
2019-02-04 23:48:21 +00:00
|
|
|
entry.classList.add('threadEntry')
|
|
|
|
|
2019-03-03 06:26:55 +00:00
|
|
|
entry.onclick = function(event){
|
|
|
|
if (event.target.classList.contains('deleteBtn')){
|
|
|
|
return
|
|
|
|
}
|
2019-04-09 17:30:54 +00:00
|
|
|
openThread(entry.getAttribute('data-hash'), senderInput.value, dateStr.innerText, resp['meta']['validSig'], entry.getAttribute('data-pubkey'), subjectLine.innerText)
|
2019-02-04 23:48:21 +00:00
|
|
|
}
|
2019-03-03 06:26:55 +00:00
|
|
|
|
|
|
|
deleteBtn.onclick = function(){
|
|
|
|
entry.parentNode.removeChild(entry);
|
|
|
|
deleteMessage(entry.getAttribute('data-hash'))
|
|
|
|
}
|
2019-02-04 23:48:21 +00:00
|
|
|
|
|
|
|
}.bind(bHash))
|
|
|
|
}
|
|
|
|
|
2019-02-01 06:38:12 +00:00
|
|
|
function getInbox(){
|
2019-02-03 18:19:50 +00:00
|
|
|
var showed = false
|
2019-02-04 23:48:21 +00:00
|
|
|
var requested = ''
|
2019-02-01 06:38:12 +00:00
|
|
|
for(var i = 0; i < pms.length; i++) {
|
2019-02-03 18:19:50 +00:00
|
|
|
if (pms[i].trim().length == 0){
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
threadPlaceholder.style.display = 'none'
|
|
|
|
showed = true
|
|
|
|
}
|
2019-02-28 03:02:44 +00:00
|
|
|
loadInboxEntries(pms[i])
|
2019-02-01 06:38:12 +00:00
|
|
|
}
|
2019-02-03 18:19:50 +00:00
|
|
|
if (! showed){
|
|
|
|
threadPlaceholder.style.display = 'block'
|
|
|
|
}
|
2019-02-08 18:53:28 +00:00
|
|
|
}
|
2019-02-01 06:38:12 +00:00
|
|
|
|
2019-02-08 18:53:28 +00:00
|
|
|
function getSentbox(){
|
2019-03-03 06:26:55 +00:00
|
|
|
fetch('/mail/getsentbox', {
|
2019-02-08 18:53:28 +00:00
|
|
|
headers: {
|
|
|
|
"token": webpass
|
|
|
|
}})
|
2019-02-09 06:32:11 +00:00
|
|
|
.then((resp) => resp.json()) // Transform the data into json
|
|
|
|
.then(function(resp) {
|
|
|
|
var keys = [];
|
|
|
|
var entry = document.createElement('div')
|
|
|
|
for(var k in resp) keys.push(k);
|
2019-02-28 03:02:44 +00:00
|
|
|
if (keys.length == 0){
|
|
|
|
threadPart.innerHTML = "nothing to show here yet."
|
|
|
|
}
|
2019-04-09 17:30:54 +00:00
|
|
|
for (var i = 0; i < keys.length; i++) (function(i, resp){
|
2019-02-09 06:32:11 +00:00
|
|
|
var entry = document.createElement('div')
|
2019-03-05 03:16:33 +00:00
|
|
|
var obj = resp[i]
|
2019-02-09 06:32:11 +00:00
|
|
|
var toLabel = document.createElement('span')
|
|
|
|
toLabel.innerText = 'To: '
|
|
|
|
var toEl = document.createElement('input')
|
2019-03-11 05:10:37 +00:00
|
|
|
var sentDate = document.createElement('span')
|
|
|
|
var humanDate = new Date(0)
|
|
|
|
humanDate.setUTCSeconds(resp[i]['date'])
|
2019-02-09 06:32:11 +00:00
|
|
|
var preview = document.createElement('span')
|
2019-03-03 06:26:55 +00:00
|
|
|
var deleteBtn = document.createElement('button')
|
2019-03-05 03:16:33 +00:00
|
|
|
var message = resp[i]['message']
|
2019-03-03 06:26:55 +00:00
|
|
|
deleteBtn.classList.add('deleteBtn', 'dangerBtn')
|
|
|
|
deleteBtn.innerText = 'X'
|
2019-02-09 06:32:11 +00:00
|
|
|
toEl.readOnly = true
|
2019-03-11 05:10:37 +00:00
|
|
|
sentDate.innerText = humanDate
|
2019-03-05 03:16:33 +00:00
|
|
|
if (resp[i]['name'] == null){
|
|
|
|
toEl.value = resp[i]['peer']
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
toEl.value = resp[i]['name']
|
|
|
|
}
|
2019-03-03 06:26:55 +00:00
|
|
|
preview.innerText = '(' + resp[i]['subject'] + ')'
|
|
|
|
entry.setAttribute('data-hash', resp[i]['hash'])
|
|
|
|
entry.appendChild(deleteBtn)
|
2019-02-09 06:32:11 +00:00
|
|
|
entry.appendChild(toLabel)
|
|
|
|
entry.appendChild(toEl)
|
|
|
|
entry.appendChild(preview)
|
2019-03-11 05:10:37 +00:00
|
|
|
entry.appendChild(sentDate)
|
2019-04-09 17:30:54 +00:00
|
|
|
|
|
|
|
threadPart.appendChild(entry)
|
|
|
|
|
|
|
|
entry.onclick = function(e){
|
|
|
|
if (e.target.classList.contains('deleteBtn')){
|
|
|
|
deleteMessage(e.target.parentNode.getAttribute('data-hash'))
|
|
|
|
e.target.parentNode.parentNode.removeChild(e.target.parentNode)
|
|
|
|
return
|
2019-03-03 06:26:55 +00:00
|
|
|
}
|
2019-04-09 17:30:54 +00:00
|
|
|
showSentboxWindow()
|
2019-02-09 06:32:11 +00:00
|
|
|
}
|
2019-04-09 17:30:54 +00:00
|
|
|
})(i, resp)
|
2019-02-09 06:32:11 +00:00
|
|
|
threadPart.appendChild(entry)
|
|
|
|
}.bind(threadPart))
|
|
|
|
}
|
|
|
|
|
|
|
|
function showSentboxWindow(to, content){
|
|
|
|
document.getElementById('toID').value = to
|
|
|
|
document.getElementById('sentboxDisplayText').innerText = content
|
|
|
|
overlay('sentboxDisplay')
|
2019-02-01 06:38:12 +00:00
|
|
|
}
|
|
|
|
|
2019-03-03 06:26:55 +00:00
|
|
|
function refreshPms(){
|
2019-03-02 19:17:18 +00:00
|
|
|
fetch('/mail/getinbox', {
|
2019-02-01 06:38:12 +00:00
|
|
|
headers: {
|
|
|
|
"token": webpass
|
|
|
|
}})
|
|
|
|
.then((resp) => resp.text()) // Transform the data into json
|
|
|
|
.then(function(data) {
|
|
|
|
pms = data.split(',')
|
2019-03-04 19:03:35 +00:00
|
|
|
getInbox()
|
2019-02-01 06:38:12 +00:00
|
|
|
})
|
2019-03-03 06:26:55 +00:00
|
|
|
}
|
2019-02-01 06:38:12 +00:00
|
|
|
|
2019-02-04 00:31:03 +00:00
|
|
|
tabBtns.onclick = function(event){
|
|
|
|
var children = tabBtns.children
|
|
|
|
for (var i = 0; i < children.length; i++) {
|
|
|
|
var btn = children[i]
|
|
|
|
btn.classList.remove('activeTab')
|
|
|
|
}
|
|
|
|
event.target.classList.add('activeTab')
|
|
|
|
setActiveTab(event.target.innerText.toLowerCase())
|
|
|
|
}
|
|
|
|
|
|
|
|
var idStrings = document.getElementsByClassName('myPub')
|
|
|
|
for (var i = 0; i < idStrings.length; i++){
|
|
|
|
if (idStrings[i].tagName.toLowerCase() == 'input'){
|
2019-02-26 04:19:37 +00:00
|
|
|
idStrings[i].value = myPub
|
2019-02-04 00:31:03 +00:00
|
|
|
}
|
|
|
|
else{
|
2019-02-26 04:19:37 +00:00
|
|
|
idStrings[i].innerText = myPub
|
2019-02-04 00:31:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var i = 0; i < document.getElementsByClassName('refresh').length; i++){
|
|
|
|
document.getElementsByClassName('refresh')[i].style.float = 'right'
|
2019-02-05 06:29:06 +00:00
|
|
|
}
|
|
|
|
|
2019-02-26 04:19:37 +00:00
|
|
|
fetch('/friends/list', {
|
|
|
|
headers: {
|
|
|
|
"token": webpass
|
|
|
|
}})
|
|
|
|
.then((resp) => resp.json()) // Transform the data into json
|
|
|
|
.then(function(resp) {
|
|
|
|
var friendSelectParent = document.getElementById('friendSelect')
|
|
|
|
var keys = [];
|
|
|
|
var friend
|
|
|
|
for(var k in resp) keys.push(k);
|
|
|
|
|
|
|
|
friendSelectParent.appendChild(document.createElement('option'))
|
|
|
|
for (var i = 0; i < keys.length; i++) {
|
|
|
|
var option = document.createElement("option")
|
|
|
|
var name = resp[keys[i]]['name']
|
|
|
|
option.value = keys[i]
|
|
|
|
if (name.length == 0){
|
|
|
|
option.text = keys[i]
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
option.text = name
|
|
|
|
}
|
|
|
|
friendSelectParent.appendChild(option)
|
|
|
|
}
|
|
|
|
|
|
|
|
for (var i = 0; i < keys.length; i++){
|
|
|
|
|
|
|
|
//friendSelectParent
|
|
|
|
//alert(resp[keys[i]]['name'])
|
|
|
|
}
|
2019-03-03 06:26:55 +00:00
|
|
|
})
|
2019-03-11 05:10:37 +00:00
|
|
|
setActiveTab('inbox')
|
|
|
|
|
|
|
|
setInterval(function(){mailPing()}, 10000)
|
|
|
|
mailPing()
|