mirror of
https://github.com/golang/go
synced 2024-11-18 18:44:42 -07:00
193 lines
5.7 KiB
Go
193 lines
5.7 KiB
Go
|
package protocol
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"encoding/json"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"strings"
|
||
|
"time"
|
||
|
|
||
|
"golang.org/x/tools/internal/jsonrpc2"
|
||
|
)
|
||
|
|
||
|
type loggingStream struct {
|
||
|
stream jsonrpc2.Stream
|
||
|
log io.Writer
|
||
|
}
|
||
|
|
||
|
// LoggingStream returns a stream that does LSP protocol logging too
|
||
|
func LoggingStream(str jsonrpc2.Stream, w io.Writer) jsonrpc2.Stream {
|
||
|
return &loggingStream{str, w}
|
||
|
}
|
||
|
|
||
|
func (s *loggingStream) Read(ctx context.Context) ([]byte, int64, error) {
|
||
|
data, count, err := s.stream.Read(ctx)
|
||
|
if err == nil {
|
||
|
logIn(s.log, data)
|
||
|
}
|
||
|
return data, count, err
|
||
|
}
|
||
|
|
||
|
func (s *loggingStream) Write(ctx context.Context, data []byte) (int64, error) {
|
||
|
logOut(s.log, data)
|
||
|
count, err := s.stream.Write(ctx, data)
|
||
|
return count, err
|
||
|
}
|
||
|
|
||
|
// Combined has all the fields of both Request and Response.
|
||
|
// We can decode this and then work out which it is.
|
||
|
type Combined struct {
|
||
|
VersionTag jsonrpc2.VersionTag `json:"jsonrpc"`
|
||
|
ID *jsonrpc2.ID `json:"id,omitempty"`
|
||
|
Method string `json:"method"`
|
||
|
Params *json.RawMessage `json:"params,omitempty"`
|
||
|
Result *json.RawMessage `json:"result,omitempty"`
|
||
|
Error *jsonrpc2.Error `json:"error,omitempty"`
|
||
|
}
|
||
|
|
||
|
type req struct {
|
||
|
method string
|
||
|
start time.Time
|
||
|
}
|
||
|
|
||
|
var (
|
||
|
// remember to delete the entries after responses are seen TODO
|
||
|
clientCalls = make(map[string]req)
|
||
|
serverCalls = make(map[string]req)
|
||
|
)
|
||
|
|
||
|
const eor = "\r\n\r\n\r\n"
|
||
|
|
||
|
func strID(x *jsonrpc2.ID) string {
|
||
|
if x == nil {
|
||
|
// should never happen, but we need a number
|
||
|
return "999999999"
|
||
|
}
|
||
|
if x.Name != "" {
|
||
|
return x.Name
|
||
|
}
|
||
|
return fmt.Sprintf("%d", x.Number)
|
||
|
}
|
||
|
|
||
|
func logCommon(outfd io.Writer, data []byte) (*Combined, time.Time, string) {
|
||
|
if outfd == nil {
|
||
|
return nil, time.Time{}, ""
|
||
|
}
|
||
|
var v Combined
|
||
|
err := json.Unmarshal(data, &v)
|
||
|
if err != nil {
|
||
|
fmt.Fprintf(outfd, "Unmarshal %v\n", err)
|
||
|
panic(err) // do better
|
||
|
}
|
||
|
tm := time.Now()
|
||
|
tmfmt := tm.Format("15:04:05.000 PM")
|
||
|
return &v, tm, tmfmt
|
||
|
}
|
||
|
|
||
|
// logOut and logIn could be combined. "received"<->"Sending", serverCalls<->clientCalls
|
||
|
// but it wouldn't be a lot shorter or clearer and "shutdown" is a special case
|
||
|
|
||
|
// Writing a message to the client, log it
|
||
|
func logOut(outfd io.Writer, data []byte) {
|
||
|
v, tm, tmfmt := logCommon(outfd, data)
|
||
|
if v == nil {
|
||
|
return
|
||
|
}
|
||
|
if v.Error != nil {
|
||
|
id := strID(v.ID)
|
||
|
fmt.Fprintf(outfd, "[Error - %s] Received #%s %s%s", tmfmt, id, v.Error, eor)
|
||
|
return
|
||
|
}
|
||
|
buf := strings.Builder{}
|
||
|
id := strID(v.ID)
|
||
|
fmt.Fprintf(&buf, "[Trace - %s] ", tmfmt) // common beginning
|
||
|
if v.ID != nil && v.Method != "" && v.Params != nil {
|
||
|
fmt.Fprintf(&buf, "Received request '%s - (%s)'.\n", v.Method, id)
|
||
|
fmt.Fprintf(&buf, "Params: %s%s", *v.Params, eor)
|
||
|
serverCalls[id] = req{method: v.Method, start: tm}
|
||
|
} else if v.ID != nil && v.Method == "" && v.Params == nil {
|
||
|
elapsed := tm.Sub(clientCalls[id].start)
|
||
|
fmt.Fprintf(&buf, "Received response '%s - (%s)' in %dms.\n",
|
||
|
clientCalls[id].method, id, elapsed/time.Millisecond)
|
||
|
if v.Result == nil {
|
||
|
fmt.Fprintf(&buf, "Result: {}%s", eor)
|
||
|
} else {
|
||
|
fmt.Fprintf(&buf, "Result: %s%s", string(*v.Result), eor)
|
||
|
}
|
||
|
} else if v.ID == nil && v.Method != "" && v.Params != nil {
|
||
|
fmt.Fprintf(&buf, "Received notification '%s'.\n", v.Method)
|
||
|
fmt.Fprintf(&buf, "Params: %s%s", *v.Params, eor)
|
||
|
} else { // for completeness, as it should never happen
|
||
|
buf = strings.Builder{} // undo common Trace
|
||
|
fmt.Fprintf(&buf, "[Error - %s] on write ID?%v method:%q Params:%v Result:%v Error:%v%s",
|
||
|
tmfmt, v.ID != nil, v.Method, v.Params != nil,
|
||
|
v.Result != nil, v.Error != nil, eor)
|
||
|
p := "null"
|
||
|
if v.Params != nil {
|
||
|
p = string(*v.Params)
|
||
|
}
|
||
|
r := "null"
|
||
|
if v.Result != nil {
|
||
|
r = string(*v.Result)
|
||
|
}
|
||
|
fmt.Fprintf(&buf, "%s\n%s\n%s\r\n\r\n\r\n", p, r, v.Error)
|
||
|
}
|
||
|
outfd.Write([]byte(buf.String()))
|
||
|
}
|
||
|
|
||
|
// Got a message from the client, log it
|
||
|
func logIn(outfd io.Writer, data []byte) {
|
||
|
v, tm, tmfmt := logCommon(outfd, data)
|
||
|
if v == nil {
|
||
|
return
|
||
|
}
|
||
|
// ID Method Params => Sending request
|
||
|
// ID !Method Result(might be null, but !Params) => Sending response (could we get an Error?)
|
||
|
// !ID Method Params => Sending notification
|
||
|
if v.Error != nil { // does this ever happen?
|
||
|
id := strID(v.ID)
|
||
|
fmt.Fprintf(outfd, "[Error - %s] Sent #%s %s%s", tmfmt, id, v.Error, eor)
|
||
|
return
|
||
|
}
|
||
|
buf := strings.Builder{}
|
||
|
id := strID(v.ID)
|
||
|
fmt.Fprintf(&buf, "[Trace - %s] ", tmfmt) // common beginning
|
||
|
if v.ID != nil && v.Method != "" && (v.Params != nil || v.Method == "shutdown") {
|
||
|
fmt.Fprintf(&buf, "Sending request '%s - (%s)'.\n", v.Method, id)
|
||
|
x := "{}"
|
||
|
if v.Params != nil {
|
||
|
x = string(*v.Params)
|
||
|
}
|
||
|
fmt.Fprintf(&buf, "Params: %s%s", x, eor)
|
||
|
clientCalls[id] = req{method: v.Method, start: tm}
|
||
|
} else if v.ID != nil && v.Method == "" && v.Params == nil {
|
||
|
elapsed := tm.Sub(serverCalls[id].start)
|
||
|
fmt.Fprintf(&buf, "Sending response '%s - (%s)' in %dms.\n",
|
||
|
serverCalls[id].method, id, elapsed/time.Millisecond)
|
||
|
if v.Result == nil {
|
||
|
fmt.Fprintf(&buf, "Result: {}%s", eor)
|
||
|
} else {
|
||
|
fmt.Fprintf(&buf, "Result: %s%s", string(*v.Result), eor)
|
||
|
}
|
||
|
} else if v.ID == nil && v.Method != "" && v.Params != nil {
|
||
|
fmt.Fprintf(&buf, "Sending notification '%s'.\n", v.Method)
|
||
|
fmt.Fprintf(&buf, "Params: %s%s", *v.Params, eor)
|
||
|
} else { // for completeness, as it should never happen
|
||
|
buf = strings.Builder{} // undo common Trace
|
||
|
fmt.Fprintf(&buf, "[Error - %s] on read ID?%v method:%q Params:%v Result:%v Error:%v%s",
|
||
|
tmfmt, v.ID != nil, v.Method, v.Params != nil,
|
||
|
v.Result != nil, v.Error != nil, eor)
|
||
|
p := "null"
|
||
|
if v.Params != nil {
|
||
|
p = string(*v.Params)
|
||
|
}
|
||
|
r := "null"
|
||
|
if v.Result != nil {
|
||
|
r = string(*v.Result)
|
||
|
}
|
||
|
fmt.Fprintf(&buf, "%s\n%s\n%s\r\n\r\n\r\n", p, r, v.Error)
|
||
|
}
|
||
|
outfd.Write([]byte(buf.String()))
|
||
|
}
|