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

Wasm support #5

Merged
merged 6 commits into from
Oct 16, 2024
Merged
Show file tree
Hide file tree
Changes from all 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 Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -5,5 +5,6 @@ members = [
"sologger-log-context",
"sologger-log-transport",
"sologger-log-transformer",
"sologger-log-transformer-wasm",
"sologger"
]
8 changes: 7 additions & 1 deletion sologger-log-context/Cargo.toml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
[package]
name = "sologger_log_context"
version = "0.2.0"
version = "0.2.1"
edition = "2021"
authors = ["Will Kennedy"]
description = "Provides functionality to turn raw logs output by Solana RPCs into structured logs for specified program IDs"
Expand All @@ -12,6 +12,7 @@ categories = ["development-tools::debugging"]
exclude=["/tests", "/examples", "/benches", "/docs", "/target", "/.github", "/.gitignore", "/.gitattributes", "/.gitmodules", "/.travis.yml", "/.cargo-ok", "/.git", "/.idea"]

[lib]
crate-type = ["cdylib", "rlib"]
doctest = false

[dependencies]
Expand All @@ -22,9 +23,14 @@ serde_json = {version = "1.0.128"}
serde_derive = "1.0.210"
bs58 = "0.5.1"
lazy_static = "1.5.0"
wasm-bindgen = "0.2"
serde-wasm-bindgen = "0.6.5"
web-sys = { version = "0.3", features = ["console"] }
console_error_panic_hook = "0.1"

[dev-dependencies]
criterion = "0.5.1"
wasm-bindgen-test = "0.3.45"

[[bench]]
name = "log_context_benchmarks"
Expand Down
1 change: 1 addition & 0 deletions sologger-log-context/pkg/.gitignore
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
*
150 changes: 150 additions & 0 deletions sologger-log-context/pkg/README.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,150 @@
# sologger-log-context

**Overview**

This library provides functionality to turn raw logs output by Solana RPCs into structured logs for specified program IDs.

**Usage**

```rust
//Provide the ProgramSelector with the Program IDs for which you want to parse logs.
//If you want to parse logs for all programs, use ProgramsSelector::new(&["*".to_string()])
let programs_selector = ProgramsSelector::new(&["9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7".to_string()]);
//Provide the raw logs, transfer error, programs selector, slot, and signature to the LogContext::parse_logs function.
let log_contexts = LogContext::parse_logs(&logs, "".to_string(), &programs_selector, 1, "12345".to_string());
```

These json schema for the log context can be found here: [LogContext-schema](LogContext-schema.json)

For example, if we have a list of raw logs retrieved from the Solana RPC, we can parse them into structured logs using the LogContext::parse_logs function. The first parameter is the raw logs, the second parameter is the program ID, the third parameter is the programs selector, the fourth parameter is the slot, and the fifth parameter is the signature.

The LogContext::parse_logs function returns a vector of LogContexts. Each LogContext contains a vector of LogMessages. Each LogMessage contains a vector of LogFields. Each LogField contains a key and a value.

Here is an example of raw logs retrieved from the Solana RPC:

```text
Program 9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7 invoke [1]
Program log: Instruction: Initialize
Program 11111111111111111111111111111111 invoke [2]
Program 11111111111111111111111111111111 success
Program log: Initialized new event. Current value
Program 9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7 consumed 59783 of 200000 compute units
Program 9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7 success
Program AbcdefGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL invoke [1]
Program log: Create
Program AbcdefGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL consumed 5475 of 200000 compute units
Program failed to complete: Invoked an instruction with data that is too large (12178014311288245306 > 10240)
Program AbcdefGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL failed: Program failed to complete
```

Here is an example of the structured logs that are returned from the LogContext::parse_logs function:

