PHP WebShell

Текущая директория: /opt/BitGoJS/modules/sdk-core/dist/src/bitgo/utils

Просмотр файла: opengpgUtils.js

"use strict";
/* eslint-disable @typescript-eslint/ban-ts-comment */
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    var desc = Object.getOwnPropertyDescriptor(m, k);
    if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
      desc = { enumerable: true, get: function() { return m[k]; } };
    }
    Object.defineProperty(o, k2, desc);
}) : (function(o, m, k, k2) {
    if (k2 === undefined) k2 = k;
    o[k2] = m[k];
}));
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
    Object.defineProperty(o, "default", { enumerable: true, value: v });
}) : function(o, v) {
    o["default"] = v;
});
var __importStar = (this && this.__importStar) || (function () {
    var ownKeys = function(o) {
        ownKeys = Object.getOwnPropertyNames || function (o) {
            var ar = [];
            for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
            return ar;
        };
        return ownKeys(o);
    };
    return function (mod) {
        if (mod && mod.__esModule) return mod;
        var result = {};
        if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
        __setModuleDefault(result, mod);
        return result;
    };
})();
var __importDefault = (this && this.__importDefault) || function (mod) {
    return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.getBitgoGpgPubKey = getBitgoGpgPubKey;
exports.verifyPrimaryUserWrapper = verifyPrimaryUserWrapper;
exports.verifyShareProof = verifyShareProof;
exports.verifySharedDataProof = verifySharedDataProof;
exports.createSharedDataProof = createSharedDataProof;
exports.createShareProof = createShareProof;
exports.encryptText = encryptText;
exports.encryptAndSignText = encryptAndSignText;
exports.readSignedMessage = readSignedMessage;
exports.signText = signText;
exports.verifySignature = verifySignature;
exports.generateGPGKeyPair = generateGPGKeyPair;
const pgp = __importStar(require("openpgp"));
const openpgp_1 = require("openpgp");
const _ = __importStar(require("lodash"));
const secp256k1_1 = require("@bitgo/secp256k1");
const crypto_1 = __importDefault(require("crypto"));
const sodium = require('libsodium-wrappers-sumo');
/**
 * Fetches BitGo's public gpg key used in MPC flows
 * @param {BitGoBase} bitgo BitGo object
 * @return {Key} public gpg key
 */
async function getBitgoGpgPubKey(bitgo) {
    const constants = await bitgo.fetchConstants();
    if (!constants.mpc || !constants.mpc.bitgoPublicKey) {
        throw new Error('Unable to create MPC keys - bitgoPublicKey is missing from constants');
    }
    const bitgoPublicKeyStr = constants.mpc.bitgoPublicKey;
    const bitgoMPCv2PublicKeyStr = constants.mpc.bitgoMPCv2PublicKey
        ? await (0, openpgp_1.readKey)({ armoredKey: constants.mpc.bitgoMPCv2PublicKey })
        : undefined;
    return { mpcV1: await (0, openpgp_1.readKey)({ armoredKey: bitgoPublicKeyStr }), mpcV2: bitgoMPCv2PublicKeyStr };
}
/**
 * Verifies the primary user on a GPG key using a reference key representing the user to be checked.
 * Allows a verification without a date check by wrapping verifyPrimaryUser of openpgp.
 * @param {Key} pubKey gpg key to check the primary user of.
 * @param {Key} primaryUser gpg key of the user to check.
 * @param {boolean} checkDates If false, disable date checks in the openpgp call to check the primary user.
 * @return {KeyValidityDict} list of users checked and whether each passed as a primary user in pubKey or not.
 */
async function verifyPrimaryUserWrapper(pubKey, primaryUser, checkDates) {
    if (checkDates) {
        return await pubKey.verifyPrimaryUser([primaryUser]);
    }
    else {
        return await pubKey.verifyPrimaryUser([primaryUser], null);
    }
}
/**
 * Verify an Eddsa or Ecdsa KeyShare Proof.
 *
 * @param senderPubKey public key of the sender of the privateShareProof
 * @param privateShareProof u value proof
 * @param uValue u value from an Eddsa keyshare
 * @param algo
 * @return {boolean} whether uValue proof actually was signed by sender as part of their subkeys
 */
async function verifyShareProof(senderPubKey, privateShareProof, uValue, algo) {
    const decodedProof = await pgp.readKey({ armoredKey: privateShareProof });
    const senderGpgKey = await pgp.readKey({ armoredKey: senderPubKey });
    if (!(await verifyPrimaryUserWrapper(decodedProof, senderGpgKey, true))[0].valid) {
        return false;
    }
    const proofSubkeys = decodedProof.getSubkeys()[1];
    if (algo === 'eddsa') {
        const decodedUValueProof = Buffer.from(proofSubkeys.keyPacket.publicParams['Q'].slice(1)).toString('hex');
        const rawUValueProof = Buffer.from(sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(uValue, 'hex'))).toString('hex');
        return decodedUValueProof === rawUValueProof;
    }
    else if (algo === 'ecdsa') {
        const decodedUValueProof = Buffer.from(proofSubkeys.keyPacket.publicParams['Q']).toString('hex');
        const rawUValueProof = secp256k1_1.ecc.pointFromScalar(Buffer.from(uValue, 'hex'), false);
        return rawUValueProof !== null && decodedUValueProof === Buffer.from(rawUValueProof).toString('hex');
    }
    else {
        throw new Error('Invalid algorithm provided');
    }
}
/**
 * Verify a shared data proof.
 *
 * @param senderPubKeyArm public key of the signer of the key with proof data
 * @param keyWithNotation signed reciever key with notation data
 * @param dataToVerify data to be checked against notation data in the signed key
 * @return {boolean} whether proof is valid
 */
