plugin_test.go 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. // Copyright 2018-present the CoreDHCP Authors. All rights reserved
  2. // This source code is licensed under the MIT license found in the
  3. // LICENSE file in the root directory of this source tree.
  4. package rangeplugin
  5. import (
  6. "fmt"
  7. "net"
  8. "testing"
  9. "github.com/insomniacslk/dhcp/dhcpv4"
  10. "github.com/stretchr/testify/assert"
  11. "github.com/stretchr/testify/mock"
  12. )
  13. // mockAllocator is a simple mock for testing
  14. type mockAllocator struct {
  15. mock.Mock
  16. }
  17. func (m *mockAllocator) Allocate(hint net.IPNet) (net.IPNet, error) {
  18. return m.Called(hint).Get(0).(net.IPNet), nil
  19. }
  20. func (m *mockAllocator) Free(ip net.IPNet) error {
  21. m.Called(ip)
  22. return nil
  23. }
  24. type mockFailingAllocator struct {
  25. mock.Mock
  26. }
  27. func (m *mockFailingAllocator) Allocate(hint net.IPNet) (net.IPNet, error) {
  28. args := m.Called(hint)
  29. return args.Get(0).(net.IPNet), args.Error(1)
  30. }
  31. func (m *mockFailingAllocator) Free(ip net.IPNet) error {
  32. args := m.Called(ip)
  33. return args.Error(0)
  34. }
  35. func TestHandler4Release(t *testing.T) {
  36. db, dbErr := testDBSetup()
  37. if dbErr != nil {
  38. t.Fatalf("Failed to set up test DB: %v", dbErr)
  39. }
  40. mockAlloc := &mockAllocator{}
  41. pl := PluginState{
  42. leasedb: db,
  43. Recordsv4: make(map[string]*Record),
  44. allocator: mockAlloc,
  45. }
  46. loadedRecords, loadErr := loadRecords(db)
  47. if loadErr != nil {
  48. t.Fatalf("Failed to load records: %v", loadErr)
  49. }
  50. pl.Recordsv4 = loadedRecords
  51. // Create a DHCP RELEASE request using existing test data
  52. hwaddr, _ := net.ParseMAC(records[1].mac)
  53. req := &dhcpv4.DHCPv4{
  54. ClientHWAddr: hwaddr,
  55. }
  56. req.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeRelease))
  57. resp := &dhcpv4.DHCPv4{}
  58. // Verify record exists before release
  59. record, exists := pl.Recordsv4[hwaddr.String()]
  60. assert.True(t, exists, "Record should exist before release")
  61. expectedIPNet := net.IPNet{IP: record.IP}
  62. mockAlloc.On("Free", expectedIPNet).Return(nil)
  63. // Call Handler4 with RELEASE message
  64. result, stop := pl.Handler4(req, resp)
  65. assert.Nil(t, result, "Should return nil response for RELEASE")
  66. assert.True(t, stop, "Should return true to stop processing")
  67. _, exists = pl.Recordsv4[hwaddr.String()]
  68. assert.False(t, exists, "Record should be removed from memory after release")
  69. parsedRecords, parseErr := loadRecords(pl.leasedb)
  70. if parseErr != nil {
  71. t.Fatalf("Failed to load records after release: %v", parseErr)
  72. }
  73. _, exists = parsedRecords[hwaddr.String()]
  74. assert.False(t, exists, "Record should be removed from storage after release")
  75. mockAlloc.AssertExpectations(t)
  76. mockAlloc.AssertNotCalled(t, "Allocate")
  77. }
  78. func TestHandler4ReleaseAllocatorError(t *testing.T) {
  79. db, parseErr := testDBSetup()
  80. if parseErr != nil {
  81. t.Fatalf("Failed to set up test DB: %v", parseErr)
  82. }
  83. mockAlloc := &mockFailingAllocator{}
  84. pl := PluginState{
  85. leasedb: db,
  86. Recordsv4: make(map[string]*Record),
  87. allocator: mockAlloc,
  88. }
  89. loadedRecords, err := loadRecords(db)
  90. if err != nil {
  91. t.Fatalf("Failed to load records: %v", err)
  92. }
  93. pl.Recordsv4 = loadedRecords
  94. hwaddr, _ := net.ParseMAC(records[1].mac)
  95. req := &dhcpv4.DHCPv4{
  96. ClientHWAddr: hwaddr,
  97. }
  98. req.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeRelease))
  99. resp := &dhcpv4.DHCPv4{}
  100. record := pl.Recordsv4[hwaddr.String()]
  101. expectedIPNet := net.IPNet{IP: record.IP}
  102. expectedError := fmt.Errorf("mock allocator free failure")
  103. mockAlloc.On("Free", expectedIPNet).Return(expectedError)
  104. // Call Handler4 - this should fail on allocator.Free()
  105. result, stop := pl.Handler4(req, resp)
  106. assert.Nil(t, result, "Should return nil on allocator failure")
  107. assert.True(t, stop, "Should stop processing on allocator failure")
  108. _, exists := pl.Recordsv4[hwaddr.String()]
  109. assert.False(t, exists, "Record should be removed from memory even on allocator failure")
  110. parsedRecords, parseErr := loadRecords(pl.leasedb)
  111. if parseErr != nil {
  112. t.Fatalf("Failed to load records after release: %v", parseErr)
  113. }
  114. _, exists = parsedRecords[hwaddr.String()]
  115. assert.False(t, exists, "Record should be removed from storage even on allocator failure")
  116. mockAlloc.AssertExpectations(t)
  117. mockAlloc.AssertNotCalled(t, "Allocate")
  118. }
  119. func TestHandler4ReleaseStorageError(t *testing.T) {
  120. db, parseErr := testDBSetup()
  121. if parseErr != nil {
  122. t.Fatalf("Failed to set up test DB: %v", parseErr)
  123. }
  124. mockAlloc := &mockAllocator{}
  125. pl := PluginState{
  126. leasedb: db,
  127. Recordsv4: make(map[string]*Record),
  128. allocator: mockAlloc,
  129. }
  130. loadedRecords, err := loadRecords(db)
  131. if err != nil {
  132. t.Fatalf("Failed to load records: %v", err)
  133. }
  134. pl.Recordsv4 = loadedRecords
  135. hwaddr, _ := net.ParseMAC(records[1].mac)
  136. req := &dhcpv4.DHCPv4{
  137. ClientHWAddr: hwaddr,
  138. }
  139. req.UpdateOption(dhcpv4.OptMessageType(dhcpv4.MessageTypeRelease))
  140. resp := &dhcpv4.DHCPv4{}
  141. // Close the database to simulate storage failure
  142. db.Close()
  143. result, stop := pl.Handler4(req, resp)
  144. assert.Nil(t, result, "Should return nil on storage failure")
  145. assert.True(t, stop, "Should stop processing on storage failure")
  146. _, exists := pl.Recordsv4[hwaddr.String()]
  147. assert.True(t, exists, "Record should still exist in memory after storage failure")
  148. mockAlloc.AssertNotCalled(t, "Free")
  149. mockAlloc.AssertNotCalled(t, "Allocate")
  150. }