2019-12-17 17:38:03 +02:00
|
|
|
// Package handler implements service debug handler embedded in go-micro services
|
2019-08-06 18:53:14 +02:00
|
|
|
package handler
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2022-10-20 13:00:50 +02:00
|
|
|
"errors"
|
|
|
|
"io"
|
2019-08-06 18:53:14 +02:00
|
|
|
"time"
|
|
|
|
|
2021-10-12 13:55:53 +02:00
|
|
|
"go-micro.dev/v4/client"
|
|
|
|
"go-micro.dev/v4/debug/log"
|
|
|
|
proto "go-micro.dev/v4/debug/proto"
|
|
|
|
"go-micro.dev/v4/debug/stats"
|
|
|
|
"go-micro.dev/v4/debug/trace"
|
2019-08-06 18:53:14 +02:00
|
|
|
)
|
|
|
|
|
2022-09-30 16:27:07 +02:00
|
|
|
// NewHandler returns an instance of the Debug Handler.
|
2020-05-24 19:45:57 +02:00
|
|
|
func NewHandler(c client.Client) *Debug {
|
2020-01-29 18:05:58 +02:00
|
|
|
return &Debug{
|
|
|
|
log: log.DefaultLog,
|
|
|
|
stats: stats.DefaultStats,
|
2020-01-30 00:40:43 +02:00
|
|
|
trace: trace.DefaultTracer,
|
2020-01-29 18:05:58 +02:00
|
|
|
}
|
|
|
|
}
|
2019-08-06 18:53:14 +02:00
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
var _ proto.DebugHandler = (*Debug)(nil)
|
|
|
|
|
2019-11-26 19:04:44 +02:00
|
|
|
type Debug struct {
|
2022-09-30 16:27:07 +02:00
|
|
|
// must honor the debug handler
|
2019-11-28 13:05:35 +02:00
|
|
|
proto.DebugHandler
|
2019-12-18 20:36:42 +02:00
|
|
|
// the logger for retrieving logs
|
2019-11-28 13:36:38 +02:00
|
|
|
log log.Log
|
2019-12-18 20:36:42 +02:00
|
|
|
// the stats collector
|
|
|
|
stats stats.Stats
|
2020-01-24 23:29:29 +02:00
|
|
|
// the tracer
|
2020-01-29 17:45:11 +02:00
|
|
|
trace trace.Tracer
|
2019-11-26 19:04:44 +02:00
|
|
|
}
|
|
|
|
|
2019-08-06 18:53:14 +02:00
|
|
|
func (d *Debug) Health(ctx context.Context, req *proto.HealthRequest, rsp *proto.HealthResponse) error {
|
|
|
|
rsp.Status = "ok"
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
func (d *Debug) MessageBus(ctx context.Context, stream proto.Debug_MessageBusStream) error {
|
|
|
|
for {
|
|
|
|
_, err := stream.Recv()
|
|
|
|
if errors.Is(err, io.EOF) {
|
|
|
|
return nil
|
|
|
|
} else if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
rsp := proto.BusMsg{
|
|
|
|
Msg: "Request received!",
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := stream.Send(&rsp); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-06 18:53:14 +02:00
|
|
|
func (d *Debug) Stats(ctx context.Context, req *proto.StatsRequest, rsp *proto.StatsResponse) error {
|
2019-12-18 20:36:42 +02:00
|
|
|
stats, err := d.stats.Read()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(stats) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// write the response values
|
|
|
|
rsp.Timestamp = uint64(stats[0].Timestamp)
|
|
|
|
rsp.Started = uint64(stats[0].Started)
|
|
|
|
rsp.Uptime = uint64(stats[0].Uptime)
|
|
|
|
rsp.Memory = stats[0].Memory
|
|
|
|
rsp.Gc = stats[0].GC
|
|
|
|
rsp.Threads = stats[0].Threads
|
|
|
|
rsp.Requests = stats[0].Requests
|
|
|
|
rsp.Errors = stats[0].Errors
|
|
|
|
|
2019-08-06 18:53:14 +02:00
|
|
|
return nil
|
|
|
|
}
|
2019-11-26 17:39:55 +02:00
|
|
|
|
2020-01-24 23:29:29 +02:00
|
|
|
func (d *Debug) Trace(ctx context.Context, req *proto.TraceRequest, rsp *proto.TraceResponse) error {
|
|
|
|
traces, err := d.trace.Read(trace.ReadTrace(req.Id))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2020-02-12 12:57:17 +02:00
|
|
|
for _, t := range traces {
|
|
|
|
var typ proto.SpanType
|
2022-09-30 20:32:55 +02:00
|
|
|
|
2020-02-12 12:57:17 +02:00
|
|
|
switch t.Type {
|
|
|
|
case trace.SpanTypeRequestInbound:
|
|
|
|
typ = proto.SpanType_INBOUND
|
|
|
|
case trace.SpanTypeRequestOutbound:
|
|
|
|
typ = proto.SpanType_OUTBOUND
|
|
|
|
}
|
2022-09-30 20:32:55 +02:00
|
|
|
|
2020-01-24 23:29:29 +02:00
|
|
|
rsp.Spans = append(rsp.Spans, &proto.Span{
|
2020-02-12 12:57:17 +02:00
|
|
|
Trace: t.Trace,
|
|
|
|
Id: t.Id,
|
|
|
|
Parent: t.Parent,
|
|
|
|
Name: t.Name,
|
|
|
|
Started: uint64(t.Started.UnixNano()),
|
|
|
|
Duration: uint64(t.Duration.Nanoseconds()),
|
|
|
|
Type: typ,
|
|
|
|
Metadata: t.Metadata,
|
2020-01-24 23:29:29 +02:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-10-20 13:00:50 +02:00
|
|
|
func (d *Debug) Log(ctx context.Context, req *proto.LogRequest, stream proto.Debug_LogStream) error {
|
2019-12-02 16:55:35 +02:00
|
|
|
|
2019-11-28 20:08:48 +02:00
|
|
|
var options []log.ReadOption
|
|
|
|
|
2019-12-01 15:15:10 +02:00
|
|
|
since := time.Unix(req.Since, 0)
|
2019-11-27 20:38:26 +02:00
|
|
|
if !since.IsZero() {
|
2019-11-28 20:08:48 +02:00
|
|
|
options = append(options, log.Since(since))
|
|
|
|
}
|
|
|
|
|
|
|
|
count := int(req.Count)
|
|
|
|
if count > 0 {
|
|
|
|
options = append(options, log.Count(count))
|
2019-11-27 20:38:26 +02:00
|
|
|
}
|
|
|
|
|
2019-11-30 14:39:29 +02:00
|
|
|
if req.Stream {
|
2019-12-17 17:38:03 +02:00
|
|
|
// TODO: we need to figure out how to close the log stream
|
2019-12-01 15:15:10 +02:00
|
|
|
// It seems like when a client disconnects,
|
2019-11-30 14:39:29 +02:00
|
|
|
// the connection stays open until some timeout expires
|
|
|
|
// or something like that; that means the map of streams
|
2019-12-01 15:15:10 +02:00
|
|
|
// might end up leaking memory if not cleaned up properly
|
2019-12-17 18:56:55 +02:00
|
|
|
lgStream, err := d.log.Stream()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer lgStream.Stop()
|
2019-12-17 17:38:03 +02:00
|
|
|
|
2019-12-17 18:56:55 +02:00
|
|
|
for record := range lgStream.Chan() {
|
2019-12-18 20:36:42 +02:00
|
|
|
// copy metadata
|
|
|
|
metadata := make(map[string]string)
|
|
|
|
for k, v := range record.Metadata {
|
|
|
|
metadata[k] = v
|
|
|
|
}
|
|
|
|
// send record
|
|
|
|
if err := stream.Send(&proto.Record{
|
|
|
|
Timestamp: record.Timestamp.Unix(),
|
|
|
|
Message: record.Message.(string),
|
|
|
|
Metadata: metadata,
|
|
|
|
}); err != nil {
|
2019-11-30 14:39:29 +02:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
2019-12-17 17:38:03 +02:00
|
|
|
|
2019-11-30 14:39:29 +02:00
|
|
|
// done streaming, return
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2019-11-28 20:08:48 +02:00
|
|
|
// get the log records
|
2019-12-17 18:56:55 +02:00
|
|
|
records, err := d.log.Read(options...)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2019-12-17 17:38:03 +02:00
|
|
|
|
2019-11-30 14:39:29 +02:00
|
|
|
// send all the logs downstream
|
2019-11-27 20:38:26 +02:00
|
|
|
for _, record := range records {
|
2019-12-18 20:36:42 +02:00
|
|
|
// copy metadata
|
|
|
|
metadata := make(map[string]string)
|
|
|
|
for k, v := range record.Metadata {
|
|
|
|
metadata[k] = v
|
|
|
|
}
|
|
|
|
// send record
|
|
|
|
if err := stream.Send(&proto.Record{
|
|
|
|
Timestamp: record.Timestamp.Unix(),
|
|
|
|
Message: record.Message.(string),
|
|
|
|
Metadata: metadata,
|
|
|
|
}); err != nil {
|
2019-11-30 14:39:29 +02:00
|
|
|
return err
|
2019-11-27 20:38:26 +02:00
|
|
|
}
|
2019-11-30 14:39:29 +02:00
|
|
|
}
|
2019-11-27 20:38:26 +02:00
|
|
|
|
2019-11-30 14:39:29 +02:00
|
|
|
return nil
|
|
|
|
}
|