improve console output

This commit is contained in:
Dominic Breuker
2018-02-12 23:39:04 +01:00
parent 22df28ae1d
commit 949e206ac1
6 changed files with 200 additions and 156 deletions

View File

@@ -1,7 +1,7 @@
package main package main
import "github.com/dominicbreuker/pspy/internal/dev" import "github.com/dominicbreuker/pspy/internal/cmd"
func main() { func main() {
dev.Monitor() cmd.Monitor()
} }

View File

@@ -1,4 +1,4 @@
package dev package cmd
import ( import (
"log" "log"
@@ -9,26 +9,18 @@ import (
"github.com/dominicbreuker/pspy/internal/walker" "github.com/dominicbreuker/pspy/internal/walker"
) )
type Process struct { const MaxInt = int(^uint(0) >> 1)
pid int
ppid int
state rune
pgrp int
sid int
binary string
}
func Monitor() { func Monitor() {
watch() watch([]string{"/tmp"}, nil)
} }
func watch() { func watch(rdirs, dirs []string) {
maxWatchers, err := inotify.WatcherLimit() maxWatchers, err := inotify.WatcherLimit()
if err != nil { if err != nil {
log.Printf("Can't get inotify watcher limit...: %v\n", err) log.Printf("Can't get inotify watcher limit...: %v\n", err)
} }
log.Printf("Watcher limit: %d\n", maxWatchers) log.Printf("Inotify watcher limit: %d (/proc/sys/fs/inotify/max_user_watches)\n", maxWatchers)
ping := make(chan struct{}) ping := make(chan struct{})
in, err := inotify.NewInotify(ping) in, err := inotify.NewInotify(ping)
@@ -37,9 +29,37 @@ func watch() {
} }
defer in.Close() defer in.Close()
dirCh, errCh := walker.Walk("/tmp") for _, dir := range rdirs {
addWatchers(dir, MaxInt, in, maxWatchers)
}
for _, dir := range dirs {
addWatchers(dir, 0, in, maxWatchers)
}
log.Printf("Inotify watchers set up: %s\n", in)
procList := process.NewProcList()
ticker := time.NewTicker(100 * time.Millisecond)
for {
select {
case <-ticker.C:
refresh(in, procList)
case <-ping:
refresh(in, procList)
}
}
}
func addWatchers(dir string, depth int, in *inotify.Inotify, maxWatchers int) {
dirCh, errCh, doneCh := walker.Walk(dir, depth)
loop: loop:
for { for {
if in.NumWatchers() >= maxWatchers {
close(doneCh)
break loop
}
select { select {
case dir, ok := <-dirCh: case dir, ok := <-dirCh:
if !ok { if !ok {
@@ -52,21 +72,6 @@ loop:
log.Printf("Error walking filesystem: %v", err) log.Printf("Error walking filesystem: %v", err)
} }
} }
log.Printf("Inotify set up: %s\n", in)
procList := process.NewProcList()
ticker := time.NewTicker(100 * time.Millisecond).C
for {
select {
case <-ticker:
refresh(in, procList)
case <-ping:
refresh(in, procList)
}
}
} }
func refresh(in *inotify.Inotify, pl *process.ProcList) { func refresh(in *inotify.Inotify, pl *process.ProcList) {
@@ -74,6 +79,6 @@ func refresh(in *inotify.Inotify, pl *process.ProcList) {
if err := pl.Refresh(); err != nil { if err := pl.Refresh(); err != nil {
log.Printf("ERROR refreshing process list: %v", err) log.Printf("ERROR refreshing process list: %v", err)
} }
time.Sleep(10 * time.Millisecond) time.Sleep(5 * time.Millisecond)
in.UnPause() in.UnPause()
} }

88
internal/inotify/event.go Normal file
View File

@@ -0,0 +1,88 @@
package inotify
import (
"bytes"
"fmt"
"log"
"strconv"
"unsafe"
"golang.org/x/sys/unix"
)
var InotifyEvents = map[uint32]string{
unix.IN_ACCESS: "ACCESS",
unix.IN_ATTRIB: "ATTRIB",
unix.IN_CLOSE_NOWRITE: "CLOSE_NOWRITE",
unix.IN_CLOSE_WRITE: "CLOSE_WRITE",
unix.IN_CREATE: "CREATE",
unix.IN_DELETE: "DELETE",
unix.IN_DELETE_SELF: "DELETE_SELF",
unix.IN_MODIFY: "MODIFY",
unix.IN_MOVED_FROM: "MOVED_FROM",
unix.IN_MOVED_TO: "MOVED_TO",
unix.IN_MOVE_SELF: "MOVE_SELF",
unix.IN_OPEN: "OPEN",
(unix.IN_ACCESS | unix.IN_ISDIR): "ACCESS DIR",
(unix.IN_ATTRIB | unix.IN_ISDIR): "ATTRIB DIR",
(unix.IN_CLOSE_NOWRITE | unix.IN_ISDIR): "CLOSE_NOWRITE DIR",
(unix.IN_CLOSE_WRITE | unix.IN_ISDIR): "CLOSE_WRITE DIR",
(unix.IN_CREATE | unix.IN_ISDIR): "CREATE DIR",
(unix.IN_DELETE | unix.IN_ISDIR): "DELETE DIR",
(unix.IN_DELETE_SELF | unix.IN_ISDIR): "DELETE_SELF DIR",
(unix.IN_MODIFY | unix.IN_ISDIR): "MODIFY DIR",
(unix.IN_MOVED_FROM | unix.IN_ISDIR): "MOVED_FROM DIR",
(unix.IN_MOVE_SELF | unix.IN_ISDIR): "MODE_SELF DIR",
(unix.IN_OPEN | unix.IN_ISDIR): "OPEN DIR",
}
type Event struct {
name string
op string
}
func (e Event) String() string {
return fmt.Sprintf("%20s | %s", e.op, e.name)
}
func newEvent(name string, mask uint32) Event {
e := Event{name: name}
op, ok := InotifyEvents[mask]
if !ok {
op = strconv.FormatInt(int64(mask), 2)
}
e.op = op
return e
}
func eventLogger(i *Inotify, buffers chan bufRead) {
for bf := range buffers {
n := bf.n
buf := bf.buf
if n < unix.SizeofInotifyEvent {
// incomplete or erroneous read
continue
}
var ptr uint32
var name string
for ptr <= uint32(n-unix.SizeofInotifyEvent) {
sys := (*unix.InotifyEvent)(unsafe.Pointer(&buf[ptr]))
ptr += unix.SizeofInotifyEvent
watcher, ok := i.watchers[int(sys.Wd)]
if !ok {
continue
}
name = watcher.dir + "/"
if sys.Len > 0 && len(buf) >= int(ptr+sys.Len) {
name += string(bytes.TrimRight(buf[ptr:ptr+sys.Len], "\x00"))
ptr += sys.Len
}
ev := newEvent(name, sys.Mask)
log.Printf("\x1b[32;1mFS: %+v\x1b[0m", ev)
}
}
}

View File

@@ -2,9 +2,6 @@ package inotify
import ( import (
"fmt" "fmt"
"log"
"strings"
"unsafe"
"golang.org/x/sys/unix" "golang.org/x/sys/unix"
) )
@@ -57,6 +54,10 @@ func (i *Inotify) UnPause() {
i.paused = false i.paused = false
} }
func (i *Inotify) NumWatchers() int {
return len(i.watchers)
}
func (i *Inotify) String() string { func (i *Inotify) String() string {
if len(i.watchers) < 20 { if len(i.watchers) < 20 {
dirs := make([]string, 0) dirs := make([]string, 0)
@@ -75,9 +76,9 @@ type bufRead struct {
} }
func watch(i *Inotify) { func watch(i *Inotify) {
buf := make([]byte, 20*unix.SizeofInotifyEvent) buf := make([]byte, 5*unix.SizeofInotifyEvent)
buffers := make(chan bufRead) buffers := make(chan bufRead)
go verboseWatcher(i, buffers) go eventLogger(i, buffers)
for { for {
n, _ := unix.Read(i.fd, buf) n, _ := unix.Read(i.fd, buf)
if !i.paused { if !i.paused {
@@ -89,73 +90,3 @@ func watch(i *Inotify) {
} }
} }
} }
func verboseWatcher(i *Inotify, buffers chan bufRead) {
for bf := range buffers {
n := bf.n
buf := bf.buf
if n < unix.SizeofInotifyEvent {
if n == 0 {
// If EOF is received. This should really never happen.
panic(fmt.Sprintf("No bytes read from fd"))
} else if n < 0 {
// If an error occurred while reading.
log.Printf("ERROR: reading from inotify: %d", n)
} else {
// Read was too short.
log.Printf("ERROR: Short read")
}
continue
}
var offset uint32
for offset <= uint32(n-unix.SizeofInotifyEvent) {
raw := (*unix.InotifyEvent)(unsafe.Pointer(&buf[offset]))
mask := uint32(raw.Mask)
nameLen := uint32(raw.Len)
name := i.watchers[int(raw.Wd)].dir
if nameLen > 0 {
bytes := (*[unix.PathMax]byte)(unsafe.Pointer(&buf[offset+unix.SizeofInotifyEvent]))
if uint32(len(bytes)) > nameLen {
name += "/" + strings.TrimRight(string(bytes[0:nameLen]), "\000")
}
}
ev := newEvent(name, mask)
log.Printf("### %+v", ev)
offset += unix.SizeofInotifyEvent + nameLen
}
}
}
type Event struct {
name string
op string
}
func (e Event) String() string {
return fmt.Sprintf("%10s | %s", e.op, e.name)
}
func newEvent(name string, mask uint32) Event {
e := Event{name: name}
if mask&unix.IN_CREATE == unix.IN_CREATE || mask&unix.IN_MOVED_TO == unix.IN_MOVED_TO {
e.op = "CREATE"
}
if mask&unix.IN_DELETE_SELF == unix.IN_DELETE_SELF || mask&unix.IN_DELETE == unix.IN_DELETE {
e.op = "REMOVE"
}
if mask&unix.IN_MODIFY == unix.IN_MODIFY {
e.op = "WRITE"
}
if mask&unix.IN_MOVE_SELF == unix.IN_MOVE_SELF || mask&unix.IN_MOVED_FROM == unix.IN_MOVED_FROM {
e.op = "RENAME"
}
if mask&unix.IN_ATTRIB == unix.IN_ATTRIB {
e.op = "CHMOD"
}
return e
}

View File

@@ -5,28 +5,49 @@ import (
"io/ioutil" "io/ioutil"
"log" "log"
"strconv" "strconv"
"strings"
) )
type ProcList map[int]string type ProcList map[int]string
// type Proc struct {
// Cmd string
// User string
// }
func NewProcList() *ProcList { func NewProcList() *ProcList {
pl := make(ProcList) pl := make(ProcList)
return &pl return &pl
} }
func (pl ProcList) Refresh() error { func (pl ProcList) Refresh() error {
pids, err := getPIDs()
if err != nil {
return err
}
for i := len(pids) - 1; i >= 0; i-- {
pid := pids[i]
_, ok := pl[pid]
if !ok {
cmd, err := getCmd(pid)
if err != nil {
cmd = "???" // process probably terminated
}
uid, err := getUID(pid)
if err != nil {
uid = "???"
}
log.Printf("\x1b[31;1mCMD: UID=%-4s PID=%-6d | %s\x1b[0m\n", uid, pid, cmd)
pl[pid] = cmd
}
}
return nil
}
func getPIDs() ([]int, error) {
proc, err := ioutil.ReadDir("/proc") proc, err := ioutil.ReadDir("/proc")
if err != nil { if err != nil {
return fmt.Errorf("opening proc dir: %v", err) return nil, fmt.Errorf("opening proc dir: %v", err)
} }
pids := make([]int, 0) pids := make([]int, 0)
for _, f := range proc { for _, f := range proc {
if f.IsDir() { if f.IsDir() {
name := f.Name() name := f.Name()
@@ -37,20 +58,7 @@ func (pl ProcList) Refresh() error {
pids = append(pids, pid) pids = append(pids, pid)
} }
} }
return pids, nil
for i := len(pids) - 1; i >= 0; i-- {
pid := pids[i]
_, ok := pl[pid]
if !ok {
cmd, err := getCmd(pid)
if err != nil {
cmd = "UNKNOWN" // process probably terminated
}
log.Printf("New process: %5d: %s\n", pid, cmd)
pl[pid] = cmd
}
}
return nil
} }
func getCmd(pid int) (string, error) { func getCmd(pid int) (string, error) {
@@ -66,3 +74,20 @@ func getCmd(pid int) (string, error) {
} }
return string(cmd), nil return string(cmd), nil
} }
func getUID(pid int) (string, error) {
statPath := fmt.Sprintf("/proc/%d/status", pid)
stat, err := ioutil.ReadFile(statPath)
if err != nil {
return "", err
}
lines := strings.Split(string(stat), "\n")
if len(lines) < 9 {
return "", fmt.Errorf("no uid information")
}
uidL := strings.Split(lines[8], "\t")
if len(uidL) < 2 {
return "", fmt.Errorf("uid line read incomplete")
}
return uidL[1], nil
}

View File

@@ -3,33 +3,31 @@ package walker
import ( import (
"fmt" "fmt"
"io/ioutil" "io/ioutil"
"path/filepath"
) )
func Walk(root string) (dirCh chan string, errCh chan error) { func Walk(root string, depth int) (dirCh chan string, errCh chan error, doneCh chan struct{}) {
dirCh = make(chan string) dirCh = make(chan string)
errCh = make(chan error) errCh = make(chan error)
dirs := make([]string, 1) doneCh = make(chan struct{})
dirs[0] = root
go func() { go func() {
dirCh <- root descent(root, depth-1, dirCh, errCh, doneCh)
}()
go func() {
for {
if len(dirs) == 0 {
break
}
dirs = descent(dirs, dirCh, errCh)
}
close(dirCh) close(dirCh)
close(errCh)
}() }()
return dirCh, errCh return dirCh, errCh, doneCh
}
func descent(dir string, depth int, dirCh chan string, errCh chan error, doneCh chan struct{}) {
select {
case dirCh <- dir:
case <-doneCh:
return
}
if depth < 0 {
return
} }
func descent(dirs []string, dirCh chan string, errCh chan error) []string {
next := make([]string, 0)
for _, dir := range dirs {
ls, err := ioutil.ReadDir(dir) ls, err := ioutil.ReadDir(dir)
if err != nil { if err != nil {
errCh <- fmt.Errorf("opening dir %s: %v", dir, err) errCh <- fmt.Errorf("opening dir %s: %v", dir, err)
@@ -37,11 +35,8 @@ func descent(dirs []string, dirCh chan string, errCh chan error) []string {
for _, e := range ls { for _, e := range ls {
if e.IsDir() { if e.IsDir() {
newDir := dir + e.Name() + "/" newDir := filepath.Join(dir, e.Name())
dirCh <- newDir descent(newDir, depth-1, dirCh, errCh, doneCh)
next = append(next, newDir)
} }
} }
} }
return next
}