storage_test.go 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241
  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. "database/sql"
  7. "fmt"
  8. "net"
  9. "testing"
  10. "time"
  11. "github.com/stretchr/testify/assert"
  12. )
  13. func testDBSetup() (*sql.DB, error) {
  14. db, err := loadDB(":memory:")
  15. if err != nil {
  16. return nil, err
  17. }
  18. for _, record := range records {
  19. stmt, err := db.Prepare("insert into leases4(mac, ip, expiry, hostname) values (?, ?, ?, ?)")
  20. if err != nil {
  21. return nil, fmt.Errorf("failed to prepare insert statement: %w", err)
  22. }
  23. defer stmt.Close()
  24. if _, err := stmt.Exec(record.mac, record.ip.IP.String(), record.ip.expires, record.ip.hostname); err != nil {
  25. return nil, fmt.Errorf("failed to insert record into test db: %w", err)
  26. }
  27. }
  28. return db, nil
  29. }
  30. var expire = int(time.Date(2000, 01, 01, 00, 00, 00, 00, time.UTC).Unix())
  31. var records = []struct {
  32. mac string
  33. ip *Record
  34. }{
  35. {"02:00:00:00:00:00", &Record{IP: net.IPv4(10, 0, 0, 0), expires: expire, hostname: "zero"}},
  36. {"02:00:00:00:00:01", &Record{IP: net.IPv4(10, 0, 0, 1), expires: expire, hostname: "one"}},
  37. {"02:00:00:00:00:02", &Record{IP: net.IPv4(10, 0, 0, 2), expires: expire, hostname: "two"}},
  38. {"02:00:00:00:00:03", &Record{IP: net.IPv4(10, 0, 0, 3), expires: expire, hostname: "three"}},
  39. {"02:00:00:00:00:04", &Record{IP: net.IPv4(10, 0, 0, 4), expires: expire, hostname: "four"}},
  40. {"02:00:00:00:00:05", &Record{IP: net.IPv4(10, 0, 0, 5), expires: expire, hostname: "five"}},
  41. }
  42. func TestLoadRecords(t *testing.T) {
  43. db, err := testDBSetup()
  44. if err != nil {
  45. t.Fatalf("Failed to set up test DB: %v", err)
  46. }
  47. parsedRec, err := loadRecords(db)
  48. if err != nil {
  49. t.Fatalf("Failed to load records from file: %v", err)
  50. }
  51. mapRec := make(map[string]*Record)
  52. for _, rec := range records {
  53. var (
  54. ip, mac, hostname string
  55. expiry int
  56. )
  57. if err := db.QueryRow("select mac, ip, expiry, hostname from leases4 where mac = ?", rec.mac).Scan(&mac, &ip, &expiry, &hostname); err != nil {
  58. t.Fatalf("record not found for mac=%s: %v", rec.mac, err)
  59. }
  60. mapRec[mac] = &Record{IP: net.ParseIP(ip), expires: expiry, hostname: hostname}
  61. }
  62. assert.Equal(t, mapRec, parsedRec, "Loaded records differ from what's in the DB")
  63. }
  64. func TestWriteRecords(t *testing.T) {
  65. pl := PluginState{}
  66. if err := pl.registerBackingDB(":memory:"); err != nil {
  67. t.Fatalf("Could not setup file")
  68. }
  69. mapRec := make(map[string]*Record)
  70. for _, rec := range records {
  71. hwaddr, err := net.ParseMAC(rec.mac)
  72. if err != nil {
  73. // bug in testdata
  74. panic(err)
  75. }
  76. if err := pl.saveIPAddress(hwaddr, rec.ip); err != nil {
  77. t.Errorf("Failed to save ip for %s: %v", hwaddr, err)
  78. }
  79. mapRec[hwaddr.String()] = &Record{IP: rec.ip.IP, expires: rec.ip.expires, hostname: rec.ip.hostname}
  80. }
  81. parsedRec, err := loadRecords(pl.leasedb)
  82. if err != nil {
  83. t.Fatal(err)
  84. }
  85. assert.Equal(t, mapRec, parsedRec, "Loaded records differ from what's in the DB")
  86. }
  87. func TestFreeIPAddress(t *testing.T) {
  88. db, err := testDBSetup()
  89. if err != nil {
  90. t.Fatalf("Failed to set up test DB: %v", err)
  91. }
  92. pl := PluginState{leasedb: db}
  93. hwaddr, err := net.ParseMAC(records[1].mac)
  94. if err != nil {
  95. t.Fatalf("Failed to parse MAC address: %v", err)
  96. }
  97. record := records[1].ip
  98. parsedRecords, err := loadRecords(pl.leasedb)
  99. if err != nil {
  100. t.Fatalf("Failed to load records: %v", err)
  101. }
  102. _, exists := parsedRecords[hwaddr.String()]
  103. assert.True(t, exists, "Record should exist before deletion")
  104. // Now free the IP address
  105. if err := pl.freeIPAddress(hwaddr, record); err != nil {
  106. t.Errorf("Failed to free IP address: %v", err)
  107. }
  108. parsedRecords, err = loadRecords(pl.leasedb)
  109. if err != nil {
  110. t.Fatalf("Failed to load records after deletion: %v", err)
  111. }
  112. _, exists = parsedRecords[hwaddr.String()]
  113. assert.False(t, exists, "Record should not exist after deletion")
  114. }
  115. func TestFreeIPAddressNonExistent(t *testing.T) {
  116. pl := PluginState{}
  117. if err := pl.registerBackingDB(":memory:"); err != nil {
  118. t.Fatalf("Could not setup file")
  119. }
  120. hwaddr, err := net.ParseMAC("02:00:00:00:00:99")
  121. if err != nil {
  122. t.Fatalf("Failed to parse MAC address: %v", err)
  123. }
  124. record := &Record{
  125. IP: net.IPv4(10, 0, 0, 99),
  126. expires: expire,
  127. hostname: "non-existent",
  128. }
  129. err = pl.freeIPAddress(hwaddr, record)
  130. assert.NoError(t, err, "Freeing a non-existent IP address should not return an error")
  131. parsedRecords, err := loadRecords(pl.leasedb)
  132. if err != nil {
  133. t.Fatalf("Failed to load records: %v", err)
  134. }
  135. assert.Empty(t, parsedRecords, "Database should be empty")
  136. }
  137. func TestFreeIPAddressVerifyDeletion(t *testing.T) {
  138. db, err := testDBSetup()
  139. if err != nil {
  140. t.Fatalf("Failed to set up test DB: %v", err)
  141. }
  142. pl := PluginState{leasedb: db}
  143. parsedRecords, err := loadRecords(pl.leasedb)
  144. if err != nil {
  145. t.Fatalf("Failed to load records: %v", err)
  146. }
  147. assert.Len(t, parsedRecords, 6, "Should have 6 records from testDBSetup")
  148. // Delete the middle record (records[2] = "02:00:00:00:00:02" with IP 10.0.0.2)
  149. hwaddrToDelete, _ := net.ParseMAC(records[2].mac)
  150. recordToDelete := records[2].ip
  151. if err := pl.freeIPAddress(hwaddrToDelete, recordToDelete); err != nil {
  152. t.Errorf("Failed to free IP address: %v", err)
  153. }
  154. parsedRecords, err = loadRecords(pl.leasedb)
  155. if err != nil {
  156. t.Fatalf("Failed to load records after deletion: %v", err)
  157. }
  158. assert.Len(t, parsedRecords, 5, "Should have 5 records after deletion")
  159. _, exists := parsedRecords[hwaddrToDelete.String()]
  160. assert.False(t, exists, "Deleted record should not exist")
  161. // Verify some other records still exist
  162. otherMacs := []string{records[1].mac, records[3].mac}
  163. for _, mac := range otherMacs {
  164. _, exists := parsedRecords[mac]
  165. assert.True(t, exists, "Other records should still exist: %s", mac)
  166. }
  167. }
  168. func TestFreeIPAddressExecutionError(t *testing.T) {
  169. // This test triggers a statement execution failure using a SQLite trigger
  170. // that aborts DELETE operations for records[0]
  171. db, err := testDBSetup()
  172. if err != nil {
  173. t.Fatalf("Failed to set up test database: %v", err)
  174. }
  175. defer db.Close()
  176. const triggerErrorMsg = "Custom deletion prevention trigger"
  177. // Create a trigger that will cause DELETE operations to fail for records[0]
  178. triggerSQL := fmt.Sprintf(`
  179. CREATE TRIGGER prevent_delete
  180. BEFORE DELETE ON leases4
  181. WHEN OLD.mac = '%s'
  182. BEGIN
  183. SELECT RAISE(ABORT, '%s');
  184. END
  185. `, records[0].mac, triggerErrorMsg)
  186. _, err = db.Exec(triggerSQL)
  187. if err != nil {
  188. t.Fatalf("Failed to create trigger: %v", err)
  189. }
  190. pl := PluginState{leasedb: db}
  191. hwaddr, err := net.ParseMAC(records[0].mac)
  192. if err != nil {
  193. t.Fatalf("Failed to parse MAC address: %v", err)
  194. }
  195. record := records[0].ip
  196. err = pl.freeIPAddress(hwaddr, record)
  197. assert.Error(t, err, "Should return error due to trigger preventing deletion")
  198. assert.Contains(t, err.Error(), "record delete failed", "Error should indicate record delete failure")
  199. assert.Contains(t, err.Error(), triggerErrorMsg, "Error should contain trigger message")
  200. }