Skip to content

Commit e760d20

Browse files
Phil Sutterummakynes
Phil Sutter
authored andcommitted
tests/py: Use libnftables instead of calling nft binary
This adds a simple nftables Python class in py/nftables.py which gives access to libnftables API via ctypes module. nft-test.py is extended to make use of the above class instead of calling nft binary. Since command line formatting had to be touched anyway, this patch also streamlines things a bit by introducing __str__ methods to classes Table and Chain and making extensive use of format strings instead of onerously adding all string parts together. Since the called commands don't see a shell anymore, all shell meta character escaping done in testcases is removed. The visible effects of this change are: * Four new warnings in ip/flowtable.t due to changing objref IDs (will be addressed later in a patch to libnftnl). * Reported command line in warning and error messages changed slightly for obvious reasons. * Reduction of a full test run's runtime by a factor of four. Status diff after running with 'time': < 83 test files, 77 files passed, 1724 unit tests, 0 error, 33 warning < 87.23user 696.13system 15:11.82elapsed 85%CPU (0avgtext+0avgdata 9604maxresident)k < 8inputs+36800outputs (0major+35171235minor)pagefaults 0swaps > 83 test files, 77 files passed, 1724 unit tests, 4 error, 33 warning > 6.80user 30.18system 3:45.86elapsed 16%CPU (0avgtext+0avgdata 14064maxresident)k > 0inputs+35808outputs (0major+2874minor)pagefaults 0swaps Signed-off-by: Phil Sutter <[email protected]> Signed-off-by: Pablo Neira Ayuso <[email protected]>
1 parent 4885331 commit e760d20

19 files changed

+339
-120
lines changed

py/.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -0,0 +1 @@
1+
*.pyc

py/nftables.py

