package google import ( "fmt" "github.com/docker/machine/state" "os/exec" "path" log "github.com/Sirupsen/logrus" "github.com/codegangsta/cli" "github.com/docker/machine/drivers" "github.com/docker/machine/ssh" ) const ( dockerConfigDir = "/etc/docker" ) // Driver is a struct compatible with the docker.hosts.drivers.Driver interface. type Driver struct { MachineName string Zone string MachineType string Scopes string DiskSize int AuthTokenPath string storePath string UserName string Project string CaCertPath string PrivateKeyPath string sshKeyPath string publicSSHKeyPath string SwarmMaster bool SwarmHost string SwarmDiscovery string } // CreateFlags are the command line flags used to create a driver. type CreateFlags struct { Zone *string MachineType *string UserName *string Project *string Scopes *string DiskSize *int } func init() { drivers.Register("google", &drivers.RegisteredDriver{ New: NewDriver, GetCreateFlags: GetCreateFlags, }) } // RegisterCreateFlags registers the flags this driver adds to // "docker hosts create" func GetCreateFlags() []cli.Flag { return []cli.Flag{ cli.StringFlag{ Name: "google-zone", Usage: "GCE Zone", Value: "us-central1-a", EnvVar: "GOOGLE_ZONE", }, cli.StringFlag{ Name: "google-machine-type", Usage: "GCE Machine Type", Value: "f1-micro", EnvVar: "GOOGLE_MACHINE_TYPE", }, cli.StringFlag{ Name: "google-username", Usage: "GCE User Name", Value: "docker-user", EnvVar: "GOOGLE_USERNAME", }, cli.StringFlag{ Name: "google-project", Usage: "GCE Project", EnvVar: "GOOGLE_PROJECT", }, cli.StringFlag{ Name: "google-auth-token", Usage: "GCE oAuth token", EnvVar: "GOOGLE_AUTH_TOKEN", }, cli.StringFlag{ Name: "google-scopes", Usage: "GCE Scopes (comma-separated if multiple scopes)", Value: "https://www.googleapis.com/auth/devstorage.read_only,https://www.googleapis.com/auth/logging.write", EnvVar: "GOOGLE_SCOPES", }, cli.IntFlag{ Name: "google-disk-size", Usage: "GCE Instance Disk Size (in GB)", Value: 10, EnvVar: "GOOGLE_DISK_SIZE", }, } } // NewDriver creates a Driver with the specified storePath. func NewDriver(machineName string, storePath string, caCert string, privateKey string) (drivers.Driver, error) { return &Driver{ MachineName: machineName, storePath: storePath, CaCertPath: caCert, PrivateKeyPath: privateKey, sshKeyPath: path.Join(storePath, "id_rsa"), publicSSHKeyPath: path.Join(storePath, "id_rsa.pub"), }, nil } // DriverName returns the name of the driver. func (driver *Driver) DriverName() string { return "google" } // SetConfigFromFlags initializes the driver based on the command line flags. func (driver *Driver) SetConfigFromFlags(flags drivers.DriverOptions) error { driver.Zone = flags.String("google-zone") driver.MachineType = flags.String("google-machine-type") driver.DiskSize = flags.Int("google-disk-size") driver.UserName = flags.String("google-username") driver.AuthTokenPath = flags.String("google-auth-token") driver.Project = flags.String("google-project") driver.Scopes = flags.String("google-scopes") driver.SwarmMaster = flags.Bool("swarm-master") driver.SwarmHost = flags.String("swarm-host") driver.SwarmDiscovery = flags.String("swarm-discovery") if driver.Project == "" { return fmt.Errorf("Please specify the Google Cloud Project name using the option --google-project.") } return nil } func (driver *Driver) initApis() (*ComputeUtil, error) { return newComputeUtil(driver) } func (d *Driver) PreCreateCheck() error { return nil } // Create creates a GCE VM instance acting as a docker host. func (driver *Driver) Create() error { c, err := newComputeUtil(driver) if err != nil { return err } log.Infof("Creating host...") // Check if the instance already exists. There will be an error if the instance // doesn't exist, so just check instance for nil. if instance, _ := c.instance(); instance != nil { return fmt.Errorf("Instance %v already exists.", driver.MachineName) } log.Infof("Generating SSH Key") if err := ssh.GenerateSSHKey(driver.sshKeyPath); err != nil { return err } return c.createInstance(driver) } // GetURL returns the URL of the remote docker daemon. func (driver *Driver) GetURL() (string, error) { ip, err := driver.GetIP() if err != nil { return "", err } url := fmt.Sprintf("tcp://%s:2376", ip) return url, nil } // GetIP returns the IP address of the GCE instance. func (driver *Driver) GetIP() (string, error) { c, err := newComputeUtil(driver) if err != nil { return "", err } return c.ip() } // GetState returns a docker.hosts.state.State value representing the current state of the host. func (driver *Driver) GetState() (state.State, error) { c, err := newComputeUtil(driver) if err != nil { return state.None, err } // All we care about is whether the disk exists, so we just check disk for a nil value. // There will be no error if disk is not nil. disk, _ := c.disk() instance, _ := c.instance() if instance == nil && disk == nil { return state.None, nil } if instance == nil && disk != nil { return state.Stopped, nil } switch instance.Status { case "PROVISIONING", "STAGING": return state.Starting, nil case "RUNNING": return state.Running, nil case "STOPPING", "STOPPED", "TERMINATED": return state.Stopped, nil } return state.None, nil } // Start creates a GCE instance and attaches it to the existing disk. func (driver *Driver) Start() error { c, err := newComputeUtil(driver) if err != nil { return err } return c.createInstance(driver) } // Stop deletes the GCE instance, but keeps the disk. func (driver *Driver) Stop() error { c, err := newComputeUtil(driver) if err != nil { return err } return c.deleteInstance() } // Remove deletes the GCE instance and the disk. func (driver *Driver) Remove() error { c, err := newComputeUtil(driver) if err != nil { return err } s, err := driver.GetState() if err != nil { return err } if s == state.Running { if err := c.deleteInstance(); err != nil { return err } } return c.deleteDisk() } // Restart deletes and recreates the GCE instance, keeping the disk. func (driver *Driver) Restart() error { c, err := newComputeUtil(driver) if err != nil { return err } if err := c.deleteInstance(); err != nil { return err } return c.createInstance(driver) } // Kill deletes the GCE instance, but keeps the disk. func (driver *Driver) Kill() error { return driver.Stop() } func (d *Driver) StartDocker() error { log.Debug("Starting Docker...") cmd, err := d.GetSSHCommand("sudo service docker start") if err != nil { return err } if err := cmd.Run(); err != nil { return err } return nil } func (d *Driver) StopDocker() error { log.Debug("Stopping Docker...") cmd, err := d.GetSSHCommand("sudo service docker stop") if err != nil { return err } if err := cmd.Run(); err != nil { return err } return nil } func (d *Driver) GetDockerConfigDir() string { return dockerConfigDir } // GetSSHCommand returns a command that will run over SSH on the GCE instance. func (driver *Driver) GetSSHCommand(args ...string) (*exec.Cmd, error) { ip, err := driver.GetIP() if err != nil { return nil, err } return ssh.GetSSHCommand(ip, 22, driver.UserName, driver.sshKeyPath, args...), nil } // Upgrade upgrades the docker daemon on the host to the latest version. func (driver *Driver) Upgrade() error { c, err := newComputeUtil(driver) if err != nil { return err } return c.updateDocker(driver) }