Flowtables implementation (#201)
Added flowtables implementation | Added flow_offload expression | Added tests
This commit is contained in:
parent
3ad45c080c
commit
c89a57ce19
|
@ -128,6 +128,8 @@ func exprsFromBytes(fam byte, ad *netlink.AttributeDecoder, b []byte) ([]Any, er
|
|||
e = &Connlimit{}
|
||||
case "queue":
|
||||
e = &Queue{}
|
||||
case "flow_offload":
|
||||
e = &FlowOffload{}
|
||||
}
|
||||
if e == nil {
|
||||
// TODO: introduce an opaque expression type so that users know
|
||||
|
|
|
@ -0,0 +1,59 @@
|
|||
// Copyright 2018 Google LLC. 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 expr
|
||||
|
||||
import (
|
||||
"encoding/binary"
|
||||
|
||||
"github.com/mdlayher/netlink"
|
||||
"golang.org/x/sys/unix"
|
||||
)
|
||||
|
||||
const NFTNL_EXPR_FLOW_TABLE_NAME = 1
|
||||
|
||||
type FlowOffload struct {
|
||||
Name string
|
||||
}
|
||||
|
||||
func (e *FlowOffload) marshal(fam byte) ([]byte, error) {
|
||||
data, err := netlink.MarshalAttributes([]netlink.Attribute{
|
||||
{Type: NFTNL_EXPR_FLOW_TABLE_NAME, Data: []byte(e.Name)},
|
||||
})
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return netlink.MarshalAttributes([]netlink.Attribute{
|
||||
{Type: unix.NFTA_EXPR_NAME, Data: []byte("flow_offload\x00")},
|
||||
{Type: unix.NLA_F_NESTED | unix.NFTA_EXPR_DATA, Data: data},
|
||||
})
|
||||
}
|
||||
|
||||
func (e *FlowOffload) unmarshal(fam byte, data []byte) error {
|
||||
ad, err := netlink.NewAttributeDecoder(data)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
ad.ByteOrder = binary.BigEndian
|
||||
for ad.Next() {
|
||||
switch ad.Type() {
|
||||
case NFTNL_EXPR_FLOW_TABLE_NAME:
|
||||
e.Name = ad.String()
|
||||
}
|
||||
}
|
||||
|
||||
return ad.Err()
|
||||
}
|
|
@ -0,0 +1,306 @@
|
|||
// Copyright 2018 Google LLC. 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 nftables
|
||||
|
||||
import (
|
||||
"encoding/binary"
|
||||
"fmt"
|
||||
|
||||
"github.com/google/nftables/binaryutil"
|
||||
"github.com/mdlayher/netlink"
|
||||
"golang.org/x/sys/unix"
|
||||
)
|
||||
|
||||
const (
|
||||
// not in ztypes_linux.go, added here
|
||||
// https://cs.opensource.google/go/x/sys/+/c6bc011c:unix/ztypes_linux.go;l=1870-1892
|
||||
NFT_MSG_NEWFLOWTABLE = 0x16
|
||||
NFT_MSG_GETFLOWTABLE = 0x17
|
||||
NFT_MSG_DELFLOWTABLE = 0x18
|
||||
)
|
||||
|
||||
const (
|
||||
// not in ztypes_linux.go, added here
|
||||
// https://git.netfilter.org/libnftnl/tree/include/linux/netfilter/nf_tables.h?id=84d12cfacf8ddd857a09435f3d982ab6250d250c#n1634
|
||||
_ = iota
|
||||
NFTA_FLOWTABLE_TABLE
|
||||
NFTA_FLOWTABLE_NAME
|
||||
NFTA_FLOWTABLE_HOOK
|
||||
NFTA_FLOWTABLE_USE
|
||||
NFTA_FLOWTABLE_HANDLE
|
||||
NFTA_FLOWTABLE_PAD
|
||||
NFTA_FLOWTABLE_FLAGS
|
||||
)
|
||||
|
||||
const (
|
||||
// not in ztypes_linux.go, added here
|
||||
// https://git.netfilter.org/libnftnl/tree/include/linux/netfilter/nf_tables.h?id=84d12cfacf8ddd857a09435f3d982ab6250d250c#n1657
|
||||
_ = iota
|
||||
NFTA_FLOWTABLE_HOOK_NUM
|
||||
NFTA_FLOWTABLE_PRIORITY
|
||||
NFTA_FLOWTABLE_DEVS
|
||||
)
|
||||
|
||||
const (
|
||||
// not in ztypes_linux.go, added here, used for flowtable device name specification
|
||||
// https://git.netfilter.org/libnftnl/tree/include/linux/netfilter/nf_tables.h?id=84d12cfacf8ddd857a09435f3d982ab6250d250c#n1709
|
||||
NFTA_DEVICE_NAME = 1
|
||||
)
|
||||
|
||||
type FlowtableFlags uint32
|
||||
|
||||
const (
|
||||
_ FlowtableFlags = iota
|
||||
FlowtableFlagsHWOffload
|
||||
FlowtableFlagsCounter
|
||||
FlowtableFlagsMask = (FlowtableFlagsHWOffload | FlowtableFlagsCounter)
|
||||
)
|
||||
|
||||
type FlowtableHook uint32
|
||||
|
||||
func FlowtableHookRef(h FlowtableHook) *FlowtableHook {
|
||||
return &h
|
||||
}
|
||||
|
||||
var (
|
||||
// Only ingress is supported
|
||||
// https://github.com/torvalds/linux/blob/b72018ab8236c3ae427068adeb94bdd3f20454ec/net/netfilter/nf_tables_api.c#L7378-L7379
|
||||
FlowtableHookIngress *FlowtableHook = FlowtableHookRef(unix.NF_NETDEV_INGRESS)
|
||||
)
|
||||
|
||||
type FlowtablePriority int32
|
||||
|
||||
func FlowtablePriorityRef(p FlowtablePriority) *FlowtablePriority {
|
||||
return &p
|
||||
}
|
||||
|
||||
var (
|
||||
// As per man page:
|
||||
// The priority can be a signed integer or filter which stands for 0. Addition and subtraction can be used to set relative priority, e.g. filter + 5 equals to 5.
|
||||
// https://git.netfilter.org/nftables/tree/doc/nft.txt?id=8c600a843b7c0c1cc275ecc0603bd1fc57773e98#n712
|
||||
FlowtablePriorityFilter *FlowtablePriority = FlowtablePriorityRef(0)
|
||||
)
|
||||
|
||||
type Flowtable struct {
|
||||
Table *Table
|
||||
Name string
|
||||
Hooknum *FlowtableHook
|
||||
Priority *FlowtablePriority
|
||||
Devices []string
|
||||
Use uint32
|
||||
// Bitmask flags, can be HW_OFFLOAD or COUNTER
|
||||
// https://git.netfilter.org/libnftnl/tree/include/linux/netfilter/nf_tables.h?id=84d12cfacf8ddd857a09435f3d982ab6250d250c#n1621
|
||||
Flags FlowtableFlags
|
||||
Handle uint64
|
||||
}
|
||||
|
||||
func (cc *Conn) AddFlowtable(f *Flowtable) *Flowtable {
|
||||
cc.mu.Lock()
|
||||
defer cc.mu.Unlock()
|
||||
|
||||
data := cc.marshalAttr([]netlink.Attribute{
|
||||
{Type: NFTA_FLOWTABLE_TABLE, Data: []byte(f.Table.Name)},
|
||||
{Type: NFTA_FLOWTABLE_NAME, Data: []byte(f.Name)},
|
||||
{Type: NFTA_FLOWTABLE_FLAGS, Data: binaryutil.BigEndian.PutUint32(uint32(f.Flags))},
|
||||
})
|
||||
|
||||
if f.Hooknum == nil {
|
||||
f.Hooknum = FlowtableHookIngress
|
||||
}
|
||||
|
||||
if f.Priority == nil {
|
||||
f.Priority = FlowtablePriorityFilter
|
||||
}
|
||||
|
||||
hookAttr := []netlink.Attribute{
|
||||
{Type: NFTA_FLOWTABLE_HOOK_NUM, Data: binaryutil.BigEndian.PutUint32(uint32(*f.Hooknum))},
|
||||
{Type: NFTA_FLOWTABLE_PRIORITY, Data: binaryutil.BigEndian.PutUint32(uint32(*f.Priority))},
|
||||
}
|
||||
if len(f.Devices) > 0 {
|
||||
devs := make([]netlink.Attribute, len(f.Devices))
|
||||
for i, d := range f.Devices {
|
||||
devs[i] = netlink.Attribute{Type: NFTA_DEVICE_NAME, Data: []byte(d)}
|
||||
}
|
||||
hookAttr = append(hookAttr, netlink.Attribute{
|
||||
Type: unix.NLA_F_NESTED | NFTA_FLOWTABLE_DEVS,
|
||||
Data: cc.marshalAttr(devs),
|
||||
})
|
||||
}
|
||||
data = append(data, cc.marshalAttr([]netlink.Attribute{
|
||||
{Type: unix.NLA_F_NESTED | NFTA_FLOWTABLE_HOOK, Data: cc.marshalAttr(hookAttr)},
|
||||
})...)
|
||||
|
||||
cc.messages = append(cc.messages, netlink.Message{
|
||||
Header: netlink.Header{
|
||||
Type: netlink.HeaderType((unix.NFNL_SUBSYS_NFTABLES << 8) | NFT_MSG_NEWFLOWTABLE),
|
||||
Flags: netlink.Request | netlink.Acknowledge | netlink.Create,
|
||||
},
|
||||
Data: append(extraHeader(uint8(f.Table.Family), 0), data...),
|
||||
})
|
||||
|
||||
return f
|
||||
}
|
||||
|
||||
func (cc *Conn) DelFlowtable(f *Flowtable) {
|
||||
cc.mu.Lock()
|
||||
defer cc.mu.Unlock()
|
||||
|
||||
data := cc.marshalAttr([]netlink.Attribute{
|
||||
{Type: NFTA_FLOWTABLE_TABLE, Data: []byte(f.Table.Name)},
|
||||
{Type: NFTA_FLOWTABLE_NAME, Data: []byte(f.Name)},
|
||||
})
|
||||
|
||||
cc.messages = append(cc.messages, netlink.Message{
|
||||
Header: netlink.Header{
|
||||
Type: netlink.HeaderType((unix.NFNL_SUBSYS_NFTABLES << 8) | NFT_MSG_DELFLOWTABLE),
|
||||
Flags: netlink.Request | netlink.Acknowledge,
|
||||
},
|
||||
Data: append(extraHeader(uint8(f.Table.Family), 0), data...),
|
||||
})
|
||||
}
|
||||
|
||||
func (cc *Conn) ListFlowtables(t *Table) ([]*Flowtable, error) {
|
||||
reply, err := cc.getFlowtables(t)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var fts []*Flowtable
|
||||
for _, msg := range reply {
|
||||
f, err := ftsFromMsg(msg)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
f.Table = t
|
||||
fts = append(fts, f)
|
||||
}
|
||||
|
||||
return fts, nil
|
||||
}
|
||||
|
||||
func (cc *Conn) getFlowtables(t *Table) ([]netlink.Message, error) {
|
||||
conn, closer, err := cc.netlinkConn()
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
defer func() { _ = closer() }()
|
||||
|
||||
attrs := []netlink.Attribute{
|
||||
{Type: NFTA_FLOWTABLE_TABLE, Data: []byte(t.Name + "\x00")},
|
||||
}
|
||||
data, err := netlink.MarshalAttributes(attrs)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
message := netlink.Message{
|
||||
Header: netlink.Header{
|
||||
Type: netlink.HeaderType((unix.NFNL_SUBSYS_NFTABLES << 8) | NFT_MSG_GETFLOWTABLE),
|
||||
Flags: netlink.Request | netlink.Acknowledge | netlink.Dump,
|
||||
},
|
||||
Data: append(extraHeader(uint8(t.Family), 0), data...),
|
||||
}
|
||||
|
||||
if _, err := conn.SendMessages([]netlink.Message{message}); err != nil {
|
||||
return nil, fmt.Errorf("SendMessages: %v", err)
|
||||
}
|
||||
|
||||
reply, err := receiveAckAware(conn, message.Header.Flags)
|
||||
if err != nil {
|
||||
return nil, fmt.Errorf("Receive: %v", err)
|
||||
}
|
||||
|
||||
return reply, nil
|
||||
}
|
||||
|
||||
func ftsFromMsg(msg netlink.Message) (*Flowtable, error) {
|
||||
flowHeaderType := netlink.HeaderType((unix.NFNL_SUBSYS_NFTABLES << 8) | NFT_MSG_NEWFLOWTABLE)
|
||||
if got, want := msg.Header.Type, flowHeaderType; got != want {
|
||||
return nil, fmt.Errorf("unexpected header type: got %v, want %v", got, want)
|
||||
}
|
||||
ad, err := netlink.NewAttributeDecoder(msg.Data[4:])
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
ad.ByteOrder = binary.BigEndian
|
||||
|
||||
var ft Flowtable
|
||||
for ad.Next() {
|
||||
switch ad.Type() {
|
||||
case NFTA_FLOWTABLE_NAME:
|
||||
ft.Name = ad.String()
|
||||
case NFTA_FLOWTABLE_USE:
|
||||
ft.Use = ad.Uint32()
|
||||
case NFTA_FLOWTABLE_HANDLE:
|
||||
ft.Handle = ad.Uint64()
|
||||
case NFTA_FLOWTABLE_FLAGS:
|
||||
ft.Flags = FlowtableFlags(ad.Uint32())
|
||||
case NFTA_FLOWTABLE_HOOK:
|
||||
ad.Do(func(b []byte) error {
|
||||
ft.Hooknum, ft.Priority, ft.Devices, err = ftsHookFromMsg(b)
|
||||
return err
|
||||
})
|
||||
}
|
||||
}
|
||||
return &ft, nil
|
||||
}
|
||||
|
||||
func ftsHookFromMsg(b []byte) (*FlowtableHook, *FlowtablePriority, []string, error) {
|
||||
ad, err := netlink.NewAttributeDecoder(b)
|
||||
if err != nil {
|
||||
return nil, nil, nil, err
|
||||
}
|
||||
|
||||
ad.ByteOrder = binary.BigEndian
|
||||
|
||||
var hooknum FlowtableHook
|
||||
var prio FlowtablePriority
|
||||
var devices []string
|
||||
|
||||
for ad.Next() {
|
||||
switch ad.Type() {
|
||||
case NFTA_FLOWTABLE_HOOK_NUM:
|
||||
hooknum = FlowtableHook(ad.Uint32())
|
||||
case NFTA_FLOWTABLE_PRIORITY:
|
||||
prio = FlowtablePriority(ad.Uint32())
|
||||
case NFTA_FLOWTABLE_DEVS:
|
||||
ad.Do(func(b []byte) error {
|
||||
devices, err = devsFromMsg(b)
|
||||
return err
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
return &hooknum, &prio, devices, nil
|
||||
}
|
||||
|
||||
func devsFromMsg(b []byte) ([]string, error) {
|
||||
ad, err := netlink.NewAttributeDecoder(b)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
ad.ByteOrder = binary.BigEndian
|
||||
|
||||
devs := make([]string, 0)
|
||||
for ad.Next() {
|
||||
switch ad.Type() {
|
||||
case NFTA_DEVICE_NAME:
|
||||
devs = append(devs, ad.String())
|
||||
}
|
||||
}
|
||||
|
||||
return devs, nil
|
||||
}
|
228
nftables_test.go
228
nftables_test.go
|
@ -2884,6 +2884,234 @@ func TestSetElementsInterval(t *testing.T) {
|
|||
}
|
||||
}
|
||||
|
||||
func TestCreateListFlowtable(t *testing.T) {
|
||||
c, newNS := openSystemNFTConn(t)
|
||||
defer cleanupSystemNFTConn(t, newNS)
|
||||
c.FlushRuleset()
|
||||
defer c.FlushRuleset()
|
||||
|
||||
filter := &nftables.Table{
|
||||
Family: nftables.TableFamilyIPv4,
|
||||
Name: "filter",
|
||||
}
|
||||
|
||||
flowtable := &nftables.Flowtable{
|
||||
Table: filter,
|
||||
Name: "flowtable_test",
|
||||
}
|
||||
|
||||
c.AddTable(filter)
|
||||
c.AddFlowtable(flowtable)
|
||||
|
||||
if err := c.Flush(); err != nil {
|
||||
t.Fatalf("c.Flush() failed: %v", err)
|
||||
}
|
||||
|
||||
flowtables, err := c.ListFlowtables(filter)
|
||||
if err != nil {
|
||||
t.Fatalf("c.ListFlowtables() failed: %v", err)
|
||||
}
|
||||
|
||||
if got, want := len(flowtables), 1; got != want {
|
||||
t.Fatalf("flowtable entry length mismatch: got %d, want %d", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestCreateListFlowtableWithDevices(t *testing.T) {
|
||||
c, newNS := openSystemNFTConn(t)
|
||||
defer cleanupSystemNFTConn(t, newNS)
|
||||
c.FlushRuleset()
|
||||
defer c.FlushRuleset()
|
||||
|
||||
filter := &nftables.Table{
|
||||
Family: nftables.TableFamilyIPv4,
|
||||
Name: "filter",
|
||||
}
|
||||
|
||||
lo, err := net.InterfaceByName("lo")
|
||||
if err != nil {
|
||||
t.Fatalf("net.InterfaceByName() failed: %v", err)
|
||||
}
|
||||
|
||||
flowtable := &nftables.Flowtable{
|
||||
Table: filter,
|
||||
Name: "flowtable_test",
|
||||
Devices: []string{lo.Name},
|
||||
Hooknum: nftables.FlowtableHookIngress,
|
||||
Priority: nftables.FlowtablePriorityRef(5),
|
||||
}
|
||||
|
||||
c.AddTable(filter)
|
||||
c.AddFlowtable(flowtable)
|
||||
|
||||
if err := c.Flush(); err != nil {
|
||||
t.Fatalf("c.Flush() failed: %v", err)
|
||||
}
|
||||
|
||||
flowtables, err := c.ListFlowtables(filter)
|
||||
if err != nil {
|
||||
t.Fatalf("c.ListFlowtables() failed: %v", err)
|
||||
}
|
||||
|
||||
if got, want := len(flowtables), 1; got != want {
|
||||
t.Fatalf("flowtable entry length mismatch: got %d, want %d", got, want)
|
||||
}
|
||||
|
||||
sysFlowtable := flowtables[0]
|
||||
if got, want := sysFlowtable.Table, flowtable.Table; got != want {
|
||||
t.Errorf("flowtables table mismatch: got %v, want %v", got, want)
|
||||
}
|
||||
|
||||
if got, want := sysFlowtable.Name, flowtable.Name; got != want {
|
||||
t.Errorf("flowtables name mismatch: got %s, want %s", got, want)
|
||||
}
|
||||
|
||||
if len(sysFlowtable.Devices) != 1 {
|
||||
t.Fatalf("expected 1 device in flowtable, got %d", len(sysFlowtable.Devices))
|
||||
}
|
||||
|
||||
if got, want := sysFlowtable.Devices, flowtable.Devices; !reflect.DeepEqual(got, want) {
|
||||
t.Errorf("flowtables device mismatch: got %v, want %v", got, want)
|
||||
}
|
||||
|
||||
if got, want := *sysFlowtable.Hooknum, *flowtable.Hooknum; got != want {
|
||||
t.Errorf("flowtables hook mismatch: got %v, want %v", got, want)
|
||||
}
|
||||
|
||||
if got, want := *sysFlowtable.Priority, *flowtable.Priority; got != want {
|
||||
t.Errorf("flowtables prio mismatch: got %v, want %v", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestCreateDeleteFlowtable(t *testing.T) {
|
||||
c, newNS := openSystemNFTConn(t)
|
||||
defer cleanupSystemNFTConn(t, newNS)
|
||||
c.FlushRuleset()
|
||||
defer c.FlushRuleset()
|
||||
|
||||
filter := &nftables.Table{
|
||||
Family: nftables.TableFamilyIPv4,
|
||||
Name: "filter",
|
||||
}
|
||||
|
||||
flowtable := &nftables.Flowtable{
|
||||
Table: filter,
|
||||
Name: "flowtable_test",
|
||||
}
|
||||
|
||||
c.AddTable(filter)
|
||||
c.AddFlowtable(flowtable)
|
||||
flowtable.Name = "flowtable_test_to_delete"
|
||||
c.AddFlowtable(flowtable)
|
||||
|
||||
if err := c.Flush(); err != nil {
|
||||
t.Fatalf("c.Flush() failed: %v", err)
|
||||
}
|
||||
|
||||
flowtables, err := c.ListFlowtables(filter)
|
||||
if err != nil {
|
||||
t.Fatalf("c.ListFlowtables() failed: %v", err)
|
||||
}
|
||||
|
||||
if got, want := len(flowtables), 2; got != want {
|
||||
t.Fatalf("flowtable entry length mismatch: got %d, want %d", got, want)
|
||||
}
|
||||
|
||||
c.DelFlowtable(flowtable)
|
||||
if err := c.Flush(); err != nil {
|
||||
t.Fatalf("c.Flush() failed: %v", err)
|
||||
}
|
||||
|
||||
flowtables, err = c.ListFlowtables(filter)
|
||||
if err != nil {
|
||||
t.Fatalf("c.ListFlowtables() after deletion failed: %v", err)
|
||||
}
|
||||
|
||||
if got, want := len(flowtables), 1; got != want {
|
||||
t.Errorf("flowtable entry length mismatch: got %d, want %d", got, want)
|
||||
}
|
||||
|
||||
if got, removed := flowtables[0].Name, flowtable.Name; got == removed {
|
||||
t.Errorf("wrong flowtable entry deleted")
|
||||
}
|
||||
}
|
||||
|
||||
func TestOffload(t *testing.T) {
|
||||
c, newNS := openSystemNFTConn(t)
|
||||
defer cleanupSystemNFTConn(t, newNS)
|
||||
c.FlushRuleset()
|
||||
defer c.FlushRuleset()
|
||||
|
||||
filter := &nftables.Table{
|
||||
Family: nftables.TableFamilyIPv4,
|
||||
Name: "filter",
|
||||
}
|
||||
|
||||
accept := nftables.ChainPolicyAccept
|
||||
forward := &nftables.Chain{
|
||||
Name: "forward",
|
||||
Table: filter,
|
||||
Type: nftables.ChainTypeFilter,
|
||||
Priority: nftables.ChainPriorityFilter,
|
||||
Hooknum: nftables.ChainHookForward,
|
||||
Policy: &accept,
|
||||
}
|
||||
|
||||
flowtable := &nftables.Flowtable{
|
||||
Table: filter,
|
||||
Name: "flowtable_test",
|
||||
}
|
||||
|
||||
c.AddTable(filter)
|
||||
c.AddChain(forward)
|
||||
c.AddFlowtable(flowtable)
|
||||
|
||||
if err := c.Flush(); err != nil {
|
||||
t.Fatalf("c.Flush() failed: %v", err)
|
||||
}
|
||||
|
||||
rule := &nftables.Rule{
|
||||
Table: filter,
|
||||
Chain: forward,
|
||||
Exprs: []expr.Any{
|
||||
&expr.Payload{
|
||||
OperationType: expr.PayloadLoad,
|
||||
Base: expr.PayloadBaseNetworkHeader,
|
||||
Len: 1,
|
||||
Offset: 9,
|
||||
DestRegister: 1,
|
||||
},
|
||||
&expr.Cmp{
|
||||
Op: expr.CmpOpEq,
|
||||
Register: 1,
|
||||
Data: []byte{0x06, 0x00, 0x00, 0x00},
|
||||
},
|
||||
&expr.FlowOffload{
|
||||
Name: flowtable.Name,
|
||||
},
|
||||
},
|
||||
}
|
||||
c.AddRule(rule)
|
||||
|
||||
if err := c.Flush(); err != nil {
|
||||
t.Fatalf("c.Flush() offload failed: %v", err)
|
||||
}
|
||||
|
||||
rules, err := c.GetRule(filter, forward)
|
||||
if err != nil {
|
||||
t.Fatalf("c.GetRule() failed: %v", err)
|
||||
}
|
||||
|
||||
if got, want := len(rules), 1; got != want {
|
||||
t.Fatalf("rule count mismatch: got %d, want %d", got, want)
|
||||
}
|
||||
|
||||
sysRule := rules[0]
|
||||
if got, want := sysRule.Exprs, rule.Exprs; !reflect.DeepEqual(got, want) {
|
||||
t.Errorf("rule content mismatch: got %v, want %v", got, want)
|
||||
}
|
||||
}
|
||||
|
||||
func TestFlushChain(t *testing.T) {
|
||||
// Create a new network namespace to test these operations,
|
||||
// and tear down the namespace at test completion.
|
||||
|
|
Loading…
Reference in New Issue