Improved logging
* Added dependency on logrus. * Command line flags are parsed in order to obtain the name of the configuration file, the log level and the instance identifier. * Logging in various places : configuration loader, API, data access. Privilege computations and subsequent actions do not write proper logs yet.
This commit is contained in:
parent
e065dab3a2
commit
dcd3f920c9
3 changed files with 199 additions and 71 deletions
|
@ -52,10 +52,14 @@ go build
|
||||||
To Do
|
To Do
|
||||||
------
|
------
|
||||||
|
|
||||||
* Proper logging.
|
* Proper logging, work in progress:
|
||||||
|
* Add logs to the privilege computations and related API calls.
|
||||||
|
* Sending logs to... well, Graylog... through CLI switches.
|
||||||
|
* Writing logs to a file.
|
||||||
|
* Document command line flags.
|
||||||
|
* Cache LDAP username lookups
|
||||||
* Add TLS options (skip checks / specify CA) for the Graylog API.
|
* Add TLS options (skip checks / specify CA) for the Graylog API.
|
||||||
* Read object ownership using `grn_permissions` to preserve privileges on users'
|
* Read object ownership using `grn_permissions` to preserve privileges on users'
|
||||||
own objects
|
own objects
|
||||||
* Support granting ownership on objects
|
* Support granting ownership on objects
|
||||||
* Cleaner CLI
|
|
||||||
* Use goroutines ? Maybe.
|
* Use goroutines ? Maybe.
|
||||||
|
|
1
go.mod
1
go.mod
|
@ -4,6 +4,7 @@ go 1.15
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/go-ldap/ldap v3.0.3+incompatible
|
github.com/go-ldap/ldap v3.0.3+incompatible
|
||||||
|
github.com/sirupsen/logrus v1.7.0
|
||||||
gopkg.in/asn1-ber.v1 v1.0.0-20181015200546-f715ec2f112d // indirect
|
gopkg.in/asn1-ber.v1 v1.0.0-20181015200546-f715ec2f112d // indirect
|
||||||
gopkg.in/yaml.v2 v2.2.2
|
gopkg.in/yaml.v2 v2.2.2
|
||||||
)
|
)
|
||||||
|
|
243
main.go
243
main.go
|
@ -5,19 +5,35 @@ import (
|
||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"crypto/x509"
|
"crypto/x509"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io"
|
"io"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"log"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"os"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/go-ldap/ldap"
|
"github.com/go-ldap/ldap"
|
||||||
|
"github.com/sirupsen/logrus"
|
||||||
"gopkg.in/yaml.v2"
|
"gopkg.in/yaml.v2"
|
||||||
)
|
)
|
||||||
|
|
||||||
type (
|
type (
|
||||||
|
// This structure contains all values that may be set from the command line.
|
||||||
|
cliFlags struct {
|
||||||
|
// The path to the configuration file.
|
||||||
|
cfgFile string
|
||||||
|
// The name of the instance, to be used in logs.
|
||||||
|
instance string
|
||||||
|
// The log level.
|
||||||
|
logLevel string
|
||||||
|
}
|
||||||
|
|
||||||
|
// LDAP connection encapsulation, including a logger.
|
||||||
|
ldapConn struct {
|
||||||
|
conn *ldap.Conn
|
||||||
|
log *logrus.Entry
|
||||||
|
}
|
||||||
|
|
||||||
/* *
|
/* *
|
||||||
* CONFIGURATION DATA *
|
* CONFIGURATION DATA *
|
||||||
* */
|
* */
|
||||||
|
@ -28,7 +44,6 @@ type (
|
||||||
Port uint16
|
Port uint16
|
||||||
Tls string
|
Tls string
|
||||||
TlsNoVerify bool `yaml:"tls_skip_verify"`
|
TlsNoVerify bool `yaml:"tls_skip_verify"`
|
||||||
TlsAllowCnOnly bool `yaml:"tls_allow_cn_only"`
|
|
||||||
CaChain string `yaml:"cachain"`
|
CaChain string `yaml:"cachain"`
|
||||||
BindUser string `yaml:"bind_user"`
|
BindUser string `yaml:"bind_user"`
|
||||||
BindPassword string `yaml:"bind_password"`
|
BindPassword string `yaml:"bind_password"`
|
||||||
|
@ -109,19 +124,35 @@ var (
|
||||||
"stream:read": {"streams:read:%s"},
|
"stream:read": {"streams:read:%s"},
|
||||||
"stream:write": {"streams:read:%s", "streams:edit:%s", "streams:changestate:%s"},
|
"stream:write": {"streams:read:%s", "streams:edit:%s", "streams:changestate:%s"},
|
||||||
}
|
}
|
||||||
|
// The logging context.
|
||||||
|
log *logrus.Entry
|
||||||
)
|
)
|
||||||
|
|
||||||
// Load and check the configuration file
|
// Check group/privilege mapping configuration
|
||||||
func loadConfiguration() (configuration Configuration) {
|
func checkPrivMapping(cfg GroupMapping, log *logrus.Entry) {
|
||||||
var cfgFile string
|
for group, info := range cfg {
|
||||||
if len(os.Args) < 2 {
|
log := log.WithField("group", group)
|
||||||
cfgFile = "graylog-groups.yml"
|
for index, priv := range info.Privileges {
|
||||||
} else {
|
log := log.WithField("entry", index)
|
||||||
cfgFile = os.Args[1]
|
if !graylogItems[priv.Type] {
|
||||||
|
log.WithField("item", priv.Type).
|
||||||
|
Fatal("Invalid Graylog item")
|
||||||
}
|
}
|
||||||
cfgData, err := ioutil.ReadFile(cfgFile)
|
if _, ok := privLevels[priv.Level]; !ok {
|
||||||
|
log.WithField("level", priv.Type).
|
||||||
|
Fatal("Invalid privilege level")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Load and check the configuration file
|
||||||
|
func loadConfiguration(flags cliFlags) (configuration Configuration) {
|
||||||
|
log := log.WithField("config", flags.cfgFile)
|
||||||
|
log.Trace("Loading configuration")
|
||||||
|
cfgData, err := ioutil.ReadFile(flags.cfgFile)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("could not load configuration: %v", err)
|
log.WithField("error", err).Fatal("Could not load configuration")
|
||||||
}
|
}
|
||||||
|
|
||||||
configuration = Configuration{
|
configuration = Configuration{
|
||||||
|
@ -132,29 +163,26 @@ func loadConfiguration() (configuration Configuration) {
|
||||||
}
|
}
|
||||||
err = yaml.Unmarshal(cfgData, &configuration)
|
err = yaml.Unmarshal(cfgData, &configuration)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("could not parse configuration: %v", err)
|
log.WithField("error", err).Fatal("Could not parse configuration")
|
||||||
}
|
|
||||||
|
|
||||||
for _, info := range configuration.Mapping {
|
|
||||||
for _, priv := range info.Privileges {
|
|
||||||
if !graylogItems[priv.Type] {
|
|
||||||
log.Fatalf("invalid Graylog item %s", priv.Type)
|
|
||||||
}
|
|
||||||
if _, ok := privLevels[priv.Level]; !ok {
|
|
||||||
log.Fatalf("invalid privilege level %s", priv.Level)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
checkPrivMapping(configuration.Mapping, log)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Execute a Graylog API request, returning the status code and the body
|
// Execute a Graylog API request, returning the status code and the body
|
||||||
func executeApiCall(cfg GraylogConfig, method string, path string, data io.Reader) (status int, body []byte) {
|
func executeApiCall(cfg GraylogConfig, method string, path string, data io.Reader) (status int, body []byte) {
|
||||||
|
log := log.WithFields(logrus.Fields{
|
||||||
|
"base": cfg.ApiBase,
|
||||||
|
"username": cfg.Username,
|
||||||
|
"method": method,
|
||||||
|
"path": path,
|
||||||
|
})
|
||||||
|
log.Trace("Executing Graylog API call")
|
||||||
client := &http.Client{}
|
client := &http.Client{}
|
||||||
request, err := http.NewRequest(method, fmt.Sprintf("%s/%s", cfg.ApiBase, path), data)
|
request, err := http.NewRequest(method, fmt.Sprintf("%s/%s", cfg.ApiBase, path), data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("could not create HTTP request: %v", err)
|
log.WithField("error", err).Fatal("Could not create HTTP request")
|
||||||
}
|
}
|
||||||
request.SetBasicAuth(cfg.Username, cfg.Password)
|
request.SetBasicAuth(cfg.Username, cfg.Password)
|
||||||
if data != nil {
|
if data != nil {
|
||||||
|
@ -163,27 +191,29 @@ func executeApiCall(cfg GraylogConfig, method string, path string, data io.Reade
|
||||||
request.Header.Add("X-Requested-By", "graylog-groups")
|
request.Header.Add("X-Requested-By", "graylog-groups")
|
||||||
response, err := client.Do(request)
|
response, err := client.Do(request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("could not execute %s %s request on Graylog at %s: %v", method, path, cfg.ApiBase, err)
|
log.WithField("error", err).Fatal("Could not execute HTTP request")
|
||||||
}
|
}
|
||||||
defer response.Body.Close()
|
defer response.Body.Close()
|
||||||
status = response.StatusCode
|
status = response.StatusCode
|
||||||
body, err = ioutil.ReadAll(response.Body)
|
body, err = ioutil.ReadAll(response.Body)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("could not read Graylog response: %v", err)
|
log.WithField("error", err).Fatal("Could not read Graylog response")
|
||||||
}
|
}
|
||||||
|
log.WithField("status", status).Trace("Executed Graylog API call")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the list of Graylog users that have been imported from LDAP
|
// Get the list of Graylog users that have been imported from LDAP
|
||||||
func getGraylogUsers(configuration GraylogConfig) (users []GraylogUser) {
|
func getGraylogUsers(configuration GraylogConfig) (users []GraylogUser) {
|
||||||
|
log.Trace("Getting users from the Graylog API")
|
||||||
status, body := executeApiCall(configuration, "GET", "users", nil)
|
status, body := executeApiCall(configuration, "GET", "users", nil)
|
||||||
if status != 200 {
|
if status != 200 {
|
||||||
log.Fatalf("could not read users: status code %v", status)
|
log.WithField("status", status).Fatal("Could not read users")
|
||||||
}
|
}
|
||||||
|
|
||||||
data := GlUsers{}
|
data := GlUsers{}
|
||||||
if err := json.Unmarshal(body, &data); err != nil {
|
if err := json.Unmarshal(body, &data); err != nil {
|
||||||
log.Fatalf("could not parse Graylog's user response: %v", err)
|
log.WithField("error", err).Fatal("Could not parse Graylog's user list")
|
||||||
}
|
}
|
||||||
|
|
||||||
users = make([]GraylogUser, 0)
|
users = make([]GraylogUser, 0)
|
||||||
|
@ -192,84 +222,128 @@ func getGraylogUsers(configuration GraylogConfig) (users []GraylogUser) {
|
||||||
users = append(users, item.GraylogUser)
|
users = append(users, item.GraylogUser)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
log.WithField("users", len(users)).Info("Obtained users from the Graylog API")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Establish a connection to the LDAP server
|
// Establish a connection to the LDAP server
|
||||||
func getLdapConnection(cfg LdapConfig) (conn *ldap.Conn) {
|
func getLdapConnection(cfg LdapConfig) (conn ldapConn) {
|
||||||
|
dest := fmt.Sprintf("%s:%d", cfg.Host, cfg.Port)
|
||||||
|
log := log.WithFields(logrus.Fields{
|
||||||
|
"ldap_server": dest,
|
||||||
|
"ldap_tls": cfg.Tls,
|
||||||
|
})
|
||||||
|
log.Trace("Establishing LDAP connection")
|
||||||
|
|
||||||
tlsConfig := &tls.Config{
|
tlsConfig := &tls.Config{
|
||||||
InsecureSkipVerify: cfg.TlsNoVerify,
|
InsecureSkipVerify: cfg.TlsNoVerify,
|
||||||
}
|
}
|
||||||
if cfg.Tls != "no" && cfg.CaChain != "" {
|
if cfg.Tls != "no" && cfg.CaChain != "" {
|
||||||
|
log := log.WithField("cachain", cfg.CaChain)
|
||||||
data, err := ioutil.ReadFile(cfg.CaChain)
|
data, err := ioutil.ReadFile(cfg.CaChain)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("failed to read CA certificate chain from %s", cfg.CaChain)
|
log.WithField("error", err).Fatal("Failed to read CA certificate chain")
|
||||||
}
|
}
|
||||||
pool := x509.NewCertPool()
|
pool := x509.NewCertPool()
|
||||||
if !pool.AppendCertsFromPEM(data) {
|
if !pool.AppendCertsFromPEM(data) {
|
||||||
log.Fatalf("could not add CA certificates from %s", cfg.CaChain)
|
log.Fatal("Could not add CA certificates")
|
||||||
}
|
}
|
||||||
tlsConfig.RootCAs = pool
|
tlsConfig.RootCAs = pool
|
||||||
}
|
}
|
||||||
|
|
||||||
var err error
|
var err error
|
||||||
dest := fmt.Sprintf("%s:%d", cfg.Host, cfg.Port)
|
var lc *ldap.Conn
|
||||||
if cfg.Tls == "yes" {
|
if cfg.Tls == "yes" {
|
||||||
conn, err = ldap.DialTLS("tcp", dest, tlsConfig)
|
lc, err = ldap.DialTLS("tcp", dest, tlsConfig)
|
||||||
} else {
|
} else {
|
||||||
conn, err = ldap.Dial("tcp", dest)
|
lc, err = ldap.Dial("tcp", dest)
|
||||||
|
}
|
||||||
|
conn = ldapConn{
|
||||||
|
conn: lc,
|
||||||
|
log: log,
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("failed to connect to LDAP server %s: %v", cfg.Host, err)
|
conn.log.WithField("error", err).Fatal("Failed to connect to the LDAP server")
|
||||||
}
|
}
|
||||||
|
|
||||||
if cfg.Tls == "starttls" {
|
if cfg.Tls == "starttls" {
|
||||||
err = conn.StartTLS(tlsConfig)
|
err = lc.StartTLS(tlsConfig)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
conn.Close()
|
lc.Close()
|
||||||
log.Fatalf("LDAP server %s, StartTLS failed: %v", cfg.Host, err)
|
conn.log.WithField("error", err).Fatal("StartTLS failed")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if cfg.BindUser != "" {
|
||||||
|
conn.log = conn.log.WithField("ldap_user", cfg.BindUser)
|
||||||
|
err := lc.Bind(cfg.BindUser, cfg.BindPassword)
|
||||||
|
if err != nil {
|
||||||
|
conn.close()
|
||||||
|
conn.log.WithField("error", err).Fatal("Could not bind")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Debug("LDAP connection established")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Run a LDAP query to obtain a single object.
|
// Run a LDAP query to obtain a single object.
|
||||||
func executeQuery(conn *ldap.Conn, dn string, attrs []string) (bool, *ldap.Entry) {
|
func (conn ldapConn) query(dn string, attrs []string) (bool, *ldap.Entry) {
|
||||||
|
log := conn.log.WithFields(logrus.Fields{
|
||||||
|
"dn": dn,
|
||||||
|
"attributes": attrs,
|
||||||
|
})
|
||||||
|
log.Trace("Accessing DN")
|
||||||
req := ldap.NewSearchRequest(
|
req := ldap.NewSearchRequest(
|
||||||
dn,
|
dn,
|
||||||
ldap.ScopeBaseObject, ldap.NeverDerefAliases, 1, 0, false,
|
ldap.ScopeBaseObject, ldap.NeverDerefAliases, 1, 0, false,
|
||||||
"(objectClass=*)", attrs, nil)
|
"(objectClass=*)", attrs, nil)
|
||||||
res, err := conn.Search(req)
|
res, err := conn.conn.Search(req)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ldapError, ok := err.(*ldap.Error)
|
ldapError, ok := err.(*ldap.Error)
|
||||||
if ok && ldapError.ResultCode == ldap.LDAPResultNoSuchObject {
|
if ok && ldapError.ResultCode == ldap.LDAPResultNoSuchObject {
|
||||||
|
log.Trace("DN not found")
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
log.Fatalf("LDAP search for %s: %v", dn, err)
|
log.WithField("error", err).Fatal("LDAP query failed")
|
||||||
}
|
}
|
||||||
if len(res.Entries) > 1 {
|
if len(res.Entries) > 1 {
|
||||||
log.Printf("LDAP search for %s returned more than 1 record", dn)
|
log.WithField("results", len(res.Entries)).
|
||||||
|
Warning("LDAP search returned more than 1 record")
|
||||||
return false, nil
|
return false, nil
|
||||||
}
|
}
|
||||||
|
log.Trace("Obtained LDAP object")
|
||||||
return true, res.Entries[0]
|
return true, res.Entries[0]
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Close a LDAP connection
|
||||||
|
func (conn ldapConn) close() {
|
||||||
|
conn.log.Trace("Closing LDAP connection")
|
||||||
|
conn.conn.Close()
|
||||||
|
}
|
||||||
|
|
||||||
// Read a username from a LDAP record based on a DN.
|
// Read a username from a LDAP record based on a DN.
|
||||||
func readUsernameFromLdap(dn string, conn *ldap.Conn, attr string) (bool, string) {
|
func readUsernameFromLdap(dn string, conn ldapConn, attr string) (bool, string) {
|
||||||
ok, res := executeQuery(conn, dn, []string{attr})
|
log := conn.log.WithFields(logrus.Fields{
|
||||||
|
"dn": dn,
|
||||||
|
"attribute": attr,
|
||||||
|
})
|
||||||
|
log.Trace("Converting DN to username")
|
||||||
|
ok, res := conn.query(dn, []string{attr})
|
||||||
if !ok {
|
if !ok {
|
||||||
return false, ""
|
return false, ""
|
||||||
}
|
}
|
||||||
values := res.GetAttributeValues(attr)
|
values := res.GetAttributeValues(attr)
|
||||||
if len(values) != 1 {
|
if len(values) != 1 {
|
||||||
log.Printf("LDAP search for %s: attribute %s has %d values", dn, attr, len(values))
|
log.WithField("count", len(values)).
|
||||||
|
Warning("Attribute does not have 1 value exactly.")
|
||||||
return false, ""
|
return false, ""
|
||||||
}
|
}
|
||||||
|
log.WithField("username", values[0]).Trace("Mapped DN to username")
|
||||||
return true, values[0]
|
return true, values[0]
|
||||||
}
|
}
|
||||||
|
|
||||||
// Extract an username from something that may be an username or a DN.
|
// Extract an username from something that may be an username or a DN.
|
||||||
func usernameFromMember(member string, conn *ldap.Conn, config LdapConfig) (bool, string) {
|
func usernameFromMember(member string, conn ldapConn, config LdapConfig) (bool, string) {
|
||||||
eqPos := strings.Index(member, "=")
|
eqPos := strings.Index(member, "=")
|
||||||
if eqPos == -1 {
|
if eqPos == -1 {
|
||||||
return true, member
|
return true, member
|
||||||
|
@ -289,8 +363,10 @@ func usernameFromMember(member string, conn *ldap.Conn, config LdapConfig) (bool
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read the list of members from a LDAP group
|
// Read the list of members from a LDAP group
|
||||||
func getGroupMembers(group string, conn *ldap.Conn, config LdapConfig) (members []string) {
|
func getGroupMembers(group string, conn ldapConn, config LdapConfig) (members []string) {
|
||||||
ok, entry := executeQuery(conn, group, config.MemberFields)
|
log := conn.log.WithField("group", group)
|
||||||
|
log.Trace("Obtaining group members")
|
||||||
|
ok, entry := conn.query(group, config.MemberFields)
|
||||||
if !ok {
|
if !ok {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -307,20 +383,14 @@ func getGroupMembers(group string, conn *ldap.Conn, config LdapConfig) (members
|
||||||
}
|
}
|
||||||
break
|
break
|
||||||
}
|
}
|
||||||
|
log.WithField("members", members).Info("Obtained group members")
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read the list of group members from the LDAP server for all groups in the mapping section.
|
// Read the list of group members from the LDAP server for all groups in the mapping section.
|
||||||
func readLdapGroups(configuration Configuration) (groups GroupMembers) {
|
func readLdapGroups(configuration Configuration) (groups GroupMembers) {
|
||||||
conn := getLdapConnection(configuration.Ldap)
|
conn := getLdapConnection(configuration.Ldap)
|
||||||
defer conn.Close()
|
defer conn.close()
|
||||||
|
|
||||||
if configuration.Ldap.BindUser != "" {
|
|
||||||
err := conn.Bind(configuration.Ldap.BindUser, configuration.Ldap.BindPassword)
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("LDAP server %s, could not bind: %v", configuration.Ldap.Host, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
groups = make(GroupMembers)
|
groups = make(GroupMembers)
|
||||||
for group := range configuration.Mapping {
|
for group := range configuration.Mapping {
|
||||||
|
@ -398,10 +468,14 @@ func computePrivileges(mapping GroupMapping, membership []string) (privileges []
|
||||||
|
|
||||||
// Delete a Graylog user account
|
// Delete a Graylog user account
|
||||||
func deleteAccount(cfg GraylogConfig, user string) {
|
func deleteAccount(cfg GraylogConfig, user string) {
|
||||||
log.Printf("DELETING ACCOUNT %s", user)
|
log := log.WithField("user", user)
|
||||||
|
log.Warning("Deleting Graylog account")
|
||||||
code, body := executeApiCall(cfg, "DELETE", fmt.Sprintf("/users/%s", user), nil)
|
code, body := executeApiCall(cfg, "DELETE", fmt.Sprintf("/users/%s", user), nil)
|
||||||
if code != 204 {
|
if code != 204 {
|
||||||
log.Fatalf("could not delete user %s: code %d, body '%s'", user, code, string(body))
|
log.WithFields(logrus.Fields{
|
||||||
|
"status": code,
|
||||||
|
"body": string(body),
|
||||||
|
}).Fatal("Could not delete user")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -470,8 +544,57 @@ func applyMapping(cfg Configuration, users []GraylogUser, groups GroupMembers) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Parse command line options.
|
||||||
|
func parseCommandLine() cliFlags {
|
||||||
|
flags := cliFlags{}
|
||||||
|
flag.StringVar(&flags.cfgFile, "c", "graylog-groups.yml", "Configuration file.")
|
||||||
|
flag.StringVar(&flags.instance, "i", "", "Instance identifier.")
|
||||||
|
flag.StringVar(&flags.logLevel, "L", "", "Log level for the logrus library.")
|
||||||
|
flag.Parse()
|
||||||
|
return flags
|
||||||
|
}
|
||||||
|
|
||||||
|
// Initialize the logging context.
|
||||||
|
func getLoggingContext(instance string) *logrus.Entry {
|
||||||
|
logFields := logrus.Fields{
|
||||||
|
"application": "graylog",
|
||||||
|
"component": "graylog-groups",
|
||||||
|
}
|
||||||
|
if instance != "" {
|
||||||
|
logFields["instance"] = instance
|
||||||
|
}
|
||||||
|
return logrus.WithFields(logFields)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configure the log level
|
||||||
|
func configureLogLevel(cliLevel string) {
|
||||||
|
var lvl logrus.Level
|
||||||
|
if cliLevel == "" {
|
||||||
|
lvl = logrus.InfoLevel
|
||||||
|
} else {
|
||||||
|
var err error
|
||||||
|
lvl, err = logrus.ParseLevel(cliLevel)
|
||||||
|
if err != nil {
|
||||||
|
log.WithFields(logrus.Fields{
|
||||||
|
"level": cliLevel,
|
||||||
|
}).Warning("Invalid log level on command line")
|
||||||
|
lvl = logrus.InfoLevel
|
||||||
|
}
|
||||||
|
}
|
||||||
|
log.Logger.SetLevel(lvl)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Configure the logging library based on the various command line flags.
|
||||||
|
func configureLogging(flags cliFlags) {
|
||||||
|
log = getLoggingContext(flags.instance)
|
||||||
|
configureLogLevel(flags.logLevel)
|
||||||
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
configuration := loadConfiguration()
|
flags := parseCommandLine()
|
||||||
|
configureLogging(flags)
|
||||||
|
log.Debug("Starting synchronization")
|
||||||
|
configuration := loadConfiguration(flags)
|
||||||
glUsers := getGraylogUsers(configuration.Graylog)
|
glUsers := getGraylogUsers(configuration.Graylog)
|
||||||
groups := readLdapGroups(configuration)
|
groups := readLdapGroups(configuration)
|
||||||
applyMapping(configuration, glUsers, groups)
|
applyMapping(configuration, glUsers, groups)
|
||||||
|
|
Loading…
Reference in a new issue