mt_racket 1.0.12a
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/README +76 -0
- data/examples/arp-send +24 -0
- data/examples/arp-send2 +30 -0
- data/examples/cdp +39 -0
- data/examples/cdp-spew +52 -0
- data/examples/dhcp +42 -0
- data/examples/dhcp-spew +48 -0
- data/examples/dns +38 -0
- data/examples/egp +30 -0
- data/examples/hsrp +43 -0
- data/examples/hsrp_takeover +69 -0
- data/examples/icmp-recv +34 -0
- data/examples/icmp-spew +50 -0
- data/examples/icmpv6 +84 -0
- data/examples/icmpv6-spew +50 -0
- data/examples/igmpv1 +27 -0
- data/examples/igmpv2 +27 -0
- data/examples/igrp-send +25 -0
- data/examples/ipv6 +35 -0
- data/examples/nat-pimp +32 -0
- data/examples/ntp +38 -0
- data/examples/ntp2 +42 -0
- data/examples/sctp +32 -0
- data/examples/stp-send +21 -0
- data/examples/synflood +147 -0
- data/examples/tcp +43 -0
- data/examples/tcp2udp +65 -0
- data/examples/udp +46 -0
- data/examples/vrrp +34 -0
- data/examples/vtp +28 -0
- data/lib/racket/l2/eightotwodotthree.rb +48 -0
- data/lib/racket/l2/ethernet.rb +62 -0
- data/lib/racket/l2/llc.rb +50 -0
- data/lib/racket/l2/misc.rb +67 -0
- data/lib/racket/l2/snap.rb +40 -0
- data/lib/racket/l2/vlan.rb +61 -0
- data/lib/racket/l2/vtp.rb +124 -0
- data/lib/racket/l2.rb +30 -0
- data/lib/racket/l3/arp.rb +63 -0
- data/lib/racket/l3/cdp.rb +85 -0
- data/lib/racket/l3/egp.rb +53 -0
- data/lib/racket/l3/ipv4.rb +132 -0
- data/lib/racket/l3/ipv6.rb +66 -0
- data/lib/racket/l3/misc.rb +159 -0
- data/lib/racket/l3/stp.rb +81 -0
- data/lib/racket/l3.rb +30 -0
- data/lib/racket/l4/gre.rb +65 -0
- data/lib/racket/l4/icmp.rb +295 -0
- data/lib/racket/l4/icmpv6.rb +446 -0
- data/lib/racket/l4/igmpv1.rb +79 -0
- data/lib/racket/l4/igmpv2.rb +76 -0
- data/lib/racket/l4/igrp.rb +138 -0
- data/lib/racket/l4/misc.rb +35 -0
- data/lib/racket/l4/sctp.rb +163 -0
- data/lib/racket/l4/tcp.rb +152 -0
- data/lib/racket/l4/udp.rb +81 -0
- data/lib/racket/l4/vrrp.rb +95 -0
- data/lib/racket/l4.rb +30 -0
- data/lib/racket/l5/bootp.rb +106 -0
- data/lib/racket/l5/dns.rb +110 -0
- data/lib/racket/l5/hsrp.rb +73 -0
- data/lib/racket/l5/misc.rb +35 -0
- data/lib/racket/l5/nat-pmp.rb +41 -0
- data/lib/racket/l5/ntp.rb +59 -0
- data/lib/racket/l5.rb +30 -0
- data/lib/racket/misc/lv.rb +108 -0
- data/lib/racket/misc/misc.rb +62 -0
- data/lib/racket/misc/orderedhash.rb +63 -0
- data/lib/racket/misc/raw.rb +35 -0
- data/lib/racket/misc/tlv.rb +103 -0
- data/lib/racket/misc/vt.rb +114 -0
- data/lib/racket/misc.rb +30 -0
- data/lib/racket/racket.rb +166 -0
- data/lib/racket/racketpart.rb +66 -0
- data/lib/racket.rb +4 -0
- data/test/l2/ts_ethernet.rb +22 -0
- data/test/l2/ts_misc.rb +23 -0
- data/test/l2/ts_vlan.rb +15 -0
- data/test/l3/ts_ipv4.rb +44 -0
- data/test/l3/ts_ipv6.rb +26 -0
- data/test/l3/ts_misc.rb +31 -0
- data/test/l4/ts_icmp.rb +38 -0
- data/test/l4/ts_tcp.rb +55 -0
- data/test/l4/ts_udp.rb +40 -0
- data/test/misc/ts_lv.rb +59 -0
- data/test/misc/ts_orderedhash.rb +33 -0
- data/test/misc/ts_tlv.rb +47 -0
- data/test/misc/ts_vt.rb +56 -0
- data/test/ts_all.rb +14 -0
- metadata +153 -0
@@ -0,0 +1,35 @@
|
|
1
|
+
# $Id:$
|
2
|
+
#
|
3
|
+
# Copyright (c) 2008, Jon Hart
|
4
|
+
# All rights reserved.
|
5
|
+
#
|
6
|
+
# Redistribution and use in source and binary forms, with or without
|
7
|
+
# modification, are permitted provided that the following conditions are met:
|
8
|
+
# * Redistributions of source code must retain the above copyright
|
9
|
+
# notice, this list of conditions and the following disclaimer.
|
10
|
+
# * Redistributions in binary form must reproduce the above copyright
|
11
|
+
# notice, this list of conditions and the following disclaimer in the
|
12
|
+
# documentation and/or other materials provided with the distribution.
|
13
|
+
# * Neither the name of the <organization> nor the
|
14
|
+
# names of its contributors may be used to endorse or promote products
|
15
|
+
# derived from this software without specific prior written permission.
|
16
|
+
#
|
17
|
+
# THIS SOFTWARE IS PROVIDED BY Jon Hart ``AS IS'' AND ANY
|
18
|
+
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
19
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
20
|
+
# DISCLAIMED. IN NO EVENT SHALL Jon Hart BE LIABLE FOR ANY
|
21
|
+
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
22
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
23
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
24
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
25
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
26
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
27
|
+
#
|
28
|
+
module Racket
|
29
|
+
# A raw part of a packet that only has a payload
|
30
|
+
class Raw < RacketPart
|
31
|
+
rest :payload
|
32
|
+
end
|
33
|
+
end
|
34
|
+
# vim: set ts=2 et sw=2:
|
35
|
+
|
@@ -0,0 +1,103 @@
|
|
1
|
+
# $Id: tlv.rb 14 2008-03-02 05:42:30Z warchild $
|
2
|
+
#
|
3
|
+
# Copyright (c) 2008, Jon Hart
|
4
|
+
# All rights reserved.
|
5
|
+
#
|
6
|
+
# Redistribution and use in source and binary forms, with or without
|
7
|
+
# modification, are permitted provided that the following conditions are met:
|
8
|
+
# * Redistributions of source code must retain the above copyright
|
9
|
+
# notice, this list of conditions and the following disclaimer.
|
10
|
+
# * Redistributions in binary form must reproduce the above copyright
|
11
|
+
# notice, this list of conditions and the following disclaimer in the
|
12
|
+
# documentation and/or other materials provided with the distribution.
|
13
|
+
# * Neither the name of the <organization> nor the
|
14
|
+
# names of its contributors may be used to endorse or promote products
|
15
|
+
# derived from this software without specific prior written permission.
|
16
|
+
#
|
17
|
+
# THIS SOFTWARE IS PROVIDED BY Jon Hart ``AS IS'' AND ANY
|
18
|
+
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
19
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
20
|
+
# DISCLAIMED. IN NO EVENT SHALL Jon Hart BE LIABLE FOR ANY
|
21
|
+
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
22
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
23
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
24
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
25
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
26
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
27
|
+
#
|
28
|
+
module Racket
|
29
|
+
module Misc
|
30
|
+
# Simple class for your average type, length, value datastructure.
|
31
|
+
# Everything after the TLV is stuff into +rest+
|
32
|
+
class TLV
|
33
|
+
attr_accessor :type, :length, :value, :rest, :lbytes, :tlinclude
|
34
|
+
|
35
|
+
# Create a new TLV which requires +ts+ bytes for the type field
|
36
|
+
# and +ls+ bytes for the length field, where (optionally) the value
|
37
|
+
# in +length+ is a multiple of +lbytes+ and (optionally) whether or not
|
38
|
+
# the length field indicates the total length of the TLV of just that of
|
39
|
+
# the value
|
40
|
+
def initialize(ts, ls, lbytes=1, tlinclude=false)
|
41
|
+
@ts = ts
|
42
|
+
@ls = ls
|
43
|
+
@lbytes = lbytes
|
44
|
+
@tlinclude = tlinclude
|
45
|
+
end
|
46
|
+
|
47
|
+
# Given +data+, return the type, length, value and rest
|
48
|
+
# values as dictated by this instance.
|
49
|
+
def decode(data)
|
50
|
+
s = "#{punpack_string(@ts)}#{punpack_string(@ls)}"
|
51
|
+
type, length, tmp = data.unpack("#{s}a*")
|
52
|
+
if (type.nil? or length.nil?)
|
53
|
+
nil
|
54
|
+
else
|
55
|
+
elength = (length * lbytes) - (@tlinclude ? (@ls + @ts) : 0)
|
56
|
+
value, rest = tmp.unpack("a#{elength}a*")
|
57
|
+
if (value.empty? and length > 0)
|
58
|
+
nil
|
59
|
+
else
|
60
|
+
[type, length, value, rest]
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
def decode!(data)
|
66
|
+
@type, @length, @value, @rest = self.decode(data)
|
67
|
+
end
|
68
|
+
|
69
|
+
# Return a string suitable for use elswhere.
|
70
|
+
def encode
|
71
|
+
s = "#{punpack_string(@ts)}#{punpack_string(@ls)}"
|
72
|
+
[@type, @length, @value].pack("#{s}a*")
|
73
|
+
end
|
74
|
+
|
75
|
+
def to_s
|
76
|
+
encode
|
77
|
+
end
|
78
|
+
|
79
|
+
def to_str
|
80
|
+
encode
|
81
|
+
end
|
82
|
+
|
83
|
+
private
|
84
|
+
# XXX: make this handle arbitrarily sized fields
|
85
|
+
def punpack_string(size)
|
86
|
+
s = ""
|
87
|
+
case size
|
88
|
+
when 1
|
89
|
+
s << "C"
|
90
|
+
when 2
|
91
|
+
s << "n"
|
92
|
+
when 4
|
93
|
+
s << "N"
|
94
|
+
else
|
95
|
+
raise ArgumentError, "Size #{s} is not a supported conversion size"
|
96
|
+
end
|
97
|
+
s
|
98
|
+
end
|
99
|
+
|
100
|
+
end
|
101
|
+
end
|
102
|
+
end
|
103
|
+
# vim: set ts=2 et sw=2:
|
@@ -0,0 +1,114 @@
|
|
1
|
+
# $Id: vt.rb 14 2008-03-02 05:42:30Z warchild $
|
2
|
+
#
|
3
|
+
# Copyright (c) 2008, Jon Hart
|
4
|
+
# All rights reserved.
|
5
|
+
#
|
6
|
+
# Redistribution and use in source and binary forms, with or without
|
7
|
+
# modification, are permitted provided that the following conditions are met:
|
8
|
+
# * Redistributions of source code must retain the above copyright
|
9
|
+
# notice, this list of conditions and the following disclaimer.
|
10
|
+
# * Redistributions in binary form must reproduce the above copyright
|
11
|
+
# notice, this list of conditions and the following disclaimer in the
|
12
|
+
# documentation and/or other materials provided with the distribution.
|
13
|
+
# * Neither the name of the <organization> nor the
|
14
|
+
# names of its contributors may be used to endorse or promote products
|
15
|
+
# derived from this software without specific prior written permission.
|
16
|
+
#
|
17
|
+
# THIS SOFTWARE IS PROVIDED BY Jon Hart ``AS IS'' AND ANY
|
18
|
+
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
19
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
20
|
+
# DISCLAIMED. IN NO EVENT SHALL Jon Hart BE LIABLE FOR ANY
|
21
|
+
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
22
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
23
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
24
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
25
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
26
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
27
|
+
#
|
28
|
+
module Racket
|
29
|
+
module Misc
|
30
|
+
# Simple class to represent a datastructure that is made up of a
|
31
|
+
# null terminted string followed by an arbitrary number of
|
32
|
+
# arbitrarily sized values, followed by a "rest" field.
|
33
|
+
class VT
|
34
|
+
# the value for this VT object
|
35
|
+
attr_accessor :value
|
36
|
+
# the array of types for this VT object
|
37
|
+
attr_accessor :types
|
38
|
+
# everything else
|
39
|
+
attr_accessor :rest
|
40
|
+
|
41
|
+
# Create a new VT which consists of a null terminated string
|
42
|
+
# followed by some number of arbitrarily sized values, as
|
43
|
+
# specified by +args+
|
44
|
+
def initialize(*args)
|
45
|
+
@lengths = args
|
46
|
+
@types = []
|
47
|
+
@value = ""
|
48
|
+
end
|
49
|
+
|
50
|
+
# Given +data+, return the value and an array
|
51
|
+
# of the types as dictated by this instance
|
52
|
+
def decode(data)
|
53
|
+
null = data.index("\x0")
|
54
|
+
value = data.unpack("a#{null}")[0]
|
55
|
+
data = data.slice(null+1, data.length)
|
56
|
+
|
57
|
+
n = 0
|
58
|
+
types = []
|
59
|
+
@lengths.each do |l|
|
60
|
+
types[n] = data.unpack("#{punpack_string(l)}")[0]
|
61
|
+
data = data.slice(l, data.length)
|
62
|
+
n += 1
|
63
|
+
end
|
64
|
+
|
65
|
+
[value, types, data]
|
66
|
+
end
|
67
|
+
|
68
|
+
# Given +data+, set the +value+ and +types+ array
|
69
|
+
# accordingly
|
70
|
+
def decode!(data)
|
71
|
+
@value, @types, @rest = self.decode(data)
|
72
|
+
end
|
73
|
+
|
74
|
+
# Return a string suitable for use elsewhere
|
75
|
+
def encode
|
76
|
+
s = "#{@value}\000"
|
77
|
+
|
78
|
+
n = 0
|
79
|
+
@lengths.each do |l|
|
80
|
+
s << [@types[n]].pack("#{punpack_string(l)}")
|
81
|
+
n += 1
|
82
|
+
end
|
83
|
+
s
|
84
|
+
end
|
85
|
+
|
86
|
+
def to_s
|
87
|
+
encode
|
88
|
+
end
|
89
|
+
|
90
|
+
def to_str
|
91
|
+
encode
|
92
|
+
end
|
93
|
+
|
94
|
+
private
|
95
|
+
# XXX: make this handle arbitrarily sized fields
|
96
|
+
def punpack_string(size)
|
97
|
+
s = ""
|
98
|
+
case size
|
99
|
+
when 1
|
100
|
+
s << "C"
|
101
|
+
when 2
|
102
|
+
s << "n"
|
103
|
+
when 4
|
104
|
+
s << "N"
|
105
|
+
else
|
106
|
+
raise ArgumentError, "Size #{s} not supported"
|
107
|
+
end
|
108
|
+
s
|
109
|
+
end
|
110
|
+
|
111
|
+
end
|
112
|
+
end
|
113
|
+
end
|
114
|
+
# vim: set ts=2 et sw=2:
|
data/lib/racket/misc.rb
ADDED
@@ -0,0 +1,30 @@
|
|
1
|
+
# $Id$
|
2
|
+
#
|
3
|
+
# Copyright (c) 2008, Jon Hart
|
4
|
+
# All rights reserved.
|
5
|
+
#
|
6
|
+
# Redistribution and use in source and binary forms, with or without
|
7
|
+
# modification, are permitted provided that the following conditions are met:
|
8
|
+
# * Redistributions of source code must retain the above copyright
|
9
|
+
# notice, this list of conditions and the following disclaimer.
|
10
|
+
# * Redistributions in binary form must reproduce the above copyright
|
11
|
+
# notice, this list of conditions and the following disclaimer in the
|
12
|
+
# documentation and/or other materials provided with the distribution.
|
13
|
+
# * Neither the name of the <organization> nor the
|
14
|
+
# names of its contributors may be used to endorse or promote products
|
15
|
+
# derived from this software without specific prior written permission.
|
16
|
+
#
|
17
|
+
# THIS SOFTWARE IS PROVIDED BY Jon Hart ``AS IS'' AND ANY
|
18
|
+
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
19
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
20
|
+
# DISCLAIMED. IN NO EVENT SHALL Jon Hart BE LIABLE FOR ANY
|
21
|
+
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
22
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
23
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
24
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
25
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
26
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
27
|
+
|
28
|
+
Dir.glob(File.join(File.dirname(__FILE__), 'misc/*.rb')).each { |f| require f }
|
29
|
+
|
30
|
+
# vim: set ts=2 et sw=2:
|
@@ -0,0 +1,166 @@
|
|
1
|
+
# $Id: racket.rb 14 2008-03-02 05:42:30Z warchild $
|
2
|
+
#
|
3
|
+
# Copyright (c) 2008, Jon Hart
|
4
|
+
# All rights reserved.
|
5
|
+
#
|
6
|
+
# Redistribution and use in source and binary forms, with or without
|
7
|
+
# modification, are permitted provided that the following conditions are met:
|
8
|
+
# * Redistributions of source code must retain the above copyright
|
9
|
+
# notice, this list of conditions and the following disclaimer.
|
10
|
+
# * Redistributions in binary form must reproduce the above copyright
|
11
|
+
# notice, this list of conditions and the following disclaimer in the
|
12
|
+
# documentation and/or other materials provided with the distribution.
|
13
|
+
# * Neither the name of the <organization> nor the
|
14
|
+
# names of its contributors may be used to endorse or promote products
|
15
|
+
# derived from this software without specific prior written permission.
|
16
|
+
#
|
17
|
+
# THIS SOFTWARE IS PROVIDED BY Jon Hart ``AS IS'' AND ANY
|
18
|
+
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
19
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
20
|
+
# DISCLAIMED. IN NO EVENT SHALL Jon Hart BE LIABLE FOR ANY
|
21
|
+
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
22
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
23
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
24
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
25
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
26
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
27
|
+
#
|
28
|
+
|
29
|
+
require 'socket'
|
30
|
+
require 'racket/racketpart'
|
31
|
+
require 'racket/misc'
|
32
|
+
require 'racket/l2'
|
33
|
+
require 'racket/l3'
|
34
|
+
require 'racket/l4'
|
35
|
+
require 'racket/l5'
|
36
|
+
|
37
|
+
module Racket
|
38
|
+
class Racket
|
39
|
+
|
40
|
+
attr_accessor :iface, :mtu, :timeout
|
41
|
+
attr_accessor :layers, :payload
|
42
|
+
|
43
|
+
@@loaded_pcaprub = false
|
44
|
+
begin
|
45
|
+
require 'pcaprub'
|
46
|
+
@@loaded_pcaprub = true
|
47
|
+
rescue ::LoadError
|
48
|
+
end
|
49
|
+
|
50
|
+
|
51
|
+
def initialize(payload="")
|
52
|
+
@layers = []
|
53
|
+
@mtu = 1500
|
54
|
+
@timeout = 10
|
55
|
+
@payload = payload
|
56
|
+
1.upto(7) do |l|
|
57
|
+
self.class.send(:define_method, "layer#{l}", lambda { @layers[l] })
|
58
|
+
self.class.send(:define_method, "l#{l}", lambda { @layers[l] })
|
59
|
+
self.class.send(:define_method, "layer#{l}=", lambda { |x| @layers[l] = x; })
|
60
|
+
self.class.send(:define_method, "l#{l}=", lambda { |x| @layers[l] = x; })
|
61
|
+
end
|
62
|
+
end
|
63
|
+
|
64
|
+
# Assemble all the pieces of this Racket as a string, ready for sending.
|
65
|
+
def pack
|
66
|
+
last_payload = ""
|
67
|
+
orig_payload = ""
|
68
|
+
@layers.compact.reverse.each do |l|
|
69
|
+
# save the original payload
|
70
|
+
orig_payload = l.payload
|
71
|
+
# tack on the last payload in
|
72
|
+
# case fix needs it...
|
73
|
+
l.payload += last_payload
|
74
|
+
if (l.autofix?)
|
75
|
+
l.fix!
|
76
|
+
end
|
77
|
+
|
78
|
+
if (l.payload == orig_payload + last_payload)
|
79
|
+
# payload was not modified by fix, so reset it to what
|
80
|
+
# it used to be
|
81
|
+
l.payload = orig_payload
|
82
|
+
else
|
83
|
+
# payload was modified by fix. chop off what we added.
|
84
|
+
# XXX: this assumes that what we added is still at the end.
|
85
|
+
# XXX: this is not always true
|
86
|
+
l.payload = l.payload.slice(0, l.payload.length - last_payload.length)
|
87
|
+
end
|
88
|
+
|
89
|
+
# save this layer for the next guy
|
90
|
+
last_payload += l
|
91
|
+
|
92
|
+
end
|
93
|
+
|
94
|
+
payload = ""
|
95
|
+
@layers.compact.each do |l|
|
96
|
+
payload += l
|
97
|
+
end
|
98
|
+
payload
|
99
|
+
end
|
100
|
+
|
101
|
+
# return a pretty interpretation of this packet
|
102
|
+
def pretty
|
103
|
+
s = ""
|
104
|
+
@layers.compact.each do |l|
|
105
|
+
s << "#{l.class}: "
|
106
|
+
s << l.pretty
|
107
|
+
s << "\n"
|
108
|
+
end
|
109
|
+
s
|
110
|
+
end
|
111
|
+
|
112
|
+
|
113
|
+
# Attempt to figure out which of send2 or send3 needs to be called.
|
114
|
+
def sendpacket
|
115
|
+
if (@layers[2])
|
116
|
+
send2
|
117
|
+
else
|
118
|
+
send3
|
119
|
+
end
|
120
|
+
end
|
121
|
+
|
122
|
+
# Write raw layer2 frames
|
123
|
+
def send2
|
124
|
+
if(not @@loaded_pcaprub)
|
125
|
+
raise RuntimeError, "Could not initialize the pcaprub library (You need the pcaprub gem)"
|
126
|
+
end
|
127
|
+
begin
|
128
|
+
|
129
|
+
p = Pcap::open_live(@iface, @mtu, false, @timeout)
|
130
|
+
rescue Exception => e
|
131
|
+
puts "Pcap: can't open device '#{@iface}' (#{e})"
|
132
|
+
return
|
133
|
+
end
|
134
|
+
|
135
|
+
begin
|
136
|
+
b = p.inject(pack)
|
137
|
+
#p.pcap_close
|
138
|
+
return b
|
139
|
+
rescue Exception => e
|
140
|
+
puts "Pcap: error while sending packet on '#{@iface}' (#{e})"
|
141
|
+
end
|
142
|
+
end
|
143
|
+
|
144
|
+
# Write raw layer3 frames
|
145
|
+
def send3
|
146
|
+
begin
|
147
|
+
s = Socket.open(Socket::PF_INET, Socket::SOCK_RAW, Socket::IPPROTO_RAW)
|
148
|
+
|
149
|
+
if (Socket.const_defined?('SOL_IP'))
|
150
|
+
s.setsockopt(Socket::SOL_IP, Socket::IP_HDRINCL, true)
|
151
|
+
else
|
152
|
+
# BSD
|
153
|
+
s.setsockopt(Socket::IPPROTO_IP, Socket::IP_HDRINCL, true)
|
154
|
+
end
|
155
|
+
# Takeshi: set specific network interface
|
156
|
+
s.setsockopt(Socket::SOL_SOCKET, Socket::SO_BINDTODEVICE, @iface)
|
157
|
+
rescue Errno::EPERM
|
158
|
+
raise ArgumentError, "Must run #{$0} as root."
|
159
|
+
end
|
160
|
+
|
161
|
+
return s.send(pack, 0, Socket.pack_sockaddr_in(1024, @layers[3].dst_ip))
|
162
|
+
end
|
163
|
+
end
|
164
|
+
end
|
165
|
+
|
166
|
+
# vim: set ts=2 et sw=2:
|
@@ -0,0 +1,66 @@
|
|
1
|
+
# $Id: racketpart.rb 14 2008-03-02 05:42:30Z warchild $
|
2
|
+
#
|
3
|
+
# Copyright (c) 2008, Jon Hart
|
4
|
+
# All rights reserved.
|
5
|
+
#
|
6
|
+
# Redistribution and use in source and binary forms, with or without
|
7
|
+
# modification, are permitted provided that the following conditions are met:
|
8
|
+
# * Redistributions of source code must retain the above copyright
|
9
|
+
# notice, this list of conditions and the following disclaimer.
|
10
|
+
# * Redistributions in binary form must reproduce the above copyright
|
11
|
+
# notice, this list of conditions and the following disclaimer in the
|
12
|
+
# documentation and/or other materials provided with the distribution.
|
13
|
+
# * Neither the name of the <organization> nor the
|
14
|
+
# names of its contributors may be used to endorse or promote products
|
15
|
+
# derived from this software without specific prior written permission.
|
16
|
+
#
|
17
|
+
# THIS SOFTWARE IS PROVIDED BY Jon Hart ``AS IS'' AND ANY
|
18
|
+
# EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
19
|
+
# WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
20
|
+
# DISCLAIMED. IN NO EVENT SHALL Jon Hart BE LIABLE FOR ANY
|
21
|
+
# DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
22
|
+
# (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
23
|
+
# LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
24
|
+
# ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
25
|
+
# (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
26
|
+
# SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
27
|
+
#
|
28
|
+
require 'rubygems'
|
29
|
+
require 'bit-struct'
|
30
|
+
|
31
|
+
module Racket
|
32
|
+
# Every Racket object is made up of numerous sub-parts, namely layers of the OSI stack.
|
33
|
+
class RacketPart < BitStruct
|
34
|
+
|
35
|
+
# Boolean indicating whether or not this instance should be
|
36
|
+
# automatically "fixed" prior to be packed and sent.
|
37
|
+
attr_accessor :autofix
|
38
|
+
|
39
|
+
# Should this instance be automatically fixed
|
40
|
+
# prior to being packed and sent?
|
41
|
+
def autofix?
|
42
|
+
@autofix
|
43
|
+
end
|
44
|
+
|
45
|
+
def initialize(*args)
|
46
|
+
@autofix = true
|
47
|
+
super(*args)
|
48
|
+
end
|
49
|
+
|
50
|
+
|
51
|
+
# Print out all of the fields and all of their values
|
52
|
+
def pretty
|
53
|
+
s = ""
|
54
|
+
self.fields.each do |f|
|
55
|
+
unless (f.name == "payload")
|
56
|
+
s += "#{f.name}=#{self.send(f.name)} "
|
57
|
+
end
|
58
|
+
end
|
59
|
+
s.gsub(/ $/, '')
|
60
|
+
end
|
61
|
+
|
62
|
+
def fix!
|
63
|
+
end
|
64
|
+
end
|
65
|
+
end
|
66
|
+
# vim: set ts=2 et sw=2:
|
data/lib/racket.rb
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
# $Id$
|
2
|
+
|
3
|
+
$:.unshift File.join(File.dirname(__FILE__), "../..", "lib")
|
4
|
+
|
5
|
+
require 'test/unit'
|
6
|
+
require 'racket'
|
7
|
+
|
8
|
+
class TestEthernet < Test::Unit::TestCase
|
9
|
+
def test_init
|
10
|
+
assert_nothing_raised() { Racket::L2::Ethernet.new }
|
11
|
+
assert_nothing_raised() { Racket::L2::Ethernet.new(Racket::Misc.randstring(30)) }
|
12
|
+
end
|
13
|
+
|
14
|
+
def test_attrs
|
15
|
+
binary = "\x00\x00\x24\xc1\x8c\x09\x00\x30\x1b\xa0\x63\x16\x08\x00\x45\x10\x00\x3c\x2f\xdf\x40\x00\x40\x06\x89\x17\xc0\xa8\x00\x64\xc0\xa8\x00\x01\x99\xb7\x00\x35\x29\x39\x28\x66\x00\x00\x00\x00\xa0\x02\x16\xd0\xbc\x04\x00\x00\x02\x04\x05\xb4\x04\x02\x08\x0a\x00\x31\x07\xb9\x00\x00\x00\x00\x01\x03\x03\x07"
|
16
|
+
e = Racket::L2::Ethernet.new(binary)
|
17
|
+
assert_equal(e.src_mac, "00:30:1b:a0:63:16")
|
18
|
+
assert_equal(e.dst_mac, "00:00:24:c1:8c:09")
|
19
|
+
assert_equal(e.ethertype, 2048)
|
20
|
+
end
|
21
|
+
end
|
22
|
+
# vim: set ts=2 et sw=2:
|
data/test/l2/ts_misc.rb
ADDED
@@ -0,0 +1,23 @@
|
|
1
|
+
# $Id$
|
2
|
+
|
3
|
+
$:.unshift File.join(File.dirname(__FILE__), "../..", "lib")
|
4
|
+
|
5
|
+
require 'test/unit'
|
6
|
+
require 'racket'
|
7
|
+
|
8
|
+
class TestL2Misc < Test::Unit::TestCase
|
9
|
+
def test_convert
|
10
|
+
(0..rand(1024)).each {
|
11
|
+
len = rand(512)+1
|
12
|
+
mac = Racket::L2::Misc.randommac(len)
|
13
|
+
assert_equal(mac.length, (len*2) + (len-1))
|
14
|
+
long = Racket::L2::Misc.mac2long(mac)
|
15
|
+
assert_equal(mac, Racket::L2::Misc.long2mac(long, len))
|
16
|
+
assert_equal(long, Racket::L2::Misc.mac2long(mac))
|
17
|
+
mac = "00:11:22:33:44:55"
|
18
|
+
string = Racket::L2::Misc.mac2string(mac)
|
19
|
+
assert_equal(mac, Racket::L2::Misc.string2mac(string))
|
20
|
+
}
|
21
|
+
end
|
22
|
+
end
|
23
|
+
# vim: set ts=2 et sw=2:
|
data/test/l2/ts_vlan.rb
ADDED
@@ -0,0 +1,15 @@
|
|
1
|
+
# $Id$
|
2
|
+
|
3
|
+
$:.unshift File.join(File.dirname(__FILE__), "../..", "lib")
|
4
|
+
|
5
|
+
require 'test/unit'
|
6
|
+
require 'racket'
|
7
|
+
|
8
|
+
class TestVLAN < Test::Unit::TestCase
|
9
|
+
def test_init
|
10
|
+
assert_nothing_raised() { Racket::L2::VLAN.new }
|
11
|
+
assert_nothing_raised() { Racket::L2::VLAN.new(Racket::Misc.randstring(30)) }
|
12
|
+
end
|
13
|
+
|
14
|
+
end
|
15
|
+
# vim: set ts=2 et sw=2:
|
data/test/l3/ts_ipv4.rb
ADDED
@@ -0,0 +1,44 @@
|
|
1
|
+
# $Id$
|
2
|
+
|
3
|
+
$:.unshift File.join(File.dirname(__FILE__), "../..", "lib")
|
4
|
+
|
5
|
+
require 'test/unit'
|
6
|
+
require 'racket'
|
7
|
+
|
8
|
+
class TestIPv4 < Test::Unit::TestCase
|
9
|
+
def test_init
|
10
|
+
assert_nothing_raised() { Racket::L3::IPv4.new }
|
11
|
+
assert_nothing_raised() { Racket::L3::IPv4.new(Racket::Misc.randstring(30)) }
|
12
|
+
end
|
13
|
+
|
14
|
+
def test_attrs
|
15
|
+
binary = "\x45\x10\x00\x3c\x2f\xdf\x40\x00\x40\x06\x89\x17\xc0\xa8\x00\x64\xc0\xa8\x00\x01\x99\xb7\x00\x35\x29\x39\x28\x66\x00\x00\x00\x00\xa0\x02\x16\xd0\xbc\x04\x00\x00\x02\x04\x05\xb4\x04\x02\x08\x0a\x00\x31\x07\xb9\x00\x00\x00\x00\x01\x03\x03\x07"
|
16
|
+
i = Racket::L3::IPv4.new(binary)
|
17
|
+
assert_equal(i.version, 4)
|
18
|
+
assert_equal(i.hlen, 5)
|
19
|
+
assert_equal(i.tos, 16)
|
20
|
+
assert_equal(i.len, 60)
|
21
|
+
assert_equal(i.id, 12255)
|
22
|
+
assert_equal(i.flags, 2)
|
23
|
+
assert_equal(i.foffset, 0)
|
24
|
+
assert_equal(i.ttl, 64)
|
25
|
+
assert_equal(i.protocol, 6)
|
26
|
+
assert_equal(i.checksum, 35095)
|
27
|
+
assert_equal(i.src_ip, "192.168.0.100")
|
28
|
+
assert_equal(i.dst_ip, "192.168.0.1")
|
29
|
+
|
30
|
+
i.src_ip = "1.2.3.4"
|
31
|
+
i.fix!
|
32
|
+
assert_equal(i.checksum, 17950)
|
33
|
+
end
|
34
|
+
|
35
|
+
def test_convert
|
36
|
+
(0..512).each {
|
37
|
+
long = rand(2**32)
|
38
|
+
ipv4 = Racket::L3::Misc.long2ipv4(long)
|
39
|
+
assert_equal(long, Racket::L3::Misc.ipv42long(ipv4))
|
40
|
+
assert_equal(ipv4, Racket::L3::Misc.long2ipv4(long))
|
41
|
+
}
|
42
|
+
end
|
43
|
+
end
|
44
|
+
# vim: set ts=2 et sw=2:
|
data/test/l3/ts_ipv6.rb
ADDED
@@ -0,0 +1,26 @@
|
|
1
|
+
# $Id$
|
2
|
+
|
3
|
+
$:.unshift File.join(File.dirname(__FILE__), "../..", "lib")
|
4
|
+
|
5
|
+
require 'test/unit'
|
6
|
+
require 'racket'
|
7
|
+
|
8
|
+
class TestIPV6 < Test::Unit::TestCase
|
9
|
+
def test_init
|
10
|
+
assert_nothing_raised() { Racket::L3::IPv6.new }
|
11
|
+
assert_nothing_raised() { Racket::L3::IPv6.new(Racket::Misc.randstring(30)) }
|
12
|
+
end
|
13
|
+
|
14
|
+
def test_convert
|
15
|
+
long = rand(0xFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF)
|
16
|
+
ipv6 = Racket::L3::Misc.long2ipv6(long)
|
17
|
+
|
18
|
+
assert_equal(long, Racket::L3::Misc.ipv62long(ipv6))
|
19
|
+
assert_equal(ipv6, Racket::L3::Misc.long2ipv6(long))
|
20
|
+
# assert_nothing_raised { Racket::L3::IPv6.new("a:b:c::1") }
|
21
|
+
assert_nothing_raised { Racket::L3::Misc.soll_mcast_addr6(ipv6) }
|
22
|
+
mac = Racket::L2::Misc.randommac
|
23
|
+
assert_nothing_raised { Racket::L3::Misc.soll_mcast_mac(mac) }
|
24
|
+
end
|
25
|
+
end
|
26
|
+
# vim: set ts=2 et sw=2:
|
data/test/l3/ts_misc.rb
ADDED
@@ -0,0 +1,31 @@
|
|
1
|
+
# $Id$
|
2
|
+
|
3
|
+
$:.unshift File.join(File.dirname(__FILE__), "../..", "lib")
|
4
|
+
|
5
|
+
require 'test/unit'
|
6
|
+
require 'racket'
|
7
|
+
|
8
|
+
class TestL3Misc < Test::Unit::TestCase
|
9
|
+
def test_misc
|
10
|
+
assert_nothing_raised {
|
11
|
+
Racket::L3::Misc.checksum(Racket::Misc.randstring(rand(2048)))
|
12
|
+
}
|
13
|
+
|
14
|
+
mac = "6e:f1:eb:b7:c8:72"
|
15
|
+
ll = "fe80::6cf1:ebff:feb7:c872"
|
16
|
+
assert_equal(ll, Racket::L3::Misc.linklocaladdr(mac))
|
17
|
+
|
18
|
+
ipv4 = Racket::L3::Misc.randomipv4
|
19
|
+
lipv4 = Racket::L3::Misc.ipv42long(ipv4)
|
20
|
+
ipv6 = Racket::L3::Misc.randomipv6
|
21
|
+
lipv6 = Racket::L3::Misc.ipv62long(ipv6)
|
22
|
+
|
23
|
+
assert_equal(ipv4, Racket::L3::Misc.long2ipv4(lipv4))
|
24
|
+
assert_equal(ipv6, Racket::L3::Misc.long2ipv6(lipv6))
|
25
|
+
|
26
|
+
ipv6 = "a:0:0:0f:0:1"
|
27
|
+
zipv6 = "a::f:0:1"
|
28
|
+
assert_equal(zipv6, Racket::L3::Misc.compressipv6(ipv6))
|
29
|
+
end
|
30
|
+
end
|
31
|
+
# vim: set ts=2 et sw=2:
|