Format node bindings with clang-format (#268)

This commit is contained in:
Matthew Keil 2023-03-30 12:20:25 -04:00 committed by GitHub
parent dd5ec7ba7e
commit e9c9c912f6
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 404 additions and 340 deletions

View File

@ -27,9 +27,13 @@ jobs:
uses: actions/setup-node@v1 uses: actions/setup-node@v1
with: with:
node-version: ${{matrix.node}} node-version: ${{matrix.node}}
- name: Check formatting
if: matrix.os == 'ubuntu-latest'
working-directory: bindings/node.js
run: make format
- name: Build/test bindings - name: Build/test bindings
working-directory: bindings/node.js working-directory: bindings/node.js
run: make run: make build test bundle
- name: Install distribution - name: Install distribution
working-directory: bindings/node.js/dist working-directory: bindings/node.js/dist
run: npm install run: npm install

View File

@ -0,0 +1,19 @@
#
# This file is a copy/paste from the one at the root of this repo. It was
# placed here for convenience as not all IDE's will pickup the settings
# in the root file and there was a fight between the IDE formatting and the
# official repo settings. Please keep this file in sync with the root file
#
BasedOnStyle: llvm
IndentWidth: 4
AllowAllParametersOfDeclarationOnNextLine: True
AllowShortFunctionsOnASingleLine: None
AllowShortIfStatementsOnASingleLine: OnlyFirstIf
AlignAfterOpenBracket: BlockIndent
AlignEscapedNewlines: DontAlign
AlwaysBreakAfterDefinitionReturnType: None
BinPackArguments: False
BinPackParameters: False
PenaltyReturnTypeOnItsOwnLine: 1000
PenaltyBreakAssignment: 100

View File

@ -62,6 +62,7 @@ test: install
.PHONY: format .PHONY: format
format: install format: install
@$(YARN) prettier --loglevel=warn --write . @$(YARN) prettier --loglevel=warn --write .
@clang-format -i src/kzg.cxx
# Publish package to npm (requires an auth token) # Publish package to npm (requires an auth token)
.PHONY: publish .PHONY: publish

View File

@ -1,29 +1,28 @@
#include <stdio.h>
#include <iostream>
#include <sstream> // std::ostringstream
#include <string_view>
#include <napi.h>
#include "c_kzg_4844.h"
#include "blst.h" #include "blst.h"
#include "c_kzg_4844.h"
#include <iostream>
#include <napi.h>
#include <sstream> // std::ostringstream
#include <stdio.h>
#include <string_view>
/** /**
* Convert C_KZG_RET to a string representation for error messages. * Convert C_KZG_RET to a string representation for error messages.
*/ */
std::string from_c_kzg_ret(C_KZG_RET ret) { std::string from_c_kzg_ret(C_KZG_RET ret) {
switch (ret) { switch (ret) {
case C_KZG_RET::C_KZG_OK: case C_KZG_RET::C_KZG_OK:
return "C_KZG_OK"; return "C_KZG_OK";
case C_KZG_RET::C_KZG_BADARGS: case C_KZG_RET::C_KZG_BADARGS:
return "C_KZG_BADARGS"; return "C_KZG_BADARGS";
case C_KZG_RET::C_KZG_ERROR: case C_KZG_RET::C_KZG_ERROR:
return "C_KZG_ERROR"; return "C_KZG_ERROR";
case C_KZG_RET::C_KZG_MALLOC: case C_KZG_RET::C_KZG_MALLOC:
return "C_KZG_MALLOC"; return "C_KZG_MALLOC";
default: default:
std::ostringstream msg; std::ostringstream msg;
msg << "UNKNOWN (" << ret << ")"; msg << "UNKNOWN (" << ret << ")";
return msg.str(); return msg.str();
} }
} }
@ -42,8 +41,8 @@ std::string from_c_kzg_ret(C_KZG_RET ret) {
* `napi_get_instance_data` or `Napi::Env::GetInstanceData`. * `napi_get_instance_data` or `Napi::Env::GetInstanceData`.
*/ */
typedef struct { typedef struct {
bool is_setup; bool is_setup;
KZGSettings settings; KZGSettings settings;
} KzgAddonData; } KzgAddonData;
/** /**
@ -58,11 +57,11 @@ typedef struct {
* @param[in] data Pointer KzgAddonData stored by the runtime * @param[in] data Pointer KzgAddonData stored by the runtime
* @param[in] hint (unused) * @param[in] hint (unused)
*/ */
void delete_kzg_addon_data(napi_env /*env*/, void *data, void* /*hint*/) { void delete_kzg_addon_data(napi_env /*env*/, void *data, void * /*hint*/) {
if (((KzgAddonData*)data)->is_setup) { if (((KzgAddonData *)data)->is_setup) {
free_trusted_setup(&((KzgAddonData*)data)->settings); free_trusted_setup(&((KzgAddonData *)data)->settings);
} }
free(data); free(data);
} }
/** /**
@ -82,12 +81,16 @@ void delete_kzg_addon_data(napi_env /*env*/, void *data, void* /*hint*/) {
* @return - Pointer to the KZGSettings * @return - Pointer to the KZGSettings
*/ */
KZGSettings *get_kzg_settings(Napi::Env &env, const Napi::CallbackInfo &info) { KZGSettings *get_kzg_settings(Napi::Env &env, const Napi::CallbackInfo &info) {
KzgAddonData *data = env.GetInstanceData<KzgAddonData>(); KzgAddonData *data = env.GetInstanceData<KzgAddonData>();
if (!data->is_setup) { if (!data->is_setup) {
Napi::Error::New(env, "Must run loadTrustedSetup before running any other c-kzg functions").ThrowAsJavaScriptException(); Napi::Error::New(
return nullptr; env,
} "Must run loadTrustedSetup before running any other c-kzg functions"
return &(data->settings); )
.ThrowAsJavaScriptException();
return nullptr;
}
return &(data->settings);
} }
/** /**
@ -121,98 +124,113 @@ inline uint8_t *get_bytes(
const Napi::Env &env, const Napi::Env &env,
const Napi::Value &val, const Napi::Value &val,
size_t length, size_t length,
std::string_view name) std::string_view name
{ ) {
if (!val.IsTypedArray() || val.As<Napi::TypedArray>().TypedArrayType() != napi_uint8_array) { if (!val.IsTypedArray() ||
std::ostringstream msg; val.As<Napi::TypedArray>().TypedArrayType() != napi_uint8_array) {
msg << "Expected " << name << " to be a Uint8Array"; std::ostringstream msg;
Napi::TypeError::New(env, msg.str()).ThrowAsJavaScriptException(); msg << "Expected " << name << " to be a Uint8Array";
return nullptr; Napi::TypeError::New(env, msg.str()).ThrowAsJavaScriptException();
} return nullptr;
Napi::Uint8Array array = val.As<Napi::Uint8Array>(); }
if (array.ByteLength() != length) { Napi::Uint8Array array = val.As<Napi::Uint8Array>();
std::ostringstream msg; if (array.ByteLength() != length) {
msg << "Expected " << name << " to be " << length << " bytes"; std::ostringstream msg;
Napi::TypeError::New(env, msg.str()).ThrowAsJavaScriptException(); msg << "Expected " << name << " to be " << length << " bytes";
return nullptr; Napi::TypeError::New(env, msg.str()).ThrowAsJavaScriptException();
} return nullptr;
return array.Data(); }
return array.Data();
} }
inline Blob *get_blob(const Napi::Env &env, const Napi::Value &val) { inline Blob *get_blob(const Napi::Env &env, const Napi::Value &val) {
return reinterpret_cast<Blob *>(get_bytes(env, val, BYTES_PER_BLOB, "blob")); return reinterpret_cast<Blob *>(get_bytes(env, val, BYTES_PER_BLOB, "blob")
);
} }
inline Bytes32 *get_bytes32(const Napi::Env &env, const Napi::Value &val, std::string_view name) { inline Bytes32 *get_bytes32(
return reinterpret_cast<Bytes32 *>(get_bytes(env, val, BYTES_PER_FIELD_ELEMENT, name)); const Napi::Env &env, const Napi::Value &val, std::string_view name
) {
return reinterpret_cast<Bytes32 *>(
get_bytes(env, val, BYTES_PER_FIELD_ELEMENT, name)
);
} }
inline Bytes48 *get_bytes48(const Napi::Env &env, const Napi::Value &val, std::string_view name) { inline Bytes48 *get_bytes48(
return reinterpret_cast<Bytes48 *>(get_bytes(env, val, BYTES_PER_COMMITMENT, name)); const Napi::Env &env, const Napi::Value &val, std::string_view name
) {
return reinterpret_cast<Bytes48 *>(
get_bytes(env, val, BYTES_PER_COMMITMENT, name)
);
} }
Napi::Value LoadTrustedSetup(const Napi::CallbackInfo& info) { Napi::Value LoadTrustedSetup(const Napi::CallbackInfo &info) {
Napi::Env env = info.Env(); Napi::Env env = info.Env();
// Check if the trusted setup is already loaded // Check if the trusted setup is already loaded
KzgAddonData *data = env.GetInstanceData<KzgAddonData>(); KzgAddonData *data = env.GetInstanceData<KzgAddonData>();
if (data->is_setup) { if (data->is_setup) {
Napi::Error::New(env, "Error trusted setup is already loaded").ThrowAsJavaScriptException(); Napi::Error::New(env, "Error trusted setup is already loaded")
.ThrowAsJavaScriptException();
return env.Undefined();
}
// Open the trusted setup file
std::string file_path = info[0].As<Napi::String>().Utf8Value();
FILE *file_handle = fopen(file_path.c_str(), "r");
if (file_handle == nullptr) {
Napi::Error::New(env, "Error opening trusted setup file: " + file_path)
.ThrowAsJavaScriptException();
return env.Undefined();
}
// Load the trusted setup from that file
C_KZG_RET ret = load_trusted_setup_file(&(data->settings), file_handle);
// Close the trusted setup file
fclose(file_handle);
// Check that loading the trusted setup was successful
if (ret != C_KZG_OK) {
std::ostringstream msg;
msg << "Error loading trusted setup file: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
return env.Undefined();
}
data->is_setup = true;
return env.Undefined(); return env.Undefined();
}
// Open the trusted setup file
std::string file_path = info[0].As<Napi::String>().Utf8Value();
FILE *file_handle = fopen(file_path.c_str(), "r");
if (file_handle == nullptr) {
Napi::Error::New(env, "Error opening trusted setup file: " + file_path).ThrowAsJavaScriptException();
return env.Undefined();
}
// Load the trusted setup from that file
C_KZG_RET ret = load_trusted_setup_file(&(data->settings), file_handle);
// Close the trusted setup file
fclose(file_handle);
// Check that loading the trusted setup was successful
if (ret != C_KZG_OK) {
std::ostringstream msg;
msg << "Error loading trusted setup file: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
return env.Undefined();
}
data->is_setup = true;
return env.Undefined();
} }
/** /**
* Convert a blob to a KZG commitment. * Convert a blob to a KZG commitment.
* *
* @param[in] {Blob} blob - The blob representing the polynomial to be committed to * @param[in] {Blob} blob - The blob representing the polynomial to be
* committed to
* *
* @return {KZGCommitment} - The resulting commitment * @return {KZGCommitment} - The resulting commitment
* *
* @throws {TypeError} - For invalid arguments or failure of the native library * @throws {TypeError} - For invalid arguments or failure of the native library
*/ */
Napi::Value BlobToKzgCommitment(const Napi::CallbackInfo& info) { Napi::Value BlobToKzgCommitment(const Napi::CallbackInfo &info) {
Napi::Env env = info.Env(); Napi::Env env = info.Env();
Blob *blob = get_blob(env, info[0]); Blob *blob = get_blob(env, info[0]);
if (blob == nullptr) { if (blob == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGSettings *kzg_settings = get_kzg_settings(env, info); KZGSettings *kzg_settings = get_kzg_settings(env, info);
if (kzg_settings == nullptr) { if (kzg_settings == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGCommitment commitment; KZGCommitment commitment;
C_KZG_RET ret = blob_to_kzg_commitment(&commitment, blob, kzg_settings); C_KZG_RET ret = blob_to_kzg_commitment(&commitment, blob, kzg_settings);
if (ret != C_KZG_OK) { if (ret != C_KZG_OK) {
std::ostringstream msg; std::ostringstream msg;
msg << "Failed to convert blob to commitment: " << from_c_kzg_ret(ret) ; msg << "Failed to convert blob to commitment: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException(); Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
return env.Undefined(); return env.Undefined();
} }
return Napi::Buffer<uint8_t>::Copy(env, reinterpret_cast<uint8_t *>(&commitment), BYTES_PER_COMMITMENT); return Napi::Buffer<uint8_t>::Copy(
env, reinterpret_cast<uint8_t *>(&commitment), BYTES_PER_COMMITMENT
);
} }
/** /**
@ -222,49 +240,48 @@ Napi::Value BlobToKzgCommitment(const Napi::CallbackInfo& info) {
* @param[in] {Bytes32} zBytes - The generator z-value for the evaluation points * @param[in] {Bytes32} zBytes - The generator z-value for the evaluation points
* *
* @return {ProofResult} - Tuple containing the resulting proof and evaluation * @return {ProofResult} - Tuple containing the resulting proof and evaluation
* of the polynomial at the evaluation point z * of the polynomial at the evaluation point z
* *
* @throws {TypeError} - for invalid arguments or failure of the native library * @throws {TypeError} - for invalid arguments or failure of the native library
*/ */
Napi::Value ComputeKzgProof(const Napi::CallbackInfo& info) { Napi::Value ComputeKzgProof(const Napi::CallbackInfo &info) {
Napi::Env env = info.Env(); Napi::Env env = info.Env();
Blob *blob = get_blob(env, info[0]); Blob *blob = get_blob(env, info[0]);
if (blob == nullptr) { if (blob == nullptr) {
return env.Undefined(); return env.Null();
} }
Bytes32 *z_bytes = get_bytes32(env, info[1], "zBytes"); Bytes32 *z_bytes = get_bytes32(env, info[1], "zBytes");
if (z_bytes == nullptr) { if (z_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGSettings *kzg_settings = get_kzg_settings(env, info); KZGSettings *kzg_settings = get_kzg_settings(env, info);
if (kzg_settings == nullptr) { if (kzg_settings == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGProof proof; KZGProof proof;
Bytes32 y_out; Bytes32 y_out;
C_KZG_RET ret = compute_kzg_proof( C_KZG_RET ret = compute_kzg_proof(
&proof, &proof, &y_out, blob, z_bytes, kzg_settings
&y_out, );
blob,
z_bytes,
kzg_settings
);
if (ret != C_KZG_OK) { if (ret != C_KZG_OK) {
std::ostringstream msg; std::ostringstream msg;
msg << "Failed to compute proof: " << from_c_kzg_ret(ret) ; msg << "Failed to compute proof: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException(); Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
return env.Undefined(); return env.Undefined();
} }
Napi::Array tuple = Napi::Array::New(env, 2); Napi::Array tuple = Napi::Array::New(env, 2);
tuple[(uint32_t)0] = Napi::Buffer<uint8_t>::Copy(env, reinterpret_cast<uint8_t *>(&proof), BYTES_PER_PROOF); tuple[(uint32_t)0] = Napi::Buffer<uint8_t>::Copy(
tuple[(uint32_t)1] = Napi::Buffer<uint8_t>::Copy(env, reinterpret_cast<uint8_t *>(&y_out), BYTES_PER_FIELD_ELEMENT); env, reinterpret_cast<uint8_t *>(&proof), BYTES_PER_PROOF
return tuple; );
tuple[(uint32_t)1] = Napi::Buffer<uint8_t>::Copy(
env, reinterpret_cast<uint8_t *>(&y_out), BYTES_PER_FIELD_ELEMENT
);
return tuple;
} }
/** /**
* Given a blob, return the KZG proof that is used to verify it against the * Given a blob, return the KZG proof that is used to verify it against the
* commitment. * commitment.
@ -276,43 +293,43 @@ Napi::Value ComputeKzgProof(const Napi::CallbackInfo& info) {
* *
* @throws {TypeError} - for invalid arguments or failure of the native library * @throws {TypeError} - for invalid arguments or failure of the native library
*/ */
Napi::Value ComputeBlobKzgProof(const Napi::CallbackInfo& info) { Napi::Value ComputeBlobKzgProof(const Napi::CallbackInfo &info) {
Napi::Env env = info.Env(); Napi::Env env = info.Env();
Blob *blob = get_blob(env, info[0]); Blob *blob = get_blob(env, info[0]);
if (blob == nullptr) { if (blob == nullptr) {
return env.Undefined(); return env.Null();
} }
Bytes48 *commitment_bytes = get_bytes48(env, info[1], "commitmentBytes"); Bytes48 *commitment_bytes = get_bytes48(env, info[1], "commitmentBytes");
if (commitment_bytes == nullptr) { if (commitment_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGSettings *kzg_settings = get_kzg_settings(env, info); KZGSettings *kzg_settings = get_kzg_settings(env, info);
if (kzg_settings == nullptr) { if (kzg_settings == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGProof proof; KZGProof proof;
C_KZG_RET ret = compute_blob_kzg_proof( C_KZG_RET ret = compute_blob_kzg_proof(
&proof, &proof, blob, commitment_bytes, kzg_settings
blob, );
commitment_bytes,
kzg_settings
);
if (ret != C_KZG_OK) { if (ret != C_KZG_OK) {
std::ostringstream msg; std::ostringstream msg;
msg << "Error in computeBlobKzgProof: " << from_c_kzg_ret(ret) ; msg << "Error in computeBlobKzgProof: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException(); Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
return env.Undefined(); return env.Undefined();
} }
return Napi::Buffer<uint8_t>::Copy(env, reinterpret_cast<uint8_t *>(&proof), BYTES_PER_PROOF); return Napi::Buffer<uint8_t>::Copy(
env, reinterpret_cast<uint8_t *>(&proof), BYTES_PER_PROOF
);
} }
/** /**
* Verify a KZG poof claiming that `p(z) == y`. * Verify a KZG poof claiming that `p(z) == y`.
* *
* @param[in] {Bytes48} commitmentBytes - The serialized commitment corresponding to polynomial p(x) * @param[in] {Bytes48} commitmentBytes - The serialized commitment
* corresponding to polynomial p(x)
* @param[in] {Bytes32} zBytes - The serialized evaluation point * @param[in] {Bytes32} zBytes - The serialized evaluation point
* @param[in] {Bytes32} yBytes - The serialized claimed evaluation result * @param[in] {Bytes32} yBytes - The serialized claimed evaluation result
* @param[in] {Bytes48} proofBytes - The serialized KZG proof * @param[in] {Bytes48} proofBytes - The serialized KZG proof
@ -321,47 +338,42 @@ Napi::Value ComputeBlobKzgProof(const Napi::CallbackInfo& info) {
* *
* @throws {TypeError} - for invalid arguments or failure of the native library * @throws {TypeError} - for invalid arguments or failure of the native library
*/ */
Napi::Value VerifyKzgProof(const Napi::CallbackInfo& info) { Napi::Value VerifyKzgProof(const Napi::CallbackInfo &info) {
Napi::Env env = info.Env(); Napi::Env env = info.Env();
Bytes48 *commitment_bytes = get_bytes48(env, info[0], "commitmentBytes"); Bytes48 *commitment_bytes = get_bytes48(env, info[0], "commitmentBytes");
if (commitment_bytes == nullptr) { if (commitment_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
Bytes32 *z_bytes = get_bytes32(env, info[1], "zBytes"); Bytes32 *z_bytes = get_bytes32(env, info[1], "zBytes");
if (z_bytes == nullptr) { if (z_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
Bytes32 *y_bytes = get_bytes32(env, info[2], "yBytes"); Bytes32 *y_bytes = get_bytes32(env, info[2], "yBytes");
if (y_bytes == nullptr) { if (y_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
Bytes48 *proof_bytes = get_bytes48(env, info[3], "proofBytes"); Bytes48 *proof_bytes = get_bytes48(env, info[3], "proofBytes");
if (proof_bytes == nullptr) { if (proof_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGSettings *kzg_settings = get_kzg_settings(env, info); KZGSettings *kzg_settings = get_kzg_settings(env, info);
if (kzg_settings == nullptr) { if (kzg_settings == nullptr) {
return env.Undefined(); return env.Null();
} }
bool out; bool out;
C_KZG_RET ret = verify_kzg_proof( C_KZG_RET ret = verify_kzg_proof(
&out, &out, commitment_bytes, z_bytes, y_bytes, proof_bytes, kzg_settings
commitment_bytes, );
z_bytes,
y_bytes,
proof_bytes,
kzg_settings
);
if (ret != C_KZG_OK) { if (ret != C_KZG_OK) {
std::ostringstream msg; std::ostringstream msg;
msg << "Failed to verify KZG proof: " << from_c_kzg_ret(ret) ; msg << "Failed to verify KZG proof: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException(); Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
return env.Undefined(); return env.Undefined();
} }
return Napi::Boolean::New(env, out); return Napi::Boolean::New(env, out);
} }
/** /**
@ -376,173 +388,201 @@ Napi::Value VerifyKzgProof(const Napi::CallbackInfo& info) {
* *
* @throws {TypeError} - for invalid arguments or failure of the native library * @throws {TypeError} - for invalid arguments or failure of the native library
*/ */
Napi::Value VerifyBlobKzgProof(const Napi::CallbackInfo& info) { Napi::Value VerifyBlobKzgProof(const Napi::CallbackInfo &info) {
Napi::Env env = info.Env(); Napi::Env env = info.Env();
Blob *blob_bytes = get_blob(env, info[0]); Blob *blob_bytes = get_blob(env, info[0]);
if (blob_bytes == nullptr) { if (blob_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
Bytes48 *commitment_bytes = get_bytes48(env, info[1], "commitmentBytes"); Bytes48 *commitment_bytes = get_bytes48(env, info[1], "commitmentBytes");
if (commitment_bytes == nullptr) { if (commitment_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
Bytes48 *proof_bytes = get_bytes48(env, info[2], "proofBytes"); Bytes48 *proof_bytes = get_bytes48(env, info[2], "proofBytes");
if (proof_bytes == nullptr) { if (proof_bytes == nullptr) {
return env.Undefined(); return env.Null();
} }
KZGSettings *kzg_settings = get_kzg_settings(env, info); KZGSettings *kzg_settings = get_kzg_settings(env, info);
if (kzg_settings == nullptr) { if (kzg_settings == nullptr) {
return env.Undefined(); return env.Null();
} }
bool out; bool out;
C_KZG_RET ret = verify_blob_kzg_proof( C_KZG_RET ret = verify_blob_kzg_proof(
&out, &out, blob_bytes, commitment_bytes, proof_bytes, kzg_settings
blob_bytes, );
commitment_bytes,
proof_bytes,
kzg_settings);
if (ret != C_KZG_OK) { if (ret != C_KZG_OK) {
std::ostringstream msg; std::ostringstream msg;
msg << "Error in verifyBlobKzgProof: " << from_c_kzg_ret(ret) ; msg << "Error in verifyBlobKzgProof: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException(); Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
return env.Undefined(); return env.Undefined();
} }
return Napi::Boolean::New(env, out); return Napi::Boolean::New(env, out);
} }
/** /**
* Given an array of blobs and their proofs, verify that they corresponds to their * Given an array of blobs and their proofs, verify that they corresponds to
* provided commitment. * their provided commitment.
* *
* @remark blobs[0] relates to commitmentBytes[0] and proofBytes[0] * @remark blobs[0] relates to commitmentBytes[0] and proofBytes[0]
* *
* @param[in] {Blob} blobs - An array of serialized blobs to verify * @param[in] {Blob} blobs - An array of serialized blobs to verify
* @param[in] {Bytes48} commitmentBytes - An array of serialized commitments to verify * @param[in] {Bytes48} commitmentBytes - An array of serialized commitments to
* @param[in] {Bytes48} proofBytes - An array of serialized KZG proofs for verification * verify
* @param[in] {Bytes48} proofBytes - An array of serialized KZG proofs for
* verification
* *
* @return {boolean} - true/false depending on batch validity * @return {boolean} - true/false depending on batch validity
* *
* @throws {TypeError} - for invalid arguments or failure of the native library * @throws {TypeError} - for invalid arguments or failure of the native library
*/ */
Napi::Value VerifyBlobKzgProofBatch(const Napi::CallbackInfo& info) { Napi::Value VerifyBlobKzgProofBatch(const Napi::CallbackInfo &info) {
Napi::Env env = info.Env(); Napi::Env env = info.Env();
C_KZG_RET ret; C_KZG_RET ret;
Blob *blobs = NULL; Blob *blobs = NULL;
Bytes48 *commitments = NULL; Bytes48 *commitments = NULL;
Bytes48 *proofs = NULL; Bytes48 *proofs = NULL;
Napi::Value result = env.Null(); Napi::Value result = env.Null();
if (!(info[0].IsArray() && info[1].IsArray() && info[2].IsArray())) { if (!(info[0].IsArray() && info[1].IsArray() && info[2].IsArray())) {
Napi::Error::New(env, "Blobs, commitments, and proofs must all be arrays").ThrowAsJavaScriptException(); Napi::Error::New(
return result; env, "Blobs, commitments, and proofs must all be arrays"
} )
Napi::Array blobs_param = info[0].As<Napi::Array>(); .ThrowAsJavaScriptException();
Napi::Array commitments_param = info[1].As<Napi::Array>(); return result;
Napi::Array proofs_param = info[2].As<Napi::Array>();
KZGSettings *kzg_settings = get_kzg_settings(env, info);
if (kzg_settings == nullptr) {
return env.Undefined();
}
uint32_t count = blobs_param.Length();
if (count != commitments_param.Length() || count != proofs_param.Length()) {
Napi::Error::New(env, "Requires equal number of blobs/commitments/proofs").ThrowAsJavaScriptException();
return result;
}
blobs = (Blob *)calloc(count, sizeof(Blob));
if (blobs == nullptr) {
Napi::Error::New(env, "Error while allocating memory for blobs").ThrowAsJavaScriptException();
goto out;
}
commitments = (Bytes48 *)calloc(count, sizeof(Bytes48));
if (commitments == nullptr) {
Napi::Error::New(env, "Error while allocating memory for commitments").ThrowAsJavaScriptException();
goto out;
}
proofs = (Bytes48 *)calloc(count, sizeof(Bytes48));
if (proofs == nullptr) {
Napi::Error::New(env, "Error while allocating memory for proofs").ThrowAsJavaScriptException();
goto out;
}
for (uint32_t index = 0; index < count; index++) {
// add HandleScope here to release reference to temp values
// after each iteration since data is being memcpy
Napi::HandleScope scope{env};
Blob *blob = get_blob(env, blobs_param[index]);
if (blob == nullptr) {
goto out;
} }
memcpy(&blobs[index], blob, BYTES_PER_BLOB); Napi::Array blobs_param = info[0].As<Napi::Array>();
Bytes48 *commitment = get_bytes48(env, commitments_param[index], "commitmentBytes"); Napi::Array commitments_param = info[1].As<Napi::Array>();
if (commitment == nullptr) { Napi::Array proofs_param = info[2].As<Napi::Array>();
goto out; KZGSettings *kzg_settings = get_kzg_settings(env, info);
if (kzg_settings == nullptr) {
return env.Null();
} }
memcpy(&commitments[index], commitment, BYTES_PER_COMMITMENT); uint32_t count = blobs_param.Length();
Bytes48 *proof = get_bytes48(env, proofs_param[index], "proofBytes"); if (count != commitments_param.Length() || count != proofs_param.Length()) {
if (proof == nullptr) { Napi::Error::New(
goto out; env, "Requires equal number of blobs/commitments/proofs"
)
.ThrowAsJavaScriptException();
return result;
}
blobs = (Blob *)calloc(count, sizeof(Blob));
if (blobs == nullptr) {
Napi::Error::New(env, "Error while allocating memory for blobs")
.ThrowAsJavaScriptException();
goto out;
}
commitments = (Bytes48 *)calloc(count, sizeof(Bytes48));
if (commitments == nullptr) {
Napi::Error::New(env, "Error while allocating memory for commitments")
.ThrowAsJavaScriptException();
goto out;
}
proofs = (Bytes48 *)calloc(count, sizeof(Bytes48));
if (proofs == nullptr) {
Napi::Error::New(env, "Error while allocating memory for proofs")
.ThrowAsJavaScriptException();
goto out;
} }
memcpy(&proofs[index], proof, BYTES_PER_PROOF);
}
bool out; for (uint32_t index = 0; index < count; index++) {
ret = verify_blob_kzg_proof_batch( // add HandleScope here to release reference to temp values
&out, // after each iteration since data is being memcpy
blobs, Napi::HandleScope scope{env};
commitments, Blob *blob = get_blob(env, blobs_param[index]);
proofs, if (blob == nullptr) {
count, goto out;
kzg_settings }
); memcpy(&blobs[index], blob, BYTES_PER_BLOB);
Bytes48 *commitment = get_bytes48(
env, commitments_param[index], "commitmentBytes"
);
if (commitment == nullptr) {
goto out;
}
memcpy(&commitments[index], commitment, BYTES_PER_COMMITMENT);
Bytes48 *proof = get_bytes48(env, proofs_param[index], "proofBytes");
if (proof == nullptr) {
goto out;
}
memcpy(&proofs[index], proof, BYTES_PER_PROOF);
}
if (ret != C_KZG_OK) { bool out;
std::ostringstream msg; ret = verify_blob_kzg_proof_batch(
msg << "Error in verifyBlobKzgProofBatch: " << from_c_kzg_ret(ret) ; &out, blobs, commitments, proofs, count, kzg_settings
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException(); );
goto out;
}
result = Napi::Boolean::New(env, out); if (ret != C_KZG_OK) {
std::ostringstream msg;
msg << "Error in verifyBlobKzgProofBatch: " << from_c_kzg_ret(ret);
Napi::Error::New(env, msg.str()).ThrowAsJavaScriptException();
goto out;
}
result = Napi::Boolean::New(env, out);
out: out:
free(blobs); free(blobs);
free(commitments); free(commitments);
free(proofs); free(proofs);
return result; return result;
} }
Napi::Object Init(Napi::Env env, Napi::Object exports) {
KzgAddonData *data = (KzgAddonData *)malloc(sizeof(KzgAddonData));
if (data == nullptr) {
Napi::Error::New(env, "Error allocating memory for kzg setup handle")
.ThrowAsJavaScriptException();
return exports;
}
data->is_setup = false;
napi_status status = napi_set_instance_data(
env, data, delete_kzg_addon_data, NULL
);
if (status != napi_ok) {
Napi::Error::New(env, "Error setting kzg bindings instance data")
.ThrowAsJavaScriptException();
return exports;
}
Napi::Object Init(Napi::Env env, Napi::Object exports) { // Functions
KzgAddonData* data = (KzgAddonData*)malloc(sizeof(KzgAddonData)); exports["loadTrustedSetup"] = Napi::Function::New(
if (data == nullptr) { env, LoadTrustedSetup, "setup"
Napi::Error::New(env, "Error allocating memory for kzg setup handle").ThrowAsJavaScriptException(); );
exports["blobToKzgCommitment"] = Napi::Function::New(
env, BlobToKzgCommitment, "blobToKzgCommitment"
);
exports["computeKzgProof"] = Napi::Function::New(
env, ComputeKzgProof, "computeKzgProof"
);
exports["computeBlobKzgProof"] = Napi::Function::New(
env, ComputeBlobKzgProof, "computeBlobKzgProof"
);
exports["verifyKzgProof"] = Napi::Function::New(
env, VerifyKzgProof, "verifyKzgProof"
);
exports["verifyBlobKzgProof"] = Napi::Function::New(
env, VerifyBlobKzgProof, "verifyBlobKzgProof"
);
exports["verifyBlobKzgProofBatch"] = Napi::Function::New(
env, VerifyBlobKzgProofBatch, "verifyBlobKzgProofBatch"
);
// Constants
exports["BYTES_PER_BLOB"] = Napi::Number::New(env, BYTES_PER_BLOB);
exports["BYTES_PER_COMMITMENT"] = Napi::Number::New(
env, BYTES_PER_COMMITMENT
);
exports["BYTES_PER_FIELD_ELEMENT"] = Napi::Number::New(
env, BYTES_PER_FIELD_ELEMENT
);
exports["BYTES_PER_PROOF"] = Napi::Number::New(env, BYTES_PER_PROOF);
exports["FIELD_ELEMENTS_PER_BLOB"] = Napi::Number::New(
env, FIELD_ELEMENTS_PER_BLOB
);
return exports; return exports;
}
data->is_setup = false;
napi_status status = napi_set_instance_data(env, data, delete_kzg_addon_data, NULL);
if (status != napi_ok) {
Napi::Error::New(env, "Error setting kzg bindings instance data").ThrowAsJavaScriptException();
return exports;
}
// Functions
exports["loadTrustedSetup"] = Napi::Function::New(env, LoadTrustedSetup, "setup");
exports["blobToKzgCommitment"] = Napi::Function::New(env, BlobToKzgCommitment, "blobToKzgCommitment");
exports["computeKzgProof"] = Napi::Function::New(env, ComputeKzgProof, "computeKzgProof");
exports["computeBlobKzgProof"] = Napi::Function::New(env, ComputeBlobKzgProof, "computeBlobKzgProof");
exports["verifyKzgProof"] = Napi::Function::New(env, VerifyKzgProof, "verifyKzgProof");
exports["verifyBlobKzgProof"] = Napi::Function::New(env, VerifyBlobKzgProof, "verifyBlobKzgProof");
exports["verifyBlobKzgProofBatch"] = Napi::Function::New(env, VerifyBlobKzgProofBatch, "verifyBlobKzgProofBatch");
// Constants
exports["BYTES_PER_BLOB"] = Napi::Number::New(env, BYTES_PER_BLOB);
exports["BYTES_PER_COMMITMENT"] = Napi::Number::New(env, BYTES_PER_COMMITMENT);
exports["BYTES_PER_FIELD_ELEMENT"] = Napi::Number::New(env, BYTES_PER_FIELD_ELEMENT);
exports["BYTES_PER_PROOF"] = Napi::Number::New(env, BYTES_PER_PROOF);
exports["FIELD_ELEMENTS_PER_BLOB"] = Napi::Number::New(env, FIELD_ELEMENTS_PER_BLOB);
return exports;
} }
NODE_API_MODULE(addon, Init) NODE_API_MODULE(addon, Init)