async function verifySharedDataProof(senderPubKeyArm, keyWithNotation, dataToVerify) {
    const senderPubKey = await pgp.readKey({ armoredKey: senderPubKeyArm });
    const signedKey = await pgp.readKey({ armoredKey: keyWithNotation });
    if (!(await verifyPrimaryUserWrapper(signedKey, senderPubKey, false).then((values) => _.some(values, (value) => value.valid)))) {
        return false;
    }
    const primaryUser = await signedKey.getPrimaryUser(null);
    const anyInvalidProof = _.some(
    // @ts-ignore
    primaryUser.user.otherCertifications[0].rawNotations, (notation) => dataToVerify.find((i) => i.name === notation.name)?.value !== Buffer.from(notation.value).toString());
    return !anyInvalidProof;
}
/**
 * Creates a proof through adding notation data to a GPG ceritifying signature.
 *
 * @param privateKeyArmored gpg private key in armor format of the sender
 * @param publicKeyToCertArmored gpg public key in armor fomrat of the reciever
 * @param notations data to be proofed
 * @return {string} keyshare proof
 */
async function createSharedDataProof(privateKeyArmored, publicKeyToCertArmored, notations) {
    const certifyingKey = await pgp.readKey({ armoredKey: privateKeyArmored });
    const publicKeyToCert = await pgp.readKey({ armoredKey: publicKeyToCertArmored });
    const dateTime = new Date();
    // UserId Packet.
    const userIdPkt = new pgp.UserIDPacket();
    const primaryUser = await publicKeyToCert.getPrimaryUser(null);
    // @ts-ignore
    userIdPkt.userID = primaryUser.user.userID.userID;
    // Signature packet.
    const signaturePacket = new pgp.SignaturePacket();
    signaturePacket.signatureType = pgp.enums.signature.certPositive;
    signaturePacket.publicKeyAlgorithm = pgp.enums.publicKey.ecdsa;
    signaturePacket.hashAlgorithm = pgp.enums.hash.sha256;
    // @ts-ignore
    signaturePacket.issuerFingerprint = await primaryUser.user.mainKey.keyPacket.getFingerprintBytes();
    // @ts-ignore
    signaturePacket.issuerKeyID = primaryUser.user.mainKey.keyPacket.keyID;
    // @ts-ignore
    signaturePacket.signingKeyID = primaryUser.user.mainKey.keyPacket.keyID;
    // @ts-ignore
    signaturePacket.signersUserID = primaryUser.user.userID.userID;
    // @ts-ignore
    signaturePacket.features = [1];
    notations.forEach(({ name, value }) => {
        signaturePacket.rawNotations.push({
            name: name,
            value: new Uint8Array(Buffer.from(value)),
            humanReadable: true,
            critical: false,
        });
    });
    // Prepare signing data.
    const keydataToSign = {};
    // @ts-ignore
    keydataToSign.key = publicKeyToCert.keyPacket;
    // @ts-ignore
    keydataToSign.userID = userIdPkt;
    // Sign the data (create certification).
    // @ts-ignore
    await signaturePacket.sign(certifyingKey.keyPacket, keydataToSign, dateTime);
    // Assemble packets together.
    const publicKeyToCertPkts = publicKeyToCert.toPacketList();
    const newKeyPktList = new pgp.PacketList();
    newKeyPktList.push(...publicKeyToCertPkts.slice(0, 3), signaturePacket, ...publicKeyToCertPkts.slice(3));
    // @ts-ignore
    const newPubKey = new pgp.PublicKey(newKeyPktList);
    return newPubKey.armor().replace(/\r\n/g, '\n');
}
/**
 * Creates a KeyShare Proof based on given algo.
 *
 * Creates an EdDSA KeyShare Proof by appending an ed25519 subkey (auth) to an armored gpg private key.
 * Creates an ECDSA KeyShare Proof by Append a secp256k1 subkey (auth) to a PGP keychain.
 *
 * @param privateArmor gpg private key in armor format
 * @param uValue u value from an Eddsa keyshare
 * @param algo algo to use, eddsa or ecdsa
 * @return {string} keyshare proof
 */
