mirror of
https://github.com/octoleo/syncthing.git
synced 2025-01-22 14:48:30 +00:00
9ce6a73f42
This reverts commit dc6a10dff4bddab178118cc0e32a56925118293b.
322 lines
8.8 KiB
Go
322 lines
8.8 KiB
Go
// Copyright (C) 2019 The Syncthing Authors.
|
|
//
|
|
// This Source Code Form is subject to the terms of the Mozilla Public
|
|
// License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
// You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
package main
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"errors"
|
|
"io"
|
|
"log"
|
|
"regexp"
|
|
"strings"
|
|
"sync"
|
|
|
|
raven "github.com/getsentry/raven-go"
|
|
"github.com/maruel/panicparse/v2/stack"
|
|
)
|
|
|
|
const reportServer = "https://crash.syncthing.net/report/"
|
|
|
|
var loader = newGithubSourceCodeLoader()
|
|
|
|
func init() {
|
|
raven.SetSourceCodeLoader(loader)
|
|
}
|
|
|
|
var (
|
|
clients = make(map[string]*raven.Client)
|
|
clientsMut sync.Mutex
|
|
)
|
|
|
|
type sentryService struct {
|
|
dsn string
|
|
inbox chan sentryRequest
|
|
}
|
|
|
|
type sentryRequest struct {
|
|
reportID string
|
|
userID string
|
|
data []byte
|
|
}
|
|
|
|
func (s *sentryService) Serve(ctx context.Context) {
|
|
for {
|
|
select {
|
|
case req := <-s.inbox:
|
|
pkt, err := parseCrashReport(req.reportID, req.data)
|
|
if err != nil {
|
|
log.Println("Failed to parse crash report:", err)
|
|
continue
|
|
}
|
|
if err := sendReport(s.dsn, pkt, req.userID); err != nil {
|
|
log.Println("Failed to send crash report:", err)
|
|
}
|
|
|
|
case <-ctx.Done():
|
|
return
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *sentryService) Send(reportID, userID string, data []byte) bool {
|
|
select {
|
|
case s.inbox <- sentryRequest{reportID, userID, data}:
|
|
return true
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
func sendReport(dsn string, pkt *raven.Packet, userID string) error {
|
|
pkt.Interfaces = append(pkt.Interfaces, &raven.User{ID: userID})
|
|
|
|
clientsMut.Lock()
|
|
defer clientsMut.Unlock()
|
|
|
|
cli, ok := clients[dsn]
|
|
if !ok {
|
|
var err error
|
|
cli, err = raven.New(dsn)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
clients[dsn] = cli
|
|
}
|
|
|
|
// The client sets release and such on the packet before sending, in the
|
|
// misguided idea that it knows this better than than the packet we give
|
|
// it. So we copy the values from the packet to the client first...
|
|
cli.SetRelease(pkt.Release)
|
|
cli.SetEnvironment(pkt.Environment)
|
|
|
|
defer cli.Wait()
|
|
_, errC := cli.Capture(pkt, nil)
|
|
return <-errC
|
|
}
|
|
|
|
func parseCrashReport(path string, report []byte) (*raven.Packet, error) {
|
|
parts := bytes.SplitN(report, []byte("\n"), 2)
|
|
if len(parts) != 2 {
|
|
return nil, errors.New("no first line")
|
|
}
|
|
|
|
version, err := parseVersion(string(parts[0]))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
report = parts[1]
|
|
|
|
foundPanic := false
|
|
var subjectLine []byte
|
|
for {
|
|
parts = bytes.SplitN(report, []byte("\n"), 2)
|
|
if len(parts) != 2 {
|
|
return nil, errors.New("no panic line found")
|
|
}
|
|
|
|
line := parts[0]
|
|
report = parts[1]
|
|
|
|
if foundPanic {
|
|
// The previous line was our "Panic at ..." header. We are now
|
|
// at the beginning of the real panic trace and this is our
|
|
// subject line.
|
|
subjectLine = line
|
|
break
|
|
} else if bytes.HasPrefix(line, []byte("Panic at")) {
|
|
foundPanic = true
|
|
}
|
|
}
|
|
|
|
r := bytes.NewReader(report)
|
|
ctx, _, err := stack.ScanSnapshot(r, io.Discard, stack.DefaultOpts())
|
|
if err != nil && err != io.EOF {
|
|
return nil, err
|
|
}
|
|
if ctx == nil || len(ctx.Goroutines) == 0 {
|
|
return nil, errors.New("no goroutines found")
|
|
}
|
|
|
|
// Lock the source code loader to the version we are processing here.
|
|
if version.commit != "" {
|
|
// We have a commit hash, so we know exactly which source to use
|
|
loader.LockWithVersion(version.commit)
|
|
} else if strings.HasPrefix(version.tag, "v") {
|
|
// Lets hope the tag is close enough
|
|
loader.LockWithVersion(version.tag)
|
|
} else {
|
|
// Last resort
|
|
loader.LockWithVersion("main")
|
|
}
|
|
defer loader.Unlock()
|
|
|
|
var trace raven.Stacktrace
|
|
for _, gr := range ctx.Goroutines {
|
|
if gr.First {
|
|
trace.Frames = make([]*raven.StacktraceFrame, len(gr.Stack.Calls))
|
|
for i, sc := range gr.Stack.Calls {
|
|
trace.Frames[len(trace.Frames)-1-i] = raven.NewStacktraceFrame(0, sc.Func.Name, sc.RemoteSrcPath, sc.Line, 3, nil)
|
|
}
|
|
break
|
|
}
|
|
}
|
|
|
|
pkt := packet(version, "crash")
|
|
pkt.Message = string(subjectLine)
|
|
pkt.Extra = raven.Extra{
|
|
"url": reportServer + path,
|
|
}
|
|
pkt.Interfaces = []raven.Interface{&trace}
|
|
pkt.Fingerprint = crashReportFingerprint(pkt.Message)
|
|
|
|
return pkt, nil
|
|
}
|
|
|
|
var (
|
|
indexRe = regexp.MustCompile(`\[[-:0-9]+\]`)
|
|
sizeRe = regexp.MustCompile(`(length|capacity) [0-9]+`)
|
|
ldbPosRe = regexp.MustCompile(`(\(pos=)([0-9]+)\)`)
|
|
ldbChecksumRe = regexp.MustCompile(`(want=0x)([a-z0-9]+)( got=0x)([a-z0-9]+)`)
|
|
ldbFileRe = regexp.MustCompile(`(\[file=)([0-9]+)(\.ldb\])`)
|
|
ldbInternalKeyRe = regexp.MustCompile(`(internal key ")[^"]+(", len=)[0-9]+`)
|
|
ldbPathRe = regexp.MustCompile(`(open|write|read) .+[\\/].+[\\/]index[^\\/]+[\\/][^\\/]+: `)
|
|
)
|
|
|
|
func sanitizeMessageLDB(message string) string {
|
|
message = ldbPosRe.ReplaceAllString(message, "${1}x)")
|
|
message = ldbFileRe.ReplaceAllString(message, "${1}x${3}")
|
|
message = ldbChecksumRe.ReplaceAllString(message, "${1}X${3}X")
|
|
message = ldbInternalKeyRe.ReplaceAllString(message, "${1}x${2}x")
|
|
message = ldbPathRe.ReplaceAllString(message, "$1 x: ")
|
|
return message
|
|
}
|
|
|
|
func crashReportFingerprint(message string) []string {
|
|
// Do not fingerprint on the stack in case of db corruption or fatal
|
|
// db io error - where it occurs doesn't matter.
|
|
orig := message
|
|
message = sanitizeMessageLDB(message)
|
|
if message != orig {
|
|
return []string{message}
|
|
}
|
|
|
|
message = indexRe.ReplaceAllString(message, "[x]")
|
|
message = sizeRe.ReplaceAllString(message, "$1 x")
|
|
|
|
// {{ default }} is what sentry uses as a fingerprint by default. While
|
|
// never specified, the docs point at this being some hash derived from the
|
|
// stack trace. Here we include the filtered panic message on top of that.
|
|
// https://docs.sentry.io/platforms/go/data-management/event-grouping/sdk-fingerprinting/#basic-example
|
|
return []string{"{{ default }}", message}
|
|
}
|
|
|
|
// syncthing v1.1.4-rc.1+30-g6aaae618-dirty-crashrep "Erbium Earthworm" (go1.12.5 darwin-amd64) jb@kvin.kastelo.net 2019-05-23 16:08:14 UTC [foo, bar]
|
|
// or, somewhere along the way the "+" in the version tag disappeared:
|
|
// syncthing v1.23.7-dev.26.gdf7b56ae.dirty-stversionextra "Fermium Flea" (go1.20.5 darwin-arm64) jb@ok.kastelo.net 2023-07-12 06:55:26 UTC [Some Wrapper, purego, stnoupgrade]
|
|
var (
|
|
longVersionRE = regexp.MustCompile(`syncthing\s+(v[^\s]+)\s+"([^"]+)"\s\(([^\s]+)\s+([^-]+)-([^)]+)\)\s+([^\s]+)[^\[]*(?:\[(.+)\])?$`)
|
|
gitExtraRE = regexp.MustCompile(`\.\d+\.g[0-9a-f]+`) // ".1.g6aaae618"
|
|
gitExtraSepRE = regexp.MustCompile(`[.-]`) // dot or dash
|
|
)
|
|
|
|
type version struct {
|
|
version string // "v1.1.4-rc.1+30-g6aaae618-dirty-crashrep"
|
|
tag string // "v1.1.4-rc.1"
|
|
commit string // "6aaae618", blank when absent
|
|
codename string // "Erbium Earthworm"
|
|
runtime string // "go1.12.5"
|
|
goos string // "darwin"
|
|
goarch string // "amd64"
|
|
builder string // "jb@kvin.kastelo.net"
|
|
extra []string // "foo", "bar"
|
|
}
|
|
|
|
func (v version) environment() string {
|
|
if v.commit != "" {
|
|
return "Development"
|
|
}
|
|
if strings.Contains(v.tag, "-rc.") {
|
|
return "Candidate"
|
|
}
|
|
if strings.Contains(v.tag, "-") {
|
|
return "Beta"
|
|
}
|
|
return "Stable"
|
|
}
|
|
|
|
func parseVersion(line string) (version, error) {
|
|
m := longVersionRE.FindStringSubmatch(line)
|
|
if len(m) == 0 {
|
|
return version{}, errors.New("unintelligeble version string")
|
|
}
|
|
|
|
v := version{
|
|
version: m[1],
|
|
codename: m[2],
|
|
runtime: m[3],
|
|
goos: m[4],
|
|
goarch: m[5],
|
|
builder: m[6],
|
|
}
|
|
|
|
// Split the version tag into tag and commit. This is old style
|
|
// v1.2.3-something.4+11-g12345678 or newer with just dots
|
|
// v1.2.3-something.4.11.g12345678 or v1.2.3-dev.11.g12345678.
|
|
parts := []string{v.version}
|
|
if strings.Contains(v.version, "+") {
|
|
parts = strings.Split(v.version, "+")
|
|
} else {
|
|
idxs := gitExtraRE.FindStringIndex(v.version)
|
|
if len(idxs) > 0 {
|
|
parts = []string{v.version[:idxs[0]], v.version[idxs[0]+1:]}
|
|
}
|
|
}
|
|
v.tag = parts[0]
|
|
if len(parts) > 1 {
|
|
fields := gitExtraSepRE.Split(parts[1], -1)
|
|
if len(fields) >= 2 && strings.HasPrefix(fields[1], "g") {
|
|
v.commit = fields[1][1:]
|
|
}
|
|
}
|
|
|
|
if len(m) >= 8 && m[7] != "" {
|
|
tags := strings.Split(m[7], ",")
|
|
for i := range tags {
|
|
tags[i] = strings.TrimSpace(tags[i])
|
|
}
|
|
v.extra = tags
|
|
}
|
|
|
|
return v, nil
|
|
}
|
|
|
|
func packet(version version, reportType string) *raven.Packet {
|
|
pkt := &raven.Packet{
|
|
Platform: "go",
|
|
Release: version.tag,
|
|
Environment: version.environment(),
|
|
Tags: raven.Tags{
|
|
raven.Tag{Key: "version", Value: version.version},
|
|
raven.Tag{Key: "tag", Value: version.tag},
|
|
raven.Tag{Key: "codename", Value: version.codename},
|
|
raven.Tag{Key: "runtime", Value: version.runtime},
|
|
raven.Tag{Key: "goos", Value: version.goos},
|
|
raven.Tag{Key: "goarch", Value: version.goarch},
|
|
raven.Tag{Key: "builder", Value: version.builder},
|
|
raven.Tag{Key: "report_type", Value: reportType},
|
|
},
|
|
}
|
|
if version.commit != "" {
|
|
pkt.Tags = append(pkt.Tags, raven.Tag{Key: "commit", Value: version.commit})
|
|
}
|
|
for _, tag := range version.extra {
|
|
pkt.Tags = append(pkt.Tags, raven.Tag{Key: tag, Value: "1"})
|
|
}
|
|
return pkt
|
|
}
|