mirror of
https://github.com/tailscale/tailscale.git
synced 2025-06-08 08:48:35 +00:00
186 lines
4.9 KiB
Go
186 lines
4.9 KiB
Go
package ippool
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/netip"
|
|
"testing"
|
|
"time"
|
|
|
|
"github.com/hashicorp/raft"
|
|
"go4.org/netipx"
|
|
"tailscale.com/tailcfg"
|
|
"tailscale.com/tsconsensus"
|
|
"tailscale.com/util/must"
|
|
)
|
|
|
|
func makeSetFromPrefix(pfx netip.Prefix) *netipx.IPSet {
|
|
var ipsb netipx.IPSetBuilder
|
|
ipsb.AddPrefix(pfx)
|
|
return must.Get(ipsb.IPSet())
|
|
}
|
|
|
|
type FakeConsensus struct {
|
|
ipp *ConsensusIPPool
|
|
}
|
|
|
|
func (c *FakeConsensus) ExecuteCommand(cmd tsconsensus.Command) (tsconsensus.CommandResult, error) {
|
|
b, err := json.Marshal(cmd)
|
|
if err != nil {
|
|
return tsconsensus.CommandResult{}, err
|
|
}
|
|
result := c.ipp.Apply(&raft.Log{Data: b})
|
|
return result.(tsconsensus.CommandResult), nil
|
|
}
|
|
|
|
func makePool(pfx netip.Prefix) *ConsensusIPPool {
|
|
ipp := &ConsensusIPPool{
|
|
IPSet: makeSetFromPrefix(pfx),
|
|
}
|
|
ipp.consensus = &FakeConsensus{ipp: ipp}
|
|
return ipp
|
|
}
|
|
|
|
func TestConsensusIPForDomain(t *testing.T) {
|
|
pfx := netip.MustParsePrefix("100.64.0.0/16")
|
|
ipp := makePool(pfx)
|
|
from := tailcfg.NodeID(1)
|
|
|
|
a, err := ipp.IPForDomain(from, "example.com")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !pfx.Contains(a) {
|
|
t.Fatalf("expected %v to be in the prefix %v", a, pfx)
|
|
}
|
|
|
|
b, err := ipp.IPForDomain(from, "a.example.com")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if !pfx.Contains(b) {
|
|
t.Fatalf("expected %v to be in the prefix %v", b, pfx)
|
|
}
|
|
if b == a {
|
|
t.Fatalf("same address issued twice %v, %v", a, b)
|
|
}
|
|
|
|
c, err := ipp.IPForDomain(from, "example.com")
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if c != a {
|
|
t.Fatalf("expected %v to be remembered as the addr for example.com, but got %v", a, c)
|
|
}
|
|
}
|
|
|
|
func TestConsensusPoolExhaustion(t *testing.T) {
|
|
ipp := makePool(netip.MustParsePrefix("100.64.0.0/31"))
|
|
from := tailcfg.NodeID(1)
|
|
|
|
subdomains := []string{"a", "b", "c"}
|
|
for i, sd := range subdomains {
|
|
_, err := ipp.IPForDomain(from, fmt.Sprintf("%s.example.com", sd))
|
|
if i < 2 && err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
expected := "ip pool exhausted"
|
|
if i == 2 && err.Error() != expected {
|
|
t.Fatalf("expected error to be '%s', got '%s'", expected, err.Error())
|
|
}
|
|
}
|
|
}
|
|
|
|
func TestConsensusPoolExpiry(t *testing.T) {
|
|
ipp := makePool(netip.MustParsePrefix("100.64.0.0/31"))
|
|
firstIP := netip.MustParseAddr("100.64.0.0")
|
|
secondIP := netip.MustParseAddr("100.64.0.1")
|
|
timeOfUse := time.Now()
|
|
beforeTimeOfUse := timeOfUse.Add(-2 * time.Hour)
|
|
afterTimeOfUse := timeOfUse.Add(2 * time.Hour)
|
|
from := tailcfg.NodeID(1)
|
|
|
|
// the pool is unused, we get an address, and it's marked as being used at timeOfUse
|
|
aAddr, err := ipp.applyCheckoutAddr(from, "a.example.com", timeOfUse, timeOfUse)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if aAddr.Compare(firstIP) != 0 {
|
|
t.Fatalf("expected %s, got %s", firstIP, aAddr)
|
|
}
|
|
d, ok := ipp.DomainForIP(from, firstIP, timeOfUse)
|
|
if !ok {
|
|
t.Fatal("expected addr to be found")
|
|
}
|
|
if d != "a.example.com" {
|
|
t.Fatalf("expected aAddr to look up to a.example.com, got: %s", d)
|
|
}
|
|
|
|
// the time before which we will reuse addresses is prior to timeOfUse, so no reuse
|
|
bAddr, err := ipp.applyCheckoutAddr(from, "b.example.com", beforeTimeOfUse, timeOfUse)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if bAddr.Compare(secondIP) != 0 {
|
|
t.Fatalf("expected %s, got %s", secondIP, bAddr)
|
|
}
|
|
|
|
// the time before which we will reuse addresses is after timeOfUse, so reuse addresses that were marked as used at timeOfUse.
|
|
cAddr, err := ipp.applyCheckoutAddr(from, "c.example.com", afterTimeOfUse, timeOfUse)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if cAddr.Compare(firstIP) != 0 {
|
|
t.Fatalf("expected %s, got %s", firstIP, cAddr)
|
|
}
|
|
d, ok = ipp.DomainForIP(from, firstIP, timeOfUse)
|
|
if !ok {
|
|
t.Fatal("expected addr to be found")
|
|
}
|
|
if d != "c.example.com" {
|
|
t.Fatalf("expected firstIP to look up to c.example.com, got: %s", d)
|
|
}
|
|
|
|
// the addr remains associated with c.example.com
|
|
cAddrAgain, err := ipp.applyCheckoutAddr(from, "c.example.com", beforeTimeOfUse, timeOfUse)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
if cAddrAgain.Compare(cAddr) != 0 {
|
|
t.Fatalf("expected cAddrAgain to be cAddr, but they are different. cAddrAgain=%s cAddr=%s", cAddrAgain, cAddr)
|
|
}
|
|
d, ok = ipp.DomainForIP(from, firstIP, timeOfUse)
|
|
if !ok {
|
|
t.Fatal("expected addr to be found")
|
|
}
|
|
if d != "c.example.com" {
|
|
t.Fatalf("expected firstIP to look up to c.example.com, got: %s", d)
|
|
}
|
|
}
|
|
|
|
func TestConsensusDomainForIP(t *testing.T) {
|
|
ipp := makePool(netip.MustParsePrefix("100.64.0.0/16"))
|
|
from := tailcfg.NodeID(1)
|
|
domain := "example.com"
|
|
now := time.Now()
|
|
|
|
d, ok := ipp.DomainForIP(from, netip.MustParseAddr("100.64.0.1"), now)
|
|
if d != "" {
|
|
t.Fatalf("expected an empty string if the addr is not found but got %s", d)
|
|
}
|
|
if ok {
|
|
t.Fatalf("expected domain to not be found for IP, as it has never been looked up")
|
|
}
|
|
a, err := ipp.IPForDomain(from, domain)
|
|
if err != nil {
|
|
t.Fatal(err)
|
|
}
|
|
d2, ok := ipp.DomainForIP(from, a, now)
|
|
if d2 != domain {
|
|
t.Fatalf("expected %s but got %s", domain, d2)
|
|
}
|
|
if !ok {
|
|
t.Fatalf("expected domain to be found for IP that was handed out for it")
|
|
}
|
|
}
|