2015-09-03 20:51:04 -04:00
|
|
|
package daemon
|
|
|
|
|
|
|
|
import (
|
2015-11-21 13:45:34 -05:00
|
|
|
"strings"
|
2015-09-03 20:51:04 -04:00
|
|
|
"sync"
|
|
|
|
|
2015-11-21 13:45:34 -05:00
|
|
|
"github.com/Sirupsen/logrus"
|
2015-09-03 20:51:04 -04:00
|
|
|
"github.com/docker/docker/container"
|
2015-11-21 13:45:34 -05:00
|
|
|
"github.com/docker/docker/pkg/graphdb"
|
2015-09-03 20:51:04 -04:00
|
|
|
)
|
|
|
|
|
|
|
|
// linkIndex stores link relationships between containers, including their specified alias
|
|
|
|
// The alias is the name the parent uses to reference the child
|
|
|
|
type linkIndex struct {
|
|
|
|
// idx maps a parent->alias->child relationship
|
|
|
|
idx map[*container.Container]map[string]*container.Container
|
|
|
|
// childIdx maps child->parent->aliases
|
|
|
|
childIdx map[*container.Container]map[*container.Container]map[string]struct{}
|
|
|
|
mu sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func newLinkIndex() *linkIndex {
|
|
|
|
return &linkIndex{
|
|
|
|
idx: make(map[*container.Container]map[string]*container.Container),
|
|
|
|
childIdx: make(map[*container.Container]map[*container.Container]map[string]struct{}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// link adds indexes for the passed in parent/child/alias relationships
|
|
|
|
func (l *linkIndex) link(parent, child *container.Container, alias string) {
|
|
|
|
l.mu.Lock()
|
|
|
|
|
|
|
|
if l.idx[parent] == nil {
|
|
|
|
l.idx[parent] = make(map[string]*container.Container)
|
|
|
|
}
|
|
|
|
l.idx[parent][alias] = child
|
|
|
|
if l.childIdx[child] == nil {
|
|
|
|
l.childIdx[child] = make(map[*container.Container]map[string]struct{})
|
|
|
|
}
|
|
|
|
if l.childIdx[child][parent] == nil {
|
|
|
|
l.childIdx[child][parent] = make(map[string]struct{})
|
|
|
|
}
|
|
|
|
l.childIdx[child][parent][alias] = struct{}{}
|
|
|
|
|
|
|
|
l.mu.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
// unlink removes the requested alias for the given parent/child
|
|
|
|
func (l *linkIndex) unlink(alias string, child, parent *container.Container) {
|
|
|
|
l.mu.Lock()
|
|
|
|
delete(l.idx[parent], alias)
|
|
|
|
delete(l.childIdx[child], parent)
|
|
|
|
l.mu.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
// children maps all the aliases-> children for the passed in parent
|
|
|
|
// aliases here are the aliases the parent uses to refer to the child
|
|
|
|
func (l *linkIndex) children(parent *container.Container) map[string]*container.Container {
|
|
|
|
l.mu.Lock()
|
|
|
|
children := l.idx[parent]
|
|
|
|
l.mu.Unlock()
|
|
|
|
return children
|
|
|
|
}
|
|
|
|
|
|
|
|
// parents maps all the aliases->parent for the passed in child
|
|
|
|
// aliases here are the aliases the parents use to refer to the child
|
|
|
|
func (l *linkIndex) parents(child *container.Container) map[string]*container.Container {
|
|
|
|
l.mu.Lock()
|
|
|
|
|
|
|
|
parents := make(map[string]*container.Container)
|
|
|
|
for parent, aliases := range l.childIdx[child] {
|
|
|
|
for alias := range aliases {
|
|
|
|
parents[alias] = parent
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
l.mu.Unlock()
|
|
|
|
return parents
|
|
|
|
}
|
|
|
|
|
|
|
|
// delete deletes all link relationships referencing this container
|
|
|
|
func (l *linkIndex) delete(container *container.Container) {
|
|
|
|
l.mu.Lock()
|
|
|
|
for _, child := range l.idx[container] {
|
|
|
|
delete(l.childIdx[child], container)
|
|
|
|
}
|
|
|
|
delete(l.idx, container)
|
|
|
|
delete(l.childIdx, container)
|
|
|
|
l.mu.Unlock()
|
|
|
|
}
|
2015-11-21 13:45:34 -05:00
|
|
|
|
|
|
|
// migrateLegacySqliteLinks migrates sqlite links to use links from HostConfig
|
|
|
|
// when sqlite links were used, hostConfig.Links was set to nil
|
|
|
|
func (daemon *Daemon) migrateLegacySqliteLinks(db *graphdb.Database, container *container.Container) error {
|
|
|
|
// if links is populated (or an empty slice), then this isn't using sqlite links and can be skipped
|
|
|
|
if container.HostConfig == nil || container.HostConfig.Links != nil {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
logrus.Debugf("migrating legacy sqlite link info for container: %s", container.ID)
|
|
|
|
|
|
|
|
fullName := container.Name
|
|
|
|
if fullName[0] != '/' {
|
|
|
|
fullName = "/" + fullName
|
|
|
|
}
|
|
|
|
|
|
|
|
// don't use a nil slice, this ensures that the check above will skip once the migration has completed
|
|
|
|
links := []string{}
|
|
|
|
children, err := db.Children(fullName, 0)
|
|
|
|
if err != nil {
|
|
|
|
if !strings.Contains(err.Error(), "Cannot find child for") {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// else continue... it's ok if we didn't find any children, it'll just be nil and we can continue the migration
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, child := range children {
|
|
|
|
c, err := daemon.GetContainer(child.Entity.ID())
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
links = append(links, c.Name+":"+child.Edge.Name)
|
|
|
|
}
|
|
|
|
|
|
|
|
container.HostConfig.Links = links
|
|
|
|
return container.WriteHostConfig()
|
|
|
|
}
|