This library uses dissectors. Each dissector corresponds to a protocol. They are used to build and dissect packages. They are located in the layers folder.
Each layer is represented by a structure with several fields corresponding to the protocol. For each field, you must indicate which type it is.
Let's take the ARP layer as an example:
// ARP Layer
type ARP struct {
Hwtype uint16
Ptype uint16
Hwlen uint8
Plen uint8
Opcode uint16
Hwsrc net.HardwareAddr
Psrc net.IP
Hwdst net.HardwareAddr
Pdst net.IP
}
As you can see, the ARP layer is a structure with different fields.
This name must be the same as that used in the structure.
The following functions must be defined for each layer: GetName()
, SetDefault()
, Build()
, Dissect()
, BindLayer()
and <LayerName>Layer()
.
This function is used to specify the layer name. It is used in the Show()
function to retrieve the name of the current layer. The function returns a string of characters.
func (a *ARP) GetName() string {
return "ARP"
}
The purpose of this function is to define default values for each protocol field.
func (a *ARP) SetDefault() {
ifaces, _ := net.Interfaces()
netAddr, _ := net.InterfaceAddrs()
a.Hwtype = 1
a.Ptype = 0x0800
a.Hwlen = 6
a.Plen = 4
a.Opcode = 1
a.Hwsrc = ifaces[1].HardwareAddr
a.Psrc = netAddr[1].(*net.IPNet).IP.To4()
a.Hwdst = net.HardwareAddr{0x00, 0x00, 0x00, 0x00, 0x00, 0x00}
a.Pdst = net.IP{0, 0, 0, 0}
}
This function is used when creating a packet using the ARP layer. It returns a corresponding array of bytes containing the values of each field.
func (a *ARP) Build() *buffer.ProtoBuff {
// Process
}
The first step is to initialize a buffer containing the bytes of our fields.
// Initiate the buffer
var buffer buffer.ProtoBuff
Next, we need to add the value of each of our fields inside the buffer. The Add()
function takes three positional arguments. The first is the field name, the second is its value and the third is for enumeration.
// Add Hwtype field into the buffer
buffer.Add("Hwtype", a.Hwtype, HARDWARE_TYPES[a.Hwtype])
// Add Ptype field into the buffer
buffer.Add("Ptype", a.Ptype, ETHERTYPE[a.Ptype])
// Add Hwlen field into the buffer
buffer.Add("Hwlen", a.Hwlen, nil)
// Add Plen field into the buffer
buffer.Add("Plen", a.Plen, nil)
// Add Opcode field into the buffer
buffer.Add("Opcode", a.Opcode, OPCODE[a.Opcode])
// Add Hwsrc field into the buffer
buffer.Add("Hwsrc", a.Hwsrc, nil)
// Add Psrc field into the buffer
buffer.Add("Psrc", a.Psrc, nil)
// Add Hwdst field into the buffer
buffer.Add("Hwdst", a.Hwdst, nil)
// Add Pdst field into the buffer
buffer.Add("Pdst", a.Pds, nil)
Finally, we return the buffer.
return &buffer
This function converts an array of bytes into a layer. It takes as argument a buffer corresponding to the undissected bytes and returns the buffer.
func (a *ARP) Dissect(buffer *buffer.ProtoBuff) *buffer.ProtoBuff {
// Process
return buffer
}
Bytes are inserted for each field in the layer. Be careful with the type !
// Inserts bytes in Hwtype
a.Hwtype = buffer.NextUint16()
// Inserts bytes in Ptype
a.Ptype = buffer.NextUint16()
// Inserts byte in Hwlen
a.Hwlen = buffer.NextUint8()
// Inserts byte in Plen
a.Plen = buffer.NextUint8()
// Inserts bytes in Opcode
a.Opcode = buffer.NextUint16()
// Inserts bytes in Hwsrc
a.Hwsrc = buffer.Next(int(a.Hwlen))
// Inserts bytes in Psrc
a.Psrc = buffer.Next(int(a.Plen))
// Inserts bytes in Hwdst
a.Hwdst = buffer.Next(int(a.Hwlen))
// Inserts bytes in Pdst
a.Pdst = buffer.Next(int(a.Plen))
This function is used to indicate the next layer. It returns a Layer strucuture corresponding to the next layer.
func (a *ARP) BindLayer() Layer {
return nil
}
If there is no next layer, the value nil
is returned.
Let's imagine that the next layer is IP (this is an example, of course ^^), then we'd have :
// Condition on the field indicating the next layer
if a.field == value {
return &IP{}
}
Once you've understood this, it's important to specify the condition for indicating the presence of our layer in the BindLayer()
function of the previous layer !
This is the function that will be called when the layer is created. It instantiates the layer and applies default values to the fields by calling the SetDefault()
function. Finally, it returns the layer, which can then be manipulated by the user.
func ARPLayer() ARP {
arp := ARP{}
arp.SetDefault()
return arp
}