This commit is contained in:
buffermet 2024-10-12 17:35:08 +02:00
commit 32995aada3
2 changed files with 107 additions and 24 deletions

View file

@ -136,7 +136,7 @@ func (mod *EventsStream) Render(output io.Writer, e session.Event) {
mod.viewGatewayEvent(output, e) mod.viewGatewayEvent(output, e)
} else if strings.HasPrefix(e.Tag, "zeroconf.") { } else if strings.HasPrefix(e.Tag, "zeroconf.") {
mod.viewZeroConfEvent(output, e) mod.viewZeroConfEvent(output, e)
} else if e.Tag != "tick" && e.Tag != "session.started" && e.Tag != "session.stopped" { } else if !strings.HasPrefix(e.Tag, "tick") && e.Tag != "session.started" && e.Tag != "session.stopped" {
fmt.Fprintf(output, "[%s] [%s] %v\n", e.Time.Format(mod.timeFormat), tui.Green(e.Tag), e) fmt.Fprintf(output, "[%s] [%s] %v\n", e.Time.Format(mod.timeFormat), tui.Green(e.Tag), e)
} }
} }

View file

@ -1,43 +1,71 @@
package ticker package ticker
import ( import (
"errors"
"strconv"
"time" "time"
"github.com/bettercap/bettercap/v2/session" "github.com/bettercap/bettercap/v2/session"
) )
type Ticker struct { type Params struct {
session.SessionModule
Period time.Duration Period time.Duration
Commands []string Commands []string
Running bool
}
type Ticker struct {
session.SessionModule
main Params
named map[string]*Params
} }
func NewTicker(s *session.Session) *Ticker { func NewTicker(s *session.Session) *Ticker {
mod := &Ticker{ mod := &Ticker{
SessionModule: session.NewSessionModule("ticker", s), SessionModule: session.NewSessionModule("ticker", s),
named: make(map[string]*Params),
} }
mod.AddParam(session.NewStringParameter("ticker.commands", mod.AddParam(session.NewStringParameter("ticker.commands",
"clear; net.show; events.show 20", "clear; net.show; events.show 20",
"", "",
"List of commands separated by a ;")) "List of commands for the main ticker separated by a ;"))
mod.AddParam(session.NewIntParameter("ticker.period", mod.AddParam(session.NewIntParameter("ticker.period",
"1", "1",
"Ticker period in seconds")) "Main ticker period in seconds"))
mod.AddHandler(session.NewModuleHandler("ticker on", "", mod.AddHandler(session.NewModuleHandler("ticker on", "",
"Start the ticker.", "Start the main ticker.",
func(args []string) error { func(args []string) error {
return mod.Start() return mod.Start()
})) }))
mod.AddHandler(session.NewModuleHandler("ticker off", "", mod.AddHandler(session.NewModuleHandler("ticker off", "",
"Stop the ticker.", "Stop the maint icker.",
func(args []string) error { func(args []string) error {
return mod.Stop() return mod.Stop()
})) }))
mod.AddHandler(session.NewModuleHandler("ticker.create <name> <period> <commands>",
`(?i)^ticker\.create\s+([^\s]+)\s+(\d+)\s+(.+)$`,
"Create and start a named ticker.",
func(args []string) error {
if period, err := strconv.Atoi(args[1]); err != nil {
return err
} else {
return mod.createNamed(args[0], period, args[2])
}
}))
mod.AddHandler(session.NewModuleHandler("ticker.destroy <name>",
`(?i)^ticker\.destroy\s+([^\s]+)$`,
"Stop a named ticker.",
func(args []string) error {
return mod.destroyNamed(args[0])
}))
return mod return mod
} }
@ -66,38 +94,93 @@ func (mod *Ticker) Configure() error {
return err return err
} }
mod.Commands = session.ParseCommands(commands) mod.main = Params{
mod.Period = time.Duration(period) * time.Second Commands: session.ParseCommands(commands),
Period: time.Duration(period) * time.Second,
Running: true,
}
return nil return nil
} }
type TickEvent struct{} type TickEvent struct{}
func (mod *Ticker) worker(name string, params *Params) {
isMain := name == "main"
eventName := "tick"
if isMain {
mod.Info("main ticker running with period %.fs", params.Period.Seconds())
} else {
eventName = "ticker." + name
mod.Info("ticker '%s' running with period %.fs", name, params.Period.Seconds())
}
tick := time.NewTicker(params.Period)
for range tick.C {
if !params.Running {
break
}
session.I.Events.Add(eventName, TickEvent{})
for _, cmd := range params.Commands {
if err := mod.Session.Run(cmd); err != nil {
mod.Error("%s", err)
}
}
}
if isMain {
mod.Info("main ticker stopped")
} else {
mod.Info("ticker '%s' stopped", name)
}
}
func (mod *Ticker) Start() error { func (mod *Ticker) Start() error {
if err := mod.Configure(); err != nil { if err := mod.Configure(); err != nil {
return err return err
} }
return mod.SetRunning(true, func() { return mod.SetRunning(true, func() {
mod.Info("running with period %.fs", mod.Period.Seconds()) mod.worker("main", &mod.main)
tick := time.NewTicker(mod.Period)
for range tick.C {
if !mod.Running() {
break
}
session.I.Events.Add("tick", TickEvent{})
for _, cmd := range mod.Commands {
if err := mod.Session.Run(cmd); err != nil {
mod.Error("%s", err)
}
}
}
}) })
} }
func (mod *Ticker) Stop() error { func (mod *Ticker) Stop() error {
mod.main.Running = false
for _, params := range mod.named {
params.Running = false
}
return mod.SetRunning(false, nil) return mod.SetRunning(false, nil)
} }
func (mod *Ticker) createNamed(name string, period int, commands string) error {
if _, found := mod.named[name]; found {
return errors.New("ticker '" + name + "' already exists")
}
params := &Params{
Commands: session.ParseCommands(commands),
Period: time.Duration(period) * time.Second,
Running: true,
}
mod.named[name] = params
go mod.worker(name, params)
return nil
}
func (mod *Ticker) destroyNamed(name string) error {
if _, found := mod.named[name]; !found {
return errors.New("ticker '" + name + "' not found")
}
mod.named[name].Running = false
delete(mod.named, name)
return nil
}