2022-05-11 05:13:13 +00:00
|
|
|
package steward
|
|
|
|
|
|
|
|
import (
|
|
|
|
"crypto/sha256"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
2022-05-26 05:13:34 +00:00
|
|
|
"io"
|
2022-05-11 05:13:13 +00:00
|
|
|
"log"
|
|
|
|
"os"
|
2022-05-26 05:13:34 +00:00
|
|
|
"path/filepath"
|
2022-05-11 05:13:13 +00:00
|
|
|
"sort"
|
|
|
|
"strings"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/fxamacker/cbor/v2"
|
|
|
|
"github.com/go-playground/validator/v10"
|
|
|
|
)
|
|
|
|
|
|
|
|
// // centralAuth
|
|
|
|
// type centralAuth struct {
|
|
|
|
// authorization *authorization
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// // newCentralAuth
|
|
|
|
// func newCentralAuth() *centralAuth {
|
|
|
|
// c := centralAuth{
|
|
|
|
// authorization: newAuthorization(),
|
|
|
|
// }
|
|
|
|
//
|
|
|
|
// return &c
|
|
|
|
// }
|
|
|
|
|
|
|
|
// --------------------------------------
|
|
|
|
|
2022-05-18 07:21:14 +00:00
|
|
|
type accessLists struct {
|
2022-05-11 05:13:13 +00:00
|
|
|
// Holds the editable structures for ACL handling.
|
|
|
|
schemaMain *schemaMain
|
|
|
|
// Holds the generated based on the editable structures for ACL handling.
|
|
|
|
schemaGenerated *schemaGenerated
|
|
|
|
validator *validator.Validate
|
2022-05-19 09:53:25 +00:00
|
|
|
errorKernel *errorKernel
|
2022-05-19 18:00:25 +00:00
|
|
|
configuration *Configuration
|
2022-06-01 11:04:34 +00:00
|
|
|
pki *pki
|
2022-05-11 05:13:13 +00:00
|
|
|
}
|
|
|
|
|
2022-06-01 11:04:34 +00:00
|
|
|
func newAccessLists(pki *pki, errorKernel *errorKernel, configuration *Configuration) *accessLists {
|
2022-05-18 07:21:14 +00:00
|
|
|
a := accessLists{
|
2022-05-26 05:13:34 +00:00
|
|
|
schemaMain: newSchemaMain(configuration),
|
2022-05-11 05:13:13 +00:00
|
|
|
schemaGenerated: newSchemaGenerated(),
|
|
|
|
validator: validator.New(),
|
2022-05-19 09:53:25 +00:00
|
|
|
errorKernel: errorKernel,
|
2022-05-19 18:00:25 +00:00
|
|
|
configuration: configuration,
|
2022-06-01 11:04:34 +00:00
|
|
|
pki: pki,
|
2022-05-11 05:13:13 +00:00
|
|
|
}
|
|
|
|
|
2022-05-26 05:13:34 +00:00
|
|
|
// The main acl map gets loaded from disk in the newSchemaMain function, but since that
|
|
|
|
// function do not have access to the generated map we have to generate it here.
|
|
|
|
a.generateACLsForAllNodes()
|
|
|
|
|
2022-05-11 05:13:13 +00:00
|
|
|
return &a
|
|
|
|
}
|
|
|
|
|
2022-05-18 09:26:06 +00:00
|
|
|
// type node string
|
2022-05-11 05:13:13 +00:00
|
|
|
type command string
|
|
|
|
type nodeGroup string
|
|
|
|
type commandGroup string
|
|
|
|
|
|
|
|
// schemaMain is the structure that holds the user editable parts for creating ACL's.
|
|
|
|
type schemaMain struct {
|
2022-05-18 09:26:06 +00:00
|
|
|
ACLMap map[Node]map[Node]map[command]struct{}
|
2022-05-26 05:13:34 +00:00
|
|
|
ACLMapFilePath string
|
2022-05-18 09:26:06 +00:00
|
|
|
NodeGroupMap map[nodeGroup]map[Node]struct{}
|
2022-05-11 05:13:13 +00:00
|
|
|
CommandGroupMap map[commandGroup]map[command]struct{}
|
|
|
|
mu sync.Mutex
|
|
|
|
}
|
|
|
|
|
2022-05-26 05:13:34 +00:00
|
|
|
func newSchemaMain(configuration *Configuration) *schemaMain {
|
2022-05-11 05:13:13 +00:00
|
|
|
s := schemaMain{
|
2022-05-18 09:26:06 +00:00
|
|
|
ACLMap: make(map[Node]map[Node]map[command]struct{}),
|
2022-05-26 05:13:34 +00:00
|
|
|
ACLMapFilePath: filepath.Join(configuration.DatabaseFolder, "central_aclmap.txt"),
|
2022-05-18 09:26:06 +00:00
|
|
|
NodeGroupMap: make(map[nodeGroup]map[Node]struct{}),
|
2022-05-11 05:13:13 +00:00
|
|
|
CommandGroupMap: make(map[commandGroup]map[command]struct{}),
|
|
|
|
}
|
2022-05-26 05:13:34 +00:00
|
|
|
|
|
|
|
// Load ACLMap from disk if present.
|
|
|
|
func() {
|
|
|
|
if _, err := os.Stat(s.ACLMapFilePath); os.IsNotExist(err) {
|
|
|
|
er := fmt.Errorf("error: newSchemaMain: no file for ACLMap found %v: %v", s.ACLMapFilePath, err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
|
|
|
|
// If no aclmap is present on disk we just return from this
|
|
|
|
// function without loading any values.
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
fh, err := os.Open(s.ACLMapFilePath)
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: newSchemaMain: failed to open file for reading %v: %v", s.ACLMapFilePath, err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
b, err := io.ReadAll(fh)
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: newSchemaMain: failed to ReadAll file %v: %v", s.ACLMapFilePath, err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Unmarshal the data read from disk.
|
|
|
|
err = json.Unmarshal(b, &s.ACLMap)
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: newSchemaMain: failed to unmarshal content from file %v: %v", s.ACLMapFilePath, err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate the aclGenerated map happens in the function where this function is called.
|
|
|
|
}()
|
2022-05-11 05:13:13 +00:00
|
|
|
return &s
|
|
|
|
}
|
|
|
|
|
|
|
|
// schemaGenerated is the structure that holds all the generated ACL's
|
|
|
|
// to be sent to nodes.
|
|
|
|
// The ACL's here are generated from the schemaMain.ACLMap.
|
|
|
|
type schemaGenerated struct {
|
2022-05-18 09:26:06 +00:00
|
|
|
ACLsToConvert map[Node]map[Node]map[command]struct{}
|
|
|
|
GeneratedACLsMap map[Node]HostACLsSerializedWithHash
|
2022-05-11 05:13:13 +00:00
|
|
|
mu sync.Mutex
|
|
|
|
}
|
|
|
|
|
|
|
|
func newSchemaGenerated() *schemaGenerated {
|
|
|
|
s := schemaGenerated{
|
2022-05-18 09:26:06 +00:00
|
|
|
ACLsToConvert: map[Node]map[Node]map[command]struct{}{},
|
|
|
|
GeneratedACLsMap: make(map[Node]HostACLsSerializedWithHash),
|
2022-05-11 05:13:13 +00:00
|
|
|
}
|
|
|
|
return &s
|
|
|
|
}
|
|
|
|
|
|
|
|
// HostACLsSerializedWithHash holds the serialized representation node specific ACL's in the authSchema.
|
|
|
|
// There is also a sha256 hash of the data.
|
|
|
|
type HostACLsSerializedWithHash struct {
|
2022-05-24 10:00:38 +00:00
|
|
|
// data is all the ACL's for a specific node serialized serialized into cbor.
|
2022-05-11 05:13:13 +00:00
|
|
|
Data []byte
|
|
|
|
// hash is the sha256 hash of the ACL's.
|
|
|
|
// With maps the order are not guaranteed, so A sorted appearance
|
|
|
|
// of the ACL map for a host node is used when creating the hash,
|
|
|
|
// so the hash stays the same unless the ACL is changed.
|
|
|
|
Hash [32]byte
|
|
|
|
}
|
|
|
|
|
|
|
|
// commandAsSlice will convert the given argument into a slice representation.
|
|
|
|
// If the argument is a group, then all the members of that group will be expanded into
|
|
|
|
// the slice.
|
|
|
|
// If the argument is not a group kind of value, then only a slice with that single
|
|
|
|
// value is returned.
|
2022-05-18 09:26:06 +00:00
|
|
|
func (a *accessLists) nodeAsSlice(n Node) []Node {
|
|
|
|
nodes := []Node{}
|
2022-05-11 05:13:13 +00:00
|
|
|
|
|
|
|
// Check if we are given a nodeGroup variable, and if we are, get all the
|
|
|
|
// nodes for that group.
|
|
|
|
if strings.HasPrefix(string(n), "grp_nodes_") {
|
|
|
|
for nd := range a.schemaMain.NodeGroupMap[nodeGroup(n)] {
|
|
|
|
nodes = append(nodes, nd)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// No group found meaning a single node was given as an argument.
|
2022-05-18 09:26:06 +00:00
|
|
|
nodes = []Node{n}
|
2022-05-11 05:13:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return nodes
|
|
|
|
}
|
|
|
|
|
|
|
|
// commandAsSlice will convert the given argument into a slice representation.
|
|
|
|
// If the argument is a group, then all the members of that group will be expanded into
|
|
|
|
// the slice.
|
|
|
|
// If the argument is not a group kind of value, then only a slice with that single
|
|
|
|
// value is returned.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) commandAsSlice(c command) []command {
|
2022-05-11 05:13:13 +00:00
|
|
|
commands := []command{}
|
|
|
|
|
|
|
|
// Check if we are given a nodeGroup variable, and if we are, get all the
|
|
|
|
// nodes for that group.
|
|
|
|
if strings.HasPrefix(string(c), "grp_commands_") {
|
|
|
|
for cmd := range a.schemaMain.CommandGroupMap[commandGroup(c)] {
|
|
|
|
commands = append(commands, cmd)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// No group found meaning a single node was given as an argument, so we
|
|
|
|
// just put the single node given as the only value in the slice.
|
|
|
|
commands = []command{c}
|
|
|
|
}
|
|
|
|
|
|
|
|
return commands
|
|
|
|
}
|
|
|
|
|
2022-05-18 12:43:35 +00:00
|
|
|
// aclAddCommand will add a command for a fromNode.
|
2022-05-11 05:13:13 +00:00
|
|
|
// If the node or the fromNode do not exist they will be created.
|
|
|
|
// The json encoded schema for a node and the hash of those data
|
|
|
|
// will also be generated.
|
2022-05-18 12:43:35 +00:00
|
|
|
func (a *accessLists) aclAddCommand(host Node, source Node, cmd command) {
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
|
|
|
|
// Check if node exists in map.
|
|
|
|
if _, ok := a.schemaMain.ACLMap[host]; !ok {
|
|
|
|
// log.Printf("info: did not find node=%v in map, creating map[fromnode]map[command]struct{}\n", n)
|
2022-05-18 09:26:06 +00:00
|
|
|
a.schemaMain.ACLMap[host] = make(map[Node]map[command]struct{})
|
2022-05-11 05:13:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check if also source node exists in map
|
|
|
|
if _, ok := a.schemaMain.ACLMap[host][source]; !ok {
|
|
|
|
// log.Printf("info: did not find node=%v in map, creating map[fromnode]map[command]struct{}\n", fn)
|
|
|
|
a.schemaMain.ACLMap[host][source] = make(map[command]struct{})
|
|
|
|
}
|
|
|
|
|
|
|
|
a.schemaMain.ACLMap[host][source][cmd] = struct{}{}
|
|
|
|
// err := a.generateJSONForHostOrGroup(n)
|
|
|
|
err := a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: addCommandForFromNode: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
// fmt.Printf(" * DEBUG: aclNodeFromnodeCommandAdd: a.schemaMain.ACLMap=%v\n", a.schemaMain.ACLMap)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// aclDeleteCommand will delete the specified command from the fromnode.
|
2022-05-18 09:26:06 +00:00
|
|
|
func (a *accessLists) aclDeleteCommand(host Node, source Node, cmd command) error {
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
|
|
|
|
// Check if node exists in map.
|
|
|
|
if _, ok := a.schemaMain.ACLMap[host]; !ok {
|
|
|
|
return fmt.Errorf("authSchema: no such node=%v to delete on in schema exists", host)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := a.schemaMain.ACLMap[host][source]; !ok {
|
|
|
|
return fmt.Errorf("authSchema: no such fromnode=%v to delete on in schema for node=%v exists", source, host)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := a.schemaMain.ACLMap[host][source][cmd]; !ok {
|
|
|
|
return fmt.Errorf("authSchema: no such command=%v from fromnode=%v to delete on in schema for node=%v exists", cmd, source, host)
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(a.schemaMain.ACLMap[host][source], cmd)
|
|
|
|
|
|
|
|
err := a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: aclNodeFromNodeCommandDelete: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// aclDeleteSource will delete specified source node and all commands specified for it.
|
2022-05-18 09:26:06 +00:00
|
|
|
func (a *accessLists) aclDeleteSource(host Node, source Node) error {
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
|
|
|
|
// Check if node exists in map.
|
|
|
|
if _, ok := a.schemaMain.ACLMap[host]; !ok {
|
|
|
|
return fmt.Errorf("authSchema: no such node=%v to delete on in schema exists", host)
|
|
|
|
}
|
|
|
|
|
|
|
|
if _, ok := a.schemaMain.ACLMap[host][source]; !ok {
|
|
|
|
return fmt.Errorf("authSchema: no such fromnode=%v to delete on in schema for node=%v exists", source, host)
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(a.schemaMain.ACLMap[host], source)
|
|
|
|
|
|
|
|
err := a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: aclNodeFromnodeDelete: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// generateACLsForAllNodes will generate a json encoded representation of the node specific
|
|
|
|
// map values of authSchema, along with a hash of the data.
|
|
|
|
//
|
|
|
|
// Will range over all the host elements defined in the ACL, create a new authParser for each one,
|
|
|
|
// and run a small state machine on each element to create the final ACL result to be used at host
|
|
|
|
// nodes.
|
|
|
|
// The result will be written to the schemaGenerated.ACLsToConvert map.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) generateACLsForAllNodes() error {
|
2022-05-26 05:13:34 +00:00
|
|
|
// We first one to save the current main ACLMap.
|
|
|
|
func() {
|
|
|
|
fh, err := os.OpenFile(a.schemaMain.ACLMapFilePath, os.O_CREATE|os.O_TRUNC|os.O_RDWR, 0600)
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: generateACLsForAllNodes: opening file for writing: %v, err: %v", a.schemaMain.ACLMapFilePath, err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
defer fh.Close()
|
|
|
|
|
|
|
|
// a.schemaMain.mu.Lock()
|
|
|
|
// defer a.schemaMain.mu.Unlock()
|
|
|
|
enc := json.NewEncoder(fh)
|
|
|
|
enc.Encode(a.schemaMain.ACLMap)
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: generateACLsForAllNodes: encoding json to file failed: %v, err: %v", a.schemaMain.ACLMapFilePath, err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaGenerated.mu.Lock()
|
|
|
|
defer a.schemaGenerated.mu.Unlock()
|
|
|
|
|
2022-05-18 09:26:06 +00:00
|
|
|
a.schemaGenerated.ACLsToConvert = make(map[Node]map[Node]map[command]struct{})
|
2022-05-11 05:13:13 +00:00
|
|
|
|
|
|
|
// Rangle all ACL's. Both for single hosts, and group of hosts.
|
|
|
|
// ACL's that are for a group of hosts will be generated split
|
|
|
|
// out in it's indivial host name, and that current ACL will
|
|
|
|
// be added to the individual host in the ACLsToConvert map to
|
|
|
|
// built a complete picture of what the ACL's looks like for each
|
|
|
|
// individual hosts.
|
|
|
|
for n := range a.schemaMain.ACLMap {
|
|
|
|
//a.schemaGenerated.ACLsToConvert = make(map[node]map[node]map[command]struct{})
|
|
|
|
ap := newAuthParser(n, a)
|
|
|
|
ap.parse()
|
|
|
|
}
|
|
|
|
|
2022-05-19 18:00:25 +00:00
|
|
|
inf := fmt.Errorf("generateACLsFor all nodes, ACLsToConvert contains: %#v", a.schemaGenerated.ACLsToConvert)
|
|
|
|
a.errorKernel.logConsoleOnlyIfDebug(inf, a.configuration)
|
|
|
|
|
2022-05-11 05:13:13 +00:00
|
|
|
// ACLsToConvert got the complete picture of what ACL's that
|
|
|
|
// are defined for each individual host node.
|
|
|
|
// Range this map, and generate a JSON representation of all
|
|
|
|
// the ACL's each host.
|
|
|
|
func() {
|
2022-05-19 06:27:12 +00:00
|
|
|
// If the map to generate from map is empty we want to also set the generatedACLsMap
|
|
|
|
// to empty so we can make sure that no more generated ACL's exists to be distributed.
|
|
|
|
if len(a.schemaGenerated.ACLsToConvert) == 0 {
|
|
|
|
a.schemaGenerated.GeneratedACLsMap = make(map[Node]HostACLsSerializedWithHash)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-05-11 05:13:13 +00:00
|
|
|
for n, m := range a.schemaGenerated.ACLsToConvert {
|
2022-05-19 18:00:25 +00:00
|
|
|
//fmt.Printf("\n ################ DEBUG: RANGE in generate: n=%v, m=%v\n", n, m)
|
2022-05-11 05:13:13 +00:00
|
|
|
|
|
|
|
// cbor marshal the data of the ACL map to store for the host node.
|
|
|
|
cb, err := cbor.Marshal(m)
|
|
|
|
if err != nil {
|
2022-05-24 13:51:36 +00:00
|
|
|
er := fmt.Errorf("error: failed to generate cbor for host in schemaGenerated: %v", err)
|
2022-05-11 05:13:13 +00:00
|
|
|
log.Printf("%v\n", er)
|
|
|
|
os.Exit(1)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create the hash for the data for the host node.
|
|
|
|
hash := func() [32]byte {
|
|
|
|
sns := a.nodeMapToSlice(n)
|
|
|
|
|
|
|
|
b, err := cbor.Marshal(sns)
|
|
|
|
if err != nil {
|
|
|
|
err := fmt.Errorf("error: authSchema, json for hash: %v", err)
|
|
|
|
log.Printf("%v\n", err)
|
|
|
|
return [32]byte{}
|
|
|
|
}
|
|
|
|
|
|
|
|
hash := sha256.Sum256(b)
|
|
|
|
return hash
|
|
|
|
}()
|
|
|
|
|
|
|
|
// Store both the cbor marshaled data and the hash in a structure.
|
|
|
|
hostSerialized := HostACLsSerializedWithHash{
|
|
|
|
Data: cb,
|
|
|
|
Hash: hash,
|
|
|
|
}
|
|
|
|
|
|
|
|
// and then store the cbor encoded data and the hash in the generated map.
|
|
|
|
a.schemaGenerated.GeneratedACLsMap[n] = hostSerialized
|
|
|
|
|
|
|
|
}
|
|
|
|
}()
|
2022-05-19 18:00:25 +00:00
|
|
|
|
|
|
|
inf = fmt.Errorf("generateACLsFor all nodes, GeneratedACLsMap contains: %#v", a.schemaGenerated.GeneratedACLsMap)
|
|
|
|
a.errorKernel.logConsoleOnlyIfDebug(inf, a.configuration)
|
2022-05-11 05:13:13 +00:00
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// sourceNode is used to convert the ACL map structure of a host into a slice,
|
|
|
|
// and we then use the slice representation of the ACL to create the hash for
|
|
|
|
// a specific host node.
|
|
|
|
type sourceNode struct {
|
2022-05-18 09:26:06 +00:00
|
|
|
HostNode Node
|
2022-05-11 05:13:13 +00:00
|
|
|
SourceCommands []sourceNodeCommands
|
|
|
|
}
|
|
|
|
|
|
|
|
// sourceNodeCommand is used to convert the ACL map structure of a host into a slice,
|
|
|
|
// and we then use the slice representation of the ACL to create the hash for
|
|
|
|
// a specific host node.
|
|
|
|
type sourceNodeCommands struct {
|
2022-05-18 09:26:06 +00:00
|
|
|
Source Node
|
2022-05-11 05:13:13 +00:00
|
|
|
Commands []command
|
|
|
|
}
|
|
|
|
|
|
|
|
// nodeMapToSlice will return a sourceNode structure, with the map sourceNode part
|
|
|
|
// of the map converted into a slice. Both the from node, and the commands
|
|
|
|
// defined for each sourceNode are sorted.
|
|
|
|
// This function is used when creating the hash of the nodeMap since we can not
|
|
|
|
// guarantee the order of a hash map, but we can with a slice.
|
2022-05-18 09:26:06 +00:00
|
|
|
func (a *accessLists) nodeMapToSlice(host Node) sourceNode {
|
2022-05-11 05:13:13 +00:00
|
|
|
srcNodes := sourceNode{
|
|
|
|
HostNode: host,
|
|
|
|
}
|
|
|
|
|
|
|
|
for sn, commandMap := range a.schemaGenerated.ACLsToConvert[host] {
|
|
|
|
srcC := sourceNodeCommands{
|
|
|
|
Source: sn,
|
|
|
|
}
|
|
|
|
|
|
|
|
for cmd := range commandMap {
|
|
|
|
srcC.Commands = append(srcC.Commands, cmd)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort all the commands.
|
|
|
|
sort.SliceStable(srcC.Commands, func(i, j int) bool {
|
|
|
|
return srcC.Commands[i] < srcC.Commands[j]
|
|
|
|
})
|
|
|
|
|
|
|
|
srcNodes.SourceCommands = append(srcNodes.SourceCommands, srcC)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Sort all the source nodes.
|
|
|
|
sort.SliceStable(srcNodes.SourceCommands, func(i, j int) bool {
|
|
|
|
return srcNodes.SourceCommands[i].Source < srcNodes.SourceCommands[j].Source
|
|
|
|
})
|
|
|
|
|
|
|
|
// fmt.Printf(" * nodeMapToSlice: fromNodes: %#v\n", fns)
|
|
|
|
|
|
|
|
return srcNodes
|
|
|
|
}
|
|
|
|
|
|
|
|
// groupNodesAddNode adds a node to a group. If the group does
|
|
|
|
// not exist it will be created.
|
2022-05-18 09:26:06 +00:00
|
|
|
func (a *accessLists) groupNodesAddNode(ng nodeGroup, n Node) {
|
2022-05-11 05:13:13 +00:00
|
|
|
err := a.validator.Var(ng, "startswith=grp_nodes_")
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error: group name do not start with grp_nodes_: %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
if _, ok := a.schemaMain.NodeGroupMap[ng]; !ok {
|
2022-05-18 09:26:06 +00:00
|
|
|
a.schemaMain.NodeGroupMap[ng] = make(map[Node]struct{})
|
2022-05-11 05:13:13 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
a.schemaMain.NodeGroupMap[ng][n] = struct{}{}
|
|
|
|
|
|
|
|
// fmt.Printf(" * groupNodesAddNode: After adding to group node looks like: %+v\n", a.schemaMain.NodeGroupMap)
|
|
|
|
|
|
|
|
err = a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: groupNodesAddNode: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// groupNodesDeleteNode deletes a node from a group in the map.
|
2022-05-18 09:26:06 +00:00
|
|
|
func (a *accessLists) groupNodesDeleteNode(ng nodeGroup, n Node) {
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
if _, ok := a.schemaMain.NodeGroupMap[ng][n]; !ok {
|
|
|
|
log.Printf("info: no such node with name=%v found in group=%v\n", ng, n)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(a.schemaMain.NodeGroupMap[ng], n)
|
|
|
|
|
|
|
|
//fmt.Printf(" * After deleting nodeGroup map looks like: %+v\n", a.schemaMain.NodeGroupMap)
|
|
|
|
|
|
|
|
err := a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: groupNodesDeleteNode: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// groupNodesDeleteGroup deletes a nodeGroup from map.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) groupNodesDeleteGroup(ng nodeGroup) {
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
if _, ok := a.schemaMain.NodeGroupMap[ng]; !ok {
|
|
|
|
log.Printf("info: no such group found: %v\n", ng)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(a.schemaMain.NodeGroupMap, ng)
|
|
|
|
|
|
|
|
//fmt.Printf(" * After deleting nodeGroup map looks like: %+v\n", a.schemaMain.NodeGroupMap)
|
|
|
|
|
|
|
|
err := a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: groupNodesDeleteGroup: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----
|
|
|
|
|
|
|
|
// groupCommandsAddCommand adds a command to a group. If the group does
|
|
|
|
// not exist it will be created.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) groupCommandsAddCommand(cg commandGroup, c command) {
|
2022-05-11 05:13:13 +00:00
|
|
|
err := a.validator.Var(cg, "startswith=grp_commands_")
|
|
|
|
if err != nil {
|
|
|
|
log.Printf("error: group name do not start with grp_commands_ : %v\n", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
if _, ok := a.schemaMain.CommandGroupMap[cg]; !ok {
|
|
|
|
a.schemaMain.CommandGroupMap[cg] = make(map[command]struct{})
|
|
|
|
}
|
|
|
|
|
|
|
|
a.schemaMain.CommandGroupMap[cg][c] = struct{}{}
|
|
|
|
|
|
|
|
//fmt.Printf(" * groupCommandsAddCommand: After adding command=%v to command group=%v map looks like: %+v\n", c, cg, a.schemaMain.CommandGroupMap)
|
|
|
|
|
|
|
|
err = a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: groupCommandsAddCommand: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// groupCommandsDeleteCommand deletes a command from a group in the map.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) groupCommandsDeleteCommand(cg commandGroup, c command) {
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
if _, ok := a.schemaMain.CommandGroupMap[cg][c]; !ok {
|
|
|
|
log.Printf("info: no such command with name=%v found in group=%v\n", c, cg)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(a.schemaMain.CommandGroupMap[cg], c)
|
|
|
|
|
|
|
|
//fmt.Printf(" * After deleting command=%v from group=%v map looks like: %+v\n", c, cg, a.schemaMain.CommandGroupMap)
|
|
|
|
|
|
|
|
err := a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: groupCommandsDeleteCommand: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// groupCommandDeleteGroup deletes a commandGroup map.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) groupCommandDeleteGroup(cg commandGroup) {
|
2022-05-11 05:13:13 +00:00
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
if _, ok := a.schemaMain.CommandGroupMap[cg]; !ok {
|
|
|
|
log.Printf("info: no such group found: %v\n", cg)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
delete(a.schemaMain.CommandGroupMap, cg)
|
|
|
|
|
|
|
|
//fmt.Printf(" * After deleting commandGroup=%v map looks like: %+v\n", cg, a.schemaMain.CommandGroupMap)
|
|
|
|
|
|
|
|
err := a.generateACLsForAllNodes()
|
|
|
|
if err != nil {
|
|
|
|
er := fmt.Errorf("error: groupCommandDeleteGroup: %v", err)
|
|
|
|
log.Printf("%v\n", er)
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// exportACLs will export the current content of the main ACLMap in JSON format.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) exportACLs() ([]byte, error) {
|
2022-05-11 05:13:13 +00:00
|
|
|
|
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
|
|
|
|
js, err := json.Marshal(a.schemaMain.ACLMap)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("error: failed to marshal schemaMain.ACLMap: %v", err)
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return js, nil
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// importACLs will import and replace all current ACL's with the ACL's provided as input.
|
2022-05-18 07:21:14 +00:00
|
|
|
func (a *accessLists) importACLs(js []byte) error {
|
2022-05-11 05:13:13 +00:00
|
|
|
|
|
|
|
a.schemaMain.mu.Lock()
|
|
|
|
defer a.schemaMain.mu.Unlock()
|
|
|
|
|
2022-05-18 09:26:06 +00:00
|
|
|
m := make(map[Node]map[Node]map[command]struct{})
|
2022-05-11 05:13:13 +00:00
|
|
|
|
|
|
|
err := json.Unmarshal(js, &m)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error: failed to unmarshal into ACLMap: %v", err)
|
|
|
|
}
|
|
|
|
|
|
|
|
a.schemaMain.ACLMap = m
|
|
|
|
|
|
|
|
return nil
|
|
|
|
|
|
|
|
}
|