// apparmor.d - Full set of apparmor profiles // Copyright (C) 2021-2024 Alexandre Pujol // SPDX-License-Identifier: GPL-2.0-only package builder import ( "fmt" "slices" "strings" "github.com/roddhjav/apparmor.d/pkg/paths" "github.com/roddhjav/apparmor.d/pkg/prebuild" ) var ( // Build the profiles with the following directive applied Builds = []Builder{} // Available builders Builders = map[string]Builder{} ) // Main directive interface type Builder interface { prebuild.BaseInterface Apply(opt *Option, profile string) (string, error) } // Builder options type Option struct { Name string File *paths.Path } func NewOption(file *paths.Path) *Option { return &Option{ Name: strings.TrimSuffix(file.Base(), ".apparmor.d"), File: file, } } func Register(names ...string) { for _, name := range names { if b, present := Builders[name]; present { if !slices.Contains(Builds, b) { Builds = append(Builds, b) } } else { panic(fmt.Sprintf("Unknown builder: %s", name)) } } } func Unregister(names ...string) { for _, name := range names { for i, b := range Builds { if b.Name() == name { Builds = slices.Delete(Builds, i, i+1) break } } } } func RegisterBuilder(d Builder) { Builders[d.Name()] = d } func Run(file *paths.Path, profile string) (string, error) { var err error opt := NewOption(file) for _, b := range Builds { profile, err = b.Apply(opt, profile) if err != nil { return "", fmt.Errorf("%s %s: %w", b.Name(), opt.File, err) } } return profile, nil }