```json
{
"log_messages":[
"Instruction: Initialize",
"Initialized new event. Current value"
],
"data_logs":[

],
"raw_logs":[
"Program 9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7 invoke [1]",
"Program log: Instruction: Initialize",
"Program log: Initialized new event. Current value",
"Program 9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7 consumed 59783 of 200000 compute units",
"Program 9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7 success"
],
"errors":[

],
"transaction_error":"",
"program_id":"9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7",
"parent_program_id":"",
"depth":1,
"id":0,
"instruction_index":0,
"invoke_result":"",
"slot":1,
"signature":"12345"
}
```

```json
{
"log_messages":[

],
"data_logs":[

],
"raw_logs":[
"Program 11111111111111111111111111111111 invoke [2]",
"Program 11111111111111111111111111111111 success"
],
"errors":[

],
"transaction_error":"",
"program_id":"11111111111111111111111111111111",
"parent_program_id":"9RX7oz3WN5VRTqekBBHBvEJFVMNRnrCmVy7S6B6S5oU7",
"depth":2,
"id":1,
"instruction_index":0,
"invoke_result":"",
"slot":1,
"signature":"12345"
}
```

```json
{
"log_messages":[
"Create"
],
"data_logs":[

],
"raw_logs":[
"Program AbcdefGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL invoke [1]",
"Program log: Create",
"Program AbcdefGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL consumed 5475 of 200000 compute units",
"Program failed to complete: Invoked an instruction with data that is too large (12178014311288245306 > 10240)",
"Program AbcdefGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL failed: Program failed to complete"
],
"errors":[
"Invoked an instruction with data that is too large (12178014311288245306 > 10240)",
"Program failed to complete"
],
"transaction_error":"",
"program_id":"AbcdefGPvbdGVxr1b2hvZbsiqW5xWH25efTNsLJA8knL",
"parent_program_id":"",
"depth":1,
"id":2,
"instruction_index":1,
"invoke_result":"",
"slot":1,
"signature":"12345"
}
```

**Additional Usage**

The LogContext also provides utility to retrieve specific information from a log line.

- get_program_data: Returns the data mentioned in the provided log (for logs prefixed with "Program data: ")
- parse_logs_from_string: Parses the provided payload and returns a vector of LogContexts. The payload in this case is the raw JSON response as a string from the Solana RPC log_subscription endpoint.
- has_errors: Returns true if the log contains a program error

**Technical Details**

The parsing of the raw logs is done using a regular expression. The regular expression is defined in the LogContext::get_log_regex function. The regular expression is defined as follows:

