157 lines
5.8 KiB
157 lines
5.8 KiB
"use strict";
|
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
});
|
|
};
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
const zbase32 = require("./zbase32");
|
|
const lightning_1 = require("./lightning");
|
|
const path = require("path");
|
|
const env = process.env.NODE_ENV || 'development';
|
|
const config = require(path.join(__dirname, '../../config/app.json'))[env];
|
|
/*
|
|
Lightning Data Access Token
|
|
Base64 strings separated by dots:
|
|
{host}.{muid}.{buyerPubKey}.{exp}.{metadata}.{signature}
|
|
|
|
- host: web host for data (ascii->base64)
|
|
- muid: ID of media
|
|
- buyerPubKey
|
|
- exp: unix timestamp expiration (encoded into 4 bytes)
|
|
- meta: key/value pairs, url query encoded (alphabetically ordered, ascii->base64)
|
|
- signature of all that (concatenated bytes of each)
|
|
*/
|
|
function tokenFromTerms({ host, muid, ttl, pubkey, meta }) {
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
const theHost = host || config.media_host || '';
|
|
const pubkeyBytes = Buffer.from(pubkey, 'hex');
|
|
const pubkey64 = urlBase64FromBytes(pubkeyBytes);
|
|
const now = Math.floor(Date.now() / 1000);
|
|
const exp = ttl ? now + (60 * 60 * 24 * 365) : 0;
|
|
const ldat = startLDAT(theHost, muid, pubkey64, exp, meta);
|
|
if (pubkey != '') {
|
|
const sig = yield lightning_1.signBuffer(ldat.bytes);
|
|
const sigBytes = zbase32.decode(sig);
|
|
return ldat.terms + "." + urlBase64FromBytes(sigBytes);
|
|
}
|
|
else {
|
|
return ldat.terms;
|
|
}
|
|
});
|
|
}
|
|
exports.tokenFromTerms = tokenFromTerms;
|
|
// host.muid.pk.exp.meta
|
|
function startLDAT(host, muid, pk, exp, meta = {}) {
|
|
const empty = Buffer.from([]);
|
|
var hostBuf = Buffer.from(host, 'ascii');
|
|
var muidBuf = Buffer.from(muid, 'base64');
|
|
var pkBuf = pk ? Buffer.from(pk, 'base64') : empty;
|
|
var expBuf = exp ? Buffer.from(exp.toString(16), 'hex') : empty;
|
|
var metaBuf = meta ? Buffer.from(serializeMeta(meta), 'ascii') : empty;
|
|
const totalLength = hostBuf.length + muidBuf.length + pkBuf.length + expBuf.length + metaBuf.length;
|
|
const buf = Buffer.concat([hostBuf, muidBuf, pkBuf, expBuf, metaBuf], totalLength);
|
|
let terms = `${urlBase64(hostBuf)}.${urlBase64(muidBuf)}.${urlBase64(pkBuf)}.${urlBase64(expBuf)}.${urlBase64(metaBuf)}`;
|
|
return { terms, bytes: buf };
|
|
}
|
|
exports.startLDAT = startLDAT;
|
|
const termKeys = [{
|
|
key: 'host',
|
|
func: buf => buf.toString('ascii')
|
|
}, {
|
|
key: 'muid',
|
|
func: buf => urlBase64(buf)
|
|
}, {
|
|
key: 'pubkey',
|
|
func: buf => buf.toString('hex')
|
|
}, {
|
|
key: 'ts',
|
|
func: buf => parseInt('0x' + buf.toString('hex'))
|
|
}, {
|
|
key: 'meta',
|
|
func: buf => {
|
|
const ascii = buf.toString('ascii');
|
|
return ascii ? deserializeMeta(ascii) : {}; // parse this
|
|
}
|
|
}, {
|
|
key: 'sig',
|
|
func: buf => urlBase64(buf)
|
|
}];
|
|
function parseLDAT(ldat) {
|
|
const a = ldat.split('.');
|
|
const o = {};
|
|
termKeys.forEach((t, i) => {
|
|
if (a[i])
|
|
o[t.key] = t.func(Buffer.from(a[i], 'base64'));
|
|
});
|
|
return o;
|
|
}
|
|
exports.parseLDAT = parseLDAT;
|
|
function testLDAT() {
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
console.log('testLDAT');
|
|
const terms = {
|
|
host: '',
|
|
ttl: 31536000,
|
|
muid: 'qFSOa50yWeGSG8oelsMvctLYdejPRD090dsypBSx_xg=',
|
|
pubkey: '0373ca36a331d8fd847f190908715a34997b15dc3c5d560ca032cf3412fcf494e4',
|
|
meta: {
|
|
amt: 100,
|
|
ttl: 31536000,
|
|
dim: '1500x1300'
|
|
}
|
|
};
|
|
const token = yield tokenFromTerms(terms);
|
|
console.log(token);
|
|
const terms2 = {
|
|
host: '',
|
|
ttl: 0,
|
|
muid: 'qFSOa50yWeGSG8oelsMvctLYdejPRD090dsypBSx_xg=',
|
|
pubkey: '',
|
|
meta: {
|
|
amt: 100,
|
|
ttl: 31536000,
|
|
}
|
|
};
|
|
const token2 = yield tokenFromTerms(terms2);
|
|
console.log(token2);
|
|
console.log(parseLDAT(token2));
|
|
});
|
|
}
|
|
exports.testLDAT = testLDAT;
|
|
function serializeMeta(obj) {
|
|
var str = [];
|
|
for (var p in obj) {
|
|
if (obj.hasOwnProperty(p)) {
|
|
str.push(encodeURIComponent(p) + "=" + encodeURIComponent(obj[p]));
|
|
}
|
|
}
|
|
str.sort((a, b) => (a > b ? 1 : -1));
|
|
return str.join("&");
|
|
}
|
|
function deserializeMeta(str) {
|
|
const json = str && str.length > 2 ? JSON.parse('{"' + str.replace(/&/g, '","').replace(/=/g, '":"') + '"}', function (key, value) { return key === "" ? value : decodeURIComponent(value); }) : {};
|
|
const ret = {};
|
|
for (let [k, v] of Object.entries(json)) {
|
|
const value = (typeof v === 'string' && parseInt(v)) || v;
|
|
ret[k] = value;
|
|
}
|
|
return ret;
|
|
}
|
|
function urlBase64(buf) {
|
|
return buf.toString('base64').replace(/\//g, '_').replace(/\+/g, '-');
|
|
}
|
|
exports.urlBase64 = urlBase64;
|
|
function urlBase64FromBytes(buf) {
|
|
return Buffer.from(buf).toString('base64').replace(/\//g, '_').replace(/\+/g, '-');
|
|
}
|
|
exports.urlBase64FromBytes = urlBase64FromBytes;
|
|
function urlBase64FromAscii(ascii) {
|
|
return Buffer.from(ascii, 'ascii').toString('base64').replace(/\//g, '_').replace(/\+/g, '-');
|
|
}
|
|
exports.urlBase64FromAscii = urlBase64FromAscii;
|
|
//# sourceMappingURL=ldat.js.map
|