package system import ( "fmt" "io/ioutil" "log" "os" "os/exec" "path" "path/filepath" "strings" "github.com/coreos/coreos-cloudinit/third_party/github.com/coreos/go-systemd/dbus" ) func NewUnitManager(root string) UnitManager { return &systemd{root} } type systemd struct { root string } // fakeMachineID is placed on non-usr CoreOS images and should // never be used as a true MachineID const fakeMachineID = "42000000000000000000000000000042" // PlaceUnit writes a unit file at the provided destination, creating // parent directories as necessary. func (s *systemd) PlaceUnit(u *Unit, dst string) error { dir := filepath.Dir(dst) if _, err := os.Stat(dir); os.IsNotExist(err) { if err := os.MkdirAll(dir, os.FileMode(0755)); err != nil { return err } } file := File{ Path: filepath.Base(dst), Content: u.Content, RawFilePermissions: "0644", } _, err := WriteFile(&file, dir) if err != nil { return err } return nil } func (s *systemd) EnableUnitFile(unit string, runtime bool) error { conn, err := dbus.New() if err != nil { return err } units := []string{unit} _, _, err = conn.EnableUnitFiles(units, runtime, true) return err } func (s *systemd) RunUnitCommand(command, unit string) (string, error) { conn, err := dbus.New() if err != nil { return "", err } var fn func(string, string) (string, error) switch command { case "start": fn = conn.StartUnit case "stop": fn = conn.StopUnit case "restart": fn = conn.RestartUnit case "reload": fn = conn.ReloadUnit case "try-restart": fn = conn.TryRestartUnit case "reload-or-restart": fn = conn.ReloadOrRestartUnit case "reload-or-try-restart": fn = conn.ReloadOrTryRestartUnit default: return "", fmt.Errorf("Unsupported systemd command %q", command) } return fn(unit, "replace") } func (s *systemd) DaemonReload() error { conn, err := dbus.New() if err != nil { return err } return conn.Reload() } // MaskUnit masks the given Unit by symlinking its unit file to // /dev/null, analogous to `systemctl mask`. // N.B.: Unlike `systemctl mask`, this function will *remove any existing unit // file at the location*, to ensure that the mask will succeed. func (s *systemd) MaskUnit(unit *Unit) error { masked := unit.Destination(s.root) if _, err := os.Stat(masked); os.IsNotExist(err) { if err := os.MkdirAll(path.Dir(masked), os.FileMode(0755)); err != nil { return err } } else if err := os.Remove(masked); err != nil { return err } return os.Symlink("/dev/null", masked) } // UnmaskUnit is analogous to systemd's unit_file_unmask. If the file // associated with the given Unit is empty or appears to be a symlink to // /dev/null, it is removed. func (s *systemd) UnmaskUnit(unit *Unit) error { masked := unit.Destination(s.root) ne, err := nullOrEmpty(masked) if os.IsNotExist(err) { return nil } else if err != nil { return err } if !ne { log.Printf("%s is not null or empty, refusing to unmask", masked) return nil } return os.Remove(masked) } // nullOrEmpty checks whether a given path appears to be an empty regular file // or a symlink to /dev/null func nullOrEmpty(path string) (bool, error) { fi, err := os.Stat(path) if err != nil { return false, err } m := fi.Mode() if m.IsRegular() && fi.Size() <= 0 { return true, nil } if m&os.ModeCharDevice > 0 { return true, nil } return false, nil } func ExecuteScript(scriptPath string) (string, error) { props := []dbus.Property{ dbus.PropDescription("Unit generated and executed by coreos-cloudinit on behalf of user"), dbus.PropExecStart([]string{"/bin/bash", scriptPath}, false), } base := path.Base(scriptPath) name := fmt.Sprintf("coreos-cloudinit-%s.service", base) log.Printf("Creating transient systemd unit '%s'", name) conn, err := dbus.New() if err != nil { return "", err } _, err = conn.StartTransientUnit(name, "replace", props...) return name, err } func SetHostname(hostname string) error { return exec.Command("hostnamectl", "set-hostname", hostname).Run() } func Hostname() (string, error) { return os.Hostname() } func MachineID(root string) string { contents, _ := ioutil.ReadFile(path.Join(root, "etc", "machine-id")) id := strings.TrimSpace(string(contents)) if id == fakeMachineID { id = "" } return id }