This repository has been archived by the owner on Jan 14, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy pathclient.go
94 lines (82 loc) · 2.48 KB
/
client.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
package obsws
import (
"strconv"
"sync"
"time"
"github.com/gorilla/websocket"
"github.com/mitchellh/mapstructure"
)
const bufferSize = 100
var (
receiveTimeout = time.Duration(0)
messageID = 0
lock = sync.Mutex{}
)
// Client is the interface to obs-websocket.
// Client{Host: "localhost", Port: 4444} will probably work if you haven't configured OBS.
type Client struct {
Host string // Host (probably "localhost").
Port int // Port (OBS default is 4444).
Password string // Password (OBS default is "").
conn *websocket.Conn // Underlying connection to OBS.
receiveTimeout time.Duration // Maximum blocking time for receiving request responses
connected bool // True until Disconnect is called.
handlers map[string]func(e Event) // Event handlers.
respQ chan map[string]interface{} // Queue of received responses.
}
// poll listens for responses/events.
// This function blocks until Disconnect is called.
func (c *Client) poll() {
Logger.Println("started polling")
for c.connected {
m := make(map[string]interface{})
if err := c.conn.ReadJSON(&m); err != nil {
if !c.connected {
return
} else if websocket.IsUnexpectedCloseError(err) {
c.Disconnect()
}
Logger.Println("read from WS:", err)
continue
}
if _, ok := m["message-id"]; ok {
c.handleResponse(m)
} else {
c.handleEvent(m)
}
}
}
// Connected returns wheter or not the client is connected.
func (c Client) Connected() bool {
return c.connected
}
// SetReceiveTimeout sets the maximum blocking time for receiving request responses.
// If set to 0 (the default), there is no timeout.
func SetReceiveTimeout(timeout time.Duration) {
receiveTimeout = timeout
}
// GetMessageID generates a string that the client has not yet used.
func GetMessageID() string {
lock.Lock()
messageID++
id := strconv.Itoa(messageID)
lock.Unlock()
return id
}
// mapToStruct serializes a map into a struct.
func mapToStruct(data map[string]interface{}, dest interface{}) error {
decoder, err := mapstructure.NewDecoder(&mapstructure.DecoderConfig{
TagName: "json",
Result: dest,
})
if err != nil {
Logger.Println("initializing decoder:", err)
return err
}
if err = decoder.Decode(data); err != nil {
Logger.Printf("unmarshalling map -> %T: %v", dest, err)
Logger.Printf("input: %#v\n", data)
return err
}
return nil
}