[regex-vis](https://regex-vis.com/?r=%28%3F%3CprogramInvoke%3E%5EProgram+%28%3F%3CinvokeProgramId%3E%5B1-9A-HJ-NP-Za-km-z%5D%7B32%2C%7D%29+invoke+%5C%5B%28%3F%3Clevel%3E%5Cd%2B%29%5C%5D%24%29%7C%28%3F%3CprogramSuccessResult%3E%5EProgram+%28%3F%3CsuccessResultProgramId%3E%5B1-9A-HJ-NP-Za-km-z%5D%7B32%2C%7D%29+success%24%29%7C%28%3F%3CprogramFailedResult%3E%5EProgram+%28%3F%3CfailedResultProgramId%3E%5B1-9A-HJ-NP-Za-km-z%5D%7B32%2C%7D%29+failed%3A+%28%3F%3CfailedResultErr%3E.*%29%24%29%7C%28%3F%3CprogramCompleteFailedResult%3E%5EProgram+failed+to+complete%3A+%28%3F%3CfailedCompleteError%3E.*%29%24%29%7C%28%3F%3CprogramLog%3E%5E%5EProgram+log%3A+%28%3F%3ClogMessage%3E.*%29%24%29%7C%28%3F%3CprogramData%3E%5EProgram+data%3A+%28%3F%3Cdata%3E.*%29%24%29%7C%28%3F%3CprogramConsumed%3E%5EProgram+%28%3F%3CconsumedProgramId%3E%5B1-9A-HJ-NP-Za-km-z%5D%7B32%2C%7D%29+consumed+%28%3F%3CconsumedComputeUnits%3E%5Cd*%29+of+%28%3F%3CallComputedUnits%3E%5Cd*%29+compute+units%24%29%7C%28%3F%3CprogramConsumption%3E%5E%5EProgram+consumption%3A+%28%3F%3CcomputeUnitsRemaining%3E.*%29%24%29%7C%28%3F%3ClogTruncated%3E%5ELog+truncated%24%29%7C%28%3F%3CprogramReturn%3E%5EProgram+return%3A+%28%3F%3CreturnProgramId%3E%5B1-9A-HJ-NP-Za-km-z%5D%7B32%2C%7D%29+%28%3F%3CreturnMessage%3E.*%29%24%29&e=0)

```regexp
(?<programInvoke>^Program (?<invokeProgramId>[1-9A-HJ-NP-Za-km-z]{32,}) invoke \[(?<level>\d+)\]$)|(?<programSuccessResult>^Program (?<successResultProgramId>[1-9A-HJ-NP-Za-km-z]{32,}) success$)|(?<programFailedResult>^Program (?<failedResultProgramId>[1-9A-HJ-NP-Za-km-z]{32,}) failed: (?<failedResultErr>.*)$)|(?<programCompleteFailedResult>^Program failed to complete: (?<failedCompleteError>.*)$)|(?<programLog>^^Program log: (?<logMessage>.*)$)|(?<programData>^Program data: (?<data>.*)$)|(?<programConsumed>^Program (?<consumedProgramId>[1-9A-HJ-NP-Za-km-z]{32,}) consumed (?<consumedComputeUnits>\d*) of (?<allComputedUnits>\d*) compute units$)|(?<programConsumption>^^Program consumption: (?<computeUnitsRemaining>.*)$)|(?<logTruncated>^Log truncated$)|(?<programReturn>^Program return: (?<returnProgramId>[1-9A-HJ-NP-Za-km-z]{32,}) (?<returnMessage>.*)$)
```

The LogContext attempts to loop through the raw logs returned from the Solana websocket log subscription frames, or groups of logs retrieved from a specific transaction or block.
If logs are provided that are out of order or not from a contained unit such as a block, transaction or websocket frame, then the LogContext will most likely fail.
32 changes: 32 additions & 0 deletions sologger-log-context/pkg/package.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
{
"name": "sologger_log_context",
"type": "module",
"collaborators": [
"Will Kennedy"
],
"description": "Provides functionality to turn raw logs output by Solana RPCs into structured logs for specified program IDs",
"version": "0.2.0",
"license": "Apache-2.0",
"repository": {
"type": "git",
"url": "https://github.com/brytelands/sologger"
},
"files": [
"sologger_log_context_bg.wasm",
"sologger_log_context.js",
"sologger_log_context_bg.js",
"sologger_log_context.d.ts"
],
"main": "sologger_log_context.js",
"types": "sologger_log_context.d.ts",
"sideEffects": [
"./sologger_log_context.js",
"./snippets/*"
],
"keywords": [
"solana",
"logging",
"json",
"parsing"
]
}
23 changes: 23 additions & 0 deletions sologger-log-context/pkg/sologger_log_context.d.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
/* tslint:disable */
/* eslint-disable */
/**
* @param {(string)[]} logs
* @param {(string)[]} program_ids
* @returns {any}
*/
export function parse_logs_basic(logs: (string)[], program_ids: (string)[]): any;
export class WasmLogParser {
free(): void;
/**
* @param {(string)[]} program_ids
*/
constructor(program_ids: (string)[]);
/**
* @param {(string)[]} logs
* @param {string} transaction_error
* @param {bigint} slot
* @param {string} signature
* @returns {any}
*/
parse_logs(logs: (string)[], transaction_error: string, slot: bigint, signature: string): any;
}
4 changes: 4 additions & 0 deletions sologger-log-context/pkg/sologger_log_context.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,4 @@
import * as wasm from "./sologger_log_context_bg.wasm";
export * from "./sologger_log_context_bg.js";
import { __wbg_set_wasm } from "./sologger_log_context_bg.js";
__wbg_set_wasm(wasm);
Loading
Loading