state/print: collect and output state information of all users
The -state flag now outputs state of all users. The old behaviour can be accessed via the -state-current flag, user is selected via -u. Signed-off-by: Ophestra Umiker <cat@ophivana.moe>
This commit is contained in:
parent
60e4846542
commit
83af555c97
|
@ -0,0 +1,104 @@
|
||||||
|
package state
|
||||||
|
|
||||||
|
import (
|
||||||
|
"flag"
|
||||||
|
"fmt"
|
||||||
|
"os"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"text/tabwriter"
|
||||||
|
|
||||||
|
"git.ophivana.moe/cat/fortify/internal/system"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
stateActionEarly bool
|
||||||
|
stateActionEarlyC bool
|
||||||
|
)
|
||||||
|
|
||||||
|
func init() {
|
||||||
|
flag.BoolVar(&stateActionEarly, "state", false, "print state information of active launchers")
|
||||||
|
flag.BoolVar(&stateActionEarlyC, "state-current", false, "print state information of active launchers for the specified user")
|
||||||
|
}
|
||||||
|
|
||||||
|
func Early() {
|
||||||
|
var w *tabwriter.Writer
|
||||||
|
|
||||||
|
switch {
|
||||||
|
case stateActionEarly:
|
||||||
|
if runDir, err := os.ReadDir(system.V.RunDir); err != nil {
|
||||||
|
fmt.Println("Error reading runtime directory:", err)
|
||||||
|
} else {
|
||||||
|
for _, e := range runDir {
|
||||||
|
if !e.IsDir() {
|
||||||
|
if system.V.Verbose {
|
||||||
|
fmt.Println("Skipped non-directory entry", e.Name())
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err = strconv.Atoi(e.Name()); err != nil {
|
||||||
|
if system.V.Verbose {
|
||||||
|
fmt.Println("Skipped non-uid entry", e.Name())
|
||||||
|
}
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
printLauncherState(e.Name(), &w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
case stateActionEarlyC:
|
||||||
|
printLauncherState(u.Uid, &w)
|
||||||
|
default:
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if w != nil {
|
||||||
|
if err := w.Flush(); err != nil {
|
||||||
|
fmt.Println("warn: error formatting output:", err)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fmt.Println("No information available.")
|
||||||
|
}
|
||||||
|
|
||||||
|
os.Exit(0)
|
||||||
|
}
|
||||||
|
|
||||||
|
func printLauncherState(uid string, w **tabwriter.Writer) {
|
||||||
|
launchers, err := readLaunchers(uid)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("Error reading launchers:", err)
|
||||||
|
os.Exit(1)
|
||||||
|
}
|
||||||
|
|
||||||
|
if *w == nil {
|
||||||
|
*w = tabwriter.NewWriter(os.Stdout, 0, 1, 4, ' ', 0)
|
||||||
|
|
||||||
|
if !system.V.Verbose {
|
||||||
|
_, _ = fmt.Fprintln(*w, "\tUID\tPID\tEnablements\tLauncher\tCommand")
|
||||||
|
} else {
|
||||||
|
_, _ = fmt.Fprintln(*w, "\tUID\tPID\tArgv")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, state := range launchers {
|
||||||
|
enablementsDescription := strings.Builder{}
|
||||||
|
for i := Enablement(0); i < enableLength; i++ {
|
||||||
|
if state.Capability.Has(i) {
|
||||||
|
enablementsDescription.WriteString(", " + i.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if enablementsDescription.Len() == 0 {
|
||||||
|
enablementsDescription.WriteString("none")
|
||||||
|
}
|
||||||
|
|
||||||
|
if !system.V.Verbose {
|
||||||
|
_, _ = fmt.Fprintf(*w, "\t%s\t%d\t%s\t%s\t%s\n",
|
||||||
|
uid, state.PID, strings.TrimPrefix(enablementsDescription.String(), ", "), state.Launcher,
|
||||||
|
state.Command)
|
||||||
|
} else {
|
||||||
|
_, _ = fmt.Fprintf(*w, "\t%s\t%d\t%s\n",
|
||||||
|
uid, state.PID, state.Argv)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,11 @@
|
||||||
package state
|
package state
|
||||||
|
|
||||||
|
var (
|
||||||
|
cleanupCandidate []string
|
||||||
|
enablements *Enablements
|
||||||
|
xcbActionComplete bool
|
||||||
|
)
|
||||||
|
|
||||||
func RegisterRevertPath(p string) {
|
func RegisterRevertPath(p string) {
|
||||||
cleanupCandidate = append(cleanupCandidate, p)
|
cleanupCandidate = append(cleanupCandidate, p)
|
||||||
}
|
}
|
||||||
|
|
|
@ -3,15 +3,11 @@ package state
|
||||||
import (
|
import (
|
||||||
"encoding/gob"
|
"encoding/gob"
|
||||||
"errors"
|
"errors"
|
||||||
"flag"
|
|
||||||
"fmt"
|
|
||||||
"io/fs"
|
"io/fs"
|
||||||
"os"
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"path"
|
"path"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
|
||||||
"text/tabwriter"
|
|
||||||
|
|
||||||
"git.ophivana.moe/cat/fortify/internal/system"
|
"git.ophivana.moe/cat/fortify/internal/system"
|
||||||
)
|
)
|
||||||
|
@ -20,11 +16,7 @@ import (
|
||||||
// this and launcher should eventually be replaced by a server process
|
// this and launcher should eventually be replaced by a server process
|
||||||
|
|
||||||
var (
|
var (
|
||||||
stateActionEarly bool
|
|
||||||
statePath string
|
statePath string
|
||||||
cleanupCandidate []string
|
|
||||||
xcbActionComplete bool
|
|
||||||
enablements *Enablements
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type launcherState struct {
|
type launcherState struct {
|
||||||
|
@ -35,55 +27,6 @@ type launcherState struct {
|
||||||
Capability Enablements
|
Capability Enablements
|
||||||
}
|
}
|
||||||
|
|
||||||
func init() {
|
|
||||||
flag.BoolVar(&stateActionEarly, "state", false, "query state value of current active launchers")
|
|
||||||
}
|
|
||||||
|
|
||||||
func Early() {
|
|
||||||
if !stateActionEarly {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
launchers, err := readLaunchers(u.Uid)
|
|
||||||
if err != nil {
|
|
||||||
fmt.Println("Error reading launchers:", err)
|
|
||||||
os.Exit(1)
|
|
||||||
}
|
|
||||||
|
|
||||||
stdout := tabwriter.NewWriter(os.Stdout, 0, 1, 4, ' ', 0)
|
|
||||||
if !system.V.Verbose {
|
|
||||||
_, _ = fmt.Fprintln(stdout, "\tPID\tEnablements\tLauncher\tCommand")
|
|
||||||
} else {
|
|
||||||
_, _ = fmt.Fprintln(stdout, "\tPID\tArgv")
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, state := range launchers {
|
|
||||||
enablementsDescription := strings.Builder{}
|
|
||||||
for i := Enablement(0); i < enableLength; i++ {
|
|
||||||
if state.Capability.Has(i) {
|
|
||||||
enablementsDescription.WriteString(", " + i.String())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if enablementsDescription.Len() == 0 {
|
|
||||||
enablementsDescription.WriteString("none")
|
|
||||||
}
|
|
||||||
|
|
||||||
if !system.V.Verbose {
|
|
||||||
_, _ = fmt.Fprintf(stdout, "\t%d\t%s\t%s\t%s\n",
|
|
||||||
state.PID, strings.TrimPrefix(enablementsDescription.String(), ", "), state.Launcher,
|
|
||||||
state.Command)
|
|
||||||
} else {
|
|
||||||
_, _ = fmt.Fprintf(stdout, "\t%d\t%s\n",
|
|
||||||
state.PID, state.Argv)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if err = stdout.Flush(); err != nil {
|
|
||||||
fmt.Println("warn: error formatting output:", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
os.Exit(0)
|
|
||||||
}
|
|
||||||
|
|
||||||
// SaveProcess called after process start, before wait
|
// SaveProcess called after process start, before wait
|
||||||
func SaveProcess(uid string, cmd *exec.Cmd) error {
|
func SaveProcess(uid string, cmd *exec.Cmd) error {
|
||||||
statePath = path.Join(system.V.RunDir, uid, strconv.Itoa(cmd.Process.Pid))
|
statePath = path.Join(system.V.RunDir, uid, strconv.Itoa(cmd.Process.Pid))
|
||||||
|
|
Loading…
Reference in New Issue