@deso-core/identity
Advanced tools
Comparing version 0.0.14 to 0.0.15
{ | ||
"name": "@deso-core/identity", | ||
"version": "0.0.14", | ||
"version": "0.0.15", | ||
"type": "commonjs", | ||
@@ -13,3 +13,3 @@ "repository": { | ||
"bs58": "^5.0.0", | ||
"deso-protocol-types": "0.5.0" | ||
"deso-protocol-types": "0.5.7" | ||
}, | ||
@@ -16,0 +16,0 @@ "main": "./src/index.js", |
@@ -71,3 +71,3 @@ # @deso-core/identity | ||
const event = state.event; | ||
// The current user object contains the user's current permissions | ||
@@ -137,7 +137,5 @@ // (TransactionCountLimitMap). This value will be updated when the logged in | ||
// Encrypt plain text. Likely you would be using the messagingPrivateKey found on the | ||
// identity user's derived key to be used for encrypted chat or messaging applications. | ||
// Returns a promise that resolves to a hex encoded encrypted string. | ||
const encryptedMessageHex = await encrypt( | ||
senderMessagingPrivateSeedHex, | ||
// Encrypt plain text with the recipients public key. This can be subsequently | ||
// decrypted using the recipient's private key. | ||
const encryptedMessageHex = await identity.encryptMessage( | ||
recipientPublicKeyBase58Check, | ||
@@ -147,7 +145,16 @@ plaintextMsg | ||
// Decrypt cipher text. Returns a promise that resolves to a decrypted, plaintext string. | ||
const decryptedMessagePlaintext = await decrypt( | ||
recipientMessagingPrivateSeedHex, | ||
senderPublicKeyBase58Check, | ||
hexEncodedCipherText | ||
// Decrypt a message returned from any of the message endpoints of the deso | ||
// backend messages api. If it is a group message you will need to fetch the | ||
// groups the user is a member of and provide them. If it's known that the | ||
// message is not a a group message you can pass an empty array for the groups | ||
// parameter. | ||
// | ||
// See the api docs for sending and receiving messages here: | ||
// https://docs.deso.org/deso-backend/api/messages-endpoints | ||
// | ||
// See the api docs for access groups here: | ||
// https://docs.deso.org/deso-backend/api/access-group-endpoints | ||
const decryptedMessagePlaintext = await identity.decryptMessage( | ||
message, | ||
accessGroups | ||
); | ||
@@ -154,0 +161,0 @@ ``` |
import { Identity } from './lib/identity'; | ||
export declare const identity: Identity; | ||
export * from './lib/crypto-utils'; | ||
export * from './lib/error-types'; | ||
export * from './lib/types'; |
@@ -8,4 +8,5 @@ "use strict"; | ||
exports.identity = new identity_1.Identity(window, api_1.api); | ||
(0, tslib_1.__exportStar)(require("./lib/crypto-utils"), exports); | ||
(0, tslib_1.__exportStar)(require("./lib/error-types"), exports); | ||
(0, tslib_1.__exportStar)(require("./lib/types"), exports); | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzL2lkZW50aXR5L3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7O0FBQUEsbUNBQWdDO0FBQ2hDLDZDQUEwQztBQUU3QixRQUFBLFFBQVEsR0FBRyxJQUFJLG1CQUFRLENBQUMsTUFBTSxFQUFFLFNBQUcsQ0FBQyxDQUFDO0FBQ2xELGlFQUFrQztBQUNsQywyREFBNEIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBhcGkgfSBmcm9tICcuL2xpYi9hcGknO1xuaW1wb3J0IHsgSWRlbnRpdHkgfSBmcm9tICcuL2xpYi9pZGVudGl0eSc7XG5cbmV4cG9ydCBjb25zdCBpZGVudGl0eSA9IG5ldyBJZGVudGl0eSh3aW5kb3csIGFwaSk7XG5leHBvcnQgKiBmcm9tICcuL2xpYi9lcnJvci10eXBlcyc7XG5leHBvcnQgKiBmcm9tICcuL2xpYi90eXBlcyc7XG4iXX0= | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9saWJzL2lkZW50aXR5L3NyYy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiOzs7O0FBQUEsbUNBQWdDO0FBQ2hDLDZDQUEwQztBQUU3QixRQUFBLFFBQVEsR0FBRyxJQUFJLG1CQUFRLENBQUMsTUFBTSxFQUFFLFNBQUcsQ0FBQyxDQUFDO0FBQ2xELGtFQUFtQztBQUNuQyxpRUFBa0M7QUFDbEMsMkRBQTRCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgYXBpIH0gZnJvbSAnLi9saWIvYXBpJztcbmltcG9ydCB7IElkZW50aXR5IH0gZnJvbSAnLi9saWIvaWRlbnRpdHknO1xuXG5leHBvcnQgY29uc3QgaWRlbnRpdHkgPSBuZXcgSWRlbnRpdHkod2luZG93LCBhcGkpO1xuZXhwb3J0ICogZnJvbSAnLi9saWIvY3J5cHRvLXV0aWxzJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL2Vycm9yLXR5cGVzJztcbmV4cG9ydCAqIGZyb20gJy4vbGliL3R5cGVzJztcbiJdfQ== |
@@ -5,3 +5,9 @@ import { jwtAlgorithm, KeyPair, Network } from './types'; | ||
} | ||
export declare const keygen: () => KeyPair; | ||
export declare const keygen: (seed?: string | Uint8Array | undefined) => KeyPair; | ||
/** | ||
* | ||
* @param data could be a hex string or a byte array (Uint8Array) | ||
* @returns | ||
*/ | ||
export declare const sha256X2: (data: Uint8Array | string) => Promise<Uint8Array>; | ||
export declare const publicKeyToBase58Check: (publicKeyBytes: Uint8Array, options?: Base58CheckOptions | undefined) => Promise<string>; | ||
@@ -16,5 +22,12 @@ export interface SignOptions { | ||
}) => Promise<string>; | ||
export declare const encrypt: (senderSeedHex: string, recipientPublicKeyBase58Check: string, plaintext: string) => Promise<string>; | ||
export declare const encryptChatMessage: (senderSeedHex: string, recipientPublicKeyBase58Check: string, message: string) => Promise<string>; | ||
/** | ||
* @param publicEncryptionKey could be in raw bytes or base58check format | ||
* @param plaintext | ||
* @returns cipher text as a hex string | ||
*/ | ||
export declare const encrypt: (publicKey: Uint8Array | string, plaintext: string) => Promise<string>; | ||
export declare const bs58PublicKeyToBytes: (str: string) => Promise<Uint8Array>; | ||
export declare const decrypt: (recipientSeedHex: string, senderPublicKeyBase58Check: string, cipherTextHex: string) => Promise<string>; | ||
export declare const decryptChatMessage: (recipientSeedHex: string, publicDecryptionKey: string, cipherTextHex: string) => Promise<string>; | ||
export declare const decrypt: (privateDecryptionKey: Uint8Array | string, cipherTextHex: string) => Promise<string>; | ||
export declare const getSharedPrivateKey: (privKey: Uint8Array, pubKey: Uint8Array) => Promise<Uint8Array>; | ||
@@ -24,2 +37,3 @@ export declare const decodePublicKey: (publicKeyBase58Check: string) => Promise<Uint8Array>; | ||
export declare const kdf: (secret: Uint8Array, outputLength: number) => Promise<Uint8Array>; | ||
export declare function deriveAccessGroupKeyPair(privateKeyHex: string, groupKeyName: string): Promise<KeyPair>; | ||
export {}; |
"use strict"; | ||
Object.defineProperty(exports, "__esModule", { value: true }); | ||
exports.kdf = exports.getSharedSecret = exports.decodePublicKey = exports.getSharedPrivateKey = exports.decrypt = exports.bs58PublicKeyToBytes = exports.encrypt = exports.getSignedJWT = exports.signTx = exports.publicKeyToBase58Check = exports.keygen = void 0; | ||
exports.deriveAccessGroupKeyPair = exports.kdf = exports.getSharedSecret = exports.decodePublicKey = exports.getSharedPrivateKey = exports.decrypt = exports.decryptChatMessage = exports.bs58PublicKeyToBytes = exports.encrypt = exports.encryptChatMessage = exports.getSignedJWT = exports.signTx = exports.publicKeyToBase58Check = exports.sha256X2 = exports.keygen = void 0; | ||
const secp256k1_1 = require("@noble/secp256k1"); | ||
@@ -43,6 +43,9 @@ const bs58 = require("bs58"); | ||
// https://github.com/w3c/webcrypto/issues/82 | ||
const keygen = () => { | ||
// ecUtils.randomBytes uses window.crypto.getRandomValues in the browser or | ||
// crypto.randomBytes in node. | ||
const privateKey = secp256k1_1.utils.randomBytes(32); | ||
// | ||
// If you don't provide a seed, a random one will be generated for you and a | ||
// random key pair will be returned. If you do provide a seed, it should be a | ||
// randomly generated 32 byte value (Uint8Array of length 32 or hex string of | ||
// length 64) | ||
const keygen = (seed) => { | ||
const privateKey = seed ? normalizeSeed(seed) : secp256k1_1.utils.randomBytes(32); | ||
const seedHex = secp256k1_1.utils.bytesToHex(privateKey); | ||
@@ -56,3 +59,20 @@ return { | ||
exports.keygen = keygen; | ||
const sha256X2 = async (data) => secp256k1_1.utils.sha256(await secp256k1_1.utils.sha256(data)); | ||
const normalizeSeed = (seed) => { | ||
if (typeof seed === 'string') { | ||
return secp256k1_1.utils.hexToBytes(seed); | ||
} | ||
else { | ||
return seed; | ||
} | ||
}; | ||
/** | ||
* | ||
* @param data could be a hex string or a byte array (Uint8Array) | ||
* @returns | ||
*/ | ||
const sha256X2 = async (data) => { | ||
const d = typeof data === 'string' ? secp256k1_1.utils.hexToBytes(data) : data; | ||
return secp256k1_1.utils.sha256(await secp256k1_1.utils.sha256(d)); | ||
}; | ||
exports.sha256X2 = sha256X2; | ||
const publicKeyToBase58Check = async (publicKeyBytes, options) => { | ||
@@ -65,3 +85,3 @@ var _a; | ||
const bytes = new Uint8Array([...prefix, ...publicKeyBytes]); | ||
const checksum = await sha256X2(bytes); | ||
const checksum = await (0, exports.sha256X2)(bytes); | ||
return bs58.encode(concatUint8Arrays([bytes, checksum], bytes.length + 4)); | ||
@@ -81,3 +101,3 @@ }; | ||
const transactionBytes = secp256k1_1.utils.hexToBytes(txHex); | ||
const hashedTxBytes = await sha256X2(transactionBytes); | ||
const hashedTxBytes = await (0, exports.sha256X2)(transactionBytes); | ||
const transactionHashHex = secp256k1_1.utils.bytesToHex(hashedTxBytes); | ||
@@ -116,3 +136,3 @@ const privateKey = secp256k1_1.utils.hexToBytes(seedHex); | ||
} | ||
const encrypt = async (senderSeedHex, recipientPublicKeyBase58Check, plaintext) => { | ||
const encryptChatMessage = async (senderSeedHex, recipientPublicKeyBase58Check, message) => { | ||
const privateKey = secp256k1_1.utils.hexToBytes(senderSeedHex); | ||
@@ -122,5 +142,17 @@ const recipientPublicKey = await (0, exports.bs58PublicKeyToBytes)(recipientPublicKeyBase58Check); | ||
const sharedPublicKey = (0, secp256k1_1.getPublicKey)(sharedPrivateKey); | ||
return (0, exports.encrypt)(sharedPublicKey, message); | ||
}; | ||
exports.encryptChatMessage = encryptChatMessage; | ||
/** | ||
* @param publicEncryptionKey could be in raw bytes or base58check format | ||
* @param plaintext | ||
* @returns cipher text as a hex string | ||
*/ | ||
const encrypt = async (publicKey, plaintext) => { | ||
const ephemPrivateKey = secp256k1_1.utils.randomBytes(32); | ||
const ephemPublicKey = (0, secp256k1_1.getPublicKey)(ephemPrivateKey); | ||
const privKey = await (0, exports.getSharedPrivateKey)(ephemPrivateKey, sharedPublicKey); | ||
const publicKeyBytes = typeof publicKey === 'string' | ||
? await (0, exports.bs58PublicKeyToBytes)(publicKey) | ||
: publicKey; | ||
const privKey = await (0, exports.getSharedPrivateKey)(ephemPrivateKey, publicKeyBytes); | ||
const encryptionKey = privKey.slice(0, 16); | ||
@@ -149,3 +181,3 @@ const iv = secp256k1_1.utils.randomBytes(16); | ||
const checksumA = bytes.slice(-4); | ||
const checksumB = await sha256X2(payload); | ||
const checksumB = await (0, exports.sha256X2)(payload); | ||
if ((checksumA[0] ^ checksumB[0]) | | ||
@@ -171,3 +203,10 @@ (checksumA[1] ^ checksumB[1]) | | ||
}; | ||
const decrypt = async (recipientSeedHex, senderPublicKeyBase58Check, cipherTextHex) => { | ||
const decryptChatMessage = async (recipientSeedHex, publicDecryptionKey, cipherTextHex) => { | ||
const privateKey = secp256k1_1.utils.hexToBytes(recipientSeedHex); | ||
const publicKey = await (0, exports.bs58PublicKeyToBytes)(publicDecryptionKey); | ||
const sharedPrivateKey = await (0, exports.getSharedPrivateKey)(privateKey, publicKey); | ||
return (0, exports.decrypt)(sharedPrivateKey, cipherTextHex); | ||
}; | ||
exports.decryptChatMessage = decryptChatMessage; | ||
const decrypt = async (privateDecryptionKey, cipherTextHex) => { | ||
const cipherBytes = secp256k1_1.utils.hexToBytes(cipherTextHex); | ||
@@ -181,5 +220,3 @@ const metaLength = 113; | ||
} | ||
const privateKey = secp256k1_1.utils.hexToBytes(recipientSeedHex); | ||
const publicKey = await (0, exports.bs58PublicKeyToBytes)(senderPublicKeyBase58Check); | ||
const sharedPrivateKey = await (0, exports.getSharedPrivateKey)(privateKey, publicKey); | ||
const privateKey = normalizeSeed(privateDecryptionKey); | ||
const ephemPublicKey = cipherBytes.slice(0, 65); | ||
@@ -189,12 +226,13 @@ const cipherTextLength = cipherBytes.length - metaLength; | ||
const cipherAndIv = cipherBytes.slice(65, 65 + 16 + cipherTextLength); | ||
const ciphertext = cipherAndIv.slice(16); | ||
const cipherText = cipherAndIv.slice(16); | ||
const msgMac = cipherBytes.slice(65 + 16 + cipherTextLength); | ||
const privKey = await (0, exports.getSharedPrivateKey)(sharedPrivateKey, ephemPublicKey); | ||
const encryptionKey = privKey.slice(0, 16); | ||
const macKey = await secp256k1_1.utils.sha256(privKey.slice(16)); | ||
const sharedSecretKey = await (0, exports.getSharedPrivateKey)(privateKey, ephemPublicKey); | ||
const encryptionKey = sharedSecretKey.slice(0, 16); | ||
const macKey = await secp256k1_1.utils.sha256(sharedSecretKey.slice(16)); | ||
const hmacKnownGood = await secp256k1_1.utils.hmacSha256(macKey, cipherAndIv); | ||
if (!isValidHmac(msgMac, hmacKnownGood)) | ||
if (!isValidHmac(msgMac, hmacKnownGood)) { | ||
throw new Error('incorrect MAC'); | ||
} | ||
const cryptoKey = await globalThis.crypto.subtle.importKey('raw', encryptionKey, 'AES-CTR', true, ['decrypt']); | ||
const decryptedBuffer = await globalThis.crypto.subtle.decrypt({ name: 'AES-CTR', counter: iv, length: 128 }, cryptoKey, ciphertext); | ||
const decryptedBuffer = await globalThis.crypto.subtle.decrypt({ name: 'AES-CTR', counter: iv, length: 128 }, cryptoKey, cipherText); | ||
return new TextDecoder().decode(decryptedBuffer); | ||
@@ -271,2 +309,9 @@ }; | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crypto-utils.js","sourceRoot":"","sources":["../../../../../libs/identity/src/lib/crypto-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  getPublicKey,\n  getSharedSecret as nobleGetSharedSecret,\n  Point,\n  sign as ecSign,\n  utils as ecUtils,\n} from '@noble/secp256k1';\nimport * as bs58 from 'bs58';\nimport { PUBLIC_KEY_PREFIXES } from './constants';\nimport { jwtAlgorithm, KeyPair, Network } from './types';\n\n// Browser friendly version of node's Buffer.concat.\nfunction concatUint8Arrays(arrays: Uint8Array[], length?: number) {\n  if (length === undefined) {\n    length = arrays.reduce((acc, array) => acc + array.length, 0);\n  }\n  const result = new Uint8Array(length);\n  let offset = 0;\n  for (let i = 0; i < arrays.length; i++) {\n    if (offset + arrays[i].length > length) {\n      result.set(arrays[i].slice(0, length - offset), offset);\n      break;\n    }\n    result.set(arrays[i], offset);\n    offset += arrays[i].length;\n  }\n  return result;\n}\n\nconst uvarint64ToBuf = (uint: number): Uint8Array => {\n  const result: number[] = [];\n  while (uint >= 0x80) {\n    result.push(Number((BigInt(uint) & BigInt(0xff)) | BigInt(0x80)));\n    uint = Number(BigInt(uint) >> BigInt(7));\n  }\n  result.push(uint | 0);\n\n  return new Uint8Array(result);\n};\n\ninterface Base58CheckOptions {\n  network: Network;\n}\n\n// We are not using the native web crypto API to actually generate keys\n// because it does not support the secp256k1 curve. Instead, we are using\n// https://github.com/paulmillr/noble-secp256k1 which is a browser friendly\n// alternative to the node elliptic package which is far smaller and only\n// focuses on supporting the ec algorithm we are actually interested in here.\n// If the web crypto API ever adds support for secp256k1, we should change\n// this to use it.\n//\n// See the following for more info:\n// https://github.com/w3c/webcrypto/issues/82\nexport const keygen = (): KeyPair => {\n  // ecUtils.randomBytes uses window.crypto.getRandomValues in the browser or\n  // crypto.randomBytes in node.\n  const privateKey = ecUtils.randomBytes(32);\n  const seedHex = ecUtils.bytesToHex(privateKey);\n\n  return {\n    seedHex,\n    private: privateKey,\n    public: getPublicKey(privateKey, true /* isCompressed */),\n  };\n};\n\nconst sha256X2 = async (data: Uint8Array): Promise<Uint8Array> =>\n  ecUtils.sha256(await ecUtils.sha256(data));\n\nexport const publicKeyToBase58Check = async (\n  publicKeyBytes: Uint8Array,\n  options?: Base58CheckOptions\n): Promise<string> => {\n  const prefix = PUBLIC_KEY_PREFIXES[options?.network ?? 'mainnet'].deso;\n  // This is the same as the implementation in the bs58check package, but we\n  // slightly modify it to use the browser friendly version of Buffer.concat.\n  // See: https://github.com/bitcoinjs/bs58check/blob/12b3e700f355c5c49d0be3f8fc29be6c66e753e9/base.js#L1\n  const bytes = new Uint8Array([...prefix, ...publicKeyBytes]);\n  const checksum = await sha256X2(bytes);\n  return bs58.encode(concatUint8Arrays([bytes, checksum], bytes.length + 4));\n};\n\nexport interface SignOptions {\n  isDerivedKey: boolean;\n}\n\nconst sign = (msgHashHex: string, privateKey: Uint8Array) => {\n  return ecSign(msgHashHex, privateKey, {\n    // For details about the signing options see: https://github.com/paulmillr/noble-secp256k1#signmsghash-privatekey\n    canonical: true,\n    der: true,\n    extraEntropy: true,\n    recovered: true,\n  });\n};\n\nexport const signTx = async (\n  txHex: string,\n  seedHex: string,\n  options?: SignOptions\n): Promise<string> => {\n  const transactionBytes = ecUtils.hexToBytes(txHex);\n  const hashedTxBytes = await sha256X2(transactionBytes);\n  const transactionHashHex = ecUtils.bytesToHex(hashedTxBytes);\n  const privateKey = ecUtils.hexToBytes(seedHex);\n  const [signatureBytes, recoveryParam] = await sign(\n    transactionHashHex,\n    privateKey\n  );\n\n  const signatureLength = uvarint64ToBuf(signatureBytes.length);\n\n  if (options?.isDerivedKey) {\n    signatureBytes[0] += 1 + recoveryParam;\n  }\n\n  const signedTransactionBytes = ecUtils.concatBytes(\n    transactionBytes.slice(0, -1),\n    signatureLength,\n    signatureBytes\n  );\n\n  return ecUtils.bytesToHex(signedTransactionBytes);\n};\n\nexport const getSignedJWT = async (\n  seedHex: string,\n  alg: jwtAlgorithm,\n  {\n    derivedPublicKeyBase58Check,\n    expiration,\n  }: {\n    derivedPublicKeyBase58Check?: string;\n    expiration?: number;\n  }\n): Promise<string> => {\n  const header = JSON.stringify({ alg, typ: 'JWT' });\n  const issuedAt = Math.floor(Date.now() / 1000);\n  const thirtyMinFromNow = issuedAt + 30 * 60;\n  const payload = JSON.stringify({\n    ...(derivedPublicKeyBase58Check ? { derivedPublicKeyBase58Check } : {}),\n    iat: issuedAt,\n    exp: thirtyMinFromNow,\n  });\n\n  const jwt = `${urlSafeBase64(header)}.${urlSafeBase64(payload)}`;\n\n  const [signature] = await sign(\n    ecUtils.bytesToHex(await ecUtils.sha256(new TextEncoder().encode(jwt))),\n    ecUtils.hexToBytes(seedHex)\n  );\n  const encodedSignature = derToJoseEncoding(signature);\n\n  return `${jwt}.${encodedSignature}`;\n};\n\nfunction urlSafeBase64(str: string) {\n  return window\n    .btoa(str)\n    .replace(/\\+/g, '-')\n    .replace(/\\//g, '_')\n    .replace(/=/g, '');\n}\n\nexport const encrypt = async (\n  senderSeedHex: string,\n  recipientPublicKeyBase58Check: string,\n  plaintext: string\n): Promise<string> => {\n  const privateKey = ecUtils.hexToBytes(senderSeedHex);\n  const recipientPublicKey = await bs58PublicKeyToBytes(\n    recipientPublicKeyBase58Check\n  );\n  const sharedPrivateKey = await getSharedPrivateKey(\n    privateKey,\n    recipientPublicKey\n  );\n  const sharedPublicKey = getPublicKey(sharedPrivateKey);\n  const ephemPrivateKey = ecUtils.randomBytes(32);\n  const ephemPublicKey = getPublicKey(ephemPrivateKey);\n  const privKey = await getSharedPrivateKey(ephemPrivateKey, sharedPublicKey);\n  const encryptionKey = privKey.slice(0, 16);\n  const iv = ecUtils.randomBytes(16);\n  const macKey = await ecUtils.sha256(privKey.slice(16));\n  const bytes = new TextEncoder().encode(plaintext);\n  const cryptoKey = await window.crypto.subtle.importKey(\n    'raw',\n    encryptionKey,\n    'AES-CTR',\n    true,\n    ['encrypt']\n  );\n  const cipherBytes = await window.crypto.subtle.encrypt(\n    {\n      name: 'AES-CTR',\n      counter: iv,\n      length: 128,\n    },\n    cryptoKey,\n    bytes\n  );\n  const hmac = await ecUtils.hmacSha256(\n    macKey,\n    new Uint8Array([...iv, ...new Uint8Array(cipherBytes)])\n  );\n\n  return ecUtils.bytesToHex(\n    new Uint8Array([\n      ...ephemPublicKey,\n      ...iv,\n      ...new Uint8Array(cipherBytes),\n      ...hmac,\n    ])\n  );\n};\n\nexport const bs58PublicKeyToBytes = async (str: string) => {\n  const bytes = bs58.decode(str);\n  const payload = bytes.slice(0, -4);\n  const checksumA = bytes.slice(-4);\n  const checksumB = await sha256X2(payload);\n\n  if (\n    (checksumA[0] ^ checksumB[0]) |\n    (checksumA[1] ^ checksumB[1]) |\n    (checksumA[2] ^ checksumB[2]) |\n    (checksumA[3] ^ checksumB[3])\n  ) {\n    throw new Error('Invalid checksum');\n  }\n\n  return Point.fromHex(ecUtils.bytesToHex(payload.slice(3))).toRawBytes(false);\n};\n\nconst isValidHmac = (candidate: Uint8Array, knownGood: Uint8Array) => {\n  if (candidate.length !== knownGood.length) {\n    return false;\n  }\n\n  for (let i = 0; i < knownGood.length; i++) {\n    if (candidate[i] !== knownGood[i]) {\n      return false;\n    }\n  }\n\n  return true;\n};\n\nexport const decrypt = async (\n  recipientSeedHex: string,\n  senderPublicKeyBase58Check: string,\n  cipherTextHex: string\n) => {\n  const cipherBytes = ecUtils.hexToBytes(cipherTextHex);\n  const metaLength = 113;\n\n  if (cipherBytes.length < metaLength) {\n    throw new Error('invalid cipher text. data too small.');\n  }\n\n  if (!(cipherBytes[0] >= 2 && cipherBytes[0] <= 4)) {\n    throw new Error('invalid cipher text.');\n  }\n\n  const privateKey = ecUtils.hexToBytes(recipientSeedHex);\n  const publicKey = await bs58PublicKeyToBytes(senderPublicKeyBase58Check);\n  const sharedPrivateKey = await getSharedPrivateKey(privateKey, publicKey);\n  const ephemPublicKey = cipherBytes.slice(0, 65);\n  const cipherTextLength = cipherBytes.length - metaLength;\n  const iv = cipherBytes.slice(65, 65 + 16);\n  const cipherAndIv = cipherBytes.slice(65, 65 + 16 + cipherTextLength);\n  const ciphertext = cipherAndIv.slice(16);\n  const msgMac = cipherBytes.slice(65 + 16 + cipherTextLength);\n  const privKey = await getSharedPrivateKey(sharedPrivateKey, ephemPublicKey);\n  const encryptionKey = privKey.slice(0, 16);\n  const macKey = await ecUtils.sha256(privKey.slice(16));\n  const hmacKnownGood = await ecUtils.hmacSha256(macKey, cipherAndIv);\n\n  if (!isValidHmac(msgMac, hmacKnownGood)) throw new Error('incorrect MAC');\n\n  const cryptoKey = await globalThis.crypto.subtle.importKey(\n    'raw',\n    encryptionKey,\n    'AES-CTR',\n    true,\n    ['decrypt']\n  );\n\n  const decryptedBuffer = await globalThis.crypto.subtle.decrypt(\n    { name: 'AES-CTR', counter: iv, length: 128 },\n    cryptoKey,\n    ciphertext\n  );\n\n  return new TextDecoder().decode(decryptedBuffer);\n};\n\nexport const getSharedPrivateKey = async (\n  privKey: Uint8Array,\n  pubKey: Uint8Array\n) => {\n  const sharedSecret = await getSharedSecret(privKey, pubKey);\n\n  return kdf(sharedSecret, 32);\n};\n\nexport const decodePublicKey = async (publicKeyBase58Check: string) => {\n  const decoded = await bs58PublicKeyToBytes(publicKeyBase58Check);\n  const withPrefixRemoved = decoded.slice(3);\n  const senderPubKeyHex = ecUtils.bytesToHex(withPrefixRemoved);\n\n  return Point.fromHex(senderPubKeyHex).toRawBytes(false);\n};\n\nexport const getSharedSecret = async (\n  privKey: Uint8Array,\n  pubKey: Uint8Array\n) => {\n  // passing true to compress the public key, and then slicing off the first byte\n  // matches the implementation of derive in the elliptic package.\n  // https://github.com/paulmillr/noble-secp256k1/issues/28#issuecomment-946538037\n  return nobleGetSharedSecret(privKey, pubKey, true).slice(1);\n};\n\n// taken from reference implementation in the deso chat app:\n// https://github.com/deso-protocol/access-group-messaging-app/blob/cd5c237f5e5729196aac0da161d0851bde78092c/src/services/crypto-utils.service.tsx#L91\nexport const kdf = async (secret: Uint8Array, outputLength: number) => {\n  let ctr = 1;\n  let written = 0;\n  let result = new Uint8Array();\n\n  while (written < outputLength) {\n    const hash = await ecUtils.sha256(\n      new Uint8Array([\n        ...new Uint8Array([ctr >> 24, ctr >> 16, ctr >> 8, ctr]),\n        ...secret,\n      ])\n    );\n    result = new Uint8Array([...result, ...hash]);\n    written += 32;\n    ctr += 1;\n  }\n\n  return result;\n};\n\n// This is a modified version of the derToJose function from\n// https://github.com/Brightspace/node-ecdsa-sig-formatter/blob/ca25a2fd5ae9dd85036081632936e802a47a1289/src/ecdsa-sig-formatter.js#L32\n// The original package is not browser friendly and requires node polyfills. We\n// also don't need to be quite as defensive as the original package since we\n// have full control of the input.\nfunction derToJoseEncoding(signature: Uint8Array) {\n  const paramBytes = 32;\n\n  let offset = 3;\n  const rLength = signature[offset];\n  offset += 1;\n  const rOffset = offset;\n  offset += rLength + 1;\n  const sLength = signature[offset];\n  offset += 1;\n  const sOffset = offset;\n  offset += sLength;\n\n  const rPadding = paramBytes - rLength;\n  const sPadding = paramBytes - sLength;\n\n  const outPut = new Uint8Array(rPadding + rLength + sPadding + sLength);\n\n  for (offset = 0; offset < rPadding; ++offset) {\n    outPut[offset] = 0;\n  }\n\n  outPut.set(\n    signature.slice(rOffset + Math.max(-rPadding, 0), rOffset + rLength),\n    offset\n  );\n\n  offset = paramBytes;\n\n  for (const o = offset; offset < o + sPadding; ++offset) {\n    outPut[offset] = 0;\n  }\n\n  outPut.set(\n    signature.slice(sOffset + Math.max(-sPadding, 0), sOffset + sLength),\n    offset\n  );\n\n  const outputChars = outPut.reduce(\n    (data, byte) => data + String.fromCharCode(byte),\n    ''\n  );\n\n  return urlSafeBase64(outputChars);\n}\n"]} | ||
async function deriveAccessGroupKeyPair(privateKeyHex, groupKeyName) { | ||
const secretHash = await (0, exports.sha256X2)(privateKeyHex); | ||
const keyNameHash = await (0, exports.sha256X2)(new TextEncoder().encode(groupKeyName)); | ||
const privateKey = await (0, exports.sha256X2)(new Uint8Array([...secretHash, ...keyNameHash])); | ||
return (0, exports.keygen)(privateKey); | ||
} | ||
exports.deriveAccessGroupKeyPair = deriveAccessGroupKeyPair; | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"crypto-utils.js","sourceRoot":"","sources":["../../../../../libs/identity/src/lib/crypto-utils.ts"],"names":[],"mappings":";;;AAAA,gDAM0B;AAC1B,6BAA6B;AAC7B,2CAAkD;AAGlD,oDAAoD;AACpD,SAAS,iBAAiB,CAAC,MAAoB,EAAE,MAAe;IAC9D,IAAI,MAAM,KAAK,SAAS,EAAE;QACxB,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE,CAAC,GAAG,GAAG,KAAK,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;KAC/D;IACD,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC,MAAM,CAAC,CAAC;IACtC,IAAI,MAAM,GAAG,CAAC,CAAC;IACf,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;QACtC,IAAI,MAAM,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,GAAG,MAAM,EAAE;YACtC,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,MAAM,GAAG,MAAM,CAAC,EAAE,MAAM,CAAC,CAAC;YACxD,MAAM;SACP;QACD,MAAM,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,MAAM,CAAC,CAAC;QAC9B,MAAM,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC;KAC5B;IACD,OAAO,MAAM,CAAC;AAChB,CAAC;AAED,MAAM,cAAc,GAAG,CAAC,IAAY,EAAc,EAAE;IAClD,MAAM,MAAM,GAAa,EAAE,CAAC;IAC5B,OAAO,IAAI,IAAI,IAAI,EAAE;QACnB,MAAM,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;QAClE,IAAI,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;KAC1C;IACD,MAAM,CAAC,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC;IAEtB,OAAO,IAAI,UAAU,CAAC,MAAM,CAAC,CAAC;AAChC,CAAC,CAAC;AAMF,uEAAuE;AACvE,yEAAyE;AACzE,2EAA2E;AAC3E,yEAAyE;AACzE,6EAA6E;AAC7E,0EAA0E;AAC1E,kBAAkB;AAClB,EAAE;AACF,mCAAmC;AACnC,6CAA6C;AAC7C,EAAE;AACF,4EAA4E;AAC5E,6EAA6E;AAC7E,6EAA6E;AAC7E,aAAa;AACN,MAAM,MAAM,GAAG,CAAC,IAA0B,EAAW,EAAE;IAC5D,MAAM,UAAU,GAAG,IAAI,CAAC,CAAC,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,iBAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;IACxE,MAAM,OAAO,GAAG,iBAAO,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC;IAE/C,OAAO;QACL,OAAO;QACP,OAAO,EAAE,UAAU;QACnB,MAAM,EAAE,IAAA,wBAAY,EAAC,UAAU,EAAE,IAAI,CAAC,kBAAkB,CAAC;KAC1D,CAAC;AACJ,CAAC,CAAC;AATW,QAAA,MAAM,UASjB;AAEF,MAAM,aAAa,GAAG,CAAC,IAAyB,EAAc,EAAE;IAC9D,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE;QAC5B,OAAO,iBAAO,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;KACjC;SAAM;QACL,OAAO,IAAI,CAAC;KACb;AACH,CAAC,CAAC;AAEF;;;;GAIG;AACI,MAAM,QAAQ,GAAG,KAAK,EAC3B,IAAyB,EACJ,EAAE;IACvB,MAAM,CAAC,GAAG,OAAO,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,iBAAO,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;IACrE,OAAO,iBAAO,CAAC,MAAM,CAAC,MAAM,iBAAO,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;AACjD,CAAC,CAAC;AALW,QAAA,QAAQ,YAKnB;AAEK,MAAM,sBAAsB,GAAG,KAAK,EACzC,cAA0B,EAC1B,OAA4B,EACX,EAAE;;IACnB,MAAM,MAAM,GAAG,+BAAmB,CAAC,MAAA,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,OAAO,mCAAI,SAAS,CAAC,CAAC,IAAI,CAAC;IACvE,0EAA0E;IAC1E,2EAA2E;IAC3E,uGAAuG;IACvG,MAAM,KAAK,GAAG,IAAI,UAAU,CAAC,CAAC,GAAG,MAAM,EAAE,GAAG,cAAc,CAAC,CAAC,CAAC;IAC7D,MAAM,QAAQ,GAAG,MAAM,IAAA,gBAAQ,EAAC,KAAK,CAAC,CAAC;IACvC,OAAO,IAAI,CAAC,MAAM,CAAC,iBAAiB,CAAC,CAAC,KAAK,EAAE,QAAQ,CAAC,EAAE,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,CAAC;AAC7E,CAAC,CAAC;AAXW,QAAA,sBAAsB,0BAWjC;AAMF,MAAM,IAAI,GAAG,CAAC,UAAkB,EAAE,UAAsB,EAAE,EAAE;IAC1D,OAAO,IAAA,gBAAM,EAAC,UAAU,EAAE,UAAU,EAAE;QACpC,iHAAiH;QACjH,SAAS,EAAE,IAAI;QACf,GAAG,EAAE,IAAI;QACT,YAAY,EAAE,IAAI;QAClB,SAAS,EAAE,IAAI;KAChB,CAAC,CAAC;AACL,CAAC,CAAC;AAEK,MAAM,MAAM,GAAG,KAAK,EACzB,KAAa,EACb,OAAe,EACf,OAAqB,EACJ,EAAE;IACnB,MAAM,gBAAgB,GAAG,iBAAO,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;IACnD,MAAM,aAAa,GAAG,MAAM,IAAA,gBAAQ,EAAC,gBAAgB,CAAC,CAAC;IACvD,MAAM,kBAAkB,GAAG,iBAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IAC7D,MAAM,UAAU,GAAG,iBAAO,CAAC,UAAU,CAAC,OAAO,CAAC,CAAC;IAC/C,MAAM,CAAC,cAAc,EAAE,aAAa,CAAC,GAAG,MAAM,IAAI,CAChD,kBAAkB,EAClB,UAAU,CACX,CAAC;IAEF,MAAM,eAAe,GAAG,cAAc,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC;IAE9D,IAAI,OAAO,aAAP,OAAO,uBAAP,OAAO,CAAE,YAAY,EAAE;QACzB,cAAc,CAAC,CAAC,CAAC,IAAI,CAAC,GAAG,aAAa,CAAC;KACxC;IAED,MAAM,sBAAsB,GAAG,iBAAO,CAAC,WAAW,CAChD,gBAAgB,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,EAC7B,eAAe,EACf,cAAc,CACf,CAAC;IAEF,OAAO,iBAAO,CAAC,UAAU,CAAC,sBAAsB,CAAC,CAAC;AACpD,CAAC,CAAC;AA3BW,QAAA,MAAM,UA2BjB;AAEK,MAAM,YAAY,GAAG,KAAK,EAC/B,OAAe,EACf,GAAiB,EACjB,EACE,2BAA2B,EAC3B,UAAU,GAIX,EACgB,EAAE;IACnB,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,EAAE,GAAG,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC;IACnD,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC;IAC/C,MAAM,gBAAgB,GAAG,QAAQ,GAAG,EAAE,GAAG,EAAE,CAAC;IAC5C,MAAM,OAAO,GAAG,IAAI,CAAC,SAAS,CAAC;QAC7B,GAAG,CAAC,2BAA2B,CAAC,CAAC,CAAC,EAAE,2BAA2B,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC;QACvE,GAAG,EAAE,QAAQ;QACb,GAAG,EAAE,gBAAgB;KACtB,CAAC,CAAC;IAEH,MAAM,GAAG,GAAG,GAAG,aAAa,CAAC,MAAM,CAAC,IAAI,aAAa,CAAC,OAAO,CAAC,EAAE,CAAC;IAEjE,MAAM,CAAC,SAAS,CAAC,GAAG,MAAM,IAAI,CAC5B,iBAAO,CAAC,UAAU,CAAC,MAAM,iBAAO,CAAC,MAAM,CAAC,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EACvE,iBAAO,CAAC,UAAU,CAAC,OAAO,CAAC,CAC5B,CAAC;IACF,MAAM,gBAAgB,GAAG,iBAAiB,CAAC,SAAS,CAAC,CAAC;IAEtD,OAAO,GAAG,GAAG,IAAI,gBAAgB,EAAE,CAAC;AACtC,CAAC,CAAC;AA7BW,QAAA,YAAY,gBA6BvB;AAEF,SAAS,aAAa,CAAC,GAAW;IAChC,OAAO,MAAM;SACV,IAAI,CAAC,GAAG,CAAC;SACT,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC;SACnB,OAAO,CAAC,KAAK,EAAE,GAAG,CAAC;SACnB,OAAO,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC;AACvB,CAAC;AAEM,MAAM,kBAAkB,GAAG,KAAK,EACrC,aAAqB,EACrB,6BAAqC,EACrC,OAAe,EACf,EAAE;IACF,MAAM,UAAU,GAAG,iBAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IACrD,MAAM,kBAAkB,GAAG,MAAM,IAAA,4BAAoB,EACnD,6BAA6B,CAC9B,CAAC;IACF,MAAM,gBAAgB,GAAG,MAAM,IAAA,2BAAmB,EAChD,UAAU,EACV,kBAAkB,CACnB,CAAC;IACF,MAAM,eAAe,GAAG,IAAA,wBAAY,EAAC,gBAAgB,CAAC,CAAC;IAEvD,OAAO,IAAA,eAAO,EAAC,eAAe,EAAE,OAAO,CAAC,CAAC;AAC3C,CAAC,CAAC;AAhBW,QAAA,kBAAkB,sBAgB7B;AAEF;;;;GAIG;AACI,MAAM,OAAO,GAAG,KAAK,EAC1B,SAA8B,EAC9B,SAAiB,EACA,EAAE;IACnB,MAAM,eAAe,GAAG,iBAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;IAChD,MAAM,cAAc,GAAG,IAAA,wBAAY,EAAC,eAAe,CAAC,CAAC;IACrD,MAAM,cAAc,GAClB,OAAO,SAAS,KAAK,QAAQ;QAC3B,CAAC,CAAC,MAAM,IAAA,4BAAoB,EAAC,SAAS,CAAC;QACvC,CAAC,CAAC,SAAS,CAAC;IAChB,MAAM,OAAO,GAAG,MAAM,IAAA,2BAAmB,EAAC,eAAe,EAAE,cAAc,CAAC,CAAC;IAC3E,MAAM,aAAa,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAC3C,MAAM,EAAE,GAAG,iBAAO,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;IACnC,MAAM,MAAM,GAAG,MAAM,iBAAO,CAAC,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;IACvD,MAAM,KAAK,GAAG,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;IAClD,MAAM,SAAS,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,CACpD,KAAK,EACL,aAAa,EACb,SAAS,EACT,IAAI,EACJ,CAAC,SAAS,CAAC,CACZ,CAAC;IACF,MAAM,WAAW,GAAG,MAAM,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CACpD;QACE,IAAI,EAAE,SAAS;QACf,OAAO,EAAE,EAAE;QACX,MAAM,EAAE,GAAG;KACZ,EACD,SAAS,EACT,KAAK,CACN,CAAC;IACF,MAAM,IAAI,GAAG,MAAM,iBAAO,CAAC,UAAU,CACnC,MAAM,EACN,IAAI,UAAU,CAAC,CAAC,GAAG,EAAE,EAAE,GAAG,IAAI,UAAU,CAAC,WAAW,CAAC,CAAC,CAAC,CACxD,CAAC;IAEF,OAAO,iBAAO,CAAC,UAAU,CACvB,IAAI,UAAU,CAAC;QACb,GAAG,cAAc;QACjB,GAAG,EAAE;QACL,GAAG,IAAI,UAAU,CAAC,WAAW,CAAC;QAC9B,GAAG,IAAI;KACR,CAAC,CACH,CAAC;AACJ,CAAC,CAAC;AA5CW,QAAA,OAAO,WA4ClB;AAEK,MAAM,oBAAoB,GAAG,KAAK,EAAE,GAAW,EAAE,EAAE;IACxD,MAAM,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;IAC/B,MAAM,OAAO,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC;IACnC,MAAM,SAAS,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;IAClC,MAAM,SAAS,GAAG,MAAM,IAAA,gBAAQ,EAAC,OAAO,CAAC,CAAC;IAE1C,IACE,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;QAC7B,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;QAC7B,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC;QAC7B,CAAC,SAAS,CAAC,CAAC,CAAC,GAAG,SAAS,CAAC,CAAC,CAAC,CAAC,EAC7B;QACA,MAAM,IAAI,KAAK,CAAC,kBAAkB,CAAC,CAAC;KACrC;IAED,OAAO,iBAAK,CAAC,OAAO,CAAC,iBAAO,CAAC,UAAU,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;AAC/E,CAAC,CAAC;AAhBW,QAAA,oBAAoB,wBAgB/B;AAEF,MAAM,WAAW,GAAG,CAAC,SAAqB,EAAE,SAAqB,EAAE,EAAE;IACnE,IAAI,SAAS,CAAC,MAAM,KAAK,SAAS,CAAC,MAAM,EAAE;QACzC,OAAO,KAAK,CAAC;KACd;IAED,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,SAAS,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;QACzC,IAAI,SAAS,CAAC,CAAC,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE;YACjC,OAAO,KAAK,CAAC;SACd;KACF;IAED,OAAO,IAAI,CAAC;AACd,CAAC,CAAC;AAEK,MAAM,kBAAkB,GAAG,KAAK,EACrC,gBAAwB,EACxB,mBAA2B,EAC3B,aAAqB,EACrB,EAAE;IACF,MAAM,UAAU,GAAG,iBAAO,CAAC,UAAU,CAAC,gBAAgB,CAAC,CAAC;IACxD,MAAM,SAAS,GAAG,MAAM,IAAA,4BAAoB,EAAC,mBAAmB,CAAC,CAAC;IAClE,MAAM,gBAAgB,GAAG,MAAM,IAAA,2BAAmB,EAAC,UAAU,EAAE,SAAS,CAAC,CAAC;IAC1E,OAAO,IAAA,eAAO,EAAC,gBAAgB,EAAE,aAAa,CAAC,CAAC;AAClD,CAAC,CAAC;AATW,QAAA,kBAAkB,sBAS7B;AAEK,MAAM,OAAO,GAAG,KAAK,EAC1B,oBAAyC,EACzC,aAAqB,EACrB,EAAE;IACF,MAAM,WAAW,GAAG,iBAAO,CAAC,UAAU,CAAC,aAAa,CAAC,CAAC;IACtD,MAAM,UAAU,GAAG,GAAG,CAAC;IAEvB,IAAI,WAAW,CAAC,MAAM,GAAG,UAAU,EAAE;QACnC,MAAM,IAAI,KAAK,CAAC,sCAAsC,CAAC,CAAC;KACzD;IAED,IAAI,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,EAAE;QACjD,MAAM,IAAI,KAAK,CAAC,sBAAsB,CAAC,CAAC;KACzC;IAED,MAAM,UAAU,GAAG,aAAa,CAAC,oBAAoB,CAAC,CAAC;IACvD,MAAM,cAAc,GAAG,WAAW,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IAChD,MAAM,gBAAgB,GAAG,WAAW,CAAC,MAAM,GAAG,UAAU,CAAC;IACzD,MAAM,EAAE,GAAG,WAAW,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,GAAG,EAAE,CAAC,CAAC;IAC1C,MAAM,WAAW,GAAG,WAAW,CAAC,KAAK,CAAC,EAAE,EAAE,EAAE,GAAG,EAAE,GAAG,gBAAgB,CAAC,CAAC;IACtE,MAAM,UAAU,GAAG,WAAW,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;IACzC,MAAM,MAAM,GAAG,WAAW,CAAC,KAAK,CAAC,EAAE,GAAG,EAAE,GAAG,gBAAgB,CAAC,CAAC;IAC7D,MAAM,eAAe,GAAG,MAAM,IAAA,2BAAmB,EAAC,UAAU,EAAE,cAAc,CAAC,CAAC;IAC9E,MAAM,aAAa,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;IACnD,MAAM,MAAM,GAAG,MAAM,iBAAO,CAAC,MAAM,CAAC,eAAe,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,CAAC;IAC/D,MAAM,aAAa,GAAG,MAAM,iBAAO,CAAC,UAAU,CAAC,MAAM,EAAE,WAAW,CAAC,CAAC;IAEpE,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE,aAAa,CAAC,EAAE;QACvC,MAAM,IAAI,KAAK,CAAC,eAAe,CAAC,CAAC;KAClC;IAED,MAAM,SAAS,GAAG,MAAM,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,CACxD,KAAK,EACL,aAAa,EACb,SAAS,EACT,IAAI,EACJ,CAAC,SAAS,CAAC,CACZ,CAAC;IAEF,MAAM,eAAe,GAAG,MAAM,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,OAAO,CAC5D,EAAE,IAAI,EAAE,SAAS,EAAE,OAAO,EAAE,EAAE,EAAE,MAAM,EAAE,GAAG,EAAE,EAC7C,SAAS,EACT,UAAU,CACX,CAAC;IAEF,OAAO,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,eAAe,CAAC,CAAC;AACnD,CAAC,CAAC;AA9CW,QAAA,OAAO,WA8ClB;AAEK,MAAM,mBAAmB,GAAG,KAAK,EACtC,OAAmB,EACnB,MAAkB,EAClB,EAAE;IACF,MAAM,YAAY,GAAG,MAAM,IAAA,uBAAe,EAAC,OAAO,EAAE,MAAM,CAAC,CAAC;IAE5D,OAAO,IAAA,WAAG,EAAC,YAAY,EAAE,EAAE,CAAC,CAAC;AAC/B,CAAC,CAAC;AAPW,QAAA,mBAAmB,uBAO9B;AAEK,MAAM,eAAe,GAAG,KAAK,EAAE,oBAA4B,EAAE,EAAE;IACpE,MAAM,OAAO,GAAG,MAAM,IAAA,4BAAoB,EAAC,oBAAoB,CAAC,CAAC;IACjE,MAAM,iBAAiB,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC3C,MAAM,eAAe,GAAG,iBAAO,CAAC,UAAU,CAAC,iBAAiB,CAAC,CAAC;IAE9D,OAAO,iBAAK,CAAC,OAAO,CAAC,eAAe,CAAC,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;AAC1D,CAAC,CAAC;AANW,QAAA,eAAe,mBAM1B;AAEK,MAAM,eAAe,GAAG,KAAK,EAClC,OAAmB,EACnB,MAAkB,EAClB,EAAE;IACF,+EAA+E;IAC/E,gEAAgE;IAChE,gFAAgF;IAChF,OAAO,IAAA,2BAAoB,EAAC,OAAO,EAAE,MAAM,EAAE,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;AAC9D,CAAC,CAAC;AARW,QAAA,eAAe,mBAQ1B;AAEF,4DAA4D;AAC5D,sJAAsJ;AAC/I,MAAM,GAAG,GAAG,KAAK,EAAE,MAAkB,EAAE,YAAoB,EAAE,EAAE;IACpE,IAAI,GAAG,GAAG,CAAC,CAAC;IACZ,IAAI,OAAO,GAAG,CAAC,CAAC;IAChB,IAAI,MAAM,GAAG,IAAI,UAAU,EAAE,CAAC;IAE9B,OAAO,OAAO,GAAG,YAAY,EAAE;QAC7B,MAAM,IAAI,GAAG,MAAM,iBAAO,CAAC,MAAM,CAC/B,IAAI,UAAU,CAAC;YACb,GAAG,IAAI,UAAU,CAAC,CAAC,GAAG,IAAI,EAAE,EAAE,GAAG,IAAI,EAAE,EAAE,GAAG,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC;YACxD,GAAG,MAAM;SACV,CAAC,CACH,CAAC;QACF,MAAM,GAAG,IAAI,UAAU,CAAC,CAAC,GAAG,MAAM,EAAE,GAAG,IAAI,CAAC,CAAC,CAAC;QAC9C,OAAO,IAAI,EAAE,CAAC;QACd,GAAG,IAAI,CAAC,CAAC;KACV;IAED,OAAO,MAAM,CAAC;AAChB,CAAC,CAAC;AAlBW,QAAA,GAAG,OAkBd;AAEF,4DAA4D;AAC5D,uIAAuI;AACvI,+EAA+E;AAC/E,4EAA4E;AAC5E,kCAAkC;AAClC,SAAS,iBAAiB,CAAC,SAAqB;IAC9C,MAAM,UAAU,GAAG,EAAE,CAAC;IAEtB,IAAI,MAAM,GAAG,CAAC,CAAC;IACf,MAAM,OAAO,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;IAClC,MAAM,IAAI,CAAC,CAAC;IACZ,MAAM,OAAO,GAAG,MAAM,CAAC;IACvB,MAAM,IAAI,OAAO,GAAG,CAAC,CAAC;IACtB,MAAM,OAAO,GAAG,SAAS,CAAC,MAAM,CAAC,CAAC;IAClC,MAAM,IAAI,CAAC,CAAC;IACZ,MAAM,OAAO,GAAG,MAAM,CAAC;IACvB,MAAM,IAAI,OAAO,CAAC;IAElB,MAAM,QAAQ,GAAG,UAAU,GAAG,OAAO,CAAC;IACtC,MAAM,QAAQ,GAAG,UAAU,GAAG,OAAO,CAAC;IAEtC,MAAM,MAAM,GAAG,IAAI,UAAU,CAAC,QAAQ,GAAG,OAAO,GAAG,QAAQ,GAAG,OAAO,CAAC,CAAC;IAEvE,KAAK,MAAM,GAAG,CAAC,EAAE,MAAM,GAAG,QAAQ,EAAE,EAAE,MAAM,EAAE;QAC5C,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;KACpB;IAED,MAAM,CAAC,GAAG,CACR,SAAS,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,EAAE,OAAO,GAAG,OAAO,CAAC,EACpE,MAAM,CACP,CAAC;IAEF,MAAM,GAAG,UAAU,CAAC;IAEpB,KAAK,MAAM,CAAC,GAAG,MAAM,EAAE,MAAM,GAAG,CAAC,GAAG,QAAQ,EAAE,EAAE,MAAM,EAAE;QACtD,MAAM,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;KACpB;IAED,MAAM,CAAC,GAAG,CACR,SAAS,CAAC,KAAK,CAAC,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,CAAC,CAAC,EAAE,OAAO,GAAG,OAAO,CAAC,EACpE,MAAM,CACP,CAAC;IAEF,MAAM,WAAW,GAAG,MAAM,CAAC,MAAM,CAC/B,CAAC,IAAI,EAAE,IAAI,EAAE,EAAE,CAAC,IAAI,GAAG,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,EAChD,EAAE,CACH,CAAC;IAEF,OAAO,aAAa,CAAC,WAAW,CAAC,CAAC;AACpC,CAAC;AAEM,KAAK,UAAU,wBAAwB,CAC5C,aAAqB,EACrB,YAAoB;IAEpB,MAAM,UAAU,GAAG,MAAM,IAAA,gBAAQ,EAAC,aAAa,CAAC,CAAC;IACjD,MAAM,WAAW,GAAG,MAAM,IAAA,gBAAQ,EAAC,IAAI,WAAW,EAAE,CAAC,MAAM,CAAC,YAAY,CAAC,CAAC,CAAC;IAC3E,MAAM,UAAU,GAAG,MAAM,IAAA,gBAAQ,EAC/B,IAAI,UAAU,CAAC,CAAC,GAAG,UAAU,EAAE,GAAG,WAAW,CAAC,CAAC,CAChD,CAAC;IAEF,OAAO,IAAA,cAAM,EAAC,UAAU,CAAC,CAAC;AAC5B,CAAC;AAXD,4DAWC","sourcesContent":["import {\n  getPublicKey,\n  getSharedSecret as nobleGetSharedSecret,\n  Point,\n  sign as ecSign,\n  utils as ecUtils,\n} from '@noble/secp256k1';\nimport * as bs58 from 'bs58';\nimport { PUBLIC_KEY_PREFIXES } from './constants';\nimport { jwtAlgorithm, KeyPair, Network } from './types';\n\n// Browser friendly version of node's Buffer.concat.\nfunction concatUint8Arrays(arrays: Uint8Array[], length?: number) {\n  if (length === undefined) {\n    length = arrays.reduce((acc, array) => acc + array.length, 0);\n  }\n  const result = new Uint8Array(length);\n  let offset = 0;\n  for (let i = 0; i < arrays.length; i++) {\n    if (offset + arrays[i].length > length) {\n      result.set(arrays[i].slice(0, length - offset), offset);\n      break;\n    }\n    result.set(arrays[i], offset);\n    offset += arrays[i].length;\n  }\n  return result;\n}\n\nconst uvarint64ToBuf = (uint: number): Uint8Array => {\n  const result: number[] = [];\n  while (uint >= 0x80) {\n    result.push(Number((BigInt(uint) & BigInt(0xff)) | BigInt(0x80)));\n    uint = Number(BigInt(uint) >> BigInt(7));\n  }\n  result.push(uint | 0);\n\n  return new Uint8Array(result);\n};\n\ninterface Base58CheckOptions {\n  network: Network;\n}\n\n// We are not using the native web crypto API to actually generate keys\n// because it does not support the secp256k1 curve. Instead, we are using\n// https://github.com/paulmillr/noble-secp256k1 which is a browser friendly\n// alternative to the node elliptic package which is far smaller and only\n// focuses on supporting the ec algorithm we are actually interested in here.\n// If the web crypto API ever adds support for secp256k1, we should change\n// this to use it.\n//\n// See the following for more info:\n// https://github.com/w3c/webcrypto/issues/82\n//\n// If you don't provide a seed, a random one will be generated for you and a\n// random key pair will be returned. If you do provide a seed, it should be a\n// randomly generated 32 byte value (Uint8Array of length 32 or hex string of\n// length 64)\nexport const keygen = (seed?: string | Uint8Array): KeyPair => {\n  const privateKey = seed ? normalizeSeed(seed) : ecUtils.randomBytes(32);\n  const seedHex = ecUtils.bytesToHex(privateKey);\n\n  return {\n    seedHex,\n    private: privateKey,\n    public: getPublicKey(privateKey, true /* isCompressed */),\n  };\n};\n\nconst normalizeSeed = (seed: string | Uint8Array): Uint8Array => {\n  if (typeof seed === 'string') {\n    return ecUtils.hexToBytes(seed);\n  } else {\n    return seed;\n  }\n};\n\n/**\n *\n * @param data could be a hex string or a byte array (Uint8Array)\n * @returns\n */\nexport const sha256X2 = async (\n  data: Uint8Array | string\n): Promise<Uint8Array> => {\n  const d = typeof data === 'string' ? ecUtils.hexToBytes(data) : data;\n  return ecUtils.sha256(await ecUtils.sha256(d));\n};\n\nexport const publicKeyToBase58Check = async (\n  publicKeyBytes: Uint8Array,\n  options?: Base58CheckOptions\n): Promise<string> => {\n  const prefix = PUBLIC_KEY_PREFIXES[options?.network ?? 'mainnet'].deso;\n  // This is the same as the implementation in the bs58check package, but we\n  // slightly modify it to use the browser friendly version of Buffer.concat.\n  // See: https://github.com/bitcoinjs/bs58check/blob/12b3e700f355c5c49d0be3f8fc29be6c66e753e9/base.js#L1\n  const bytes = new Uint8Array([...prefix, ...publicKeyBytes]);\n  const checksum = await sha256X2(bytes);\n  return bs58.encode(concatUint8Arrays([bytes, checksum], bytes.length + 4));\n};\n\nexport interface SignOptions {\n  isDerivedKey: boolean;\n}\n\nconst sign = (msgHashHex: string, privateKey: Uint8Array) => {\n  return ecSign(msgHashHex, privateKey, {\n    // For details about the signing options see: https://github.com/paulmillr/noble-secp256k1#signmsghash-privatekey\n    canonical: true,\n    der: true,\n    extraEntropy: true,\n    recovered: true,\n  });\n};\n\nexport const signTx = async (\n  txHex: string,\n  seedHex: string,\n  options?: SignOptions\n): Promise<string> => {\n  const transactionBytes = ecUtils.hexToBytes(txHex);\n  const hashedTxBytes = await sha256X2(transactionBytes);\n  const transactionHashHex = ecUtils.bytesToHex(hashedTxBytes);\n  const privateKey = ecUtils.hexToBytes(seedHex);\n  const [signatureBytes, recoveryParam] = await sign(\n    transactionHashHex,\n    privateKey\n  );\n\n  const signatureLength = uvarint64ToBuf(signatureBytes.length);\n\n  if (options?.isDerivedKey) {\n    signatureBytes[0] += 1 + recoveryParam;\n  }\n\n  const signedTransactionBytes = ecUtils.concatBytes(\n    transactionBytes.slice(0, -1),\n    signatureLength,\n    signatureBytes\n  );\n\n  return ecUtils.bytesToHex(signedTransactionBytes);\n};\n\nexport const getSignedJWT = async (\n  seedHex: string,\n  alg: jwtAlgorithm,\n  {\n    derivedPublicKeyBase58Check,\n    expiration,\n  }: {\n    derivedPublicKeyBase58Check?: string;\n    expiration?: number;\n  }\n): Promise<string> => {\n  const header = JSON.stringify({ alg, typ: 'JWT' });\n  const issuedAt = Math.floor(Date.now() / 1000);\n  const thirtyMinFromNow = issuedAt + 30 * 60;\n  const payload = JSON.stringify({\n    ...(derivedPublicKeyBase58Check ? { derivedPublicKeyBase58Check } : {}),\n    iat: issuedAt,\n    exp: thirtyMinFromNow,\n  });\n\n  const jwt = `${urlSafeBase64(header)}.${urlSafeBase64(payload)}`;\n\n  const [signature] = await sign(\n    ecUtils.bytesToHex(await ecUtils.sha256(new TextEncoder().encode(jwt))),\n    ecUtils.hexToBytes(seedHex)\n  );\n  const encodedSignature = derToJoseEncoding(signature);\n\n  return `${jwt}.${encodedSignature}`;\n};\n\nfunction urlSafeBase64(str: string) {\n  return window\n    .btoa(str)\n    .replace(/\\+/g, '-')\n    .replace(/\\//g, '_')\n    .replace(/=/g, '');\n}\n\nexport const encryptChatMessage = async (\n  senderSeedHex: string,\n  recipientPublicKeyBase58Check: string,\n  message: string\n) => {\n  const privateKey = ecUtils.hexToBytes(senderSeedHex);\n  const recipientPublicKey = await bs58PublicKeyToBytes(\n    recipientPublicKeyBase58Check\n  );\n  const sharedPrivateKey = await getSharedPrivateKey(\n    privateKey,\n    recipientPublicKey\n  );\n  const sharedPublicKey = getPublicKey(sharedPrivateKey);\n\n  return encrypt(sharedPublicKey, message);\n};\n\n/**\n * @param publicEncryptionKey could be in raw bytes or base58check format\n * @param plaintext\n * @returns cipher text as a hex string\n */\nexport const encrypt = async (\n  publicKey: Uint8Array | string,\n  plaintext: string\n): Promise<string> => {\n  const ephemPrivateKey = ecUtils.randomBytes(32);\n  const ephemPublicKey = getPublicKey(ephemPrivateKey);\n  const publicKeyBytes =\n    typeof publicKey === 'string'\n      ? await bs58PublicKeyToBytes(publicKey)\n      : publicKey;\n  const privKey = await getSharedPrivateKey(ephemPrivateKey, publicKeyBytes);\n  const encryptionKey = privKey.slice(0, 16);\n  const iv = ecUtils.randomBytes(16);\n  const macKey = await ecUtils.sha256(privKey.slice(16));\n  const bytes = new TextEncoder().encode(plaintext);\n  const cryptoKey = await window.crypto.subtle.importKey(\n    'raw',\n    encryptionKey,\n    'AES-CTR',\n    true,\n    ['encrypt']\n  );\n  const cipherBytes = await window.crypto.subtle.encrypt(\n    {\n      name: 'AES-CTR',\n      counter: iv,\n      length: 128,\n    },\n    cryptoKey,\n    bytes\n  );\n  const hmac = await ecUtils.hmacSha256(\n    macKey,\n    new Uint8Array([...iv, ...new Uint8Array(cipherBytes)])\n  );\n\n  return ecUtils.bytesToHex(\n    new Uint8Array([\n      ...ephemPublicKey,\n      ...iv,\n      ...new Uint8Array(cipherBytes),\n      ...hmac,\n    ])\n  );\n};\n\nexport const bs58PublicKeyToBytes = async (str: string) => {\n  const bytes = bs58.decode(str);\n  const payload = bytes.slice(0, -4);\n  const checksumA = bytes.slice(-4);\n  const checksumB = await sha256X2(payload);\n\n  if (\n    (checksumA[0] ^ checksumB[0]) |\n    (checksumA[1] ^ checksumB[1]) |\n    (checksumA[2] ^ checksumB[2]) |\n    (checksumA[3] ^ checksumB[3])\n  ) {\n    throw new Error('Invalid checksum');\n  }\n\n  return Point.fromHex(ecUtils.bytesToHex(payload.slice(3))).toRawBytes(false);\n};\n\nconst isValidHmac = (candidate: Uint8Array, knownGood: Uint8Array) => {\n  if (candidate.length !== knownGood.length) {\n    return false;\n  }\n\n  for (let i = 0; i < knownGood.length; i++) {\n    if (candidate[i] !== knownGood[i]) {\n      return false;\n    }\n  }\n\n  return true;\n};\n\nexport const decryptChatMessage = async (\n  recipientSeedHex: string,\n  publicDecryptionKey: string,\n  cipherTextHex: string\n) => {\n  const privateKey = ecUtils.hexToBytes(recipientSeedHex);\n  const publicKey = await bs58PublicKeyToBytes(publicDecryptionKey);\n  const sharedPrivateKey = await getSharedPrivateKey(privateKey, publicKey);\n  return decrypt(sharedPrivateKey, cipherTextHex);\n};\n\nexport const decrypt = async (\n  privateDecryptionKey: Uint8Array | string,\n  cipherTextHex: string\n) => {\n  const cipherBytes = ecUtils.hexToBytes(cipherTextHex);\n  const metaLength = 113;\n\n  if (cipherBytes.length < metaLength) {\n    throw new Error('invalid cipher text. data too small.');\n  }\n\n  if (!(cipherBytes[0] >= 2 && cipherBytes[0] <= 4)) {\n    throw new Error('invalid cipher text.');\n  }\n\n  const privateKey = normalizeSeed(privateDecryptionKey);\n  const ephemPublicKey = cipherBytes.slice(0, 65);\n  const cipherTextLength = cipherBytes.length - metaLength;\n  const iv = cipherBytes.slice(65, 65 + 16);\n  const cipherAndIv = cipherBytes.slice(65, 65 + 16 + cipherTextLength);\n  const cipherText = cipherAndIv.slice(16);\n  const msgMac = cipherBytes.slice(65 + 16 + cipherTextLength);\n  const sharedSecretKey = await getSharedPrivateKey(privateKey, ephemPublicKey);\n  const encryptionKey = sharedSecretKey.slice(0, 16);\n  const macKey = await ecUtils.sha256(sharedSecretKey.slice(16));\n  const hmacKnownGood = await ecUtils.hmacSha256(macKey, cipherAndIv);\n\n  if (!isValidHmac(msgMac, hmacKnownGood)) {\n    throw new Error('incorrect MAC');\n  }\n\n  const cryptoKey = await globalThis.crypto.subtle.importKey(\n    'raw',\n    encryptionKey,\n    'AES-CTR',\n    true,\n    ['decrypt']\n  );\n\n  const decryptedBuffer = await globalThis.crypto.subtle.decrypt(\n    { name: 'AES-CTR', counter: iv, length: 128 },\n    cryptoKey,\n    cipherText\n  );\n\n  return new TextDecoder().decode(decryptedBuffer);\n};\n\nexport const getSharedPrivateKey = async (\n  privKey: Uint8Array,\n  pubKey: Uint8Array\n) => {\n  const sharedSecret = await getSharedSecret(privKey, pubKey);\n\n  return kdf(sharedSecret, 32);\n};\n\nexport const decodePublicKey = async (publicKeyBase58Check: string) => {\n  const decoded = await bs58PublicKeyToBytes(publicKeyBase58Check);\n  const withPrefixRemoved = decoded.slice(3);\n  const senderPubKeyHex = ecUtils.bytesToHex(withPrefixRemoved);\n\n  return Point.fromHex(senderPubKeyHex).toRawBytes(false);\n};\n\nexport const getSharedSecret = async (\n  privKey: Uint8Array,\n  pubKey: Uint8Array\n) => {\n  // passing true to compress the public key, and then slicing off the first byte\n  // matches the implementation of derive in the elliptic package.\n  // https://github.com/paulmillr/noble-secp256k1/issues/28#issuecomment-946538037\n  return nobleGetSharedSecret(privKey, pubKey, true).slice(1);\n};\n\n// taken from reference implementation in the deso chat app:\n// https://github.com/deso-protocol/access-group-messaging-app/blob/cd5c237f5e5729196aac0da161d0851bde78092c/src/services/crypto-utils.service.tsx#L91\nexport const kdf = async (secret: Uint8Array, outputLength: number) => {\n  let ctr = 1;\n  let written = 0;\n  let result = new Uint8Array();\n\n  while (written < outputLength) {\n    const hash = await ecUtils.sha256(\n      new Uint8Array([\n        ...new Uint8Array([ctr >> 24, ctr >> 16, ctr >> 8, ctr]),\n        ...secret,\n      ])\n    );\n    result = new Uint8Array([...result, ...hash]);\n    written += 32;\n    ctr += 1;\n  }\n\n  return result;\n};\n\n// This is a modified version of the derToJose function from\n// https://github.com/Brightspace/node-ecdsa-sig-formatter/blob/ca25a2fd5ae9dd85036081632936e802a47a1289/src/ecdsa-sig-formatter.js#L32\n// The original package is not browser friendly and requires node polyfills. We\n// also don't need to be quite as defensive as the original package since we\n// have full control of the input.\nfunction derToJoseEncoding(signature: Uint8Array) {\n  const paramBytes = 32;\n\n  let offset = 3;\n  const rLength = signature[offset];\n  offset += 1;\n  const rOffset = offset;\n  offset += rLength + 1;\n  const sLength = signature[offset];\n  offset += 1;\n  const sOffset = offset;\n  offset += sLength;\n\n  const rPadding = paramBytes - rLength;\n  const sPadding = paramBytes - sLength;\n\n  const outPut = new Uint8Array(rPadding + rLength + sPadding + sLength);\n\n  for (offset = 0; offset < rPadding; ++offset) {\n    outPut[offset] = 0;\n  }\n\n  outPut.set(\n    signature.slice(rOffset + Math.max(-rPadding, 0), rOffset + rLength),\n    offset\n  );\n\n  offset = paramBytes;\n\n  for (const o = offset; offset < o + sPadding; ++offset) {\n    outPut[offset] = 0;\n  }\n\n  outPut.set(\n    signature.slice(sOffset + Math.max(-sPadding, 0), sOffset + sLength),\n    offset\n  );\n\n  const outputChars = outPut.reduce(\n    (data, byte) => data + String.fromCharCode(byte),\n    ''\n  );\n\n  return urlSafeBase64(outputChars);\n}\n\nexport async function deriveAccessGroupKeyPair(\n  privateKeyHex: string,\n  groupKeyName: string\n): Promise<KeyPair> {\n  const secretHash = await sha256X2(privateKeyHex);\n  const keyNameHash = await sha256X2(new TextEncoder().encode(groupKeyName));\n  const privateKey = await sha256X2(\n    new Uint8Array([...secretHash, ...keyNameHash])\n  );\n\n  return keygen(privateKey);\n}\n"]} |
@@ -1,2 +0,3 @@ | ||
import { APIProvider, IdentityConfiguration, IdentityDerivePayload, LoginOptions, StoredUser, SubscriberNotification, TransactionSpendingLimitResponseOptions } from './types'; | ||
import { AccessGroupEntryResponse, DecryptedMessageEntryResponse, NewMessageEntryResponse, SubmitTransactionResponse } from 'deso-protocol-types'; | ||
import { AccessGroupPrivateInfo, APIProvider, IdentityConfiguration, IdentityDerivePayload, LoginOptions, StoredUser, SubscriberNotification, TransactionSpendingLimitResponseOptions } from './types'; | ||
export declare class Identity { | ||
@@ -171,3 +172,3 @@ #private; | ||
TransactionHex: string; | ||
}): Promise<any>; | ||
}): Promise<SubmitTransactionResponse>; | ||
/** | ||
@@ -187,5 +188,4 @@ * @deprecated Use signAndSubmit instead. Since we don't support unauthorized | ||
/** | ||
* Encrypt an arbitrary string using the sender's seed hex and the recipient's | ||
* public key. Typically this would make use of the sender's | ||
* messagingPrivateKey for building chat or messaging applications. | ||
* Encrypt an arbitrary string using the recipient's | ||
* public key. | ||
* | ||
@@ -196,4 +196,3 @@ * @example | ||
* | ||
* const cipherText = await identity.encrypt( | ||
* senderPrivateMessagingSeedHex, | ||
* const cipherText = await identity.encryptMessage( | ||
* recipientPublicKeyBase58Check, | ||
@@ -204,21 +203,26 @@ * message | ||
*/ | ||
encrypt(senderSeedHex: string, recipientPublicKeyBase58Check: string, plaintext: string): Promise<string>; | ||
encryptMessage(recipientPublicKeyBase58Check: string, messagePlainText: string): Promise<string>; | ||
/** | ||
* Decrypt a hex encoded encrypted message using the recipient's private seed | ||
* hex and the sender's public key. Typically this would make use of the | ||
* recipient's messagingPrivateKey for building chat or messaging | ||
* applications. | ||
* @param message This is a message object returned any of the messages | ||
* endpoints of the DeSo backend api, could be a DM or a Group message. | ||
* @param groups This is an array of group chats the user belongs to. This is | ||
* required to decrypt group messages. | ||
* @returns | ||
*/ | ||
decryptMessage(message: NewMessageEntryResponse, groups: AccessGroupEntryResponse[]): Promise<DecryptedMessageEntryResponse>; | ||
/** | ||
* Decrypts the encrypted access group private key that we will need to use to decrypt group messages. | ||
* | ||
* @example | ||
* ```typescript | ||
* const cipherTextHex = "df6f3ff4695edb569631af4494b9f05d86c7fb8572226a356e03678aa56b7875"; | ||
* @param encryptedKeyHex | ||
* @returns returns a promise that resolves t the decrypted key pair. | ||
*/ | ||
decryptAccessGroupKeyPair(encryptedKeyHex: string): Promise<import("./types").KeyPair>; | ||
/** | ||
* Generate a key pair for an access group. This is used to encrypt and | ||
* decrypt group messages. | ||
* | ||
* const plaintext = await identity.decrypt( | ||
* recipientPrivateMessagingSeedHex, | ||
* senderPublicKeyBase58Check, | ||
* cipherTextHex | ||
* ); | ||
* ``` | ||
* @param groupName the plaintext name of the group chat | ||
* @returns a promise that resolves to the new key info. | ||
*/ | ||
decrypt(recipientSeedHex: string, senderPublicKeyBase58Check: string, cipherTextHex: string): Promise<string>; | ||
accessGroupStandardDerivation(groupName: string): Promise<AccessGroupPrivateInfo>; | ||
/** | ||
@@ -225,0 +229,0 @@ * Get a jwt token signed by the derived key issued to the currently active user. This can be used to pass to |
@@ -39,2 +39,11 @@ "use strict"; | ||
}, []); | ||
if (result.AccessGroupLimitMap) { | ||
result.AccessGroupLimitMap = Object.values(result.AccessGroupLimitMap); | ||
} | ||
if (result.AccessGroupMemberLimitMap) { | ||
result.AccessGroupMemberLimitMap = Object.values(result.AccessGroupMemberLimitMap); | ||
} | ||
if (result.AssociationLimitMap) { | ||
result.AssociationLimitMap = Object.values(result.AssociationLimitMap); | ||
} | ||
if (!result.TransactionCountLimitMap || | ||
@@ -88,2 +97,2 @@ typeof ((_b = result.TransactionCountLimitMap) === null || _b === void 0 ? void 0 : _b['AUTHORIZE_DERIVED_KEY']) === | ||
} | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGVybWlzc2lvbnMtdXRpbHMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL2lkZW50aXR5L3NyYy9saWIvcGVybWlzc2lvbnMtdXRpbHMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6Ijs7O0FBR0EsU0FBZ0IsZ0NBQWdDLENBQzlDLG1CQUF3QixFQUN4QixpQkFBc0I7SUFFdEIsSUFBSSxpQkFBaUIsR0FBRyxJQUFJLENBQUM7SUFFN0Isb0VBQW9FO0lBQ3BFLElBQUksaUJBQWlCLGFBQWpCLGlCQUFpQix1QkFBakIsaUJBQWlCLENBQUUsV0FBVyxFQUFFO1FBQ2xDLE9BQU8saUJBQWlCLENBQUM7S0FDMUI7SUFFRCxPQUFPLENBQUMsbUJBQW1CLEVBQUUsQ0FBQyxXQUFXLEVBQUUsSUFBSSxFQUFFLEVBQUU7UUFDakQsTUFBTSxTQUFTLEdBQUcsWUFBWSxDQUFDLGlCQUFpQixFQUFFLElBQUksQ0FBQyxDQUFDO1FBQ3hELElBQ0UsT0FBTyxTQUFTLEtBQUssV0FBVztZQUNoQyxDQUFDLE9BQU8sU0FBUyxLQUFLLFFBQVE7Z0JBQzVCLE9BQU8sV0FBVyxLQUFLLFFBQVE7Z0JBQy9CLFNBQVMsR0FBRyxXQUFXLENBQUM7WUFDMUIsQ0FBQyxPQUFPLFNBQVMsS0FBSyxRQUFRO2dCQUM1QixXQUFXLEtBQUssV0FBVztnQkFDM0IsU0FBUyxHQUFHLEdBQUcsQ0FBQyxFQUNsQjtZQUNBLGlCQUFpQixHQUFHLEtBQUssQ0FBQztTQUMzQjtJQUNILENBQUMsQ0FBQyxDQUFDO0lBRUgsT0FBTyxpQkFBaUIsQ0FBQztBQUMzQixDQUFDO0FBM0JELDRFQTJCQztBQUVELFNBQWdCLHFDQUFxQyxDQUNuRCxvQkFBc0U7O0lBRXRFLElBQUksb0JBQW9CLENBQUMsV0FBVyxFQUFFO1FBQ3BDLE9BQU87WUFDTCxXQUFXLEVBQUUsSUFBSTtTQUNsQixDQUFDO0tBQ0g7SUFFRCxJQUFJLENBQUEsTUFBQSxvQkFBb0IsQ0FBQyxlQUFl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bmRpbmdMaW1pdFJlc3BvbnNlIH0gZnJvbSAnZGVzby1wcm90b2NvbC10eXBlcyc7XG5pbXBvcnQgeyBUcmFuc2FjdGlvblNwZW5kaW5nTGltaXRSZXNwb25zZU9wdGlvbnMgfSBmcm9tICcuL3R5cGVzJztcblxuZXhwb3J0IGZ1bmN0aW9uIGNvbXBhcmVUcmFuc2FjdGlvblNwZW5kaW5nTGltaXRzKFxuICBleHBlY3RlZFBlcm1pc3Npb25zOiBhbnksXG4gIGFjdHVhbFBlcm1pc3Npb25zOiBhbnlcbik6IGJvb2xlYW4ge1xuICBsZXQgaGFzQWxsUGVybWlzc2lvbnMgPSB0cnVlO1xuXG4gIC8vIGlmIHRoZSBrZXkgaXMgdW5saW1pdGVkIHRoZW4gd2UgZG9uJ3QgbmVlZCB0byBjaGVjayBhbnl0aGluZyBlbHNlXG4gIGlmIChhY3R1YWxQZXJtaXNzaW9ucz8uSXNVbmxpbWl0ZWQpIHtcbiAgICByZXR1cm4gaGFzQWxsUGVybWlzc2lvbnM7XG4gIH1cblxuICB3YWxrT2JqKGV4cGVjdGVkUGVybWlzc2lvbnMsIChleHBlY3RlZFZhbCwgcGF0aCkgPT4ge1xuICAgIGNvbnN0IGFjdHVhbFZhbCA9IGdldERlZXBWYWx1ZShhY3R1YWxQZXJtaXNzaW9ucywgcGF0aCk7XG4gICAgaWYgKFxuICAgICAgdHlwZW9mIGFjdHVhbFZhbCA9PT0gJ3VuZGVmaW5lZCcgfHxcbiAgICAgICh0eXBlb2YgYWN0dWFsVmFsID09PSAnbnVtYmVyJyAmJlxuICAgICAgICB0eXBlb2YgZXhwZWN0ZWRWYWwgPT09ICdudW1iZXInICYmXG4gICAgICAgIGFjdHVhbFZhbCA8IGV4cGVjdGVkVmFsKSB8fFxuICAgICAgKHR5cGVvZiBhY3R1YWxWYWwgPT09ICdudW1iZXInICYmXG4gICAgICAgIGV4cGVjdGVkVmFsID09PSAnVU5MSU1JVEVEJyAmJlxuICAgICAgICBhY3R1YWxWYWwgPCAxZTkpXG4gICAgKSB7XG4gICAgICBoYXNBbGxQZXJtaXNzaW9ucyA9IGZhbHNlO1xuICAgIH1cbiAgfSk7XG5cbiAgcmV0dXJuIGhhc0FsbFBlcm1pc3Npb25zO1xufVxuXG5leHBvcnQgZnVuY3Rpb24gYnVpbGRUcmFuc2FjdGlvblNwZW5kaW5nTGltaXRSZXNwb25zZShcbiAgc3BlbmRpbmdMaW1pdE9wdGlvbnM6IFBhcnRpYWw8VHJhbnNhY3Rpb25TcGVuZGluZ0xpbWl0UmVzcG9uc2VPcHRpb25zPlxuKTogVHJhbnNhY3Rpb25TcGVuZGluZ0xpbWl0UmVzcG9uc2Uge1xuICBpZiAoc3BlbmRpbmdMaW1pdE9wdGlvbnMuSXNVbmxpbWl0ZWQpIHtcbiAgICByZXR1cm4ge1xuICAgICAgSXNVbmxpbWl0ZWQ6IHRydWUsXG4gICAgfTtcbiAgfVxuXG4gIGlmIChzcGVuZGluZ0xpbWl0T3B0aW9ucy5HbG9iYWxERVNPTGltaXQ/LnRvU3RyaW5nKCkgPT09ICdVTkxJTUlURUQnKSB7XG4gICAgdGhyb3cgbmV3IEVycm9yKFxuICAgICAgJ0dsb2JhbERFU09MaW1pdCBjYW5ub3QgYmUgdW5saW1pdGVkLiBZb3UgbXVzdCBzcGVjaWZ5IGEgc3BlY2lmaWMgbGltaXQsIG9yIHNldCB0aGUgSXNVbmxpbWl0ZWQgZmxhZyB0byB0cnVlLidcbiAgICApO1xuICB9XG5cbiAgY29uc3QgcmVzdWx0OiBUcmFuc2FjdGlvblNwZW5kaW5nTGltaXRSZXNwb25zZSA9IHt9O1xuXG4gIHdhbGtPYmooXG4gICAgc3BlbmRpbmdMaW1pdE9wdGlvbnMsXG4gICAgKHZhbCwgcGF0aCkgPT4ge1xuICAgICAgc2V0RGVlcFZhbHVlKHJlc3VsdCwgcGF0aCwgdmFsID09PSAnVU5MSU1JVEVEJyA/IDFlOSA6IHZhbCk7XG4gICAgfSxcbiAgICBbXVxuICApO1xuXG4gIGlmIChcbiAgICAhcmVzdWx0LlRyYW5zYWN0aW9uQ291bnRMaW1pdE1hcCB8fFxuICAgIHR5cGVvZiByZXN1bHQuVHJhbnNhY3Rpb25Db3VudExpbWl0TWFwPy5bJ0FVVEhPUklaRV9ERVJJVkVEX0tFWSddID09PVxuICAgICAgJ3VuZGVmaW5lZCdcbiAgKSB7XG4gICAgcmVzdWx0LlRyYW5zYWN0aW9uQ291bnRMaW1pdE1hcCA9IHtcbiAgICAgIC4uLnJlc3VsdC5UcmFuc2FjdGlvbkNvdW50TGltaXRNYXAsXG4gICAgICBBVVRIT1JJWkVfREVSSVZFRF9LRVk6IDEsXG4gICAgfTtcbiAgfVxuXG4gIHJldHVybiByZXN1bHQ7XG59XG5cbmZ1bmN0aW9uIHdhbGtPYmooXG4gIG5vZGU6IGFueSxcbiAgY2FsbGJhY2s6ICh2YWw6IGFueSwgcGF0aDogc3RyaW5nW10pID0+IHZvaWQsXG4gIHBhdGg6IHN0cmluZ1tdID0gW11cbikge1xuICBpZiAodHlwZW9mIG5vZGUgPT09ICdvYmplY3QnICYmIG5vZGUgIT09IG51bGwpIHtcbiAgICBjb25zdCBrZXlzID0gT2JqZWN0LmtleXMobm9kZSk7XG4gICAgZm9yIChsZXQgaSA9IDA7IGkgPCBrZXlzLmxlbmd0aDsgaSsrKSB7XG4gICAgICB3YWxrT2JqKG5vZGVba2V5c1tpXV0sIGNhbGxiYWNrLCBwYXRoLmNvbmNhdChrZXlzW2ldKSk7XG4gICAgfVxuICB9IGVsc2Uge1xuICAgIGNhbGxiYWNrKG5vZGUsIHBhdGgpO1xuICB9XG59XG5cbmZ1bmN0aW9uIGdldERlZXBWYWx1ZShvYmo6IGFueSwgcGF0aDogc3RyaW5nW10pOiBhbnkge1xuICBjb25zdCBjdXJyS2V5ID0gcGF0aFswXTtcblxuICBpZiAoXG4gICAgb2JqID09PSBudWxsIHx8XG4gICAgdHlwZW9mIG9iaiAhPT0gJ29iamVjdCcgfHxcbiAgICB0eXBlb2Ygb2JqW2N1cnJLZXldID09PSAndW5kZWZpbmVkJ1xuICApIHtcbiAgICByZXR1cm47XG4gIH1cblxuICBpZiAocGF0aC5sZW5ndGggPT09IDEpIHtcbiAgICByZXR1cm4gb2JqW2N1cnJLZXldO1xuICB9IGVsc2Uge1xuICAgIHJldHVybiBnZXREZWVwVmFsdWUob2JqW2N1cnJLZXldLCBwYXRoLnNsaWNlKDEpKTtcbiAgfVxufVxuXG5mdW5jdGlvbiBzZXREZWVwVmFsdWUob2JqOiBhbnksIHBhdGg6IHN0cmluZ1tdLCB2YWx1ZTogYW55KSB7XG4gIGNvbnN0IGN1cnJLZXkgPSBwYXRoWzBdO1xuXG4gIGlmICh0eXBlb2Ygb2JqW2N1cnJLZXldID09PSAndW5kZWZpbmVkJykge1xuICAgIG9ialtjdXJyS2V5XSA9IHt9O1xuICB9XG5cbiAgaWYgKHBhdGgubGVuZ3RoID09PSAxKSB7XG4gICAgb2JqW2N1cnJLZXldID0gdmFsdWU7XG4gIH0gZWxzZSB7XG4gICAgc2V0RGVlcFZhbHVlKG9ialtjdXJyS2V5XSwgcGF0aC5zbGljZSgxKSwgdmFsdWUpO1xuICB9XG59XG4iXX0= | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"permissions-utils.js","sourceRoot":"","sources":["../../../../../libs/identity/src/lib/permissions-utils.ts"],"names":[],"mappings":"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","sourcesContent":["import { TransactionSpendingLimitResponse } from 'deso-protocol-types';\nimport { TransactionSpendingLimitResponseOptions } from './types';\n\nexport function compareTransactionSpendingLimits(\n  expectedPermissions: any,\n  actualPermissions: any\n): boolean {\n  let hasAllPermissions = true;\n\n  // if the key is unlimited then we don't need to check anything else\n  if (actualPermissions?.IsUnlimited) {\n    return hasAllPermissions;\n  }\n\n  walkObj(expectedPermissions, (expectedVal, path) => {\n    const actualVal = getDeepValue(actualPermissions, path);\n    if (\n      typeof actualVal === 'undefined' ||\n      (typeof actualVal === 'number' &&\n        typeof expectedVal === 'number' &&\n        actualVal < expectedVal) ||\n      (typeof actualVal === 'number' &&\n        expectedVal === 'UNLIMITED' &&\n        actualVal < 1e9)\n    ) {\n      hasAllPermissions = false;\n    }\n  });\n\n  return hasAllPermissions;\n}\n\nexport function buildTransactionSpendingLimitResponse(\n  spendingLimitOptions: Partial<TransactionSpendingLimitResponseOptions>\n): TransactionSpendingLimitResponse {\n  if (spendingLimitOptions.IsUnlimited) {\n    return {\n      IsUnlimited: true,\n    };\n  }\n\n  if (spendingLimitOptions.GlobalDESOLimit?.toString() === 'UNLIMITED') {\n    throw new Error(\n      'GlobalDESOLimit cannot be unlimited. You must specify a specific limit, or set the IsUnlimited flag to true.'\n    );\n  }\n\n  const result: TransactionSpendingLimitResponse = {};\n\n  walkObj(\n    spendingLimitOptions,\n    (val, path) => {\n      setDeepValue(result, path, val === 'UNLIMITED' ? 1e9 : val);\n    },\n    []\n  );\n\n  if (result.AccessGroupLimitMap) {\n    result.AccessGroupLimitMap = Object.values(result.AccessGroupLimitMap);\n  }\n  if (result.AccessGroupMemberLimitMap) {\n    result.AccessGroupMemberLimitMap = Object.values(\n      result.AccessGroupMemberLimitMap\n    );\n  }\n  if (result.AssociationLimitMap) {\n    result.AssociationLimitMap = Object.values(result.AssociationLimitMap);\n  }\n\n  if (\n    !result.TransactionCountLimitMap ||\n    typeof result.TransactionCountLimitMap?.['AUTHORIZE_DERIVED_KEY'] ===\n      'undefined'\n  ) {\n    result.TransactionCountLimitMap = {\n      ...result.TransactionCountLimitMap,\n      AUTHORIZE_DERIVED_KEY: 1,\n    };\n  }\n  return result;\n}\n\nfunction walkObj(\n  node: any,\n  callback: (val: any, path: string[]) => void,\n  path: string[] = []\n) {\n  if (typeof node === 'object' && node !== null) {\n    const keys = Object.keys(node);\n    for (let i = 0; i < keys.length; i++) {\n      walkObj(node[keys[i]], callback, path.concat(keys[i]));\n    }\n  } else {\n    callback(node, path);\n  }\n}\n\nfunction getDeepValue(obj: any, path: string[]): any {\n  const currKey = path[0];\n\n  if (\n    obj === null ||\n    typeof obj !== 'object' ||\n    typeof obj[currKey] === 'undefined'\n  ) {\n    return;\n  }\n\n  if (path.length === 1) {\n    return obj[currKey];\n  } else {\n    return getDeepValue(obj[currKey], path.slice(1));\n  }\n}\n\nfunction setDeepValue(obj: any, path: string[], value: any) {\n  const currKey = path[0];\n  if (typeof obj[currKey] === 'undefined') {\n    obj[currKey] = {};\n  }\n\n  if (path.length === 1) {\n    obj[currKey] = value;\n  } else {\n    setDeepValue(obj[currKey], path.slice(1), value);\n  }\n}\n"]} |
@@ -55,5 +55,11 @@ import { AccessGroupLimitMapItem, AccessGroupMemberLimitMapItem, AssociationLimitMapItem, TransactionSpendingLimitResponse } from 'deso-protocol-types'; | ||
}; | ||
AssociationLimitMap?: AssociationLimitMapItem[]; | ||
AccessGroupLimitMap?: AccessGroupLimitMapItem[]; | ||
AccessGroupMemberLimitMap?: AccessGroupMemberLimitMapItem[]; | ||
AssociationLimitMap?: (Omit<AssociationLimitMapItem, 'OpCount'> & { | ||
OpCount: number | 'UNLIMITED'; | ||
})[]; | ||
AccessGroupLimitMap?: (Omit<AccessGroupLimitMapItem, 'OpCount'> & { | ||
OpCount: number | 'UNLIMITED'; | ||
})[]; | ||
AccessGroupMemberLimitMap?: (Omit<AccessGroupMemberLimitMapItem, 'OpCount'> & { | ||
OpCount: number | 'UNLIMITED'; | ||
})[]; | ||
IsUnlimited?: boolean; | ||
@@ -116,8 +122,9 @@ } | ||
} | ||
export declare type PrimaryDerivedKeyInfo = IdentityDerivePayload & { | ||
transactionSpendingLimits: TransactionSpendingLimitResponse; | ||
IsValid?: boolean; | ||
}; | ||
export declare type StoredUser = { | ||
publicKey: string; | ||
primaryDerivedKey: IdentityDerivePayload & { | ||
transactionSpendingLimits: TransactionSpendingLimitResponse; | ||
IsValid?: boolean; | ||
}; | ||
primaryDerivedKey: PrimaryDerivedKeyInfo; | ||
}; | ||
@@ -155,6 +162,11 @@ export interface IdentityUser { | ||
export interface SubscriberNotification { | ||
event: string; | ||
event: NOTIFICATION_EVENTS; | ||
currentUser: StoredUser | null; | ||
alternateUsers: Record<string, StoredUser> | null; | ||
} | ||
export interface AccessGroupPrivateInfo { | ||
AccessGroupPublicKeyBase58Check: string; | ||
AccessGroupPrivateKeyHex: string; | ||
AccessGroupKeyName: string; | ||
} | ||
export declare enum NOTIFICATION_EVENTS { | ||
@@ -161,0 +173,0 @@ /** |
@@ -76,2 +76,2 @@ "use strict"; | ||
})(NOTIFICATION_EVENTS = exports.NOTIFICATION_EVENTS || (exports.NOTIFICATION_EVENTS = {})); | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../libs/identity/src/lib/types.ts"],"names":[],"mappings":";;;AAqLA,IAAY,mBAoFX;AApFD,WAAY,mBAAmB;IAC7B;;OAEG;IACH,8CAAuB,CAAA;IAEvB;;;;OAIG;IACH,kFAA2D,CAAA;IAE3D;;;;OAIG;IACH,8EAAuD,CAAA;IAEvD;;OAEG;IACH,gFAAyD,CAAA;IAEzD;;OAEG;IACH,8EAAuD,CAAA;IAEvD;;;OAGG;IACH,0EAAmD,CAAA;IAEnD;;OAEG;IACH,kDAA2B,CAAA;IAE3B;;;OAGG;IACH,8CAAuB,CAAA;IAEvB;;OAEG;IACH,oDAA6B,CAAA;IAE7B;;;OAGG;IACH,gDAAyB,CAAA;IAEzB;;OAEG;IACH,kEAA2C,CAAA;IAE3C;;;OAGG;IACH,8DAAuC,CAAA;IAEvC;;OAEG;IACH,8EAAuD,CAAA;IAEvD;;;OAGG;IACH,0EAAmD,CAAA;IAEnD;;OAEG;IACH,gEAAyC,CAAA;AAC3C,CAAC,EApFW,mBAAmB,GAAnB,2BAAmB,KAAnB,2BAAmB,QAoF9B","sourcesContent":["import {\n  AccessGroupLimitMapItem,\n  AccessGroupMemberLimitMapItem,\n  AssociationLimitMapItem,\n  TransactionSpendingLimitResponse,\n} from 'deso-protocol-types';\n\nexport type Network = 'mainnet' | 'testnet';\n\nexport interface IdentityResponse {\n  service: 'identity';\n  method: 'derive' | 'login' | 'initialize';\n  payload?: any;\n  id?: string;\n}\n\nexport interface IdentityDerivePayload {\n  derivedSeedHex?: string;\n  derivedPublicKeyBase58Check: string;\n  publicKeyBase58Check: string;\n  btcDepositAddress: string;\n  ethDepositAddress: string;\n  expirationBlock: number;\n  network: Network;\n  accessSignature: string;\n  jwt: string;\n  derivedJwt: string;\n  messagingPublicKeyBase58Check: string;\n  messagingPrivateKey: string;\n  messagingKeyName: string;\n  messagingKeySignature: string;\n  transactionSpendingLimitHex: string;\n  signedUp: boolean;\n  publicKeyAdded?: string;\n}\n\nexport interface TransactionSpendingLimitResponseOptions {\n  GlobalDESOLimit?: number;\n  TransactionCountLimitMap?: { [key: string]: number | 'UNLIMITED' };\n  CreatorCoinOperationLimitMap?: {\n    [key: string]: { [key: string]: number | 'UNLIMITED' };\n  };\n  DAOCoinOperationLimitMap?: {\n    [key: string]: { [key: string]: number | 'UNLIMITED' };\n  };\n  NFTOperationLimitMap?: {\n    [key: string]: { [key: number]: { [key: string]: number | 'UNLIMITED' } };\n  };\n  DAOCoinLimitOrderLimitMap?: {\n    [key: string]: { [key: string]: number | 'UNLIMITED' };\n  };\n  AssociationLimitMap?: AssociationLimitMapItem[];\n  AccessGroupLimitMap?: AccessGroupLimitMapItem[];\n  AccessGroupMemberLimitMap?: AccessGroupMemberLimitMapItem[];\n  IsUnlimited?: boolean;\n}\n\nexport type jwtAlgorithm = 'ES256K' | 'ES256';\nexport interface IdentityConfiguration {\n  /**\n   * The identity domain. Defaults to https://identity.deso.org\n   */\n  identityURI?: string;\n\n  /**\n   * The current network. If not provided, we will assume mainnet.\n   */\n  network?: Network;\n\n  /**\n   * The deso node used for any api calls (get balance, derived key authorization, etc)\n   */\n  nodeURI?: string;\n\n  /**\n   * Optional redirect URI. If provided, we do a hard redirect to the identity\n   * domain and pass data via query params back to the provided uri.\n   */\n  redirectURI?: string;\n\n  /**\n   * The default permissions and spending limits that will be presented to the user\n   * during login. If not provided, we will assume no permissions.\n   */\n  spendingLimitOptions?: TransactionSpendingLimitResponseOptions;\n\n  /**\n   * The name of the app used to authorize derived keys. Defaults to unknown.\n   */\n  appName?: string;\n\n  // Since our keys are generated using the secp256k1 curve, the correct\n  // JWT algorithm header *should* be ES256K.\n  // See: https://www.rfc-editor.org/rfc/rfc8812.html#name-jose-algorithms-registratio\n  //\n  // HOWEVER, the backend jwt lib used by deso foundation -\n  // https://github.com/golang-jwt/jwt - (as well as many other jwt libraries)\n  // do not support ES256K. So instead, we default to the more widely supported ES256 algo,\n  // which can still work for verifying our signatures. But if a consumer of this lib is using a\n  // jwt lib that supports ES256K they can specify that here.\n  // See this github issue\n  // for more context: https://github.com/auth0/node-jsonwebtoken/issues/862\n  // If ES256K is ever supported by the backend jwt lib, we should change this.\n  jwtAlgorithm?: jwtAlgorithm;\n}\n\nexport interface APIProvider {\n  post: (url: string, data: any) => Promise<any>;\n  get: (url: string) => Promise<any>;\n}\n\nexport interface WindowProvider {\n  location: { search: string; pathname: string; href: string };\n  history: { replaceState: (state: any, title: string, url: string) => void };\n  localStorage: {\n    getItem: (key: string) => string | null;\n    setItem: (key: string, value: string) => void;\n    removeItem: (key: string) => void;\n  };\n  open: (\n    url: string,\n    title: string | undefined,\n    options: string\n  ) => Window | null;\n  addEventListener: (event: string, callback: (event: any) => void) => void;\n  removeEventListener: (event: string, callback: (event: any) => void) => void;\n}\n\nexport interface LoginOptions {\n  getFreeDeso: boolean;\n}\n\nexport type StoredUser = {\n  publicKey: string;\n  primaryDerivedKey: IdentityDerivePayload & {\n    transactionSpendingLimits: TransactionSpendingLimitResponse;\n    IsValid?: boolean;\n  };\n};\n\nexport interface IdentityUser {\n  accessLevel: number;\n  accessLevelHmac: string;\n  btcDepositAddress: string;\n  encryptedSeedHex: string;\n  ethDepositAddress: string;\n  derivedPublicKeyBase58Check?: string;\n  hasExtraText: boolean;\n  network: string;\n  version: number;\n}\n\nexport interface IdentityLoginPayload {\n  users: Record<string, IdentityUser>;\n  publicKeyAdded: string;\n  phoneNumberSuccess: boolean;\n  signedUp: boolean;\n}\n\nexport interface IdentityState {\n  currentUser: StoredUser | null;\n  alternateUsers: Record<string, StoredUser> | null;\n}\n\nexport interface Deferred {\n  resolve: (args: any) => void;\n  reject: (args: any) => void;\n}\n\nexport interface KeyPair {\n  seedHex: string;\n  private: Uint8Array;\n  public: Uint8Array;\n}\n\nexport interface SubscriberNotification {\n  event: string;\n  currentUser: StoredUser | null;\n  alternateUsers: Record<string, StoredUser> | null;\n}\n\nexport enum NOTIFICATION_EVENTS {\n  /**\n   * This event is fired when the consuming app initially subscribes to identity.\n   */\n  SUBSCRIBE = 'SUBSCRIBE',\n\n  /**\n   * This is an intermediate event fired AFTER the user completes an identity flow\n   * that requires a derived key authorization. This event is fired BEFORE the\n   * request to authorize the derived key is made.\n   */\n  AUTHORIZE_DERIVED_KEY_START = 'AUTHORIZE_DERIVED_KEY_START',\n\n  /**\n   * This is an intermediate event fired AFTER the user completes an identity\n   * flow that requires a derived key authorization. This event is fired AFTER\n   * the request to authorize the derived key is made.\n   */\n  AUTHORIZE_DERIVED_KEY_END = 'AUTHORIZE_DERIVED_KEY_END',\n\n  /**\n   * This event is fired if the request to authorize a derived key fails.\n   */\n  AUTHORIZE_DERIVED_KEY_FAIL = 'AUTHORIZE_DERIVED_KEY_FAIL',\n\n  /**\n   * This event is fired when the user opens the permissions approval popup.\n   */\n  REQUEST_PERMISSIONS_START = 'REQUEST_PERMISSIONS_START',\n\n  /**\n   * This event is fired when the user completes approving permissions, and\n   * comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  REQUEST_PERMISSIONS_END = 'REQUEST_PERMISSIONS_END',\n\n  /**\n   * This event is fired when the user opens the login popup.\n   */\n  LOGIN_START = 'LOGIN_START',\n\n  /**\n   * This event is fired when the user completes logging in, and\n   * comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  LOGIN_END = 'LOGIN_END',\n\n  /**\n   * This event is fired when the user opens the logout popup.\n   */\n  LOGOUT_START = 'LOGOUT_START',\n\n  /**\n   * This event is fired when the user completes logging out, and\n   * comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  LOGOUT_END = 'LOGOUT_END',\n\n  /**\n   * This event is fired when the user opens the get deso popup.\n   */\n  GET_FREE_DESO_START = 'GET_FREE_DESO_START',\n\n  /**\n   * This event is fired when the user completes the get deso flow, and comes\n   * AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  GET_FREE_DESO_END = 'GET_FREE_DESO_END',\n\n  /**\n   * This event is fired when the user opens the verify phone number popup.\n   */\n  VERIFY_PHONE_NUMBER_START = 'VERIFY_PHONE_NUMBER_START',\n\n  /**\n   * This event is fired when the user completes the verify phone number flow,\n   * and comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  VERIFY_PHONE_NUMBER_END = 'VERIFY_PHONE_NUMBER_END',\n\n  /**\n   * This event is fired when the consuming app switches the active user.\n   */\n  CHANGE_ACTIVE_USER = 'CHANGE_ACTIVE_USER',\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../libs/identity/src/lib/types.ts"],"names":[],"mappings":";;;AAoMA,IAAY,mBAoFX;AApFD,WAAY,mBAAmB;IAC7B;;OAEG;IACH,8CAAuB,CAAA;IAEvB;;;;OAIG;IACH,kFAA2D,CAAA;IAE3D;;;;OAIG;IACH,8EAAuD,CAAA;IAEvD;;OAEG;IACH,gFAAyD,CAAA;IAEzD;;OAEG;IACH,8EAAuD,CAAA;IAEvD;;;OAGG;IACH,0EAAmD,CAAA;IAEnD;;OAEG;IACH,kDAA2B,CAAA;IAE3B;;;OAGG;IACH,8CAAuB,CAAA;IAEvB;;OAEG;IACH,oDAA6B,CAAA;IAE7B;;;OAGG;IACH,gDAAyB,CAAA;IAEzB;;OAEG;IACH,kEAA2C,CAAA;IAE3C;;;OAGG;IACH,8DAAuC,CAAA;IAEvC;;OAEG;IACH,8EAAuD,CAAA;IAEvD;;;OAGG;IACH,0EAAmD,CAAA;IAEnD;;OAEG;IACH,gEAAyC,CAAA;AAC3C,CAAC,EApFW,mBAAmB,GAAnB,2BAAmB,KAAnB,2BAAmB,QAoF9B","sourcesContent":["import {\n  AccessGroupLimitMapItem,\n  AccessGroupMemberLimitMapItem,\n  AssociationLimitMapItem,\n  TransactionSpendingLimitResponse,\n} from 'deso-protocol-types';\n\nexport type Network = 'mainnet' | 'testnet';\n\nexport interface IdentityResponse {\n  service: 'identity';\n  method: 'derive' | 'login' | 'initialize';\n  payload?: any;\n  id?: string;\n}\n\nexport interface IdentityDerivePayload {\n  derivedSeedHex?: string;\n  derivedPublicKeyBase58Check: string;\n  publicKeyBase58Check: string;\n  btcDepositAddress: string;\n  ethDepositAddress: string;\n  expirationBlock: number;\n  network: Network;\n  accessSignature: string;\n  jwt: string;\n  derivedJwt: string;\n  messagingPublicKeyBase58Check: string;\n  messagingPrivateKey: string;\n  messagingKeyName: string;\n  messagingKeySignature: string;\n  transactionSpendingLimitHex: string;\n  signedUp: boolean;\n  publicKeyAdded?: string;\n}\n\nexport interface TransactionSpendingLimitResponseOptions {\n  GlobalDESOLimit?: number;\n  TransactionCountLimitMap?: { [key: string]: number | 'UNLIMITED' };\n  CreatorCoinOperationLimitMap?: {\n    [key: string]: { [key: string]: number | 'UNLIMITED' };\n  };\n  DAOCoinOperationLimitMap?: {\n    [key: string]: { [key: string]: number | 'UNLIMITED' };\n  };\n  NFTOperationLimitMap?: {\n    [key: string]: { [key: number]: { [key: string]: number | 'UNLIMITED' } };\n  };\n  DAOCoinLimitOrderLimitMap?: {\n    [key: string]: { [key: string]: number | 'UNLIMITED' };\n  };\n  AssociationLimitMap?: (Omit<AssociationLimitMapItem, 'OpCount'> & {\n    OpCount: number | 'UNLIMITED';\n  })[];\n  AccessGroupLimitMap?: (Omit<AccessGroupLimitMapItem, 'OpCount'> & {\n    OpCount: number | 'UNLIMITED';\n  })[];\n  AccessGroupMemberLimitMap?: (Omit<\n    AccessGroupMemberLimitMapItem,\n    'OpCount'\n  > & { OpCount: number | 'UNLIMITED' })[];\n  IsUnlimited?: boolean;\n}\n\nexport type jwtAlgorithm = 'ES256K' | 'ES256';\nexport interface IdentityConfiguration {\n  /**\n   * The identity domain. Defaults to https://identity.deso.org\n   */\n  identityURI?: string;\n\n  /**\n   * The current network. If not provided, we will assume mainnet.\n   */\n  network?: Network;\n\n  /**\n   * The deso node used for any api calls (get balance, derived key authorization, etc)\n   */\n  nodeURI?: string;\n\n  /**\n   * Optional redirect URI. If provided, we do a hard redirect to the identity\n   * domain and pass data via query params back to the provided uri.\n   */\n  redirectURI?: string;\n\n  /**\n   * The default permissions and spending limits that will be presented to the user\n   * during login. If not provided, we will assume no permissions.\n   */\n  spendingLimitOptions?: TransactionSpendingLimitResponseOptions;\n\n  /**\n   * The name of the app used to authorize derived keys. Defaults to unknown.\n   */\n  appName?: string;\n\n  // Since our keys are generated using the secp256k1 curve, the correct\n  // JWT algorithm header *should* be ES256K.\n  // See: https://www.rfc-editor.org/rfc/rfc8812.html#name-jose-algorithms-registratio\n  //\n  // HOWEVER, the backend jwt lib used by deso foundation -\n  // https://github.com/golang-jwt/jwt - (as well as many other jwt libraries)\n  // do not support ES256K. So instead, we default to the more widely supported ES256 algo,\n  // which can still work for verifying our signatures. But if a consumer of this lib is using a\n  // jwt lib that supports ES256K they can specify that here.\n  // See this github issue\n  // for more context: https://github.com/auth0/node-jsonwebtoken/issues/862\n  // If ES256K is ever supported by the backend jwt lib, we should change this.\n  jwtAlgorithm?: jwtAlgorithm;\n}\n\nexport interface APIProvider {\n  post: (url: string, data: any) => Promise<any>;\n  get: (url: string) => Promise<any>;\n}\n\nexport interface WindowProvider {\n  location: { search: string; pathname: string; href: string };\n  history: { replaceState: (state: any, title: string, url: string) => void };\n  localStorage: {\n    getItem: (key: string) => string | null;\n    setItem: (key: string, value: string) => void;\n    removeItem: (key: string) => void;\n  };\n  open: (\n    url: string,\n    title: string | undefined,\n    options: string\n  ) => Window | null;\n  addEventListener: (event: string, callback: (event: any) => void) => void;\n  removeEventListener: (event: string, callback: (event: any) => void) => void;\n}\n\nexport interface LoginOptions {\n  getFreeDeso: boolean;\n}\n\nexport type PrimaryDerivedKeyInfo = IdentityDerivePayload & {\n  transactionSpendingLimits: TransactionSpendingLimitResponse;\n  IsValid?: boolean;\n};\n\nexport type StoredUser = {\n  publicKey: string;\n  primaryDerivedKey: PrimaryDerivedKeyInfo;\n};\n\nexport interface IdentityUser {\n  accessLevel: number;\n  accessLevelHmac: string;\n  btcDepositAddress: string;\n  encryptedSeedHex: string;\n  ethDepositAddress: string;\n  derivedPublicKeyBase58Check?: string;\n  hasExtraText: boolean;\n  network: string;\n  version: number;\n}\n\nexport interface IdentityLoginPayload {\n  users: Record<string, IdentityUser>;\n  publicKeyAdded: string;\n  phoneNumberSuccess: boolean;\n  signedUp: boolean;\n}\n\nexport interface IdentityState {\n  currentUser: StoredUser | null;\n  alternateUsers: Record<string, StoredUser> | null;\n}\n\nexport interface Deferred {\n  resolve: (args: any) => void;\n  reject: (args: any) => void;\n}\n\nexport interface KeyPair {\n  seedHex: string;\n  private: Uint8Array;\n  public: Uint8Array;\n}\n\nexport interface SubscriberNotification {\n  event: NOTIFICATION_EVENTS;\n  currentUser: StoredUser | null;\n  alternateUsers: Record<string, StoredUser> | null;\n}\n\nexport interface AccessGroupPrivateInfo {\n  AccessGroupPublicKeyBase58Check: string;\n  AccessGroupPrivateKeyHex: string;\n  AccessGroupKeyName: string;\n}\n\nexport enum NOTIFICATION_EVENTS {\n  /**\n   * This event is fired when the consuming app initially subscribes to identity.\n   */\n  SUBSCRIBE = 'SUBSCRIBE',\n\n  /**\n   * This is an intermediate event fired AFTER the user completes an identity flow\n   * that requires a derived key authorization. This event is fired BEFORE the\n   * request to authorize the derived key is made.\n   */\n  AUTHORIZE_DERIVED_KEY_START = 'AUTHORIZE_DERIVED_KEY_START',\n\n  /**\n   * This is an intermediate event fired AFTER the user completes an identity\n   * flow that requires a derived key authorization. This event is fired AFTER\n   * the request to authorize the derived key is made.\n   */\n  AUTHORIZE_DERIVED_KEY_END = 'AUTHORIZE_DERIVED_KEY_END',\n\n  /**\n   * This event is fired if the request to authorize a derived key fails.\n   */\n  AUTHORIZE_DERIVED_KEY_FAIL = 'AUTHORIZE_DERIVED_KEY_FAIL',\n\n  /**\n   * This event is fired when the user opens the permissions approval popup.\n   */\n  REQUEST_PERMISSIONS_START = 'REQUEST_PERMISSIONS_START',\n\n  /**\n   * This event is fired when the user completes approving permissions, and\n   * comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  REQUEST_PERMISSIONS_END = 'REQUEST_PERMISSIONS_END',\n\n  /**\n   * This event is fired when the user opens the login popup.\n   */\n  LOGIN_START = 'LOGIN_START',\n\n  /**\n   * This event is fired when the user completes logging in, and\n   * comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  LOGIN_END = 'LOGIN_END',\n\n  /**\n   * This event is fired when the user opens the logout popup.\n   */\n  LOGOUT_START = 'LOGOUT_START',\n\n  /**\n   * This event is fired when the user completes logging out, and\n   * comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  LOGOUT_END = 'LOGOUT_END',\n\n  /**\n   * This event is fired when the user opens the get deso popup.\n   */\n  GET_FREE_DESO_START = 'GET_FREE_DESO_START',\n\n  /**\n   * This event is fired when the user completes the get deso flow, and comes\n   * AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  GET_FREE_DESO_END = 'GET_FREE_DESO_END',\n\n  /**\n   * This event is fired when the user opens the verify phone number popup.\n   */\n  VERIFY_PHONE_NUMBER_START = 'VERIFY_PHONE_NUMBER_START',\n\n  /**\n   * This event is fired when the user completes the verify phone number flow,\n   * and comes AFTER the intermediate AUTHORIZE_DERIVED_KEY events.\n   */\n  VERIFY_PHONE_NUMBER_END = 'VERIFY_PHONE_NUMBER_END',\n\n  /**\n   * This event is fired when the consuming app switches the active user.\n   */\n  CHANGE_ACTIVE_USER = 'CHANGE_ACTIVE_USER',\n}\n"]} |
Sorry, the diff of this file is too big to display
289392
2389
191
+ Addeddeso-protocol-types@0.5.7(transitive)
- Removeddeso-protocol-types@0.5.0(transitive)
Updateddeso-protocol-types@0.5.7