187 lines
5.3 KiB
Go
187 lines
5.3 KiB
Go
// Copyright (c) 2016 Tigera, Inc. All rights reserved.
|
|
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package model
|
|
|
|
import (
|
|
"fmt"
|
|
"regexp"
|
|
|
|
"reflect"
|
|
|
|
"github.com/projectcalico/libcalico-go/lib/errors"
|
|
"github.com/projectcalico/libcalico-go/lib/net"
|
|
"github.com/projectcalico/libcalico-go/lib/numorstring"
|
|
log "github.com/sirupsen/logrus"
|
|
)
|
|
|
|
var (
|
|
matchGlobalBGPPeer = regexp.MustCompile("^/?calico/bgp/v1/global/peer_v./([^/]+)$")
|
|
matchHostBGPPeer = regexp.MustCompile("^/?calico/bgp/v1/host/([^/]+)/peer_v./([^/]+)$")
|
|
typeBGPPeer = reflect.TypeOf(BGPPeer{})
|
|
)
|
|
|
|
type NodeBGPPeerKey struct {
|
|
Nodename string `json:"-" validate:"omitempty"`
|
|
PeerIP net.IP `json:"-" validate:"required"`
|
|
}
|
|
|
|
func (key NodeBGPPeerKey) defaultPath() (string, error) {
|
|
if key.PeerIP.IP == nil {
|
|
return "", errors.ErrorInsufficientIdentifiers{Name: "peerIP"}
|
|
}
|
|
if key.Nodename == "" {
|
|
return "", errors.ErrorInsufficientIdentifiers{Name: "node"}
|
|
}
|
|
e := fmt.Sprintf("/calico/bgp/v1/host/%s/peer_v%d/%s",
|
|
key.Nodename, key.PeerIP.Version(), key.PeerIP)
|
|
return e, nil
|
|
}
|
|
|
|
func (key NodeBGPPeerKey) defaultDeletePath() (string, error) {
|
|
return key.defaultPath()
|
|
}
|
|
|
|
func (key NodeBGPPeerKey) defaultDeleteParentPaths() ([]string, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
func (key NodeBGPPeerKey) valueType() (reflect.Type, error) {
|
|
return typeBGPPeer, nil
|
|
}
|
|
|
|
func (key NodeBGPPeerKey) String() string {
|
|
return fmt.Sprintf("BGPPeer(node=%s, ip=%s)", key.Nodename, key.PeerIP)
|
|
}
|
|
|
|
type NodeBGPPeerListOptions struct {
|
|
Nodename string
|
|
PeerIP net.IP
|
|
}
|
|
|
|
func (options NodeBGPPeerListOptions) defaultPathRoot() string {
|
|
if options.Nodename == "" {
|
|
return "/calico/bgp/v1/host"
|
|
} else if options.PeerIP.IP == nil {
|
|
return fmt.Sprintf("/calico/bgp/v1/host/%s",
|
|
options.Nodename)
|
|
} else {
|
|
return fmt.Sprintf("/calico/bgp/v1/host/%s/peer_v%d/%s",
|
|
options.Nodename, options.PeerIP.Version(), options.PeerIP)
|
|
}
|
|
}
|
|
|
|
func (options NodeBGPPeerListOptions) KeyFromDefaultPath(path string) Key {
|
|
log.Debugf("Get BGPPeer key from %s", path)
|
|
nodename := ""
|
|
peerIP := net.IP{}
|
|
ekeyb := []byte(path)
|
|
|
|
if r := matchHostBGPPeer.FindAllSubmatch(ekeyb, -1); len(r) == 1 {
|
|
nodename = string(r[0][1])
|
|
if err := peerIP.UnmarshalText(r[0][2]); err != nil {
|
|
log.WithError(err).WithField("PeerIP", r[0][2]).Error("Error unmarshalling GlobalBGPPeer IP address")
|
|
return nil
|
|
}
|
|
} else {
|
|
log.Debugf("%s didn't match regex", path)
|
|
return nil
|
|
}
|
|
|
|
if options.PeerIP.IP != nil && !options.PeerIP.Equal(peerIP.IP) {
|
|
log.Debugf("Didn't match peerIP %s != %s", options.PeerIP.String(), peerIP.String())
|
|
return nil
|
|
}
|
|
if options.Nodename != "" && nodename != options.Nodename {
|
|
log.Debugf("Didn't match hostname %s != %s", options.Nodename, nodename)
|
|
return nil
|
|
}
|
|
return NodeBGPPeerKey{PeerIP: peerIP, Nodename: nodename}
|
|
}
|
|
|
|
type GlobalBGPPeerKey struct {
|
|
PeerIP net.IP `json:"-" validate:"required"`
|
|
}
|
|
|
|
func (key GlobalBGPPeerKey) defaultPath() (string, error) {
|
|
if key.PeerIP.IP == nil {
|
|
return "", errors.ErrorInsufficientIdentifiers{Name: "peerIP"}
|
|
}
|
|
e := fmt.Sprintf("/calico/bgp/v1/global/peer_v%d/%s",
|
|
key.PeerIP.Version(), key.PeerIP)
|
|
return e, nil
|
|
}
|
|
|
|
func (key GlobalBGPPeerKey) defaultDeletePath() (string, error) {
|
|
return key.defaultPath()
|
|
}
|
|
|
|
func (key GlobalBGPPeerKey) defaultDeleteParentPaths() ([]string, error) {
|
|
return nil, nil
|
|
}
|
|
|
|
func (key GlobalBGPPeerKey) valueType() (reflect.Type, error) {
|
|
return typeBGPPeer, nil
|
|
}
|
|
|
|
func (key GlobalBGPPeerKey) String() string {
|
|
return fmt.Sprintf("BGPPeer(global, ip=%s)", key.PeerIP)
|
|
}
|
|
|
|
type GlobalBGPPeerListOptions struct {
|
|
PeerIP net.IP
|
|
}
|
|
|
|
func (options GlobalBGPPeerListOptions) defaultPathRoot() string {
|
|
if options.PeerIP.IP == nil {
|
|
return "/calico/bgp/v1/global"
|
|
} else {
|
|
return fmt.Sprintf("/calico/bgp/v1/global/peer_v%d/%s",
|
|
options.PeerIP.Version(), options.PeerIP)
|
|
}
|
|
}
|
|
|
|
func (options GlobalBGPPeerListOptions) KeyFromDefaultPath(path string) Key {
|
|
log.Debugf("Get BGPPeer key from %s", path)
|
|
peerIP := net.IP{}
|
|
ekeyb := []byte(path)
|
|
|
|
if r := matchGlobalBGPPeer.FindAllSubmatch(ekeyb, -1); len(r) == 1 {
|
|
if err := peerIP.UnmarshalText(r[0][1]); err != nil {
|
|
log.WithError(err).WithField("PeerIP", r[0][1]).Error("Error unmarshalling GlobalBGPPeer IP address")
|
|
return nil
|
|
}
|
|
} else {
|
|
log.Debugf("%s didn't match regex", path)
|
|
return nil
|
|
}
|
|
|
|
if options.PeerIP.IP != nil && !options.PeerIP.Equal(peerIP.IP) {
|
|
log.Debugf("Didn't match peerIP %s != %s", options.PeerIP.String(), peerIP.String())
|
|
return nil
|
|
}
|
|
return GlobalBGPPeerKey{PeerIP: peerIP}
|
|
}
|
|
|
|
type BGPPeer struct {
|
|
// PeerIP is the IP address of the BGP peer.
|
|
PeerIP net.IP `json:"ip"`
|
|
|
|
// ASNum is the AS number of the peer. Note that we write out the
|
|
// value as a string in to the backend, because confd templating
|
|
// converts large uints to float e notation which breaks the BIRD
|
|
// configuration.
|
|
ASNum numorstring.ASNumber `json:"as_num,string"`
|
|
}
|