475 lines
12 KiB
Go
475 lines
12 KiB
Go
//go:build js && wasm
|
|
// +build js,wasm
|
|
|
|
package wasm
|
|
|
|
import (
|
|
"encoding/hex"
|
|
"fmt"
|
|
"syscall/js"
|
|
|
|
"git.quad4.io/Networks/Reticulum-Go/pkg/common"
|
|
"git.quad4.io/Networks/Reticulum-Go/pkg/debug"
|
|
"git.quad4.io/Networks/Reticulum-Go/pkg/destination"
|
|
"git.quad4.io/Networks/Reticulum-Go/pkg/identity"
|
|
"git.quad4.io/Networks/Reticulum-Go/pkg/interfaces"
|
|
"git.quad4.io/Networks/Reticulum-Go/pkg/packet"
|
|
"git.quad4.io/Networks/Reticulum-Go/pkg/transport"
|
|
)
|
|
|
|
var (
|
|
reticulumTransport *transport.Transport
|
|
reticulumDest *destination.Destination
|
|
reticulumIdentity *identity.Identity
|
|
stats = struct {
|
|
packetsSent int
|
|
packetsReceived int
|
|
bytesSent int
|
|
bytesReceived int
|
|
}{}
|
|
packetCallback js.Value
|
|
announceHandler js.Value
|
|
)
|
|
|
|
// RegisterJSFunctions registers the Reticulum WASM API to the JavaScript global scope.
|
|
func RegisterJSFunctions() {
|
|
js.Global().Set("reticulum", js.ValueOf(map[string]interface{}{
|
|
"init": js.FuncOf(InitReticulum),
|
|
"getIdentity": js.FuncOf(GetIdentity),
|
|
"getDestination": js.FuncOf(GetDestination),
|
|
"connect": js.FuncOf(ConnectWebSocket),
|
|
"disconnect": js.FuncOf(DisconnectWebSocket),
|
|
"isConnected": js.FuncOf(IsConnected),
|
|
"requestPath": js.FuncOf(RequestPath),
|
|
"getStats": js.FuncOf(GetStats),
|
|
"setPacketCallback": js.FuncOf(SetPacketCallback),
|
|
"setAnnounceCallback": js.FuncOf(SetAnnounceCallback),
|
|
"sendData": js.FuncOf(SendDataJS),
|
|
"announce": js.FuncOf(SendAnnounceJS),
|
|
}))
|
|
}
|
|
|
|
func SetPacketCallback(this js.Value, args []js.Value) interface{} {
|
|
if len(args) > 0 && args[0].Type() == js.TypeFunction {
|
|
packetCallback = args[0]
|
|
return js.ValueOf(true)
|
|
}
|
|
return js.ValueOf(false)
|
|
}
|
|
|
|
func SetAnnounceCallback(this js.Value, args []js.Value) interface{} {
|
|
if len(args) > 0 && args[0].Type() == js.TypeFunction {
|
|
announceHandler = args[0]
|
|
return js.ValueOf(true)
|
|
}
|
|
return js.ValueOf(false)
|
|
}
|
|
|
|
func RequestPath(this js.Value, args []js.Value) interface{} {
|
|
if len(args) < 1 {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Destination hash required",
|
|
})
|
|
}
|
|
|
|
destHashHex := args[0].String()
|
|
destHash, err := hex.DecodeString(destHashHex)
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Invalid destination hash: %v", err),
|
|
})
|
|
}
|
|
|
|
if reticulumTransport == nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Reticulum not initialized",
|
|
})
|
|
}
|
|
|
|
if err := reticulumTransport.RequestPath(destHash, "", nil, true); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to request path: %v", err),
|
|
})
|
|
}
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"success": true,
|
|
})
|
|
}
|
|
|
|
func GetStats(this js.Value, args []js.Value) interface{} {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"packetsSent": stats.packetsSent,
|
|
"packetsReceived": stats.packetsReceived,
|
|
"bytesSent": stats.bytesSent,
|
|
"bytesReceived": stats.bytesReceived,
|
|
})
|
|
}
|
|
|
|
func InitReticulum(this js.Value, args []js.Value) interface{} {
|
|
if len(args) < 1 {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "WebSocket URL required",
|
|
})
|
|
}
|
|
|
|
if reticulumTransport != nil {
|
|
reticulumTransport.Close()
|
|
reticulumTransport = nil
|
|
}
|
|
|
|
wsURL := args[0].String()
|
|
appName := "wasm_core"
|
|
if len(args) >= 2 && args[1].Type() == js.TypeString {
|
|
appName = args[1].String()
|
|
}
|
|
|
|
var id *identity.Identity
|
|
var err error
|
|
|
|
// Check for existing identity in args
|
|
if len(args) >= 3 && !args[2].IsNull() && !args[2].IsUndefined() {
|
|
idHex := args[2].String()
|
|
idBytes, decodeErr := hex.DecodeString(idHex)
|
|
if decodeErr == nil && len(idBytes) == 64 {
|
|
id, err = identity.FromBytes(idBytes)
|
|
if err != nil {
|
|
debug.Log(debug.DEBUG_ERROR, "Failed to load provided identity, generating new one", "error", err)
|
|
id, err = identity.NewIdentity()
|
|
}
|
|
} else {
|
|
id, err = identity.NewIdentity()
|
|
}
|
|
} else {
|
|
id, err = identity.NewIdentity()
|
|
}
|
|
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to handle identity: %v", err),
|
|
})
|
|
}
|
|
|
|
cfg := common.DefaultConfig()
|
|
t := transport.NewTransport(cfg)
|
|
|
|
dest, err := destination.New(
|
|
id,
|
|
destination.IN,
|
|
destination.SINGLE,
|
|
appName,
|
|
t,
|
|
"browser",
|
|
)
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to create destination: %v", err),
|
|
})
|
|
}
|
|
|
|
dest.SetPacketCallback(func(data []byte, ni common.NetworkInterface) {
|
|
stats.packetsReceived++
|
|
stats.bytesReceived += len(data)
|
|
|
|
if !packetCallback.IsUndefined() {
|
|
// Convert bytes to JS Uint8Array for performance and compatibility
|
|
uint8Array := js.Global().Get("Uint8Array").New(len(data))
|
|
js.CopyBytesToJS(uint8Array, data)
|
|
packetCallback.Invoke(uint8Array)
|
|
}
|
|
})
|
|
|
|
dest.SetProofStrategy(destination.PROVE_ALL)
|
|
|
|
t.RegisterAnnounceHandler(&genericAnnounceHandler{})
|
|
|
|
wsInterface, err := interfaces.NewWebSocketInterface("wasm0", wsURL, true)
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to create WebSocket interface: %v", err),
|
|
})
|
|
}
|
|
|
|
wsInterface.SetPacketCallback(func(data []byte, ni common.NetworkInterface) {
|
|
msg := fmt.Sprintf("Received packet: %d bytes (type: 0x%02x)", len(data), data[0])
|
|
js.Global().Call("log", msg, "success")
|
|
debug.Log(debug.DEBUG_INFO, "WASM received packet", "bytes", len(data), "type", fmt.Sprintf("0x%02x", data[0]))
|
|
t.HandlePacket(data, ni)
|
|
})
|
|
|
|
if err := t.RegisterInterface("wasm0", wsInterface); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to register interface: %v", err),
|
|
})
|
|
}
|
|
|
|
if err := t.Start(); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to start transport: %v", err),
|
|
})
|
|
}
|
|
|
|
reticulumTransport = t
|
|
reticulumDest = dest
|
|
reticulumIdentity = id
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"success": true,
|
|
"identity": id.GetHexHash(),
|
|
"privateKey": hex.EncodeToString(id.GetPrivateKey()),
|
|
"destination": fmt.Sprintf("%x", dest.GetHash()),
|
|
})
|
|
}
|
|
|
|
// GetTransport returns the internal transport pointer.
|
|
func GetTransport() *transport.Transport {
|
|
return reticulumTransport
|
|
}
|
|
|
|
// GetDestinationPointer returns the internal destination pointer.
|
|
func GetDestinationPointer() *destination.Destination {
|
|
return reticulumDest
|
|
}
|
|
|
|
func GetIdentity(this js.Value, args []js.Value) interface{} {
|
|
if reticulumIdentity == nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Reticulum not initialized",
|
|
})
|
|
}
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"hash": reticulumIdentity.GetHexHash(),
|
|
})
|
|
}
|
|
|
|
func GetDestination(this js.Value, args []js.Value) interface{} {
|
|
if reticulumDest == nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Reticulum not initialized",
|
|
})
|
|
}
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"hash": fmt.Sprintf("%x", reticulumDest.GetHash()),
|
|
})
|
|
}
|
|
|
|
func IsConnected(this js.Value, args []js.Value) interface{} {
|
|
if reticulumTransport == nil {
|
|
return js.ValueOf(false)
|
|
}
|
|
|
|
ifaces := reticulumTransport.GetInterfaces()
|
|
for _, iface := range ifaces {
|
|
if iface.IsOnline() {
|
|
return js.ValueOf(true)
|
|
}
|
|
}
|
|
|
|
return js.ValueOf(false)
|
|
}
|
|
|
|
func ConnectWebSocket(this js.Value, args []js.Value) interface{} {
|
|
if reticulumTransport == nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Reticulum not initialized",
|
|
})
|
|
}
|
|
|
|
ifaces := reticulumTransport.GetInterfaces()
|
|
for name, iface := range ifaces {
|
|
if iface.IsOnline() {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"success": true,
|
|
"interface": name,
|
|
})
|
|
}
|
|
if err := iface.Start(); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to connect: %v", err),
|
|
})
|
|
}
|
|
return js.ValueOf(map[string]interface{}{
|
|
"success": true,
|
|
"interface": name,
|
|
})
|
|
}
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "WebSocket interface not found",
|
|
})
|
|
}
|
|
|
|
func DisconnectWebSocket(this js.Value, args []js.Value) interface{} {
|
|
if reticulumTransport == nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Reticulum not initialized",
|
|
})
|
|
}
|
|
|
|
ifaces := reticulumTransport.GetInterfaces()
|
|
for _, iface := range ifaces {
|
|
if err := iface.Stop(); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to stop interface: %v", err),
|
|
})
|
|
}
|
|
return js.ValueOf(map[string]interface{}{
|
|
"success": true,
|
|
})
|
|
}
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "WebSocket interface not found",
|
|
})
|
|
}
|
|
|
|
type genericAnnounceHandler struct{}
|
|
|
|
func (h *genericAnnounceHandler) AspectFilter() []string {
|
|
return nil
|
|
}
|
|
|
|
func (h *genericAnnounceHandler) ReceivePathResponses() bool {
|
|
return false
|
|
}
|
|
|
|
func (h *genericAnnounceHandler) ReceivedAnnounce(destHash []byte, ident interface{}, appData []byte, hops uint8) error {
|
|
if !announceHandler.IsUndefined() {
|
|
hashStr := hex.EncodeToString(destHash)
|
|
announceHandler.Invoke(js.ValueOf(map[string]interface{}{
|
|
"hash": hashStr,
|
|
"appData": string(appData),
|
|
"hops": int(hops),
|
|
}))
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// SendDataJS is the JS-facing wrapper for SendData
|
|
func SendDataJS(this js.Value, args []js.Value) interface{} {
|
|
if len(args) < 2 {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Destination hash and data required",
|
|
})
|
|
}
|
|
|
|
destHashHex := args[0].String()
|
|
destHash, err := hex.DecodeString(destHashHex)
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Invalid destination hash: %v", err),
|
|
})
|
|
}
|
|
|
|
// Support both string and Uint8Array data from JS
|
|
var data []byte
|
|
if args[1].Type() == js.TypeString {
|
|
data = []byte(args[1].String())
|
|
} else {
|
|
data = make([]byte, args[1].Length())
|
|
js.CopyBytesToGo(data, args[1])
|
|
}
|
|
|
|
return SendData(destHash, data)
|
|
}
|
|
|
|
// SendData is a generic function to send raw bytes to a destination
|
|
func SendData(destHash []byte, data []byte) interface{} {
|
|
if reticulumTransport == nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Reticulum not initialized",
|
|
})
|
|
}
|
|
|
|
remoteIdentity, err := identity.Recall(destHash)
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Identity not found. Wait for an announce from this peer!"),
|
|
})
|
|
}
|
|
|
|
targetDest, err := destination.FromHash(destHash, remoteIdentity, destination.SINGLE, reticulumTransport)
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to create target destination: %v", err),
|
|
})
|
|
}
|
|
|
|
encrypted, err := targetDest.Encrypt(data)
|
|
if err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Encryption failed: %v", err),
|
|
})
|
|
}
|
|
|
|
pkt := packet.NewPacket(
|
|
packet.DestinationSingle,
|
|
encrypted,
|
|
packet.PacketTypeData,
|
|
packet.ContextNone,
|
|
packet.PropagationBroadcast,
|
|
packet.HeaderType1,
|
|
nil,
|
|
true,
|
|
packet.FlagUnset,
|
|
)
|
|
pkt.DestinationHash = destHash
|
|
|
|
if err := pkt.Pack(); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Packet packing failed: %v", err),
|
|
})
|
|
}
|
|
|
|
if err := reticulumTransport.SendPacket(pkt); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Packet sending failed: %v", err),
|
|
})
|
|
}
|
|
|
|
stats.packetsSent++
|
|
stats.bytesSent += len(data)
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"success": true,
|
|
})
|
|
}
|
|
|
|
// SendAnnounceJS is the JS-facing wrapper for SendAnnounce
|
|
func SendAnnounceJS(this js.Value, args []js.Value) interface{} {
|
|
var appData []byte
|
|
if len(args) >= 1 && args[0].Type() == js.TypeString {
|
|
appData = []byte(args[0].String())
|
|
} else if len(args) >= 1 && args[0].Type() == js.TypeObject {
|
|
appData = make([]byte, args[0].Length())
|
|
js.CopyBytesToGo(appData, args[0])
|
|
}
|
|
return SendAnnounce(appData)
|
|
}
|
|
|
|
// SendAnnounce is a generic function to send an announce
|
|
func SendAnnounce(appData []byte) interface{} {
|
|
if reticulumDest == nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": "Reticulum not initialized",
|
|
})
|
|
}
|
|
|
|
if len(appData) > 0 {
|
|
reticulumDest.SetDefaultAppData(appData)
|
|
}
|
|
|
|
if err := reticulumDest.Announce(false, nil, nil); err != nil {
|
|
return js.ValueOf(map[string]interface{}{
|
|
"error": fmt.Sprintf("Failed to send announce: %v", err),
|
|
})
|
|
}
|
|
|
|
return js.ValueOf(map[string]interface{}{
|
|
"success": true,
|
|
})
|
|
}
|
|
|