hotfix: hotfix 2 for tcp.proxy

This commit is contained in:
evilsocket 2025-08-18 19:14:05 +02:00
commit 42da612113
2 changed files with 75 additions and 54 deletions

View file

@ -1,6 +1,7 @@
package tcp_proxy package tcp_proxy
import ( import (
"encoding/json"
"net" "net"
"strings" "strings"
@ -62,24 +63,15 @@ func (s *TcpProxyScript) OnData(from, to net.Addr, data []byte, callback func(ca
} }
func toByteArray(ret interface{}) []byte { func toByteArray(ret interface{}) []byte {
// Handle different array types that otto.Export() might return // this approach is a bit hacky but it handles all cases
switch v := ret.(type) {
case []interface{}: // serialize ret to JSON
// Mixed type array if jsonData, err := json.Marshal(ret); err == nil {
result := make([]byte, len(v)) // attempt to deserialize as []float64
for i, elem := range v { var back2Array []float64
if num, ok := toNumber(elem); ok && num >= 0 && num <= 255 { if err := json.Unmarshal(jsonData, &back2Array); err == nil {
result[i] = byte(num) result := make([]byte, len(back2Array))
} else { for i, num := range back2Array {
log.Error("array element at index %d is not a valid byte value %+v", i, elem)
return nil
}
}
return result
case []int64:
// Array of integers
result := make([]byte, len(v))
for i, num := range v {
if num >= 0 && num <= 255 { if num >= 0 && num <= 255 {
result[i] = byte(num) result[i] = byte(num)
} else { } else {
@ -88,34 +80,12 @@ func toByteArray(ret interface{}) []byte {
} }
} }
return result return result
case []float64:
// Array of floats
result := make([]byte, len(v))
for i, num := range v {
if num >= 0 && num <= 255 {
result[i] = byte(num)
} else { } else {
log.Error("array element at index %d is not a valid byte value %f", i, num) log.Error("failed to deserialize %+v to []float64: %v", ret, err)
return nil
}
}
return result
default:
log.Error("unexpected array type returned from onData: %T, value = %+v", ret, ret)
return nil
} }
} else {
log.Error("failed to serialize %+v to JSON: %v", ret, err)
} }
// toNumber tries to convert an interface{} to a float64 return nil
func toNumber(v interface{}) (float64, bool) {
switch n := v.(type) {
case float64:
return n, true
case int64:
return float64(n), true
case int:
return float64(n), true
default:
return 0, false
}
} }

View file

@ -116,3 +116,54 @@ func TestOnData_ReturnsDynamicArray(t *testing.T) {
} }
} }
} }
func TestOnData_ReturnsMixedArray(t *testing.T) {
jsCode := `
function charToInt(value) {
return value.charCodeAt()
}
function onData(from, to, data) {
st_data = String.fromCharCode.apply(null, data)
if( st_data.indexOf("mysearch") != -1 ) {
payload = "mypayload";
st_data = st_data.replace("mysearch", payload);
res_int_arr = st_data.split("").map(charToInt) // []uint16
res_int_arr[0] = payload.length + 1; // first index is float64 and rest []uint16
return res_int_arr;
}
return data;
}
`
plug, err := plugin.Parse(jsCode)
if err != nil {
t.Fatalf("Failed to parse plugin: %v", err)
}
script := &TcpProxyScript{
Plugin: plug,
doOnData: plug.HasFunc("onData"),
}
from := &net.TCPAddr{IP: net.ParseIP("192.168.1.1"), Port: 1234}
to := &net.TCPAddr{IP: net.ParseIP("192.168.1.6"), Port: 5678}
data := []byte("Hello mysearch world")
result := script.OnData(from, to, data, nil)
expected := []byte("\x0aello mypayload world")
if result == nil {
t.Fatal("Expected non-nil result when callback returns array of integers")
}
if len(result) != len(expected) {
t.Fatalf("Expected result length %d, got %d", len(expected), len(result))
}
for i, b := range result {
if b != expected[i] {
t.Errorf("Expected byte at index %d to be %d, got %d", i, expected[i], b)
}
}
}