2023-05-21 18:37:59 +02:00
|
|
|
package db
|
2021-08-02 22:57:45 +02:00
|
|
|
|
|
|
|
import (
|
2022-09-02 00:05:18 +02:00
|
|
|
"net/netip"
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
"github.com/juanfont/headscale/hscontrol/types"
|
|
|
|
"github.com/juanfont/headscale/hscontrol/util"
|
2022-09-02 00:05:18 +02:00
|
|
|
"go4.org/netipx"
|
2021-08-02 22:57:45 +02:00
|
|
|
"gopkg.in/check.v1"
|
|
|
|
)
|
|
|
|
|
|
|
|
func (s *Suite) TestGetAvailableIp(c *check.C) {
|
2023-05-21 18:37:59 +02:00
|
|
|
ips, err := db.getAvailableIPs()
|
2021-08-02 22:57:45 +02:00
|
|
|
|
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-09-02 00:05:18 +02:00
|
|
|
expected := netip.MustParseAddr("10.27.0.1")
|
2021-08-02 22:57:45 +02:00
|
|
|
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(ips), check.Equals, 1)
|
|
|
|
c.Assert(ips[0].String(), check.Equals, expected.String())
|
2021-08-02 22:57:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Suite) TestGetUsedIps(c *check.C) {
|
2023-05-21 18:37:59 +02:00
|
|
|
ips, err := db.getAvailableIPs()
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
user, err := db.CreateUser("test-ip")
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
pak, err := db.CreatePreAuthKey(user.Name, false, false, nil, nil)
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
_, err = db.GetMachine("test", "testmachine")
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.NotNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
machine := types.Machine{
|
2021-08-02 22:57:45 +02:00
|
|
|
ID: 0,
|
|
|
|
MachineKey: "foo",
|
|
|
|
NodeKey: "bar",
|
|
|
|
DiscoKey: "faa",
|
2022-06-14 11:33:01 +02:00
|
|
|
Hostname: "testmachine",
|
2023-01-17 20:36:46 +01:00
|
|
|
UserID: user.ID,
|
2023-05-21 18:37:59 +02:00
|
|
|
RegisterMethod: util.RegisterMethodAuthKey,
|
2021-08-02 22:57:45 +02:00
|
|
|
AuthKeyID: uint(pak.ID),
|
2022-01-16 14:16:59 +01:00
|
|
|
IPAddresses: ips,
|
2021-08-02 22:57:45 +02:00
|
|
|
}
|
2023-05-21 18:37:59 +02:00
|
|
|
db.db.Save(&machine)
|
2021-08-02 22:57:45 +02:00
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
usedIps, err := db.getUsedIPs()
|
2021-08-02 22:57:45 +02:00
|
|
|
|
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-09-02 00:05:18 +02:00
|
|
|
expected := netip.MustParseAddr("10.27.0.1")
|
|
|
|
expectedIPSetBuilder := netipx.IPSetBuilder{}
|
2022-02-25 09:28:22 +01:00
|
|
|
expectedIPSetBuilder.Add(expected)
|
|
|
|
expectedIPSet, _ := expectedIPSetBuilder.IPSet()
|
2021-08-02 22:57:45 +02:00
|
|
|
|
2022-02-25 09:28:22 +01:00
|
|
|
c.Assert(usedIps.Equal(expectedIPSet), check.Equals, true)
|
|
|
|
c.Assert(usedIps.Contains(expected), check.Equals, true)
|
2021-08-02 23:39:18 +02:00
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
machine1, err := db.GetMachineByID(0)
|
2021-08-02 23:39:18 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(machine1.IPAddresses), check.Equals, 1)
|
|
|
|
c.Assert(machine1.IPAddresses[0], check.Equals, expected)
|
2023-05-21 18:37:59 +02:00
|
|
|
|
|
|
|
c.Assert(channelUpdates, check.Equals, int32(0))
|
2021-08-02 22:57:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Suite) TestGetMultiIp(c *check.C) {
|
2023-05-21 18:37:59 +02:00
|
|
|
user, err := db.CreateUser("test-ip-multi")
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2021-11-15 17:16:04 +01:00
|
|
|
for index := 1; index <= 350; index++ {
|
2023-05-21 18:37:59 +02:00
|
|
|
db.ipAllocationMutex.Lock()
|
2022-02-25 09:28:22 +01:00
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
ips, err := db.getAvailableIPs()
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
pak, err := db.CreatePreAuthKey(user.Name, false, false, nil, nil)
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
_, err = db.GetMachine("test", "testmachine")
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.NotNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
machine := types.Machine{
|
2021-11-15 17:16:04 +01:00
|
|
|
ID: uint64(index),
|
2021-08-02 22:57:45 +02:00
|
|
|
MachineKey: "foo",
|
|
|
|
NodeKey: "bar",
|
|
|
|
DiscoKey: "faa",
|
2022-06-14 11:33:01 +02:00
|
|
|
Hostname: "testmachine",
|
2023-01-17 20:36:46 +01:00
|
|
|
UserID: user.ID,
|
2023-05-21 18:37:59 +02:00
|
|
|
RegisterMethod: util.RegisterMethodAuthKey,
|
2021-08-02 22:57:45 +02:00
|
|
|
AuthKeyID: uint(pak.ID),
|
2022-01-16 14:16:59 +01:00
|
|
|
IPAddresses: ips,
|
2021-08-02 22:57:45 +02:00
|
|
|
}
|
2023-05-21 18:37:59 +02:00
|
|
|
db.db.Save(&machine)
|
2022-02-25 09:28:22 +01:00
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
db.ipAllocationMutex.Unlock()
|
2021-08-02 22:57:45 +02:00
|
|
|
}
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
usedIps, err := db.getUsedIPs()
|
2022-02-25 09:28:22 +01:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-09-02 00:05:18 +02:00
|
|
|
expected0 := netip.MustParseAddr("10.27.0.1")
|
|
|
|
expected9 := netip.MustParseAddr("10.27.0.10")
|
|
|
|
expected300 := netip.MustParseAddr("10.27.0.45")
|
2021-08-02 22:57:45 +02:00
|
|
|
|
2022-09-02 00:05:18 +02:00
|
|
|
notExpectedIPSetBuilder := netipx.IPSetBuilder{}
|
2022-02-25 09:28:22 +01:00
|
|
|
notExpectedIPSetBuilder.Add(expected0)
|
|
|
|
notExpectedIPSetBuilder.Add(expected9)
|
|
|
|
notExpectedIPSetBuilder.Add(expected300)
|
|
|
|
notExpectedIPSet, err := notExpectedIPSetBuilder.IPSet()
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-02-25 09:28:22 +01:00
|
|
|
// We actually expect it to be a lot larger
|
|
|
|
c.Assert(usedIps.Equal(notExpectedIPSet), check.Equals, false)
|
2021-08-02 22:57:45 +02:00
|
|
|
|
2022-02-25 09:28:22 +01:00
|
|
|
c.Assert(usedIps.Contains(expected0), check.Equals, true)
|
|
|
|
c.Assert(usedIps.Contains(expected9), check.Equals, true)
|
|
|
|
c.Assert(usedIps.Contains(expected300), check.Equals, true)
|
2021-08-02 22:57:45 +02:00
|
|
|
|
2021-08-02 23:39:18 +02:00
|
|
|
// Check that we can read back the IPs
|
2023-05-21 18:37:59 +02:00
|
|
|
machine1, err := db.GetMachineByID(1)
|
2021-08-02 23:39:18 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(machine1.IPAddresses), check.Equals, 1)
|
2021-11-15 17:16:04 +01:00
|
|
|
c.Assert(
|
2022-01-16 14:16:59 +01:00
|
|
|
machine1.IPAddresses[0],
|
2021-11-15 17:16:04 +01:00
|
|
|
check.Equals,
|
2022-09-02 00:05:18 +02:00
|
|
|
netip.MustParseAddr("10.27.0.1"),
|
2021-11-15 17:16:04 +01:00
|
|
|
)
|
2021-08-02 23:39:18 +02:00
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
machine50, err := db.GetMachineByID(50)
|
2021-08-02 23:39:18 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(machine50.IPAddresses), check.Equals, 1)
|
2021-11-15 17:16:04 +01:00
|
|
|
c.Assert(
|
2022-01-16 14:16:59 +01:00
|
|
|
machine50.IPAddresses[0],
|
2021-11-15 17:16:04 +01:00
|
|
|
check.Equals,
|
2022-09-02 00:05:18 +02:00
|
|
|
netip.MustParseAddr("10.27.0.50"),
|
2021-11-15 17:16:04 +01:00
|
|
|
)
|
2021-08-02 23:39:18 +02:00
|
|
|
|
2022-09-02 00:05:18 +02:00
|
|
|
expectedNextIP := netip.MustParseAddr("10.27.1.95")
|
2023-05-21 18:37:59 +02:00
|
|
|
nextIP, err := db.getAvailableIPs()
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(nextIP), check.Equals, 1)
|
|
|
|
c.Assert(nextIP[0].String(), check.Equals, expectedNextIP.String())
|
2021-08-02 22:57:45 +02:00
|
|
|
|
|
|
|
// If we call get Available again, we should receive
|
|
|
|
// the same IP, as it has not been reserved.
|
2023-05-21 18:37:59 +02:00
|
|
|
nextIP2, err := db.getAvailableIPs()
|
2021-08-02 22:57:45 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(nextIP2), check.Equals, 1)
|
|
|
|
c.Assert(nextIP2[0].String(), check.Equals, expectedNextIP.String())
|
2023-05-21 18:37:59 +02:00
|
|
|
|
|
|
|
c.Assert(channelUpdates, check.Equals, int32(0))
|
2021-08-02 22:57:45 +02:00
|
|
|
}
|
2021-08-03 10:26:28 +02:00
|
|
|
|
|
|
|
func (s *Suite) TestGetAvailableIpMachineWithoutIP(c *check.C) {
|
2023-05-21 18:37:59 +02:00
|
|
|
ips, err := db.getAvailableIPs()
|
2021-08-03 10:26:28 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-09-03 23:46:14 +02:00
|
|
|
expected := netip.MustParseAddr("10.27.0.1")
|
2021-08-03 10:26:28 +02:00
|
|
|
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(ips), check.Equals, 1)
|
|
|
|
c.Assert(ips[0].String(), check.Equals, expected.String())
|
2021-08-03 10:26:28 +02:00
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
user, err := db.CreateUser("test-ip")
|
2021-08-03 10:26:28 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
pak, err := db.CreatePreAuthKey(user.Name, false, false, nil, nil)
|
2021-08-03 10:26:28 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
_, err = db.GetMachine("test", "testmachine")
|
2021-08-03 10:26:28 +02:00
|
|
|
c.Assert(err, check.NotNil)
|
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
machine := types.Machine{
|
2021-08-03 10:26:28 +02:00
|
|
|
ID: 0,
|
|
|
|
MachineKey: "foo",
|
|
|
|
NodeKey: "bar",
|
|
|
|
DiscoKey: "faa",
|
2022-06-14 11:33:01 +02:00
|
|
|
Hostname: "testmachine",
|
2023-01-17 20:36:46 +01:00
|
|
|
UserID: user.ID,
|
2023-05-21 18:37:59 +02:00
|
|
|
RegisterMethod: util.RegisterMethodAuthKey,
|
2021-08-03 10:26:28 +02:00
|
|
|
AuthKeyID: uint(pak.ID),
|
|
|
|
}
|
2023-05-21 18:37:59 +02:00
|
|
|
db.db.Save(&machine)
|
2021-08-03 10:26:28 +02:00
|
|
|
|
2023-05-21 18:37:59 +02:00
|
|
|
ips2, err := db.getAvailableIPs()
|
2021-08-03 10:26:28 +02:00
|
|
|
c.Assert(err, check.IsNil)
|
|
|
|
|
2022-01-16 14:16:59 +01:00
|
|
|
c.Assert(len(ips2), check.Equals, 1)
|
|
|
|
c.Assert(ips2[0].String(), check.Equals, expected.String())
|
2023-05-21 18:37:59 +02:00
|
|
|
|
|
|
|
c.Assert(channelUpdates, check.Equals, int32(0))
|
2021-08-03 10:26:28 +02:00
|
|
|
}
|