refractor(build): move builder tasks to the builder sub package. Add tests.

This commit is contained in:
Alexandre Pujol 2024-03-25 23:16:00 +00:00
parent 08d4110c2a
commit 16f00ebfc7
Failed to generate hash of commit
7 changed files with 516 additions and 0 deletions

View file

@ -0,0 +1,35 @@
// apparmor.d - Full set of apparmor profiles
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
// SPDX-License-Identifier: GPL-2.0-only
package builder
import (
"github.com/roddhjav/apparmor.d/pkg/prebuild/cfg"
"github.com/roddhjav/apparmor.d/pkg/util"
)
var (
regAbi4To3 = util.ToRegexRepl([]string{ // Currently Abi3 -> Abi4
`abi/3.0`, `abi/4.0`,
`# userns,`, `userns,`,
`# mqueue`, `mqueue`,
})
)
type ABI3 struct {
cfg.Base
}
func init() {
RegisterBuilder(&ABI3{
Base: cfg.Base{
Keyword: "abi3",
Msg: "Convert all profiles from abi 4.0 to abi 3.0",
},
})
}
func (b ABI3) Apply(profile string) string {
return regAbi4To3.Replace(profile)
}

View file

@ -0,0 +1,48 @@
// apparmor.d - Full set of apparmor profiles
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
// SPDX-License-Identifier: GPL-2.0-only
package builder
import (
"regexp"
"slices"
"strings"
"github.com/roddhjav/apparmor.d/pkg/prebuild/cfg"
)
var (
regFlags = regexp.MustCompile(`flags=\(([^)]+)\)`)
regProfileHeader = regexp.MustCompile(` {`)
)
type Complain struct {
cfg.Base
}
func init() {
RegisterBuilder(&Complain{
Base: cfg.Base{
Keyword: "complain",
Msg: "Set complain flag on all profiles",
},
})
}
func (b Complain) Apply(profile string) string {
flags := []string{}
matches := regFlags.FindStringSubmatch(profile)
if len(matches) != 0 {
flags = strings.Split(matches[1], ",")
if slices.Contains(flags, "complain") {
return profile
}
}
flags = append(flags, "complain")
strFlags := " flags=(" + strings.Join(flags, ",") + ") {"
// Remove all flags definition, then set manifest' flags
profile = regFlags.ReplaceAllLiteralString(profile, "")
return regProfileHeader.ReplaceAllLiteralString(profile, strFlags)
}

View file

@ -0,0 +1,39 @@
// apparmor.d - Full set of apparmor profiles
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
// SPDX-License-Identifier: GPL-2.0-only
package builder
import (
"fmt"
"github.com/roddhjav/apparmor.d/pkg/prebuild/cfg"
)
var (
// Build the profiles with the following directive applied
Builds = []Builder{}
// Available builders
Builders = map[string]Builder{}
)
// Main directive interface
type Builder interface {
cfg.BaseInterface
Apply(profile string) string
}
func Register(names ...string) {
for _, name := range names {
if b, present := Builders[name]; present {
Builds = append(Builds, b)
} else {
panic(fmt.Sprintf("Unknown builder: %s", name))
}
}
}
func RegisterBuilder(d Builder) {
Builders[d.Name()] = d
}

View file

