From a7a2111266e5739b5783b32b16292823968ba051 Mon Sep 17 00:00:00 2001 From: Patrick Schork <354473+pschork@users.noreply.github.com> Date: Wed, 29 May 2024 17:12:29 -0700 Subject: [PATCH 1/4] batchgen --- tools/batchgen/Makefile | 12 ++ tools/batchgen/cmd/main.go | 232 +++++++++++++++++++++++++++++++++++++ 2 files changed, 244 insertions(+) create mode 100644 tools/batchgen/Makefile create mode 100644 tools/batchgen/cmd/main.go diff --git a/tools/batchgen/Makefile b/tools/batchgen/Makefile new file mode 100644 index 0000000000..ccb9b06290 --- /dev/null +++ b/tools/batchgen/Makefile @@ -0,0 +1,12 @@ +build: clean + go mod tidy + go build -o ./bin/batchgen ./cmd + +clean: + rm -rf ./bin + +proto: + protoc --go_out=. store_chunks.proto + +run: build + ./bin/batchgen localhost:32005 diff --git a/tools/batchgen/cmd/main.go b/tools/batchgen/cmd/main.go new file mode 100644 index 0000000000..33c3658dbe --- /dev/null +++ b/tools/batchgen/cmd/main.go @@ -0,0 +1,232 @@ +package main + +import ( + "context" + "log" + "os" + + //"errors" + //"time" + + commonpb "github.com/Layr-Labs/eigenda/api/grpc/common" + pb "github.com/Layr-Labs/eigenda/api/grpc/node" + "github.com/Layr-Labs/eigenda/core" + "github.com/Layr-Labs/eigenda/encoding" + "github.com/consensys/gnark-crypto/ecc/bn254/fp" + + "google.golang.org/grpc" + "google.golang.org/grpc/credentials/insecure" + "google.golang.org/protobuf/proto" +) + +var ( + encodedChunk = []byte{42, 255, 129, 3, 1, 1, 5, 67, 104, 117, 110, 107, 1, 255, 130, 0, 1, 2, 1, 6, 67, 111, 101, 102, 102, 115, 1, 255, 134, 0, 1, 5, 80, 114, 111, 111, 102, 1, 255, 136, 0, 0, 0, 25, 255, 133, 2, 1, 1, 10, 91, 93, 98, 110, 50, 53, 52, 46, 70, 114, 1, 255, 134, 0, 1, 255, 132, 0, 0, 18, 255, 131, 1, 1, 1, 2, 70, 114, 1, 255, 132, 0, 1, 6, 1, 8, 0, 0, 35, 255, 135, 3, 1, 1, 7, 71, 49, 80, 111, 105, 110, 116, 1, 255, 136, 0, 1, 2, 1, 1, 88, 1, 255, 138, 0, 1, 1, 89, 1, 255, 138, 0, 0, 0, 23, 255, 137, 1, 1, 1, 7, 69, 108, 101, 109, 101, 110, 116, 1, 255, 138, 0, 1, 6, 1, 8, 0, 0, 254, 4, 243, 255, 130, 1, 32, 4, 248, 186, 196, 96, 34, 212, 35, 97, 83, 248, 121, 9, 252, 220, 181, 118, 97, 134, 248, 186, 26, 225, 204, 191, 144, 133, 234, 248, 7, 223, 191, 156, 83, 115, 21, 36, 4, 248, 43, 196, 225, 43, 61, 88, 43, 49, 248, 28, 200, 121, 122, 178, 119, 200, 17, 248, 29, 172, 61, 194, 130, 114, 50, 171, 248, 33, 141, 185, 47, 11, 129, 128, 116, 4, 248, 246, 236, 255, 207, 43, 92, 176, 63, 248, 103, 179, 139, 80, 75, 57, 128, 89, 248, 107, 170, 70, 254, 95, 17, 101, 158, 248, 8, 106, 82, 82, 25, 78, 95, 104, 4, 248, 28, 125, 21, 116, 243, 255, 206, 10, 248, 153, 249, 156, 88, 61, 254, 171, 171, 248, 103, 66, 131, 8, 12, 165, 173, 173, 248, 36, 227, 189, 242, 180, 18, 171, 208, 4, 248, 19, 159, 205, 146, 86, 81, 57, 28, 248, 161, 130, 249, 92, 236, 82, 103, 4, 248, 84, 44, 63, 43, 249, 88, 187, 12, 248, 42, 121, 83, 118, 55, 127, 180, 134, 4, 248, 193, 39, 155, 110, 195, 113, 118, 46, 248, 47, 92, 162, 69, 188, 120, 94, 161, 248, 101, 214, 253, 103, 243, 8, 246, 176, 248, 41, 1, 238, 37, 43, 132, 228, 244, 4, 248, 70, 34, 194, 33, 68, 87, 108, 180, 248, 203, 230, 97, 137, 162, 177, 142, 23, 248, 101, 25, 216, 255, 137, 96, 240, 73, 248, 40, 50, 167, 154, 63, 108, 55, 240, 4, 248, 78, 40, 51, 224, 193, 131, 8, 90, 248, 162, 203, 245, 119, 83, 125, 219, 33, 248, 85, 109, 106, 231, 162, 152, 229, 110, 248, 38, 189, 66, 40, 176, 177, 114, 84, 4, 248, 193, 67, 43, 158, 218, 245, 83, 116, 248, 100, 165, 217, 161, 166, 209, 98, 172, 248, 231, 23, 45, 28, 225, 102, 143, 157, 248, 20, 12, 146, 122, 104, 126, 51, 235, 4, 248, 19, 118, 59, 144, 83, 246, 144, 229, 248, 203, 168, 161, 194, 137, 34, 191, 157, 248, 252, 196, 212, 78, 99, 166, 6, 225, 248, 29, 41, 54, 112, 125, 128, 240, 209, 4, 248, 24, 175, 53, 2, 113, 155, 113, 233, 248, 162, 189, 238, 198, 233, 31, 199, 239, 248, 205, 162, 128, 190, 163, 250, 181, 226, 248, 40, 205, 5, 117, 16, 49, 205, 45, 4, 248, 78, 49, 135, 21, 90, 93, 196, 50, 248, 115, 105, 77, 122, 222, 27, 224, 166, 248, 44, 0, 255, 63, 67, 184, 234, 235, 248, 45, 88, 39, 211, 138, 80, 43, 243, 4, 248, 244, 239, 154, 119, 68, 204, 215, 5, 248, 53, 82, 219, 150, 72, 243, 20, 147, 248, 141, 131, 101, 73, 11, 218, 234, 89, 248, 25, 246, 203, 17, 86, 91, 107, 199, 4, 248, 111, 106, 155, 101, 22, 163, 231, 214, 248, 86, 123, 235, 222, 87, 192, 80, 167, 248, 107, 38, 156, 175, 73, 123, 184, 189, 248, 23, 12, 154, 39, 153, 2, 158, 213, 4, 248, 40, 166, 62, 99, 6, 145, 128, 237, 248, 77, 160, 235, 64, 123, 181, 120, 66, 248, 116, 0, 126, 221, 26, 18, 100, 74, 248, 46, 92, 161, 252, 177, 177, 191, 127, 4, 248, 227, 144, 223, 154, 232, 249, 22, 233, 248, 53, 82, 148, 149, 84, 76, 107, 93, 248, 71, 251, 7, 58, 156, 200, 102, 4, 248, 3, 147, 75, 172, 199, 222, 109, 87, 4, 248, 169, 207, 109, 252, 37, 85, 158, 78, 248, 237, 12, 207, 255, 117, 62, 171, 3, 248, 43, 93, 155, 238, 136, 102, 150, 139, 248, 40, 174, 6, 46, 62, 50, 174, 104, 4, 248, 156, 217, 228, 156, 76, 202, 37, 121, 248, 80, 44, 200, 177, 237, 112, 103, 44, 248, 211, 172, 202, 164, 34, 242, 190, 204, 248, 15, 241, 94, 33, 88, 13, 34, 66, 4, 248, 198, 229, 9, 111, 155, 117, 84, 125, 248, 69, 115, 47, 6, 35, 132, 39, 86, 248, 243, 113, 79, 216, 240, 35, 72, 75, 248, 7, 29, 38, 85, 134, 106, 213, 236, 4, 248, 8, 8, 251, 11, 97, 66, 8, 55, 248, 159, 67, 100, 214, 31, 167, 88, 221, 248, 151, 110, 49, 190, 136, 249, 55, 217, 248, 47, 94, 78, 30, 0, 220, 176, 125, 4, 248, 246, 81, 132, 144, 151, 161, 113, 102, 248, 229, 8, 10, 180, 28, 223, 222, 8, 248, 158, 88, 212, 24, 77, 31, 96, 232, 248, 41, 65, 45, 216, 25, 224, 221, 4, 4, 248, 11, 189, 86, 122, 64, 254, 107, 253, 248, 242, 174, 32, 144, 43, 116, 187, 77, 248, 16, 163, 127, 128, 4, 233, 82, 168, 248, 4, 90, 126, 233, 232, 220, 81, 74, 4, 248, 54, 17, 20, 36, 220, 10, 168, 78, 248, 77, 61, 41, 4, 95, 154, 130, 70, 248, 37, 180, 163, 188, 242, 88, 81, 28, 248, 37, 195, 179, 103, 195, 0, 252, 30, 4, 248, 148, 154, 198, 22, 110, 201, 164, 240, 248, 242, 100, 163, 103, 30, 185, 139, 205, 248, 198, 168, 87, 116, 135, 219, 11, 230, 248, 43, 163, 196, 37, 51, 32, 130, 241, 4, 248, 160, 22, 80, 69, 111, 126, 3, 23, 248, 76, 89, 182, 79, 244, 245, 155, 42, 248, 144, 203, 89, 203, 85, 216, 109, 139, 248, 36, 125, 246, 94, 210, 7, 236, 50, 4, 248, 244, 42, 154, 219, 137, 78, 64, 167, 248, 73, 57, 191, 50, 122, 120, 124, 249, 248, 192, 102, 139, 159, 135, 150, 18, 35, 248, 40, 167, 252, 247, 112, 215, 52, 61, 4, 248, 151, 181, 121, 81, 121, 147, 227, 13, 248, 236, 181, 178, 176, 243, 4, 136, 195, 248, 62, 97, 145, 239, 166, 114, 175, 107, 248, 23, 91, 75, 217, 198, 192, 155, 92, 4, 248, 182, 191, 150, 70, 229, 96, 122, 14, 248, 134, 0, 111, 72, 36, 162, 244, 220, 248, 168, 72, 14, 253, 239, 166, 139, 197, 248, 44, 139, 158, 151, 191, 127, 27, 222, 4, 248, 74, 171, 39, 27, 36, 31, 102, 30, 248, 41, 77, 140, 191, 229, 182, 30, 16, 248, 219, 194, 193, 143, 239, 141, 47, 73, 248, 23, 1, 236, 49, 51, 57, 155, 228, 4, 248, 128, 145, 254, 105, 104, 55, 224, 206, 248, 195, 70, 112, 120, 42, 171, 202, 23, 248, 242, 232, 247, 249, 215, 77, 208, 121, 248, 29, 0, 45, 26, 151, 224, 199, 214, 4, 248, 235, 253, 108, 246, 112, 139, 56, 187, 248, 214, 211, 157, 43, 210, 247, 57, 203, 248, 150, 28, 35, 231, 169, 220, 146, 139, 248, 48, 54, 207, 130, 116, 140, 125, 197, 4, 248, 23, 120, 154, 57, 66, 85, 149, 5, 248, 170, 172, 192, 127, 230, 130, 224, 17, 248, 117, 98, 19, 140, 134, 78, 47, 98, 248, 40, 206, 62, 254, 165, 238, 160, 130, 1, 1, 4, 248, 164, 40, 240, 180, 149, 114, 87, 82, 248, 195, 115, 109, 187, 95, 132, 65, 10, 248, 176, 59, 100, 197, 207, 37, 161, 253, 248, 10, 19, 137, 98, 39, 77, 128, 20, 1, 4, 248, 213, 212, 69, 58, 138, 39, 69, 249, 248, 99, 187, 162, 108, 114, 239, 78, 157, 248, 62, 166, 165, 148, 83, 202, 37, 169, 248, 47, 253, 18, 76, 216, 168, 22, 21, 0, 0} + opID [32]byte +) + +func main() { + req, _, _, _, _ := makeStoreChunksRequest(100, 10) + + host := os.Args[1] + log.Println("Connecting to operator", "operator", host) + conn, err := grpc.Dial(host, grpc.WithTransportCredentials(insecure.NewCredentials())) + if err != nil { + log.Println("Batchgen cannot connect to operator dispersal socket", "dispersal_socket", host, "err", err) + return + } + defer conn.Close() + + client := pb.NewDispersalClient(conn) + //opt := grpc.MaxCallSendMsgSize(60 * 1024 * 1024 * 1024) + + log.Println("sending chunks to operator", "operator", host, "request message size", proto.Size(req)) + reply, err := client.StoreChunks(context.Background(), req) + + if err != nil { + log.Println("Reply err", err) + return + } + log.Println("Reply", reply) + log.Println("Signature", reply.GetSignature()) +} + +func makeStoreChunksRequest(quorumThreshold, adversaryThreshold uint8) (*pb.StoreChunksRequest, [32]byte, [32]byte, []*core.BlobHeader, []*pb.BlobHeader) { + var commitX, commitY fp.Element + _, err := commitX.SetString("21661178944771197726808973281966770251114553549453983978976194544185382599016") + if err != nil { + log.Println("Error setting commitX") + } + _, err = commitY.SetString("9207254729396071334325696286939045899948985698134704137261649190717970615186") + if err != nil { + log.Println("Error setting commitY") + } + + commitment := &encoding.G1Commitment{ + X: commitX, + Y: commitY, + } + var lengthXA0, lengthXA1, lengthYA0, lengthYA1 fp.Element + _, err = lengthXA0.SetString("10857046999023057135944570762232829481370756359578518086990519993285655852781") + _, err = lengthXA1.SetString("11559732032986387107991004021392285783925812861821192530917403151452391805634") + _, err = lengthYA0.SetString("8495653923123431417604973247489272438418190587263600148770280649306958101930") + _, err = lengthYA1.SetString("4082367875863433681332203403145435568316851327593401208105741076214120093531") + + var lengthProof, lengthCommitment encoding.G2Commitment + lengthProof.X.A0 = lengthXA0 + lengthProof.X.A1 = lengthXA1 + lengthProof.Y.A0 = lengthYA0 + lengthProof.Y.A1 = lengthYA1 + + lengthCommitment = lengthProof + + quorumHeader := &core.BlobQuorumInfo{ + SecurityParam: core.SecurityParam{ + QuorumID: 0, + ConfirmationThreshold: quorumThreshold, + AdversaryThreshold: adversaryThreshold, + }, + ChunkLength: 10, + } + + quorumHeader1 := &core.BlobQuorumInfo{ + SecurityParam: core.SecurityParam{ + QuorumID: 1, + ConfirmationThreshold: 65, + AdversaryThreshold: 15, + }, + ChunkLength: 10, + } + + blobHeaders := []*core.BlobHeader{ + { + BlobCommitments: encoding.BlobCommitments{ + Commitment: commitment, + LengthCommitment: &lengthCommitment, + LengthProof: &lengthProof, + Length: 48, + }, + QuorumInfos: []*core.BlobQuorumInfo{quorumHeader, quorumHeader1}, + }, + { + BlobCommitments: encoding.BlobCommitments{ + Commitment: commitment, + LengthCommitment: &lengthCommitment, + LengthProof: &lengthProof, + Length: 50, + }, + QuorumInfos: []*core.BlobQuorumInfo{quorumHeader}, + }, + } + batchHeader := core.BatchHeader{ + BatchRoot: [32]byte{0}, + ReferenceBlockNumber: 1, + } + + _, err = batchHeader.SetBatchRoot(blobHeaders) + + batchHeaderHash, err := batchHeader.GetBatchHeaderHash() + + blobHeaderProto0 := blobHeaderToProto(blobHeaders[0]) + blobHeaderProto1 := blobHeaderToProto(blobHeaders[1]) + + req := &pb.StoreChunksRequest{ + BatchHeader: &pb.BatchHeader{ + BatchRoot: batchHeader.BatchRoot[:], + ReferenceBlockNumber: uint32(batchHeader.ReferenceBlockNumber), + }, + Blobs: []*pb.Blob{ + { + Header: blobHeaderProto0, + Bundles: []*pb.Bundle{ + { + Chunks: [][]byte{encodedChunk}, + }, + { + // Empty bundle for the second quorum + Chunks: [][]byte{}, + }, + }, + }, + { + Header: blobHeaderProto1, + Bundles: []*pb.Bundle{ + { + Chunks: [][]byte{encodedChunk}, + }, + }, + }, + }, + } + blobHeadersProto := []*pb.BlobHeader{blobHeaderProto0, blobHeaderProto1} + + return req, batchHeaderHash, batchHeader.BatchRoot, blobHeaders, blobHeadersProto +} + +func blobHeaderToProto(blobHeader *core.BlobHeader) *pb.BlobHeader { + var lengthCommitment, lengthProof pb.G2Commitment + if blobHeader.LengthCommitment != nil { + lengthCommitment.XA0 = blobHeader.LengthCommitment.X.A0.Marshal() + lengthCommitment.XA1 = blobHeader.LengthCommitment.X.A1.Marshal() + lengthCommitment.YA0 = blobHeader.LengthCommitment.Y.A0.Marshal() + lengthCommitment.YA1 = blobHeader.LengthCommitment.Y.A1.Marshal() + } + if blobHeader.LengthProof != nil { + lengthProof.XA0 = blobHeader.LengthProof.X.A0.Marshal() + lengthProof.XA1 = blobHeader.LengthProof.X.A1.Marshal() + lengthProof.YA0 = blobHeader.LengthProof.Y.A0.Marshal() + lengthProof.YA1 = blobHeader.LengthProof.Y.A1.Marshal() + } + quorumHeaders := make([]*pb.BlobQuorumInfo, len(blobHeader.QuorumInfos)) + for i, quorumInfo := range blobHeader.QuorumInfos { + quorumHeaders[i] = &pb.BlobQuorumInfo{ + QuorumId: uint32(quorumInfo.QuorumID), + ConfirmationThreshold: uint32(quorumInfo.ConfirmationThreshold), + AdversaryThreshold: uint32(quorumInfo.AdversaryThreshold), + ChunkLength: uint32(quorumInfo.ChunkLength), + } + } + + return &pb.BlobHeader{ + Commitment: &commonpb.G1Commitment{ + X: blobHeader.Commitment.X.Marshal(), + Y: blobHeader.Commitment.Y.Marshal(), + }, + LengthCommitment: &lengthCommitment, + LengthProof: &lengthProof, + Length: uint32(blobHeader.Length), + QuorumHeaders: quorumHeaders, + } +} + +func generateRequest(size int64) (*pb.StoreChunksRequest, error) { + log.Println("Building request of size", size) + // Create a BatchHeader + batchHeader := &pb.BatchHeader{ + BatchRoot: []byte("root_hash_example"), + ReferenceBlockNumber: 12345, + } + + // Create a BlobHeader + blobHeader := &pb.BlobHeader{ + // Add fields here if necessary + } + + // Create a Bundle + bundle := &pb.Bundle{ + Chunks: [][]byte{[]byte("chunk1"), []byte("chunk2"), []byte("chunk3")}, + } + + // Create a Blob + blob := &pb.Blob{ + Header: blobHeader, + Bundles: []*pb.Bundle{bundle}, + } + + // Create the StoreChunksRequest + storeChunksRequest := &pb.StoreChunksRequest{ + BatchHeader: batchHeader, + Blobs: []*pb.Blob{blob}, + } + + return storeChunksRequest, nil +} From e9eddd133a89cbab2ff367b0347d3f03097ffa04 Mon Sep 17 00:00:00 2001 From: Patrick Schork <354473+pschork@users.noreply.github.com> Date: Wed, 29 May 2024 17:39:57 -0700 Subject: [PATCH 2/4] Add scaling factor to encoded chunk Add goroutine --- tools/batchgen/Makefile | 2 +- tools/batchgen/cmd/main.go | 74 +++++++++++++++++++++++++------------- 2 files changed, 51 insertions(+), 25 deletions(-) diff --git a/tools/batchgen/Makefile b/tools/batchgen/Makefile index ccb9b06290..cf52ef5b2a 100644 --- a/tools/batchgen/Makefile +++ b/tools/batchgen/Makefile @@ -9,4 +9,4 @@ proto: protoc --go_out=. store_chunks.proto run: build - ./bin/batchgen localhost:32005 + ./bin/batchgen 10000 localhost:32005 localhost:32005 diff --git a/tools/batchgen/cmd/main.go b/tools/batchgen/cmd/main.go index 33c3658dbe..4428faa810 100644 --- a/tools/batchgen/cmd/main.go +++ b/tools/batchgen/cmd/main.go @@ -4,9 +4,7 @@ import ( "context" "log" "os" - - //"errors" - //"time" + "strconv" commonpb "github.com/Layr-Labs/eigenda/api/grpc/common" pb "github.com/Layr-Labs/eigenda/api/grpc/node" @@ -25,32 +23,60 @@ var ( ) func main() { - req, _, _, _, _ := makeStoreChunksRequest(100, 10) - - host := os.Args[1] - log.Println("Connecting to operator", "operator", host) - conn, err := grpc.Dial(host, grpc.WithTransportCredentials(insecure.NewCredentials())) + factor, err := strconv.Atoi(os.Args[1]) if err != nil { - log.Println("Batchgen cannot connect to operator dispersal socket", "dispersal_socket", host, "err", err) + log.Println("failed to convert factor to int:", err) return } - defer conn.Close() + req, _, _, _, _ := makeStoreChunksRequest(100, 10, factor) - client := pb.NewDispersalClient(conn) - //opt := grpc.MaxCallSendMsgSize(60 * 1024 * 1024 * 1024) + hosts := os.Args[2:] // assuming multiple hosts are passed as command line arguments + results := make(chan *pb.StoreChunksReply, len(hosts)) // channel to collect results + errors := make(chan error, len(hosts)) // channel to collect errors - log.Println("sending chunks to operator", "operator", host, "request message size", proto.Size(req)) - reply, err := client.StoreChunks(context.Background(), req) + for _, host := range hosts { + go func(host string) { + conn, err := grpc.Dial(host, grpc.WithTransportCredentials(insecure.NewCredentials())) + if err != nil { + errors <- err + return + } + defer conn.Close() - if err != nil { - log.Println("Reply err", err) - return + client := pb.NewDispersalClient(conn) + opt := grpc.MaxCallSendMsgSize(60 * 1024 * 1024 * 1024) + + log.Println("sending chunks to operator", host, "request message size", proto.Size(req)) + reply, err := client.StoreChunks(context.Background(), req, opt) + if err != nil { + errors <- err + } else { + results <- reply + } + }(host) + } + + // Wait for all goroutines to finish and collect results + for i := 0; i < len(hosts); i++ { + select { + case result := <-results: + log.Println("reply", result) + case err := <-errors: + log.Println("error", err) + } } - log.Println("Reply", reply) - log.Println("Signature", reply.GetSignature()) } -func makeStoreChunksRequest(quorumThreshold, adversaryThreshold uint8) (*pb.StoreChunksRequest, [32]byte, [32]byte, []*core.BlobHeader, []*pb.BlobHeader) { +func makeStoreChunksRequest(quorumThreshold, adversaryThreshold uint8, factor int) (*pb.StoreChunksRequest, [32]byte, [32]byte, []*core.BlobHeader, []*pb.BlobHeader) { + // Create a new slice with the required length + newSize := len(encodedChunk) * factor + newEncodedChunk := make([]byte, newSize) + + // Fill the new slice with repeated data from the original slice + for i := 0; i < factor; i++ { + copy(newEncodedChunk[i*len(encodedChunk):], encodedChunk) + } + var commitX, commitY fp.Element _, err := commitX.SetString("21661178944771197726808973281966770251114553549453983978976194544185382599016") if err != nil { @@ -139,11 +165,11 @@ func makeStoreChunksRequest(quorumThreshold, adversaryThreshold uint8) (*pb.Stor Header: blobHeaderProto0, Bundles: []*pb.Bundle{ { - Chunks: [][]byte{encodedChunk}, + Chunks: [][]byte{newEncodedChunk}, }, { - // Empty bundle for the second quorum - Chunks: [][]byte{}, + // second quorum + Chunks: [][]byte{newEncodedChunk}, }, }, }, @@ -151,7 +177,7 @@ func makeStoreChunksRequest(quorumThreshold, adversaryThreshold uint8) (*pb.Stor Header: blobHeaderProto1, Bundles: []*pb.Bundle{ { - Chunks: [][]byte{encodedChunk}, + Chunks: [][]byte{newEncodedChunk}, }, }, }, From 07d1fb96a00bc7b7e2f9170303b5c641b0da8dc3 Mon Sep 17 00:00:00 2001 From: Patrick Schork <354473+pschork@users.noreply.github.com> Date: Wed, 5 Jun 2024 02:10:49 -0700 Subject: [PATCH 3/4] wip --- node/grpc/server.go | 3 - node/node.go | 15 +++++ tools/batchgen/Makefile | 6 +- tools/batchgen/cmd/main.go | 119 ++++++++++++++++------------------ tools/batchgen/flags/flags.go | 45 +++++++++++++ 5 files changed, 119 insertions(+), 69 deletions(-) create mode 100644 tools/batchgen/flags/flags.go diff --git a/node/grpc/server.go b/node/grpc/server.go index 73dabc436f..93b24989d9 100644 --- a/node/grpc/server.go +++ b/node/grpc/server.go @@ -164,9 +164,6 @@ func (s *Server) validateStoreChunkRequest(in *pb.StoreChunksRequest) error { if in.GetBatchHeader().GetBatchRoot() == nil { return api.NewInvalidArgError("missing batch_root in request") } - if in.GetBatchHeader().GetReferenceBlockNumber() == 0 { - return api.NewInvalidArgError("missing reference_block_number in request") - } if len(in.GetBlobs()) == 0 { return api.NewInvalidArgError("missing blobs in request") diff --git a/node/node.go b/node/node.go index dff4651629..198a12be52 100644 --- a/node/node.go +++ b/node/node.go @@ -347,6 +347,21 @@ func (n *Node) ProcessBatch(ctx context.Context, header *core.BatchHeader, blobs storeChan <- storeResult{err: nil, keys: keys, latency: time.Since(start)} }(n) + // Detect batch probe request + if header.ReferenceBlockNumber == 0 { + log.Info("Batch probe request detected", "batchHeaderHash", batchHeaderHashHex) + result := <-storeChan + if result.keys != nil { + log.Debug("Rolling back the probe request key/value entries stored in database", "number of entires", len(*result.keys), "batchHeaderHash", batchHeaderHash) + if deleteKeysErr := n.Store.DeleteKeys(ctx, result.keys); deleteKeysErr != nil { + log.Error("Failed to delete the batch probe that should be rolled back", "batchHeaderHash", batchHeaderHashHex, "err", deleteKeysErr) + } + } + // Sign batch header hash and return + sig := n.KeyPair.SignMessage(batchHeaderHash) + return sig, nil + } + // Validate batch. stageTimer := time.Now() err = n.ValidateBatch(ctx, header, blobs) diff --git a/tools/batchgen/Makefile b/tools/batchgen/Makefile index cf52ef5b2a..e27e4763dc 100644 --- a/tools/batchgen/Makefile +++ b/tools/batchgen/Makefile @@ -5,8 +5,6 @@ build: clean clean: rm -rf ./bin -proto: - protoc --go_out=. store_chunks.proto - run: build - ./bin/batchgen 10000 localhost:32005 localhost:32005 + ./bin/batchgen 1000 10 localhost:32005 + diff --git a/tools/batchgen/cmd/main.go b/tools/batchgen/cmd/main.go index 4428faa810..c8f92b7ded 100644 --- a/tools/batchgen/cmd/main.go +++ b/tools/batchgen/cmd/main.go @@ -2,6 +2,7 @@ package main import ( "context" + "fmt" "log" "os" "strconv" @@ -10,7 +11,9 @@ import ( pb "github.com/Layr-Labs/eigenda/api/grpc/node" "github.com/Layr-Labs/eigenda/core" "github.com/Layr-Labs/eigenda/encoding" + "github.com/Layr-Labs/eigenda/tools/batchgen/flags" "github.com/consensys/gnark-crypto/ecc/bn254/fp" + "github.com/urfave/cli" "google.golang.org/grpc" "google.golang.org/grpc/credentials/insecure" @@ -18,51 +21,76 @@ import ( ) var ( + version = "" + gitCommit = "" + gitDate = "" + encodedChunk = []byte{42, 255, 129, 3, 1, 1, 5, 67, 104, 117, 110, 107, 1, 255, 130, 0, 1, 2, 1, 6, 67, 111, 101, 102, 102, 115, 1, 255, 134, 0, 1, 5, 80, 114, 111, 111, 102, 1, 255, 136, 0, 0, 0, 25, 255, 133, 2, 1, 1, 10, 91, 93, 98, 110, 50, 53, 52, 46, 70, 114, 1, 255, 134, 0, 1, 255, 132, 0, 0, 18, 255, 131, 1, 1, 1, 2, 70, 114, 1, 255, 132, 0, 1, 6, 1, 8, 0, 0, 35, 255, 135, 3, 1, 1, 7, 71, 49, 80, 111, 105, 110, 116, 1, 255, 136, 0, 1, 2, 1, 1, 88, 1, 255, 138, 0, 1, 1, 89, 1, 255, 138, 0, 0, 0, 23, 255, 137, 1, 1, 1, 7, 69, 108, 101, 109, 101, 110, 116, 1, 255, 138, 0, 1, 6, 1, 8, 0, 0, 254, 4, 243, 255, 130, 1, 32, 4, 248, 186, 196, 96, 34, 212, 35, 97, 83, 248, 121, 9, 252, 220, 181, 118, 97, 134, 248, 186, 26, 225, 204, 191, 144, 133, 234, 248, 7, 223, 191, 156, 83, 115, 21, 36, 4, 248, 43, 196, 225, 43, 61, 88, 43, 49, 248, 28, 200, 121, 122, 178, 119, 200, 17, 248, 29, 172, 61, 194, 130, 114, 50, 171, 248, 33, 141, 185, 47, 11, 129, 128, 116, 4, 248, 246, 236, 255, 207, 43, 92, 176, 63, 248, 103, 179, 139, 80, 75, 57, 128, 89, 248, 107, 170, 70, 254, 95, 17, 101, 158, 248, 8, 106, 82, 82, 25, 78, 95, 104, 4, 248, 28, 125, 21, 116, 243, 255, 206, 10, 248, 153, 249, 156, 88, 61, 254, 171, 171, 248, 103, 66, 131, 8, 12, 165, 173, 173, 248, 36, 227, 189, 242, 180, 18, 171, 208, 4, 248, 19, 159, 205, 146, 86, 81, 57, 28, 248, 161, 130, 249, 92, 236, 82, 103, 4, 248, 84, 44, 63, 43, 249, 88, 187, 12, 248, 42, 121, 83, 118, 55, 127, 180, 134, 4, 248, 193, 39, 155, 110, 195, 113, 118, 46, 248, 47, 92, 162, 69, 188, 120, 94, 161, 248, 101, 214, 253, 103, 243, 8, 246, 176, 248, 41, 1, 238, 37, 43, 132, 228, 244, 4, 248, 70, 34, 194, 33, 68, 87, 108, 180, 248, 203, 230, 97, 137, 162, 177, 142, 23, 248, 101, 25, 216, 255, 137, 96, 240, 73, 248, 40, 50, 167, 154, 63, 108, 55, 240, 4, 248, 78, 40, 51, 224, 193, 131, 8, 90, 248, 162, 203, 245, 119, 83, 125, 219, 33, 248, 85, 109, 106, 231, 162, 152, 229, 110, 248, 38, 189, 66, 40, 176, 177, 114, 84, 4, 248, 193, 67, 43, 158, 218, 245, 83, 116, 248, 100, 165, 217, 161, 166, 209, 98, 172, 248, 231, 23, 45, 28, 225, 102, 143, 157, 248, 20, 12, 146, 122, 104, 126, 51, 235, 4, 248, 19, 118, 59, 144, 83, 246, 144, 229, 248, 203, 168, 161, 194, 137, 34, 191, 157, 248, 252, 196, 212, 78, 99, 166, 6, 225, 248, 29, 41, 54, 112, 125, 128, 240, 209, 4, 248, 24, 175, 53, 2, 113, 155, 113, 233, 248, 162, 189, 238, 198, 233, 31, 199, 239, 248, 205, 162, 128, 190, 163, 250, 181, 226, 248, 40, 205, 5, 117, 16, 49, 205, 45, 4, 248, 78, 49, 135, 21, 90, 93, 196, 50, 248, 115, 105, 77, 122, 222, 27, 224, 166, 248, 44, 0, 255, 63, 67, 184, 234, 235, 248, 45, 88, 39, 211, 138, 80, 43, 243, 4, 248, 244, 239, 154, 119, 68, 204, 215, 5, 248, 53, 82, 219, 150, 72, 243, 20, 147, 248, 141, 131, 101, 73, 11, 218, 234, 89, 248, 25, 246, 203, 17, 86, 91, 107, 199, 4, 248, 111, 106, 155, 101, 22, 163, 231, 214, 248, 86, 123, 235, 222, 87, 192, 80, 167, 248, 107, 38, 156, 175, 73, 123, 184, 189, 248, 23, 12, 154, 39, 153, 2, 158, 213, 4, 248, 40, 166, 62, 99, 6, 145, 128, 237, 248, 77, 160, 235, 64, 123, 181, 120, 66, 248, 116, 0, 126, 221, 26, 18, 100, 74, 248, 46, 92, 161, 252, 177, 177, 191, 127, 4, 248, 227, 144, 223, 154, 232, 249, 22, 233, 248, 53, 82, 148, 149, 84, 76, 107, 93, 248, 71, 251, 7, 58, 156, 200, 102, 4, 248, 3, 147, 75, 172, 199, 222, 109, 87, 4, 248, 169, 207, 109, 252, 37, 85, 158, 78, 248, 237, 12, 207, 255, 117, 62, 171, 3, 248, 43, 93, 155, 238, 136, 102, 150, 139, 248, 40, 174, 6, 46, 62, 50, 174, 104, 4, 248, 156, 217, 228, 156, 76, 202, 37, 121, 248, 80, 44, 200, 177, 237, 112, 103, 44, 248, 211, 172, 202, 164, 34, 242, 190, 204, 248, 15, 241, 94, 33, 88, 13, 34, 66, 4, 248, 198, 229, 9, 111, 155, 117, 84, 125, 248, 69, 115, 47, 6, 35, 132, 39, 86, 248, 243, 113, 79, 216, 240, 35, 72, 75, 248, 7, 29, 38, 85, 134, 106, 213, 236, 4, 248, 8, 8, 251, 11, 97, 66, 8, 55, 248, 159, 67, 100, 214, 31, 167, 88, 221, 248, 151, 110, 49, 190, 136, 249, 55, 217, 248, 47, 94, 78, 30, 0, 220, 176, 125, 4, 248, 246, 81, 132, 144, 151, 161, 113, 102, 248, 229, 8, 10, 180, 28, 223, 222, 8, 248, 158, 88, 212, 24, 77, 31, 96, 232, 248, 41, 65, 45, 216, 25, 224, 221, 4, 4, 248, 11, 189, 86, 122, 64, 254, 107, 253, 248, 242, 174, 32, 144, 43, 116, 187, 77, 248, 16, 163, 127, 128, 4, 233, 82, 168, 248, 4, 90, 126, 233, 232, 220, 81, 74, 4, 248, 54, 17, 20, 36, 220, 10, 168, 78, 248, 77, 61, 41, 4, 95, 154, 130, 70, 248, 37, 180, 163, 188, 242, 88, 81, 28, 248, 37, 195, 179, 103, 195, 0, 252, 30, 4, 248, 148, 154, 198, 22, 110, 201, 164, 240, 248, 242, 100, 163, 103, 30, 185, 139, 205, 248, 198, 168, 87, 116, 135, 219, 11, 230, 248, 43, 163, 196, 37, 51, 32, 130, 241, 4, 248, 160, 22, 80, 69, 111, 126, 3, 23, 248, 76, 89, 182, 79, 244, 245, 155, 42, 248, 144, 203, 89, 203, 85, 216, 109, 139, 248, 36, 125, 246, 94, 210, 7, 236, 50, 4, 248, 244, 42, 154, 219, 137, 78, 64, 167, 248, 73, 57, 191, 50, 122, 120, 124, 249, 248, 192, 102, 139, 159, 135, 150, 18, 35, 248, 40, 167, 252, 247, 112, 215, 52, 61, 4, 248, 151, 181, 121, 81, 121, 147, 227, 13, 248, 236, 181, 178, 176, 243, 4, 136, 195, 248, 62, 97, 145, 239, 166, 114, 175, 107, 248, 23, 91, 75, 217, 198, 192, 155, 92, 4, 248, 182, 191, 150, 70, 229, 96, 122, 14, 248, 134, 0, 111, 72, 36, 162, 244, 220, 248, 168, 72, 14, 253, 239, 166, 139, 197, 248, 44, 139, 158, 151, 191, 127, 27, 222, 4, 248, 74, 171, 39, 27, 36, 31, 102, 30, 248, 41, 77, 140, 191, 229, 182, 30, 16, 248, 219, 194, 193, 143, 239, 141, 47, 73, 248, 23, 1, 236, 49, 51, 57, 155, 228, 4, 248, 128, 145, 254, 105, 104, 55, 224, 206, 248, 195, 70, 112, 120, 42, 171, 202, 23, 248, 242, 232, 247, 249, 215, 77, 208, 121, 248, 29, 0, 45, 26, 151, 224, 199, 214, 4, 248, 235, 253, 108, 246, 112, 139, 56, 187, 248, 214, 211, 157, 43, 210, 247, 57, 203, 248, 150, 28, 35, 231, 169, 220, 146, 139, 248, 48, 54, 207, 130, 116, 140, 125, 197, 4, 248, 23, 120, 154, 57, 66, 85, 149, 5, 248, 170, 172, 192, 127, 230, 130, 224, 17, 248, 117, 98, 19, 140, 134, 78, 47, 98, 248, 40, 206, 62, 254, 165, 238, 160, 130, 1, 1, 4, 248, 164, 40, 240, 180, 149, 114, 87, 82, 248, 195, 115, 109, 187, 95, 132, 65, 10, 248, 176, 59, 100, 197, 207, 37, 161, 253, 248, 10, 19, 137, 98, 39, 77, 128, 20, 1, 4, 248, 213, 212, 69, 58, 138, 39, 69, 249, 248, 99, 187, 162, 108, 114, 239, 78, 157, 248, 62, 166, 165, 148, 83, 202, 37, 169, 248, 47, 253, 18, 76, 216, 168, 22, 21, 0, 0} opID [32]byte ) func main() { - factor, err := strconv.Atoi(os.Args[1]) + app := cli.NewApp() + app.Version = fmt.Sprintf("%s,%s,%s", version, gitCommit, gitDate) + app.Name = "batchgen" + app.Description = "Synthetic batch generator & dispersal tool" + app.Usage = "" + app.Flags = flags.Flags + app.Action = batchGeneratorMain + if err := app.Run(os.Args); err != nil { + log.Fatalf("runtime failure: %v", err) + } +} + +func batchGeneratorMain() { + threads, err := strconv.Atoi(os.Args[2]) if err != nil { - log.Println("failed to convert factor to int:", err) - return + log.Fatalf("failed to convert threads to int:", err) } - req, _, _, _, _ := makeStoreChunksRequest(100, 10, factor) - hosts := os.Args[2:] // assuming multiple hosts are passed as command line arguments - results := make(chan *pb.StoreChunksReply, len(hosts)) // channel to collect results - errors := make(chan error, len(hosts)) // channel to collect errors + factor, err := strconv.Atoi(os.Args[1]) + if err != nil { + log.Fatalf("failed to convert batch scaling factor to int:", err) + } - for _, host := range hosts { - go func(host string) { - conn, err := grpc.Dial(host, grpc.WithTransportCredentials(insecure.NewCredentials())) - if err != nil { - errors <- err - return - } - defer conn.Close() + hosts := os.Args[3:] + results := make(chan *pb.StoreChunksReply, len(hosts)) + errors := make(chan error, len(hosts)) - client := pb.NewDispersalClient(conn) - opt := grpc.MaxCallSendMsgSize(60 * 1024 * 1024 * 1024) + req, _, _, _, _ := makeStoreChunksRequest(100, 10, factor) + reqSizeMB := float64(proto.Size(req)) / 1024.0 / 1024.0 + totalSizeMB := reqSizeMB * float64(threads) * float64(len(hosts)) + log.Println("request size", reqSizeMB, "total data sent", totalSizeMB, "MB") - log.Println("sending chunks to operator", host, "request message size", proto.Size(req)) - reply, err := client.StoreChunks(context.Background(), req, opt) - if err != nil { - errors <- err - } else { - results <- reply - } - }(host) + for _, host := range hosts { + for i := 0; i < threads; i++ { + go func(host string, i int) { + conn, err := grpc.Dial(host, grpc.WithTransportCredentials(insecure.NewCredentials())) + if err != nil { + errors <- err + return + } + defer conn.Close() + + client := pb.NewDispersalClient(conn) + opt := grpc.MaxCallSendMsgSize(60 * 1024 * 1024 * 1024) // matches batcher max msg size + + log.Println("host", host, "thread", i, "size", reqSizeMB, "MB") + reply, err := client.StoreChunks(context.Background(), req, opt) + if err != nil { + errors <- err + } else { + results <- reply + } + }(host, i) + } } // Wait for all goroutines to finish and collect results for i := 0; i < len(hosts); i++ { select { - case result := <-results: - log.Println("reply", result) case err := <-errors: - log.Println("error", err) + log.Println("ERR", err) } } } @@ -145,7 +173,7 @@ func makeStoreChunksRequest(quorumThreshold, adversaryThreshold uint8, factor in } batchHeader := core.BatchHeader{ BatchRoot: [32]byte{0}, - ReferenceBlockNumber: 1, + ReferenceBlockNumber: 0, // Block 0 triggers probe mode on the node } _, err = batchHeader.SetBatchRoot(blobHeaders) @@ -223,36 +251,3 @@ func blobHeaderToProto(blobHeader *core.BlobHeader) *pb.BlobHeader { QuorumHeaders: quorumHeaders, } } - -func generateRequest(size int64) (*pb.StoreChunksRequest, error) { - log.Println("Building request of size", size) - // Create a BatchHeader - batchHeader := &pb.BatchHeader{ - BatchRoot: []byte("root_hash_example"), - ReferenceBlockNumber: 12345, - } - - // Create a BlobHeader - blobHeader := &pb.BlobHeader{ - // Add fields here if necessary - } - - // Create a Bundle - bundle := &pb.Bundle{ - Chunks: [][]byte{[]byte("chunk1"), []byte("chunk2"), []byte("chunk3")}, - } - - // Create a Blob - blob := &pb.Blob{ - Header: blobHeader, - Bundles: []*pb.Bundle{bundle}, - } - - // Create the StoreChunksRequest - storeChunksRequest := &pb.StoreChunksRequest{ - BatchHeader: batchHeader, - Blobs: []*pb.Blob{blob}, - } - - return storeChunksRequest, nil -} diff --git a/tools/batchgen/flags/flags.go b/tools/batchgen/flags/flags.go new file mode 100644 index 0000000000..84c092b70f --- /dev/null +++ b/tools/batchgen/flags/flags.go @@ -0,0 +1,45 @@ +package flags + +import ( + "github.com/Layr-Labs/eigenda/common" + "github.com/urfave/cli" +) + +const ( + FlagPrefix = "batchgen" + envPrefix = "BATCHGEN" +) + +var ( + /* Required Flags */ + + NumThreadsFlag = cli.UintFlag{ + Name: "host-threads", + Usage: "Number of host threads in parallel", + Required: false, + Value: 1, + EnvVar: "HOST_THREADS", + } + ScalingFactorFlag = cli.UintFlag{ + Name: "scaling-factor", + Usage: "Scaling factor applied to default batch size", + Required: false, + Value: 10, + EnvVar: "SCALING_FACTOR", + } +) + +var requiredFlags = []cli.Flag{ + NumThreadsFlag, + ScalingFactorFlag, +} + +var optionalFlags = []cli.Flag{} + +// Flags contains the list of configuration options available to the binary. +var Flags []cli.Flag + +func init() { + Flags = append(requiredFlags, optionalFlags...) + Flags = append(Flags, common.LoggerCLIFlags(envPrefix, FlagPrefix)...) +} From ca4684c5b5b3faa2e88a562abebf1202b4cc8c3e Mon Sep 17 00:00:00 2001 From: Patrick Schork <354473+pschork@users.noreply.github.com> Date: Thu, 6 Jun 2024 22:27:04 -0700 Subject: [PATCH 4/4] Adds urfave/cli --- tools/batchgen/Makefile | 2 +- tools/batchgen/cmd/main.go | 24 ++++++++++-------------- tools/batchgen/flags/flags.go | 23 +++++++++++++++-------- 3 files changed, 26 insertions(+), 23 deletions(-) diff --git a/tools/batchgen/Makefile b/tools/batchgen/Makefile index e27e4763dc..7afd620ad4 100644 --- a/tools/batchgen/Makefile +++ b/tools/batchgen/Makefile @@ -6,5 +6,5 @@ clean: rm -rf ./bin run: build - ./bin/batchgen 1000 10 localhost:32005 + ./bin/batchgen --host localhost:32005 diff --git a/tools/batchgen/cmd/main.go b/tools/batchgen/cmd/main.go index c8f92b7ded..8e8f7ccade 100644 --- a/tools/batchgen/cmd/main.go +++ b/tools/batchgen/cmd/main.go @@ -5,7 +5,6 @@ import ( "fmt" "log" "os" - "strconv" commonpb "github.com/Layr-Labs/eigenda/api/grpc/common" pb "github.com/Layr-Labs/eigenda/api/grpc/node" @@ -38,32 +37,26 @@ func main() { app.Flags = flags.Flags app.Action = batchGeneratorMain if err := app.Run(os.Args); err != nil { - log.Fatalf("runtime failure: %v", err) + log.Fatal(err) } } -func batchGeneratorMain() { - threads, err := strconv.Atoi(os.Args[2]) - if err != nil { - log.Fatalf("failed to convert threads to int:", err) - } - - factor, err := strconv.Atoi(os.Args[1]) - if err != nil { - log.Fatalf("failed to convert batch scaling factor to int:", err) - } +func batchGeneratorMain(ctx *cli.Context) { + threads := int(ctx.GlobalUint(flags.NumThreadsFlag.Name)) + factor := int(ctx.GlobalUint(flags.ScalingFactorFlag.Name)) + hosts := ctx.GlobalStringSlice(flags.HostsFlag.Name) - hosts := os.Args[3:] results := make(chan *pb.StoreChunksReply, len(hosts)) errors := make(chan error, len(hosts)) req, _, _, _, _ := makeStoreChunksRequest(100, 10, factor) reqSizeMB := float64(proto.Size(req)) / 1024.0 / 1024.0 totalSizeMB := reqSizeMB * float64(threads) * float64(len(hosts)) - log.Println("request size", reqSizeMB, "total data sent", totalSizeMB, "MB") for _, host := range hosts { + log.Printf("host: %v", host) for i := 0; i < threads; i++ { + log.Printf("i: %v", i) go func(host string, i int) { conn, err := grpc.Dial(host, grpc.WithTransportCredentials(insecure.NewCredentials())) if err != nil { @@ -93,6 +86,9 @@ func batchGeneratorMain() { log.Println("ERR", err) } } + + log.Println("request size", reqSizeMB, "total data sent", totalSizeMB, "MB") + os.Exit(0) } func makeStoreChunksRequest(quorumThreshold, adversaryThreshold uint8, factor int) (*pb.StoreChunksRequest, [32]byte, [32]byte, []*core.BlobHeader, []*pb.BlobHeader) { diff --git a/tools/batchgen/flags/flags.go b/tools/batchgen/flags/flags.go index 84c092b70f..26411f8d0f 100644 --- a/tools/batchgen/flags/flags.go +++ b/tools/batchgen/flags/flags.go @@ -6,7 +6,7 @@ import ( ) const ( - FlagPrefix = "batchgen" + FlagPrefix = "" envPrefix = "BATCHGEN" ) @@ -14,28 +14,35 @@ var ( /* Required Flags */ NumThreadsFlag = cli.UintFlag{ - Name: "host-threads", + Name: common.PrefixFlag(FlagPrefix, "threads"), Usage: "Number of host threads in parallel", Required: false, Value: 1, - EnvVar: "HOST_THREADS", + EnvVar: common.PrefixEnvVar(envPrefix, "HOST_THREADS"), } ScalingFactorFlag = cli.UintFlag{ - Name: "scaling-factor", + Name: common.PrefixFlag(FlagPrefix, "scaling-factor"), Usage: "Scaling factor applied to default batch size", Required: false, - Value: 10, - EnvVar: "SCALING_FACTOR", + Value: 100, + EnvVar: common.PrefixEnvVar(envPrefix, "SCALING_FACTOR"), + } + HostsFlag = cli.StringSliceFlag{ + Name: common.PrefixFlag(FlagPrefix, "host"), + Usage: "host:port", + Required: true, } ) var requiredFlags = []cli.Flag{ + HostsFlag, +} + +var optionalFlags = []cli.Flag{ NumThreadsFlag, ScalingFactorFlag, } -var optionalFlags = []cli.Flag{} - // Flags contains the list of configuration options available to the binary. var Flags []cli.Flag