async function createShareProof(privateArmor, uValue, algo) {
    const privateKey = await (0, openpgp_1.readKey)({ armoredKey: privateArmor });
    const dateTime = new Date();
    // @ts-ignore - type inconsistency, this ctor supports a date param: https://docs.openpgpjs.org/SecretSubkeyPacket.html
    const secretSubkeyPacket = new pgp.SecretSubkeyPacket(dateTime);
    secretSubkeyPacket.algorithm = pgp.enums.publicKey[algo];
    // @ts-ignore - same as above
    secretSubkeyPacket.isEncrypted = false;
    let oid;
    let Q;
    if (algo === 'eddsa') {
        await sodium.ready;
        const subKeyVal = Buffer.from(sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(uValue, 'hex'), 'uint8array'));
        // Sub-key (encryption key) packet.
        oid = [0x2b, 0x06, 0x01, 0x04, 0x01, 0xda, 0x47, 0x0f, 0x01];
        // @ts-ignore
        oid.write = () => new Uint8Array(Buffer.from('092b06010401da470f01', 'hex'));
        Q = new Uint8Array([0x40, ...subKeyVal]);
    }
    else if (algo === 'ecdsa') {
        oid = [0x2b, 0x81, 0x04, 0x00, 0x0a];
        // @ts-ignore - same as above
        oid.write = () => new Uint8Array(Buffer.from('052b8104000a', 'hex'));
        Q = secp256k1_1.ecc.pointFromScalar(new Uint8Array(Buffer.from(uValue, 'hex')), false);
    }
    secretSubkeyPacket.publicParams = {
        oid,
        Q,
    };
    // @ts-ignore - same as above
    await secretSubkeyPacket.computeFingerprintAndKeyID();
    // Sub-key signature packet.
    const subKeydataToSign = {
        key: privateKey.keyPacket,
        bind: secretSubkeyPacket,
    };
    const subkeySignaturePacket = new pgp.SignaturePacket();
    subkeySignaturePacket.signatureType = pgp.enums.signature.subkeyBinding;
    subkeySignaturePacket.publicKeyAlgorithm = pgp.enums.publicKey.ecdsa;
    subkeySignaturePacket.hashAlgorithm = pgp.enums.hash.sha256;
    subkeySignaturePacket.keyFlags = new Uint8Array([pgp.enums.keyFlags.authentication]);
    // Sign the subkey
    // @ts-ignore - sign supports arbitrary data for 2nd param: https://docs.openpgpjs.org/SignaturePacket.html
    await subkeySignaturePacket.sign(privateKey.keyPacket, subKeydataToSign, dateTime);
    // Assemble packets together.
    const newKeyPktList = new pgp.PacketList();
    const privateKeyPkts = privateKey.toPacketList();
    privateKeyPkts.forEach((packet) => newKeyPktList.push(packet));
    newKeyPktList.push(secretSubkeyPacket, subkeySignaturePacket);
    // @ts-ignore - supports packet list as ctor param: https://docs.openpgpjs.org/PrivateKey.html
    const newPubKey = new pgp.PrivateKey(newKeyPktList).toPublic();
    if (!(await verifyPrimaryUserWrapper(newPubKey, privateKey, true))[0].valid) {
        throw new Error('Incorrect signature');
    }
    return newPubKey.armor().replace(/\r\n/g, '\n');
}
/**
 * Encrypts string using gpg key
 * @DEPRECATED - should use encryptAndSignText instead for added security
 *
 * @param text string to encrypt
 * @param key encryption key
 * @return {string} encrypted string
 *
 * TODO(BG-47170): Delete once gpg signatures are fully supported
 */
