From 78d75f4d4be95787bd3e8b1bf28172059d09273d Mon Sep 17 00:00:00 2001 From: Sean Karlage Date: Tue, 14 Aug 2018 22:53:45 -0700 Subject: BSDP: Add code to generate ACKs for INFORMs --- dhcpv4/bsdp/bsdp.go | 90 ++++++++++++ dhcpv4/bsdp/bsdp_test.go | 204 ++++++++++++++++++++++++++ dhcpv4/bsdp/types.go | 2 +- dhcpv4/bsdp/vendor_class_identifier_darwin.go | 2 +- 4 files changed, 296 insertions(+), 2 deletions(-) diff --git a/dhcpv4/bsdp/bsdp.go b/dhcpv4/bsdp/bsdp.go index 27de68e..2909ccd 100644 --- a/dhcpv4/bsdp/bsdp.go +++ b/dhcpv4/bsdp/bsdp.go @@ -14,6 +14,20 @@ import ( // prefer this BSDP-specific option over the DHCP standard option. const MaxDHCPMessageSize = 1500 +// AppleVendorID is the string constant set in the vendor class identifier (DHCP +// option 60) that is sent by the server. +const AppleVendorID = "AAPLBSDPC" + +// ReplyConfig is a struct containing some common configuration values for a +// BSDP reply (ACK). +type ReplyConfig struct { + ServerIP net.IP + ServerHostname, BootFileName string + ServerPriority int + Images []BootImage + DefaultImage, SelectedImage *BootImage +} + // ParseBootImageListFromAck parses the list of boot images presented in the // ACK[LIST] packet and returns them as a list of BootImages. func ParseBootImageListFromAck(ack dhcpv4.DHCPv4) ([]BootImage, error) { @@ -139,3 +153,79 @@ func InformSelectForAck(ack dhcpv4.DHCPv4, replyPort uint16, selectedImage BootI d.AddOption(&OptVendorSpecificInformation{vendorOpts}) return d, nil } + +// NewReplyForInformList constructs an ACK for the INFORM[LIST] packet `inform` +// with additional options in `config`. +func NewReplyForInformList(inform *dhcpv4.DHCPv4, config ReplyConfig) (*dhcpv4.DHCPv4, error) { + if config.DefaultImage == nil { + return nil, errors.New("NewReplyForInformList: no default boot image ID set") + } + if len(config.Images) == 0 { + return nil, errors.New("NewReplyForInformList: no boot images provided") + } + reply, err := dhcpv4.NewReplyFromRequest(inform) + if err != nil { + return nil, err + } + reply.SetClientIPAddr(inform.ClientIPAddr()) + reply.SetYourIPAddr(net.IPv4zero) + reply.SetGatewayIPAddr(inform.GatewayIPAddr()) + reply.SetServerIPAddr(config.ServerIP) + reply.SetServerHostName([]byte(config.ServerHostname)) + + reply.AddOption(&dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}) + reply.AddOption(&dhcpv4.OptServerIdentifier{ServerID: config.ServerIP}) + reply.AddOption(&dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}) + + // BSDP opts. + vendorOpts := []dhcpv4.Option{ + &OptMessageType{Type: MessageTypeList}, + &OptServerPriority{Priority: config.ServerPriority}, + &OptDefaultBootImageID{ID: config.DefaultImage.ID}, + &OptBootImageList{Images: config.Images}, + } + if config.SelectedImage != nil { + vendorOpts = append(vendorOpts, &OptSelectedBootImageID{ID: config.SelectedImage.ID}) + } + reply.AddOption(&OptVendorSpecificInformation{Options: vendorOpts}) + + reply.AddOption(&dhcpv4.OptionGeneric{OptionCode: dhcpv4.OptionEnd}) + return reply, nil +} + +// NewReplyForInformSelect constructs an ACK for the INFORM[Select] packet +// `inform` with additional options in `config`. +func NewReplyForInformSelect(inform *dhcpv4.DHCPv4, config ReplyConfig) (*dhcpv4.DHCPv4, error) { + if config.SelectedImage == nil { + return nil, errors.New("NewReplyForInformSelect: no selected boot image ID set") + } + if len(config.Images) == 0 { + return nil, errors.New("NewReplyForInformSelect: no boot images provided") + } + reply, err := dhcpv4.NewReplyFromRequest(inform) + if err != nil { + return nil, err + } + + reply.SetClientIPAddr(inform.ClientIPAddr()) + reply.SetYourIPAddr(net.IPv4zero) + reply.SetGatewayIPAddr(inform.GatewayIPAddr()) + reply.SetServerIPAddr(config.ServerIP) + reply.SetServerHostName([]byte(config.ServerHostname)) + reply.SetBootFileName([]byte(config.BootFileName)) + + reply.AddOption(&dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}) + reply.AddOption(&dhcpv4.OptServerIdentifier{ServerID: config.ServerIP}) + reply.AddOption(&dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}) + + // BSDP opts. + reply.AddOption(&OptVendorSpecificInformation{ + Options: []dhcpv4.Option{ + &OptMessageType{Type: MessageTypeSelect}, + &OptSelectedBootImageID{ID: config.SelectedImage.ID}, + }, + }) + + reply.AddOption(&dhcpv4.OptionGeneric{OptionCode: dhcpv4.OptionEnd}) + return reply, nil +} diff --git a/dhcpv4/bsdp/bsdp_test.go b/dhcpv4/bsdp/bsdp_test.go index ad2a265..f457631 100644 --- a/dhcpv4/bsdp/bsdp_test.go +++ b/dhcpv4/bsdp/bsdp_test.go @@ -1,9 +1,11 @@ package bsdp import ( + "net" "testing" "github.com/insomniacslk/dhcp/dhcpv4" + "github.com/insomniacslk/dhcp/iana" "github.com/stretchr/testify/require" ) @@ -49,3 +51,205 @@ func TestNeedsReplyPort(t *testing.T) { require.False(t, needsReplyPort(0)) require.False(t, needsReplyPort(dhcpv4.ClientPort)) } + +// TODO(get9): Remove when #99 lands. +func newInform() *dhcpv4.DHCPv4 { + p, _ := dhcpv4.New() + p.SetClientIPAddr(net.IP{1, 2, 3, 4}) + p.SetGatewayIPAddr(net.IP{4, 3, 2, 1}) + p.SetHwType(iana.HwTypeEthernet) + hwAddr := [16]byte{1, 2, 3, 4, 5, 6} + p.SetClientHwAddr(hwAddr[:]) + p.SetHwAddrLen(6) + return p +} + +func TestNewReplyForInformList_NoDefaultImage(t *testing.T) { + inform := newInform() + _, err := NewReplyForInformList(inform, ReplyConfig{}) + require.Error(t, err) +} + +func TestNewReplyForInformList_NoImages(t *testing.T) { + inform := newInform() + fakeImage := BootImage{ + ID: BootImageID{ImageType: BootImageTypeMacOSX}, + } + _, err := NewReplyForInformList(inform, ReplyConfig{ + Images: []BootImage{}, + DefaultImage: &fakeImage, + }) + require.Error(t, err) +} + +// TODO (get9): clean up when #99 lands. +func TestNewReplyForInformList(t *testing.T) { + inform := newInform() + images := []BootImage{ + BootImage{ + ID: BootImageID{ + IsInstall: true, + ImageType: BootImageTypeMacOSX, + Index: 0x7070, + }, + Name: "image-1", + }, + BootImage{ + ID: BootImageID{ + IsInstall: true, + ImageType: BootImageTypeMacOSX, + Index: 0x8080, + }, + Name: "image-2", + }, + } + config := ReplyConfig{ + Images: images, + DefaultImage: &images[0], + ServerIP: net.IP{9, 9, 9, 9}, + ServerHostname: "bsdp.foo.com", + ServerPriority: 0x7070, + } + ack, err := NewReplyForInformList(inform, config) + require.NoError(t, err) + require.Equal(t, net.IP{1, 2, 3, 4}, ack.ClientIPAddr()) + require.Equal(t, net.IPv4zero, ack.YourIPAddr()) + require.Equal(t, net.IP{4, 3, 2, 1}, ack.GatewayIPAddr()) + require.Equal(t, "bsdp.foo.com", ack.ServerHostNameToString()) + + // Validate options. + require.Equal( + t, + &dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}, + ack.GetOneOption(dhcpv4.OptionDHCPMessageType).(*dhcpv4.OptMessageType), + ) + require.Equal( + t, + &dhcpv4.OptServerIdentifier{ServerID: net.IP{9, 9, 9, 9}}, + ack.GetOneOption(dhcpv4.OptionServerIdentifier).(*dhcpv4.OptServerIdentifier), + ) + require.Equal( + t, + &dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}, + ack.GetOneOption(dhcpv4.OptionClassIdentifier).(*dhcpv4.OptClassIdentifier), + ) + require.NotNil(t, ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation)) + require.Equal(t, &dhcpv4.OptionGeneric{OptionCode: dhcpv4.OptionEnd}, ack.Options()[len(ack.Options())-1]) + + vendorOpts := ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation).(*OptVendorSpecificInformation) + require.Equal( + t, + &OptMessageType{Type: MessageTypeList}, + vendorOpts.GetOneOption(OptionMessageType).(*OptMessageType), + ) + require.Equal( + t, + &OptServerPriority{Priority: 0x7070}, + vendorOpts.GetOneOption(OptionServerPriority).(*OptServerPriority), + ) + require.Equal( + t, + &OptDefaultBootImageID{ID: images[0].ID}, + vendorOpts.GetOneOption(OptionDefaultBootImageID).(*OptDefaultBootImageID), + ) + require.Equal( + t, + &OptBootImageList{Images: images}, + vendorOpts.GetOneOption(OptionBootImageList).(*OptBootImageList), + ) + + // Add in selected boot image, ensure it's in the generated ACK. + config.SelectedImage = &images[0] + ack, err = NewReplyForInformList(inform, config) + require.NoError(t, err) + vendorOpts = ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation).(*OptVendorSpecificInformation) + require.Equal( + t, + &OptSelectedBootImageID{ID: images[0].ID}, + vendorOpts.GetOneOption(OptionSelectedBootImageID).(*OptSelectedBootImageID), + ) +} + +func TestNewReplyForInformSelect_NoSelectedImage(t *testing.T) { + inform := newInform() + _, err := NewReplyForInformSelect(inform, ReplyConfig{}) + require.Error(t, err) +} + +func TestNewReplyForInformSelect_NoImages(t *testing.T) { + inform := newInform() + fakeImage := BootImage{ + ID: BootImageID{ImageType: BootImageTypeMacOSX}, + } + _, err := NewReplyForInformSelect(inform, ReplyConfig{ + Images: []BootImage{}, + SelectedImage: &fakeImage, + }) + require.Error(t, err) +} + +func TestNewReplyForInformSelect(t *testing.T) { + inform := newInform() + images := []BootImage{ + BootImage{ + ID: BootImageID{ + IsInstall: true, + ImageType: BootImageTypeMacOSX, + Index: 0x7070, + }, + Name: "image-1", + }, + BootImage{ + ID: BootImageID{ + IsInstall: true, + ImageType: BootImageTypeMacOSX, + Index: 0x8080, + }, + Name: "image-2", + }, + } + config := ReplyConfig{ + Images: images, + SelectedImage: &images[0], + ServerIP: net.IP{9, 9, 9, 9}, + ServerHostname: "bsdp.foo.com", + ServerPriority: 0x7070, + } + ack, err := NewReplyForInformSelect(inform, config) + require.NoError(t, err) + require.Equal(t, net.IP{1, 2, 3, 4}, ack.ClientIPAddr()) + require.Equal(t, net.IPv4zero, ack.YourIPAddr()) + require.Equal(t, net.IP{4, 3, 2, 1}, ack.GatewayIPAddr()) + require.Equal(t, "bsdp.foo.com", ack.ServerHostNameToString()) + + // Validate options. + require.Equal( + t, + &dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}, + ack.GetOneOption(dhcpv4.OptionDHCPMessageType).(*dhcpv4.OptMessageType), + ) + require.Equal( + t, + &dhcpv4.OptServerIdentifier{ServerID: net.IP{9, 9, 9, 9}}, + ack.GetOneOption(dhcpv4.OptionServerIdentifier).(*dhcpv4.OptServerIdentifier), + ) + require.Equal( + t, + &dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}, + ack.GetOneOption(dhcpv4.OptionClassIdentifier).(*dhcpv4.OptClassIdentifier), + ) + require.NotNil(t, ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation)) + require.Equal(t, &dhcpv4.OptionGeneric{OptionCode: dhcpv4.OptionEnd}, ack.Options()[len(ack.Options())-1]) + + vendorOpts := ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation).(*OptVendorSpecificInformation) + require.Equal( + t, + &OptMessageType{Type: MessageTypeSelect}, + vendorOpts.GetOneOption(OptionMessageType).(*OptMessageType), + ) + require.Equal( + t, + &OptSelectedBootImageID{ID: images[0].ID}, + vendorOpts.GetOneOption(OptionSelectedBootImageID).(*OptSelectedBootImageID), + ) +} diff --git a/dhcpv4/bsdp/types.go b/dhcpv4/bsdp/types.go index 4f1c140..ac4b05b 100644 --- a/dhcpv4/bsdp/types.go +++ b/dhcpv4/bsdp/types.go @@ -6,7 +6,7 @@ import "github.com/insomniacslk/dhcp/dhcpv4" // on non-darwin hosts where the vendor class identifier cannot be determined. // It should mostly be used for debugging if testing BSDP on a non-darwin // system. -const DefaultMacOSVendorClassIdentifier = "AAPLBSDP/i386/MacMini6,1" +const DefaultMacOSVendorClassIdentifier = AppleVendorID + "/i386/MacMini6,1" // Options (occur as sub-options of DHCP option 43). const ( diff --git a/dhcpv4/bsdp/vendor_class_identifier_darwin.go b/dhcpv4/bsdp/vendor_class_identifier_darwin.go index 530d843..d1b4c2b 100644 --- a/dhcpv4/bsdp/vendor_class_identifier_darwin.go +++ b/dhcpv4/bsdp/vendor_class_identifier_darwin.go @@ -13,5 +13,5 @@ func MakeVendorClassIdentifier() (string, error) { if err != nil { return "", err } - return fmt.Sprintf("AAPLBSDPC/i386/%s", hwModel), nil + return fmt.Sprintf("%s/i386/%s", AppleVendorID, hwModel), nil } -- cgit v1.2.3 From d614fa996b8fb38528191fe52c258037ae9c539a Mon Sep 17 00:00:00 2001 From: Sean Karlage Date: Wed, 15 Aug 2018 10:20:05 -0700 Subject: Add nil check for Images slice --- dhcpv4/bsdp/bsdp.go | 4 ++-- dhcpv4/bsdp/bsdp_test.go | 11 +++++++++++ 2 files changed, 13 insertions(+), 2 deletions(-) diff --git a/dhcpv4/bsdp/bsdp.go b/dhcpv4/bsdp/bsdp.go index 2909ccd..3402b0c 100644 --- a/dhcpv4/bsdp/bsdp.go +++ b/dhcpv4/bsdp/bsdp.go @@ -160,7 +160,7 @@ func NewReplyForInformList(inform *dhcpv4.DHCPv4, config ReplyConfig) (*dhcpv4.D if config.DefaultImage == nil { return nil, errors.New("NewReplyForInformList: no default boot image ID set") } - if len(config.Images) == 0 { + if config.Images == nil || len(config.Images) == 0 { return nil, errors.New("NewReplyForInformList: no boot images provided") } reply, err := dhcpv4.NewReplyFromRequest(inform) @@ -199,7 +199,7 @@ func NewReplyForInformSelect(inform *dhcpv4.DHCPv4, config ReplyConfig) (*dhcpv4 if config.SelectedImage == nil { return nil, errors.New("NewReplyForInformSelect: no selected boot image ID set") } - if len(config.Images) == 0 { + if config.Images == nil || len(config.Images) == 0 { return nil, errors.New("NewReplyForInformSelect: no boot images provided") } reply, err := dhcpv4.NewReplyFromRequest(inform) diff --git a/dhcpv4/bsdp/bsdp_test.go b/dhcpv4/bsdp/bsdp_test.go index f457631..fc1aedf 100644 --- a/dhcpv4/bsdp/bsdp_test.go +++ b/dhcpv4/bsdp/bsdp_test.go @@ -80,6 +80,11 @@ func TestNewReplyForInformList_NoImages(t *testing.T) { DefaultImage: &fakeImage, }) require.Error(t, err) + + _, err = NewReplyForInformList(inform, ReplyConfig{ + Images: nil, + SelectedImage: &fakeImage, + }) } // TODO (get9): clean up when #99 lands. @@ -186,6 +191,12 @@ func TestNewReplyForInformSelect_NoImages(t *testing.T) { SelectedImage: &fakeImage, }) require.Error(t, err) + + _, err = NewReplyForInformSelect(inform, ReplyConfig{ + Images: nil, + SelectedImage: &fakeImage, + }) + require.Error(t, err) } func TestNewReplyForInformSelect(t *testing.T) { -- cgit v1.2.3 From 206c8032f6b6c954019ca752df3b703da4597b07 Mon Sep 17 00:00:00 2001 From: Sean Karlage Date: Wed, 15 Aug 2018 15:25:14 -0700 Subject: refactor tests to take advantage of new helper functions --- dhcpv4/bsdp/bsdp_test.go | 257 ++++++++++++++++++----------------------------- 1 file changed, 98 insertions(+), 159 deletions(-) diff --git a/dhcpv4/bsdp/bsdp_test.go b/dhcpv4/bsdp/bsdp_test.go index 6029d49..7da4dfa 100644 --- a/dhcpv4/bsdp/bsdp_test.go +++ b/dhcpv4/bsdp/bsdp_test.go @@ -9,6 +9,14 @@ import ( "github.com/stretchr/testify/require" ) +func RequireHasOption(t *testing.T, opts dhcpv4.OptionGetter, opt dhcpv4.Option) { + require.NotNil(t, opts, "must pass list of options") + require.NotNil(t, opt, "must pass option") + require.True(t, dhcpv4.HasOption(opts, opt.Code())) + actual := opts.GetOneOption(opt.Code()) + require.Equal(t, opt, actual) +} + func TestParseBootImageListFromAck(t *testing.T) { expectedBootImages := []BootImage{ BootImage{ @@ -52,96 +60,6 @@ func TestNeedsReplyPort(t *testing.T) { require.False(t, needsReplyPort(dhcpv4.ClientPort)) } -func TestNewReplyForInformSelect_NoSelectedImage(t *testing.T) { - inform := newInform() - _, err := NewReplyForInformSelect(inform, ReplyConfig{}) - require.Error(t, err) -} - -func TestNewReplyForInformSelect_NoImages(t *testing.T) { - inform := newInform() - fakeImage := BootImage{ - ID: BootImageID{ImageType: BootImageTypeMacOSX}, - } - _, err := NewReplyForInformSelect(inform, ReplyConfig{ - Images: []BootImage{}, - SelectedImage: &fakeImage, - }) - require.Error(t, err) - - _, err = NewReplyForInformSelect(inform, ReplyConfig{ - Images: nil, - SelectedImage: &fakeImage, - }) - require.Error(t, err) -} - -func TestNewReplyForInformSelect(t *testing.T) { - inform := newInform() - images := []BootImage{ - BootImage{ - ID: BootImageID{ - IsInstall: true, - ImageType: BootImageTypeMacOSX, - Index: 0x7070, - }, - Name: "image-1", - }, - BootImage{ - ID: BootImageID{ - IsInstall: true, - ImageType: BootImageTypeMacOSX, - Index: 0x8080, - }, - Name: "image-2", - }, - } - config := ReplyConfig{ - Images: images, - SelectedImage: &images[0], - ServerIP: net.IP{9, 9, 9, 9}, - ServerHostname: "bsdp.foo.com", - ServerPriority: 0x7070, - } - ack, err := NewReplyForInformSelect(inform, config) - require.NoError(t, err) - require.Equal(t, net.IP{1, 2, 3, 4}, ack.ClientIPAddr()) - require.Equal(t, net.IPv4zero, ack.YourIPAddr()) - require.Equal(t, net.IP{4, 3, 2, 1}, ack.GatewayIPAddr()) - require.Equal(t, "bsdp.foo.com", ack.ServerHostNameToString()) - - // Validate options. - require.Equal( - t, - &dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}, - ack.GetOneOption(dhcpv4.OptionDHCPMessageType).(*dhcpv4.OptMessageType), - ) - require.Equal( - t, - &dhcpv4.OptServerIdentifier{ServerID: net.IP{9, 9, 9, 9}}, - ack.GetOneOption(dhcpv4.OptionServerIdentifier).(*dhcpv4.OptServerIdentifier), - ) - require.Equal( - t, - &dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}, - ack.GetOneOption(dhcpv4.OptionClassIdentifier).(*dhcpv4.OptClassIdentifier), - ) - require.NotNil(t, ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation)) - require.Equal(t, &dhcpv4.OptionGeneric{OptionCode: dhcpv4.OptionEnd}, ack.Options()[len(ack.Options())-1]) - - vendorOpts := ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation).(*OptVendorSpecificInformation) - require.Equal( - t, - &OptMessageType{Type: MessageTypeSelect}, - vendorOpts.GetOneOption(OptionMessageType).(*OptMessageType), - ) - require.Equal( - t, - &OptSelectedBootImageID{ID: images[0].ID}, - vendorOpts.GetOneOption(OptionSelectedBootImageID).(*OptSelectedBootImageID), - ) -} - func TestNewInformList_NoReplyPort(t *testing.T) { hwAddr := net.HardwareAddr{1, 2, 3, 4, 5, 6} localIP := net.IPv4(10, 10, 11, 11) @@ -232,16 +150,10 @@ func TestInformSelectForAck_Broadcast(t *testing.T) { require.True(t, dhcpv4.HasOption(m, dhcpv4.OptionVendorSpecificInformation)) opt = m.GetOneOption(dhcpv4.OptionVendorSpecificInformation) vendorInfo := opt.(*OptVendorSpecificInformation) - require.True(t, dhcpv4.HasOption(vendorInfo, OptionMessageType)) - opt = vendorInfo.GetOneOption(OptionMessageType) - require.Equal(t, MessageTypeSelect, opt.(*OptMessageType).Type) + RequireHasOption(t, vendorInfo, &OptMessageType{Type: MessageTypeSelect}) require.True(t, dhcpv4.HasOption(vendorInfo, OptionVersion)) - require.True(t, dhcpv4.HasOption(vendorInfo, OptionSelectedBootImageID)) - opt = vendorInfo.GetOneOption(OptionSelectedBootImageID) - require.Equal(t, bootImage.ID, opt.(*OptSelectedBootImageID).ID) - require.True(t, dhcpv4.HasOption(vendorInfo, OptionServerIdentifier)) - opt = vendorInfo.GetOneOption(OptionServerIdentifier) - require.True(t, serverID.Equal(opt.(*OptServerIdentifier).ServerID)) + RequireHasOption(t, vendorInfo, &OptSelectedBootImageID{ID: bootImage.ID}) + RequireHasOption(t, vendorInfo, &OptServerIdentifier{ServerID: serverID}) } func TestInformSelectForAck_NoServerID(t *testing.T) { @@ -304,31 +216,17 @@ func TestInformSelectForAck_ReplyPort(t *testing.T) { require.True(t, dhcpv4.HasOption(m, dhcpv4.OptionVendorSpecificInformation)) opt := m.GetOneOption(dhcpv4.OptionVendorSpecificInformation) vendorInfo := opt.(*OptVendorSpecificInformation) - require.True(t, dhcpv4.HasOption(vendorInfo, OptionReplyPort)) - opt = vendorInfo.GetOneOption(OptionReplyPort) - require.Equal(t, replyPort, opt.(*OptReplyPort).Port) -} - -// TODO(get9): Remove when #99 lands. -func newInform() *dhcpv4.DHCPv4 { - p, _ := dhcpv4.New() - p.SetClientIPAddr(net.IP{1, 2, 3, 4}) - p.SetGatewayIPAddr(net.IP{4, 3, 2, 1}) - p.SetHwType(iana.HwTypeEthernet) - hwAddr := [16]byte{1, 2, 3, 4, 5, 6} - p.SetClientHwAddr(hwAddr[:]) - p.SetHwAddrLen(6) - return p + RequireHasOption(t, vendorInfo, &OptReplyPort{Port: replyPort}) } func TestNewReplyForInformList_NoDefaultImage(t *testing.T) { - inform := newInform() + inform, _ := NewInformList(net.HardwareAddr{1, 2, 3, 4, 5, 6}, net.IP{1, 2, 3, 4}, dhcpv4.ClientPort) _, err := NewReplyForInformList(inform, ReplyConfig{}) require.Error(t, err) } func TestNewReplyForInformList_NoImages(t *testing.T) { - inform := newInform() + inform, _ := NewInformList(net.HardwareAddr{1, 2, 3, 4, 5, 6}, net.IP{1, 2, 3, 4}, dhcpv4.ClientPort) fakeImage := BootImage{ ID: BootImageID{ImageType: BootImageTypeMacOSX}, } @@ -345,9 +243,8 @@ func TestNewReplyForInformList_NoImages(t *testing.T) { require.Error(t, err) } -// TODO (get9): clean up when #99 lands. func TestNewReplyForInformList(t *testing.T) { - inform := newInform() + inform, _ := NewInformList(net.HardwareAddr{1, 2, 3, 4, 5, 6}, net.IP{1, 2, 3, 4}, dhcpv4.ClientPort) images := []BootImage{ BootImage{ ID: BootImageID{ @@ -377,58 +274,100 @@ func TestNewReplyForInformList(t *testing.T) { require.NoError(t, err) require.Equal(t, net.IP{1, 2, 3, 4}, ack.ClientIPAddr()) require.Equal(t, net.IPv4zero, ack.YourIPAddr()) - require.Equal(t, net.IP{4, 3, 2, 1}, ack.GatewayIPAddr()) require.Equal(t, "bsdp.foo.com", ack.ServerHostNameToString()) // Validate options. - require.Equal( - t, - &dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}, - ack.GetOneOption(dhcpv4.OptionDHCPMessageType).(*dhcpv4.OptMessageType), - ) - require.Equal( - t, - &dhcpv4.OptServerIdentifier{ServerID: net.IP{9, 9, 9, 9}}, - ack.GetOneOption(dhcpv4.OptionServerIdentifier).(*dhcpv4.OptServerIdentifier), - ) - require.Equal( - t, - &dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}, - ack.GetOneOption(dhcpv4.OptionClassIdentifier).(*dhcpv4.OptClassIdentifier), - ) + RequireHasOption(t, ack, &dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}) + RequireHasOption(t, ack, &dhcpv4.OptServerIdentifier{ServerID: net.IP{9, 9, 9, 9}}) + RequireHasOption(t, ack, &dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}) require.NotNil(t, ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation)) + + // Ensure options terminated with End option. require.Equal(t, &dhcpv4.OptionGeneric{OptionCode: dhcpv4.OptionEnd}, ack.Options()[len(ack.Options())-1]) + // Vendor-specific options. vendorOpts := ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation).(*OptVendorSpecificInformation) - require.Equal( - t, - &OptMessageType{Type: MessageTypeList}, - vendorOpts.GetOneOption(OptionMessageType).(*OptMessageType), - ) - require.Equal( - t, - &OptServerPriority{Priority: 0x7070}, - vendorOpts.GetOneOption(OptionServerPriority).(*OptServerPriority), - ) - require.Equal( - t, - &OptDefaultBootImageID{ID: images[0].ID}, - vendorOpts.GetOneOption(OptionDefaultBootImageID).(*OptDefaultBootImageID), - ) - require.Equal( - t, - &OptBootImageList{Images: images}, - vendorOpts.GetOneOption(OptionBootImageList).(*OptBootImageList), - ) + RequireHasOption(t, vendorOpts, &OptMessageType{Type: MessageTypeList}) + RequireHasOption(t, vendorOpts, &OptDefaultBootImageID{ID: images[0].ID}) + RequireHasOption(t, vendorOpts, &OptServerPriority{Priority: 0x7070}) + RequireHasOption(t, vendorOpts, &OptBootImageList{Images: images}) // Add in selected boot image, ensure it's in the generated ACK. config.SelectedImage = &images[0] ack, err = NewReplyForInformList(inform, config) require.NoError(t, err) vendorOpts = ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation).(*OptVendorSpecificInformation) - require.Equal( - t, - &OptSelectedBootImageID{ID: images[0].ID}, - vendorOpts.GetOneOption(OptionSelectedBootImageID).(*OptSelectedBootImageID), - ) + RequireHasOption(t, vendorOpts, &OptSelectedBootImageID{ID: images[0].ID}) +} + +func TestNewReplyForInformSelect_NoSelectedImage(t *testing.T) { + inform, _ := NewInformList(net.HardwareAddr{1, 2, 3, 4, 5, 6}, net.IP{1, 2, 3, 4}, dhcpv4.ClientPort) + _, err := NewReplyForInformSelect(inform, ReplyConfig{}) + require.Error(t, err) +} + +func TestNewReplyForInformSelect_NoImages(t *testing.T) { + inform, _ := NewInformList(net.HardwareAddr{1, 2, 3, 4, 5, 6}, net.IP{1, 2, 3, 4}, dhcpv4.ClientPort) + fakeImage := BootImage{ + ID: BootImageID{ImageType: BootImageTypeMacOSX}, + } + _, err := NewReplyForInformSelect(inform, ReplyConfig{ + Images: []BootImage{}, + SelectedImage: &fakeImage, + }) + require.Error(t, err) + + _, err = NewReplyForInformSelect(inform, ReplyConfig{ + Images: nil, + SelectedImage: &fakeImage, + }) + require.Error(t, err) +} + +func TestNewReplyForInformSelect(t *testing.T) { + inform, _ := NewInformList(net.HardwareAddr{1, 2, 3, 4, 5, 6}, net.IP{1, 2, 3, 4}, dhcpv4.ClientPort) + images := []BootImage{ + BootImage{ + ID: BootImageID{ + IsInstall: true, + ImageType: BootImageTypeMacOSX, + Index: 0x7070, + }, + Name: "image-1", + }, + BootImage{ + ID: BootImageID{ + IsInstall: true, + ImageType: BootImageTypeMacOSX, + Index: 0x8080, + }, + Name: "image-2", + }, + } + config := ReplyConfig{ + Images: images, + SelectedImage: &images[0], + ServerIP: net.IP{9, 9, 9, 9}, + ServerHostname: "bsdp.foo.com", + ServerPriority: 0x7070, + } + ack, err := NewReplyForInformSelect(inform, config) + require.NoError(t, err) + require.Equal(t, net.IP{1, 2, 3, 4}, ack.ClientIPAddr()) + require.Equal(t, net.IPv4zero, ack.YourIPAddr()) + require.Equal(t, "bsdp.foo.com", ack.ServerHostNameToString()) + + // Validate options. + RequireHasOption(t, ack, &dhcpv4.OptMessageType{MessageType: dhcpv4.MessageTypeAck}) + RequireHasOption(t, ack, &dhcpv4.OptServerIdentifier{ServerID: net.IP{9, 9, 9, 9}}) + RequireHasOption(t, ack, &dhcpv4.OptServerIdentifier{ServerID: net.IP{9, 9, 9, 9}}) + RequireHasOption(t, ack, &dhcpv4.OptClassIdentifier{Identifier: AppleVendorID}) + require.NotNil(t, ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation)) + + // Ensure options are terminated with End option. + require.Equal(t, &dhcpv4.OptionGeneric{OptionCode: dhcpv4.OptionEnd}, ack.Options()[len(ack.Options())-1]) + + vendorOpts := ack.GetOneOption(dhcpv4.OptionVendorSpecificInformation).(*OptVendorSpecificInformation) + RequireHasOption(t, vendorOpts, &OptMessageType{Type: MessageTypeSelect}) + RequireHasOption(t, vendorOpts, &OptSelectedBootImageID{ID: images[0].ID}) } -- cgit v1.2.3