mirror of
https://github.com/rclone/rclone.git
synced 2025-01-14 18:28:24 +01:00
71 lines
1.8 KiB
Go
71 lines
1.8 KiB
Go
|
// Copyright (C) 2020 Storj Labs, Inc.
|
||
|
// See LICENSE for copying information.
|
||
|
|
||
|
// Package rpcpeer implements context.Context peer tagging.
|
||
|
package rpcpeer
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"crypto/tls"
|
||
|
"net"
|
||
|
|
||
|
"github.com/zeebo/errs"
|
||
|
|
||
|
"storj.io/common/internal/grpchook"
|
||
|
"storj.io/drpc/drpcctx"
|
||
|
)
|
||
|
|
||
|
// Error is the class of errors returned by this package.
|
||
|
var Error = errs.Class("rpcpeer")
|
||
|
|
||
|
// Peer represents an rpc peer.
|
||
|
type Peer struct {
|
||
|
Addr net.Addr
|
||
|
State tls.ConnectionState
|
||
|
}
|
||
|
|
||
|
// peerKey is used as a unique value for context keys.
|
||
|
type peerKey struct{}
|
||
|
|
||
|
// NewContext returns a new context with the peer associated as a value.
|
||
|
func NewContext(ctx context.Context, peer *Peer) context.Context {
|
||
|
return context.WithValue(ctx, peerKey{}, peer)
|
||
|
}
|
||
|
|
||
|
// FromContext returns the peer that was previously associated by NewContext.
|
||
|
func FromContext(ctx context.Context) (*Peer, error) {
|
||
|
if peer, ok := ctx.Value(peerKey{}).(*Peer); ok {
|
||
|
return peer, nil
|
||
|
} else if peer, drpcErr := drpcInternalFromContext(ctx); drpcErr == nil {
|
||
|
return peer, nil
|
||
|
} else if addr, state, grpcErr := grpchook.InternalFromContext(ctx); grpcErr == nil {
|
||
|
return &Peer{Addr: addr, State: state}, nil
|
||
|
} else {
|
||
|
if grpcErr == grpchook.ErrNotHooked {
|
||
|
grpcErr = nil
|
||
|
}
|
||
|
return nil, errs.Combine(drpcErr, grpcErr)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// drpcInternalFromContext returns a peer from the context using drpc.
|
||
|
func drpcInternalFromContext(ctx context.Context) (*Peer, error) {
|
||
|
tr, ok := drpcctx.Transport(ctx)
|
||
|
if !ok {
|
||
|
return nil, Error.New("unable to get drpc peer from context")
|
||
|
}
|
||
|
|
||
|
conn, ok := tr.(interface {
|
||
|
RemoteAddr() net.Addr
|
||
|
ConnectionState() tls.ConnectionState
|
||
|
})
|
||
|
if !ok {
|
||
|
return nil, Error.New("drpc transport does not have required methods")
|
||
|
}
|
||
|
|
||
|
return &Peer{
|
||
|
Addr: conn.RemoteAddr(),
|
||
|
State: conn.ConnectionState(),
|
||
|
}, nil
|
||
|
}
|