mirror of
https://github.com/roddhjav/apparmor.d.git
synced 2025-01-24 20:08:11 +01:00
123 lines
3.2 KiB
Go
123 lines
3.2 KiB
Go
// aa-log - Review AppArmor generated messages
|
|
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
|
|
// SPDX-License-Identifier: GPL-2.0-only
|
|
|
|
package main
|
|
|
|
import (
|
|
"flag"
|
|
"fmt"
|
|
"io"
|
|
"os"
|
|
"slices"
|
|
"strings"
|
|
|
|
"github.com/roddhjav/apparmor.d/pkg/logs"
|
|
)
|
|
|
|
const usage = `aa-log [-h] [--systemd] [--file file] [--rules | --raw] [profile]
|
|
|
|
Review AppArmor generated messages in a colorful way. It supports logs from
|
|
auditd, systemd, syslog as well as dbus session events.
|
|
|
|
It can be given an optional profile name to filter the output with.
|
|
|
|
Default logs are read from '/var/log/audit/audit.log'. Other files in
|
|
'/var/log/audit/' can easily be checked: 'aa-log -f 1' parses 'audit.log.1'
|
|
|
|
Options:
|
|
-h, --help Show this help message and exit.
|
|
-f, --file FILE Set a logfile or a suffix to the default log file.
|
|
-s, --systemd Parse systemd logs from journalctl.
|
|
-r, --rules Convert the log into AppArmor rules.
|
|
-R, --raw Print the raw log without any formatting.
|
|
-S, --since DATE Show entries not older than the specified date.
|
|
|
|
`
|
|
|
|
// Command line options
|
|
var (
|
|
help bool
|
|
rules bool
|
|
path string
|
|
systemd bool
|
|
raw bool
|
|
since string
|
|
)
|
|
|
|
func aaLog(logger string, path string, profile string) error {
|
|
var err error
|
|
var file io.Reader
|
|
|
|
switch logger {
|
|
case "auditd":
|
|
file, err = logs.GetAuditLogs(path)
|
|
case "systemd":
|
|
file, err = logs.GetJournalctlLogs(path, since, !slices.Contains(logs.LogFiles, path))
|
|
default:
|
|
err = fmt.Errorf("Logger %s not supported.", logger)
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
if raw {
|
|
fmt.Print(strings.Join(logs.GetApparmorLogs(file, profile), "\n") + "\n")
|
|
return nil
|
|
}
|
|
|
|
aaLogs := logs.New(file, profile)
|
|
if rules {
|
|
profiles := aaLogs.ParseToProfiles()
|
|
for _, p := range profiles {
|
|
p.Merge(nil)
|
|
p.Sort()
|
|
p.Format()
|
|
fmt.Print(p.String() + "\n\n")
|
|
}
|
|
} else {
|
|
fmt.Print(aaLogs.String())
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func init() {
|
|
flag.BoolVar(&help, "h", false, "Show this help message and exit.")
|
|
flag.BoolVar(&help, "help", false, "Show this help message and exit.")
|
|
flag.StringVar(&path, "f", "", "Set a logfile or a suffix to the default log file.")
|
|
flag.StringVar(&path, "file", "", "Set a logfile or a suffix to the default log file.")
|
|
flag.BoolVar(&systemd, "s", false, "Parse systemd logs from journalctl.")
|
|
flag.BoolVar(&systemd, "systemd", false, "Parse systemd logs from journalctl.")
|
|
flag.BoolVar(&rules, "r", false, "Convert the log into AppArmor rules.")
|
|
flag.BoolVar(&rules, "rules", false, "Convert the log into AppArmor rules.")
|
|
flag.BoolVar(&raw, "R", false, "Print the raw log without any formatting.")
|
|
flag.BoolVar(&raw, "raw", false, "Print the raw log without any formatting.")
|
|
flag.StringVar(&since, "S", "", "Display logs since the START time.")
|
|
flag.StringVar(&since, "since", "", "Display logs since the START time.")
|
|
}
|
|
|
|
func main() {
|
|
flag.Usage = func() { fmt.Print(usage) }
|
|
flag.Parse()
|
|
if help {
|
|
flag.Usage()
|
|
os.Exit(0)
|
|
}
|
|
|
|
profile := ""
|
|
if len(flag.Args()) >= 1 {
|
|
profile = flag.Args()[0]
|
|
}
|
|
|
|
logger := "auditd"
|
|
if systemd {
|
|
logger = "systemd"
|
|
}
|
|
|
|
path = logs.SelectLogFile(path)
|
|
err := aaLog(logger, path, profile)
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
os.Exit(1)
|
|
}
|
|
}
|