diff --git a/finalize-plugins/shell-final.go b/finalize-plugins/shell-final.go new file mode 100644 index 0000000..cce250b --- /dev/null +++ b/finalize-plugins/shell-final.go @@ -0,0 +1,105 @@ +package main + +import ( + "C" + "encoding/json" + "fmt" + "github.com/vanilla-os/vib/api" + "os" + "os/exec" + "strings" +) + +type Shell struct { + Name string `json:"name"` + Type string `json:"type"` + Commands []string `json:"commands"` + Cwd string `json:"cwd"` +} + +//export PlugInfo +func PlugInfo() *C.char { + plugininfo := &api.PluginInfo{Name: "shell-final", Type: api.FinalizePlugin} + pluginjson, err := json.Marshal(plugininfo) + if err != nil { + return C.CString(fmt.Sprintf("ERROR: %s", err.Error())) + } + return C.CString(string(pluginjson)) +} + +//export PluginScope +func PluginScope() int32 { // int32 is defined as GoInt32 in cgo which is the same as a C int + return api.IMAGENAME | api.FS | api.RECIPE +} + +func parsePath(path string, data *api.ScopeData) string { + path = strings.ReplaceAll(path, "$PROJROOT", data.Recipe.ParentPath) + path = strings.ReplaceAll(path, "$FSROOT", data.FS) + return path +} + +func baseCommand(command string, data *api.ScopeData) string { + commandParts := strings.Split(command, " ") + if strings.Contains(commandParts[0], "/") { + return parsePath(commandParts[0], data) + } else { + command, err := exec.LookPath(commandParts[0]) + if err != nil { + return commandParts[0] + } + return command + } +} + +func getArgs(command string, data *api.ScopeData) []string { + commandParts := strings.Split(parsePath(command, data), " ") + return commandParts[1:] +} + +func genCommand(command string, data *api.ScopeData) []string { + baseCommand := baseCommand(command, data) + args := getArgs(command, data) + return append(append(append([]string{"-c", "'"}, strings.Join(args, " ")), baseCommand), "'") +} + +//export FinalizeBuild +func FinalizeBuild(moduleInterface *C.char, extraData *C.char) *C.char { + var module *Shell + var data *api.ScopeData + + err := json.Unmarshal([]byte(C.GoString(moduleInterface)), &module) + if err != nil { + return C.CString(fmt.Sprintf("ERROR: %s", err.Error())) + } + + err = json.Unmarshal([]byte(C.GoString(extraData)), &data) + if err != nil { + return C.CString(fmt.Sprintf("ERROR: %s", err.Error())) + } + + for _, command := range module.Commands { + fmt.Println("shell-final:: bash ", "-c ", command) + + cmd := exec.Command( + "bash", "-c", parsePath(command, data), + ) + cmd.Stdin = os.Stdin + cmd.Stdout = os.Stdout + cmd.Stderr = os.Stderr + cmd.Env = os.Environ() + if len(strings.TrimSpace(module.Cwd)) == 0 { + cmd.Dir = data.Recipe.ParentPath + } else { + cmd.Dir = parsePath(module.Cwd, data) + } + + err = cmd.Run() + if err != nil { + return C.CString(fmt.Sprintf("ERROR: %s", err.Error())) + } + } + + return C.CString("") +} + +func main() {} diff --git a/finalize-plugins/sysext.go b/finalize-plugins/sysext.go index e972ff5..5b4d0c2 100644 --- a/finalize-plugins/sysext.go +++ b/finalize-plugins/sysext.go @@ -51,7 +51,7 @@ func FinalizeBuild(moduleInterface *C.char, extraData *C.char) *C.char { var extensionRelease strings.Builder fmt.Fprintf(&extensionRelease, "ID=%s\n", module.OSReleaseID) - fmt.Fprintf(&extensionRelease, "VersionID=%s\n", module.OSReleaseVersionID) + fmt.Fprintf(&extensionRelease, "VERSION_ID=%s\n", module.OSReleaseVersionID) err = os.MkdirAll(filepath.Join(data.FS, "usr/lib/extension-release.d"), 0o777) if err != nil { diff --git a/finalize-plugins/systemd-repart.go b/finalize-plugins/systemd-repart.go index 170969f..8d6a331 100644 --- a/finalize-plugins/systemd-repart.go +++ b/finalize-plugins/systemd-repart.go @@ -7,14 +7,19 @@ import ( "github.com/vanilla-os/vib/api" "os" "os/exec" + "strings" ) type SystemdRepart struct { - Name string `json:"name"` - Type string `json:"type"` - Output string `json:"output"` - Size string `json:"size"` - Seed string `json:"seed"` + Name string `json:"name"` + Type string `json:"type"` + Output string `json:"output"` + Json string `json:"json"` + SpecOutput string `json:"spec_output"` + Size string `json:"size"` + Seed string `json:"seed"` + Split bool `json:"split"` + DeferPartitions []string `json:"defer_partitions"` } //export PlugInfo @@ -51,8 +56,12 @@ func FinalizeBuild(moduleInterface *C.char, extraData *C.char) *C.char { if err != nil { return C.CString(fmt.Sprintf("ERROR: %s", err.Error())) } - cmd := exec.Command( - repart, + + if len(strings.TrimSpace(module.Json)) == 0 { + module.Json = "off" + } + + args := []string{ "--definitions=definitions", "--empty=create", fmt.Sprintf("--size=%s", module.Size), @@ -60,11 +69,27 @@ func FinalizeBuild(moduleInterface *C.char, extraData *C.char) *C.char { "--discard=no", "--offline=true", "--no-pager", + fmt.Sprintf("--split=%t", module.Split), fmt.Sprintf("--seed=%s", module.Seed), fmt.Sprintf("--root=%s", data.FS), module.Output, + fmt.Sprintf("--json=%s", module.Json), + } + + if len(module.DeferPartitions) > 0 { + args = append(args, fmt.Sprintf("--defer-partitions=%s", strings.Join(module.DeferPartitions, ","))) + } + + cmd := exec.Command( + repart, + args..., ) - cmd.Stdout = os.Stdout + jsonFile, err := os.Create(module.SpecOutput) + if err != nil { + return C.CString(fmt.Sprintf("ERROR: %s", err.Error())) + } + defer jsonFile.Close() + cmd.Stdout = jsonFile cmd.Stderr = os.Stderr cmd.Dir = data.Recipe.ParentPath