Skip to content
Snippets Groups Projects
update.go 3.73 KiB
Newer Older
  • Learn to ignore specific revisions
  • package packet
    
    import (
    	"bytes"
    	"fmt"
    
    	"github.com/taktv6/tflow2/convert"
    )
    
    type BGPUpdate struct {
    	WithdrawnRoutesLen uint16
    	WithdrawnRoutes    *NLRI
    	TotalPathAttrLen   uint16
    	PathAttributes     *PathAttribute
    	NLRI               *NLRI
    }
    
    func (b *BGPUpdate) SerializeUpdate(opt *Options) ([]byte, error) {
    	budget := MaxLen - MinLen
    	nlriLen := 0
    	buf := bytes.NewBuffer(nil)
    
    	withdrawBuf := bytes.NewBuffer(nil)
    	for withdraw := b.WithdrawnRoutes; withdraw != nil; withdraw = withdraw.Next {
    		if opt.AddPathRX {
    			nlriLen = int(withdraw.serializeAddPath(withdrawBuf))
    		} else {
    			nlriLen = int(withdraw.serialize(withdrawBuf))
    		}
    
    		budget -= nlriLen
    		if budget < 0 {
    			return nil, fmt.Errorf("update too long")
    		}
    	}
    
    	pathAttributesBuf := bytes.NewBuffer(nil)
    	for pa := b.PathAttributes; pa != nil; pa = pa.Next {
    
    Oliver Herms's avatar
    Oliver Herms committed
    		paLen := int(pa.Serialize(pathAttributesBuf, opt))
    
    		budget -= paLen
    		if budget < 0 {
    			return nil, fmt.Errorf("update too long")
    		}
    	}
    
    	nlriBuf := bytes.NewBuffer(nil)
    	for nlri := b.NLRI; nlri != nil; nlri = nlri.Next {
    		if opt.AddPathRX {
    			nlriLen = int(nlri.serializeAddPath(withdrawBuf))
    		} else {
    			nlriLen = int(nlri.serialize(withdrawBuf))
    		}
    
    		budget -= nlriLen
    		if budget < 0 {
    			return nil, fmt.Errorf("update too long")
    		}
    	}
    
    	withdrawnRoutesLen := withdrawBuf.Len()
    	if withdrawnRoutesLen > 65535 {
    		return nil, fmt.Errorf("Invalid Withdrawn Routes Length: %d", withdrawnRoutesLen)
    	}
    
    	totalPathAttributesLen := pathAttributesBuf.Len()
    	if totalPathAttributesLen > 65535 {
    		return nil, fmt.Errorf("Invalid Total Path Attribute Length: %d", totalPathAttributesLen)
    	}
    
    	totalLength := 2 + withdrawnRoutesLen + totalPathAttributesLen + 2 + nlriBuf.Len() + 19
    	if totalLength > 4096 {
    		return nil, fmt.Errorf("Update too long: %d bytes", totalLength)
    	}
    
    	serializeHeader(buf, uint16(totalLength), UpdateMsg)
    
    	buf.Write(convert.Uint16Byte(uint16(withdrawnRoutesLen)))
    	buf.Write(withdrawBuf.Bytes())
    
    	buf.Write(convert.Uint16Byte(uint16(totalPathAttributesLen)))
    	buf.Write(pathAttributesBuf.Bytes())
    
    	buf.Write(nlriBuf.Bytes())
    
    	return buf.Bytes(), nil
    }
    
    func (b *BGPUpdate) SerializeUpdateAddPath(opt *Options) ([]byte, error) {
    	budget := MaxLen - MinLen
    	buf := bytes.NewBuffer(nil)
    
    	withdrawBuf := bytes.NewBuffer(nil)
    	for withdraw := b.WithdrawnRoutes; withdraw != nil; withdraw = withdraw.Next {
    		nlriLen := int(withdraw.serialize(withdrawBuf))
    		budget -= nlriLen
    		if budget < 0 {
    			return nil, fmt.Errorf("update too long")
    		}
    	}
    
    	pathAttributesBuf := bytes.NewBuffer(nil)
    	for pa := b.PathAttributes; pa != nil; pa = pa.Next {
    
    Oliver Herms's avatar
    Oliver Herms committed
    		paLen := int(pa.Serialize(pathAttributesBuf, opt))
    
    		budget -= paLen
    		if budget < 0 {
    			return nil, fmt.Errorf("update too long")
    		}
    	}
    
    	nlriBuf := bytes.NewBuffer(nil)
    	for nlri := b.NLRI; nlri != nil; nlri = nlri.Next {
    		nlriLen := int(nlri.serialize(nlriBuf))
    		budget -= nlriLen
    		if budget < 0 {
    			return nil, fmt.Errorf("update too long")
    		}
    	}
    
    	withdrawnRoutesLen := withdrawBuf.Len()
    	if withdrawnRoutesLen > 65535 {
    		return nil, fmt.Errorf("Invalid Withdrawn Routes Length: %d", withdrawnRoutesLen)
    	}
    
    	totalPathAttributesLen := pathAttributesBuf.Len()
    	if totalPathAttributesLen > 65535 {
    		return nil, fmt.Errorf("Invalid Total Path Attribute Length: %d", totalPathAttributesLen)
    	}
    
    	totalLength := 2 + withdrawnRoutesLen + totalPathAttributesLen + 2 + nlriBuf.Len() + 19
    	if totalLength > 4096 {
    		return nil, fmt.Errorf("Update too long: %d bytes", totalLength)
    	}
    
    	serializeHeader(buf, uint16(totalLength), UpdateMsg)
    
    	buf.Write(convert.Uint16Byte(uint16(withdrawnRoutesLen)))
    	buf.Write(withdrawBuf.Bytes())
    
    	buf.Write(convert.Uint16Byte(uint16(totalPathAttributesLen)))
    	buf.Write(pathAttributesBuf.Bytes())
    
    	buf.Write(nlriBuf.Bytes())
    
    	return buf.Bytes(), nil
    }