bettercap/session/events.go
2021-01-27 00:17:25 +01:00

201 lines
3.4 KiB
Go

package session
import (
"fmt"
"os"
"sort"
"sync"
"time"
"github.com/evilsocket/islazy/log"
"github.com/evilsocket/islazy/tui"
)
type Event struct {
Tag string `json:"tag"`
Time time.Time `json:"time"`
Data interface{} `json:"data"`
}
type LogMessage struct {
Level log.Verbosity
Message string
}
func NewEvent(tag string, data interface{}) Event {
return Event{
Tag: tag,
Time: time.Now(),
Data: data,
}
}
func (e Event) Label() string {
m := e.Data.(LogMessage)
label := log.LevelName(m.Level)
color := log.LevelColor(m.Level)
return color + label + tui.RESET
}
type EventBus <-chan Event
type PrintCallback func(format string, args ...interface{})
type PrintWriter struct {
pool *EventPool
}
func (w PrintWriter) Write(p []byte) (n int, err error) {
w.pool.Printf("%s", string(p))
return len(p), nil
}
type EventPool struct {
*sync.Mutex
debug bool
silent bool
events []Event
listeners []chan Event
printLock sync.Mutex
printCbs []PrintCallback
Stdout PrintWriter
}
func NewEventPool(debug bool, silent bool) *EventPool {
pool := &EventPool{
Mutex: &sync.Mutex{},
debug: debug,
silent: silent,
events: make([]Event, 0),
listeners: make([]chan Event, 0),
printCbs: make([]PrintCallback, 0),
}
pool.Stdout = PrintWriter{
pool: pool,
}
return pool
}
func (p *EventPool) OnPrint(cb PrintCallback) {
p.printLock.Lock()
defer p.printLock.Unlock()
p.printCbs = append(p.printCbs, cb)
}
func (p *EventPool) Listen() EventBus {
p.Lock()
defer p.Unlock()
l := make(chan Event)
// make sure, without blocking, the new listener
// will receive all the queued events
go func() {
for i := len(p.events) - 1; i >= 0; i-- {
defer func() {
recover()
}()
l <- p.events[i]
}
}()
p.listeners = append(p.listeners, l)
return l
}
func (p *EventPool) Unlisten(listener EventBus) {
p.Lock()
defer p.Unlock()
for i, l := range p.listeners {
if l == listener {
close(l)
p.listeners = append(p.listeners[:i], p.listeners[i+1:]...)
return
}
}
}
func (p *EventPool) SetSilent(s bool) {
p.Lock()
defer p.Unlock()
p.silent = s
}
func (p *EventPool) SetDebug(d bool) {
p.Lock()
defer p.Unlock()
p.debug = d
}
func (p *EventPool) Add(tag string, data interface{}) {
p.Lock()
defer p.Unlock()
e := NewEvent(tag, data)
p.events = append([]Event{e}, p.events...)
// broadcast the event to every listener
for _, l := range p.listeners {
// do not block!
go func(ch chan Event) {
// channel might be closed
defer func() {
if recover() != nil {
}
}()
ch <- e
}(l)
}
}
func (p *EventPool) Printf(format string, a ...interface{}) {
p.printLock.Lock()
defer p.printLock.Unlock()
for _, cb := range p.printCbs {
cb(format, a...)
}
fmt.Printf(format, a...)
}
func (p *EventPool) Log(level log.Verbosity, format string, args ...interface{}) {
if level == log.DEBUG && !p.debug {
return
} else if level < log.ERROR && p.silent {
return
}
message := fmt.Sprintf(format, args...)
p.Add("sys.log", LogMessage{
level,
message,
})
if level == log.FATAL {
fmt.Fprintf(os.Stderr, "%s\n", message)
os.Exit(1)
}
}
func (p *EventPool) Clear() {
p.Lock()
defer p.Unlock()
p.events = make([]Event, 0)
}
func (p *EventPool) Sorted() []Event {
p.Lock()
defer p.Unlock()
sort.Slice(p.events, func(i, j int) bool {
return p.events[i].Time.Before(p.events[j].Time)
})
return p.events
}