mirror of
https://github.com/Rucknium/xmrpeers.git
synced 2025-01-22 02:24:34 +00:00
Add get.p2p.log()
This commit is contained in:
parent
1d4c64b7d8
commit
db758fbca9
5 changed files with 250 additions and 18 deletions
|
@ -15,6 +15,8 @@ License: GPL (>= 2)
|
|||
Encoding: UTF-8
|
||||
LazyData: true
|
||||
Imports:
|
||||
R.utils,
|
||||
data.table,
|
||||
readr,
|
||||
stringr,
|
||||
pingr,
|
||||
|
@ -22,7 +24,8 @@ Imports:
|
|||
IP,
|
||||
parallelly,
|
||||
future,
|
||||
future.apply
|
||||
future.apply,
|
||||
qs
|
||||
Suggests:
|
||||
testthat (>= 3.0.0),
|
||||
withr
|
||||
|
|
|
@ -1,3 +1,5 @@
|
|||
exportPattern("^[[:alpha:]]+")
|
||||
# exportPattern("^[[:alpha:]]+")
|
||||
export("ping.peers")
|
||||
export("get.p2p.log")
|
||||
importFrom("utils", "read.csv")
|
||||
importFrom("stats", "complete.cases")
|
||||
|
|
198
R/parse.R
Normal file
198
R/parse.R
Normal file
|
@ -0,0 +1,198 @@
|
|||
|
||||
|
||||
|
||||
|
||||
|
||||
#' Parse p2p log files
|
||||
#'
|
||||
#' @param bitmonero.dir Directory location of the log files
|
||||
#' @param output.file Optional file to output compressed data
|
||||
#'
|
||||
#' @return
|
||||
#' data.frame of each transaction gossip message for each peer.
|
||||
#' @export
|
||||
#'
|
||||
#' @examples
|
||||
#' \dontrun{
|
||||
#' get.p2p.log()
|
||||
#' }
|
||||
get.p2p.log <- function(bitmonero.dir = "~/.bitmonero", output.file = NULL) {
|
||||
|
||||
# Sys.setenv(VROOM_CONNECTION_SIZE = formatC(131072 * 10000, format = "fg"))
|
||||
# Fixes occasional issues with https://github.com/tidyverse/vroom/issues/364
|
||||
# formatC() is used since VROOM_CONNECTION_SIZE cannot interpret scientific notation
|
||||
|
||||
bitmonero.dir <- path.expand(bitmonero.dir)
|
||||
bitmonero.dir <- gsub("/+$", "", bitmonero.dir)
|
||||
# Remove trailing slash(es) if they exist
|
||||
|
||||
files.in.dir <- list.files(bitmonero.dir)
|
||||
bitmonero.files <- files.in.dir[grepl("(^bitmonero[.]log)|(^monero[.]log)", files.in.dir, ignore.case = TRUE)]
|
||||
# This one does not have "$" at the end of each string like ping.peers()
|
||||
# since we want all log files
|
||||
|
||||
get.time <- function(x) {
|
||||
x <- stringr::str_extract(x, "^[0-9]{4}-[0-9]{2}-[0-9]{2}\\s[0-9]{2}:[0-9]{2}:[0-9]{2}\\S*")
|
||||
as.POSIXct(strptime(x, format = "%Y-%m-%d %H:%M:%OS"))
|
||||
}
|
||||
|
||||
cat(base::date(), " Reading ", length(bitmonero.files), " log files...\n", sep = "")
|
||||
|
||||
monero.log <- lapply(bitmonero.files, function(x) {
|
||||
x <- data.table::fread(paste0(bitmonero.dir, "/", x),
|
||||
header = FALSE, sep = NULL, blank.lines.skip = FALSE)[[1]]
|
||||
# Do [[1]] since data.table::fread() produces a data.frame. We want a vector.
|
||||
times <- get.time(x)
|
||||
x[!is.na(times)]
|
||||
})
|
||||
# Doing get.time() and excluding invalid lines prevents problems when
|
||||
# re-reading the data to monero.log.timing
|
||||
monero.log.timing <- lapply(seq_along(monero.log), function(x) {
|
||||
suppressWarnings(readr::read_tsv(I(monero.log[[x]]), col_names = c("time", "thread"),
|
||||
col_types = c("T", "c"), col_select = 1:2, skip_empty_rows = FALSE))
|
||||
# skip_empty_rows = FALSE is very important for this will not align with monero.log
|
||||
# Do not re-read the file from storage. Must wrap in I()
|
||||
# suppresswarnings() because we get
|
||||
# # One or more parsing issues, call `problems()` on your data frame for details, e.g.:
|
||||
# # dat <- vroom(...)
|
||||
# # problems(dat)
|
||||
# but we cannot view the problem beacuse: https://github.com/tidyverse/readr/issues/1477
|
||||
# It is probably because a few lines have more than the normal number of tab characters.
|
||||
# It likely has no effect since we only get the first two columns.
|
||||
})
|
||||
|
||||
beginning.log.times <- vector("numeric", length(monero.log))
|
||||
|
||||
for (i in seq_along(beginning.log.times)) {
|
||||
if (grepl("^[0-9]{4}-", monero.log[[i]][1])) {
|
||||
first.date.line <- 1
|
||||
} else {
|
||||
first.date.line <- grep("^[0-9]{4}-", monero.log[[i]])[1]
|
||||
# Get first appearance of a date in the log file
|
||||
}
|
||||
beginning.log.times[i] <- get.time(monero.log[[i]][first.date.line])
|
||||
}
|
||||
|
||||
monero.log <- unlist(monero.log[order(beginning.log.times)])
|
||||
# Put monero logs in correct order and concatenate them
|
||||
|
||||
monero.log.timing <- as.data.frame(do.call(rbind, monero.log.timing[order(beginning.log.times)]))
|
||||
|
||||
n.out.of.order <- sum(monero.log.timing$time[-nrow(monero.log.timing)] <= monero.log.timing$time[-1])
|
||||
|
||||
if (! n.out.of.order > 0) {
|
||||
warning(paste0(n.out.of.order, " log lines were not originally in the correct time order. Probably this\n is fixed automatically."))
|
||||
}
|
||||
|
||||
keep.rows <- (! duplicated(monero.log)) & complete.cases(monero.log.timing)
|
||||
monero.log <- monero.log[keep.rows]
|
||||
monero.log.timing <- monero.log.timing[keep.rows, ]
|
||||
monero.log <- monero.log[order(monero.log.timing$thread, monero.log.timing$time)]
|
||||
rm(monero.log.timing, keep.rows)
|
||||
# Log messages from different peers can sometimes print at roughly the same time.
|
||||
# We need to link the tx_hash of each log line to its NOTIFY_NEW_TRANSACTIONS
|
||||
# line that has the IP address of the peer. So we will order the log lines
|
||||
# first by thread number and then by time so that each set of tx_hashes
|
||||
# is kept together.
|
||||
# moneromooo: "Use the thread column to disentangle."
|
||||
# https://libera.monerologs.net/monero-dev/20240317#c348611
|
||||
|
||||
cat(base::date(), " Finished reading ", length(bitmonero.files), " log files.\n", sep = "")
|
||||
|
||||
monero.log <- monero.log[grepl("net.p2p.msg", monero.log, fixed = TRUE)]
|
||||
|
||||
# TODO: Must do the fluffy blocks before this
|
||||
|
||||
monero.log <- monero.log[
|
||||
grepl("Received NOTIFY_NEW_TRANSACTIONS", monero.log, fixed = TRUE) |
|
||||
grepl("Including transaction", monero.log, fixed = TRUE)
|
||||
]
|
||||
|
||||
notify.lines <- grep("Received NOTIFY_NEW_TRANSACTIONS", monero.log, fixed = TRUE)
|
||||
|
||||
get.number.of.txs <- function(x) {
|
||||
x <- stringr::str_extract(x, "([0-9]+) txes[)]", group = 1)
|
||||
as.numeric(x)
|
||||
}
|
||||
|
||||
number.of.txs <- get.number.of.txs(monero.log[notify.lines])
|
||||
|
||||
get.tx.hash <- function(x) {
|
||||
stringr::str_extract(x, "Including transaction <([:xdigit:]{64})>", group = 1)
|
||||
}
|
||||
|
||||
tx.hashes <- get.tx.hash(monero.log)
|
||||
|
||||
rle.result <- rle(!is.na(tx.hashes))
|
||||
# The "Including transaction <tx_hash>" will not be printed if the tx fails
|
||||
# the cryptonote::parse_and_validate_tx_from_blob() check or if the peer sends
|
||||
# duplicate txs in the message:
|
||||
# https://github.com/monero-project/monero/blob/c8214782fb2a769c57382a999eaf099691c836e7/src/cryptonote_protocol/cryptonote_protocol_handler.inl#L990
|
||||
# https://github.com/monero-project/monero/blob/c8214782fb2a769c57382a999eaf099691c836e7/src/cryptonote_basic/cryptonote_format_utils.cpp#L224
|
||||
# Therefore, we compute the "run length" of whether there are tx_hashes in the
|
||||
# log line. (If it does not have the tx_hash, then it must be the
|
||||
# "NOTIFY_NEW_TRANSACTIONS" message because all other lines were exlcuded above.)
|
||||
# The run length is used to create the final output data.
|
||||
|
||||
for (i in which( (! rle.result$values) & rle.result$lengths > 1)) {
|
||||
# This fixes the rare case that the number of txs received in a message
|
||||
# is zero, or if all the txs in a message did not pass validation
|
||||
# so none of them were printed to the log.
|
||||
how.many <- rle.result$lengths[i] - 1
|
||||
rle.result$lengths <- append(rle.result$lengths, rep(0, how.many), after = i)
|
||||
rle.result$values <- append(rle.result$values, rep(TRUE, how.many), after = i)
|
||||
}
|
||||
|
||||
cat(base::date(), " ", sum(rle.result$lengths[rle.result$values] != number.of.txs),
|
||||
" log messages not aligned. Repairing...\n", sep = "")
|
||||
|
||||
run.lengths <- rle.result$lengths[rle.result$values]
|
||||
# When "values" is TRUE
|
||||
|
||||
number.of.txs.corrected<- run.lengths
|
||||
|
||||
peer_ip_port <- get.peer.ip.port.direction(monero.log[notify.lines], "Received NOTIFY_NEW_TRANSACTIONS")
|
||||
|
||||
tx.data <- data.frame(
|
||||
time = rep(get.time(monero.log[notify.lines]), times = number.of.txs.corrected + 1),
|
||||
# Do not get the actual time of the log message. Get the time that the
|
||||
# notify message was received.
|
||||
gossip.msg.id = rep(1:length(notify.lines), times = number.of.txs.corrected + 1),
|
||||
tx.hash = get.tx.hash(monero.log),
|
||||
stringsAsFactors = FALSE)
|
||||
|
||||
stopifnot( all(diff(tx.data$gossip.msg.id)[is.na(tx.data$tx.hash)[-1]] == 1) )
|
||||
# Checks to make sure each transmission of txs from each node has its
|
||||
# own gossip.msg.id, i.e. that there is no misalignment in the data.frame
|
||||
|
||||
tx.data <- cbind(tx.data, peer_ip_port[tx.data$gossip.msg.id, ])
|
||||
|
||||
tx.data <- tx.data[!is.na(tx.data$tx.hash), ]
|
||||
# Remove rows that were the NOTIFY_NEW_TRANSACTIONS message
|
||||
rownames(tx.data) <- NULL
|
||||
|
||||
if (!is.null(output.file)) {
|
||||
cat(base::date(), " Writing data to ", output.file, " with highest compression settings...\n", sep = "")
|
||||
qs::qsave(tx.data, output.file, preset = "custom", algorithm = "zstd_stream", compress_level = 22, shuffle_control = 15)
|
||||
}
|
||||
|
||||
tx.data
|
||||
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
# Suggested by
|
||||
# https://r-pkgs.org/dependencies-in-practice.html#how-to-not-use-a-package-in-imports
|
||||
# We need the R.utils package to enable reading compressed log file with
|
||||
# data.table::fread(), but we do not explicitly load it anywhere. R CMD check
|
||||
# creates a NOTE if it is not loaded anywhere
|
||||
ignore_unused_imports <- function() {
|
||||
R.utils::decompressFile
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
37
R/ping.R
37
R/ping.R
|
@ -1,6 +1,25 @@
|
|||
|
||||
|
||||
|
||||
|
||||
|
||||
# This function is not exported
|
||||
get.peer.ip.port.direction <- function(x, notification) {
|
||||
x <- stringr::str_extract(x,
|
||||
paste0("\t\\[+([^\\[\\]]+)\\]*:([0-9]+) ([INCOUT]{3})\\] ", notification),
|
||||
group = 1:3)
|
||||
if (!is.matrix(x)) {
|
||||
x <- matrix(x, ncol = 3, byrow = TRUE)
|
||||
# stringr::str_extract() creates an atomic vector instead of a matrix
|
||||
# if the x has only one element
|
||||
}
|
||||
x <- as.data.frame(x)
|
||||
colnames(x) <- c("ip", "port", "direction")
|
||||
x
|
||||
}
|
||||
|
||||
|
||||
|
||||
#' Ping peer nodes for latency measurement
|
||||
#'
|
||||
#' @param bitmonero.dir .bitmonero directory where the monero.log file is.
|
||||
|
@ -18,7 +37,7 @@
|
|||
#' }
|
||||
ping.peers <- function(bitmonero.dir = "~/.bitmonero", output.file = "monero_peer_pings.csv", sleep = 10, ping.count = 5, threads = NULL) {
|
||||
|
||||
Sys.setenv(VROOM_CONNECTION_SIZE = formatC(131072 * 100, format = "fg"))
|
||||
Sys.setenv(VROOM_CONNECTION_SIZE = formatC(131072 * 10000, format = "fg"))
|
||||
# Fixes occasional issues with https://github.com/tidyverse/vroom/issues/364
|
||||
# formatC() is used since VROOM_CONNECTION_SIZE cannot interpret scientific notation
|
||||
|
||||
|
@ -92,21 +111,7 @@ ping.peers <- function(bitmonero.dir = "~/.bitmonero", output.file = "monero_pee
|
|||
next
|
||||
}
|
||||
|
||||
get.peer.ip.port.direction <- function(x) {
|
||||
x <- stringr::str_extract(x,
|
||||
"\t\\[+([^\\[\\]]+)\\]*:([0-9]+) ([INCOUT]{3})\\] Received NOTIFY_NEW_TRANSACTIONS",
|
||||
group = c(1, 2, 3))
|
||||
if (!is.matrix(x)) {
|
||||
x <- matrix(x, ncol = 3, byrow = TRUE)
|
||||
# stringr::str_extract() creates an atomic vector instaed of a matrix
|
||||
# if the x has only one element
|
||||
}
|
||||
x <- as.data.frame(x)
|
||||
colnames(x) <- c("ip", "port", "direction")
|
||||
x
|
||||
}
|
||||
|
||||
peers <- get.peer.ip.port.direction(tail.file[ip.lines])
|
||||
peers <- get.peer.ip.port.direction(tail.file[ip.lines], "Received NOTIFY_NEW_TRANSACTIONS")
|
||||
peers <- unique(peers)
|
||||
missed.parsing <- (! complete.cases(peers))
|
||||
if (any(missed.parsing)) {
|
||||
|
|
24
man/get.p2p.log.Rd
Normal file
24
man/get.p2p.log.Rd
Normal file
|
@ -0,0 +1,24 @@
|
|||
% Generated by roxygen2: do not edit by hand
|
||||
% Please edit documentation in R/parse.R
|
||||
\name{get.p2p.log}
|
||||
\alias{get.p2p.log}
|
||||
\title{Parse p2p log files}
|
||||
\usage{
|
||||
get.p2p.log(bitmonero.dir = "~/.bitmonero", output.file = NULL)
|
||||
}
|
||||
\arguments{
|
||||
\item{bitmonero.dir}{Directory location of the log files}
|
||||
|
||||
\item{output.file}{Optional file to output compressed data}
|
||||
}
|
||||
\value{
|
||||
data.frame of each transaction gossip message for each peer.
|
||||
}
|
||||
\description{
|
||||
Parse p2p log files
|
||||
}
|
||||
\examples{
|
||||
\dontrun{
|
||||
get.p2p.log()
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue