You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
113 lines
3.2 KiB
113 lines
3.2 KiB
import { IIdentifier } from "@veramo/core";
|
|
import { DEFAULT_DID_PROVIDER_NAME } from "../veramo/setup";
|
|
import { getRandomBytesSync } from "ethereum-cryptography/random";
|
|
import { entropyToMnemonic } from "ethereum-cryptography/bip39";
|
|
import { wordlist } from "ethereum-cryptography/bip39/wordlists/english";
|
|
import { HDNode } from "@ethersproject/hdnode";
|
|
import * as didJwt from "did-jwt";
|
|
import * as u8a from "uint8arrays";
|
|
|
|
/**
|
|
*
|
|
*
|
|
* @param {string} address
|
|
* @param {string} publicHex
|
|
* @param {string} privateHex
|
|
* @param {string} derivationPath
|
|
* @return {*} {Omit<IIdentifier, 'provider'>}
|
|
*/
|
|
export const newIdentifier = (
|
|
address: string,
|
|
publicHex: string,
|
|
privateHex: string,
|
|
derivationPath: string
|
|
): Omit<IIdentifier, keyof "provider"> => {
|
|
return {
|
|
did: DEFAULT_DID_PROVIDER_NAME + ":" + address,
|
|
keys: [
|
|
{
|
|
kid: publicHex,
|
|
kms: "local",
|
|
meta: { derivationPath: derivationPath },
|
|
privateKeyHex: privateHex,
|
|
publicKeyHex: publicHex,
|
|
type: "Secp256k1",
|
|
},
|
|
],
|
|
provider: DEFAULT_DID_PROVIDER_NAME,
|
|
services: [],
|
|
};
|
|
};
|
|
|
|
/**
|
|
*
|
|
*
|
|
* @param {string} mnemonic
|
|
* @return {*} {[string, string, string, string]}
|
|
*/
|
|
export const deriveAddress = (
|
|
mnemonic: string
|
|
): [string, string, string, string] => {
|
|
const UPORT_ROOT_DERIVATION_PATH = "m/7696500'/0'/0'/0'";
|
|
mnemonic = mnemonic.trim().toLowerCase();
|
|
|
|
const hdnode: HDNode = HDNode.fromMnemonic(mnemonic);
|
|
const rootNode: HDNode = hdnode.derivePath(UPORT_ROOT_DERIVATION_PATH);
|
|
const privateHex = rootNode.privateKey.substring(2); // original starts with '0x'
|
|
const publicHex = rootNode.publicKey.substring(2); // original starts with '0x'
|
|
const address = rootNode.address;
|
|
|
|
return [address, privateHex, publicHex, UPORT_ROOT_DERIVATION_PATH];
|
|
};
|
|
|
|
/**
|
|
*
|
|
*
|
|
* @return {*} {string}
|
|
*/
|
|
export const createIdentifier = (): string => {
|
|
const entropy: Uint8Array = getRandomBytesSync(32);
|
|
const mnemonic = entropyToMnemonic(entropy, wordlist);
|
|
|
|
return mnemonic;
|
|
};
|
|
|
|
/**
|
|
* Retreive an access token
|
|
*
|
|
* @param {IIdentifier} identifier
|
|
* @return {*}
|
|
*/
|
|
export const accessToken = async (identifier: IIdentifier) => {
|
|
const did: string = identifier.did;
|
|
const privateKeyHex: string = identifier.keys[0].privateKeyHex as string;
|
|
const input = privateKeyHex.startsWith("0x")
|
|
? privateKeyHex.substring(2)
|
|
: privateKeyHex;
|
|
const privateKeyBytes = u8a.fromString(input.toLowerCase(), "base16");
|
|
|
|
const signer = didJwt.ES256KSigner(privateKeyBytes, true);
|
|
|
|
const nowEpoch = Math.floor(Date.now() / 1000);
|
|
const endEpoch = nowEpoch + 60; // add one minute
|
|
|
|
const uportTokenPayload = { exp: endEpoch, iat: nowEpoch, iss: did };
|
|
const alg = undefined; // defaults to 'ES256K', more standardized but harder to verify vs ES256K-R
|
|
const jwt: string = await didJwt.createJWT(uportTokenPayload, {
|
|
alg,
|
|
issuer: did,
|
|
signer,
|
|
});
|
|
return jwt;
|
|
};
|
|
|
|
export const sign = async (privateKeyHex: string) => {
|
|
const input = privateKeyHex.startsWith("0x")
|
|
? privateKeyHex.substring(2)
|
|
: privateKeyHex;
|
|
const privateKeyBytes = u8a.fromString(input.toLowerCase(), "base16");
|
|
|
|
const signer = didJwt.ES256KSigner(privateKeyBytes, true);
|
|
|
|
return signer;
|
|
};
|