//go:build !wasm package rui import ( "fmt" "net/http" "strconv" "strings" "sync" "github.com/gorilla/websocket" ) type wsBrige struct { conn *websocket.Conn answer map[int]chan DataObject answerID int answerMutex sync.Mutex closed bool buffer strings.Builder } var upgrader = websocket.Upgrader{ ReadBufferSize: 1024, WriteBufferSize: 8096, } func CreateSocketBrige(w http.ResponseWriter, req *http.Request) webBrige { conn, err := upgrader.Upgrade(w, req, nil) if err != nil { ErrorLog(err.Error()) return nil } brige := new(wsBrige) brige.answerID = 1 brige.answer = make(map[int]chan DataObject) brige.conn = conn brige.closed = false return brige } func (brige *wsBrige) close() { brige.closed = true brige.conn.Close() } func (brige *wsBrige) argToString(arg any) (string, bool) { switch arg := arg.(type) { case string: arg = strings.ReplaceAll(arg, "\\", `\\`) arg = strings.ReplaceAll(arg, "'", `\'`) arg = strings.ReplaceAll(arg, "\n", `\n`) arg = strings.ReplaceAll(arg, "\r", `\r`) arg = strings.ReplaceAll(arg, "\t", `\t`) arg = strings.ReplaceAll(arg, "\b", `\b`) arg = strings.ReplaceAll(arg, "\f", `\f`) arg = strings.ReplaceAll(arg, "\v", `\v`) return `'` + arg + `'`, true case rune: switch arg { case '\t': return `'\t'`, true case '\r': return `'\r'`, true case '\n': return `'\n'`, true case '\b': return `'\b'`, true case '\f': return `'\f'`, true case '\v': return `'\v'`, true case '\'': return `'\''`, true case '\\': return `'\\'`, true } if arg < ' ' { return fmt.Sprintf(`'\x%02d'`, int(arg)), true } return `'` + string(arg) + `'`, true case bool: if arg { return "true", true } else { return "false", true } case float32: return fmt.Sprintf("%g", float64(arg)), true case float64: return fmt.Sprintf("%g", arg), true default: if n, ok := isInt(arg); ok { return fmt.Sprintf("%d", n), true } } ErrorLog("Unsupported agument type") return "", false } func (brige *wsBrige) runFunc(funcName string, args ...any) bool { brige.buffer.Reset() brige.buffer.WriteString(funcName) brige.buffer.WriteRune('(') for i, arg := range args { argText, ok := brige.argToString(arg) if !ok { return false } if i > 0 { brige.buffer.WriteString(", ") } brige.buffer.WriteString(argText) } brige.buffer.WriteString(");") if err := brige.conn.WriteMessage(websocket.TextMessage, []byte(brige.buffer.String())); err != nil { ErrorLog(err.Error()) return false } return true } func (brige *wsBrige) readMessage() (string, bool) { _, p, err := brige.conn.ReadMessage() if err != nil { if !brige.closed { ErrorLog(err.Error()) } return "", false } return string(p), true } func (brige *wsBrige) writeMessage(script string) bool { if ProtocolInDebugLog { DebugLog("Run script:") DebugLog(script) } if err := brige.conn.WriteMessage(websocket.TextMessage, []byte(script)); err != nil { ErrorLog(err.Error()) return false } return true } func (brige *wsBrige) runGetterScript(script string) DataObject { brige.answerMutex.Lock() answerID := brige.answerID brige.answerID++ brige.answerMutex.Unlock() answer := make(chan DataObject) brige.answer[answerID] = answer errorText := "" if brige.conn != nil { script = "var answerID = " + strconv.Itoa(answerID) + ";\n" + script if ProtocolInDebugLog { DebugLog("\n" + script) } err := brige.conn.WriteMessage(websocket.TextMessage, []byte(script)) if err == nil { return <-answer } errorText = err.Error() } else { if ProtocolInDebugLog { DebugLog("\n" + script) } errorText = "No connection" } result := NewDataObject("error") result.SetPropertyValue("text", errorText) delete(brige.answer, answerID) return result } func (brige *wsBrige) answerReceived(answer DataObject) { if text, ok := answer.PropertyValue("answerID"); ok { if id, err := strconv.Atoi(text); err == nil { if chanel, ok := brige.answer[id]; ok { chanel <- answer delete(brige.answer, id) } else { ErrorLog("Bad answerID = " + text + " (chan not found)") } } else { ErrorLog("Invalid answerID = " + text) } } else { ErrorLog("answerID not found") } } func (brige *wsBrige) remoteAddr() string { return brige.conn.RemoteAddr().String() }