mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2024-12-22 19:50:36 +00:00
117 lines
3.1 KiB
Go
117 lines
3.1 KiB
Go
|
package diagnostics
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"io/fs"
|
||
|
"net/http"
|
||
|
"os"
|
||
|
"path/filepath"
|
||
|
"strconv"
|
||
|
|
||
|
"github.com/urfave/cli/v2"
|
||
|
|
||
|
"github.com/ledgerwatch/erigon/turbo/logging"
|
||
|
)
|
||
|
|
||
|
func SetupLogsAccess(ctx *cli.Context) {
|
||
|
dirPath := ctx.String(logging.LogDirPathFlag.Name)
|
||
|
if dirPath == "" {
|
||
|
return
|
||
|
}
|
||
|
http.HandleFunc("/debug/metrics/logs/list", func(w http.ResponseWriter, r *http.Request) {
|
||
|
w.Header().Set("Access-Control-Allow-Origin", "*")
|
||
|
writeLogsList(w, dirPath)
|
||
|
})
|
||
|
http.HandleFunc("/debug/metrics/logs/read", func(w http.ResponseWriter, r *http.Request) {
|
||
|
w.Header().Set("Access-Control-Allow-Origin", "*")
|
||
|
writeLogsRead(w, r, dirPath)
|
||
|
})
|
||
|
}
|
||
|
|
||
|
func writeLogsList(w io.Writer, dirPath string) {
|
||
|
entries, err := os.ReadDir(dirPath)
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(w, "ERROR: listing directory %s: %v\n", dirPath, err)
|
||
|
return
|
||
|
}
|
||
|
//nolint: prealloc
|
||
|
var infos []fs.FileInfo
|
||
|
for _, entry := range entries {
|
||
|
fileInfo, err := os.Stat(filepath.Join(dirPath, entry.Name()))
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(w, "ERROR: stat file %s: %v\n", entry.Name(), err)
|
||
|
return
|
||
|
}
|
||
|
if fileInfo.IsDir() {
|
||
|
continue
|
||
|
}
|
||
|
infos = append(infos, fileInfo)
|
||
|
}
|
||
|
fmt.Fprintf(w, "SUCCESS\n")
|
||
|
for _, fileInfo := range infos {
|
||
|
fmt.Fprintf(w, "%s | %d\n", fileInfo.Name(), fileInfo.Size())
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func writeLogsRead(w io.Writer, r *http.Request, dirPath string) {
|
||
|
if err := r.ParseForm(); err != nil {
|
||
|
fmt.Fprintf(w, "ERROR: parsing arguments: %v\n", err)
|
||
|
return
|
||
|
}
|
||
|
file := r.Form.Get("file")
|
||
|
if file == "" {
|
||
|
fmt.Fprintf(w, "ERROR: file argument is required - specify the name of log file to read")
|
||
|
return
|
||
|
}
|
||
|
fileInfo, err := os.Stat(filepath.Join(dirPath, file))
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(w, "ERROR: stat file %s: %v\n", file, err)
|
||
|
return
|
||
|
}
|
||
|
if fileInfo.IsDir() {
|
||
|
fmt.Fprintf(w, "ERROR: %s is a directory, needs to be a file", file)
|
||
|
return
|
||
|
}
|
||
|
offsetStr := r.Form.Get("offset")
|
||
|
if offsetStr == "" {
|
||
|
fmt.Fprintf(w, "ERROR: offset argument is required - specify where to start reading in the file")
|
||
|
return
|
||
|
}
|
||
|
offset, err := strconv.ParseInt(offsetStr, 10, 64)
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(w, "ERROR: offset %s is not a Uint64 number: %v\n", offsetStr, err)
|
||
|
return
|
||
|
}
|
||
|
if offset < 0 {
|
||
|
fmt.Fprintf(w, "ERROR: offset %d must be non-negative\n", offset)
|
||
|
return
|
||
|
}
|
||
|
if offset > fileInfo.Size() {
|
||
|
fmt.Fprintf(w, "ERROR: offset %d must be no greater than file size %d\n", offset, fileInfo.Size())
|
||
|
return
|
||
|
}
|
||
|
f, err := os.Open(filepath.Join(dirPath, file))
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(w, "ERROR: opening file %s: %v\n", file, err)
|
||
|
return
|
||
|
}
|
||
|
var buf [16 * 1024]byte
|
||
|
if _, err := f.Seek(offset, 0); err != nil {
|
||
|
fmt.Fprintf(w, "ERROR: seeking in file: %s to %d: %v\n", file, offset, err)
|
||
|
return
|
||
|
}
|
||
|
var n int
|
||
|
var readTotal int
|
||
|
for n, err = f.Read(buf[readTotal:]); err == nil && readTotal < len(buf); n, err = f.Read(buf[readTotal:]) {
|
||
|
readTotal += n
|
||
|
}
|
||
|
if err != nil && !errors.Is(err, io.EOF) {
|
||
|
fmt.Fprintf(w, "ERROR: reading in file: %s at %d: %v\n", file, readTotal, err)
|
||
|
return
|
||
|
}
|
||
|
fmt.Fprintf(w, "SUCCESS: %d-%d/%d\n", offset, offset+int64(readTotal), fileInfo.Size())
|
||
|
w.Write(buf[:readTotal])
|
||
|
}
|