mirror of https://github.com/docker/docs.git
230 lines
5.9 KiB
Go
230 lines
5.9 KiB
Go
package provision
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"text/template"
|
|
|
|
"github.com/docker/machine/drivers"
|
|
"github.com/docker/machine/libmachine/auth"
|
|
"github.com/docker/machine/libmachine/engine"
|
|
"github.com/docker/machine/libmachine/provision/pkgaction"
|
|
"github.com/docker/machine/libmachine/swarm"
|
|
"github.com/docker/machine/log"
|
|
"github.com/docker/machine/ssh"
|
|
"github.com/docker/machine/utils"
|
|
)
|
|
|
|
func init() {
|
|
Register("Ubuntu", &RegisteredProvisioner{
|
|
New: NewUbuntuProvisioner,
|
|
})
|
|
}
|
|
|
|
func NewUbuntuProvisioner(d drivers.Driver) Provisioner {
|
|
return &UbuntuProvisioner{
|
|
packages: []string{
|
|
"curl",
|
|
},
|
|
Driver: d,
|
|
}
|
|
}
|
|
|
|
type UbuntuProvisioner struct {
|
|
packages []string
|
|
OsReleaseInfo *OsRelease
|
|
Driver drivers.Driver
|
|
AuthOptions auth.AuthOptions
|
|
EngineOptions engine.EngineOptions
|
|
SwarmOptions swarm.SwarmOptions
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) Service(name string, action pkgaction.ServiceAction) error {
|
|
command := fmt.Sprintf("sudo service %s %s", name, action.String())
|
|
|
|
if _, err := provisioner.SSHCommand(command); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) Package(name string, action pkgaction.PackageAction) error {
|
|
var packageAction string
|
|
|
|
switch action {
|
|
case pkgaction.Install:
|
|
packageAction = "install"
|
|
case pkgaction.Remove:
|
|
packageAction = "remove"
|
|
case pkgaction.Upgrade:
|
|
packageAction = "upgrade"
|
|
}
|
|
|
|
// TODO: This should probably have a const
|
|
switch name {
|
|
case "docker":
|
|
name = "lxc-docker"
|
|
}
|
|
|
|
command := fmt.Sprintf("DEBIAN_FRONTEND=noninteractive sudo -E apt-get %s -y %s", packageAction, name)
|
|
|
|
if _, err := provisioner.SSHCommand(command); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) dockerDaemonResponding() bool {
|
|
if _, err := provisioner.SSHCommand("sudo docker version"); err != nil {
|
|
log.Warn("Error getting SSH command to check if the daemon is up: %s", err)
|
|
return false
|
|
}
|
|
|
|
// The daemon is up if the command worked. Carry on.
|
|
return true
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) Provision(swarmOptions swarm.SwarmOptions, authOptions auth.AuthOptions, engineOptions engine.EngineOptions) error {
|
|
provisioner.SwarmOptions = swarmOptions
|
|
provisioner.AuthOptions = authOptions
|
|
provisioner.EngineOptions = engineOptions
|
|
if err := provisioner.SetHostname(provisioner.Driver.GetMachineName()); err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, pkg := range provisioner.packages {
|
|
if err := provisioner.Package(pkg, pkgaction.Install); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
|
|
if err := installDockerGeneric(provisioner); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := utils.WaitFor(provisioner.dockerDaemonResponding); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := makeDockerOptionsDir(provisioner); err != nil {
|
|
return err
|
|
}
|
|
|
|
provisioner.AuthOptions = setRemoteAuthOptions(provisioner)
|
|
|
|
if err := ConfigureAuth(provisioner); err != nil {
|
|
return err
|
|
}
|
|
|
|
if err := configureSwarm(provisioner, swarmOptions); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) Hostname() (string, error) {
|
|
output, err := provisioner.SSHCommand("hostname")
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
|
|
var so bytes.Buffer
|
|
if _, err := so.ReadFrom(output.Stdout); err != nil {
|
|
return "", err
|
|
}
|
|
|
|
return so.String(), nil
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) SetHostname(hostname string) error {
|
|
if _, err := provisioner.SSHCommand(fmt.Sprintf(
|
|
"sudo hostname %s && echo %q | sudo tee /etc/hostname",
|
|
hostname,
|
|
hostname,
|
|
)); err != nil {
|
|
return err
|
|
}
|
|
|
|
// ubuntu/debian use 127.0.1.1 for non "localhost" loopback hostnames: https://www.debian.org/doc/manuals/debian-reference/ch05.en.html#_the_hostname_resolution
|
|
if _, err := provisioner.SSHCommand(fmt.Sprintf(
|
|
"if grep -xq 127.0.1.1.* /etc/hosts; then sudo sed -i 's/^127.0.1.1.*/127.0.1.1 %s/g' /etc/hosts; else echo '127.0.1.1 %s' | sudo tee -a /etc/hosts; fi",
|
|
hostname,
|
|
hostname,
|
|
)); err != nil {
|
|
return err
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) GetDockerOptionsDir() string {
|
|
return "/etc/docker"
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) SSHCommand(args string) (ssh.Output, error) {
|
|
return drivers.RunSSHCommandFromDriver(provisioner.Driver, args)
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) CompatibleWithHost() bool {
|
|
return provisioner.OsReleaseInfo.Id == "ubuntu"
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) GetAuthOptions() auth.AuthOptions {
|
|
return provisioner.AuthOptions
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) SetOsReleaseInfo(info *OsRelease) {
|
|
provisioner.OsReleaseInfo = info
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) GenerateDockerOptions(dockerPort int) (*DockerOptions, error) {
|
|
var (
|
|
engineCfg bytes.Buffer
|
|
)
|
|
|
|
driverNameLabel := fmt.Sprintf("provider=%s", provisioner.Driver.DriverName())
|
|
provisioner.EngineOptions.Labels = append(provisioner.EngineOptions.Labels, driverNameLabel)
|
|
|
|
engineConfigTmpl := `
|
|
DOCKER_OPTS='
|
|
-H tcp://0.0.0.0:{{.DockerPort}}
|
|
-H unix:///var/run/docker.sock
|
|
--storage-driver {{.EngineOptions.StorageDriver}}
|
|
--tlsverify
|
|
--tlscacert {{.AuthOptions.CaCertRemotePath}}
|
|
--tlscert {{.AuthOptions.ServerCertRemotePath}}
|
|
--tlskey {{.AuthOptions.ServerKeyRemotePath}}
|
|
{{ range .EngineOptions.Labels }}--label {{.}}
|
|
{{ end }}{{ range .EngineOptions.InsecureRegistry }}--insecure-registry {{.}}
|
|
{{ end }}{{ range .EngineOptions.RegistryMirror }}--registry-mirror {{.}}
|
|
{{ end }}{{ range .EngineOptions.ArbitraryFlags }}--{{.}}
|
|
{{ end }}
|
|
'
|
|
`
|
|
t, err := template.New("engineConfig").Parse(engineConfigTmpl)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
engineConfigContext := EngineConfigContext{
|
|
DockerPort: dockerPort,
|
|
AuthOptions: provisioner.AuthOptions,
|
|
EngineOptions: provisioner.EngineOptions,
|
|
}
|
|
|
|
t.Execute(&engineCfg, engineConfigContext)
|
|
|
|
daemonOptsDir := "/etc/default/docker"
|
|
return &DockerOptions{
|
|
EngineOptions: engineCfg.String(),
|
|
EngineOptionsPath: daemonOptsDir,
|
|
}, nil
|
|
}
|
|
|
|
func (provisioner *UbuntuProvisioner) GetDriver() drivers.Driver {
|
|
return provisioner.Driver
|
|
}
|