ip-algo: rename pod field to app; image from pod spec
Build flock Image / build (push) Has been cancelled

The `pod` field hashed pod.Name, which differs per replica because of
the ReplicaSet pod-template-hash + 5-char random suffix. With
namespace,pod,image, all replicas of the same Deployment got distinct
hextets even though they were the same workload.

Replace `pod` with `app` — a stable workload identifier derived from
the controller chain:

  - Deployment → ReplicaSet → Pod: strip the pod-template-hash suffix
    from the RS name (`traefik-789df685f` → `traefik`).
  - StatefulSet/DaemonSet/Job → Pod: use controller name as-is.
  - Bare pod: pod name.

Image now comes from pod.Spec.Containers[0].Image (the spec'd
reference). 64-hex-char values are treated as sha256 digests and
parsed as before; everything else (image:tag, short SHA) is FNV-1a-64'd
as a string. This makes `traefik:v3.5` deterministic across replicas
without needing the runtime-resolved digest.

Net effect: namespace,app,image yields identical hextets across all
replicas of the same Deployment except the trailing random N nibble.

embed.Values.Pod → App; AllocRequest.Pod kept for log context only,
new App and Image fields drive the embed call. handlers.go computes
both via deriveAppName + podImageRef helpers.

Tests: 7 new TestDeriveAppName_* cases (Deploy/STS/DS/bare/RS-without-
hash/non-controller-owner) + TestPodImageRef. Existing fuzz seeds
updated for the new keyword.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
Donavan Fritz
2026-04-25 11:42:06 -05:00
parent c860e9351b
commit 65b2fb5b17
9 changed files with 284 additions and 56 deletions
+2 -2
View File
@@ -299,8 +299,8 @@ func tryParseIPAlgo(s string) []embed.Field {
switch part {
case string(embed.FieldNamespace):
f = embed.FieldNamespace
case string(embed.FieldPod):
f = embed.FieldPod
case string(embed.FieldApp):
f = embed.FieldApp
case string(embed.FieldImage):
f = embed.FieldImage
default:
+7 -7
View File
@@ -179,10 +179,10 @@ func TestParseAnnotations_BoolCaseInsensitive(t *testing.T) {
// "all three mean unset".
func TestResolveIPAlgo_PodWins(t *testing.T) {
pod := map[string]string{annotationPrefix + annIPAlgo: "namespace,pod"}
pod := map[string]string{annotationPrefix + annIPAlgo: "namespace,app"}
node := map[string]string{annotationPrefix + annIPAlgo: "image"}
got := ResolveIPAlgo(pod, node, nil)
want := []embed.Field{embed.FieldNamespace, embed.FieldPod}
want := []embed.Field{embed.FieldNamespace, embed.FieldApp}
if !equalFields(got, want) {
t.Fatalf("got %v, want %v", got, want)
}
@@ -210,9 +210,9 @@ func TestResolveIPAlgo_PodEmptyFallsToNode(t *testing.T) {
func TestResolveIPAlgo_PodInvalidFallsToNode(t *testing.T) {
for _, podVal := range []string{"namespace,bogus", "ns", ",", "namespace,namespace"} {
pod := map[string]string{annotationPrefix + annIPAlgo: podVal}
node := map[string]string{annotationPrefix + annIPAlgo: "pod"}
node := map[string]string{annotationPrefix + annIPAlgo: "app"}
got := ResolveIPAlgo(pod, node, nil)
want := []embed.Field{embed.FieldPod}
want := []embed.Field{embed.FieldApp}
if !equalFields(got, want) {
t.Fatalf("podVal=%q: got %v, want %v", podVal, got, want)
}
@@ -243,16 +243,16 @@ func TestResolveIPAlgo_NilNodeMap(t *testing.T) {
}
func TestResolveIPAlgo_Whitespace(t *testing.T) {
pod := map[string]string{annotationPrefix + annIPAlgo: " namespace , pod "}
pod := map[string]string{annotationPrefix + annIPAlgo: " namespace , app "}
got := ResolveIPAlgo(pod, nil, nil)
want := []embed.Field{embed.FieldNamespace, embed.FieldPod}
want := []embed.Field{embed.FieldNamespace, embed.FieldApp}
if !equalFields(got, want) {
t.Fatalf("got %v, want %v", got, want)
}
}
func TestResolveIPAlgo_DuplicateInvalidates(t *testing.T) {
pod := map[string]string{annotationPrefix + annIPAlgo: "pod,pod"}
pod := map[string]string{annotationPrefix + annIPAlgo: "app,app"}
node := map[string]string{annotationPrefix + annIPAlgo: "namespace"}
got := ResolveIPAlgo(pod, node, nil)
want := []embed.Field{embed.FieldNamespace}
+78
View File
@@ -5,13 +5,89 @@ import (
"fmt"
"log/slog"
"net"
"strings"
"time"
flockcni "code.fritzlab.net/fritzlab/flock/pkg/cni"
cnitypes "github.com/containernetworking/cni/pkg/types"
current "github.com/containernetworking/cni/pkg/types/100"
corev1 "k8s.io/api/core/v1"
)
// podTemplateHashLabel is the well-known label Kubernetes attaches to
// every Pod owned by a ReplicaSet so the ReplicaSet name can be
// reconstructed as "<deploy>-<hash>". We use it to peel the hash back off
// in deriveAppName.
const podTemplateHashLabel = "pod-template-hash"
// deriveAppName returns the stable workload identifier for a Pod — the
// name of the topmost stable controller, with the pod-template-hash
// stripped for ReplicaSet-owned pods.
//
// The rule maps to Kubernetes pod-name generation:
//
// Deployment → ReplicaSet → Pod pod owner is RS named "<deploy>-<hash>";
// strip the trailing "-<hash>" to recover
// the Deployment name.
// StatefulSet → Pod pod owner is the STS itself; use as-is.
// DaemonSet → Pod pod owner is the DS itself; use as-is.
// Job → Pod pod owner is the Job itself; use as-is.
// (bare pod) → Pod no controller owner; fall back to pod name.
//
// All replicas of the same workload converge on the same return value,
// which is the property the ip-algo `app` field needs.
func deriveAppName(pod *corev1.Pod) string {
owner := controllerOwner(pod)
if owner == nil {
return pod.Name
}
if owner.Kind == "ReplicaSet" {
if hash, ok := pod.Labels[podTemplateHashLabel]; ok && hash != "" {
suffix := "-" + hash
if strings.HasSuffix(owner.Name, suffix) {
return strings.TrimSuffix(owner.Name, suffix)
}
}
// Custom controller named the RS something that doesn't match
// the pod-template-hash convention. Falling back to the RS name
// keeps replicas of the same RS aligned, which is the second-
// best correctness we can offer.
return owner.Name
}
return owner.Name
}
// controllerOwner returns the OwnerReference flagged with Controller=true,
// or nil if none. Kubernetes guarantees at most one controller per object.
func controllerOwner(pod *corev1.Pod) *metav1OwnerLite {
for i := range pod.OwnerReferences {
o := &pod.OwnerReferences[i]
if o.Controller != nil && *o.Controller {
return &metav1OwnerLite{Kind: o.Kind, Name: o.Name}
}
}
return nil
}
// metav1OwnerLite is the slice of OwnerReference we actually consult,
// kept tiny so it can be returned by value-pointer cheaply.
type metav1OwnerLite struct {
Kind string
Name string
}
// podImageRef returns a deterministic image reference for the embed
// `image` field. We use the first container's spec'd image — this is
// stable across replicas of the same Deployment without requiring the
// runtime-resolved digest. Empty string if the pod has no containers,
// in which case the embed package falls back to FNV(containerID).
func podImageRef(pod *corev1.Pod) string {
if len(pod.Spec.Containers) == 0 {
return ""
}
return pod.Spec.Containers[0].Image
}
// PodHandler is the platform-agnostic ADD/DEL/CHECK implementation. It
// resolves the Pod from the informer cache, parses annotations, allocates
// from IPAM, programs netns (or skips on non-Linux build), and persists
@@ -68,11 +144,13 @@ func (h *PodHandler) Add(ctx context.Context, req flockcni.Request) (*current.Re
ContainerID: req.ContainerID,
Namespace: args.PodNamespace,
Pod: args.PodName,
App: deriveAppName(pod),
WantV6: parsed.WantV6,
WantV4: parsed.WantV4,
AnnCIDR6: parsed.CIDR6,
AnnCIDR4: parsed.CIDR4,
IPAlgo: ipAlgo,
Image: podImageRef(pod),
}
res, err := h.IPAM.Allocate(allocReq)
if err != nil {
+108
View File
@@ -0,0 +1,108 @@
package agent
import (
"testing"
corev1 "k8s.io/api/core/v1"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
)
func ptrBool(b bool) *bool { return &b }
func mkPod(name string, owner *metav1.OwnerReference, labels map[string]string, image string) *corev1.Pod {
p := &corev1.Pod{
ObjectMeta: metav1.ObjectMeta{Name: name, Labels: labels},
}
if owner != nil {
p.OwnerReferences = []metav1.OwnerReference{*owner}
}
if image != "" {
p.Spec.Containers = []corev1.Container{{Image: image}}
}
return p
}
func TestDeriveAppName_DeploymentReplicaSet(t *testing.T) {
owner := &metav1.OwnerReference{
Kind: "ReplicaSet",
Name: "traefik-789df685f",
Controller: ptrBool(true),
}
pod := mkPod("traefik-789df685f-hqvfl", owner,
map[string]string{podTemplateHashLabel: "789df685f"}, "")
if got := deriveAppName(pod); got != "traefik" {
t.Fatalf("got %q, want %q", got, "traefik")
}
}
func TestDeriveAppName_StatefulSet(t *testing.T) {
owner := &metav1.OwnerReference{
Kind: "StatefulSet",
Name: "gitea",
Controller: ptrBool(true),
}
pod := mkPod("gitea-0", owner, nil, "")
if got := deriveAppName(pod); got != "gitea" {
t.Fatalf("got %q, want %q", got, "gitea")
}
}
func TestDeriveAppName_DaemonSet(t *testing.T) {
owner := &metav1.OwnerReference{
Kind: "DaemonSet",
Name: "flock-agent",
Controller: ptrBool(true),
}
pod := mkPod("flock-agent-abcde", owner, nil, "")
if got := deriveAppName(pod); got != "flock-agent" {
t.Fatalf("got %q, want %q", got, "flock-agent")
}
}
func TestDeriveAppName_BarePod(t *testing.T) {
pod := mkPod("standalone", nil, nil, "")
if got := deriveAppName(pod); got != "standalone" {
t.Fatalf("got %q, want %q", got, "standalone")
}
}
// TestDeriveAppName_RSWithoutTemplateHash — ReplicaSet owners that don't
// follow the standard "<deploy>-<hash>" naming convention (e.g. a custom
// controller) keep the RS name as-is. All replicas of that RS still align,
// which is the second-best correctness offer.
func TestDeriveAppName_RSWithoutTemplateHash(t *testing.T) {
owner := &metav1.OwnerReference{
Kind: "ReplicaSet",
Name: "weird-rs-name",
Controller: ptrBool(true),
}
pod := mkPod("weird-rs-name-xyz", owner, nil, "")
if got := deriveAppName(pod); got != "weird-rs-name" {
t.Fatalf("got %q, want %q", got, "weird-rs-name")
}
}
func TestDeriveAppName_NonControllerOwnerIgnored(t *testing.T) {
// OwnerReference without Controller=true must be ignored — only the
// controller owner is the canonical workload.
owner := &metav1.OwnerReference{
Kind: "Foo",
Name: "irrelevant",
// Controller pointer left nil.
}
pod := mkPod("solo", owner, nil, "")
if got := deriveAppName(pod); got != "solo" {
t.Fatalf("got %q, want %q", got, "solo")
}
}
func TestPodImageRef(t *testing.T) {
pod := mkPod("p", nil, nil, "traefik:v3.5")
if got := podImageRef(pod); got != "traefik:v3.5" {
t.Fatalf("got %q, want %q", got, "traefik:v3.5")
}
empty := mkPod("p", nil, nil, "")
if got := podImageRef(empty); got != "" {
t.Fatalf("got %q, want \"\"", got)
}
}
+16 -8
View File
@@ -67,7 +67,13 @@ func (cryptoRand) PickIndex(n int) int {
type AllocRequest struct {
ContainerID string
Namespace string
Pod string
// Pod is the literal pod name (used for logging only — not embedded).
Pod string
// App is the stable workload identity for the FieldApp embed field —
// typically the owning Deployment / StatefulSet / DaemonSet name.
// Computed by the handler; falls back to Pod when no usable owner is
// found (bare pods).
App string
// WantV6 / WantV4 are the post-merge address family selection (pod
// annotation > NodeConfig.Spec.Defaults > built-in baseline of
// dual-stack). At least one MUST be true; Allocate rejects the request
@@ -78,12 +84,14 @@ type AllocRequest struct {
// means "use any of the node's CIDRs".
AnnCIDR6 []*net.IPNet
AnnCIDR4 []*net.IPNet
// IPAlgo comes from the ip-algo annotation. Empty means random IID.
// IPAlgo comes from the resolved ip-algo precedence chain. Empty means
// random IID.
IPAlgo []embed.Field
// ImageDigest is the sha256 manifest digest (with or without "sha256:"
// prefix). If empty, embed.Values.ImageFallback = ContainerID is used
// for ip-algo fields that reference image.
ImageDigest string
// Image is the spec'd image reference (typically
// pod.Spec.Containers[0].Image). When 64 hex chars, treated as a
// sha256 digest; otherwise FNV-1a-64'd as a string. Empty falls back
// to FNV(ContainerID) for ip-algo fields that reference image.
Image string
}
// AllocResult is what the IPAM hands back to the CNI ADD.
@@ -210,8 +218,8 @@ func (i *IPAM) allocV6(cidr *net.IPNet, req AllocRequest) (net.IP, error) {
} else {
ip, err = embed.Embed(cidr, req.IPAlgo, embed.Values{
Namespace: req.Namespace,
Pod: req.Pod,
Image: req.ImageDigest,
App: req.App,
Image: req.Image,
ImageFallback: req.ContainerID,
}, i.randSrc.NibbleN())
}
+2 -2
View File
@@ -148,8 +148,8 @@ func TestIPAM_AllocV6_WithEmbed(t *testing.T) {
}
i.randSrc = &fakeRand{nibbles: []byte{0xe}}
res, err := i.Allocate(AllocRequest{
ContainerID: "c1", Namespace: "mail", Pod: "stalwart-0", WantV6: true,
IPAlgo: []embed.Field{embed.FieldNamespace, embed.FieldPod, embed.FieldImage},
ContainerID: "c1", Namespace: "mail", Pod: "stalwart-0", App: "stalwart", WantV6: true,
IPAlgo: []embed.Field{embed.FieldNamespace, embed.FieldApp, embed.FieldImage},
})
if err != nil {
t.Fatalf("Allocate: %v", err)