forked from docker-archive/classicswarm
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathdependency.go
82 lines (70 loc) · 2.27 KB
/
dependency.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
package filter
import (
"fmt"
"strings"
"github.com/docker/swarm/cluster"
"github.com/docker/swarm/scheduler/node"
)
// DependencyFilter co-schedules dependent containers on the same node.
type DependencyFilter struct {
}
// Name returns the name of the filter
func (f *DependencyFilter) Name() string {
return "dependency"
}
// Filter is exported
func (f *DependencyFilter) Filter(config *cluster.ContainerConfig, nodes []*node.Node) ([]*node.Node, error) {
if len(nodes) == 0 {
return nodes, nil
}
// Volumes
volumes := []string{}
for _, volume := range config.HostConfig.VolumesFrom {
volumes = append(volumes, strings.SplitN(volume, ":", 2)[0])
}
// Extract containers from links.
links := []string{}
for _, link := range config.HostConfig.Links {
links = append(links, strings.SplitN(link, ":", 2)[0])
}
// Check if --net points to a container.
net := []string{}
if strings.HasPrefix(config.HostConfig.NetworkMode, "container:") {
net = append(net, strings.TrimPrefix(config.HostConfig.NetworkMode, "container:"))
}
candidates := []*node.Node{}
for _, node := range nodes {
if f.check(volumes, node) &&
f.check(links, node) &&
f.check(net, node) {
candidates = append(candidates, node)
}
}
if len(candidates) == 0 {
return nil, fmt.Errorf("Unable to find a node fulfilling all dependencies: %s", f.String(config))
}
return candidates, nil
}
// Get a string representation of the dependencies found in the container config.
func (f *DependencyFilter) String(config *cluster.ContainerConfig) string {
dependencies := []string{}
for _, volume := range config.HostConfig.VolumesFrom {
dependencies = append(dependencies, fmt.Sprintf("--volumes-from=%s", volume))
}
for _, link := range config.HostConfig.Links {
dependencies = append(dependencies, fmt.Sprintf("--link=%s", link))
}
if strings.HasPrefix(config.HostConfig.NetworkMode, "container:") {
dependencies = append(dependencies, fmt.Sprintf("--net=%s", config.HostConfig.NetworkMode))
}
return strings.Join(dependencies, " ")
}
// Ensure that the node contains all dependent containers.
func (f *DependencyFilter) check(dependencies []string, node *node.Node) bool {
for _, dependency := range dependencies {
if node.Container(dependency) == nil {
return false
}
}
return true
}