mirror of
https://github.com/AlexandreRouma/wiscast.git
synced 2026-04-20 08:22:42 +00:00
progress
This commit is contained in:
171
display.go
171
display.go
@@ -1,49 +1,122 @@
|
||||
package main
|
||||
|
||||
// Packages
|
||||
import "github.com/gorilla/websocket"
|
||||
import "errors"
|
||||
import "log"
|
||||
import "net/http"
|
||||
import "sync"
|
||||
import "time"
|
||||
import "github.com/gorilla/websocket"
|
||||
|
||||
// Display instance
|
||||
type Display struct {
|
||||
// WebSocket used to communicate with the display
|
||||
sock *websocket.Conn
|
||||
sockSendMtx sync.Mutex
|
||||
|
||||
// User currently connected to the display
|
||||
user *User
|
||||
|
||||
// One-time-pass currently shown on the display
|
||||
otpMtx sync.Mutex
|
||||
otp string
|
||||
|
||||
// Channel to pass the answer from the display coroutine to the user couroutine
|
||||
answerCh chan string
|
||||
}
|
||||
|
||||
// Helper function to flush channels
|
||||
func chFlush(ch *chan string) {
|
||||
for {
|
||||
empty := false
|
||||
select {
|
||||
// If data is available, read it and try again
|
||||
case <-*ch:
|
||||
continue
|
||||
|
||||
// If no data is available, stop reading
|
||||
default:
|
||||
empty = true
|
||||
}
|
||||
if empty { break }
|
||||
}
|
||||
}
|
||||
|
||||
// Helper function to read from a channel with a timeout
|
||||
func chReadTimeout(ch *chan string, timeoutMS int) (string, error) {
|
||||
select {
|
||||
// If data is available, return it with no error
|
||||
case data := <-*ch:
|
||||
return data, nil
|
||||
|
||||
// If no data has been received and the timeout is reached, return an error
|
||||
case <-time.After(timeoutMS * time.Millisecond):
|
||||
return "", errors.New("timeout")
|
||||
}
|
||||
}
|
||||
|
||||
// List of all connected displays
|
||||
var displaysLck sync.Mutex
|
||||
var displays map[string]*Display
|
||||
var displays = map[string]*Display{}
|
||||
|
||||
// Get the display back to its idle state
|
||||
func (this *Display) reset() {
|
||||
// Acquire the sending mutex
|
||||
this.sockSendMtx.Lock()
|
||||
|
||||
// Send a reset command
|
||||
this.sock.WriteMessage(websocket.TextMessage, encodeMessage(Message{
|
||||
mtype: "reset",
|
||||
}))
|
||||
|
||||
// Release the sending mutex
|
||||
this.sockSendMtx.Unlock()
|
||||
}
|
||||
|
||||
// Switch the display to streaming mode
|
||||
func (this *Display) stream() {
|
||||
// Acquire the sending mutex
|
||||
this.sockSendMtx.Lock()
|
||||
|
||||
// Send a show-pin command
|
||||
this.sock.WriteMessage(websocket.TextMessage, encodeMessage(Message{
|
||||
mtype: "stream",
|
||||
}))
|
||||
|
||||
// Release the sending mutex
|
||||
this.sockSendMtx.Unlock()
|
||||
}
|
||||
|
||||
// Send a WebRTC offer to the display and get an answer
|
||||
func (this *Display) webRTCOffer(offer string, timeoutMS int) string {
|
||||
// TODO
|
||||
return ""
|
||||
func (this *Display) webRTCOffer(offer string, timeoutMS int) (string, error) {
|
||||
// Flush the answer channel
|
||||
chFlush(&this.answerCh)
|
||||
|
||||
// Acquire the sending mutex
|
||||
this.sockSendMtx.Lock()
|
||||
|
||||
// Send the offer
|
||||
this.sock.WriteMessage(websocket.TextMessage, encodeMessage(Message{
|
||||
mtype: "webrtc-offer",
|
||||
arguments: map[string]interface{}{
|
||||
"offer": offer,
|
||||
},
|
||||
}))
|
||||
|
||||
// Release the sending mutex
|
||||
this.sockSendMtx.Unlock()
|
||||
|
||||
// TODO: Close the connection if the display failed to respond?
|
||||
|
||||
// Receive the answer
|
||||
return chReadTimeout(&this.answerCh, CONF_TIMEOUT_MS)
|
||||
}
|
||||
|
||||
// Send an ICE candiate to the display
|
||||
func (this *Display) iceCandidate(candidate string) {
|
||||
// Acquire the sending mutex
|
||||
this.sockSendMtx.Lock()
|
||||
|
||||
// Send the candidate
|
||||
sendMessage(this.sock, Message{
|
||||
mtype: "ice-candidate",
|
||||
@@ -51,9 +124,95 @@ func (this *Display) iceCandidate(candidate string) {
|
||||
"candidate": candidate,
|
||||
},
|
||||
})
|
||||
|
||||
// Release the sending mutex
|
||||
this.sockSendMtx.Unlock()
|
||||
}
|
||||
|
||||
// Connection handler for displays
|
||||
func displayHandler(sock *websocket.Conn, dispID string) {
|
||||
func displayHandler(sock *websocket.Conn, dispID string, otp string) {
|
||||
// Create the display object
|
||||
disp := Display{ sock: sock, otp: otp }
|
||||
|
||||
// Acquire the sending mutex
|
||||
disp.sockSendMtx.Lock()
|
||||
|
||||
// Acquire the display list
|
||||
displaysLck.Lock()
|
||||
|
||||
// Check that a display with that ID doesn't already exist
|
||||
if displays[dispID] != nil {
|
||||
// Release the display list
|
||||
displaysLck.Unlock()
|
||||
|
||||
// Send back an error
|
||||
sendErrorMessage(sock, http.StatusConflict)
|
||||
|
||||
// Release the sending mutex
|
||||
disp.sockSendMtx.Unlock()
|
||||
}
|
||||
|
||||
// Add the display to the list
|
||||
displays[dispID] = &disp
|
||||
|
||||
// Release the display list
|
||||
displaysLck.Unlock()
|
||||
|
||||
// Send back the config for the display to use
|
||||
sendMessage(sock, Message{
|
||||
mtype: "config",
|
||||
arguments: map[string]interface{}{
|
||||
"timeout": CONF_TIMEOUT_MS,
|
||||
"iceServers": CONF_ICE_SERVERS,
|
||||
},
|
||||
})
|
||||
|
||||
// Release the sending mutex
|
||||
disp.sockSendMtx.Unlock()
|
||||
|
||||
// Log the new display
|
||||
log.Println("New display: ID='" + dispID + "', OTP='" + otp + "'")
|
||||
|
||||
// Message loop
|
||||
for {
|
||||
// Receive a message
|
||||
msg, err := recvMessage(sock, 0)
|
||||
|
||||
// Give up on the connection if there was an error
|
||||
if (err != nil) { break }
|
||||
|
||||
// Handle the message depending on its type
|
||||
switch msg.mtype {
|
||||
case "otp":
|
||||
// Check that the message contains an OTP
|
||||
otp, valid := msg.arguments["otp"].(string)
|
||||
if (!valid) { break }
|
||||
|
||||
// Acquire the display's OTP
|
||||
disp.otpMtx.Lock()
|
||||
|
||||
// Update the OTP
|
||||
disp.otp = otp
|
||||
|
||||
// Release the display's OTP
|
||||
disp.otpMtx.Unlock()
|
||||
|
||||
case "answer":
|
||||
// Check that the message contains an answer
|
||||
answer, valid := msg.arguments["answer"].(string)
|
||||
if (!valid) { break }
|
||||
|
||||
// Send the answer through the display's answer channel
|
||||
disp.answerCh <- answer
|
||||
|
||||
case "ice-candidate":
|
||||
// TODO
|
||||
|
||||
default:
|
||||
// Give up
|
||||
break
|
||||
}
|
||||
}
|
||||
|
||||
// TODO: Gracefull disconnect the connected user if there is one
|
||||
}
|
||||
Reference in New Issue
Block a user