+224
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,224 @@
1+
import json
2+
from ctypes import *
3+
import sys
4+
5+
class Nftables:
6+
"""A class representing libnftables interface"""
7+
8+
debug_flags = {
9+
"scanner": 0x1,
10+
"parser": 0x2,
11+
"eval": 0x4,
12+
"netlink": 0x8,
13+
"mnl": 0x10,
14+
"proto-ctx": 0x20,
15+
"segtree": 0x40,
16+
}
17+
18+
numeric_levels = {
19+
"none": 0,
20+
"addr": 1,
21+
"port": 2,
22+
"all": 3,
23+
}
24+
25+
def __init__(self, sofile="libnftables.so"):
26+
"""Instantiate a new Nftables class object.
27+
28+
Accepts a shared object file to open, by default standard search path
29+
is searched for a file named 'libnftables.so'.
30+
31+
After loading the library using ctypes module, a new nftables context
32+
is requested from the library and buffering of output and error streams
33+
is turned on.
34+
"""
35+
lib = cdll.LoadLibrary(sofile)
36+
37+
### API function definitions
38+
39+
self.nft_ctx_new = lib.nft_ctx_new
40+
self.nft_ctx_new.restype = c_void_p
41+
self.nft_ctx_new.argtypes = [c_int]
42+
43+
self.nft_ctx_output_get_handle = lib.nft_ctx_output_get_handle
44+
self.nft_ctx_output_get_handle.restype = c_bool
45+
self.nft_ctx_output_get_handle.argtypes = [c_void_p]
46+
47+
self.nft_ctx_output_set_handle = lib.nft_ctx_output_set_handle
48+
self.nft_ctx_output_set_handle.argtypes = [c_void_p, c_bool]
49+
50+
self.nft_ctx_output_get_numeric = lib.nft_ctx_output_get_numeric
51+
self.nft_ctx_output_get_numeric.restype = c_int
52+
self.nft_ctx_output_get_numeric.argtypes = [c_void_p]
53+
54+
self.nft_ctx_output_set_numeric = lib.nft_ctx_output_set_numeric
55+
self.nft_ctx_output_set_numeric.argtypes = [c_void_p, c_int]
56+
57+
self.nft_ctx_output_get_stateless = lib.nft_ctx_output_get_stateless
58+
self.nft_ctx_output_get_stateless.restype = c_bool
59+
self.nft_ctx_output_get_stateless.argtypes = [c_void_p]
60+
61+
self.nft_ctx_output_set_stateless = lib.nft_ctx_output_set_stateless
62+
self.nft_ctx_output_set_stateless.argtypes = [c_void_p, c_bool]
63+
64+
self.nft_ctx_output_get_debug = lib.nft_ctx_output_get_debug
65+
self.nft_ctx_output_get_debug.restype = c_int
66+
self.nft_ctx_output_get_debug.argtypes = [c_void_p]
67+
68+
self.nft_ctx_output_set_debug = lib.nft_ctx_output_set_debug
69+
self.nft_ctx_output_set_debug.argtypes = [c_void_p, c_int]
70+
71+
self.nft_ctx_buffer_output = lib.nft_ctx_buffer_output
72+
self.nft_ctx_buffer_output.restype = c_int
73+
self.nft_ctx_buffer_output.argtypes = [c_void_p]
74+
75+
self.nft_ctx_get_output_buffer = lib.nft_ctx_get_output_buffer
76+
self.nft_ctx_get_output_buffer.restype = c_char_p
77+
self.nft_ctx_get_output_buffer.argtypes = [c_void_p]
78+
79+
self.nft_ctx_buffer_error = lib.nft_ctx_buffer_error
80+
self.nft_ctx_buffer_error.restype = c_int
81+
self.nft_ctx_buffer_error.argtypes = [c_void_p]
82+
83+
self.nft_ctx_get_error_buffer = lib.nft_ctx_get_error_buffer
84+
self.nft_ctx_get_error_buffer.restype = c_char_p
85+
self.nft_ctx_get_error_buffer.argtypes = [c_void_p]
86+
87+
self.nft_run_cmd_from_buffer = lib.nft_run_cmd_from_buffer
88+
self.nft_run_cmd_from_buffer.restype = c_int
89+
self.nft_run_cmd_from_buffer.argtypes = [c_void_p, c_char_p, c_int]
90+
91+
self.nft_ctx_free = lib.nft_ctx_free
92+
lib.nft_ctx_free.argtypes = [c_void_p]
93+
94+
# initialize libnftables context
95+
self.__ctx = self.nft_ctx_new(0)
96+
self.nft_ctx_buffer_output(self.__ctx)
97+
self.nft_ctx_buffer_error(self.__ctx)
98+
99+
def get_handle_output(self):
100+
"""Get the current state of handle output.
101+
102+
Returns a boolean indicating whether handle output is active or not.
103+
"""
104+
return self.nft_ctx_output_get_handle(self.__ctx)
105+
106+
def set_handle_output(self, val):
107+
"""Enable or disable handle output.
108+
109+
Accepts a boolean turning handle output on or off.
110+
111+
Returns the previous value.
112+
"""
113+
old = self.get_handle_output()
114+
self.nft_ctx_output_set_handle(self.__ctx, val)
115+
return old
116+
117+
def get_numeric_output(self):
118+
"""Get the current state of numeric output.
119+
120+
Returns a boolean indicating whether boolean output is active or not.
121+
"""
122+
return self.nft_ctx_output_get_numeric(self.__ctx)
123+
124+
def set_numeric_output(self, val):
125+
"""Enable or disable numeric output.
126+
127+
Accepts a boolean turning numeric output on or off.
128+
129+
Returns the previous value.
130+
"""
131+
old = self.get_numeric_output()
132+
133+
if type(val) is str:
134+
val = self.numeric_levels[val]
135+
self.nft_ctx_output_set_numeric(self.__ctx, val)
136+
137+
return old
138+
139+
def get_stateless_output(self):
140+
"""Get the current state of stateless output.
141+
142+
Returns a boolean indicating whether stateless output is active or not.
143+
"""
144+
return self.nft_ctx_output_get_stateless(self.__ctx)
145+
146+
def set_stateless_output(self, val):
147+
"""Enable or Disable stateless output.
148+
149+
Accepts a boolean turning stateless output either on or off.
150+
151+
Returns the previous value.
152+
"""
153+
old = self.get_stateless_output()
154+
self.nft_ctx_output_set_stateless(self.__ctx, val)
155+
return old
156+
157+
def get_debug(self):
158+
"""Get currently active debug flags.
159+
160+
Returns a set of flag names. See set_debug() for details.
161+
"""
162+
val = self.nft_ctx_output_get_debug(self.__ctx)
163+
164+
names = []
165+
for n,v in self.debug_flags.items():
166+
if val & v:
167+
names.append(n)
168+
val &= ~v
169+
if val:
170+
names.append(val)
171+
172+
return names
173+
174+
def set_debug(self, values):
175+
"""Set debug output flags.
176+
177+
Accepts either a single flag or a set of flags. Each flag might be
178+
given either as string or integer value as shown in the following
179+
table:
180+
181+
Name | Value (hex)
182+
-----------------------
183+
scanner | 0x1
184+
parser | 0x2
185+
eval | 0x4
186+
netlink | 0x8
187+
mnl | 0x10
188+
proto-ctx | 0x20
189+
segtree | 0x40
190+
191+
Returns a set of previously active debug flags, as returned by
192+
get_debug() method.
193+
"""
194+
old = self.get_debug()
195+
196+
if type(values) in [str, int]:
197+
values = [values]
198+
199+
val = 0
200+
for v in values:
201+
if type(v) is str:
202+
v = self.debug_flags[v]
203+
val |= v
204+
205+
self.nft_ctx_output_set_debug(self.__ctx, val)
206+
207+
return old
208+
209+
def cmd(self, cmdline):
210+
"""Run a simple nftables command via libnftables.
211+
212+
Accepts a string containing an nftables command just like what one
213+
would enter into an interactive nftables (nft -i) session.
214+
215+
Returns a tuple (rc, output, error):
216+
rc -- return code as returned by nft_run_cmd_from_buffer() fuction
217+
output -- a string containing output written to stdout
218+
error -- a string containing output written to stderr
219+
"""
220+
rc = self.nft_run_cmd_from_buffer(self.__ctx, cmdline, len(cmdline))
221+
output = self.nft_ctx_get_output_buffer(self.__ctx)
222+
error = self.nft_ctx_get_error_buffer(self.__ctx)
223+
224+
return (rc, output, error)

