forked from Mirrors/keyoxide-web
Improve profile page
This commit is contained in:
parent
d763ea1f27
commit
6847d2a285
3 changed files with 141 additions and 51 deletions
|
@ -172,8 +172,7 @@ async function displayProfile(opts) {
|
||||||
let userData = keyData.user.user.userId;
|
let userData = keyData.user.user.userId;
|
||||||
let feedback = "", notation, isVerified, verifications = [];
|
let feedback = "", notation, isVerified, verifications = [];
|
||||||
|
|
||||||
document.body.querySelector('#profile--name').innerHTML = userData.name;
|
document.body.querySelector('#profileName').innerHTML = userData.name;
|
||||||
document.body.querySelector('#profile--email').innerHTML = userData.email;
|
|
||||||
|
|
||||||
for (var i = 0; i < keyData.notations.length; i++) {
|
for (var i = 0; i < keyData.notations.length; i++) {
|
||||||
notation = keyData.notations[i];
|
notation = keyData.notations[i];
|
||||||
|
@ -182,27 +181,55 @@ async function displayProfile(opts) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate feedback
|
// Generate feedback
|
||||||
|
feedback += `<div class="profileDataItem profileDataItem--separator">`;
|
||||||
|
feedback += `<div class="profileDataItem__label"></div>`;
|
||||||
|
feedback += `<div class="profileDataItem__value">general information</div>`;
|
||||||
|
feedback += `</div>`;
|
||||||
|
feedback += `<div class="profileDataItem">`;
|
||||||
|
feedback += `<div class="profileDataItem__label">email</div>`;
|
||||||
|
feedback += `<div class="profileDataItem__value"><a href="mailto:${userData.email}">${userData.email}</a></div>`;
|
||||||
|
feedback += `</div>`;
|
||||||
|
feedback += `<div class="profileDataItem">`;
|
||||||
|
feedback += `<div class="profileDataItem__label">fingerprint</div>`;
|
||||||
|
feedback += `<div class="profileDataItem__value"><a href="https://keys.openpgp.org/pks/lookup?op=get&options=mr&search=0x${keyData.fingerprint}">${keyData.fingerprint}</a></div>`;
|
||||||
|
feedback += `</div>`;
|
||||||
|
|
||||||
|
feedback += `<div class="profileDataItem profileDataItem--separator">`;
|
||||||
|
feedback += `<div class="profileDataItem__label"></div>`;
|
||||||
|
feedback += `<div class="profileDataItem__value">proofs</div>`;
|
||||||
|
feedback += `</div>`;
|
||||||
for (var i = 0; i < verifications.length; i++) {
|
for (var i = 0; i < verifications.length; i++) {
|
||||||
feedback += `${verifications[i].type}: <a href="${verifications[i].url}">${verifications[i].display}</a>: ${verifications[i].isVerified}<br>`;
|
// feedback += `${verifications[i].type}: <a href="${verifications[i].url}">${verifications[i].display}</a>: ${verifications[i].isVerified}<br>`;
|
||||||
|
feedback += `<div class="profileDataItem">`;
|
||||||
|
feedback += `<div class="profileDataItem__label">${verifications[i].type}</div>`;
|
||||||
|
feedback += `<div class="profileDataItem__value">`;
|
||||||
|
feedback += `<a class="proofDisplay" href="${verifications[i].url}">${verifications[i].display}</a>`;
|
||||||
|
if (verifications[i].isVerified) {
|
||||||
|
feedback += `<a class="proofUrl proofUrl--verified" href="${verifications[i].proofUrl}">verified ✔</a>`;
|
||||||
|
} else {
|
||||||
|
feedback += `<a class="proofUrl" href="${verifications[i].proofUrl}">proof</a>`;
|
||||||
|
}
|
||||||
|
feedback += `</div>`;
|
||||||
|
feedback += `</div>`;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Display feedback
|
// Display feedback
|
||||||
document.body.querySelector('#profile--proofs').innerHTML = feedback;
|
document.body.querySelector('#profileData').innerHTML = feedback;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function verifyProof(url, fingerprint) {
|
async function verifyProof(url, fingerprint) {
|
||||||
// Init
|
// Init
|
||||||
let reVerify, urlFetch, output = {url: url, type: null, isVerified: false, display: null};
|
let reVerify, output = {url: url, type: null, proofUrl: url, proofUrlFetch: null, isVerified: false, display: null};
|
||||||
|
|
||||||
// DNS
|
// DNS
|
||||||
if (/^dns:/.test(url)) {
|
if (/^dns:/.test(url)) {
|
||||||
output.type = "dns";
|
output.type = "website";
|
||||||
let domain = url.replace(/dns:/, '').replace(/\?type=TXT/, '');
|
output.display = url.replace(/dns:/, '').replace(/\?type=TXT/, '');
|
||||||
urlFetch = `https://dns.google.com/resolve?name=${domain}&type=TXT`;
|
output.proofUrlFetch = `https://dns.google.com/resolve?name=${output.display}&type=TXT`;
|
||||||
output.display = domain;
|
output.url = `https://${output.display}`;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
response = await fetch(urlFetch, {
|
response = await fetch(output.proofUrlFetch, {
|
||||||
headers: {
|
headers: {
|
||||||
Accept: 'application/json'
|
Accept: 'application/json'
|
||||||
},
|
},
|
||||||
|
@ -226,8 +253,10 @@ async function verifyProof(url, fingerprint) {
|
||||||
// HN
|
// HN
|
||||||
if (/^https:\/\/news.ycombinator.com/.test(url)) {
|
if (/^https:\/\/news.ycombinator.com/.test(url)) {
|
||||||
output.type = "hn";
|
output.type = "hn";
|
||||||
|
output.display = url.replace(/https:\/\/news.ycombinator.com\/user\?id=/, "");
|
||||||
|
output.proofUrlFetch = `https://hacker-news.firebaseio.com/v0/user/${output.display}.json`;
|
||||||
try {
|
try {
|
||||||
response = await fetch(urlFetch, {
|
response = await fetch(output.proofUrlFetch, {
|
||||||
headers: {
|
headers: {
|
||||||
Accept: 'application/json'
|
Accept: 'application/json'
|
||||||
},
|
},
|
||||||
|
@ -238,11 +267,9 @@ async function verifyProof(url, fingerprint) {
|
||||||
}
|
}
|
||||||
json = await response.json();
|
json = await response.json();
|
||||||
reVerify = new RegExp(`openpgp4fpr:${fingerprint}`);
|
reVerify = new RegExp(`openpgp4fpr:${fingerprint}`);
|
||||||
json.Answer.forEach((item, i) => {
|
if (reVerify.test(json.about)) {
|
||||||
if (reVerify.test(item.data)) {
|
|
||||||
output.isVerified = true;
|
output.isVerified = true;
|
||||||
}
|
}
|
||||||
});
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
} finally {
|
} finally {
|
||||||
return output;
|
return output;
|
||||||
|
@ -251,8 +278,11 @@ async function verifyProof(url, fingerprint) {
|
||||||
// Reddit
|
// Reddit
|
||||||
if (/^https:\/\/www.reddit.com\/user/.test(url)) {
|
if (/^https:\/\/www.reddit.com\/user/.test(url)) {
|
||||||
output.type = "reddit";
|
output.type = "reddit";
|
||||||
|
output.display = url.replace(/^https:\/\/www.reddit.com\/user\//, "").replace(/\/comments\/.*/, "");
|
||||||
|
output.url = `https://www.reddit.com/user/${output.display}`;
|
||||||
|
output.proofUrlFetch = url.replace(/\/[a-zA-Z0-9_]*\/$/, ".json");
|
||||||
try {
|
try {
|
||||||
response = await fetch(urlFetch, {
|
response = await fetch(output.proofUrlFetch, {
|
||||||
headers: {
|
headers: {
|
||||||
Accept: 'application/json'
|
Accept: 'application/json'
|
||||||
},
|
},
|
||||||
|
@ -262,7 +292,8 @@ async function verifyProof(url, fingerprint) {
|
||||||
throw new Error('Response failed: ' + response.status);
|
throw new Error('Response failed: ' + response.status);
|
||||||
}
|
}
|
||||||
json = await response.json();
|
json = await response.json();
|
||||||
reVerify = new RegExp(`openpgp4fpr:${fingerprint}`);
|
console.log(json);
|
||||||
|
reVerify = new RegExp(`Verifying my OpenPGP key: openpgp4fpr:${fingerprint}`);
|
||||||
json.Answer.forEach((item, i) => {
|
json.Answer.forEach((item, i) => {
|
||||||
if (reVerify.test(item.data)) {
|
if (reVerify.test(item.data)) {
|
||||||
output.isVerified = true;
|
output.isVerified = true;
|
||||||
|
@ -276,8 +307,12 @@ async function verifyProof(url, fingerprint) {
|
||||||
// Github
|
// Github
|
||||||
if (/^https:\/\/gist.github.com/.test(url)) {
|
if (/^https:\/\/gist.github.com/.test(url)) {
|
||||||
output.type = "github";
|
output.type = "github";
|
||||||
|
output.display = url.replace(/^https:\/\/gist.github.com\//, "").replace(/\/[a-zA-Z0-9]*$/, "");
|
||||||
|
output.url = `https://github.com/${output.display}`;
|
||||||
|
let gistId = url.replace(/^https:\/\/gist.github.com\/[a-zA-Z0-9_-]*\//, "");
|
||||||
|
output.proofUrlFetch = `https://api.github.com/gists/${gistId}`;
|
||||||
try {
|
try {
|
||||||
response = await fetch(urlFetch, {
|
response = await fetch(output.proofUrlFetch, {
|
||||||
headers: {
|
headers: {
|
||||||
Accept: 'application/json'
|
Accept: 'application/json'
|
||||||
},
|
},
|
||||||
|
@ -287,12 +322,10 @@ async function verifyProof(url, fingerprint) {
|
||||||
throw new Error('Response failed: ' + response.status);
|
throw new Error('Response failed: ' + response.status);
|
||||||
}
|
}
|
||||||
json = await response.json();
|
json = await response.json();
|
||||||
reVerify = new RegExp(`openpgp4fpr:${fingerprint}`);
|
reVerify = new RegExp(`[Verifying my OpenPGP key: openpgp4fpr:${fingerprint}]`);
|
||||||
json.Answer.forEach((item, i) => {
|
if (reVerify.test(json.files["openpgp.md"].content)) {
|
||||||
if (reVerify.test(item.data)) {
|
|
||||||
output.isVerified = true;
|
output.isVerified = true;
|
||||||
}
|
}
|
||||||
});
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
} finally {
|
} finally {
|
||||||
return output;
|
return output;
|
||||||
|
@ -316,6 +349,7 @@ async function verifyProof(url, fingerprint) {
|
||||||
if (item.value === fingerprint) {
|
if (item.value === fingerprint) {
|
||||||
output.type = "mastodon";
|
output.type = "mastodon";
|
||||||
output.display = json.url;
|
output.display = json.url;
|
||||||
|
output.proofUrlFetch = json.url;
|
||||||
output.isVerified = true;
|
output.isVerified = true;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
|
@ -20,7 +20,7 @@ nav a {
|
||||||
footer {
|
footer {
|
||||||
color: #777;
|
color: #777;
|
||||||
margin: 64px 0;
|
margin: 64px 0;
|
||||||
padding: 32px;
|
padding: 0 32px;
|
||||||
}
|
}
|
||||||
.container {
|
.container {
|
||||||
max-width: 720px;
|
max-width: 720px;
|
||||||
|
@ -91,3 +91,78 @@ input[type="submit"] {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
margin: 0 0 8px;
|
margin: 0 0 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.container--profile {
|
||||||
|
margin-top: 64px;
|
||||||
|
}
|
||||||
|
.container--profile .content {
|
||||||
|
padding-top: 32px;
|
||||||
|
font-size: 1.2em;
|
||||||
|
}
|
||||||
|
.container--profile footer {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.profileDataItem {
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
.profileDataItem--separator {
|
||||||
|
margin-top: 1em;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
.profileDataItem__label {
|
||||||
|
display: inline-block;
|
||||||
|
position: relative;
|
||||||
|
flex: 1;
|
||||||
|
min-height: 32px;
|
||||||
|
padding: 0 8px;
|
||||||
|
max-width: 20%;
|
||||||
|
color: #777;
|
||||||
|
text-align: right;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
.profileDataItem__value {
|
||||||
|
display: inline-block;
|
||||||
|
flex: 1;
|
||||||
|
min-height: 32px;
|
||||||
|
max-width: 100%;
|
||||||
|
padding: 0 8px;
|
||||||
|
}
|
||||||
|
.profileDataItem__value a {
|
||||||
|
display: inline-block;
|
||||||
|
max-width: 100%;
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
.profileDataItem__value a.proofDisplay {
|
||||||
|
margin-right: 16px;
|
||||||
|
}
|
||||||
|
.profileDataItem__value a.proofUrl {
|
||||||
|
color: #777;
|
||||||
|
/* text-decoration: none; */
|
||||||
|
}
|
||||||
|
.profileDataItem__value a.proofUrl.proofUrl--verified {
|
||||||
|
color: #499539;
|
||||||
|
}
|
||||||
|
.profileDataItem__value a.proofUrl:hover {
|
||||||
|
text-decoration: underline;
|
||||||
|
}
|
||||||
|
@media (max-width: 680px) {
|
||||||
|
.profileDataItem {
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
.profileDataItem__label {
|
||||||
|
max-width: 100%;
|
||||||
|
min-height: 28px;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
.profileDataItem__value {
|
||||||
|
min-height: 28px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -3,41 +3,22 @@
|
||||||
<head>
|
<head>
|
||||||
<meta charset="utf-8">
|
<meta charset="utf-8">
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
<title>Encrypt - Keyoxide</title>
|
<title>Keyoxide</title>
|
||||||
<script async defer data-domain="keyoxide.org" src="https://plausible.io/js/plausible.js"></script>
|
<script async defer data-domain="keyoxide.org" src="https://plausible.io/js/plausible.js"></script>
|
||||||
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/normalize.css@8.0.1/normalize.css">
|
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/normalize.css@8.0.1/normalize.css">
|
||||||
<link rel="stylesheet" href="/assets/styles.css">
|
<link rel="stylesheet" href="/assets/styles.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<header>
|
<div class="container container--profile">
|
||||||
<div class="container">
|
<h1 id="profileName"></h1>
|
||||||
<a href="/">Keyoxide</a>
|
|
||||||
</div>
|
|
||||||
</header>
|
|
||||||
|
|
||||||
<div class="container">
|
|
||||||
<h1>Profile</h1>
|
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<h3></h3>
|
<span id="profileUid" style="display: none;">%UID%</span>
|
||||||
<p>Input: <span id="profileUid">%UID%</span></p>
|
<div id="profileData"></div>
|
||||||
<p>Name: <span id="profile--name"></span></p>
|
|
||||||
<p>Email address: <span id="profile--email">/span></p>
|
|
||||||
<p>Proofs: <span id="profile--proofs">/span></p>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<footer>
|
<footer>
|
||||||
<p>
|
<p>
|
||||||
<a href="/">Keyoxide</a> makes basic cryptography operations accessible to regular humans.
|
Page generated by <a href="/">Keyoxide</a>.
|
||||||
<br>
|
|
||||||
Made by <a href="https://yarmo.eu">Yarmo Mackenbach</a>.
|
|
||||||
<br>
|
|
||||||
Code hosted on <a href="https://codeberg.org/yarmo/keyoxide">Codeberg</a> (<a href="https://drone.private.foss.best/yarmo/keyoxide/">drone CI/CD</a>).
|
|
||||||
<br>
|
|
||||||
Uses <a href="https://github.com/openpgpjs/openpgpjs">openpgp.js</a> (version <a href="https://github.com/openpgpjs/openpgpjs/releases/tag/v4.10.4">4.10.4</a>).
|
|
||||||
<br>
|
|
||||||
Privacy-friendly public usage stats by <a href="https://plausible.io/keyoxide.org">Plausible.io</a>.
|
|
||||||
<br>
|
|
||||||
Because SO2 + 2NaOH → Na2SO3 + H2O
|
|
||||||
</p>
|
</p>
|
||||||
</footer>
|
</footer>
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Reference in a new issue