Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: PRT - Result generic parser #1711

Merged
merged 23 commits into from
Oct 9, 2024
Merged
Show file tree
Hide file tree
Changes from 6 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions proto/lavanet/lava/spec/api_collection.proto
Original file line number Diff line number Diff line change
Expand Up @@ -83,6 +83,7 @@ message ParseDirective {
string function_template = 2;
BlockParser result_parsing = 3 [(gogoproto.nullable) = false];
string api_name = 4;
repeated GenericParser parsers = 5 [(gogoproto.nullable) = false];
}

message BlockParser {
Expand Down
112 changes: 58 additions & 54 deletions protocol/chainlib/chain_fetcher.go
Original file line number Diff line number Diff line change
Expand Up @@ -188,68 +188,71 @@ func (cf *ChainFetcher) Verify(ctx context.Context, verification VerificationCon
)
}

parsedResult, err := parser.ParseFromReply(parserInput, parsing.ResultParsing)
if err != nil {
return utils.LavaFormatWarning("[-] verify failed to parse result", err, []utils.Attribute{
{Key: "chainId", Value: chainId},
{Key: "nodeUrl", Value: proxyUrl.Url},
{Key: "Method", Value: parsing.GetApiName()},
{Key: "Response", Value: string(reply.RelayReply.Data)},
}...)
parsedInput := parser.ParseBlockFromReply(parserInput, parsing.ResultParsing, parsing.Parsers)
if parsedInput.GetBlockRaw() == "" {
return utils.LavaFormatWarning("[-] verify failed to parse result", err,
utils.LogAttr("chainId", chainId),
utils.LogAttr("nodeUrl", proxyUrl.Url),
utils.LogAttr("Method", parsing.GetApiName()),
utils.LogAttr("Response", string(reply.RelayReply.Data)),
)
}
if verification.LatestDistance != 0 && latestBlock != 0 && verification.ParseDirective.FunctionTag != spectypes.FUNCTION_TAG_GET_BLOCK_BY_NUM {
parsedResultAsNumber, err := strconv.ParseUint(parsedResult, 0, 64)
if err != nil {
return utils.LavaFormatWarning("[-] verify failed to parse result as number", err, []utils.Attribute{
{Key: "chainId", Value: chainId},
{Key: "nodeUrl", Value: proxyUrl.Url},
{Key: "Method", Value: parsing.GetApiName()},
{Key: "Response", Value: string(reply.RelayReply.Data)},
{Key: "parsedResult", Value: parsedResult},
}...)
parsedResultAsNumber := parsedInput.GetBlock()
if parsedResultAsNumber == spectypes.NOT_APPLICABLE {
return utils.LavaFormatWarning("[-] verify failed to parse result as number", err,
utils.LogAttr("chainId", chainId),
utils.LogAttr("nodeUrl", proxyUrl.Url),
utils.LogAttr("Method", parsing.GetApiName()),
utils.LogAttr("Response", string(reply.RelayReply.Data)),
utils.LogAttr("rawParsedBlock", parsedInput.GetBlockRaw()),
)
}
if parsedResultAsNumber > latestBlock {
return utils.LavaFormatWarning("[-] verify failed parsed result is greater than latestBlock", err, []utils.Attribute{
{Key: "chainId", Value: chainId},
{Key: "nodeUrl", Value: proxyUrl.Url},
{Key: "Method", Value: parsing.GetApiName()},
{Key: "latestBlock", Value: latestBlock},
{Key: "parsedResult", Value: parsedResultAsNumber},
}...)
uint64ParsedResultAsNumber := uint64(parsedResultAsNumber)
if uint64ParsedResultAsNumber > latestBlock {
return utils.LavaFormatWarning("[-] verify failed parsed result is greater than latestBlock", err,
utils.LogAttr("chainId", chainId),
utils.LogAttr("nodeUrl", proxyUrl.Url),
utils.LogAttr("Method", parsing.GetApiName()),
utils.LogAttr("latestBlock", latestBlock),
utils.LogAttr("parsedResult", uint64ParsedResultAsNumber),
)
}
if latestBlock-parsedResultAsNumber < verification.LatestDistance {
return utils.LavaFormatWarning("[-] verify failed expected block distance is not sufficient", err, []utils.Attribute{
{Key: "chainId", Value: chainId},
{Key: "nodeUrl", Value: proxyUrl.Url},
{Key: "Method", Value: parsing.GetApiName()},
{Key: "latestBlock", Value: latestBlock},
{Key: "parsedResult", Value: parsedResultAsNumber},
{Key: "expected", Value: verification.LatestDistance},
}...)
if latestBlock-uint64ParsedResultAsNumber < verification.LatestDistance {
return utils.LavaFormatWarning("[-] verify failed expected block distance is not sufficient", err,
utils.LogAttr("chainId", chainId),
utils.LogAttr("nodeUrl", proxyUrl.Url),
utils.LogAttr("Method", parsing.GetApiName()),
utils.LogAttr("latestBlock", latestBlock),
utils.LogAttr("parsedResult", uint64ParsedResultAsNumber),
utils.LogAttr("expected", verification.LatestDistance),
)
}
}
// some verifications only want the response to be valid, and don't care about the value
if verification.Value != "*" && verification.Value != "" && verification.ParseDirective.FunctionTag != spectypes.FUNCTION_TAG_GET_BLOCK_BY_NUM {
if parsedResult != verification.Value {
return utils.LavaFormatWarning("[-] verify failed expected and received are different", err, []utils.Attribute{
{Key: "chainId", Value: chainId},
{Key: "nodeUrl", Value: proxyUrl.Url},
{Key: "parsedResult", Value: parsedResult},
{Key: "verification.Value", Value: verification.Value},
{Key: "Method", Value: parsing.GetApiName()},
{Key: "Extension", Value: verification.Extension},
{Key: "Addon", Value: verification.Addon},
{Key: "Verification", Value: verification.Name},
}...)
rawBlock := parsedInput.GetBlockRaw()
if rawBlock != verification.Value {
return utils.LavaFormatWarning("[-] verify failed expected and received are different", err,
utils.LogAttr("chainId", chainId),
utils.LogAttr("nodeUrl", proxyUrl.Url),
utils.LogAttr("rawParsedBlock", rawBlock),
utils.LogAttr("verification.Value", verification.Value),
utils.LogAttr("Method", parsing.GetApiName()),
utils.LogAttr("Extension", verification.Extension),
utils.LogAttr("Addon", verification.Addon),
utils.LogAttr("Verification", verification.Name),
)
}
}
utils.LavaFormatInfo("[+] verified successfully",
utils.Attribute{Key: "chainId", Value: chainId},
utils.Attribute{Key: "nodeUrl", Value: proxyUrl.Url},
utils.Attribute{Key: "verification", Value: verification.Name},
utils.Attribute{Key: "value", Value: parser.CapStringLen(parsedResult)},
utils.Attribute{Key: "verificationKey", Value: verification.VerificationKey},
utils.Attribute{Key: "apiInterface", Value: cf.endpoint.ApiInterface},
utils.LogAttr("chainId", chainId),
utils.LogAttr("nodeUrl", proxyUrl.Url),
utils.LogAttr("verification", verification.Name),
utils.LogAttr("block", parsedInput.GetBlock()),
utils.LogAttr("rawBlock", parsedInput.GetBlockRaw()),
utils.LogAttr("verificationKey", verification.VerificationKey),
utils.LogAttr("apiInterface", cf.endpoint.ApiInterface),
)
return nil
}
Expand Down Expand Up @@ -292,8 +295,9 @@ func (cf *ChainFetcher) FetchLatestBlockNum(ctx context.Context) (int64, error)
{Key: "error", Value: err},
}...)
}
blockNum, err := parser.ParseBlockFromReply(parserInput, parsing.ResultParsing)
if err != nil {
parsedInput := parser.ParseBlockFromReply(parserInput, parsing.ResultParsing, parsing.Parsers)
blockNum := parsedInput.GetBlock()
if blockNum == spectypes.NOT_APPLICABLE {
return spectypes.NOT_APPLICABLE, utils.LavaFormatDebug(tagName+" Failed to parse Response", []utils.Attribute{
{Key: "chainId", Value: chainId},
{Key: "nodeUrl", Value: proxyUrl.Url},
Expand Down Expand Up @@ -355,7 +359,7 @@ func (cf *ChainFetcher) FetchBlockHashByNum(ctx context.Context, blockNum int64)
}...)
}

res, err := parser.ParseFromReplyAndDecode(parserInput, parsing.ResultParsing)
res, err := parser.ParseBlockHashFromReplyAndDecode(parserInput, parsing.ResultParsing, parsing.Parsers)
if err != nil {
return "", utils.LavaFormatDebug(tagName+" Failed ParseMessageResponse", []utils.Attribute{
{Key: "error", Value: err},
Expand Down
10 changes: 4 additions & 6 deletions protocol/chainlib/grpc_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -217,9 +217,8 @@ func TestParsingRequestedBlocksHeadersGrpc(t *testing.T) {
require.NoError(t, err)
parserInput, err := FormatResponseForParsing(reply.RelayReply, chainMessage)
require.NoError(t, err)
blockNum, err := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing)
require.NoError(t, err)
require.Equal(t, test.block, blockNum)
parsedInput := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing, nil)
require.Equal(t, test.block, parsedInput.GetBlock())
})
}
}
Expand Down Expand Up @@ -287,9 +286,8 @@ func TestSettingBlocksHeadersGrpc(t *testing.T) {
require.NoError(t, err)
parserInput, err := FormatResponseForParsing(reply.RelayReply, chainMessage)
require.NoError(t, err)
blockNum, err := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing)
require.NoError(t, err)
require.Equal(t, test.block, blockNum)
parsedInput := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing, nil)
require.Equal(t, test.block, parsedInput.GetBlock())
})
}
}
5 changes: 3 additions & 2 deletions protocol/chainlib/jsonRPC_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -187,8 +187,9 @@ func TestJsonRpcChainProxy(t *testing.T) {
require.NoError(t, err)

_, err = chainFetcher.FetchBlockHashByNum(ctx, block)
errMsg := "GET_BLOCK_BY_NUM Failed ParseMessageResponse {error:invalid parser input format"
require.True(t, err.Error()[:len(errMsg)] == errMsg, err.Error())
actualErrMsg := "GET_BLOCK_BY_NUM Failed ParseMessageResponse {error:blockParsing - parse failed {error:invalid parser input format,"
expectedErrMsg := err.Error()[:len(actualErrMsg)]
require.Equal(t, actualErrMsg, expectedErrMsg, err.Error())
}

func TestAddonAndVerifications(t *testing.T) {
Expand Down
10 changes: 4 additions & 6 deletions protocol/chainlib/rest_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -214,9 +214,8 @@ func TestParsingRequestedBlocksHeadersRest(t *testing.T) {
require.NoError(t, err)
parserInput, err := FormatResponseForParsing(reply.RelayReply, chainMessage)
require.NoError(t, err)
blockNum, err := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing)
require.NoError(t, err)
require.Equal(t, test.block, blockNum)
parsedInput := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing, nil)
require.Equal(t, test.block, parsedInput.GetBlock())
})
}
}
Expand Down Expand Up @@ -289,9 +288,8 @@ func TestSettingRequestedBlocksHeadersRest(t *testing.T) {
require.NoError(t, err)
parserInput, err := FormatResponseForParsing(reply.RelayReply, chainMessage)
require.NoError(t, err)
blockNum, err := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing)
require.NoError(t, err)
require.Equal(t, test.block, blockNum)
parsedInput := parser.ParseBlockFromReply(parserInput, parsingForCrafting.ResultParsing, nil)
require.Equal(t, test.block, parsedInput.GetBlock())
})
}
}
Expand Down
Loading
Loading