tests/py/any/ct.t

+7-7
Original file line numberDiff line numberDiff line change
@@ -75,19 +75,19 @@ ct expiration != {33-55};ok;ct expiration != { 33s-55s}
7575

7676
ct helper "ftp";ok
7777
ct helper "12345678901234567";fail
78-
ct helper '""';fail
78+
ct helper "";fail
7979

8080
ct state . ct mark { new . 0x12345678};ok
8181
ct state . ct mark { new . 0x12345678, new . 0x34127856, established . 0x12785634};ok
8282
ct direction . ct mark { original . 0x12345678};ok
8383
ct state . ct mark vmap { new . 0x12345678 : drop};ok
8484

85-
ct original bytes \> 100000;ok;ct original bytes > 100000
86-
ct reply packets \< 100;ok;ct reply packets < 100
87-
ct bytes \> 100000;ok;ct bytes > 100000
85+
ct original bytes > 100000;ok
86+
ct reply packets < 100;ok
87+
ct bytes > 100000;ok
8888

89-
ct avgpkt \> 200;ok;ct avgpkt > 200
90-
ct original avgpkt \< 500;ok;ct original avgpkt < 500
89+
ct avgpkt > 200;ok
90+
ct original avgpkt < 500;ok
9191

9292
# bogus direction
9393
ct both bytes gt 1;fail
@@ -107,7 +107,7 @@ ct mark original;fail
107107

108108
ct event set new;ok
109109
ct event set new or related or destroy or foobar;fail
110-
ct event set 'new | related | destroy | label';ok;ct event set new,related,destroy,label
110+
ct event set new | related | destroy | label;ok;ct event set new,related,destroy,label
111111
ct event set new,related,destroy,label;ok
112112
ct event set new,destroy;ok
113113
ct event set 1;ok;ct event set new

tests/py/any/ct.t.payload

+6-6
Original file line numberDiff line numberDiff line change
@@ -343,31 +343,31 @@ ip test-ip4 output
343343
[ lookup reg 1 set __map%d dreg 1 ]
344344
[ ct set mark with reg 1 ]
345345

346-
# ct original bytes \> 100000
346+
# ct original bytes > 100000
347347
ip test-ip4 output
348348
[ ct load bytes => reg 1 , dir original ]
349349
[ byteorder reg 1 = hton(reg 1, 8, 8) ]
350350
[ cmp gt reg 1 0x00000000 0xa0860100 ]
351351

352-
# ct reply packets \< 100
352+
# ct reply packets < 100
353353
ip test-ip4 output
354354
[ ct load packets => reg 1 , dir reply ]
355355
[ byteorder reg 1 = hton(reg 1, 8, 8) ]
356356
[ cmp lt reg 1 0x00000000 0x64000000 ]
357357

358-
# ct bytes \> 100000
358+
# ct bytes > 100000
359359
ip test-ip4 output
360360
[ ct load bytes => reg 1 ]
361361
[ byteorder reg 1 = hton(reg 1, 8, 8) ]
362362
[ cmp gt reg 1 0x00000000 0xa0860100 ]
363363