async function encryptText(text, key) {
    const messageToEncrypt = await (0, openpgp_1.createMessage)({
        text,
    });
    return await (0, openpgp_1.encrypt)({
        message: messageToEncrypt,
        encryptionKeys: [key],
        format: 'armored',
        config: {
            rejectCurves: new Set(),
            showVersion: false,
            showComment: false,
        },
    });
}
/**
 * Encrypts and signs a string
 * @param text string to encrypt and sign
 * @param publicArmor public key to encrypt with
 * @param privateArmor private key to sign with
 */
async function encryptAndSignText(text, publicArmor, privateArmor) {
    const publicKey = await (0, openpgp_1.readKey)({ armoredKey: publicArmor });
    const privateKey = await (0, openpgp_1.readPrivateKey)({ armoredKey: privateArmor });
    const message = await (0, openpgp_1.createMessage)({ text });
    const signedMessage = await (0, openpgp_1.encrypt)({
        message,
        encryptionKeys: publicKey,
        signingKeys: privateKey,
        format: 'armored',
        config: {
            rejectCurves: new Set(),
            showVersion: false,
            showComment: false,
        },
    });
    return signedMessage;
}
/**
 * Reads a signed and encrypted message
 *
 * @param signed signed and encrypted message
 * @param publicArmor public key to verify signature
 * @param privateArmor private key to decrypt message
 */
async function readSignedMessage(signed, publicArmor, privateArmor) {
    const publicKey = await (0, openpgp_1.readKey)({ armoredKey: publicArmor });
    const privateKey = await (0, openpgp_1.readPrivateKey)({ armoredKey: privateArmor });
    const message = await (0, openpgp_1.readMessage)({ armoredMessage: signed });
    const decrypted = await (0, openpgp_1.decrypt)({
        message,
        verificationKeys: publicKey,
        decryptionKeys: privateKey,
        expectSigned: true,
        config: { rejectCurves: new Set() },
    });
    return decrypted.data;
}
/**
 * Generates a signature
 *
 * @param text string to generate a signature for
 * @param privateArmor private key as armored string
 * @return {string} armored signature string
 */
async function signText(text, privateArmor) {
    const privateKey = await (0, openpgp_1.readPrivateKey)({ armoredKey: privateArmor });
    const message = await (0, openpgp_1.createMessage)({ text });
    const signature = await (0, openpgp_1.sign)({
        message,
        signingKeys: privateKey,
        format: 'armored',
        detached: true,
    });
    return signature;
}
/**
 * Verifies signature was generated by the public key and matches the expected text
 *
 * @param text text that the signature was for
 * @param armoredSignature signed message as an armored string
 * @param publicArmor public key that generated the signature
 */
async function verifySignature(text, armoredSignature, publicArmor) {
    const publicKey = await (0, openpgp_1.readKey)({ armoredKey: publicArmor });
    const signature = await (0, openpgp_1.readSignature)({ armoredSignature });
    const message = await (0, openpgp_1.createMessage)({ text });
    const verificationResult = await (0, openpgp_1.verify)({
        message,
        signature,
        verificationKeys: publicKey,
    });
    if (verificationResult.signatures.length !== 1) {
        throw new Error('Invalid number of signatures');
    }
    try {
        await verificationResult.signatures[0].verified;
        return text === verificationResult.data;
    }
    catch {
        return false;
    }
}
/**
 * Generate a GPG key pair
 *
 * @param: keyCurve the curve to create a key with
 * @param: username name of the user (optional)
 * @param: email email of the user (optional)
 */
