Get count of each ledger entry type from history archive
This example connects to the Stellar History Archive and initializes a checkpoint change reader with ingest.NewCheckpointChangeReader
to read ledger entry changes from a history archive snapshot occurring at a given checkpoint ledger. It iterates over the ledger changes, processing different entry types such as Account
, ClaimableBalance
, Trustline
, and Offer
, using the ingest.Change
structure. The code tracks statistics in an entriesInfo
struct, counting the total entries and specific types.
- Go
// Filename: ledger_entries.go
package main
import (
"context"
"encoding/json"
"fmt"
"github.com/stellar/go/historyarchive"
"github.com/stellar/go/ingest"
"github.com/stellar/go/network"
"github.com/stellar/go/support/storage"
"github.com/stellar/go/xdr"
"io"
)
func panicIf(err error) {
if err != nil {
panic(fmt.Errorf("An error occurred, panicking: %s\n", err))
}
}
// It helps to have a type to distinguish between the type of key and value, when defining a map
type AccountIdStr string
type entriesInfo struct {
TotalEntries int32
NumAccounts int32
NumAccountsWithTrustlines int32
NumClaimableBalances int32
NumOffers int32
}
func main() {
archiveURLs := network.PublicNetworkhistoryArchiveURLs
networkPassphrase := network.PublicNetworkPassphrase
// Open a history archive using our existing configuration details.
historyArchive, err := historyarchive.NewArchivePool(
archiveURLs,
historyarchive.ArchiveOptions{
NetworkPassphrase: networkPassphrase,
ConnectOptions: storage.ConnectOptions{
S3Region: "us-west-1",
UnsignedRequests: false,
},
},
)
panicIf(err)
// We pass 48921599 because given a checkpoint frequency of 64 ledgers (the
// default in `ConnectOptions`, above), 48921599+1 mod 64 == 0. Incompatible
// sequence numbers will likely result in 404 errors.
// Ledger seq 48921599 corresponds to about 8 years
var reader *ingest.CheckpointChangeReader
// NOTE: Connecting to pubnet and reading the checkpoint might take a while.
reader, err = ingest.NewCheckpointChangeReader(context.Background(), historyArchive, 48921599)
panicIf(err)
defer reader.Close()
trustlinesPerAccount := make(map[AccountIdStr]int32)
info := entriesInfo{}
for {
var entry ingest.Change
entry, err = reader.Read()
if err == io.EOF {
break
}
panicIf(err)
info.TotalEntries++
switch entry.Type {
case xdr.LedgerEntryTypeClaimableBalance:
info.NumClaimableBalances++
case xdr.LedgerEntryTypeAccount:
info.NumAccounts++
case xdr.LedgerEntryTypeTrustline:
accountId := entry.Post.Data.MustTrustLine().AccountId
trustlinesPerAccount[AccountIdStr(accountId.Address())]++
case xdr.LedgerEntryTypeOffer:
info.NumOffers++
default:
// there are other types as well, but we are not including them in this example
}
// Print progress after every 1000 entries are read
if info.TotalEntries%1000 == 0 {
fmt.Printf("Processed %d ledger entry changes...\r", info.TotalEntries)
}
}
info.NumAccountsWithTrustlines = int32(len(trustlinesPerAccount))
fmt.Println()
// Marshal the struct into JSON with indentation
jsonData, err := json.MarshalIndent(&info, "", " ")
panicIf(err)
// Print the indented JSON
fmt.Println(string(jsonData))
}
Sample Response:
>> go run ./ledger_entries.go
Processed 40123000 ledger entry changes...
{
"TotalEntries": 40123409,
"NumAccounts": 7699007,
"NumAccountsWithTrustlines": 3414587,
"NumClaimableBalances": 7559287,
"NumOffers": 1053870
}
Did you find this page helpful?