Compare commits

...

8 Commits

11 changed files with 225 additions and 53 deletions

View File

@ -2,4 +2,7 @@ module ectl
go 1.23.4 go 1.23.4
require rsc.io/getopt v0.0.0-20170811000552-20be20937449 // indirect require (
gopkg.in/yaml.v3 v3.0.1 // indirect
rsc.io/getopt v0.0.0-20170811000552-20be20937449 // indirect
)

View File

@ -1,2 +1,5 @@
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA=
gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM=
rsc.io/getopt v0.0.0-20170811000552-20be20937449 h1:UukjJOsjQH0DIuyyrcod6CXHS6cdaMMuJmrt+SN1j4A= rsc.io/getopt v0.0.0-20170811000552-20be20937449 h1:UukjJOsjQH0DIuyyrcod6CXHS6cdaMMuJmrt+SN1j4A=
rsc.io/getopt v0.0.0-20170811000552-20be20937449/go.mod h1:dhCdeqAxkyt5u3/sKRkUXuHaMXUu1Pt13GTQAM2xnig= rsc.io/getopt v0.0.0-20170811000552-20be20937449/go.mod h1:dhCdeqAxkyt5u3/sKRkUXuHaMXUu1Pt13GTQAM2xnig=

View File

@ -3,6 +3,7 @@ package main
import ( import (
"flag" "flag"
"fmt" "fmt"
"gopkg.in/yaml.v3"
"log" "log"
"net" "net"
"os" "os"
@ -64,8 +65,7 @@ func main() {
fmt.Printf("Usage: ectl service %s <service>\n", flag.Args()[1]) fmt.Printf("Usage: ectl service %s <service>\n", flag.Args()[1])
return return
} else if flag.Args()[1] == "start" { } else if flag.Args()[1] == "start" {
flag.Args()[2] = strings.TrimSuffix(flag.Args()[2], ".esv") if _, err := os.Stat(path.Join(runstatedir, "esvm", flag.Args()[2])); err != nil {
if _, err := os.Stat(path.Join(sysconfdir, "esvm/services/", flag.Args()[2]+".esv")); err != nil {
log.Fatalf("Could not start service! Error: %s\n", err) log.Fatalf("Could not start service! Error: %s\n", err)
} }
@ -86,8 +86,7 @@ func main() {
fmt.Println("Service started successfully!") fmt.Println("Service started successfully!")
return return
} else if flag.Args()[1] == "stop" { } else if flag.Args()[1] == "stop" {
flag.Args()[2] = strings.TrimSuffix(flag.Args()[2], ".esv") if _, err := os.Stat(path.Join(runstatedir, "esvm", flag.Args()[2])); err != nil {
if _, err := os.Stat(path.Join(sysconfdir, "esvm/services/", flag.Args()[2]+".esv")); err != nil {
log.Fatalf("Could not stop service! Error: %s\n", err) log.Fatalf("Could not stop service! Error: %s\n", err)
} }
@ -107,9 +106,8 @@ func main() {
fmt.Println("Service stopped successfully!") fmt.Println("Service stopped successfully!")
return return
} else if flag.Args()[1] == "restart" || flag.Args()[1] == "reload" { } else if flag.Args()[1] == "restart" || flag.Args()[1] == "reload" {
flag.Args()[2] = strings.TrimSuffix(flag.Args()[2], ".esv") if _, err := os.Stat(path.Join(runstatedir, "esvm", flag.Args()[2])); err != nil {
if _, err := os.Stat(path.Join(sysconfdir, "esvm/services/", flag.Args()[2]+".esv")); err != nil { log.Fatalf("Could not restart service! Error: %s\n", err)
log.Fatalf("Could not stop service! Error: %s\n", err)
} }
_, err := socket.Write([]byte("restart " + flag.Args()[2])) _, err := socket.Write([]byte("restart " + flag.Args()[2]))
@ -127,10 +125,105 @@ func main() {
} }
fmt.Println("Service restarted successfully!") fmt.Println("Service restarted successfully!")
return return
} else if flag.Args()[1] == "enable" {
// Check if service exists
found := false
entries, err := os.ReadDir(path.Join(sysconfdir, "esvm/services/"))
if err != nil {
log.Fatalf("Could not enable service! Error: %s\n", err)
}
type minimalServiceStruct struct {
Name string `yaml:"name"`
}
for _, entry := range entries {
if entry.IsDir() || !strings.HasSuffix(entry.Name(), ".esv") {
continue
}
bytes, err := os.ReadFile(path.Join(sysconfdir, "esvm/services", entry.Name()))
if err != nil {
log.Fatalf("Could not enable service! Error: %s\n", err)
}
sv := minimalServiceStruct{Name: ""}
err = yaml.Unmarshal(bytes, &sv)
if err != nil {
log.Fatalf("Could not enable service! Error: %s\n", err)
}
if sv.Name == flag.Args()[2] {
found = true
break
}
}
if !found {
log.Fatalf("Service does not exist!")
}
if _, err := os.Stat(path.Join(sysconfdir, "esvm/enabled_services")); err != nil {
err := os.WriteFile(path.Join(sysconfdir, "esvm/enabled_services"), []byte(flag.Args()[2]+"\n"), 0644)
if err != nil {
log.Fatalf("Could not enable service! Error: %s\n", err)
}
return
}
file, err := os.ReadFile(path.Join(sysconfdir, "esvm/enabled_services"))
if err != nil {
log.Fatalf("Could not enable service! Error: %s\n", err)
}
for _, line := range strings.Split(string(file), "\n") {
if strings.TrimSpace(line) == flag.Args()[2] {
fmt.Println("Service is already enabled!")
return
}
}
err = os.WriteFile(path.Join(sysconfdir, "esvm/enabled_services"), []byte(string(file)+flag.Args()[2]+"\n"), 0644)
if err != nil {
log.Fatalf("Could not enable service! Error: %s\n", err)
}
fmt.Printf("Service (%s) has been enabled!\n", flag.Args()[2])
return
} else if flag.Args()[1] == "disable" {
if _, err := os.Stat(path.Join(sysconfdir, "esvm/enabled_services")); err != nil {
fmt.Println("Service is already disabled!")
return
}
file, err := os.ReadFile(path.Join(sysconfdir, "esvm/enabled_services"))
if err != nil {
log.Fatalf("Could not disable service! Error: %s\n", err)
}
lines := strings.Split(string(file), "\n")
found := false
for i := len(lines) - 1; i >= 0; i-- {
line := strings.TrimSpace(lines[i])
if strings.TrimSpace(line) == flag.Args()[2] {
lines = append(lines[:i], lines[i+1:]...)
found = true
} else if strings.TrimSpace(line) == "" {
lines = append(lines[:i], lines[i+1:]...)
}
}
if !found {
fmt.Println("Service is already disabled!")
return
}
err = os.WriteFile(path.Join(sysconfdir, "esvm/enabled_services"), []byte(strings.Join(lines, "\n")+"\n"), 0644)
if err != nil {
log.Fatalf("Could not disable service! Error: %s\n", err)
}
fmt.Printf("Service (%s) has been disabled!\n", flag.Args()[2])
return
} else if flag.Args()[1] == "status" { } else if flag.Args()[1] == "status" {
flag.Args()[2] = strings.TrimSuffix(flag.Args()[2], ".esv") if _, err := os.Stat(path.Join(runstatedir, "esvm", flag.Args()[2])); err != nil {
if _, err := os.Stat(path.Join(sysconfdir, "esvm/services/", flag.Args()[2]+".esv")); err != nil { log.Fatalf("Could not get service status! Error: %s\n", err)
log.Fatalf("Could not stop service! Error: %s\n", err)
} }
var state uint64 var state uint64

View File

@ -1,6 +1,7 @@
package main package main
import ( import (
"errors"
"flag" "flag"
"fmt" "fmt"
"log" "log"
@ -8,6 +9,7 @@ import (
"os/exec" "os/exec"
"os/signal" "os/signal"
"path" "path"
"strings"
"syscall" "syscall"
"time" "time"
) )
@ -40,6 +42,8 @@ func main() {
mountVirtualFilesystems() mountVirtualFilesystems()
// Mount filesystems in fstab // Mount filesystems in fstab
mountFilesystems() mountFilesystems()
// Set hostname
setHostname()
// Start service manager // Start service manager
startServiceManager() startServiceManager()
@ -73,14 +77,14 @@ func mountVirtualFilesystems() {
panic(err) panic(err)
} }
// Mount /dev/pts // Mount /dev/pts
if err := os.Mkdir("/dev/pts", 0755); err != nil { if err := os.Mkdir("/dev/pts", 0755); err != nil && !errors.Is(err, os.ErrExist) {
panic(err) panic(err)
} }
if err := syscall.Mount("devpts", "/dev/pts", "devpts", commonFlags, "gid=5,mode=620,ptmxmode=000"); err != nil { if err := syscall.Mount("devpts", "/dev/pts", "devpts", commonFlags, "gid=5,mode=620,ptmxmode=000"); err != nil {
panic(err) panic(err)
} }
// Mount /dev/shm // Mount /dev/shm
if err := os.Mkdir("/dev/shm", 0755); err != nil { if err := os.Mkdir("/dev/shm", 0755); err != nil && !errors.Is(err, os.ErrExist) {
panic(err) panic(err)
} }
if err := syscall.Mount("shm", "/dev/shm", "tmpfs", commonFlags|syscall.MS_NODEV, "inode64"); err != nil { if err := syscall.Mount("shm", "/dev/shm", "tmpfs", commonFlags|syscall.MS_NODEV, "inode64"); err != nil {
@ -116,6 +120,8 @@ func startServiceManager() {
fmt.Print("Initializing service manager... ") fmt.Print("Initializing service manager... ")
cmd := exec.Command("/sbin/esvm", path.Join(runstatedir, "esvm"), path.Join(sysconfdir, "esvm")) cmd := exec.Command("/sbin/esvm", path.Join(runstatedir, "esvm"), path.Join(sysconfdir, "esvm"))
cmd.Stdout = os.Stdout
cmd.Stderr = os.Stderr
err := cmd.Start() err := cmd.Start()
if err != nil { if err != nil {
log.Println("Could not initialize service manager!") log.Println("Could not initialize service manager!")
@ -165,6 +171,25 @@ func stopServiceManager() {
fmt.Println("Done.") fmt.Println("Done.")
} }
func setHostname() {
fmt.Print("Setting hostname... ")
bytes, err := os.ReadFile("/etc/hostname")
if err != nil {
log.Println("Could not set hostname!")
return
}
hostname := strings.TrimSpace(string(bytes))
if err := syscall.Sethostname([]byte(hostname)); err != nil {
log.Println("Could not set hostname!")
return
}
fmt.Println("Done.")
}
func waitZombieProcesses() { func waitZombieProcesses() {
for { for {
if wpid, _ := syscall.Wait4(-1, nil, syscall.WNOHANG, nil); wpid <= 0 { if wpid, _ := syscall.Wait4(-1, nil, syscall.WNOHANG, nil); wpid <= 0 {

View File

@ -11,6 +11,7 @@ import (
"os/exec" "os/exec"
"os/signal" "os/signal"
"path" "path"
"slices"
"strconv" "strconv"
"strings" "strings"
"syscall" "syscall"
@ -29,17 +30,18 @@ const (
) )
type EnitService struct { type EnitService struct {
Name string `yaml:"name"` Name string `yaml:"name"`
Description string `yaml:"description,omitempty"` Description string `yaml:"description,omitempty"`
Dependencies []string `yaml:"dependencies,omitempty"` Dependencies []string `yaml:"dependencies,omitempty"`
Type string `yaml:"type"` Type string `yaml:"type"`
StartCmd string `yaml:"start_cmd"` StartCmd string `yaml:"start_cmd"`
ExitMethod string `yaml:"exit_method"` ExitMethod string `yaml:"exit_method"`
StopCmd string `yaml:"stop_cmd,omitempty"` CrashOnSafeExit bool `yaml:"crash_on_safe_exit"`
Restart bool `yaml:"restart,omitempty"` StopCmd string `yaml:"stop_cmd,omitempty"`
ServiceRunPath string Restart string `yaml:"restart,omitempty"`
restartCount int ServiceRunPath string
stopChannel chan bool restartCount int
stopChannel chan bool
} }
// Build-time variables // Build-time variables
@ -49,6 +51,7 @@ var runtimeServiceDir string
var serviceConfigDir string var serviceConfigDir string
var Services = make([]EnitService, 0) var Services = make([]EnitService, 0)
var EnabledServices = make([]string, 0)
var logger *log.Logger var logger *log.Logger
var socket net.Listener var socket net.Listener
@ -137,23 +140,30 @@ func Init() {
} }
service := EnitService{ service := EnitService{
Name: "", Name: "",
Description: "", Description: "",
Dependencies: make([]string, 0), Dependencies: make([]string, 0),
Type: "", Type: "",
StartCmd: "", StartCmd: "",
ExitMethod: "", ExitMethod: "",
StopCmd: "", StopCmd: "",
Restart: false, Restart: "",
ServiceRunPath: "", CrashOnSafeExit: true,
restartCount: 0, ServiceRunPath: "",
stopChannel: make(chan bool), restartCount: 0,
stopChannel: make(chan bool),
} }
if err := yaml.Unmarshal(bytes, &service); err != nil { if err := yaml.Unmarshal(bytes, &service); err != nil {
logger.Printf("Could not read service file at %s!\n", path.Join(serviceConfigDir, "services", entry.Name())) logger.Printf("Could not read service file at %s!\n", path.Join(serviceConfigDir, "services", entry.Name()))
continue continue
} }
for _, sv := range Services {
if sv.Name == service.Name {
logger.Printf("Service with name (%s) has already been initialized!", service.Name)
}
}
switch service.Type { switch service.Type {
case "simple", "background": case "simple", "background":
default: default:
@ -168,6 +178,12 @@ func Init() {
continue continue
} }
switch service.Restart {
case "true", "always":
default:
service.Restart = "false"
}
service.ServiceRunPath = path.Join(runtimeServiceDir, service.Name) service.ServiceRunPath = path.Join(runtimeServiceDir, service.Name)
err = os.MkdirAll(path.Join(service.ServiceRunPath), 0755) err = os.MkdirAll(path.Join(service.ServiceRunPath), 0755)
if err != nil { if err != nil {
@ -185,15 +201,28 @@ func Init() {
} }
} }
// Get services that meet their dependencies // Get enabled services
if _, err := os.Stat(path.Join(serviceConfigDir, "enabled_services")); err == nil {
file, err := os.ReadFile(path.Join(serviceConfigDir, "enabled_services"))
if err != nil {
return
}
for _, line := range strings.Split(string(file), "\n") {
if line != "" {
EnabledServices = append(EnabledServices, line)
}
}
}
// Get enabled services that meet their dependencies
servicesWithMetDepends := make([]EnitService, 0) servicesWithMetDepends := make([]EnitService, 0)
for _, service := range Services { for _, service := range Services {
if len(service.GetUnmetDependencies()) == 0 { if slices.Contains(EnabledServices, service.Name) && len(service.GetUnmetDependencies()) == 0 {
servicesWithMetDepends = append(servicesWithMetDepends, service) servicesWithMetDepends = append(servicesWithMetDepends, service)
} }
} }
// Loop until all services have started or timed out // Loop until all enabled services have started or timed out
for start := time.Now(); time.Since(start) < 60*time.Second; { for start := time.Now(); time.Since(start) < 60*time.Second; {
if len(servicesWithMetDepends) == 0 { if len(servicesWithMetDepends) == 0 {
break break
@ -340,10 +369,17 @@ func (service *EnitService) StartService() error {
_ = service.setCurrentState(EnitServiceCompleted) _ = service.setCurrentState(EnitServiceCompleted)
return return
} }
logger.Printf("Service (%s) has crashed!\n", service.Name) if !service.CrashOnSafeExit {
_ = service.setCurrentState(EnitServiceCrashed) logger.Printf("Service (%s) has exited\n", service.Name)
_ = service.setCurrentState(EnitServiceStopped)
} else {
logger.Printf("Service (%s) has crashed!\n", service.Name)
_ = service.setCurrentState(EnitServiceCrashed)
}
if service.Restart && service.restartCount < 5 { if service.Restart == "always" {
_ = service.StartService()
} else if service.Restart == "true" && service.restartCount < 5 {
service.restartCount++ service.restartCount++
_ = service.StartService() _ = service.StartService()
} }

View File

@ -1,5 +1,7 @@
name: agetty-tty1 name: agetty-tty1
description: Start virtual terminal on tty1 description: Start virtual terminal on tty1
type: background type: background
start_cmd: /sbin/agetty --noclear tty1 start_cmd: /usr/bin/setsid /sbin/agetty --noclear tty1
exit_method: kill exit_method: kill
crash_on_safe_exit: false
restart: always

View File

@ -1,5 +1,7 @@
name: agetty-tty2 name: agetty-tty2
description: Start virtual terminal on tty2 description: Start virtual terminal on tty2
type: background type: background
start_cmd: /sbin/agetty --noclear tty2 start_cmd: /usr/bin/setsid /sbin/agetty tty2
exit_method: kill exit_method: kill
crash_on_safe_exit: false
restart: always

View File

@ -1,5 +1,7 @@
name: agetty-tty3 name: agetty-tty3
description: Start virtual terminal on tty3 description: Start virtual terminal on tty3
type: background type: background
start_cmd: /sbin/agetty --noclear tty3 start_cmd: /usr/bin/setsid /sbin/agetty tty3
exit_method: kill exit_method: kill
crash_on_safe_exit: false
restart: always

View File

@ -1,5 +1,7 @@
name: agetty-tty4 name: agetty-tty4
description: Start virtual terminal on tty4 description: Start virtual terminal on tty4
type: background type: background
start_cmd: /sbin/agetty --noclear tty4 start_cmd: /usr/bin/setsid /sbin/agetty tty4
exit_method: kill exit_method: kill
crash_on_safe_exit: false
restart: always

View File

@ -1,5 +1,7 @@
name: agetty-tty5 name: agetty-tty5
description: Start virtual terminal on tty5 description: Start virtual terminal on tty5
type: background type: background
start_cmd: /sbin/agetty --noclear tty5 start_cmd: /usr/bin/setsid /sbin/agetty tty5
exit_method: kill exit_method: kill
crash_on_safe_exit: false
restart: always

View File

@ -1,5 +1,7 @@
name: agetty-tty6 name: agetty-tty6
description: Start virtual terminal on tty6 description: Start virtual terminal on tty6
type: background type: background
start_cmd: /sbin/agetty --noclear tty6 start_cmd: /usr/bin/setsid /sbin/agetty tty6
exit_method: kill exit_method: kill
crash_on_safe_exit: false
restart: always