async function generateGPGKeyPair(keyCurve, username, email) {
    const randomHexString = crypto_1.default.randomBytes(12).toString('hex');
    username = username ?? randomHexString;
    email = email ?? `user-${randomHexString}@${randomHexString}.com`;
    // Allow generating secp256k1 key pairs
    pgp.config.rejectCurves = new Set();
    const gpgKey = await pgp.generateKey({
        userIDs: [
            {
                name: username,
                email,
            },
        ],
        curve: keyCurve,
    });
    return gpgKey;
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"opengpgUtils.js","sourceRoot":"","sources":["../../../../src/bitgo/utils/opengpgUtils.ts"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable @typescript-eslint/ban-ts-comment */\n\nimport * as pgp from 'openpgp';\nimport {\n  createMessage,\n  decrypt,\n  encrypt,\n  Key,\n  readKey,\n  readMessage,\n  readPrivateKey,\n  readSignature,\n  sign,\n  verify,\n} from 'openpgp';\nimport * as _ from 'lodash';\nimport { ecc as secp256k1 } from '@bitgo/secp256k1';\nimport { BitGoBase } from '../bitgoBase';\nimport crypto from 'crypto';\n\nconst sodium = require('libsodium-wrappers-sumo');\n\nexport type KeyValidityDict = {\n  keyID: pgp.KeyID;\n  valid: boolean | null;\n}[];\n\nexport type AuthEncMessage = {\n  encryptedMessage: string;\n  signature: string;\n};\n\n/**\n * Fetches BitGo's public gpg key used in MPC flows\n * @param {BitGoBase} bitgo BitGo object\n * @return {Key} public gpg key\n */\nexport async function getBitgoGpgPubKey(bitgo: BitGoBase): Promise<{ mpcV1: Key; mpcV2: Key | undefined }> {\n  const constants = await bitgo.fetchConstants();\n  if (!constants.mpc || !constants.mpc.bitgoPublicKey) {\n    throw new Error('Unable to create MPC keys - bitgoPublicKey is missing from constants');\n  }\n\n  const bitgoPublicKeyStr = constants.mpc.bitgoPublicKey as string;\n  const bitgoMPCv2PublicKeyStr = constants.mpc.bitgoMPCv2PublicKey\n    ? await readKey({ armoredKey: constants.mpc.bitgoMPCv2PublicKey as string })\n    : undefined;\n  return { mpcV1: await readKey({ armoredKey: bitgoPublicKeyStr }), mpcV2: bitgoMPCv2PublicKeyStr };\n}\n\n/**\n * Verifies the primary user on a GPG key using a reference key representing the user to be checked.\n * Allows a verification without a date check by wrapping verifyPrimaryUser of openpgp.\n * @param {Key} pubKey gpg key to check the primary user of.\n * @param {Key} primaryUser gpg key of the user to check.\n * @param {boolean} checkDates If false, disable date checks in the openpgp call to check the primary user.\n * @return {KeyValidityDict} list of users checked and whether each passed as a primary user in pubKey or not.\n */\nexport async function verifyPrimaryUserWrapper(\n  pubKey: Key,\n  primaryUser: Key,\n  checkDates: boolean\n): Promise<KeyValidityDict> {\n  if (checkDates) {\n    return await pubKey.verifyPrimaryUser([primaryUser]);\n  } else {\n    return await pubKey.verifyPrimaryUser([primaryUser], null as unknown as undefined);\n  }\n}\n\n/**\n * Verify an Eddsa or Ecdsa KeyShare Proof.\n *\n * @param senderPubKey public key of the sender of the privateShareProof\n * @param privateShareProof u value proof\n * @param uValue u value from an Eddsa keyshare\n * @param algo\n * @return {boolean} whether uValue proof actually was signed by sender as part of their subkeys\n */\nexport async function verifyShareProof(\n  senderPubKey: string,\n  privateShareProof: string,\n  uValue: string,\n  algo: 'eddsa' | 'ecdsa'\n): Promise<boolean> {\n  const decodedProof = await pgp.readKey({ armoredKey: privateShareProof });\n  const senderGpgKey = await pgp.readKey({ armoredKey: senderPubKey });\n  if (!(await verifyPrimaryUserWrapper(decodedProof, senderGpgKey, true))[0].valid) {\n    return false;\n  }\n  const proofSubkeys = decodedProof.getSubkeys()[1];\n  if (algo === 'eddsa') {\n    const decodedUValueProof = Buffer.from(proofSubkeys.keyPacket.publicParams['Q'].slice(1)).toString('hex');\n    const rawUValueProof = Buffer.from(\n      sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(uValue, 'hex'))\n    ).toString('hex');\n    return decodedUValueProof === rawUValueProof;\n  } else if (algo === 'ecdsa') {\n    const decodedUValueProof = Buffer.from(proofSubkeys.keyPacket.publicParams['Q']).toString('hex');\n    const rawUValueProof = secp256k1.pointFromScalar(Buffer.from(uValue, 'hex'), false);\n    return rawUValueProof !== null && decodedUValueProof === Buffer.from(rawUValueProof).toString('hex');\n  } else {\n    throw new Error('Invalid algorithm provided');\n  }\n}\n\n/**\n * Verify a shared data proof.\n *\n * @param senderPubKeyArm public key of the signer of the key with proof data\n * @param keyWithNotation signed reciever key with notation data\n * @param dataToVerify data to be checked against notation data in the signed key\n * @return {boolean} whether proof is valid\n */\nexport async function verifySharedDataProof(\n  senderPubKeyArm: string,\n  keyWithNotation: string,\n  dataToVerify: { name: string; value: string }[]\n): Promise<boolean> {\n  const senderPubKey = await pgp.readKey({ armoredKey: senderPubKeyArm });\n  const signedKey = await pgp.readKey({ armoredKey: keyWithNotation });\n  if (\n    !(await verifyPrimaryUserWrapper(signedKey, senderPubKey, false).then((values) =>\n      _.some(values, (value) => value.valid)\n    ))\n  ) {\n    return false;\n  }\n  const primaryUser = await signedKey.getPrimaryUser(null as unknown as undefined);\n  const anyInvalidProof = _.some(\n    // @ts-ignore\n    primaryUser.user.otherCertifications[0].rawNotations,\n    (notation) => dataToVerify.find((i) => i.name === notation.name)?.value !== Buffer.from(notation.value).toString()\n  );\n  return !anyInvalidProof;\n}\n\n/**\n * Creates a proof through adding notation data to a GPG ceritifying signature.\n *\n * @param privateKeyArmored gpg private key in armor format of the sender\n * @param publicKeyToCertArmored gpg public key in armor fomrat of the reciever\n * @param notations data to be proofed\n * @return {string} keyshare proof\n */\nexport async function createSharedDataProof(\n  privateKeyArmored: string,\n  publicKeyToCertArmored: string,\n  notations: { name: string; value: string }[]\n): Promise<string> {\n  const certifyingKey = await pgp.readKey({ armoredKey: privateKeyArmored });\n  const publicKeyToCert = await pgp.readKey({ armoredKey: publicKeyToCertArmored });\n  const dateTime = new Date();\n  // UserId Packet.\n  const userIdPkt = new pgp.UserIDPacket();\n  const primaryUser = await publicKeyToCert.getPrimaryUser(null as unknown as undefined);\n  // @ts-ignore\n  userIdPkt.userID = primaryUser.user.userID.userID;\n  // Signature packet.\n  const signaturePacket = new pgp.SignaturePacket();\n  signaturePacket.signatureType = pgp.enums.signature.certPositive;\n  signaturePacket.publicKeyAlgorithm = pgp.enums.publicKey.ecdsa;\n  signaturePacket.hashAlgorithm = pgp.enums.hash.sha256;\n  // @ts-ignore\n  signaturePacket.issuerFingerprint = await primaryUser.user.mainKey.keyPacket.getFingerprintBytes();\n  // @ts-ignore\n  signaturePacket.issuerKeyID = primaryUser.user.mainKey.keyPacket.keyID;\n  // @ts-ignore\n  signaturePacket.signingKeyID = primaryUser.user.mainKey.keyPacket.keyID;\n  // @ts-ignore\n  signaturePacket.signersUserID = primaryUser.user.userID.userID;\n  // @ts-ignore\n  signaturePacket.features = [1];\n  notations.forEach(({ name, value }) => {\n    signaturePacket.rawNotations.push({\n      name: name,\n      value: new Uint8Array(Buffer.from(value)),\n      humanReadable: true,\n      critical: false,\n    });\n  });\n\n  // Prepare signing data.\n  const keydataToSign = {};\n  // @ts-ignore\n  keydataToSign.key = publicKeyToCert.keyPacket;\n  // @ts-ignore\n  keydataToSign.userID = userIdPkt;\n\n  // Sign the data (create certification).\n  // @ts-ignore\n  await signaturePacket.sign(certifyingKey.keyPacket, keydataToSign, dateTime);\n\n  // Assemble packets together.\n  const publicKeyToCertPkts = publicKeyToCert.toPacketList();\n  const newKeyPktList = new pgp.PacketList();\n  newKeyPktList.push(...publicKeyToCertPkts.slice(0, 3), signaturePacket, ...publicKeyToCertPkts.slice(3));\n  // @ts-ignore\n  const newPubKey = new pgp.PublicKey(newKeyPktList);\n  return newPubKey.armor().replace(/\\r\\n/g, '\\n');\n}\n\n/**\n * Creates a KeyShare Proof based on given algo.\n *\n * Creates an EdDSA KeyShare Proof by appending an ed25519 subkey (auth) to an armored gpg private key.\n * Creates an ECDSA KeyShare Proof by Append a secp256k1 subkey (auth) to a PGP keychain.\n *\n * @param privateArmor gpg private key in armor format\n * @param uValue u value from an Eddsa keyshare\n * @param algo algo to use, eddsa or ecdsa\n * @return {string} keyshare proof\n */\nexport async function createShareProof(privateArmor: string, uValue: string, algo: string): Promise<string> {\n  const privateKey = await readKey({ armoredKey: privateArmor });\n  const dateTime = new Date();\n  // @ts-ignore - type inconsistency, this ctor supports a date param: https://docs.openpgpjs.org/SecretSubkeyPacket.html\n  const secretSubkeyPacket = new pgp.SecretSubkeyPacket(dateTime);\n  secretSubkeyPacket.algorithm = pgp.enums.publicKey[algo];\n  // @ts-ignore - same as above\n  secretSubkeyPacket.isEncrypted = false;\n  let oid;\n  let Q;\n  if (algo === 'eddsa') {\n    await sodium.ready;\n    const subKeyVal = Buffer.from(\n      sodium.crypto_scalarmult_ed25519_base_noclamp(Buffer.from(uValue, 'hex'), 'uint8array')\n    );\n    // Sub-key (encryption key) packet.\n    oid = [0x2b, 0x06, 0x01, 0x04, 0x01, 0xda, 0x47, 0x0f, 0x01];\n    // @ts-ignore\n    oid.write = () => new Uint8Array(Buffer.from('092b06010401da470f01', 'hex'));\n    Q = new Uint8Array([0x40, ...subKeyVal]);\n  } else if (algo === 'ecdsa') {\n    oid = [0x2b, 0x81, 0x04, 0x00, 0x0a];\n    // @ts-ignore - same as above\n    oid.write = () => new Uint8Array(Buffer.from('052b8104000a', 'hex'));\n    Q = secp256k1.pointFromScalar(new Uint8Array(Buffer.from(uValue, 'hex')), false);\n  }\n  secretSubkeyPacket.publicParams = {\n    oid,\n    Q,\n  };\n  // @ts-ignore - same as above\n  await secretSubkeyPacket.computeFingerprintAndKeyID();\n\n  // Sub-key signature packet.\n  const subKeydataToSign = {\n    key: privateKey.keyPacket,\n    bind: secretSubkeyPacket,\n  };\n  const subkeySignaturePacket = new pgp.SignaturePacket();\n  subkeySignaturePacket.signatureType = pgp.enums.signature.subkeyBinding;\n  subkeySignaturePacket.publicKeyAlgorithm = pgp.enums.publicKey.ecdsa;\n  subkeySignaturePacket.hashAlgorithm = pgp.enums.hash.sha256;\n  subkeySignaturePacket.keyFlags = new Uint8Array([pgp.enums.keyFlags.authentication]);\n\n  // Sign the subkey\n  // @ts-ignore - sign supports arbitrary data for 2nd param: https://docs.openpgpjs.org/SignaturePacket.html\n  await subkeySignaturePacket.sign(privateKey.keyPacket, subKeydataToSign, dateTime);\n\n  // Assemble packets together.\n  const newKeyPktList = new pgp.PacketList();\n  const privateKeyPkts = privateKey.toPacketList();\n  privateKeyPkts.forEach((packet) => newKeyPktList.push(packet));\n  newKeyPktList.push(secretSubkeyPacket, subkeySignaturePacket);\n  // @ts-ignore - supports packet list as ctor param: https://docs.openpgpjs.org/PrivateKey.html\n  const newPubKey = new pgp.PrivateKey(newKeyPktList).toPublic();\n\n  if (!(await verifyPrimaryUserWrapper(newPubKey, privateKey, true))[0].valid) {\n    throw new Error('Incorrect signature');\n  }\n\n  return newPubKey.armor().replace(/\\r\\n/g, '\\n');\n}\n\n/**\n * Encrypts string using gpg key\n * @DEPRECATED - should use encryptAndSignText instead for added security\n *\n * @param text string to encrypt\n * @param key encryption key\n * @return {string} encrypted string\n *\n * TODO(BG-47170): Delete once gpg signatures are fully supported\n */\nexport async function encryptText(text: string, key: Key): Promise<string> {\n  const messageToEncrypt = await createMessage({\n    text,\n  });\n  return await encrypt({\n    message: messageToEncrypt,\n    encryptionKeys: [key],\n    format: 'armored',\n    config: {\n      rejectCurves: new Set(),\n      showVersion: false,\n      showComment: false,\n    },\n  });\n}\n\n/**\n * Encrypts and signs a string\n * @param text string to encrypt and sign\n * @param publicArmor public key to encrypt with\n * @param privateArmor private key to sign with\n */\nexport async function encryptAndSignText(text: string, publicArmor: string, privateArmor: string): Promise<string> {\n  const publicKey = await readKey({ armoredKey: publicArmor });\n  const privateKey = await readPrivateKey({ armoredKey: privateArmor });\n\n  const message = await createMessage({ text });\n\n  const signedMessage = await encrypt({\n    message,\n    encryptionKeys: publicKey,\n    signingKeys: privateKey,\n    format: 'armored',\n    config: {\n      rejectCurves: new Set(),\n      showVersion: false,\n      showComment: false,\n    },\n  });\n\n  return signedMessage;\n}\n\n/**\n * Reads a signed and encrypted message\n *\n * @param signed signed and encrypted message\n * @param publicArmor public key to verify signature\n * @param privateArmor private key to decrypt message\n */\nexport async function readSignedMessage(signed: string, publicArmor: string, privateArmor: string): Promise<string> {\n  const publicKey = await readKey({ armoredKey: publicArmor });\n  const privateKey = await readPrivateKey({ armoredKey: privateArmor });\n\n  const message = await readMessage({ armoredMessage: signed });\n  const decrypted = await decrypt({\n    message,\n    verificationKeys: publicKey,\n    decryptionKeys: privateKey,\n    expectSigned: true,\n    config: { rejectCurves: new Set() },\n  });\n\n  return decrypted.data;\n}\n\n/**\n * Generates a signature\n *\n * @param text string to generate a signature for\n * @param privateArmor private key as armored string\n * @return {string} armored signature string\n */\nexport async function signText(text: string, privateArmor: string): Promise<string> {\n  const privateKey = await readPrivateKey({ armoredKey: privateArmor });\n  const message = await createMessage({ text });\n  const signature = await sign({\n    message,\n    signingKeys: privateKey,\n    format: 'armored',\n    detached: true,\n  });\n\n  return signature;\n}\n\n/**\n * Verifies signature was generated by the public key and matches the expected text\n *\n * @param text text that the signature was for\n * @param armoredSignature signed message as an armored string\n * @param publicArmor public key that generated the signature\n */\nexport async function verifySignature(text: string, armoredSignature: string, publicArmor: string): Promise<boolean> {\n  const publicKey = await readKey({ armoredKey: publicArmor });\n  const signature = await readSignature({ armoredSignature });\n  const message = await createMessage({ text });\n  const verificationResult = await verify({\n    message,\n    signature,\n    verificationKeys: publicKey,\n  });\n\n  if (verificationResult.signatures.length !== 1) {\n    throw new Error('Invalid number of signatures');\n  }\n\n  try {\n    await verificationResult.signatures[0].verified;\n    return text === verificationResult.data;\n  } catch {\n    return false;\n  }\n}\n\n/**\n * Generate a GPG key pair\n *\n * @param: keyCurve the curve to create a key with\n * @param: username name of the user (optional)\n * @param: email email of the user (optional)\n */\nexport async function generateGPGKeyPair(\n  keyCurve: pgp.EllipticCurveName,\n  username?: string | undefined,\n  email?: string | undefined\n): Promise<pgp.SerializedKeyPair<string>> {\n  const randomHexString = crypto.randomBytes(12).toString('hex');\n  username = username ?? randomHexString;\n  email = email ?? `user-${randomHexString}@${randomHexString}.com`;\n\n  // Allow generating secp256k1 key pairs\n  pgp.config.rejectCurves = new Set();\n  const gpgKey = await pgp.generateKey({\n    userIDs: [\n      {\n        name: username,\n        email,\n      },\n    ],\n    curve: keyCurve,\n  });\n\n  return gpgKey;\n}\n"]}

Выполнить команду


Для локальной разработки. Не используйте в интернете!