This removes the duplication of the kind string and cleans up things for resource creation.
298 lines
8.8 KiB
Go
298 lines
8.8 KiB
Go
// Mgmt
|
|
// Copyright (C) 2013-2017+ 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 Affero 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 Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
// +build !noaugeas
|
|
|
|
package resources
|
|
|
|
import (
|
|
"fmt"
|
|
"log"
|
|
"os"
|
|
"strings"
|
|
|
|
"github.com/purpleidea/mgmt/recwatch"
|
|
|
|
errwrap "github.com/pkg/errors"
|
|
// FIXME: we vendor go/augeas because master requires augeas 1.6.0
|
|
// and libaugeas-dev-1.6.0 is not yet available in a PPA.
|
|
"honnef.co/go/augeas"
|
|
)
|
|
|
|
const (
|
|
// NS is a namespace for augeas operations
|
|
NS = "Xmgmt"
|
|
)
|
|
|
|
func init() {
|
|
RegisterResource("augeas", func() Res { return &AugeasRes{} })
|
|
}
|
|
|
|
// AugeasRes is a resource that enables you to use the augeas resource.
|
|
// Currently only allows you to change simple files (e.g sshd_config).
|
|
type AugeasRes struct {
|
|
BaseRes `yaml:",inline"`
|
|
|
|
// File is the path to the file targeted by this resource.
|
|
File string `yaml:"file"`
|
|
|
|
// Lens is the lens used by this resource. If specified, mgmt
|
|
// will lower the augeas overhead by only loading that lens.
|
|
Lens string `yaml:"lens"`
|
|
|
|
// Sets is a list of changes that will be applied to the file, in the form of
|
|
// ["path", "value"]. mgmt will run augeas.Get() before augeas.Set(), to
|
|
// prevent changing the file when it is not needed.
|
|
Sets []AugeasSet `yaml:"sets"`
|
|
|
|
recWatcher *recwatch.RecWatcher // used to watch the changed files
|
|
}
|
|
|
|
// AugeasSet represents a key/value pair of settings to be applied.
|
|
type AugeasSet struct {
|
|
Path string `yaml:"path"` // The relative path to the value to be changed.
|
|
Value string `yaml:"value"` // The value to be set on the given Path.
|
|
}
|
|
|
|
// Default returns some sensible defaults for this resource.
|
|
func (obj *AugeasRes) Default() Res {
|
|
return &AugeasRes{
|
|
BaseRes: BaseRes{
|
|
MetaParams: DefaultMetaParams, // force a default
|
|
},
|
|
}
|
|
}
|
|
|
|
// Validate if the params passed in are valid data.
|
|
func (obj *AugeasRes) Validate() error {
|
|
if !strings.HasPrefix(obj.File, "/") {
|
|
return fmt.Errorf("the File param should start with a slash")
|
|
}
|
|
if obj.Lens != "" && !strings.HasSuffix(obj.Lens, ".lns") {
|
|
return fmt.Errorf("the Lens param should have a .lns suffix")
|
|
}
|
|
if (obj.Lens == "") != (obj.File == "") {
|
|
return fmt.Errorf("the File and Lens params must be specified together")
|
|
}
|
|
return obj.BaseRes.Validate()
|
|
}
|
|
|
|
// Init initiates the resource.
|
|
func (obj *AugeasRes) Init() error {
|
|
return obj.BaseRes.Init() // call base init, b/c we're overriding
|
|
}
|
|
|
|
// Watch is the primary listener for this resource and it outputs events.
|
|
// Taken from the File resource.
|
|
// FIXME: DRY - This is taken from the file resource
|
|
func (obj *AugeasRes) Watch() error {
|
|
var err error
|
|
obj.recWatcher, err = recwatch.NewRecWatcher(obj.File, false)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
defer obj.recWatcher.Close()
|
|
|
|
// notify engine that we're running
|
|
if err := obj.Running(); err != nil {
|
|
return err // bubble up a NACK...
|
|
}
|
|
|
|
var send = false // send event?
|
|
var exit *error
|
|
|
|
for {
|
|
if obj.debug {
|
|
log.Printf("%s: Watching: %s", obj, obj.File) // attempting to watch...
|
|
}
|
|
|
|
select {
|
|
case event, ok := <-obj.recWatcher.Events():
|
|
if !ok { // channel shutdown
|
|
return nil
|
|
}
|
|
if err := event.Error; err != nil {
|
|
return errwrap.Wrapf(err, "Unknown %s watcher error", obj)
|
|
}
|
|
if obj.debug { // don't access event.Body if event.Error isn't nil
|
|
log.Printf("%s: Event(%s): %v", obj, event.Body.Name, event.Body.Op)
|
|
}
|
|
send = true
|
|
obj.StateOK(false) // dirty
|
|
|
|
case event := <-obj.Events():
|
|
if exit, send = obj.ReadEvent(event); exit != nil {
|
|
return *exit // exit
|
|
}
|
|
//obj.StateOK(false) // dirty // these events don't invalidate state
|
|
}
|
|
|
|
// do all our event sending all together to avoid duplicate msgs
|
|
if send {
|
|
send = false
|
|
obj.Event()
|
|
}
|
|
}
|
|
}
|
|
|
|
// checkApplySet runs CheckApply for one element of the AugeasRes.Set
|
|
func (obj *AugeasRes) checkApplySet(apply bool, ag *augeas.Augeas, set AugeasSet) (bool, error) {
|
|
fullpath := fmt.Sprintf("/files/%v/%v", obj.File, set.Path)
|
|
|
|
// We do not check for errors because errors are also thrown when
|
|
// the path does not exist.
|
|
if getValue, _ := ag.Get(fullpath); set.Value == getValue {
|
|
// The value is what we expect, return directly
|
|
return true, nil
|
|
}
|
|
|
|
if !apply {
|
|
// If noop, we can return here directly. We return with
|
|
// nil even if err is not nil because it does not mean
|
|
// there is an error.
|
|
return false, nil
|
|
}
|
|
|
|
if err := ag.Set(fullpath, set.Value); err != nil {
|
|
return false, errwrap.Wrapf(err, "augeas: error while setting value")
|
|
}
|
|
|
|
return false, nil
|
|
}
|
|
|
|
// CheckApply method for Augeas resource.
|
|
func (obj *AugeasRes) CheckApply(apply bool) (bool, error) {
|
|
log.Printf("%s: CheckApply: %s", obj, obj.File)
|
|
// By default we do not set any option to augeas, we use the defaults.
|
|
opts := augeas.None
|
|
if obj.Lens != "" {
|
|
// if the lens is specified, we can speed up augeas by not
|
|
// loading everything. Without this option, augeas will try to
|
|
// read all the files it knows in the complete filesystem.
|
|
// e.g. to change /etc/ssh/sshd_config, it would load /etc/hosts, /etc/ntpd.conf, etc...
|
|
opts = augeas.NoModlAutoload
|
|
}
|
|
|
|
// Initiate augeas
|
|
ag, err := augeas.New("/", "", opts)
|
|
if err != nil {
|
|
return false, errwrap.Wrapf(err, "augeas: error while initializing")
|
|
}
|
|
defer ag.Close()
|
|
|
|
if obj.Lens != "" {
|
|
// If the lens is set, load the lens for the file we want to edit.
|
|
// We pick Xmgmt, as this name will not collide with any other lens name.
|
|
// We do not pick Mgmt as in the future there might be an Mgmt lens.
|
|
// https://github.com/hercules-team/augeas/wiki/Loading-specific-files
|
|
if err = ag.Set(fmt.Sprintf("/augeas/load/%s/lens", NS), obj.Lens); err != nil {
|
|
return false, errwrap.Wrapf(err, "augeas: error while initializing lens")
|
|
}
|
|
if err = ag.Set(fmt.Sprintf("/augeas/load/%s/incl", NS), obj.File); err != nil {
|
|
return false, errwrap.Wrapf(err, "augeas: error while initializing incl")
|
|
}
|
|
if err = ag.Load(); err != nil {
|
|
return false, errwrap.Wrapf(err, "augeas: error while loading")
|
|
}
|
|
}
|
|
|
|
checkOK := true
|
|
for _, set := range obj.Sets {
|
|
if setCheckOK, err := obj.checkApplySet(apply, &ag, set); err != nil {
|
|
return false, errwrap.Wrapf(err, "augeas: error during CheckApply of one Set")
|
|
} else if !setCheckOK {
|
|
checkOK = false
|
|
}
|
|
}
|
|
|
|
// If the state is correct or we can't apply, return early.
|
|
if checkOK || !apply {
|
|
return checkOK, nil
|
|
}
|
|
|
|
log.Printf("%s: changes needed, saving", obj)
|
|
if err = ag.Save(); err != nil {
|
|
return false, errwrap.Wrapf(err, "augeas: error while saving augeas values")
|
|
}
|
|
|
|
// FIXME: Workaround for https://github.com/dominikh/go-augeas/issues/13
|
|
// To be fixed upstream.
|
|
if obj.File != "" {
|
|
if _, err := os.Stat(obj.File); os.IsNotExist(err) {
|
|
return false, errwrap.Wrapf(err, "augeas: error: file does not exist")
|
|
}
|
|
}
|
|
|
|
return false, nil
|
|
}
|
|
|
|
// AugeasUID is the UID struct for AugeasRes.
|
|
type AugeasUID struct {
|
|
BaseUID
|
|
name string
|
|
}
|
|
|
|
// UIDs includes all params to make a unique identification of this object.
|
|
func (obj *AugeasRes) UIDs() []ResUID {
|
|
x := &AugeasUID{
|
|
BaseUID: BaseUID{Name: obj.GetName(), Kind: obj.GetKind()},
|
|
name: obj.Name,
|
|
}
|
|
return []ResUID{x}
|
|
}
|
|
|
|
// GroupCmp returns whether two resources can be grouped together or not.
|
|
func (obj *AugeasRes) GroupCmp(r Res) bool {
|
|
return false // Augeas commands can not be grouped together.
|
|
}
|
|
|
|
// Compare two resources and return if they are equivalent.
|
|
func (obj *AugeasRes) Compare(r Res) bool {
|
|
// we can only compare AugeasRes to others of the same resource kind
|
|
res, ok := r.(*AugeasRes)
|
|
if !ok {
|
|
return false
|
|
}
|
|
if !obj.BaseRes.Compare(res) { // call base Compare
|
|
return false
|
|
}
|
|
if obj.Name != res.Name {
|
|
return false
|
|
}
|
|
|
|
return true
|
|
}
|
|
|
|
// UnmarshalYAML is the custom unmarshal handler for this struct.
|
|
// It is primarily useful for setting the defaults.
|
|
func (obj *AugeasRes) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
type rawRes AugeasRes // indirection to avoid infinite recursion
|
|
|
|
def := obj.Default() // get the default
|
|
res, ok := def.(*AugeasRes) // put in the right format
|
|
if !ok {
|
|
return fmt.Errorf("could not convert to AugeasRes")
|
|
}
|
|
raw := rawRes(*res) // convert; the defaults go here
|
|
|
|
if err := unmarshal(&raw); err != nil {
|
|
return err
|
|
}
|
|
|
|
*obj = AugeasRes(raw) // restore from indirection with type conversion!
|
|
return nil
|
|
}
|