mirror of
https://github.com/golang/go
synced 2024-11-05 22:36:10 -07:00
3eebf4bf9d
Change-Id: I9aff86f7ab06e61849495b9c73553147e29343f1 Reviewed-on: https://go-review.googlesource.com/c/tools/+/226840 Run-TryBot: Ian Cottrell <iancottrell@google.com> Reviewed-by: Robert Findley <rfindley@google.com>
162 lines
5.0 KiB
Go
162 lines
5.0 KiB
Go
// 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
|
|
|
|
// LegacyHooks is a temporary measure during migration from the old Handler
|
|
// interface to the new HandleFunc.
|
|
// The intent is to delete this interface in a later cl.
|
|
type LegacyHooks interface {
|
|
// Deliver 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.
|
|
// Handlers are called synchronously, they should pass the work off to a go
|
|
// routine if they are going to take a long time.
|
|
// If Deliver returns true all subsequent handlers will be invoked with
|
|
// delivered set to true, and should not attempt to deliver the message.
|
|
Deliver(ctx context.Context, r *Request, delivered bool) bool
|
|
|
|
// Cancel is invoked for cancelled outgoing requests.
|
|
// It is okay to use the connection to send notifications, but the context will
|
|
// be in the cancelled state, so you must do it with the background context
|
|
// instead.
|
|
// If Cancel returns true all subsequent handlers will be invoked with
|
|
// cancelled set to true, and should not attempt to cancel the message.
|
|
Cancel(ctx context.Context, conn *Conn, id ID, cancelled bool) bool
|
|
|
|
// Request is called near the start of processing any request.
|
|
Request(ctx context.Context, conn *Conn, direction Direction, r *WireRequest) context.Context
|
|
}
|
|
|
|
// 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
|
|
}
|
|
}
|
|
|
|
func legacyDeliverHandler(handler Handler) Handler {
|
|
return func(ctx context.Context, req *Request) error {
|
|
if req.conn.LegacyHooks != nil {
|
|
if req.conn.LegacyHooks.Deliver(ctx, req, false) {
|
|
return nil
|
|
}
|
|
}
|
|
return handler(ctx, req)
|
|
}
|
|
}
|
|
|
|
func legacyRequestHandler(handler Handler) Handler {
|
|
return func(ctx context.Context, req *Request) error {
|
|
if req.conn.LegacyHooks != nil {
|
|
ctx = req.conn.LegacyHooks.Request(ctx, req.conn, Receive, &req.WireRequest)
|
|
}
|
|
return handler(ctx, req)
|
|
}
|
|
}
|