1
0
mirror of https://github.com/go-micro/go-micro.git synced 2025-01-17 17:44:30 +02:00

188 lines
4.0 KiB
Go
Raw Normal View History

2019-12-17 15:38:03 +00:00
// Package handler implements service debug handler embedded in go-micro services
2019-08-06 17:53:14 +01:00
package handler
import (
"context"
"errors"
"io"
2019-08-06 17:53:14 +01:00
"time"
2024-06-04 21:40:43 +01:00
"go-micro.dev/v5/client"
"go-micro.dev/v5/debug/log"
proto "go-micro.dev/v5/debug/proto"
"go-micro.dev/v5/debug/stats"
"go-micro.dev/v5/debug/trace"
2019-08-06 17:53:14 +01:00
)
2022-09-30 16:27:07 +02:00
// NewHandler returns an instance of the Debug Handler.
2020-05-24 18:45:57 +01:00
func NewHandler(c client.Client) *Debug {
return &Debug{
log: log.DefaultLog,
stats: stats.DefaultStats,
2020-01-29 22:40:43 +00:00
trace: trace.DefaultTracer,
}
}
2019-08-06 17:53:14 +01:00
var _ proto.DebugHandler = (*Debug)(nil)
type Debug struct {
2022-09-30 16:27:07 +02:00
// must honor the debug handler
2019-11-28 11:05:35 +00:00
proto.DebugHandler
2019-12-18 18:36:42 +00:00
// the logger for retrieving logs
2019-11-28 11:36:38 +00:00
log log.Log
2019-12-18 18:36:42 +00:00
// the stats collector
stats stats.Stats
2020-01-24 21:29:29 +00:00
// the tracer
2020-01-29 15:45:11 +00:00
trace trace.Tracer
}
2019-08-06 17:53:14 +01:00
func (d *Debug) Health(ctx context.Context, req *proto.HealthRequest, rsp *proto.HealthResponse) error {
rsp.Status = "ok"
return nil
}
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 17:53:14 +01:00
func (d *Debug) Stats(ctx context.Context, req *proto.StatsRequest, rsp *proto.StatsResponse) error {
2019-12-18 18:36:42 +00: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 17:53:14 +01:00
return nil
}
2020-01-24 21:29:29 +00: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
}
for _, t := range traces {
var typ proto.SpanType
2022-09-30 20:32:55 +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 21:29:29 +00:00
rsp.Spans = append(rsp.Spans, &proto.Span{
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 21:29:29 +00:00
})
}
return nil
}
func (d *Debug) Log(ctx context.Context, req *proto.LogRequest, stream proto.Debug_LogStream) error {
var options []log.ReadOption
since := time.Unix(req.Since, 0)
2019-11-27 18:38:26 +00:00
if !since.IsZero() {
options = append(options, log.Since(since))
}
count := int(req.Count)
if count > 0 {
options = append(options, log.Count(count))
2019-11-27 18:38:26 +00:00
}
if req.Stream {
2019-12-17 15:38:03 +00:00
// TODO: we need to figure out how to close the log stream
// It seems like when a client disconnects,
// the connection stays open until some timeout expires
// or something like that; that means the map of streams
// might end up leaking memory if not cleaned up properly
2019-12-17 16:56:55 +00:00
lgStream, err := d.log.Stream()
if err != nil {
return err
}
defer lgStream.Stop()
2019-12-17 15:38:03 +00:00
2019-12-17 16:56:55 +00:00
for record := range lgStream.Chan() {
2019-12-18 18:36:42 +00: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 {
return err
}
}
2019-12-17 15:38:03 +00:00
// done streaming, return
return nil
}
// get the log records
2019-12-17 16:56:55 +00:00
records, err := d.log.Read(options...)
if err != nil {
return err
}
2019-12-17 15:38:03 +00:00
// send all the logs downstream
2019-11-27 18:38:26 +00:00
for _, record := range records {
2019-12-18 18:36:42 +00: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 {
return err
2019-11-27 18:38:26 +00:00
}
}
2019-11-27 18:38:26 +00:00
return nil
}