Done with: ack '2019+' -l | xargs sed -i -e 's/2019+/2020+/g' Checked manually with: git add -p Hello to future James from 2021, and Happy Hacking!
558 lines
16 KiB
Go
558 lines
16 KiB
Go
// Mgmt
|
|
// Copyright (C) 2013-2020+ James Shubin and the project contributors
|
|
// Written by James Shubin <james@shubin.ca> and the project contributors
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU General Public License as published by
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
// (at your option) any later version.
|
|
//
|
|
// This program is distributed in the hope that it will be useful,
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
// GNU General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
// DOCS: https://godoc.org/github.com/coreos/go-systemd/dbus
|
|
|
|
package resources
|
|
|
|
import (
|
|
"fmt"
|
|
"os/user"
|
|
"path"
|
|
|
|
"github.com/purpleidea/mgmt/engine"
|
|
"github.com/purpleidea/mgmt/engine/traits"
|
|
engineUtil "github.com/purpleidea/mgmt/engine/util"
|
|
"github.com/purpleidea/mgmt/util"
|
|
"github.com/purpleidea/mgmt/util/errwrap"
|
|
|
|
systemd "github.com/coreos/go-systemd/dbus" // change namespace
|
|
systemdUtil "github.com/coreos/go-systemd/util"
|
|
"github.com/godbus/dbus" // namespace collides with systemd wrapper
|
|
)
|
|
|
|
func init() {
|
|
engine.RegisterResource("svc", func() engine.Res { return &SvcRes{} })
|
|
}
|
|
|
|
// SvcRes is a service resource for systemd units.
|
|
type SvcRes struct {
|
|
traits.Base // add the base methods without re-implementation
|
|
traits.Edgeable
|
|
traits.Groupable
|
|
traits.Refreshable
|
|
|
|
init *engine.Init
|
|
|
|
State string `yaml:"state"` // state: running, stopped, undefined
|
|
Startup string `yaml:"startup"` // enabled, disabled, undefined
|
|
Session bool `yaml:"session"` // user session (true) or system?
|
|
}
|
|
|
|
// Default returns some sensible defaults for this resource.
|
|
func (obj *SvcRes) Default() engine.Res {
|
|
return &SvcRes{}
|
|
}
|
|
|
|
// Validate checks if the resource data structure was populated correctly.
|
|
func (obj *SvcRes) Validate() error {
|
|
if obj.State != "running" && obj.State != "stopped" && obj.State != "" {
|
|
return fmt.Errorf("state must be either `running` or `stopped` or undefined")
|
|
}
|
|
if obj.Startup != "enabled" && obj.Startup != "disabled" && obj.Startup != "" {
|
|
return fmt.Errorf("startup must be either `enabled` or `disabled` or undefined")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// Init runs some startup code for this resource.
|
|
func (obj *SvcRes) Init(init *engine.Init) error {
|
|
obj.init = init // save for later
|
|
return nil
|
|
}
|
|
|
|
// Close is run by the engine to clean up after the resource is done.
|
|
func (obj *SvcRes) Close() error {
|
|
return nil
|
|
}
|
|
|
|
// Watch is the primary listener for this resource and it outputs events.
|
|
func (obj *SvcRes) Watch() error {
|
|
// obj.Name: svc name
|
|
if !systemdUtil.IsRunningSystemd() {
|
|
return fmt.Errorf("systemd is not running")
|
|
}
|
|
|
|
var conn *systemd.Conn
|
|
var bus *dbus.Conn
|
|
var err error
|
|
if obj.Session {
|
|
conn, err = systemd.NewUserConnection() // user session
|
|
} else {
|
|
// we want NewSystemConnection but New falls back to this
|
|
conn, err = systemd.New() // needs root access
|
|
}
|
|
if err != nil {
|
|
return errwrap.Wrapf(err, "failed to connect to systemd")
|
|
}
|
|
defer conn.Close()
|
|
|
|
// if we share the bus with others, we will get each others messages!!
|
|
if obj.Session {
|
|
bus, err = util.SessionBusPrivateUsable()
|
|
} else {
|
|
bus, err = util.SystemBusPrivateUsable()
|
|
}
|
|
if err != nil {
|
|
return errwrap.Wrapf(err, "failed to connect to bus")
|
|
}
|
|
defer bus.Close()
|
|
|
|
// XXX: will this detect new units?
|
|
bus.BusObject().Call("org.freedesktop.DBus.AddMatch", 0,
|
|
"type='signal',interface='org.freedesktop.systemd1.Manager',member='Reloading'")
|
|
buschan := make(chan *dbus.Signal, 10)
|
|
defer close(buschan) // NOTE: closing a chan that contains a value is ok
|
|
bus.Signal(buschan)
|
|
defer bus.RemoveSignal(buschan) // not needed here, but nice for symmetry
|
|
|
|
obj.init.Running() // when started, notify engine that we're running
|
|
|
|
var svc = fmt.Sprintf("%s.service", obj.Name()) // systemd name
|
|
var send = false // send event?
|
|
var invalid = false // does the svc exist or not?
|
|
var previous bool // previous invalid value
|
|
|
|
// TODO: do we first need to call conn.Subscribe() ?
|
|
set := conn.NewSubscriptionSet() // no error should be returned
|
|
subChannel, subErrors := set.Subscribe()
|
|
//defer close(subChannel) // cannot close receive-only channel
|
|
//defer close(subErrors) // cannot close receive-only channel
|
|
var activeSet = false
|
|
|
|
for {
|
|
// XXX: watch for an event for new units...
|
|
// XXX: detect if startup enabled/disabled value changes...
|
|
|
|
previous = invalid
|
|
invalid = false
|
|
|
|
// firstly, does svc even exist or not?
|
|
loadstate, err := conn.GetUnitProperty(svc, "LoadState")
|
|
if err != nil {
|
|
obj.init.Logf("failed to get property: %+v", err)
|
|
invalid = true
|
|
}
|
|
|
|
if !invalid {
|
|
var notFound = (loadstate.Value == dbus.MakeVariant("not-found"))
|
|
if notFound { // XXX: in the loop we'll handle changes better...
|
|
obj.init.Logf("failed to find svc")
|
|
invalid = true // XXX: ?
|
|
}
|
|
}
|
|
|
|
if previous != invalid { // if invalid changed, send signal
|
|
send = true
|
|
}
|
|
|
|
if invalid {
|
|
obj.init.Logf("waiting fo service") // waiting for svc to appear...
|
|
if activeSet {
|
|
activeSet = false
|
|
set.Remove(svc) // no return value should ever occur
|
|
}
|
|
|
|
select {
|
|
case <-buschan: // XXX: wait for new units event to unstick
|
|
// loop so that we can see the changed invalid signal
|
|
obj.init.Logf("daemon reload")
|
|
|
|
case <-obj.init.Done: // closed by the engine to signal shutdown
|
|
return nil
|
|
}
|
|
} else {
|
|
if !activeSet {
|
|
activeSet = true
|
|
set.Add(svc) // no return value should ever occur
|
|
}
|
|
|
|
obj.init.Logf("watching...") // attempting to watch...
|
|
select {
|
|
case event := <-subChannel:
|
|
|
|
obj.init.Logf("event: %+v", event)
|
|
// NOTE: the value returned is a map for some reason...
|
|
if event[svc] != nil {
|
|
// event[svc].ActiveState is not nil
|
|
|
|
switch event[svc].ActiveState {
|
|
case "active":
|
|
obj.init.Logf("started")
|
|
case "inactive":
|
|
obj.init.Logf("stopped")
|
|
case "reloading":
|
|
obj.init.Logf("reloading")
|
|
case "failed":
|
|
obj.init.Logf("failed")
|
|
case "activating":
|
|
obj.init.Logf("activating")
|
|
case "deactivating":
|
|
obj.init.Logf("deactivating")
|
|
default:
|
|
return fmt.Errorf("unknown svc state: %s", event[svc].ActiveState)
|
|
}
|
|
} else {
|
|
// svc stopped (and ActiveState is nil...)
|
|
obj.init.Logf("stopped")
|
|
}
|
|
send = true
|
|
|
|
case err := <-subErrors:
|
|
return errwrap.Wrapf(err, "unknown %s error", obj)
|
|
|
|
case <-obj.init.Done: // closed by the engine to signal shutdown
|
|
return nil
|
|
}
|
|
}
|
|
|
|
if send {
|
|
send = false
|
|
obj.init.Event() // notify engine of an event (this can block)
|
|
}
|
|
}
|
|
}
|
|
|
|
// CheckApply checks the resource state and applies the resource if the bool
|
|
// input is true. It returns error info and if the state check passed or not.
|
|
func (obj *SvcRes) CheckApply(apply bool) (bool, error) {
|
|
if !systemdUtil.IsRunningSystemd() {
|
|
return false, fmt.Errorf("systemd is not running")
|
|
}
|
|
|
|
var conn *systemd.Conn
|
|
var err error
|
|
if obj.Session {
|
|
conn, err = systemd.NewUserConnection() // user session
|
|
} else {
|
|
// we want NewSystemConnection but New falls back to this
|
|
conn, err = systemd.New() // needs root access
|
|
}
|
|
if err != nil {
|
|
return false, errwrap.Wrapf(err, "failed to connect to systemd")
|
|
}
|
|
defer conn.Close()
|
|
|
|
var svc = fmt.Sprintf("%s.service", obj.Name()) // systemd name
|
|
|
|
loadstate, err := conn.GetUnitProperty(svc, "LoadState")
|
|
if err != nil {
|
|
return false, errwrap.Wrapf(err, "failed to get load state")
|
|
}
|
|
|
|
// NOTE: we have to compare variants with other variants, they are really strings...
|
|
var notFound = (loadstate.Value == dbus.MakeVariant("not-found"))
|
|
if notFound {
|
|
return false, errwrap.Wrapf(err, "failed to find svc: %s", svc)
|
|
}
|
|
|
|
// XXX: check svc "enabled at boot" or not status...
|
|
|
|
//conn.GetUnitProperties(svc)
|
|
activestate, err := conn.GetUnitProperty(svc, "ActiveState")
|
|
if err != nil {
|
|
return false, errwrap.Wrapf(err, "failed to get active state")
|
|
}
|
|
|
|
var running = (activestate.Value == dbus.MakeVariant("active"))
|
|
var stateOK = ((obj.State == "") || (obj.State == "running" && running) || (obj.State == "stopped" && !running))
|
|
var startupOK = true // XXX: DETECT AND SET
|
|
|
|
// NOTE: if this svc resource is embedded as a composite resource inside
|
|
// of another resource using a technique such as `makeComposite()`, then
|
|
// the Init of the embedded resource is traditionally passed through and
|
|
// identical to the parent's Init. As a result, the data matches what is
|
|
// expected from the parent. (So this luckily turns out to be actually a
|
|
// thing that does help, although it is important to add the Refreshable
|
|
// trait to the parent resource, or we'll panic when we call this line.)
|
|
// It might not be recommended to use the Watch method without a thought
|
|
// to what actually happens when we would run Send(), and other methods.
|
|
var refresh = obj.init.Refresh() // do we have a pending reload to apply?
|
|
|
|
if stateOK && startupOK && !refresh {
|
|
return true, nil // we are in the correct state
|
|
}
|
|
|
|
// state is not okay, no work done, exit, but without error
|
|
if !apply {
|
|
return false, nil
|
|
}
|
|
|
|
// apply portion
|
|
obj.init.Logf("Apply")
|
|
var files = []string{svc} // the svc represented in a list
|
|
if obj.Startup == "enabled" {
|
|
_, _, err = conn.EnableUnitFiles(files, false, true)
|
|
|
|
} else if obj.Startup == "disabled" {
|
|
_, err = conn.DisableUnitFiles(files, false)
|
|
}
|
|
|
|
if err != nil {
|
|
return false, errwrap.Wrapf(err, "unable to change startup status")
|
|
}
|
|
|
|
// XXX: do we need to use a buffered channel here?
|
|
result := make(chan string, 1) // catch result information
|
|
|
|
if obj.State == "running" {
|
|
_, err = conn.StartUnit(svc, "fail", result)
|
|
if err != nil {
|
|
return false, errwrap.Wrapf(err, "failed to start unit")
|
|
}
|
|
if refresh {
|
|
obj.init.Logf("Skipping reload, due to pending start")
|
|
}
|
|
refresh = false // we did a start, so a reload is not needed
|
|
} else if obj.State == "stopped" {
|
|
_, err = conn.StopUnit(svc, "fail", result)
|
|
if err != nil {
|
|
return false, errwrap.Wrapf(err, "failed to stop unit")
|
|
}
|
|
if refresh {
|
|
obj.init.Logf("Skipping reload, due to pending stop")
|
|
}
|
|
refresh = false // we did a stop, so a reload is not needed
|
|
}
|
|
|
|
status := <-result
|
|
if &status == nil {
|
|
return false, fmt.Errorf("systemd service action result is nil")
|
|
}
|
|
switch status {
|
|
case "done":
|
|
// pass
|
|
case "failed":
|
|
return false, fmt.Errorf("svc failed (selinux?)")
|
|
default:
|
|
return false, fmt.Errorf("unknown systemd return string: %v", status)
|
|
}
|
|
|
|
if refresh { // we need to reload the service
|
|
// XXX: run a svc reload here!
|
|
obj.init.Logf("Reloading...")
|
|
}
|
|
|
|
// XXX: also set enabled on boot
|
|
|
|
return false, nil // success
|
|
}
|
|
|
|
// Cmp compares two resources and returns an error if they are not equivalent.
|
|
func (obj *SvcRes) Cmp(r engine.Res) error {
|
|
// we can only compare SvcRes to others of the same resource kind
|
|
res, ok := r.(*SvcRes)
|
|
if !ok {
|
|
return fmt.Errorf("not a %s", obj.Kind())
|
|
}
|
|
|
|
if obj.State != res.State {
|
|
return fmt.Errorf("the State differs")
|
|
}
|
|
if obj.Startup != res.Startup {
|
|
return fmt.Errorf("the Startup differs")
|
|
}
|
|
if obj.Session != res.Session {
|
|
return fmt.Errorf("the Session differs")
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
// SvcUID is the UID struct for SvcRes.
|
|
type SvcUID struct {
|
|
// NOTE: there is also a name variable in the BaseUID struct, this is
|
|
// information about where this UID came from, and is unrelated to the
|
|
// information about the resource we're matching. That data which is
|
|
// used in the IFF function, is what you see in the struct fields here.
|
|
engine.BaseUID
|
|
name string // the svc name
|
|
session bool // user session
|
|
}
|
|
|
|
// IFF aka if and only if they are equivalent, return true. If not, false.
|
|
func (obj *SvcUID) IFF(uid engine.ResUID) bool {
|
|
res, ok := uid.(*SvcUID)
|
|
if !ok {
|
|
return false
|
|
}
|
|
if obj.name != res.name {
|
|
return false
|
|
}
|
|
if obj.session != res.session {
|
|
return false
|
|
}
|
|
return true
|
|
}
|
|
|
|
// SvcResAutoEdges holds the state of the auto edge generator.
|
|
type SvcResAutoEdges struct {
|
|
data []engine.ResUID
|
|
pointer int
|
|
found bool
|
|
}
|
|
|
|
// Next returns the next automatic edge.
|
|
func (obj *SvcResAutoEdges) Next() []engine.ResUID {
|
|
if obj.found {
|
|
panic("shouldn't be called anymore!")
|
|
}
|
|
if len(obj.data) == 0 { // check length for rare scenarios
|
|
return nil
|
|
}
|
|
value := obj.data[obj.pointer]
|
|
obj.pointer++
|
|
return []engine.ResUID{value} // we return one, even though api supports N
|
|
}
|
|
|
|
// Test gets results of the earlier Next() call, & returns if we should continue!
|
|
func (obj *SvcResAutoEdges) Test(input []bool) bool {
|
|
// if there aren't any more remaining
|
|
if len(obj.data) <= obj.pointer {
|
|
return false
|
|
}
|
|
if obj.found { // already found, done!
|
|
return false
|
|
}
|
|
if len(input) != 1 { // in case we get given bad data
|
|
panic("expecting a single value")
|
|
}
|
|
if input[0] { // if a match is found, we're done!
|
|
obj.found = true // no more to find!
|
|
return false
|
|
}
|
|
return true // keep going
|
|
}
|
|
|
|
// SvcResAutoEdgesCron holds the state of the svc -> cron auto edge generator.
|
|
type SvcResAutoEdgesCron struct {
|
|
unit string // target unit
|
|
session bool // user session
|
|
}
|
|
|
|
// Next returns the next automatic edge.
|
|
func (obj *SvcResAutoEdgesCron) Next() []engine.ResUID {
|
|
// XXX: should this be true if SvcRes State == "stopped"?
|
|
reversed := false
|
|
value := &CronUID{
|
|
BaseUID: engine.BaseUID{
|
|
Kind: "CronRes",
|
|
Reversed: &reversed,
|
|
},
|
|
unit: obj.unit, // target unit
|
|
session: obj.session, // user session
|
|
}
|
|
return []engine.ResUID{value} // we return one, even though api supports N
|
|
}
|
|
|
|
// Test takes the output of the last call to Next() and outputs true if we
|
|
// should continue.
|
|
func (obj *SvcResAutoEdgesCron) Test([]bool) bool {
|
|
return false // only get one svc -> cron edge
|
|
}
|
|
|
|
// AutoEdges returns the AutoEdge interface. In this case, systemd unit file
|
|
// resources and cron (systemd-timer) resources.
|
|
func (obj *SvcRes) AutoEdges() (engine.AutoEdge, error) {
|
|
var data []engine.ResUID
|
|
var svcFiles []string
|
|
svcFiles = []string{
|
|
// root svc
|
|
fmt.Sprintf("/etc/systemd/system/%s.service", obj.Name()), // takes precedence
|
|
fmt.Sprintf("/usr/lib/systemd/system/%s.service", obj.Name()), // pkg default
|
|
}
|
|
if obj.Session {
|
|
// user svc
|
|
u, err := user.Current()
|
|
if err != nil {
|
|
return nil, errwrap.Wrapf(err, "error getting current user")
|
|
}
|
|
if u.HomeDir == "" {
|
|
return nil, fmt.Errorf("user has no home directory")
|
|
}
|
|
svcFiles = []string{
|
|
path.Join(u.HomeDir, "/.config/systemd/user/", fmt.Sprintf("%s.service", obj.Name())),
|
|
}
|
|
}
|
|
for _, x := range svcFiles {
|
|
var reversed = true
|
|
data = append(data, &FileUID{
|
|
BaseUID: engine.BaseUID{
|
|
Name: obj.Name(),
|
|
Kind: obj.Kind(),
|
|
Reversed: &reversed,
|
|
},
|
|
path: x, // what matters
|
|
})
|
|
}
|
|
|
|
fileEdge := &FileResAutoEdges{
|
|
data: data,
|
|
pointer: 0,
|
|
found: false,
|
|
}
|
|
cronEdge := &SvcResAutoEdgesCron{
|
|
session: obj.Session,
|
|
unit: fmt.Sprintf("%s.service", obj.Name()),
|
|
}
|
|
|
|
return engineUtil.AutoEdgeCombiner(fileEdge, cronEdge)
|
|
}
|
|
|
|
// UIDs includes all params to make a unique identification of this object.
|
|
// Most resources only return one, although some resources can return multiple.
|
|
func (obj *SvcRes) UIDs() []engine.ResUID {
|
|
x := &SvcUID{
|
|
BaseUID: engine.BaseUID{Name: obj.Name(), Kind: obj.Kind()},
|
|
name: obj.Name(), // svc name
|
|
session: obj.Session, // user session
|
|
}
|
|
return []engine.ResUID{x}
|
|
}
|
|
|
|
// GroupCmp returns whether two resources can be grouped together or not.
|
|
//func (obj *SvcRes) GroupCmp(r engine.GroupableRes) error {
|
|
// _, ok := r.(*SvcRes)
|
|
// if !ok {
|
|
// return fmt.Errorf("resource is not the same kind")
|
|
// }
|
|
// // TODO: depending on if the systemd service api allows batching, we
|
|
// // might be able to build this, although not sure how useful it is...
|
|
// // it might just eliminate parallelism by bunching up the graph
|
|
// return fmt.Errorf("not possible at the moment")
|
|
//}
|
|
|
|
// UnmarshalYAML is the custom unmarshal handler for this struct.
|
|
// It is primarily useful for setting the defaults.
|
|
func (obj *SvcRes) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
type rawRes SvcRes // indirection to avoid infinite recursion
|
|
|
|
def := obj.Default() // get the default
|
|
res, ok := def.(*SvcRes) // put in the right format
|
|
if !ok {
|
|
return fmt.Errorf("could not convert to SvcRes")
|
|
}
|
|
raw := rawRes(*res) // convert; the defaults go here
|
|
|
|
if err := unmarshal(&raw); err != nil {
|
|
return err
|
|
}
|
|
|
|
*obj = SvcRes(raw) // restore from indirection with type conversion!
|
|
return nil
|
|
}
|