mirror of
https://github.com/tailscale/tailscale.git
synced 2025-01-10 01:53:49 +00:00
0d19f5d421
The log ID types were moved to a separate package so that code that only depend on log ID types do not need to link in the logic for the logtail client itself. Not all code need the logtail client. Signed-off-by: Joe Tsai <joetsai@digital-static.net>
116 lines
2.6 KiB
Go
116 lines
2.6 KiB
Go
// Copyright (c) Tailscale Inc & AUTHORS
|
|
// SPDX-License-Identifier: BSD-3-Clause
|
|
|
|
// The logreprocess program tails a log and reprocesses it.
|
|
package main
|
|
|
|
import (
|
|
"bufio"
|
|
"encoding/json"
|
|
"flag"
|
|
"io"
|
|
"log"
|
|
"net/http"
|
|
"os"
|
|
"strings"
|
|
"time"
|
|
|
|
"tailscale.com/types/logid"
|
|
)
|
|
|
|
func main() {
|
|
collection := flag.String("c", "", "logtail collection name to read")
|
|
apiKey := flag.String("p", "", "logtail API key")
|
|
timeout := flag.Duration("t", 0, "timeout after which logreprocess quits")
|
|
flag.Parse()
|
|
if len(flag.Args()) != 0 {
|
|
flag.Usage()
|
|
os.Exit(1)
|
|
}
|
|
log.SetFlags(0)
|
|
|
|
if *timeout != 0 {
|
|
go func() {
|
|
<-time.After(*timeout)
|
|
log.Printf("logreprocess: timeout reached, quitting")
|
|
os.Exit(1)
|
|
}()
|
|
}
|
|
|
|
req, err := http.NewRequest("GET", "https://log.tailscale.io/c/"+*collection+"?stream=true", nil)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
req.SetBasicAuth(*apiKey, "")
|
|
resp, err := http.DefaultClient.Do(req)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
defer resp.Body.Close()
|
|
|
|
if resp.StatusCode != 200 {
|
|
b, err := io.ReadAll(resp.Body)
|
|
if err != nil {
|
|
log.Fatalf("logreprocess: read error %d: %v", resp.StatusCode, err)
|
|
}
|
|
log.Fatalf("logreprocess: read error %d: %s", resp.StatusCode, string(b))
|
|
}
|
|
|
|
tracebackCache := make(map[logid.PublicID]*ProcessedMsg)
|
|
|
|
scanner := bufio.NewScanner(resp.Body)
|
|
for scanner.Scan() {
|
|
var msg Msg
|
|
if err := json.Unmarshal(scanner.Bytes(), &msg); err != nil {
|
|
log.Fatalf("logreprocess of %q: %v", string(scanner.Bytes()), err)
|
|
}
|
|
var pMsg *ProcessedMsg
|
|
if pMsg = tracebackCache[msg.Logtail.Instance]; pMsg != nil {
|
|
pMsg.Text += "\n" + msg.Text
|
|
if strings.HasPrefix(msg.Text, "Exception: ") {
|
|
delete(tracebackCache, msg.Logtail.Instance)
|
|
} else {
|
|
continue // write later
|
|
}
|
|
} else {
|
|
pMsg = &ProcessedMsg{
|
|
OrigInstance: msg.Logtail.Instance,
|
|
Text: msg.Text,
|
|
}
|
|
pMsg.Logtail.ClientTime = msg.Logtail.ClientTime
|
|
}
|
|
|
|
if strings.HasPrefix(msg.Text, "Traceback (most recent call last):") {
|
|
tracebackCache[msg.Logtail.Instance] = pMsg
|
|
continue // write later
|
|
}
|
|
|
|
b, err := json.Marshal(pMsg)
|
|
if err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
log.Printf("%s", b)
|
|
}
|
|
if err := scanner.Err(); err != nil {
|
|
log.Fatal(err)
|
|
}
|
|
}
|
|
|
|
type Msg struct {
|
|
Logtail struct {
|
|
Instance logid.PublicID `json:"instance"`
|
|
ClientTime time.Time `json:"client_time"`
|
|
} `json:"logtail"`
|
|
|
|
Text string `json:"text"`
|
|
}
|
|
|
|
type ProcessedMsg struct {
|
|
Logtail struct {
|
|
ClientTime time.Time `json:"client_time"`
|
|
} `json:"logtail"`
|
|
|
|
OrigInstance logid.PublicID `json:"orig_instance"`
|
|
Text string `json:"text"`
|
|
}
|