mirror of
https://github.com/bettercap/bettercap.git
synced 2025-03-12 04:36:03 -07:00
201 lines
3.4 KiB
Go
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
|
|
}
|