364-
# ct avgpkt \> 200
364+
# ct avgpkt > 200
365365
ip test-ip4 output
366366
[ ct load avgpkt => reg 1 ]
367367
[ byteorder reg 1 = hton(reg 1, 8, 8) ]
368368
[ cmp gt reg 1 0x00000000 0xc8000000 ]
369369

370-
# ct original avgpkt \< 500
370+
# ct original avgpkt < 500
371371
ip test-ip4 output
372372
[ ct load avgpkt => reg 1 , dir original ]
373373
[ byteorder reg 1 = hton(reg 1, 8, 8) ]
@@ -396,7 +396,7 @@ ip test-ip4 output
396396
[ immediate reg 1 0x00000001 ]
397397
[ ct set event with reg 1 ]
398398

399-
# ct event set 'new | related | destroy | label'
399+
# ct event set new | related | destroy | label
400400
ip test-ip4 output
401401
[ immediate reg 1 0x00000407 ]
402402
[ ct set event with reg 1 ]

tests/py/any/log.t

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ log prefix aaaaa-aaaaaa group 2 snaplen 33;ok;log prefix "aaaaa-aaaaaa" group 2
2424
# The correct rule is log group 2 queue-threshold 2
2525
log group 2 queue-threshold 2;ok
2626
log group 2 snaplen 33;ok
27-
log group 2 prefix \"nft-test: \";ok;log prefix "nft-test: " group 2
27+
log group 2 prefix "nft-test: ";ok;log prefix "nft-test: " group 2
2828

2929
log flags all;ok
3030
log level debug flags ip options flags skuid;ok

tests/py/any/log.t.payload

+1-1
Original file line numberDiff line numberDiff line change
@@ -46,7 +46,7 @@ ip test-ip4 output
4646
ip test-ip4 output
4747
[ log group 2 snaplen 33 qthreshold 0 ]
4848

49-
# log group 2 prefix \"nft-test: \"
49+
# log group 2 prefix "nft-test: "
5050
ip test-ip4 output
5151
[ log prefix nft-test: group 2 snaplen 0 qthreshold 0 ]
5252

tests/py/any/meta.t

+2-2
Original file line numberDiff line numberDiff line change
@@ -70,7 +70,7 @@ meta iifname {"dummy0", "lo"};ok;iifname {"dummy0", "lo"}
7070
meta iifname != {"dummy0", "lo"};ok;iifname != {"dummy0", "lo"}
7171
meta iifname "dummy*";ok;iifname "dummy*"
7272
meta iifname "dummy\*";ok;iifname "dummy\*"
73-
meta iifname '""';fail
73+
meta iifname "";fail
7474

7575
meta iiftype {ether, ppp, ipip, ipip6, loopback, sit, ipgre};ok;iiftype {ether, ppp, ipip, ipip6, loopback, sit, ipgre}
7676
meta iiftype != {ether, ppp, ipip, ipip6, loopback, sit, ipgre};ok;iiftype != {ether, ppp, ipip, ipip6, loopback, sit, ipgre}
@@ -89,7 +89,7 @@ meta oifname != "dummy0";ok;oifname != "dummy0"
8989
meta oifname { "dummy0", "lo"};ok;oifname { "dummy0", "lo"}
9090
meta oifname "dummy*";ok;oifname "dummy*"
9191
meta oifname "dummy\*";ok;oifname "dummy\*"
92-
meta oifname '""';fail
92+
meta oifname "";fail
9393

9494
meta oiftype {ether, ppp, ipip, ipip6, loopback, sit, ipgre};ok;oiftype {ether, ppp, ipip, ipip6, loopback, sit, ipgre}
9595
meta oiftype != {ether, ppp, ipip, ipip6, loopback, sit, ipgre};ok;oiftype != {ether, ppp, ipip, ipip6, loopback, sit, ipgre}

tests/py/arp/arp.t

+1-1
Original file line numberDiff line numberDiff line change
@@ -55,4 +55,4 @@ arp operation != inreply;ok
5555
arp operation != nak;ok
5656
arp operation != reply;ok
5757

58-
meta iifname \"invalid\" arp ptype 0x0800 arp htype 1 arp hlen 6 arp plen 4 @nh,192,32 0xc0a88f10 @nh,144,48 set 0x112233445566;ok;iifname "invalid" arp htype 1 arp ptype ip arp hlen 6 arp plen 4 @nh,192,32 3232272144 @nh,144,48 set 18838586676582
58+
meta iifname "invalid" arp ptype 0x0800 arp htype 1 arp hlen 6 arp plen 4 @nh,192,32 0xc0a88f10 @nh,144,48 set 0x112233445566;ok;iifname "invalid" arp htype 1 arp ptype ip arp hlen 6 arp plen 4 @nh,192,32 3232272144 @nh,144,48 set 18838586676582

