This commit is contained in:
Hay.banz 2025-06-01 13:45:03 +02:00 committed by GitHub
commit 69e4e6ada9
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
3 changed files with 90 additions and 189 deletions

View file

@ -8,7 +8,6 @@ import (
"github.com/bettercap/bettercap/v2/core" "github.com/bettercap/bettercap/v2/core"
"github.com/bettercap/bettercap/v2/network" "github.com/bettercap/bettercap/v2/network"
"github.com/evilsocket/islazy/fs" "github.com/evilsocket/islazy/fs"
"github.com/evilsocket/islazy/str" "github.com/evilsocket/islazy/str"
) )
@ -32,143 +31,46 @@ func Make(iface *network.Endpoint) FirewallManager {
restore: false, restore: false,
redirections: make(map[string]*Redirection), redirections: make(map[string]*Redirection),
} }
firewall.forwarding = firewall.IsForwardingEnabled() firewall.forwarding = firewall.IsForwardingEnabled()
return firewall return firewall
} }
func (f LinuxFirewall) enableFeature(filename string, enable bool) error { func (f *LinuxFirewall) enableFeature(filename string, enable bool) error {
var value string value := "0"
if enable { if enable {
value = "1" value = "1"
} else {
value = "0"
} }
fd, err := os.OpenFile(filename, os.O_CREATE|os.O_APPEND|os.O_WRONLY, 0600) fd, err := os.OpenFile(filename, os.O_CREATE|os.O_WRONLY, 0600)
if err != nil { if err != nil {
return err return fmt.Errorf("failed to open file %s: %w", filename, err)
} }
defer fd.Close() defer fd.Close()
_, err = fd.WriteString(value) if _, err := fd.WriteString(value); err != nil {
return err return fmt.Errorf("failed to write to file %s: %w", filename, err)
}
return nil
} }
func (f LinuxFirewall) IsForwardingEnabled() bool { func (f *LinuxFirewall) IsForwardingEnabled() bool {
content, err := ioutil.ReadFile(IPV4ForwardingFile)
if out, err := ioutil.ReadFile(IPV4ForwardingFile); err != nil { if err != nil {
return false return false
} else {
return str.Trim(string(out)) == "1"
} }
return str.Trim(string(content)) == "1"
} }
func (f LinuxFirewall) EnableForwarding(enabled bool) error { func (f *LinuxFirewall) EnableForwarding(enabled bool) error {
if err := f.enableFeature(IPV4ForwardingFile, enabled); err != nil { if err := f.enableFeature(IPV4ForwardingFile, enabled); err != nil {
return err return err
} }
if fs.Exists(IPV6ForwardingFile) { if fs.Exists(IPV6ForwardingFile) {
return f.enableFeature(IPV6ForwardingFile, enabled) if err := f.enableFeature(IPV6ForwardingFile, enabled); err != nil {
return err
}
} }
f.restore = true f.restore = true
return nil return nil
}
func (f *LinuxFirewall) getCommandLine(r *Redirection, enabled bool) (cmdLine []string) {
action := "-A"
destination := ""
if !enabled {
action = "-D"
}
if strings.Count(r.DstAddress, ":") < 2 {
destination = r.DstAddress
} else {
destination = fmt.Sprintf("[%s]", r.DstAddress)
}
if r.SrcAddress == "" {
cmdLine = []string{
"-t", "nat",
action, "PREROUTING",
"-i", r.Interface,
"-p", r.Protocol,
"--dport", fmt.Sprintf("%d", r.SrcPort),
"-j", "DNAT",
"--to", fmt.Sprintf("%s:%d", destination, r.DstPort),
}
} else {
cmdLine = []string{
"-t", "nat",
action, "PREROUTING",
"-i", r.Interface,
"-p", r.Protocol,
"-d", r.SrcAddress,
"--dport", fmt.Sprintf("%d", r.SrcPort),
"-j", "DNAT",
"--to", fmt.Sprintf("%s:%d", destination, r.DstPort),
}
}
return
}
func (f *LinuxFirewall) EnableRedirection(r *Redirection, enabled bool) error {
cmdLine := f.getCommandLine(r, enabled)
rkey := r.String()
_, found := f.redirections[rkey]
cmd := ""
if strings.Count(r.DstAddress, ":") < 2 {
cmd = "iptables"
} else {
cmd = "ip6tables"
}
if enabled {
if found {
return fmt.Errorf("Redirection '%s' already enabled.", rkey)
}
f.redirections[rkey] = r
// accept all
if _, err := core.Exec(cmd, []string{"-P", "FORWARD", "ACCEPT"}); err != nil {
return err
} else if _, err := core.Exec(cmd, cmdLine); err != nil {
return err
}
} else {
if !found {
return nil
}
delete(f.redirections, r.String())
if _, err := core.Exec(cmd, cmdLine); err != nil {
return err
}
}
return nil
}
func (f LinuxFirewall) Restore() {
if f.restore == false {
return
}
for _, r := range f.redirections {
if err := f.EnableRedirection(r, false); err != nil {
fmt.Printf("%s", err)
}
}
if err := f.EnableForwarding(f.forwarding); err != nil {
fmt.Printf("%s", err)
}
}

View file

@ -4,7 +4,6 @@ import (
"bytes" "bytes"
"fmt" "fmt"
"io" "io"
"io/ioutil"
"net/http" "net/http"
"net/url" "net/url"
"strings" "strings"
@ -12,25 +11,22 @@ import (
"github.com/robertkrimen/otto" "github.com/robertkrimen/otto"
) )
type httpPackage struct { type httpPackage struct{}
}
type httpResponse struct { type httpResponse struct {
Error error Error error
Response *http.Response Response *http.Response
Raw []byte Raw []byte
Body string Body string
JSON interface{}
} }
// Encode encodes a string for use in a URL query.
func (c httpPackage) Encode(s string) string { func (c httpPackage) Encode(s string) string {
return url.QueryEscape(s) return url.QueryEscape(s)
} }
func (c httpPackage) Request(method string, uri string, // Request sends an HTTP request with the specified method, URL, headers, form data, or JSON payload.
headers map[string]string, func (c httpPackage) Request(method, uri string, headers map[string]string, form map[string]string, json string) httpResponse {
form map[string]string,
json string) httpResponse {
var reader io.Reader var reader io.Reader
if form != nil { if form != nil {
@ -38,14 +34,14 @@ func (c httpPackage) Request(method string, uri string,
for k, v := range form { for k, v := range form {
data.Set(k, v) data.Set(k, v)
} }
reader = bytes.NewBufferString(data.Encode()) reader = strings.NewReader(data.Encode())
} else if json != "" { } else if json != "" {
reader = strings.NewReader(json) reader = strings.NewReader(json)
} }
req, err := http.NewRequest(method, uri, reader) req, err := http.NewRequest(method, uri, reader)
if err != nil { if err != nil {
return httpResponse{Error: err} return httpResponse{Error: fmt.Errorf("failed to create request: %w", err)}
} }
if form != nil { if form != nil {
@ -55,18 +51,18 @@ func (c httpPackage) Request(method string, uri string,
} }
for name, value := range headers { for name, value := range headers {
req.Header.Add(name, value) req.Header.Set(name, value)
} }
resp, err := http.DefaultClient.Do(req) resp, err := http.DefaultClient.Do(req)
if err != nil { if err != nil {
return httpResponse{Error: err} return httpResponse{Error: fmt.Errorf("request failed: %w", err)}
} }
defer resp.Body.Close() defer resp.Body.Close()
raw, err := ioutil.ReadAll(resp.Body) raw, err := io.ReadAll(resp.Body)
if err != nil { if err != nil {
return httpResponse{Error: err} return httpResponse{Error: fmt.Errorf("failed to read response body: %w", err)}
} }
res := httpResponse{ res := httpResponse{
@ -75,82 +71,85 @@ func (c httpPackage) Request(method string, uri string,
Body: string(raw), Body: string(raw),
} }
if resp.StatusCode != http.StatusOK { if resp.StatusCode >= 400 {
res.Error = fmt.Errorf("%s", resp.Status) res.Error = fmt.Errorf("HTTP error: %s", resp.Status)
} }
return res return res
} }
// Get sends a GET request.
func (c httpPackage) Get(url string, headers map[string]string) httpResponse { func (c httpPackage) Get(url string, headers map[string]string) httpResponse {
return c.Request("GET", url, headers, nil, "") return c.Request(http.MethodGet, url, headers, nil, "")
} }
// PostForm sends a POST request with form data.
func (c httpPackage) PostForm(url string, headers map[string]string, form map[string]string) httpResponse { func (c httpPackage) PostForm(url string, headers map[string]string, form map[string]string) httpResponse {
return c.Request("POST", url, headers, form, "") return c.Request(http.MethodPost, url, headers, form, "")
} }
// PostJSON sends a POST request with a JSON payload.
func (c httpPackage) PostJSON(url string, headers map[string]string, json string) httpResponse { func (c httpPackage) PostJSON(url string, headers map[string]string, json string) httpResponse {
return c.Request("POST", url, headers, nil, json) return c.Request(http.MethodPost, url, headers, nil, json)
} }
// httpRequest processes JavaScript calls for HTTP requests.
func httpRequest(call otto.FunctionCall) otto.Value { func httpRequest(call otto.FunctionCall) otto.Value {
argv := call.ArgumentList if len(call.ArgumentList) < 2 {
argc := len(argv) return ReportError("httpRequest: expected at least 2 arguments, got %d", len(call.ArgumentList))
if argc < 2 {
return ReportError("httpRequest: expected 2 or more, %d given instead.", argc)
} }
method := argv[0].String() method := call.Argument(0).String()
url := argv[1].String() url := call.Argument(1).String()
var reader io.Reader
if len(call.ArgumentList) >= 3 {
data := call.Argument(2).String()
reader = bytes.NewBufferString(data)
}
req, err := http.NewRequest(method, url, reader)
if err != nil {
return ReportError("failed to create request for URL %s: %s", url, err)
}
if len(call.ArgumentList) > 3 {
headers, _ := call.Argument(3).Export()
if headerMap, ok := headers.(map[string]interface{}); ok {
for key, value := range headerMap {
if strValue, ok := value.(string); ok {
req.Header.Set(key, strValue)
}
}
}
}
client := &http.Client{} client := &http.Client{}
req, err := http.NewRequest(method, url, nil)
if argc >= 3 {
data := argv[2].String()
req, err = http.NewRequest(method, url, bytes.NewBuffer([]byte(data)))
if err != nil {
return ReportError("Could create request to url %s: %s", url, err)
}
if argc > 3 {
headers := argv[3].Object()
for _, key := range headers.Keys() {
v, err := headers.Get(key)
if err != nil {
return ReportError("Could add header %s to request: %s", key, err)
}
req.Header.Add(key, v.String())
}
}
} else if err != nil {
return ReportError("Could create request to url %s: %s", url, err)
}
resp, err := client.Do(req) resp, err := client.Do(req)
if err != nil { if err != nil {
return ReportError("Could not request url %s: %s", url, err) return ReportError("failed to execute request to URL %s: %s", url, err)
} }
defer resp.Body.Close() defer resp.Body.Close()
body, err := ioutil.ReadAll(resp.Body) body, err := io.ReadAll(resp.Body)
if err != nil { if err != nil {
return ReportError("Could not read response: %s", err) return ReportError("failed to read response body: %s", err)
} }
object, err := otto.New().Object("({})") responseObj, _ := otto.New().Object(`({})`)
if err != nil { responseObj.Set("body", string(body))
return ReportError("Could not create response object: %s", err)
}
err = object.Set("body", string(body)) v, err := otto.ToValue(responseObj)
if err != nil { if err != nil {
return ReportError("Could not populate response object: %s", err) return ReportError("failed to convert response to Otto value: %s", err)
}
v, err := otto.ToValue(object)
if err != nil {
return ReportError("Could not convert to object: %s", err)
} }
return v return v
} }
// ReportError formats and returns a JavaScript-compatible error.
func ReportError(format string, args ...interface{}) otto.Value {
errMessage := fmt.Sprintf(format, args...)
fmt.Println("Error:", errMessage) // Log the error
val, _ := otto.ToValue(fmt.Errorf(errMessage))
return val
}

View file

@ -6,13 +6,14 @@ import (
"time" "time"
) )
// CANDevice represents a CAN (Controller Area Network) device with activity tracking.
type CANDevice struct { type CANDevice struct {
sync.Mutex sync.Mutex
LastSeen time.Time LastSeen time.Time // Timestamp of the last activity.
Name string Name string // Name of the device.
Description string Description string // Description of the device.
Frames uint64 Frames uint64 // Number of frames sent/received.
Read uint64 Read uint64 // Total bytes read.
} }
type canDeviceJSON struct { type canDeviceJSON struct {
@ -23,18 +24,18 @@ type canDeviceJSON struct {
Read uint64 `json:"read"` Read uint64 `json:"read"`
} }
func NewCANDevice(name string, description string, payload []byte) *CANDevice { // NewCANDevice initializes a new CANDevice with the provided name, description, and payload.
dev := &CANDevice{ func NewCANDevice(name, description string, payload []byte) *CANDevice {
return &CANDevice{
LastSeen: time.Now(), LastSeen: time.Now(),
Name: name, Name: name,
Description: description, Description: description,
Read: uint64(len(payload)),
Frames: 1, Frames: 1,
Read: uint64(len(payload)),
}
} }
return dev // MarshalJSON customizes the JSON serialization of CANDevice.
}
func (dev *CANDevice) MarshalJSON() ([]byte, error) { func (dev *CANDevice) MarshalJSON() ([]byte, error) {
dev.Lock() dev.Lock()
defer dev.Unlock() defer dev.Unlock()
@ -43,21 +44,20 @@ func (dev *CANDevice) MarshalJSON() ([]byte, error) {
LastSeen: dev.LastSeen, LastSeen: dev.LastSeen,
Name: dev.Name, Name: dev.Name,
Description: dev.Description, Description: dev.Description,
Read: dev.Read,
Frames: dev.Frames, Frames: dev.Frames,
Read: dev.Read,
} }
return json.Marshal(doc) return json.Marshal(doc)
} }
// AddPayload updates the CANDevice's statistics with the new payload data.
func (dev *CANDevice) AddPayload(payload []byte) { func (dev *CANDevice) AddPayload(payload []byte) {
dev.Lock() dev.Lock()
defer dev.Unlock() defer dev.Unlock()
sz := len(payload) if len(payload) > 0 {
if payload != nil && sz > 0 { dev.Read += uint64(len(payload))
dev.Read += uint64(sz)
} }
dev.Frames++
dev.Frames += 1
} }