forked from ligato/vpp-agent
-
Notifications
You must be signed in to change notification settings - Fork 0
/
bd_interface.go
127 lines (110 loc) · 4.04 KB
/
bd_interface.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
// Copyright (c) 2018 Cisco and/or its affiliates.
//
// 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 descriptor
import (
"github.com/gogo/protobuf/proto"
"github.com/ligato/cn-infra/logging"
"github.com/pkg/errors"
interfaces "github.com/ligato/vpp-agent/api/models/vpp/interfaces"
l2 "github.com/ligato/vpp-agent/api/models/vpp/l2"
"github.com/ligato/vpp-agent/pkg/idxvpp"
kvs "github.com/ligato/vpp-agent/plugins/kvscheduler/api"
"github.com/ligato/vpp-agent/plugins/vpp/l2plugin/descriptor/adapter"
"github.com/ligato/vpp-agent/plugins/vpp/l2plugin/vppcalls"
)
const (
// BDInterfaceDescriptorName is the name of the descriptor for bindings between
// VPP bridge domains and interfaces.
BDInterfaceDescriptorName = "vpp-bd-interface"
// dependency labels
interfaceDep = "interface-exists"
)
// BDInterfaceDescriptor teaches KVScheduler how to put interface into VPP bridge
// domain.
type BDInterfaceDescriptor struct {
// dependencies
log logging.Logger
bdIndex idxvpp.NameToIndex
bdHandler vppcalls.BridgeDomainVppAPI
}
// NewBDInterfaceDescriptor creates a new instance of the BDInterface descriptor.
func NewBDInterfaceDescriptor(bdIndex idxvpp.NameToIndex, bdHandler vppcalls.BridgeDomainVppAPI, log logging.PluginLogger) *BDInterfaceDescriptor {
return &BDInterfaceDescriptor{
bdIndex: bdIndex,
bdHandler: bdHandler,
log: log.NewLogger("bd-iface-descriptor"),
}
}
// GetDescriptor returns descriptor suitable for registration (via adapter) with
// the KVScheduler.
func (d *BDInterfaceDescriptor) GetDescriptor() *adapter.BDInterfaceDescriptor {
return &adapter.BDInterfaceDescriptor{
Name: BDInterfaceDescriptorName,
KeySelector: d.IsBDInterfaceKey,
ValueTypeName: proto.MessageName(&l2.BridgeDomain_Interface{}),
Create: d.Create,
Delete: d.Delete,
Dependencies: d.Dependencies,
}
}
// IsBDInterfaceKey returns true if the key is identifying binding between
// VPP bridge domain and interface.
func (d *BDInterfaceDescriptor) IsBDInterfaceKey(key string) bool {
_, _, isBDIfaceKey := l2.ParseBDInterfaceKey(key)
return isBDIfaceKey
}
// Create puts interface into bridge domain.
func (d *BDInterfaceDescriptor) Create(key string, bdIface *l2.BridgeDomain_Interface) (metadata interface{}, err error) {
// get bridge domain index
bdName, _, _ := l2.ParseBDInterfaceKey(key)
bdMeta, found := d.bdIndex.LookupByName(bdName)
if !found {
err = errors.Errorf("failed to obtain metadata for bridge domain %s", bdName)
d.log.Error(err)
return nil, err
}
// put interface into the bridge domain
err = d.bdHandler.AddInterfaceToBridgeDomain(bdMeta.GetIndex(), bdIface)
if err != nil {
d.log.Error(err)
return nil, err
}
return nil, nil
}
// Delete removes interface from bridge domain.
func (d *BDInterfaceDescriptor) Delete(key string, bdIface *l2.BridgeDomain_Interface, metadata interface{}) error {
// get bridge domain index
bdName, _, _ := l2.ParseBDInterfaceKey(key)
bdMeta, found := d.bdIndex.LookupByName(bdName)
if !found {
err := errors.Errorf("failed to obtain metadata for bridge domain %s", bdName)
d.log.Error(err)
return err
}
err := d.bdHandler.DeleteInterfaceFromBridgeDomain(bdMeta.GetIndex(), bdIface)
if err != nil {
d.log.Error(err)
return err
}
return nil
}
// Dependencies lists the interface as the only dependency for the binding.
func (d *BDInterfaceDescriptor) Dependencies(key string, value *l2.BridgeDomain_Interface) []kvs.Dependency {
return []kvs.Dependency{
{
Label: interfaceDep,
Key: interfaces.InterfaceKey(value.Name),
},
}
}