tests/py/arp/arp.t.payload

+1-1
Original file line numberDiff line numberDiff line change
@@ -268,7 +268,7 @@ arp test-arp input
268268
[ payload load 2b @ network header + 6 => reg 1 ]
269269
[ cmp neq reg 1 0x00000200 ]
270270

271-
# meta iifname \"invalid\" arp ptype 0x0800 arp htype 1 arp hlen 6 arp plen 4 @nh,192,32 0xc0a88f10 @nh,144,48 set 0x112233445566
271+
# meta iifname "invalid" arp ptype 0x0800 arp htype 1 arp hlen 6 arp plen 4 @nh,192,32 0xc0a88f10 @nh,144,48 set 0x112233445566
272272
arp test-arp input
273273
[ meta load iifname => reg 1 ]
274274
[ cmp eq reg 1 0x61766e69 0x0064696c 0x00000000 0x00000000 ]

tests/py/arp/arp.t.payload.netdev

+1-1
Original file line numberDiff line numberDiff line change
@@ -358,7 +358,7 @@ netdev test-netdev ingress
358358
[ payload load 2b @ network header + 6 => reg 1 ]
359359
[ cmp neq reg 1 0x00000200 ]
360360

361-
# meta iifname \"invalid\" arp ptype 0x0800 arp htype 1 arp hlen 6 arp plen 4 @nh,192,32 0xc0a88f10 @nh,144,48 set 0x112233445566
361+
# meta iifname "invalid" arp ptype 0x0800 arp htype 1 arp hlen 6 arp plen 4 @nh,192,32 0xc0a88f10 @nh,144,48 set 0x112233445566
362362
netdev test-netdev ingress
363363
[ meta load iifname => reg 1 ]
364364
[ cmp eq reg 1 0x61766e69 0x0064696c 0x00000000 0x00000000 ]

tests/py/inet/tcp.t

+1-1
Original file line numberDiff line numberDiff line change
@@ -76,7 +76,7 @@ tcp flags { fin, syn, rst, psh, ack, urg, ecn, cwr} drop;ok
7676
tcp flags != { fin, urg, ecn, cwr} drop;ok
7777
tcp flags cwr;ok
7878
tcp flags != cwr;ok
79-
tcp 'flags & (syn|fin) == (syn|fin)';ok;tcp flags & (fin | syn) == fin | syn
79+
tcp flags & (syn|fin) == (syn|fin);ok;tcp flags & (fin | syn) == fin | syn
8080

8181
tcp window 22222;ok
8282
tcp window 22;ok

tests/py/inet/tcp.t.payload

+1-1
Original file line numberDiff line numberDiff line change
@@ -421,7 +421,7 @@ inet test-inet input
421421
[ payload load 1b @ transport header + 13 => reg 1 ]
422422
[ cmp neq reg 1 0x00000080 ]
423423

424-
# tcp 'flags & (syn|fin) == (syn|fin)'
424+
# tcp flags & (syn|fin) == (syn|fin)
425425
inet test-inet input
426426
[ meta load l4proto => reg 1 ]
427427
[ cmp eq reg 1 0x00000006 ]

tests/py/ip/ip.t

+3-3
Original file line numberDiff line numberDiff line change
@@ -113,10 +113,10 @@ ip daddr 192.168.0.1;ok
113113
ip daddr 192.168.0.1 drop;ok
114114
ip daddr 192.168.0.2;ok
115115

116-
ip saddr \& 0xff == 1;ok;ip saddr & 0.0.0.255 == 0.0.0.1
117-
ip saddr \& 0.0.0.255 \< 0.0.0.127;ok;ip saddr & 0.0.0.255 < 0.0.0.127
116+
ip saddr & 0xff == 1;ok;ip saddr & 0.0.0.255 == 0.0.0.1
117+
ip saddr & 0.0.0.255 < 0.0.0.127;ok
118118

119-
ip saddr \& 0xffff0000 == 0xffff0000;ok;ip saddr 255.255.0.0/16
119+
ip saddr & 0xffff0000 == 0xffff0000;ok;ip saddr 255.255.0.0/16
120120

121121
ip version 4 ip hdrlength 5;ok
122122
ip hdrlength 0;ok

0 commit comments

Comments
 (0)