2016-10-20 18:58:34 +00:00
|
|
|
package control
|
2016-08-12 18:05:16 +00:00
|
|
|
|
|
|
|
import (
|
2016-08-17 23:07:46 +00:00
|
|
|
"bytes"
|
2016-08-12 18:05:16 +00:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"os"
|
|
|
|
"os/exec"
|
2016-08-25 04:43:51 +00:00
|
|
|
"path"
|
2016-08-12 18:05:16 +00:00
|
|
|
"regexp"
|
|
|
|
"strings"
|
|
|
|
"syscall"
|
|
|
|
|
|
|
|
log "github.com/Sirupsen/logrus"
|
2016-10-20 18:58:34 +00:00
|
|
|
"github.com/codegangsta/cli"
|
2016-08-12 18:05:16 +00:00
|
|
|
"github.com/rancher/os/cmd/cloudinitexecute"
|
|
|
|
"github.com/rancher/os/config"
|
|
|
|
"github.com/rancher/os/util"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
consoleDone = "/run/console-done"
|
|
|
|
dockerHome = "/home/docker"
|
2016-08-17 23:07:46 +00:00
|
|
|
gettyCmd = "/sbin/agetty"
|
2016-08-12 18:05:16 +00:00
|
|
|
rancherHome = "/home/rancher"
|
|
|
|
startScript = "/opt/rancher/bin/start.sh"
|
|
|
|
)
|
|
|
|
|
2016-08-15 22:06:26 +00:00
|
|
|
type symlink struct {
|
|
|
|
oldname, newname string
|
|
|
|
}
|
|
|
|
|
2016-10-20 18:58:34 +00:00
|
|
|
func consoleInitAction(c *cli.Context) error {
|
2016-08-12 18:05:16 +00:00
|
|
|
cfg := config.LoadConfig()
|
|
|
|
|
2016-08-16 23:25:01 +00:00
|
|
|
if _, err := os.Stat(rancherHome); os.IsNotExist(err) {
|
|
|
|
if err := os.MkdirAll(rancherHome, 0755); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
if err := os.Chown(rancherHome, 1100, 1100); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
|
|
|
|
2016-08-16 23:25:01 +00:00
|
|
|
if _, err := os.Stat(dockerHome); os.IsNotExist(err) {
|
|
|
|
if err := os.MkdirAll(dockerHome, 0755); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
if err := os.Chown(dockerHome, 1101, 1101); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
password := config.GetCmdline("rancher.password")
|
|
|
|
cmd := exec.Command("chpasswd")
|
|
|
|
cmd.Stdin = strings.NewReader(fmt.Sprint("rancher:", password))
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd = exec.Command("bash", "-c", `sed -E -i 's/(rancher:.*:).*(:.*:.*:.*:.*:.*:.*)$/\1\2/' /etc/shadow`)
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := setupSSH(cfg); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
2016-08-22 16:50:56 +00:00
|
|
|
if err := writeRespawn(); err != nil {
|
2016-08-12 18:05:16 +00:00
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
2016-08-22 16:50:56 +00:00
|
|
|
if err := modifySshdConfig(); err != nil {
|
2016-08-12 18:05:16 +00:00
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
2016-08-22 16:50:56 +00:00
|
|
|
if err := writeOsRelease(); err != nil {
|
2016-08-12 18:05:16 +00:00
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
2016-08-15 22:06:26 +00:00
|
|
|
for _, link := range []symlink{
|
|
|
|
{"/var/lib/rancher/engine/docker", "/usr/bin/docker"},
|
|
|
|
{"/var/lib/rancher/engine/docker-containerd", "/usr/bin/docker-containerd"},
|
|
|
|
{"/var/lib/rancher/engine/docker-containerd-ctr", "/usr/bin/docker-containerd-ctr"},
|
|
|
|
{"/var/lib/rancher/engine/docker-containerd-shim", "/usr/bin/docker-containerd-shim"},
|
|
|
|
{"/var/lib/rancher/engine/dockerd", "/usr/bin/dockerd"},
|
2016-08-24 16:43:03 +00:00
|
|
|
{"/var/lib/rancher/engine/docker-proxy", "/usr/bin/docker-proxy"},
|
2016-08-15 22:06:26 +00:00
|
|
|
{"/var/lib/rancher/engine/docker-runc", "/usr/bin/docker-runc"},
|
|
|
|
} {
|
|
|
|
syscall.Unlink(link.newname)
|
2016-08-22 16:50:56 +00:00
|
|
|
if err := os.Symlink(link.oldname, link.newname); err != nil {
|
2016-08-15 22:06:26 +00:00
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-12 18:05:16 +00:00
|
|
|
cmd = exec.Command("bash", "-c", `echo 'RancherOS \n \l' > /etc/issue`)
|
2016-08-22 16:50:56 +00:00
|
|
|
if err := cmd.Run(); err != nil {
|
2016-08-12 18:05:16 +00:00
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd = exec.Command("bash", "-c", `echo $(/sbin/ifconfig | grep -B1 "inet addr" |awk '{ if ( $1 == "inet" ) { print $2 } else if ( $2 == "Link" ) { printf "%s:" ,$1 } }' |awk -F: '{ print $1 ": " $3}') >> /etc/issue`)
|
2016-08-22 16:50:56 +00:00
|
|
|
if err := cmd.Run(); err != nil {
|
2016-08-12 18:05:16 +00:00
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
cloudinitexecute.ApplyConsole(cfg)
|
|
|
|
|
2016-10-17 21:47:44 +00:00
|
|
|
if err := util.RunScript(config.CloudConfigScriptFile); err != nil {
|
2016-08-31 23:18:51 +00:00
|
|
|
log.Error(err)
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
2016-10-17 21:47:44 +00:00
|
|
|
if err := util.RunScript(startScript); err != nil {
|
2016-08-31 23:18:51 +00:00
|
|
|
log.Error(err)
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
2016-10-14 17:48:32 +00:00
|
|
|
|
|
|
|
if err := ioutil.WriteFile(consoleDone, []byte(cfg.Rancher.Console), 0644); err != nil {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
2016-10-17 21:47:44 +00:00
|
|
|
if err := util.RunScript("/etc/rc.local"); err != nil {
|
2016-08-31 23:18:51 +00:00
|
|
|
log.Error(err)
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
os.Setenv("TERM", "linux")
|
|
|
|
|
|
|
|
respawnBinPath, err := exec.LookPath("respawn")
|
|
|
|
if err != nil {
|
2016-10-20 18:58:34 +00:00
|
|
|
return err
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
|
|
|
|
2016-10-20 18:58:34 +00:00
|
|
|
return syscall.Exec(respawnBinPath, []string{"respawn", "-f", "/etc/respawn.conf"}, os.Environ())
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
|
|
|
|
2016-08-22 16:50:56 +00:00
|
|
|
func generateRespawnConf(cmdline string) string {
|
2016-08-17 23:07:46 +00:00
|
|
|
var respawnConf bytes.Buffer
|
|
|
|
|
|
|
|
for i := 1; i < 7; i++ {
|
2016-08-22 18:19:24 +00:00
|
|
|
tty := fmt.Sprintf("tty%d", i)
|
|
|
|
|
2016-08-17 23:07:46 +00:00
|
|
|
respawnConf.WriteString(gettyCmd)
|
2016-08-22 18:19:24 +00:00
|
|
|
if strings.Contains(cmdline, fmt.Sprintf("rancher.autologin=%s", tty)) {
|
2016-08-17 23:07:46 +00:00
|
|
|
respawnConf.WriteString(" --autologin rancher")
|
|
|
|
}
|
2016-08-22 18:19:24 +00:00
|
|
|
respawnConf.WriteString(fmt.Sprintf(" 115200 %s\n", tty))
|
2016-08-17 23:07:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for _, tty := range []string{"ttyS0", "ttyS1", "ttyS2", "ttyS3", "ttyAMA0"} {
|
|
|
|
if !strings.Contains(cmdline, fmt.Sprintf("console=%s", tty)) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
respawnConf.WriteString(gettyCmd)
|
2016-08-22 18:19:24 +00:00
|
|
|
if strings.Contains(cmdline, fmt.Sprintf("rancher.autologin=%s", tty)) {
|
2016-08-17 23:07:46 +00:00
|
|
|
respawnConf.WriteString(" --autologin rancher")
|
|
|
|
}
|
|
|
|
respawnConf.WriteString(fmt.Sprintf(" 115200 %s\n", tty))
|
|
|
|
}
|
|
|
|
|
|
|
|
respawnConf.WriteString("/usr/sbin/sshd -D")
|
|
|
|
|
2016-08-22 16:50:56 +00:00
|
|
|
return respawnConf.String()
|
|
|
|
}
|
|
|
|
|
|
|
|
func writeRespawn() error {
|
|
|
|
cmdline, err := ioutil.ReadFile("/proc/cmdline")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
respawn := generateRespawnConf(string(cmdline))
|
|
|
|
|
|
|
|
files, err := ioutil.ReadDir("/etc/respawn.conf.d")
|
|
|
|
if err == nil {
|
|
|
|
for _, f := range files {
|
2016-08-25 04:43:51 +00:00
|
|
|
p := path.Join("/etc/respawn.conf.d", f.Name())
|
|
|
|
content, err := ioutil.ReadFile(p)
|
2016-08-22 16:50:56 +00:00
|
|
|
if err != nil {
|
2016-09-15 23:49:39 +00:00
|
|
|
log.Errorf("Failed to read %s: %v", p, err)
|
2016-08-25 04:43:51 +00:00
|
|
|
continue
|
2016-08-22 16:50:56 +00:00
|
|
|
}
|
|
|
|
respawn += fmt.Sprintf("\n%s", string(content))
|
|
|
|
}
|
|
|
|
} else if !os.IsNotExist(err) {
|
|
|
|
log.Error(err)
|
|
|
|
}
|
|
|
|
|
|
|
|
return ioutil.WriteFile("/etc/respawn.conf", []byte(respawn), 0644)
|
2016-08-17 23:07:46 +00:00
|
|
|
}
|
|
|
|
|
2016-08-12 18:05:16 +00:00
|
|
|
func modifySshdConfig() error {
|
|
|
|
sshdConfig, err := ioutil.ReadFile("/etc/ssh/sshd_config")
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
sshdConfigString := string(sshdConfig)
|
|
|
|
|
|
|
|
for _, item := range []string{
|
|
|
|
"UseDNS no",
|
|
|
|
"PermitRootLogin no",
|
|
|
|
"ServerKeyBits 2048",
|
|
|
|
"AllowGroups docker",
|
|
|
|
} {
|
|
|
|
match, err := regexp.Match("^"+item, sshdConfig)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if !match {
|
|
|
|
sshdConfigString += fmt.Sprintf("%s\n", item)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-08-17 05:02:26 +00:00
|
|
|
return ioutil.WriteFile("/etc/ssh/sshd_config", []byte(sshdConfigString), 0644)
|
2016-08-12 18:05:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func writeOsRelease() error {
|
|
|
|
idLike := "busybox"
|
|
|
|
if osRelease, err := ioutil.ReadFile("/etc/os-release"); err == nil {
|
|
|
|
for _, line := range strings.Split(string(osRelease), "\n") {
|
|
|
|
if strings.HasPrefix(line, "ID_LIKE") {
|
|
|
|
split := strings.Split(line, "ID_LIKE")
|
|
|
|
if len(split) > 1 {
|
|
|
|
idLike = split[1]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ioutil.WriteFile("/etc/os-release", []byte(fmt.Sprintf(`
|
|
|
|
NAME="RancherOS"
|
|
|
|
VERSION=%s
|
|
|
|
ID=rancheros
|
|
|
|
ID_LIKE=%s
|
|
|
|
VERSION_ID=%s
|
|
|
|
PRETTY_NAME="RancherOS %s"
|
|
|
|
HOME_URL=
|
|
|
|
SUPPORT_URL=
|
|
|
|
BUG_REPORT_URL=
|
|
|
|
BUILD_ID=
|
|
|
|
`, config.VERSION, idLike, config.VERSION, config.VERSION)), 0644)
|
|
|
|
}
|
|
|
|
|
|
|
|
func setupSSH(cfg *config.CloudConfig) error {
|
|
|
|
for _, keyType := range []string{"rsa", "dsa", "ecdsa", "ed25519"} {
|
|
|
|
outputFile := fmt.Sprintf("/etc/ssh/ssh_host_%s_key", keyType)
|
|
|
|
outputFilePub := fmt.Sprintf("/etc/ssh/ssh_host_%s_key.pub", keyType)
|
|
|
|
|
|
|
|
if _, err := os.Stat(outputFile); err == nil {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
saved, savedExists := cfg.Rancher.Ssh.Keys[keyType]
|
|
|
|
pub, pubExists := cfg.Rancher.Ssh.Keys[keyType+"-pub"]
|
|
|
|
|
|
|
|
if savedExists && pubExists {
|
|
|
|
// TODO check permissions
|
|
|
|
if err := util.WriteFileAtomic(outputFile, []byte(saved), 0600); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := util.WriteFileAtomic(outputFilePub, []byte(pub), 0600); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
cmd := exec.Command("bash", "-c", fmt.Sprintf("ssh-keygen -f %s -N '' -t %s", outputFile, keyType))
|
|
|
|
if err := cmd.Run(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
savedBytes, err := ioutil.ReadFile(outputFile)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
pubBytes, err := ioutil.ReadFile(outputFilePub)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
config.Set(fmt.Sprintf("rancher.ssh.keys.%s", keyType), string(savedBytes))
|
|
|
|
config.Set(fmt.Sprintf("rancher.ssh.keys.%s-pub", keyType), string(pubBytes))
|
|
|
|
}
|
|
|
|
|
|
|
|
return os.MkdirAll("/var/run/sshd", 0644)
|
|
|
|
}
|