package job import ( "reflect" "testing" ) func TestMergeClientIps_EvictsStaleOldEntries(t *testing.T) { // #4077: after a ban expires, a single IP that reconnects used to get // banned again immediately because a long-disconnected IP stayed in the // DB with an ancient timestamp and kept "protecting" itself against // eviction. Guard against that regression here. old := []IPWithTimestamp{ {IP: "1.1.1.1", Timestamp: 100}, // stale — client disconnected long ago {IP: "2.2.2.2", Timestamp: 1900}, // fresh — still connecting } new := []IPWithTimestamp{ {IP: "2.2.2.2", Timestamp: 2000}, // same IP, newer log line } got := mergeClientIps(old, new, 1000) want := map[string]int64{"2.2.2.2": 2000} if !reflect.DeepEqual(got, want) { t.Fatalf("stale 1.1.1.1 should have been dropped\ngot: %v\nwant: %v", got, want) } } func TestMergeClientIps_KeepsFreshOldEntriesUnchanged(t *testing.T) { // Backwards-compat: entries that aren't stale are still carried forward, // so enforcement survives access-log rotation. old := []IPWithTimestamp{ {IP: "1.1.1.1", Timestamp: 1500}, } got := mergeClientIps(old, nil, 1000) want := map[string]int64{"1.1.1.1": 1500} if !reflect.DeepEqual(got, want) { t.Fatalf("fresh old IP should have been retained\ngot: %v\nwant: %v", got, want) } } func TestMergeClientIps_PrefersLaterTimestampForSameIp(t *testing.T) { old := []IPWithTimestamp{{IP: "1.1.1.1", Timestamp: 1500}} new := []IPWithTimestamp{{IP: "1.1.1.1", Timestamp: 1700}} got := mergeClientIps(old, new, 1000) if got["1.1.1.1"] != 1700 { t.Fatalf("expected latest timestamp 1700, got %d", got["1.1.1.1"]) } } func TestMergeClientIps_DropsStaleNewEntries(t *testing.T) { // A log line with a clock-skewed old timestamp must not resurrect a // stale IP past the cutoff. new := []IPWithTimestamp{{IP: "1.1.1.1", Timestamp: 500}} got := mergeClientIps(nil, new, 1000) if len(got) != 0 { t.Fatalf("stale new IP should have been dropped, got %v", got) } } func TestMergeClientIps_NoStaleCutoffStillWorks(t *testing.T) { // Defensive: a zero cutoff (e.g. during very first run on a fresh // install) must not over-evict. old := []IPWithTimestamp{{IP: "1.1.1.1", Timestamp: 100}} new := []IPWithTimestamp{{IP: "2.2.2.2", Timestamp: 200}} got := mergeClientIps(old, new, 0) want := map[string]int64{"1.1.1.1": 100, "2.2.2.2": 200} if !reflect.DeepEqual(got, want) { t.Fatalf("zero cutoff should keep everything\ngot: %v\nwant: %v", got, want) } } func collectIps(entries []IPWithTimestamp) []string { out := make([]string, 0, len(entries)) for _, e := range entries { out = append(out, e.IP) } return out } func TestPartitionLiveIps_SingleLiveNotStarvedByStillFreshHistoricals(t *testing.T) { // #4091: db holds A, B, C from minutes ago (still in the 30min // window) but they're not connecting anymore. only D is. old code // merged all four, sorted ascending, kept [A,B,C] and banned D // every tick. pin the new rule: only live ips count toward the limit. ipMap := map[string]int64{ "A": 1000, "B": 1100, "C": 1200, "D": 2000, } observed := map[string]bool{"D": true} live, historical := partitionLiveIps(ipMap, observed) if got := collectIps(live); !reflect.DeepEqual(got, []string{"D"}) { t.Fatalf("live set should only contain the ip observed this scan\ngot: %v\nwant: [D]", got) } if got := collectIps(historical); !reflect.DeepEqual(got, []string{"A", "B", "C"}) { t.Fatalf("historical set should contain db-only ips in ascending order\ngot: %v\nwant: [A B C]", got) } } func TestPartitionLiveIps_ConcurrentLiveIpsStillBanNewcomers(t *testing.T) { // keep the "protect original, ban newcomer" policy when several ips // are really live. with limit=1, A must stay and B must be banned. ipMap := map[string]int64{ "A": 5000, "B": 5500, } observed := map[string]bool{"A": true, "B": true} live, historical := partitionLiveIps(ipMap, observed) if got := collectIps(live); !reflect.DeepEqual(got, []string{"A", "B"}) { t.Fatalf("both live ips should be in the live set, ascending\ngot: %v\nwant: [A B]", got) } if len(historical) != 0 { t.Fatalf("no historical ips expected, got %v", historical) } } func TestPartitionLiveIps_EmptyScanLeavesDbIntact(t *testing.T) { // quiet tick: nothing observed => nothing live. everything merged // is historical. keeps the panel from wiping recent-but-idle ips. ipMap := map[string]int64{ "A": 1000, "B": 1100, } observed := map[string]bool{} live, historical := partitionLiveIps(ipMap, observed) if len(live) != 0 { t.Fatalf("no live ips expected, got %v", live) } if got := collectIps(historical); !reflect.DeepEqual(got, []string{"A", "B"}) { t.Fatalf("all merged entries should flow to historical\ngot: %v\nwant: [A B]", got) } }