@ -0,0 +1,271 @@
// apparmor.d - Full set of apparmor profiles
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
// SPDX-License-Identifier: GPL-2.0-only
package builder
import (
"slices"
"testing"
)
func TestBuilder_Apply(t *testing.T) {
tests := []struct {
name string
b Builder
profile string
want string
}{
{
name: "abi3",
b: Builders["abi3"],
profile: `
abi <abi/3.0>,
profile test {
# userns,
# mqueue r type=posix /,
}`,
want: `
abi <abi/4.0>,
profile test {
userns,
mqueue r type=posix /,
}`,
},
{
name: "complain-1",
b: Builders["complain"],
profile: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
want: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(complain) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
},
{
name: "complain-2",
b: Builders["complain"],
profile: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(complain) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
want: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(complain) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
},
{
name: "complain-3",
b: Builders["complain"],
profile: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(attach_disconnected) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
want: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(attach_disconnected,complain) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
},
{
name: "enforce-1",
b: Builders["enforce"],
profile: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
want: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
},
{
name: "enforce-2",
b: Builders["enforce"],
profile: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(complain) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
want: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
},
{
name: "complain-3",
b: Builders["enforce"],
profile: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(attach_disconnected,complain) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
want: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} flags=(attach_disconnected) {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/foo>
}`,
},
{
name: "fsp",
b: Builders["fsp"],
profile: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} {
include <abstractions/base>
@{exec_path} mr,
@{bin}/* rPUx,
@{lib}/* rUx,
include if exists <local/foo>
}`,
want: `
@{exec_path} = @{bin}/foo
profile foo @{exec_path} {
include <abstractions/base>
@{exec_path} mr,
@{bin}/* rPx,
@{lib}/* rPx,
include if exists <local/foo>
}`,
},
{
name: "userspace-1",
b: Builders["userspace"],
profile: `
@{exec_path} = @{bin}/baloo_file @{lib}/{,kf6/}baloo_file
@{exec_path} += @{lib}/@{multiarch}/{,libexec/}baloo_file
profile baloo @{exec_path} {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/baloo>
}`,
want: `
@{exec_path} = @{bin}/baloo_file @{lib}/{,kf6/}baloo_file
@{exec_path} += @{lib}/@{multiarch}/{,libexec/}baloo_file
profile baloo /{{,usr/}{,s}bin/baloo_file,{,usr/}lib{,exec,32,64}/{,kf6/}baloo_file,{,usr/}lib{,exec,32,64}/*-linux-gnu*/{,libexec/}baloo_file} {
include <abstractions/base>
@{exec_path} mr,
include if exists <local/baloo>
}`,
},
{
name: "userspace-1",
b: Builders["userspace"],
profile: `
profile foo /usr/bin/foo {
include <abstractions/base>
/usr/bin/foo mr,
include if exists <local/foo>
}`,
want: `
profile foo /usr/bin/foo {
include <abstractions/base>
/usr/bin/foo mr,
include if exists <local/foo>
}`,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
if got := tt.b.Apply(tt.profile); got != tt.want {
t.Errorf("Builder.Apply() = %v, want %v", got, tt.want)
}
})
}
}
func TestRegister(t *testing.T) {
tests := []struct {
name string
names []string
wantSuccess bool
}{
{
name: "test",
names: []string{"complain", "enforce"},
wantSuccess: true,
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
Register(tt.names...)
for _, name := range tt.names {
if got := slices.Contains(Builds, Builders[name]); got != tt.wantSuccess {
t.Errorf("Register() = %v, want %v", got, tt.wantSuccess)
}
}
})
}
}

View file

@ -0,0 +1,47 @@
// apparmor.d - Full set of apparmor profiles
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
// SPDX-License-Identifier: GPL-2.0-only
package builder
import (
"slices"
"strings"
"github.com/roddhjav/apparmor.d/pkg/prebuild/cfg"
)
type Enforce struct {
cfg.Base
}
func init() {
RegisterBuilder(&Enforce{
Base: cfg.Base{
Keyword: "enforce",
Msg: "All profiles have been enforced",
},
})
}
func (b Enforce) Apply(profile string) string {
matches := regFlags.FindStringSubmatch(profile)
if len(matches) == 0 {
return profile
}
flags := strings.Split(matches[1], ",")
idx := slices.Index(flags, "complain")
if idx == -1 {
return profile
}
flags = slices.Delete(flags, idx, idx+1)
strFlags := "{"
if len(flags) >= 1 {
strFlags = " flags=(" + strings.Join(flags, ",") + ") {"
}
// Remove all flags definition, then set new flags
profile = regFlags.ReplaceAllLiteralString(profile, "")
return regProfileHeader.ReplaceAllLiteralString(profile, strFlags)
}

View file

@ -0,0 +1,33 @@
// apparmor.d - Full set of apparmor profiles
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
// SPDX-License-Identifier: GPL-2.0-only
package builder
import (
"github.com/roddhjav/apparmor.d/pkg/prebuild/cfg"
"github.com/roddhjav/apparmor.d/pkg/util"
)
var (
regFullSystemPolicy = util.ToRegexRepl([]string{
`r(PU|U)x,`, `rPx,`,
})
)
type FullSystemPolicy struct {
cfg.Base
}
func init() {
RegisterBuilder(&FullSystemPolicy{
Base: cfg.Base{
Keyword: "fsp",
Msg: "Prevent unconfined transitions in profile rules",
},
})
}
func (b FullSystemPolicy) Apply(profile string) string {
return regFullSystemPolicy.Replace(profile)
}

View file

@ -0,0 +1,43 @@
// apparmor.d - Full set of apparmor profiles
// Copyright (C) 2021-2024 Alexandre Pujol <alexandre@pujol.io>
// SPDX-License-Identifier: GPL-2.0-only
package builder
import (
"regexp"
"strings"
"github.com/roddhjav/apparmor.d/pkg/aa"
"github.com/roddhjav/apparmor.d/pkg/prebuild/cfg"
)
var (
regAttachments = regexp.MustCompile(`(profile .* @{exec_path})`)
)
type Userspace struct {
cfg.Base
}
func init() {
RegisterBuilder(&Userspace{
Base: cfg.Base{
Keyword: "userspace",
Msg: "Bypass userspace tools restriction",
},
})
}
func (b Userspace) Apply(profile string) string {
p := aa.DefaultTunables()
p.ParseVariables(profile)
p.ResolveAttachments()
att := p.NestAttachments()
matches := regAttachments.FindAllString(profile, -1)
if len(matches) > 0 {
strheader := strings.Replace(matches[0], "@{exec_path}", att, -1)
return regAttachments.ReplaceAllLiteralString(profile, strheader)
}
return profile
}