diff --git a/example/public/index.html b/example/public/index.html new file mode 100644 index 0000000..131e9bc --- /dev/null +++ b/example/public/index.html @@ -0,0 +1,145 @@ + + + + Pocket ID Example + + +
+
+
+
+
+
+
+
+ + + + diff --git a/example/server.js b/example/server.js new file mode 100644 index 0000000..10d6084 --- /dev/null +++ b/example/server.js @@ -0,0 +1,49 @@ +"use strict"; + +import path from "path"; +import http from "http"; +import express from "express"; + +//import PocketId from "pocketid"; +import PocketId from "../index.js"; + +var app = express(); +var dirname = path.dirname(import.meta.url.slice("file:".length)); +var pubdir = path.join(dirname, "public"); + +var requirePocket = PocketId.express({ issuers: ["beta.pocketid.app", "pocketid..app"] }); +app.use("/api/auth", requirePocket); +app.get("/api/auth/info", function (req, res) { + res.json({ user: req.user }); +}); + +var checkPocket = PocketId.express({ + issuers: ["beta.pocketid.app", "pocketid..app"], + allow: true, +}); +app.use("/api/public", checkPocket); +app.get("/api/public/info", function (req, res) { + res.json({ user: req.user }); +}); + +var refresher = PocketId.refreshToken({ + issuer: "https://localhost:3000", + issuers: ["beta.pocketid.app", "pocketid..app"], +}); +app.use("/api/refresh-token", requirePocket, refresher); + +var requireAuth = PocketId.auth({ + issuers: ["localhost:3000"], +}); +app.use("/api/user", requireAuth); +app.get("/api/user/info", function (req, res) { + res.json({ user: req.user }); +}); + +app.use("/", express.static(pubdir)); + +var server = http.createServer(app); +var port = process.env.PORT || 3000; +server.listen(port, function () { + console.info(`Serving '${pubdir}' listening on`, server.address()); +}); diff --git a/index.js b/index.js index 0402ff6..e90f647 100644 --- a/index.js +++ b/index.js @@ -1,4 +1,4 @@ "use strict"; -import PocketId from "./lib/"; +import PocketId from "./lib/index.js"; export default PocketId; diff --git a/lib/index.js b/lib/index.js index b6bec93..d2c2a34 100644 --- a/lib/index.js +++ b/lib/index.js @@ -1,7 +1,249 @@ "use strict"; import Keypairs from "@root/keypairs"; +import Keyfetch from "keyfetch"; +import Request from "@root/request"; +import crypto from "crypto"; -const PocketId = {}; +var PocketId = {}; + +var submails = {}; +var rootKeypair; + +var _promise = Promise.resolve(); +PocketId._keypair = async function (opts) { + _promise = _promise.then(async function () { + var key = opts.key || process.env.PRIVATE_KEY || (rootKeypair && rootKeypair.private); + var keypair = await Keypairs.parseOrGenerate({ key: key }); + if (!key) { + console.warn("Generated random keypair:"); + console.warn(keypair.public); + } + if (!rootKeypair) { + rootKeypair = keypair; + } + return keypair; + }); + return _promise; +}; + +// TODO remove regexp one day (for performance) + +// { issuers, allow } +PocketId.express = function (opts) { + if (!opts) { + opts = {}; + } + + if (!opts.issuers) { + opts.issuers = ["beta.pocketid.app", "pocketid.app"]; + } + + async function _middleware(req, res, next) { + var [bearer, token] = (req.headers.authorization || "").split(" "); + req.jwt = token; + var decoded = await Keyfetch.jwt.verify(token, opts).catch(function (err) { + if (opts.allow) { + next(); + return; + } + + res.statusCode = 403; + res.json({ + error: "invalid authorization: " + err.message, + code: "E_AUTH", + }); + }); + if (!decoded) { + // error has been handled + return; + } + + if (!decoded.claims.jti || !decoded.claims.sub || !decoded.claims.iss) { + res.statusCode = 400; + res.json({ + error: "invalid token: missing required claims", + error_code: "E_AUTH", + }); + return; + } + + var iss = decoded.claims.iss || ""; + req.jws = decoded; + req.user = { + jti: decoded.claims.jti, + ppid: decoded.claims.sub + "@" + iss, + }; + req.user.sid = req.user.ppid + "#" + req.user.jti; + + var scheme = "https:"; + var m = iss.match(/(https?:\/\/)?localhost(:|\/|$)/); + if (m) { + scheme = m[1] || "http:"; + } + req.user.iss = scheme + "//" + iss.replace(/https?:\/\//, ""); + + await addEmail(req).catch(function (err) { + // what to do with this error? + console.warn("[pocketid.express.request-email] Error:", err); + }); + next(); + } + + return _middleware; +}; + +async function addEmail(req) { + if (submails[req.user.sid]) { + // TODO keep email cached longer than token expiration? + Object.assign(req.user, submails[req.user.sid]); + return; + } + + var resp = await Request({ + url: req.user.iss + "/api/authz/email", + headers: { + Authorization: "Bearer " + req.jwt, + }, + }); + var result = (resp.body && resp.body.result) || resp.body || {}; + console.log("authz-email:", result); + if (result.email && result.verified_at) { + submails[req.user.sid] = { + email: result.email, + verifiedAt: result.verified_at, + createdAt: new Date().toISOString(), + }; + } + Object.assign(req.user, submails[req.user.jit] || {}); + console.log("authz-user:", req.user); +} + +PocketId.refreshToken = function (opts) { + var keypair; + (async function () { + keypair = await PocketId._keypair(opts); + })(); + + if (!opts.getClaims) { + opts.getClaims = async function (req) { + return {}; + }; + } + + return async function _refreshToken(req, res) { + if (!req.user || !req.user.sid) { + res.statusCode = 400; + res.json({ + error: "invalid authorization: no user found", + code: "E_AUTH", + }); + return; + } + + var claims = await opts.getClaims(req).catch(function (err) { + res.statusCode = 500; + res.json({ + error: "invalid claims: " + err.message, + code: "E_CLAIMS", + }); + return; + }); + if (!claims) { + return; + } + + var exp = claims.exp || "15m"; + + if (!claims.sub) { + claims.sub = req.user.email; + } + if (!claims.jti) { + claims.jti = crypto + .randomBytes(16) + .toString("base64") + .replace(/\//g, "_") + .replace(/\+/g, "-") + .replace(/=/g, ""); + } + var iss = opts.issuer || opts.iss; + if (!iss) { + iss = (req.connection.encrypted ? "https:" : "http:") + "//" + req.headers.host; + } + var jwt = await Keypairs.signJwt({ + jwk: keypair.private, + iss: iss, + exp: exp, + claims: claims, + }); + res.json({ + success: true, + result: { + access_token: jwt, + }, + }); + }; +}; + +PocketId.auth = function (opts) { + var pub = opts.pub || opts.jwk; + var keypair; + (async function () { + if (!pub) { + keypair = await PocketId._keypair(opts); + } else { + keypair = { public: pub }; + } + })(); + + if (!opts.getUser) { + opts.getUser = async function (req) { + return { _test: true }; + }; + } + + return async function _auth(req, res, next) { + var [bearer, token] = (req.headers.authorization || "").split(" "); + req.jwt = token; + var _opts = Object.assign({}, opts, { jwk: keypair.public }); + var decoded = await Keyfetch.jwt.verify(token, _opts).catch(function (err) { + if (opts.allow) { + next(); + return; + } + + res.statusCode = 403; + res.json({ + error: "invalid authorization: " + err.message, + code: "E_AUTH", + }); + }); + if (!decoded) { + // error has been handled + return; + } + + req.jws = decoded; + req.user = { + id: decoded.claims.sub, + }; + + var user = await opts.getUser(req).catch(function (err) { + res.statusCode = 500; + res.json({ + error: "invalid user: " + err.message, + code: "E_USER", + }); + return; + }); + if (!user) { + // error has been handled + return; + } + + console.log("user:", Object.assign(req.user, user)); + next(); + }; +}; export default PocketId; diff --git a/package-lock.json b/package-lock.json index dfab705..1497e11 100644 --- a/package-lock.json +++ b/package-lock.json @@ -4,6 +4,11 @@ "lockfileVersion": 1, "requires": true, "dependencies": { + "@coolaj86/urequest": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/@coolaj86/urequest/-/urequest-1.3.7.tgz", + "integrity": "sha512-PPrVYra9aWvZjSCKl/x1pJ9ZpXda1652oJrPBYy5rQumJJMkmTBN3ux+sK2xAUwVvv2wnewDlaQaHLxLwSHnIA==" + }, "@root/asn1": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/@root/asn1/-/asn1-1.0.0.tgz", @@ -32,6 +37,11 @@ "resolved": "https://registry.npmjs.org/@root/pem/-/pem-1.0.4.tgz", "integrity": "sha512-rEUDiUsHtild8GfIjFE9wXtcVxeS+ehCJQBwbQQ3IVfORKHK93CFnRtkr69R75lZFjcmKYVc+AXDB+AeRFOULA==" }, + "@root/request": { + "version": "1.6.1", + "resolved": "https://registry.npmjs.org/@root/request/-/request-1.6.1.tgz", + "integrity": "sha512-8wrWyeBLRp7T8J36GkT3RODJ6zYmL0/maWlAUD5LOXT28D3TDquUepyYDKYANNA3Gc8R5ZCgf+AXvSTYpJEWwQ==" + }, "@root/x509": { "version": "0.7.2", "resolved": "https://registry.npmjs.org/@root/x509/-/x509-0.7.2.tgz", @@ -41,10 +51,396 @@ "@root/encoding": "^1.0.1" } }, + "accepts": { + "version": "1.3.7", + "resolved": "https://registry.npmjs.org/accepts/-/accepts-1.3.7.tgz", + "integrity": "sha512-Il80Qs2WjYlJIBNzNkK6KYqlVMTbZLXgHx2oT0pU/fjRHyEp+PEfEPY0R3WCwAGVOtauxh1hOxNgIf5bv7dQpA==", + "requires": { + "mime-types": "~2.1.24", + "negotiator": "0.6.2" + } + }, + "array-flatten": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/array-flatten/-/array-flatten-1.1.1.tgz", + "integrity": "sha1-ml9pkFGx5wczKPKgCJaLZOopVdI=" + }, "basetag": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/basetag/-/basetag-1.0.0.tgz", "integrity": "sha512-U9BrHuwNyBzGBA5FGYjD9TdSvvFGDmviMYKHFxY2saT7QZF6Xp/Kv6yxATp1RuEKsd1CBMcpEyl3KmikMUjwqA==" + }, + "body-parser": { + "version": "1.19.0", + "resolved": "https://registry.npmjs.org/body-parser/-/body-parser-1.19.0.tgz", + "integrity": "sha512-dhEPs72UPbDnAQJ9ZKMNTP6ptJaionhP5cBb541nXPlW60Jepo9RV/a4fX4XWW9CuFNK22krhrj1+rgzifNCsw==", + "requires": { + "bytes": "3.1.0", + "content-type": "~1.0.4", + "debug": "2.6.9", + "depd": "~1.1.2", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "on-finished": "~2.3.0", + "qs": "6.7.0", + "raw-body": "2.4.0", + "type-is": "~1.6.17" + } + }, + "bytes": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/bytes/-/bytes-3.1.0.tgz", + "integrity": "sha512-zauLjrfCG+xvoyaqLoV8bLVXXNGC4JqlxFCutSDWA6fJrTo2ZuvLYTqZ7aHBLZSMOopbzwv8f+wZcVzfVTI2Dg==" + }, + "content-disposition": { + "version": "0.5.3", + "resolved": "https://registry.npmjs.org/content-disposition/-/content-disposition-0.5.3.tgz", + "integrity": "sha512-ExO0774ikEObIAEV9kDo50o+79VCUdEB6n6lzKgGwupcVeRlhrj3qGAfwq8G6uBJjkqLrhT0qEYFcWng8z1z0g==", + "requires": { + "safe-buffer": "5.1.2" + } + }, + "content-type": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/content-type/-/content-type-1.0.4.tgz", + "integrity": "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA==" + }, + "cookie": { + "version": "0.4.0", + "resolved": "https://registry.npmjs.org/cookie/-/cookie-0.4.0.tgz", + "integrity": "sha512-+Hp8fLp57wnUSt0tY0tHEXh4voZRDnoIrZPqlo3DPiI4y9lwg/jqx+1Om94/W6ZaPDOUbnjOt/99w66zk+l1Xg==" + }, + "cookie-signature": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/cookie-signature/-/cookie-signature-1.0.6.tgz", + "integrity": "sha1-4wOogrNCzD7oylE6eZmXNNqzriw=" + }, + "debug": { + "version": "2.6.9", + "resolved": "https://registry.npmjs.org/debug/-/debug-2.6.9.tgz", + "integrity": "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA==", + "requires": { + "ms": "2.0.0" + } + }, + "depd": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/depd/-/depd-1.1.2.tgz", + "integrity": "sha1-m81S4UwJd2PnSbJ0xDRu0uVgtak=" + }, + "destroy": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/destroy/-/destroy-1.0.4.tgz", + "integrity": "sha1-l4hXRCxEdJ5CBmE+N5RiBYJqvYA=" + }, + "eckles": { + "version": "1.4.1", + "resolved": "https://registry.npmjs.org/eckles/-/eckles-1.4.1.tgz", + "integrity": "sha512-auWyk/k8oSkVHaD4RxkPadKsLUcIwKgr/h8F7UZEueFDBO7BsE4y+H6IMUDbfqKIFPg/9MxV6KcBdJCmVVcxSA==" + }, + "ee-first": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/ee-first/-/ee-first-1.1.1.tgz", + "integrity": "sha1-WQxhFWsK4vTwJVcyoViyZrxWsh0=" + }, + "encodeurl": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/encodeurl/-/encodeurl-1.0.2.tgz", + "integrity": "sha1-rT/0yG7C0CkyL1oCw6mmBslbP1k=" + }, + "escape-html": { + "version": "1.0.3", + "resolved": "https://registry.npmjs.org/escape-html/-/escape-html-1.0.3.tgz", + "integrity": "sha1-Aljq5NPQwJdN4cFpGI7wBR0dGYg=" + }, + "etag": { + "version": "1.8.1", + "resolved": "https://registry.npmjs.org/etag/-/etag-1.8.1.tgz", + "integrity": "sha1-Qa4u62XvpiJorr/qg6x9eSmbCIc=" + }, + "express": { + "version": "4.17.1", + "resolved": "https://registry.npmjs.org/express/-/express-4.17.1.tgz", + "integrity": "sha512-mHJ9O79RqluphRrcw2X/GTh3k9tVv8YcoyY4Kkh4WDMUYKRZUq0h1o0w2rrrxBqM7VoeUVqgb27xlEMXTnYt4g==", + "requires": { + "accepts": "~1.3.7", + "array-flatten": "1.1.1", + "body-parser": "1.19.0", + "content-disposition": "0.5.3", + "content-type": "~1.0.4", + "cookie": "0.4.0", + "cookie-signature": "1.0.6", + "debug": "2.6.9", + "depd": "~1.1.2", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "finalhandler": "~1.1.2", + "fresh": "0.5.2", + "merge-descriptors": "1.0.1", + "methods": "~1.1.2", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "path-to-regexp": "0.1.7", + "proxy-addr": "~2.0.5", + "qs": "6.7.0", + "range-parser": "~1.2.1", + "safe-buffer": "5.1.2", + "send": "0.17.1", + "serve-static": "1.14.1", + "setprototypeof": "1.1.1", + "statuses": "~1.5.0", + "type-is": "~1.6.18", + "utils-merge": "1.0.1", + "vary": "~1.1.2" + } + }, + "finalhandler": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/finalhandler/-/finalhandler-1.1.2.tgz", + "integrity": "sha512-aAWcW57uxVNrQZqFXjITpW3sIUQmHGG3qSb9mUah9MgMC4NeWhNOlNjXEYq3HjRAvL6arUviZGGJsBg6z0zsWA==", + "requires": { + "debug": "2.6.9", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "on-finished": "~2.3.0", + "parseurl": "~1.3.3", + "statuses": "~1.5.0", + "unpipe": "~1.0.0" + } + }, + "forwarded": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/forwarded/-/forwarded-0.1.2.tgz", + "integrity": "sha1-mMI9qxF1ZXuMBXPozszZGw/xjIQ=" + }, + "fresh": { + "version": "0.5.2", + "resolved": "https://registry.npmjs.org/fresh/-/fresh-0.5.2.tgz", + "integrity": "sha1-PYyt2Q2XZWn6g1qx+OSyOhBWBac=" + }, + "http-errors": { + "version": "1.7.2", + "resolved": "https://registry.npmjs.org/http-errors/-/http-errors-1.7.2.tgz", + "integrity": "sha512-uUQBt3H/cSIVfch6i1EuPNy/YsRSOUBXTVfZ+yR7Zjez3qjBz6i9+i4zjNaoqcoFVI4lQJ5plg63TvGfRSDCRg==", + "requires": { + "depd": "~1.1.2", + "inherits": "2.0.3", + "setprototypeof": "1.1.1", + "statuses": ">= 1.5.0 < 2", + "toidentifier": "1.0.0" + } + }, + "iconv-lite": { + "version": "0.4.24", + "resolved": "https://registry.npmjs.org/iconv-lite/-/iconv-lite-0.4.24.tgz", + "integrity": "sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA==", + "requires": { + "safer-buffer": ">= 2.1.2 < 3" + } + }, + "inherits": { + "version": "2.0.3", + "resolved": "https://registry.npmjs.org/inherits/-/inherits-2.0.3.tgz", + "integrity": "sha1-Yzwsg+PaQqUC9SRmAiSA9CCCYd4=" + }, + "ipaddr.js": { + "version": "1.9.1", + "resolved": "https://registry.npmjs.org/ipaddr.js/-/ipaddr.js-1.9.1.tgz", + "integrity": "sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g==" + }, + "keyfetch": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/keyfetch/-/keyfetch-1.2.1.tgz", + "integrity": "sha512-Zph49SMJjBiPHeut4io1xfYwgW/lM3fwHJRJ4eA3JTsIigyJUcMwZ/sMxwkAMARB45YHG2eGTqeC7JCO6iUm3A==", + "requires": { + "@coolaj86/urequest": "^1.3.7", + "eckles": "^1.4.1", + "rasha": "^1.2.4" + } + }, + "media-typer": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/media-typer/-/media-typer-0.3.0.tgz", + "integrity": "sha1-hxDXrwqmJvj/+hzgAWhUUmMlV0g=" + }, + "merge-descriptors": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/merge-descriptors/-/merge-descriptors-1.0.1.tgz", + "integrity": "sha1-sAqqVW3YtEVoFQ7J0blT8/kMu2E=" + }, + "methods": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/methods/-/methods-1.1.2.tgz", + "integrity": "sha1-VSmk1nZUE07cxSZmVoNbD4Ua/O4=" + }, + "mime": { + "version": "1.6.0", + "resolved": "https://registry.npmjs.org/mime/-/mime-1.6.0.tgz", + "integrity": "sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg==" + }, + "mime-db": { + "version": "1.44.0", + "resolved": "https://registry.npmjs.org/mime-db/-/mime-db-1.44.0.tgz", + "integrity": "sha512-/NOTfLrsPBVeH7YtFPgsVWveuL+4SjjYxaQ1xtM1KMFj7HdxlBlxeyNLzhyJVx7r4rZGJAZ/6lkKCitSc/Nmpg==" + }, + "mime-types": { + "version": "2.1.27", + "resolved": "https://registry.npmjs.org/mime-types/-/mime-types-2.1.27.tgz", + "integrity": "sha512-JIhqnCasI9yD+SsmkquHBxTSEuZdQX5BuQnS2Vc7puQQQ+8yiP5AY5uWhpdv4YL4VM5c6iliiYWPgJ/nJQLp7w==", + "requires": { + "mime-db": "1.44.0" + } + }, + "ms": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.0.0.tgz", + "integrity": "sha1-VgiurfwAvmwpAd9fmGF4jeDVl8g=" + }, + "negotiator": { + "version": "0.6.2", + "resolved": "https://registry.npmjs.org/negotiator/-/negotiator-0.6.2.tgz", + "integrity": "sha512-hZXc7K2e+PgeI1eDBe/10Ard4ekbfrrqG8Ep+8Jmf4JID2bNg7NvCPOZN+kfF574pFQI7mum2AUqDidoKqcTOw==" + }, + "on-finished": { + "version": "2.3.0", + "resolved": "https://registry.npmjs.org/on-finished/-/on-finished-2.3.0.tgz", + "integrity": "sha1-IPEzZIGwg811M3mSoWlxqi2QaUc=", + "requires": { + "ee-first": "1.1.1" + } + }, + "parseurl": { + "version": "1.3.3", + "resolved": "https://registry.npmjs.org/parseurl/-/parseurl-1.3.3.tgz", + "integrity": "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ==" + }, + "path-to-regexp": { + "version": "0.1.7", + "resolved": "https://registry.npmjs.org/path-to-regexp/-/path-to-regexp-0.1.7.tgz", + "integrity": "sha1-32BBeABfUi8V60SQ5yR6G/qmf4w=" + }, + "proxy-addr": { + "version": "2.0.6", + "resolved": "https://registry.npmjs.org/proxy-addr/-/proxy-addr-2.0.6.tgz", + "integrity": "sha512-dh/frvCBVmSsDYzw6n926jv974gddhkFPfiN8hPOi30Wax25QZyZEGveluCgliBnqmuM+UJmBErbAUFIoDbjOw==", + "requires": { + "forwarded": "~0.1.2", + "ipaddr.js": "1.9.1" + } + }, + "qs": { + "version": "6.7.0", + "resolved": "https://registry.npmjs.org/qs/-/qs-6.7.0.tgz", + "integrity": "sha512-VCdBRNFTX1fyE7Nb6FYoURo/SPe62QCaAyzJvUjwRaIsc+NePBEniHlvxFmmX56+HZphIGtV0XeCirBtpDrTyQ==" + }, + "range-parser": { + "version": "1.2.1", + "resolved": "https://registry.npmjs.org/range-parser/-/range-parser-1.2.1.tgz", + "integrity": "sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg==" + }, + "rasha": { + "version": "1.2.5", + "resolved": "https://registry.npmjs.org/rasha/-/rasha-1.2.5.tgz", + "integrity": "sha512-KxtX+/fBk+wM7O3CNgwjSh5elwFilLvqWajhr6wFr2Hd63JnKTTi43Tw+Jb1hxJQWOwoya+NZWR2xztn3hCrTw==" + }, + "raw-body": { + "version": "2.4.0", + "resolved": "https://registry.npmjs.org/raw-body/-/raw-body-2.4.0.tgz", + "integrity": "sha512-4Oz8DUIwdvoa5qMJelxipzi/iJIi40O5cGV1wNYp5hvZP8ZN0T+jiNkL0QepXs+EsQ9XJ8ipEDoiH70ySUJP3Q==", + "requires": { + "bytes": "3.1.0", + "http-errors": "1.7.2", + "iconv-lite": "0.4.24", + "unpipe": "1.0.0" + } + }, + "safe-buffer": { + "version": "5.1.2", + "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.1.2.tgz", + "integrity": "sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g==" + }, + "safer-buffer": { + "version": "2.1.2", + "resolved": "https://registry.npmjs.org/safer-buffer/-/safer-buffer-2.1.2.tgz", + "integrity": "sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg==" + }, + "send": { + "version": "0.17.1", + "resolved": "https://registry.npmjs.org/send/-/send-0.17.1.tgz", + "integrity": "sha512-BsVKsiGcQMFwT8UxypobUKyv7irCNRHk1T0G680vk88yf6LBByGcZJOTJCrTP2xVN6yI+XjPJcNuE3V4fT9sAg==", + "requires": { + "debug": "2.6.9", + "depd": "~1.1.2", + "destroy": "~1.0.4", + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "etag": "~1.8.1", + "fresh": "0.5.2", + "http-errors": "~1.7.2", + "mime": "1.6.0", + "ms": "2.1.1", + "on-finished": "~2.3.0", + "range-parser": "~1.2.1", + "statuses": "~1.5.0" + }, + "dependencies": { + "ms": { + "version": "2.1.1", + "resolved": "https://registry.npmjs.org/ms/-/ms-2.1.1.tgz", + "integrity": "sha512-tgp+dl5cGk28utYktBsrFqA7HKgrhgPsg6Z/EfhWI4gl1Hwq8B/GmY/0oXZ6nF8hDVesS/FpnYaD/kOWhYQvyg==" + } + } + }, + "serve-static": { + "version": "1.14.1", + "resolved": "https://registry.npmjs.org/serve-static/-/serve-static-1.14.1.tgz", + "integrity": "sha512-JMrvUwE54emCYWlTI+hGrGv5I8dEwmco/00EvkzIIsR7MqrHonbD9pO2MOfFnpFntl7ecpZs+3mW+XbQZu9QCg==", + "requires": { + "encodeurl": "~1.0.2", + "escape-html": "~1.0.3", + "parseurl": "~1.3.3", + "send": "0.17.1" + } + }, + "setprototypeof": { + "version": "1.1.1", + "resolved": "https://registry.npmjs.org/setprototypeof/-/setprototypeof-1.1.1.tgz", + "integrity": "sha512-JvdAWfbXeIGaZ9cILp38HntZSFSo3mWg6xGcJJsd+d4aRMOqauag1C63dJfDw7OaMYwEbHMOxEZ1lqVRYP2OAw==" + }, + "statuses": { + "version": "1.5.0", + "resolved": "https://registry.npmjs.org/statuses/-/statuses-1.5.0.tgz", + "integrity": "sha1-Fhx9rBd2Wf2YEfQ3cfqZOBR4Yow=" + }, + "toidentifier": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/toidentifier/-/toidentifier-1.0.0.tgz", + "integrity": "sha512-yaOH/Pk/VEhBWWTlhI+qXxDFXlejDGcQipMlyxda9nthulaxLZUNcUqFxokp0vcYnvteJln5FNQDRrxj3YcbVw==" + }, + "type-is": { + "version": "1.6.18", + "resolved": "https://registry.npmjs.org/type-is/-/type-is-1.6.18.tgz", + "integrity": "sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g==", + "requires": { + "media-typer": "0.3.0", + "mime-types": "~2.1.24" + } + }, + "unpipe": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/unpipe/-/unpipe-1.0.0.tgz", + "integrity": "sha1-sr9O6FFKrmFltIF4KdIbLvSZBOw=" + }, + "utils-merge": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/utils-merge/-/utils-merge-1.0.1.tgz", + "integrity": "sha1-n5VxD1CiZ5R7LMwSR0HBAoQn5xM=" + }, + "vary": { + "version": "1.1.2", + "resolved": "https://registry.npmjs.org/vary/-/vary-1.1.2.tgz", + "integrity": "sha1-IpnwLG3tMNSllhsLn3RSShj2NPw=" } } } diff --git a/package.json b/package.json index 718095e..80c438e 100644 --- a/package.json +++ b/package.json @@ -6,9 +6,13 @@ "type": "module", "dependencies": { "@root/keypairs": "^0.10.1", - "basetag": "^1.0.0" + "@root/request": "^1.6.1", + "basetag": "^1.0.0", + "keyfetch": "^1.2.1" + }, + "devDependencies": { + "express": "^4.17.1" }, - "devDependencies": {}, "scripts": { "test": "echo \"Error: no test specified\" && exit 1" },