Skip to content

Commit

Permalink
Add systemd user unit feature
Browse files Browse the repository at this point in the history
  • Loading branch information
Nemric committed Feb 10, 2022
1 parent ec526d2 commit 987794f
Show file tree
Hide file tree
Showing 12 changed files with 613 additions and 139 deletions.
2 changes: 2 additions & 0 deletions config/shared/errors/errors.go
Original file line number Diff line number Diff line change
Expand Up @@ -85,6 +85,8 @@ var (
ErrInvalidSystemdDropinExt = errors.New("invalid systemd drop-in extension")
ErrNoSystemdExt = errors.New("no systemd unit extension")
ErrInvalidInstantiatedUnit = errors.New("invalid systemd instantiated unit")
ErrInvalidUnitScope = errors.New("unit scope must be system, user or global")
ErrNoUserDefined = errors.New("when 'user' scope is used you must set at least one user")

// Misc errors
ErrSourceRequired = errors.New("source is required")
Expand Down
9 changes: 9 additions & 0 deletions config/v3_4_experimental/schema/ignition.json
Original file line number Diff line number Diff line change
Expand Up @@ -507,6 +507,15 @@
"enabled": {
"type": ["boolean", "null"]
},
"scope": {
"type": ["string", "null"]
},
"users": {
"type": "array",
"items": {
"type": "string"
}
},
"mask": {
"type": ["boolean", "null"]
},
Expand Down
11 changes: 11 additions & 0 deletions config/v3_4_experimental/translate/translate.go
Original file line number Diff line number Diff line change
Expand Up @@ -27,9 +27,20 @@ func translateIgnition(old old_types.Ignition) (ret types.Ignition) {
return
}

func translateUnit(old old_types.Unit) (ret types.Unit) {
tr := translate.NewTranslator()
tr.Translate(&old.Contents, &ret.Contents)
tr.Translate(&old.Dropins, &ret.Dropins)
tr.Translate(&old.Enabled, &ret.Enabled)
tr.Translate(&old.Mask, &ret.Mask)
tr.Translate(&old.Name, &ret.Name)
return
}

func Translate(old old_types.Config) (ret types.Config) {
tr := translate.NewTranslator()
tr.AddCustomTranslator(translateIgnition)
tr.AddCustomTranslator(translateUnit)
tr.Translate(&old, &ret)
return
}
14 changes: 9 additions & 5 deletions config/v3_4_experimental/types/schema.go
Original file line number Diff line number Diff line change
Expand Up @@ -242,13 +242,17 @@ type Timeouts struct {
}

type Unit struct {
Contents *string `json:"contents,omitempty"`
Dropins []Dropin `json:"dropins,omitempty"`
Enabled *bool `json:"enabled,omitempty"`
Mask *bool `json:"mask,omitempty"`
Name string `json:"name"`
Contents *string `json:"contents,omitempty"`
Dropins []Dropin `json:"dropins,omitempty"`
Enabled *bool `json:"enabled,omitempty"`
Mask *bool `json:"mask,omitempty"`
Name string `json:"name"`
Scope *string `json:"scope,omitempty"`
Users []UnitUser `json:"users,omitempty"`
}

type UnitUser string

type Verification struct {
Hash *string `json:"hash,omitempty"`
}
29 changes: 28 additions & 1 deletion config/v3_4_experimental/types/unit.go
Original file line number Diff line number Diff line change
Expand Up @@ -29,7 +29,13 @@ import (
)

func (u Unit) Key() string {
return u.Name
// Change unit Key to differenciate them by concatenating their scope and name
// to handle the case of same named units for differents scope
if u.Scope != nil {
return *u.Scope + "." + u.Name
} else {
return "system." + u.Name
}
}

func (d Dropin) Key() string {
Expand All @@ -43,10 +49,31 @@ func (u Unit) Validate(c cpath.ContextPath) (r report.Report) {
r.AddOnError(c, err)

r.AddOnWarn(c, validations.ValidateInstallSection(u.Name, util.IsTrue(u.Enabled), util.NilOrEmpty(u.Contents), opts))
r.AddOnError(c.Append("scope"), validateScope(u.Scope))
r.AddOnError(c, validateUsers(u))

return
}

func validateScope(scope *string) error {
if scope == nil {
return nil
}
switch *scope {
case "system", "user", "global":
return nil
default:
return errors.ErrInvalidUnitScope
}
}

func validateUsers(u Unit) error {
if u.Scope != nil && *u.Scope == "user" && u.Users == nil {
return errors.ErrNoUserDefined
}
return nil
}

func validateName(name string) error {
switch path.Ext(name) {
case ".service", ".socket", ".device", ".mount", ".automount", ".swap", ".target", ".path", ".timer", ".snapshot", ".slice", ".scope":
Expand Down
2 changes: 2 additions & 0 deletions docs/configuration-v3_4_experimental.md
Original file line number Diff line number Diff line change
Expand Up @@ -156,6 +156,8 @@ The Ignition configuration is a JSON document conforming to the following specif
* **_dropins_** (list of objects): the list of drop-ins for the unit. Every drop-in must have a unique `name`.
* **name** (string): the name of the drop-in. This must be suffixed with ".conf".
* **_contents_** (string): the contents of the drop-in.
* **_scope_** (string): Allow to define "user", "global" or "system" (default) level systemd units. cf : [systemd Documentation](https://www.freedesktop.org/software/systemd/man/systemctl.html#enable%20UNIT%E2%80%A6)
* **_users_** (list of strings): The list of users concerned by the unit if scope is set to "user".
* **_passwd_** (object): describes the desired additions to the passwd database.
* **_users_** (list of objects): the list of accounts that shall exist. All users must have a unique `name`.
* **name** (string): the username for the account.
Expand Down
119 changes: 68 additions & 51 deletions internal/exec/stages/files/units.go
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ type Preset struct {
enabled bool
instantiatable bool
instances []string
scope util.UnitScope
}

// warnOnOldSystemdVersion checks the version of Systemd
Expand Down Expand Up @@ -69,35 +70,35 @@ func (s *stage) createUnits(config types.Config) error {
if err != nil {
return err
}
key := fmt.Sprintf("%s-%s", unitName, identifier)
key := fmt.Sprintf("%s.%s-%s", util.GetUnitScope(unit), unitName, identifier)
// key := fmt.Sprintf("%s-%s", unit.Key(), identifier)
if _, ok := presets[key]; ok {
presets[key].instances = append(presets[key].instances, instance)
} else {
presets[key] = &Preset{unitName, *unit.Enabled, true, []string{instance}}
presets[key] = &Preset{unitName, *unit.Enabled, true, []string{instance}, util.GetUnitScope(unit)}
}
} else {
key := fmt.Sprintf("%s-%s", unit.Name, identifier)
if _, ok := presets[unit.Name]; !ok {
presets[key] = &Preset{unit.Name, *unit.Enabled, false, []string{}}
key := fmt.Sprintf("%s-%s", unit.Key(), identifier)
if _, ok := presets[key]; !ok {
presets[key] = &Preset{unit.Name, *unit.Enabled, false, []string{}, util.GetUnitScope(unit)}
} else {
return fmt.Errorf("%q key is already present in the presets map", key)
}
}
}
if unit.Mask != nil {
if *unit.Mask { // mask: true
relabelpath := ""
// relabelpaths := []string{}
if err := s.Logger.LogOp(
func() error {
var err error
relabelpath, err = s.MaskUnit(unit)
var err error = s.MaskUnit(unit)
return err
},
"masking unit %q", unit.Name,
"masking unit %q", unit.Key(),
); err != nil {
return err
}
s.relabel(relabelpath)

} else { // mask: false
masked, err := s.IsUnitMasked(unit)
if err != nil {
Expand All @@ -108,7 +109,7 @@ func (s *stage) createUnits(config types.Config) error {
func() error {
return s.UnmaskUnit(unit)
},
"unmasking unit %q", unit.Name,
"unmasking unit %q", unit.Key(),
); err != nil {
return err
}
Expand All @@ -118,7 +119,7 @@ func (s *stage) createUnits(config types.Config) error {
}
// if we have presets then create the systemd preset file.
if len(presets) != 0 {
if err := s.createSystemdPresetFile(presets); err != nil {
if err := s.createSystemdPresetFiles(presets); err != nil {
return err
}
}
Expand All @@ -145,30 +146,40 @@ func parseInstanceUnit(unit types.Unit) (string, string, error) {

// createSystemdPresetFile creates the presetfile for enabled/disabled
// systemd units.
func (s *stage) createSystemdPresetFile(presets map[string]*Preset) error {
if err := s.relabelPath(filepath.Join(s.DestDir, util.PresetPath)); err != nil {
return err
func (s *stage) createSystemdPresetFiles(presets map[string]*Preset) error {

//getting directories from presets file for relabling
paths := make(map[string]bool)
for _, preset := range presets {
path := filepath.Dir(s.SystemdPresetPath(preset.scope))
if _, value := paths[path]; !value {
paths[path] = false
if err := s.relabelPath(filepath.Join(s.DestDir, path)); err != nil {
return err
}
paths[path] = true
}
}
hasInstanceUnit := false
for _, value := range presets {
unitString := value.unit
if value.instantiatable {
for _, preset := range presets {
unitString := preset.unit
if preset.instantiatable {
hasInstanceUnit = true
// Let's say we have two instantiated enabled units listed under
// the systemd units i.e. [email protected], [email protected]
// then the unitString will look like "[email protected] foo bar"
unitString = fmt.Sprintf("%s %s", unitString, strings.Join(value.instances, " "))
unitString = fmt.Sprintf("%s %s", unitString, strings.Join(preset.instances, " "))
}
if value.enabled {
if preset.enabled {
if err := s.Logger.LogOp(
func() error { return s.EnableUnit(unitString) },
func() error { return s.EnableUnit(unitString, preset.scope) },
"setting preset to enabled for %q", unitString,
); err != nil {
return err
}
} else {
if err := s.Logger.LogOp(
func() error { return s.DisableUnit(unitString) },
func() error { return s.DisableUnit(unitString, preset.scope) },
"setting preset to disabled for %q", unitString,
); err != nil {
return err
Expand All @@ -191,55 +202,61 @@ func (s *stage) createSystemdPresetFile(presets map[string]*Preset) error {
// applies to the unit's dropins.
func (s *stage) writeSystemdUnit(unit types.Unit) error {
return s.Logger.LogOp(func() error {
relabeledDropinDir := false
for _, dropin := range unit.Dropins {
if dropin.Contents == nil {
continue
}
f, err := s.FileFromSystemdUnitDropin(unit, dropin)
fetchops, err := s.FilesFromSystemdUnitDropin(unit, dropin)
if err != nil {
s.Logger.Crit("error converting systemd dropin: %v", err)
return err
}
// trim off prefix since this needs to be relative to the sysroot
if !strings.HasPrefix(f.Node.Path, s.DestDir) {
panic(fmt.Sprintf("Dropin path %s isn't under prefix %s", f.Node.Path, s.DestDir))
}
relabelPath := f.Node.Path[len(s.DestDir):]
if err := s.Logger.LogOp(
func() error { return s.PerformFetch(f) },
"writing systemd drop-in %q at %q", dropin.Name, f.Node.Path,
); err != nil {
return err
}
if !relabeledDropinDir {
s.relabel(filepath.Dir(relabelPath))
relabeledDropinDir = true
for _, f := range fetchops {
relabeledDropinDir := false
// trim off prefix since this needs to be relative to the sysroot
if !strings.HasPrefix(f.Node.Path, s.DestDir) {
panic(fmt.Sprintf("Dropin path %s isn't under prefix %s", f.Node.Path, s.DestDir))
}
relabelPath := f.Node.Path[len(s.DestDir):]
if err := s.Logger.LogOp(
func() error { return s.PerformFetch(f) },
"writing systemd drop-in %q at %q", dropin.Name, f.Node.Path,
); err != nil {
return err
}
if !relabeledDropinDir {
s.relabel(filepath.Dir(relabelPath))
relabeledDropinDir = true
}
}
}

if cutil.NilOrEmpty(unit.Contents) {
return nil
}

f, err := s.FileFromSystemdUnit(unit)
fetchops, err := s.FilesFromSystemdUnit(unit)
if err != nil {
s.Logger.Crit("error converting unit: %v", err)
return err
}
// trim off prefix since this needs to be relative to the sysroot
if !strings.HasPrefix(f.Node.Path, s.DestDir) {
panic(fmt.Sprintf("Unit path %s isn't under prefix %s", f.Node.Path, s.DestDir))
}
relabelPath := f.Node.Path[len(s.DestDir):]
if err := s.Logger.LogOp(
func() error { return s.PerformFetch(f) },
"writing unit %q at %q", unit.Name, f.Node.Path,
); err != nil {
return err

for _, f := range fetchops {
// trim off prefix since this needs to be relative to the sysroot
if !strings.HasPrefix(f.Node.Path, s.DestDir) {
panic(fmt.Sprintf("Unit path %s isn't under prefix %s", f.Node.Path, s.DestDir))
}
relabelPath := f.Node.Path[len(s.DestDir):]
if err := s.Logger.LogOp(
func() error { return s.PerformFetch(f) },
"writing unit %q at %q", unit.Key(), f.Node.Path,
); err != nil {
return err
}

s.relabel(relabelPath)
}
s.relabel(relabelPath)

return nil
}, "processing unit %q", unit.Name)
}, "processing unit %q", unit.Key())
}
Loading

0 comments on commit 987794f

Please sign in to comment.