websockets!
This commit is contained in:
parent
e1be864988
commit
187238c7e4
@ -1,7 +1,8 @@
|
|||||||
package http
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"log"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
@ -24,13 +25,17 @@ func NewGinAdapter(corePort ports.CorePort, cfg config.Config) *GinAdapter {
|
|||||||
var adapter GinAdapter
|
var adapter GinAdapter
|
||||||
|
|
||||||
adapter.corePort = corePort
|
adapter.corePort = corePort
|
||||||
adapter.port = cfg.API[0].Port
|
adapter.port = cfg.API.Port
|
||||||
adapter.cfg = cfg
|
adapter.cfg = cfg
|
||||||
|
|
||||||
//gin.SetMode(gin.ReleaseMode)
|
//gin.SetMode(gin.ReleaseMode)
|
||||||
|
|
||||||
adapter.router = gin.New()
|
adapter.router = gin.New()
|
||||||
|
|
||||||
|
// websockets
|
||||||
|
manager := NewManager()
|
||||||
|
go manager.start()
|
||||||
|
|
||||||
// API
|
// API
|
||||||
adapter.router.GET("/v1/data/channel/:id", adapter.GetChannelByIDHandler)
|
adapter.router.GET("/v1/data/channel/:id", adapter.GetChannelByIDHandler)
|
||||||
adapter.router.GET("/v1/data/channels", adapter.GetChannelsHandler)
|
adapter.router.GET("/v1/data/channels", adapter.GetChannelsHandler)
|
||||||
@ -42,10 +47,34 @@ func NewGinAdapter(corePort ports.CorePort, cfg config.Config) *GinAdapter {
|
|||||||
adapter.router.GET("/v1/config/device", adapter.GetConfigDeviceHandler)
|
adapter.router.GET("/v1/config/device", adapter.GetConfigDeviceHandler)
|
||||||
adapter.router.GET("/v1/config/db", adapter.GetConfigDBHandler)
|
adapter.router.GET("/v1/config/db", adapter.GetConfigDBHandler)
|
||||||
|
|
||||||
|
adapter.router.GET("/ws", func(c *gin.Context) {
|
||||||
|
conn, err := upgrader.Upgrade(c.Writer, c.Request, nil)
|
||||||
|
if err != nil {
|
||||||
|
log.Println(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
client := &Client{
|
||||||
|
socket: conn,
|
||||||
|
send: make(chan []byte),
|
||||||
|
manager: manager,
|
||||||
|
all: false,
|
||||||
|
channels: false,
|
||||||
|
config: false,
|
||||||
|
}
|
||||||
|
|
||||||
|
manager.register <- client
|
||||||
|
|
||||||
|
go client.read()
|
||||||
|
go client.write()
|
||||||
|
})
|
||||||
|
|
||||||
adapter.router.POST("/v1/config/device", adapter.SetConfigDeviceHandler)
|
adapter.router.POST("/v1/config/device", adapter.SetConfigDeviceHandler)
|
||||||
adapter.router.POST("/v1/config/channel/:id", adapter.SetConfigChannelByIdHandler)
|
adapter.router.POST("/v1/config/channel/:id", adapter.SetConfigChannelByIdHandler)
|
||||||
adapter.router.POST("/v1/config/db", adapter.SetConfigDBHandler)
|
adapter.router.POST("/v1/config/db", adapter.SetConfigDBHandler)
|
||||||
|
|
||||||
|
// start service that dumps new data to websocket chan
|
||||||
|
go startService(manager, corePort)
|
||||||
|
|
||||||
// TH7 demo code. html file and javascript
|
// TH7 demo code. html file and javascript
|
||||||
adapter.router.GET("/", adapter.IndexHandler)
|
adapter.router.GET("/", adapter.IndexHandler)
|
||||||
adapter.router.Static("/assets", "./static")
|
adapter.router.Static("/assets", "./static")
|
@ -1,4 +1,4 @@
|
|||||||
package http
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"errors"
|
"errors"
|
||||||
@ -192,7 +192,7 @@ func (g *GinAdapter) GetConfigDeviceHandler(c *gin.Context) {
|
|||||||
func (g *GinAdapter) GetConfigDBHandler(c *gin.Context) {
|
func (g *GinAdapter) GetConfigDBHandler(c *gin.Context) {
|
||||||
var timestamp string
|
var timestamp string
|
||||||
|
|
||||||
if g.cfg.API[0].Restricted {
|
if g.cfg.API.Restricted {
|
||||||
c.String(http.StatusForbidden, ErrConfigRestrictedDB.Error())
|
c.String(http.StatusForbidden, ErrConfigRestrictedDB.Error())
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@ -275,7 +275,7 @@ func (g *GinAdapter) SetConfigDBHandler(c *gin.Context) {
|
|||||||
|
|
||||||
var req PostRequest
|
var req PostRequest
|
||||||
|
|
||||||
if g.cfg.API[0].Restricted {
|
if g.cfg.API.Restricted {
|
||||||
c.String(http.StatusForbidden, ErrConfigRestrictedDB.Error())
|
c.String(http.StatusForbidden, ErrConfigRestrictedDB.Error())
|
||||||
return
|
return
|
||||||
}
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package http
|
package api
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
41
api/service.go
Normal file
41
api/service.go
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"log"
|
||||||
|
"th7/data/core"
|
||||||
|
"th7/ports"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"encoding/json"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
CheckChannelDur = time.Millisecond * 700
|
||||||
|
StartupSleepDur = time.Second * 5
|
||||||
|
)
|
||||||
|
|
||||||
|
type ChannelsWrapper struct {
|
||||||
|
Channels []core.Channel `json:"channels"`
|
||||||
|
}
|
||||||
|
|
||||||
|
func startService(man *Manager, core ports.CorePort) {
|
||||||
|
|
||||||
|
var chwrap ChannelsWrapper
|
||||||
|
|
||||||
|
time.Sleep(StartupSleepDur)
|
||||||
|
|
||||||
|
for {
|
||||||
|
chwrap.Channels = core.GetChannels()
|
||||||
|
marshaled, err := json.Marshal(chwrap)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("Error marshaling struct:", chwrap.Channels)
|
||||||
|
}
|
||||||
|
|
||||||
|
man.broadcast <- Message{
|
||||||
|
messageType: CHANNELS,
|
||||||
|
data: []byte(marshaled),
|
||||||
|
}
|
||||||
|
time.Sleep(CheckChannelDur)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
package http
|
package api
|
||||||
|
|
||||||
import "time"
|
import "time"
|
||||||
|
|
177
api/websocket.go
Normal file
177
api/websocket.go
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
package api
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/gorilla/websocket"
|
||||||
|
)
|
||||||
|
|
||||||
|
var upgrader = websocket.Upgrader{
|
||||||
|
ReadBufferSize: 1024,
|
||||||
|
WriteBufferSize: 1024,
|
||||||
|
CheckOrigin: func(r *http.Request) bool {
|
||||||
|
return true
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
type MessageType = int
|
||||||
|
|
||||||
|
const (
|
||||||
|
ALL MessageType = iota
|
||||||
|
CHANNELS
|
||||||
|
CONFIG
|
||||||
|
)
|
||||||
|
|
||||||
|
type Message struct {
|
||||||
|
messageType MessageType
|
||||||
|
data []byte
|
||||||
|
}
|
||||||
|
|
||||||
|
type Client struct {
|
||||||
|
socket *websocket.Conn
|
||||||
|
send chan []byte
|
||||||
|
manager *Manager
|
||||||
|
channels bool
|
||||||
|
config bool
|
||||||
|
all bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type Manager struct {
|
||||||
|
clients map[*Client]bool
|
||||||
|
broadcast chan Message
|
||||||
|
register chan *Client
|
||||||
|
unregister chan *Client
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewManager() *Manager {
|
||||||
|
return &Manager{
|
||||||
|
clients: make(map[*Client]bool),
|
||||||
|
broadcast: make(chan Message),
|
||||||
|
register: make(chan *Client),
|
||||||
|
unregister: make(chan *Client),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (man *Manager) BroadcastAll(msg []byte) {
|
||||||
|
for client := range man.clients {
|
||||||
|
select {
|
||||||
|
case client.send <- msg:
|
||||||
|
default:
|
||||||
|
man.unregister <- client
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (man *Manager) BroadcastChannels(msg []byte) {
|
||||||
|
for client := range man.clients {
|
||||||
|
if !client.all && !client.channels {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case client.send <- msg:
|
||||||
|
default:
|
||||||
|
man.unregister <- client
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (man *Manager) BroadcastConfig(msg []byte) {
|
||||||
|
for client := range man.clients {
|
||||||
|
if !client.all && !client.config {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case client.send <- msg:
|
||||||
|
default:
|
||||||
|
man.unregister <- client
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (man *Manager) start() {
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
// new connection
|
||||||
|
case conn := <-man.register:
|
||||||
|
man.clients[conn] = true
|
||||||
|
|
||||||
|
// connection closed
|
||||||
|
case conn := <-man.unregister:
|
||||||
|
if _, ok := man.clients[conn]; ok {
|
||||||
|
fmt.Println("Unregistering:", conn)
|
||||||
|
close(conn.send)
|
||||||
|
delete(man.clients, conn)
|
||||||
|
}
|
||||||
|
|
||||||
|
// broadcast a message to all connected clients
|
||||||
|
case message := <-man.broadcast:
|
||||||
|
switch message.messageType {
|
||||||
|
case ALL:
|
||||||
|
man.BroadcastAll(message.data)
|
||||||
|
case CHANNELS:
|
||||||
|
man.BroadcastChannels(message.data)
|
||||||
|
case CONFIG:
|
||||||
|
man.BroadcastConfig(message.data)
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) read() {
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
c.manager.unregister <- c
|
||||||
|
_ = c.socket.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
for {
|
||||||
|
_, message, err := c.socket.ReadMessage()
|
||||||
|
if err != nil {
|
||||||
|
c.manager.unregister <- c
|
||||||
|
_ = c.socket.Close()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
|
||||||
|
switch strings.ToLower(string(message)) {
|
||||||
|
case "all":
|
||||||
|
c.all = true
|
||||||
|
case "channels":
|
||||||
|
c.channels = true
|
||||||
|
case "config":
|
||||||
|
c.config = true
|
||||||
|
default:
|
||||||
|
log.Println("received: ", message)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (c *Client) write() {
|
||||||
|
|
||||||
|
defer func() {
|
||||||
|
_ = c.socket.Close()
|
||||||
|
}()
|
||||||
|
|
||||||
|
for {
|
||||||
|
select {
|
||||||
|
case message, ok := <-c.send:
|
||||||
|
if !ok {
|
||||||
|
_ = c.socket.WriteMessage(websocket.CloseMessage, []byte{})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
err := c.socket.WriteMessage(websocket.TextMessage, message)
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println(err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
@ -59,14 +59,11 @@ func Load() (config.Config, error) {
|
|||||||
cfg.Board.NoWeb = v.GetBool("TH7.noweb")
|
cfg.Board.NoWeb = v.GetBool("TH7.noweb")
|
||||||
|
|
||||||
cfg.Channels = make([]config.Channel, 0)
|
cfg.Channels = make([]config.Channel, 0)
|
||||||
cfg.API = make([]config.API, 0)
|
cfg.API = config.API{
|
||||||
|
|
||||||
// just REST API for now ..
|
|
||||||
cfg.API = append(cfg.API, config.API{
|
|
||||||
Port: v.GetInt("API.HTTP.port"),
|
Port: v.GetInt("API.HTTP.port"),
|
||||||
Enabled: v.GetBool("API.HTTP.enabled"),
|
Enabled: v.GetBool("API.HTTP.enabled"),
|
||||||
Restricted: v.GetBool("API.HTTP.restricted"),
|
Restricted: v.GetBool("API.HTTP.restricted"),
|
||||||
})
|
}
|
||||||
|
|
||||||
for i := 1; i < 8; i++ {
|
for i := 1; i < 8; i++ {
|
||||||
var c config.Channel
|
var c config.Channel
|
||||||
|
@ -35,5 +35,5 @@ type Config struct {
|
|||||||
Board TH7 `json:"TH7"`
|
Board TH7 `json:"TH7"`
|
||||||
Channels []Channel `json:"channels"`
|
Channels []Channel `json:"channels"`
|
||||||
DB map[string]interface{} `json:"db"`
|
DB map[string]interface{} `json:"db"`
|
||||||
API []API `json:"API"`
|
API API `json:"API"`
|
||||||
}
|
}
|
||||||
|
1
go.mod
1
go.mod
@ -6,6 +6,7 @@ require (
|
|||||||
github.com/InfluxCommunity/influxdb3-go v0.4.0
|
github.com/InfluxCommunity/influxdb3-go v0.4.0
|
||||||
github.com/fatih/color v1.16.0
|
github.com/fatih/color v1.16.0
|
||||||
github.com/gin-gonic/gin v1.8.1
|
github.com/gin-gonic/gin v1.8.1
|
||||||
|
github.com/gorilla/websocket v1.5.1
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
|
2
go.sum
2
go.sum
@ -148,6 +148,8 @@ github.com/google/uuid v1.3.1 h1:KjJaJ9iWZ3jOFZIf1Lqf4laDRCasjl0BCmnEGxkdLb4=
|
|||||||
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg=
|
||||||
github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
|
github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk=
|
||||||
github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g=
|
github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g=
|
||||||
|
github.com/gorilla/websocket v1.5.1 h1:gmztn0JnHVt9JZquRuzLw3g4wouNVzKL15iLr/zn/QY=
|
||||||
|
github.com/gorilla/websocket v1.5.1/go.mod h1:x3kM2JMyaluk02fnUJpQuwD2dCS5NDG2ZHL0uE0tcaY=
|
||||||
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8=
|
||||||
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
github.com/hashicorp/hcl v1.0.0 h1:0Anlzjpi4vEasTeNFn2mLJgTSwt0+6sfsiTG8qcWGx4=
|
||||||
|
8
main.go
8
main.go
@ -6,7 +6,7 @@ import (
|
|||||||
"os"
|
"os"
|
||||||
"os/signal"
|
"os/signal"
|
||||||
"syscall"
|
"syscall"
|
||||||
"th7/api/http"
|
"th7/api"
|
||||||
"th7/config"
|
"th7/config"
|
||||||
"th7/core"
|
"th7/core"
|
||||||
"th7/db"
|
"th7/db"
|
||||||
@ -59,12 +59,12 @@ func main() {
|
|||||||
color.Unset()
|
color.Unset()
|
||||||
|
|
||||||
// if noweb is false and HTTP/REST API is enabled then start web service
|
// if noweb is false and HTTP/REST API is enabled then start web service
|
||||||
if !cfg.Board.NoWeb && cfg.API[0].Enabled {
|
if !cfg.Board.NoWeb && cfg.API.Enabled {
|
||||||
apiPort = http.NewGinAdapter(corePort, cfg)
|
apiPort = api.NewGinAdapter(corePort, cfg)
|
||||||
go apiPort.Run()
|
go apiPort.Run()
|
||||||
|
|
||||||
color.Set(color.FgHiGreen)
|
color.Set(color.FgHiGreen)
|
||||||
fmt.Printf("TH7 API is live on http://localhost:%d/\n", cfg.API[0].Port)
|
fmt.Printf("TH7 API is live on http://localhost:%d/\n", cfg.API.Port)
|
||||||
color.Unset()
|
color.Unset()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,3 +89,18 @@ updateConfig();
|
|||||||
|
|
||||||
setInterval(updateRead, 1000);
|
setInterval(updateRead, 1000);
|
||||||
setInterval(updateConfig, 10000);
|
setInterval(updateConfig, 10000);
|
||||||
|
|
||||||
|
// websocket demo
|
||||||
|
const ws = new WebSocket("ws://" + window.location.host + "/ws");
|
||||||
|
ws.onopen = (e) => {
|
||||||
|
console.log("Connection open");
|
||||||
|
ws.send("ALL");
|
||||||
|
};
|
||||||
|
|
||||||
|
ws.onmessage = (e) => {
|
||||||
|
console.log("msg recv: " + e.data);
|
||||||
|
};
|
||||||
|
|
||||||
|
ws.onclose = (e) => {
|
||||||
|
console.log("Connection close");
|
||||||
|
};
|
||||||
|
Loading…
Reference in New Issue
Block a user