2018-01-29 19:44:18 +00:00
|
|
|
// Copyright (c) 2017 Arista Networks, Inc.
|
|
|
|
// Use of this source code is governed by the Apache License 2.0
|
|
|
|
// that can be found in the COPYING file.
|
|
|
|
|
|
|
|
package gnmi
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
2018-09-05 15:36:14 +00:00
|
|
|
"fmt"
|
|
|
|
"math"
|
|
|
|
"net"
|
|
|
|
"time"
|
|
|
|
|
2018-01-29 19:44:18 +00:00
|
|
|
"io/ioutil"
|
|
|
|
"strings"
|
|
|
|
|
2018-09-05 15:36:14 +00:00
|
|
|
"github.com/aristanetworks/goarista/netns"
|
2018-01-29 19:44:18 +00:00
|
|
|
pb "github.com/openconfig/gnmi/proto/gnmi"
|
|
|
|
"google.golang.org/grpc"
|
|
|
|
"google.golang.org/grpc/credentials"
|
|
|
|
"google.golang.org/grpc/metadata"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
2018-09-05 15:36:14 +00:00
|
|
|
defaultPort = "6030"
|
2018-01-29 19:44:18 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Config is the gnmi.Client config
|
|
|
|
type Config struct {
|
|
|
|
Addr string
|
|
|
|
CAFile string
|
|
|
|
CertFile string
|
|
|
|
KeyFile string
|
|
|
|
Password string
|
|
|
|
Username string
|
|
|
|
TLS bool
|
|
|
|
}
|
|
|
|
|
2018-09-05 15:36:14 +00:00
|
|
|
// SubscribeOptions is the gNMI subscription request options
|
|
|
|
type SubscribeOptions struct {
|
|
|
|
UpdatesOnly bool
|
|
|
|
Prefix string
|
|
|
|
Mode string
|
|
|
|
StreamMode string
|
|
|
|
SampleInterval uint64
|
|
|
|
HeartbeatInterval uint64
|
|
|
|
Paths [][]string
|
|
|
|
}
|
|
|
|
|
2018-01-29 19:44:18 +00:00
|
|
|
// Dial connects to a gnmi service and returns a client
|
2018-09-05 15:36:14 +00:00
|
|
|
func Dial(cfg *Config) (pb.GNMIClient, error) {
|
2018-01-29 19:44:18 +00:00
|
|
|
var opts []grpc.DialOption
|
|
|
|
if cfg.TLS || cfg.CAFile != "" || cfg.CertFile != "" {
|
|
|
|
tlsConfig := &tls.Config{}
|
|
|
|
if cfg.CAFile != "" {
|
|
|
|
b, err := ioutil.ReadFile(cfg.CAFile)
|
|
|
|
if err != nil {
|
2018-09-05 15:36:14 +00:00
|
|
|
return nil, err
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
|
|
|
cp := x509.NewCertPool()
|
|
|
|
if !cp.AppendCertsFromPEM(b) {
|
2018-09-05 15:36:14 +00:00
|
|
|
return nil, fmt.Errorf("credentials: failed to append certificates")
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
|
|
|
tlsConfig.RootCAs = cp
|
|
|
|
} else {
|
|
|
|
tlsConfig.InsecureSkipVerify = true
|
|
|
|
}
|
|
|
|
if cfg.CertFile != "" {
|
|
|
|
if cfg.KeyFile == "" {
|
2018-09-05 15:36:14 +00:00
|
|
|
return nil, fmt.Errorf("please provide both -certfile and -keyfile")
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
|
|
|
cert, err := tls.LoadX509KeyPair(cfg.CertFile, cfg.KeyFile)
|
|
|
|
if err != nil {
|
2018-09-05 15:36:14 +00:00
|
|
|
return nil, err
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
|
|
|
tlsConfig.Certificates = []tls.Certificate{cert}
|
|
|
|
}
|
|
|
|
opts = append(opts, grpc.WithTransportCredentials(credentials.NewTLS(tlsConfig)))
|
|
|
|
} else {
|
|
|
|
opts = append(opts, grpc.WithInsecure())
|
|
|
|
}
|
|
|
|
|
|
|
|
if !strings.ContainsRune(cfg.Addr, ':') {
|
|
|
|
cfg.Addr += ":" + defaultPort
|
|
|
|
}
|
2018-09-05 15:36:14 +00:00
|
|
|
|
|
|
|
dial := func(addrIn string, time time.Duration) (net.Conn, error) {
|
|
|
|
var conn net.Conn
|
|
|
|
nsName, addr, err := netns.ParseAddress(addrIn)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
err = netns.Do(nsName, func() error {
|
|
|
|
var err error
|
|
|
|
conn, err = net.Dial("tcp", addr)
|
|
|
|
return err
|
|
|
|
})
|
|
|
|
return conn, err
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
|
|
|
|
2018-09-05 15:36:14 +00:00
|
|
|
opts = append(opts,
|
|
|
|
grpc.WithDialer(dial),
|
|
|
|
|
|
|
|
// Allows received protobuf messages to be larger than 4MB
|
|
|
|
grpc.WithDefaultCallOptions(grpc.MaxCallRecvMsgSize(math.MaxInt32)),
|
|
|
|
)
|
|
|
|
grpcconn, err := grpc.Dial(cfg.Addr, opts...)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to dial: %s", err)
|
|
|
|
}
|
|
|
|
return pb.NewGNMIClient(grpcconn), nil
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewContext returns a new context with username and password
|
|
|
|
// metadata if they are set in cfg.
|
|
|
|
func NewContext(ctx context.Context, cfg *Config) context.Context {
|
|
|
|
if cfg.Username != "" {
|
|
|
|
ctx = metadata.NewOutgoingContext(ctx, metadata.Pairs(
|
|
|
|
"username", cfg.Username,
|
|
|
|
"password", cfg.Password))
|
|
|
|
}
|
|
|
|
return ctx
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewGetRequest returns a GetRequest for the given paths
|
|
|
|
func NewGetRequest(paths [][]string) (*pb.GetRequest, error) {
|
|
|
|
req := &pb.GetRequest{
|
|
|
|
Path: make([]*pb.Path, len(paths)),
|
|
|
|
}
|
|
|
|
for i, p := range paths {
|
|
|
|
gnmiPath, err := ParseGNMIElements(p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
req.Path[i] = gnmiPath
|
|
|
|
}
|
|
|
|
return req, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewSubscribeRequest returns a SubscribeRequest for the given paths
|
2018-09-05 15:36:14 +00:00
|
|
|
func NewSubscribeRequest(subscribeOptions *SubscribeOptions) (*pb.SubscribeRequest, error) {
|
|
|
|
var mode pb.SubscriptionList_Mode
|
|
|
|
switch subscribeOptions.Mode {
|
|
|
|
case "once":
|
|
|
|
mode = pb.SubscriptionList_ONCE
|
|
|
|
case "poll":
|
|
|
|
mode = pb.SubscriptionList_POLL
|
|
|
|
case "stream":
|
|
|
|
mode = pb.SubscriptionList_STREAM
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("subscribe mode (%s) invalid", subscribeOptions.Mode)
|
|
|
|
}
|
|
|
|
|
|
|
|
var streamMode pb.SubscriptionMode
|
|
|
|
switch subscribeOptions.StreamMode {
|
|
|
|
case "on_change":
|
|
|
|
streamMode = pb.SubscriptionMode_ON_CHANGE
|
|
|
|
case "sample":
|
|
|
|
streamMode = pb.SubscriptionMode_SAMPLE
|
|
|
|
case "target_defined":
|
|
|
|
streamMode = pb.SubscriptionMode_TARGET_DEFINED
|
|
|
|
default:
|
|
|
|
return nil, fmt.Errorf("subscribe stream mode (%s) invalid", subscribeOptions.StreamMode)
|
|
|
|
}
|
|
|
|
|
|
|
|
prefixPath, err := ParseGNMIElements(SplitPath(subscribeOptions.Prefix))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-01-29 19:44:18 +00:00
|
|
|
subList := &pb.SubscriptionList{
|
2018-09-05 15:36:14 +00:00
|
|
|
Subscription: make([]*pb.Subscription, len(subscribeOptions.Paths)),
|
|
|
|
Mode: mode,
|
|
|
|
UpdatesOnly: subscribeOptions.UpdatesOnly,
|
|
|
|
Prefix: prefixPath,
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
2018-09-05 15:36:14 +00:00
|
|
|
for i, p := range subscribeOptions.Paths {
|
2018-01-29 19:44:18 +00:00
|
|
|
gnmiPath, err := ParseGNMIElements(p)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2018-09-05 15:36:14 +00:00
|
|
|
subList.Subscription[i] = &pb.Subscription{
|
|
|
|
Path: gnmiPath,
|
|
|
|
Mode: streamMode,
|
|
|
|
SampleInterval: subscribeOptions.SampleInterval,
|
|
|
|
HeartbeatInterval: subscribeOptions.HeartbeatInterval,
|
|
|
|
}
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|
2018-09-05 15:36:14 +00:00
|
|
|
return &pb.SubscribeRequest{Request: &pb.SubscribeRequest_Subscribe{
|
|
|
|
Subscribe: subList}}, nil
|
2018-01-29 19:44:18 +00:00
|
|
|
}
|