From 092342c0ff2221ca33d1e9c9bb102a0f0909798c Mon Sep 17 00:00:00 2001 From: Paul Bellamy Date: Fri, 12 Feb 2016 16:38:51 +0000 Subject: [PATCH 1/3] don't show blank ips metadata row for containers with no IP --- probe/docker/container.go | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/probe/docker/container.go b/probe/docker/container.go index cb89e6b193..7c7b604310 100644 --- a/probe/docker/container.go +++ b/probe/docker/container.go @@ -318,7 +318,10 @@ func (c *container) GetNode(hostID string, localAddrs []net.IP) report.Node { c.RLock() defer c.RUnlock() - ips := append(c.container.NetworkSettings.SecondaryIPAddresses, c.container.NetworkSettings.IPAddress) + ips := c.container.NetworkSettings.SecondaryIPAddresses + if c.container.NetworkSettings.IPAddress != "" { + ips = append(ips, c.container.NetworkSettings.IPAddress) + } // Treat all Docker IPs as local scoped. ipsWithScopes := []string{} for _, ip := range ips { From df856d78dcc798e0aaba6e7dd4ab7b6b7dedadd8 Mon Sep 17 00:00:00 2001 From: Paul Bellamy Date: Wed, 17 Feb 2016 16:42:52 +0000 Subject: [PATCH 2/3] set container IPs to host IPs for containers with --net=host --- probe/docker/container.go | 4 +++ render/topologies.go | 54 ++++++++++++++++++++++++++++++++++++++- 2 files changed, 57 insertions(+), 1 deletion(-) diff --git a/probe/docker/container.go b/probe/docker/container.go index 7c7b604310..c2f02f4a7d 100644 --- a/probe/docker/container.go +++ b/probe/docker/container.go @@ -33,6 +33,7 @@ const ( ContainerState = "docker_container_state" ContainerUptime = "docker_container_uptime" ContainerRestartCount = "docker_container_restart_count" + ContainerNetworkMode = "docker_container_network_mode" NetworkRxDropped = "network_rx_dropped" NetworkRxBytes = "network_rx_bytes" @@ -58,6 +59,8 @@ const ( StateStopped = "stopped" StatePaused = "paused" + NetworkModeHost = "host" + stopTimeout = 10 ) @@ -355,6 +358,7 @@ func (c *container) GetNode(hostID string, localAddrs []net.IP) report.Node { result = result.WithLatests(map[string]string{ ContainerUptime: uptime.String(), ContainerRestartCount: strconv.Itoa(c.container.RestartCount), + ContainerNetworkMode: c.container.HostConfig.NetworkMode, }) result = result.WithControls( RestartContainer, StopContainer, PauseContainer, AttachContainer, ExecContainer, diff --git a/render/topologies.go b/render/topologies.go index beb1d226b6..1adb663fa0 100644 --- a/render/topologies.go +++ b/render/topologies.go @@ -2,8 +2,10 @@ package render import ( "fmt" + "net" "github.com/weaveworks/scope/probe/docker" + "github.com/weaveworks/scope/probe/host" "github.com/weaveworks/scope/probe/process" "github.com/weaveworks/scope/report" ) @@ -116,6 +118,56 @@ var ContainerRenderer = MakeReduce( ), ) +type containerWithHostIPsRenderer struct { + Renderer +} + +// Render produces a process graph where the ips for host network mode are set +// to the host's IPs. +func (r containerWithHostIPsRenderer) Render(rpt report.Report) RenderableNodes { + containers := r.Renderer.Render(rpt) + hosts := MakeMap( + MapHostIdentity, + SelectHost, + ).Render(rpt) + + for id, c := range containers { + networkMode, ok := c.Node.Latest.Lookup(docker.ContainerNetworkMode) + if !ok || networkMode != docker.NetworkModeHost { + continue + } + + ips, ok := c.Node.Sets.Lookup(docker.ContainerIPs) + if ok && len(ips) > 0 { + continue + } + + h, ok := hosts[MakeHostID(report.ExtractHostID(c.Node))] + if !ok { + continue + } + + hostNetworks, ok := h.Sets.Lookup(host.LocalNetworks) + if !ok { + continue + } + + hostIPs := report.MakeStringSet() + for _, cidr := range hostNetworks { + if ip, _, err := net.ParseCIDR(cidr); err == nil { + hostIPs = hostIPs.Add(ip.String()) + } + } + + c.Sets = c.Sets.Add(docker.ContainerIPs, hostIPs) + containers[id] = c + } + + return containers +} + +var ContainerWithHostIPsRenderer = containerWithHostIPsRenderer{ContainerRenderer} + type containerWithImageNameRenderer struct { Renderer } @@ -149,7 +201,7 @@ func (r containerWithImageNameRenderer) Render(rpt report.Report) RenderableNode // ContainerWithImageNameRenderer is a Renderer which produces a container // graph where the ranks are the image names, not their IDs -var ContainerWithImageNameRenderer = containerWithImageNameRenderer{ContainerRenderer} +var ContainerWithImageNameRenderer = containerWithImageNameRenderer{ContainerWithHostIPsRenderer} // ContainerImageRenderer is a Renderer which produces a renderable container // image graph by merging the container graph and the container image topology. From 5535f5e738772a1ec310efc2daaaa45c0cc8a0ce Mon Sep 17 00:00:00 2001 From: Paul Bellamy Date: Fri, 19 Feb 2016 11:07:00 +0000 Subject: [PATCH 3/3] Review Feedback --- probe/docker/container.go | 6 +++++- render/topologies.go | 19 ++++++------------- render/topologies_test.go | 20 ++++++++++++++++++++ 3 files changed, 31 insertions(+), 14 deletions(-) diff --git a/probe/docker/container.go b/probe/docker/container.go index c2f02f4a7d..b25813a4d2 100644 --- a/probe/docker/container.go +++ b/probe/docker/container.go @@ -355,10 +355,14 @@ func (c *container) GetNode(hostID string, localAddrs []net.IP) report.Node { result = result.WithControls(UnpauseContainer) } else if c.container.State.Running { uptime := (mtime.Now().Sub(c.container.State.StartedAt) / time.Second) * time.Second + networkMode := "" + if c.container.HostConfig != nil { + networkMode = c.container.HostConfig.NetworkMode + } result = result.WithLatests(map[string]string{ ContainerUptime: uptime.String(), ContainerRestartCount: strconv.Itoa(c.container.RestartCount), - ContainerNetworkMode: c.container.HostConfig.NetworkMode, + ContainerNetworkMode: networkMode, }) result = result.WithControls( RestartContainer, StopContainer, PauseContainer, AttachContainer, ExecContainer, diff --git a/render/topologies.go b/render/topologies.go index 1adb663fa0..496787e371 100644 --- a/render/topologies.go +++ b/render/topologies.go @@ -137,35 +137,28 @@ func (r containerWithHostIPsRenderer) Render(rpt report.Report) RenderableNodes continue } - ips, ok := c.Node.Sets.Lookup(docker.ContainerIPs) - if ok && len(ips) > 0 { - continue - } - h, ok := hosts[MakeHostID(report.ExtractHostID(c.Node))] if !ok { continue } - hostNetworks, ok := h.Sets.Lookup(host.LocalNetworks) - if !ok { - continue - } - - hostIPs := report.MakeStringSet() + newIPs := report.MakeStringSet() + hostNetworks, _ := h.Sets.Lookup(host.LocalNetworks) for _, cidr := range hostNetworks { if ip, _, err := net.ParseCIDR(cidr); err == nil { - hostIPs = hostIPs.Add(ip.String()) + newIPs = newIPs.Add(ip.String()) } } - c.Sets = c.Sets.Add(docker.ContainerIPs, hostIPs) + c.Sets = c.Sets.Add(docker.ContainerIPs, newIPs) containers[id] = c } return containers } +// ContainerWithHostIPsRenderer is a Renderer which produces a container graph +// enriched with host IPs on containers where NetworkMode is Host var ContainerWithHostIPsRenderer = containerWithHostIPsRenderer{ContainerRenderer} type containerWithImageNameRenderer struct { diff --git a/render/topologies_test.go b/render/topologies_test.go index 63ce03859e..bbbcfcdf2f 100644 --- a/render/topologies_test.go +++ b/render/topologies_test.go @@ -8,6 +8,7 @@ import ( "github.com/weaveworks/scope/probe/kubernetes" "github.com/weaveworks/scope/render" "github.com/weaveworks/scope/render/expected" + "github.com/weaveworks/scope/report" "github.com/weaveworks/scope/test" "github.com/weaveworks/scope/test/fixture" ) @@ -51,6 +52,25 @@ func TestContainerFilterRenderer(t *testing.T) { } } +func TestContainerWithHostIPsRenderer(t *testing.T) { + input := fixture.Report.Copy() + input.Container.Nodes[fixture.ClientContainerNodeID] = input.Container.Nodes[fixture.ClientContainerNodeID].WithLatests(map[string]string{ + docker.ContainerNetworkMode: "host", + }) + nodes := render.ContainerWithHostIPsRenderer.Render(input) + + // Test host network nodes get the host IPs added. + haveNode, ok := nodes[render.MakeContainerID(fixture.ClientContainerID)] + if !ok { + t.Fatal("Expected output to have the client container node") + } + have, _ := haveNode.Sets.Lookup(docker.ContainerIPs) + want := report.MakeStringSet("10.10.10.0") + if !reflect.DeepEqual(want, have) { + t.Error(test.Diff(want, have)) + } +} + func TestContainerFilterRendererImageName(t *testing.T) { // Test nodes are filtered by image name as well. input := fixture.Report.Copy()