| // Copyright 2019 The Go Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style |
| // license that can be found in the LICENSE file. |
| |
| package jsonrpc2 |
| |
| import ( |
| "context" |
| "fmt" |
| "sync" |
| |
| "golang.org/x/tools/internal/telemetry/event" |
| ) |
| |
| // Handler is invoked to handle incoming requests. |
| // If the request returns false from IsNotify then the Handler must eventually |
| // call Reply on the Conn with the supplied request. |
| // The handler should return ErrNotHandled if it could not handle the request. |
| type Handler func(context.Context, *Request) error |
| |
| // Direction is used to indicate to a logger whether the logged message was being |
| // sent or received. |
| type Direction bool |
| |
| const ( |
| // Send indicates the message is outgoing. |
| Send = Direction(true) |
| // Receive indicates the message is incoming. |
| Receive = Direction(false) |
| ) |
| |
| func (d Direction) String() string { |
| switch d { |
| case Send: |
| return "send" |
| case Receive: |
| return "receive" |
| default: |
| panic("unreachable") |
| } |
| } |
| |
| // MethodNotFound is a Handler that replies to all call requests with the |
| // standard method not found response. |
| // This should normally be the final handler in a chain. |
| func MethodNotFound(ctx context.Context, r *Request) error { |
| return r.Reply(ctx, nil, NewErrorf(CodeMethodNotFound, "method %q not found", r.Method)) |
| } |
| |
| // MustReply creates a Handler that panics if the wrapped handler does |
| // not call Reply for every request that it is passed. |
| func MustReply(handler Handler) Handler { |
| return func(ctx context.Context, req *Request) error { |
| err := handler(ctx, req) |
| if req.done != nil { |
| panic(fmt.Errorf("request %q was never replied to", req.Method)) |
| } |
| return err |
| } |
| } |
| |
| // CancelHandler returns a handler that supports cancellation, and a canceller |
| // that can be used to trigger canceling in progress requests. |
| func CancelHandler(handler Handler) (Handler, Canceller) { |
| var mu sync.Mutex |
| handling := make(map[ID]context.CancelFunc) |
| wrapped := func(ctx context.Context, req *Request) error { |
| if req.ID != nil { |
| cancelCtx, cancel := context.WithCancel(ctx) |
| ctx = cancelCtx |
| mu.Lock() |
| handling[*req.ID] = cancel |
| mu.Unlock() |
| req.OnReply(func() { |
| mu.Lock() |
| delete(handling, *req.ID) |
| mu.Unlock() |
| }) |
| } |
| return handler(ctx, req) |
| } |
| return wrapped, func(id ID) { |
| mu.Lock() |
| cancel, found := handling[id] |
| mu.Unlock() |
| if found { |
| cancel() |
| } |
| } |
| } |
| |
| // AsyncHandler returns a handler that processes each request goes in its own |
| // goroutine. |
| // The handler returns immediately, without the request being processed. |
| // Each request then waits for the previous request to finish before it starts. |
| // This allows the stream to unblock at the cost of unbounded goroutines |
| // all stalled on the previous one. |
| func AsyncHandler(handler Handler) Handler { |
| nextRequest := make(chan struct{}) |
| close(nextRequest) |
| return func(ctx context.Context, req *Request) error { |
| waitForPrevious := nextRequest |
| nextRequest = make(chan struct{}) |
| unlockNext := nextRequest |
| req.OnReply(func() { close(unlockNext) }) |
| _, queueDone := event.StartSpan(ctx, "queued") |
| go func() { |
| <-waitForPrevious |
| queueDone() |
| if err := handler(ctx, req); err != nil { |
| event.Error(ctx, "jsonrpc2 async message delivery failed", err) |
| } |
| }() |
| return nil |
| } |
| } |