-
Notifications
You must be signed in to change notification settings - Fork 231
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
refactor(identity): keep index.ts file only as there's only 1 class
Former-commit-id: e25db2d
- Loading branch information
Showing
2 changed files
with
136 additions
and
140 deletions.
There are no files selected for viewing
This file was deleted.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,137 @@ | ||
import Identity from "./identity" | ||
import type { Point } from "@zk-kit/baby-jubjub" | ||
import { EdDSAPoseidon, Signature, signMessage, verifySignature } from "@zk-kit/eddsa-poseidon" | ||
import type { BigNumberish } from "@zk-kit/utils" | ||
import { bufferToHexadecimal, hexadecimalToBuffer } from "@zk-kit/utils/conversions" | ||
import { requireString } from "@zk-kit/utils/error-handlers" | ||
import { isHexadecimal } from "@zk-kit/utils/type-checks" | ||
import { poseidon2 } from "poseidon-lite/poseidon2" | ||
|
||
// eslint-disable-next-line import/prefer-default-export | ||
export { Identity } | ||
/** | ||
* The Semaphore identity is essentially an {@link https://www.rfc-editor.org/rfc/rfc8032 | EdDSA} | ||
* public/private key pair. The {@link https://github.com/privacy-scaling-explorations/zk-kit/tree/main/packages/eddsa-poseidon | EdDSA implementation} | ||
* in this library uses {@link https://eips.ethereum.org/EIPS/eip-2494 | Baby Jubjub} for public key generation | ||
* and {@link https://www.poseidon-hash.info | Poseidon} for signatures. | ||
* In addition, the commitment, i.e. the hash of the public key, is used to represent | ||
* Semaphore identities in groups, adding an additional layer of privacy and security. | ||
* The private key of the identity is stored as a hexadecimal string or text. | ||
* The other attributes are stored as stringified bigint. | ||
*/ | ||
export class Identity { | ||
// The EdDSA private key, passed as a parameter or generated randomly. | ||
private _privateKey: string | ||
// The secret scalar derived from the private key. | ||
// It is used in circuits to derive the public key. | ||
private _secretScalar: bigint | ||
// The EdDSA public key, derived from the private key. | ||
private _publicKey: Point<bigint> | ||
// The identity commitment used as a public value in Semaphore groups. | ||
private _commitment: bigint | ||
|
||
/** | ||
* Initializes the class attributes based on a given private key, which must be a hexadecimal string or a text. | ||
* Hexadecimal strings must not start with '0x' or '0X'. | ||
* If the private key is not passed as a parameter, a random hexadecimal key will be generated. | ||
* The EdDSAPoseidon class is used to generate the secret scalar and the public key. | ||
* Additionally, the constructor computes a commitment of the public key using a hash function (Poseidon). | ||
* | ||
* @example | ||
* // Generates an identity. | ||
* const { privateKey, publicKey, commitment } = new Identity("private-key") | ||
* @example | ||
* // Generates an identity with a random private key. | ||
* const { privateKey, publicKey, commitment } = new Identity() | ||
* | ||
* @param privateKey The private key used to derive the public key (hexadecimal or string). | ||
*/ | ||
constructor(privateKey?: string) { | ||
let eddsa: EdDSAPoseidon | ||
|
||
if (privateKey) { | ||
requireString(privateKey, "privateKey") | ||
|
||
this._privateKey = privateKey | ||
|
||
if (isHexadecimal(privateKey, false)) { | ||
eddsa = new EdDSAPoseidon(hexadecimalToBuffer(privateKey)) | ||
} else { | ||
eddsa = new EdDSAPoseidon(privateKey) | ||
} | ||
} else { | ||
eddsa = new EdDSAPoseidon() | ||
|
||
this._privateKey = bufferToHexadecimal(eddsa.privateKey as any) | ||
} | ||
|
||
this._secretScalar = eddsa.secretScalar | ||
this._publicKey = eddsa.publicKey | ||
this._commitment = poseidon2(this._publicKey) | ||
} | ||
|
||
/** | ||
* Returns the private key. | ||
* @returns The private key as a string (hexadecimal or text). | ||
*/ | ||
public get privateKey(): string { | ||
return this._privateKey | ||
} | ||
|
||
/** | ||
* Returns the secret scalar. | ||
* @returns The secret scalar as a string. | ||
*/ | ||
public get secretScalar(): bigint { | ||
return this._secretScalar | ||
} | ||
|
||
/** | ||
* Returns the public key as a Baby Jubjub {@link https://zkkit.pse.dev/types/_zk_kit_baby_jubjub.Point.html | Point}. | ||
* @returns The public key as a point. | ||
*/ | ||
public get publicKey(): Point<bigint> { | ||
return this._publicKey | ||
} | ||
|
||
/** | ||
* Returns the commitment hash of the public key. | ||
* @returns The commitment as a string. | ||
*/ | ||
public get commitment(): bigint { | ||
return this._commitment | ||
} | ||
|
||
/** | ||
* Generates a signature for a given message using the private key. | ||
* This method demonstrates how to sign a message and could be used | ||
* for authentication or data integrity. | ||
* | ||
* @example | ||
* const identity = new Identity() | ||
* const signature = identity.signMessage("message") | ||
* | ||
* @param message The message to be signed. | ||
* @returns A {@link https://zkkit.pse.dev/types/_zk_kit_eddsa_poseidon.Signature.html | Signature} object containing the signature components. | ||
*/ | ||
public signMessage(message: BigNumberish): Signature<bigint> { | ||
return signMessage(this.privateKey, message) | ||
} | ||
|
||
/** | ||
* Verifies a signature against a given message and public key. | ||
* This static method allows for the verification of signatures without needing | ||
* an instance of the Identity class. It's useful for cases where you only have | ||
* the public key, the message and a signature, and need to verify if they match. | ||
* | ||
* @example | ||
* const identity = new Identity() | ||
* const signature = identity.signMessage("message") | ||
* Identity.verifySignature("message", signature, identity.publicKey) | ||
* | ||
* @param message The message that was signed. | ||
* @param signature The signature to verify. | ||
* @param publicKey The public key to use for verification. | ||
* @returns A boolean indicating whether the signature is valid. | ||
*/ | ||
static verifySignature(message: BigNumberish, signature: Signature, publicKey: Point): boolean { | ||
return verifySignature(message, signature, publicKey) | ||
} | ||
} |