Compare commits
5 Commits
main
...
49880c837d
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
49880c837d | ||
|
|
4dd5a40dfe | ||
|
|
13b583a77e | ||
|
|
c9c4800d68 | ||
|
|
d793e13361 |
@@ -1,4 +1,13 @@
|
||||
package main
|
||||
|
||||
import (
|
||||
"net/http"
|
||||
"tts-room/src/server"
|
||||
)
|
||||
|
||||
func main() {
|
||||
s := server.NewServer()
|
||||
if err := http.ListenAndServe(":10000", s); err != nil {
|
||||
panic(err)
|
||||
}
|
||||
}
|
||||
|
||||
9
go.mod
9
go.mod
@@ -1,3 +1,10 @@
|
||||
module tts-room
|
||||
|
||||
go 1.22.2
|
||||
go 1.24.0
|
||||
|
||||
require (
|
||||
github.com/gorilla/websocket v1.5.3
|
||||
golang.org/x/time v0.14.0
|
||||
)
|
||||
|
||||
require github.com/google/uuid v1.6.0 // indirect
|
||||
|
||||
6
go.sum
Normal file
6
go.sum
Normal file
@@ -0,0 +1,6 @@
|
||||
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
|
||||
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
|
||||
github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg=
|
||||
github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
|
||||
golang.org/x/time v0.14.0 h1:MRx4UaLrDotUKUdCIqzPC48t1Y9hANFKIRpNx+Te8PI=
|
||||
golang.org/x/time v0.14.0/go.mod h1:eL/Oa2bBBK0TkX57Fyni+NgnyQQN4LitPmob2Hjnqw4=
|
||||
78
src/public/index.html
Normal file
78
src/public/index.html
Normal file
@@ -0,0 +1,78 @@
|
||||
<!DOCTYPE html>
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
|
||||
<meta name="viewport" content="width=device-width" />
|
||||
<script>
|
||||
window.addEventListener("load", function(evt) {
|
||||
var output = document.getElementById("output");
|
||||
var input = document.getElementById("input");
|
||||
var ws;
|
||||
|
||||
var print = function(message) {
|
||||
var d = document.createElement("div");
|
||||
d.textContent = message;
|
||||
output.appendChild(d);
|
||||
output.scroll(0, output.scrollHeight);
|
||||
};
|
||||
|
||||
ws = new WebSocket("ws://"+window.location.host+"/ws");
|
||||
ws.onopen = function(evt) {
|
||||
print("OPEN");
|
||||
}
|
||||
ws.onclose = function(evt) {
|
||||
print("CLOSE");
|
||||
ws = null;
|
||||
}
|
||||
ws.onmessage = function(evt) {
|
||||
const synth = window.speechSynthesis;
|
||||
const voices = synth.getVoices().sort(function (a, b) {
|
||||
const aname = a.name.toUpperCase();
|
||||
const bname = b.name.toUpperCase();
|
||||
|
||||
if (aname < bname) {
|
||||
return -1;
|
||||
} else if (aname == bname) {
|
||||
return 0;
|
||||
} else {
|
||||
return +1;
|
||||
}
|
||||
});
|
||||
const idx = false ? 0 : voices.length-1;
|
||||
|
||||
const utterThis = new SpeechSynthesisUtterance(evt.data);
|
||||
utterThis.voice = voices[idx];
|
||||
//utterThis.pitch = 10;
|
||||
//utterThis.rate = 10;
|
||||
window.speechSynthesis.speak(utterThis);
|
||||
print("RESPONSE: " + evt.data);
|
||||
}
|
||||
ws.onerror = function(evt) {
|
||||
print("ERROR: " + evt.data);
|
||||
}
|
||||
|
||||
document.getElementById("send").onclick = function(evt) {
|
||||
if (!ws || !input.value) {
|
||||
return false;
|
||||
}
|
||||
ws.send(JSON.stringify({
|
||||
"text": input.value
|
||||
}));
|
||||
print("SENT: " + input.value);
|
||||
input.value = "";
|
||||
return false;
|
||||
};
|
||||
});
|
||||
</script>
|
||||
</head>
|
||||
<body>
|
||||
<div style="width: 80%; height: 80%; margin: auto; display: flex; flex-direction: row;">
|
||||
<form style="flex-grow: 1;">
|
||||
<p><input id="input" type="text" value="" autofocus>
|
||||
<button id="send">Send</button>
|
||||
</form>
|
||||
<div id="output" style="flex-grow: 1; overflow-y: scroll;"></div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
5
src/server/message.go
Normal file
5
src/server/message.go
Normal file
@@ -0,0 +1,5 @@
|
||||
package server
|
||||
|
||||
type message struct {
|
||||
Text string
|
||||
}
|
||||
58
src/server/server.go
Normal file
58
src/server/server.go
Normal file
@@ -0,0 +1,58 @@
|
||||
package server
|
||||
|
||||
import (
|
||||
"log"
|
||||
"net/http"
|
||||
)
|
||||
|
||||
type Server struct {
|
||||
sessions []*session
|
||||
}
|
||||
|
||||
func NewServer() *Server {
|
||||
return &Server{}
|
||||
}
|
||||
|
||||
func (s *Server) ServeHTTP(w http.ResponseWriter, r *http.Request) {
|
||||
switch r.URL.Path {
|
||||
case "/ws":
|
||||
if err := s.WS(w, r); err != nil {
|
||||
log.Println("[ws]", err)
|
||||
}
|
||||
default:
|
||||
http.FileServer(http.Dir("./src/public")).ServeHTTP(w, r)
|
||||
}
|
||||
}
|
||||
|
||||
func (s *Server) WS(w http.ResponseWriter, r *http.Request) error {
|
||||
sess, err := newSession(w, r, nil)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
defer sess.Close()
|
||||
|
||||
sess.cb = func(m message) error {
|
||||
log.Printf("cbing to all other sessions %+v", m)
|
||||
for i := range s.sessions {
|
||||
if s.sessions[i].id != sess.id {
|
||||
select {
|
||||
case s.sessions[i].scatterc <- m:
|
||||
case <-s.sessions[i].ctx.Done():
|
||||
}
|
||||
}
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
s.sessions = append(s.sessions, sess)
|
||||
defer func() {
|
||||
for i := range s.sessions {
|
||||
if s.sessions[i].id == sess.id {
|
||||
s.sessions = append(s.sessions[:i], s.sessions[i+1:]...)
|
||||
return
|
||||
}
|
||||
}
|
||||
}()
|
||||
|
||||
return sess.Run()
|
||||
}
|
||||
104
src/server/session.go
Normal file
104
src/server/session.go
Normal file
@@ -0,0 +1,104 @@
|
||||
package server
|
||||
|
||||
import (
|
||||
"context"
|
||||
"encoding/json"
|
||||
"log"
|
||||
"net/http"
|
||||
"sync"
|
||||
|
||||
"github.com/google/uuid"
|
||||
"github.com/gorilla/websocket"
|
||||
"golang.org/x/time/rate"
|
||||
)
|
||||
|
||||
type session struct {
|
||||
ctx context.Context
|
||||
can context.CancelFunc
|
||||
ws *websocket.Conn
|
||||
wg sync.WaitGroup
|
||||
cb func(message) error
|
||||
id string
|
||||
scatterc chan (message)
|
||||
}
|
||||
|
||||
var upgrader = websocket.Upgrader{}
|
||||
|
||||
func newSession(w http.ResponseWriter, r *http.Request, cb func(message) error) (*session, error) {
|
||||
c, err := upgrader.Upgrade(w, r, nil)
|
||||
ctx, can := context.WithCancel(r.Context())
|
||||
return &session{
|
||||
ctx: ctx,
|
||||
can: can,
|
||||
ws: c,
|
||||
cb: cb,
|
||||
id: uuid.New().String(),
|
||||
scatterc: make(chan message, 20),
|
||||
}, err
|
||||
}
|
||||
|
||||
func (s *session) Close() {
|
||||
if s.ws != nil {
|
||||
s.ws.Close()
|
||||
}
|
||||
s.ws = nil
|
||||
s.can()
|
||||
s.wg.Wait()
|
||||
}
|
||||
|
||||
func (s *session) Run() error {
|
||||
defer s.Close()
|
||||
|
||||
go s.gather()
|
||||
go s.scatter()
|
||||
|
||||
<-s.ctx.Done()
|
||||
return s.ctx.Err()
|
||||
}
|
||||
|
||||
func (s *session) gather() {
|
||||
s.while(func() error {
|
||||
mt, msg, err := s.ws.ReadMessage()
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
if mt != 1 {
|
||||
return nil
|
||||
}
|
||||
|
||||
var m message
|
||||
if err := json.Unmarshal(msg, &m); err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
log.Printf("gathered %+v", m)
|
||||
return s.cb(m)
|
||||
})
|
||||
}
|
||||
|
||||
func (s *session) scatter() {
|
||||
s.while(func() error {
|
||||
select {
|
||||
case m := <-s.scatterc:
|
||||
log.Printf("scattering %+v", m)
|
||||
return s.ws.WriteMessage(1, []byte(m.Text))
|
||||
case <-s.ctx.Done():
|
||||
return s.ctx.Err()
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
func (s *session) while(foo func() error) {
|
||||
defer s.can()
|
||||
|
||||
s.wg.Add(1)
|
||||
defer s.wg.Done()
|
||||
|
||||
l := rate.NewLimiter(20, 1)
|
||||
for l.Wait(s.ctx) == nil {
|
||||
if err := foo(); err != nil {
|
||||
log.Println(err)
|
||||
return
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user