-
Notifications
You must be signed in to change notification settings - Fork 53
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Added restfull apis of mqtt and cluster.
- Loading branch information
Showing
16 changed files
with
531 additions
and
119 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
package rest | ||
|
||
type result struct { | ||
Url string `json:"url"` | ||
Data string `json:"data"` | ||
Err string `json:"err"` | ||
} | ||
|
||
type node struct { | ||
Name string `json:"name"` | ||
Addr string `json:"addr"` | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,85 @@ | ||
package rest | ||
|
||
import ( | ||
"bytes" | ||
"context" | ||
"errors" | ||
"io" | ||
"net/http" | ||
"sync" | ||
"time" | ||
) | ||
|
||
const ( | ||
HttpGet = "GET" | ||
HttpPost = "POST" | ||
HttpDelete = "DELETE" | ||
Timeout = 3 * time.Second | ||
) | ||
|
||
func fetch(ctx context.Context, method string, url string, data []byte) result { | ||
rs := result{Url: url} | ||
var req *http.Request | ||
var err error | ||
var body io.Reader | ||
if data != nil { | ||
body = bytes.NewBuffer(data) | ||
} | ||
if ctx != nil { | ||
req, err = http.NewRequestWithContext(ctx, method, url, body) | ||
} else { | ||
req, err = http.NewRequest(method, url, body) | ||
} | ||
if err != nil { | ||
rs.Err = err.Error() | ||
return rs | ||
} | ||
|
||
req.Header.Set("Content-Type", "application/json") | ||
resp, err := http.DefaultClient.Do(req) | ||
if err != nil { | ||
if ctxErr := ctx.Err(); errors.Is(ctxErr, context.DeadlineExceeded) { | ||
rs.Err = "Request timeout" | ||
} else { | ||
rs.Err = err.Error() | ||
} | ||
return rs | ||
} | ||
defer resp.Body.Close() | ||
if data, err := io.ReadAll(resp.Body); err != nil { | ||
rs.Err = err.Error() | ||
} else { | ||
if resp.StatusCode == http.StatusOK { | ||
rs.Data = string(data) | ||
} else { | ||
rs.Err = string(data) | ||
} | ||
} | ||
return rs | ||
} | ||
|
||
func fetchM(method string, urls []string, body []byte) []result { | ||
ctx, cancel := context.WithTimeout(context.Background(), Timeout) | ||
defer cancel() | ||
|
||
var wg sync.WaitGroup | ||
ch := make(chan result, len(urls)) | ||
|
||
wg.Add(len(urls)) | ||
for _, url := range urls { | ||
go func(url string) { | ||
defer wg.Done() | ||
ch <- fetch(ctx, method, url, body) | ||
}(url) | ||
} | ||
|
||
wg.Wait() | ||
close(ch) | ||
results := make([]result, len(urls)) | ||
index := 0 | ||
for rs := range ch { | ||
results[index] = rs | ||
index++ | ||
} | ||
return results | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,177 @@ | ||
package rest | ||
|
||
import ( | ||
"encoding/json" | ||
"fmt" | ||
cs "github.com/wind-c/comqtt/v2/cluster" | ||
"github.com/wind-c/comqtt/v2/cluster/discovery" | ||
rt "github.com/wind-c/comqtt/v2/mqtt/rest" | ||
"net/http" | ||
"net/netip" | ||
"strings" | ||
) | ||
|
||
type rest struct { | ||
agent *cs.Agent | ||
} | ||
|
||
func New(agent *cs.Agent) *rest { | ||
return &rest{ | ||
agent: agent, | ||
} | ||
} | ||
|
||
func (s *rest) GenHandlers() map[string]rt.Handler { | ||
return map[string]rt.Handler{ | ||
"GET /api/v1/node/config": s.viewConfig, | ||
"DELETE /api/v1/node/{name}": s.leave, | ||
"GET /api/v1/cluster/nodes": s.getNodes, | ||
"POST /api/v1/cluster/nodes": s.join, | ||
"POST /api/v1/cluster/peers": s.addRaftPeer, | ||
"DELETE /api/v1/cluster/peers/{name}": s.removeRaftPeer, | ||
"GET /api/v1/cluster/stat/online": s.getOnlineCount, | ||
"GET /api/v1/cluster/clients/{id}": s.getClient, | ||
"POST /api/v1/cluster/blacklist/{id}": s.kickClient, | ||
"DELETE /api/v1/cluster/blacklist/{id}": s.blanchClient, | ||
} | ||
} | ||
|
||
// viewConfig return the configuration parameters of this node | ||
// GET api/v1/node/config | ||
func (s *rest) viewConfig(w http.ResponseWriter, r *http.Request) { | ||
rt.Ok(w, s.agent.Config) | ||
} | ||
|
||
// getMembers return all nodes in the cluster | ||
// GET api/v1/cluster/nodes | ||
func (s *rest) getNodes(w http.ResponseWriter, r *http.Request) { | ||
rt.Ok(w, s.agent.GetMemberList()) | ||
} | ||
|
||
// join add a node to the cluster | ||
// POST api/v1/cluster/nodes | ||
func (s *rest) join(w http.ResponseWriter, r *http.Request) { | ||
var n node | ||
if err := json.NewDecoder(r.Body).Decode(&n); err != nil { | ||
rt.Error(w, http.StatusBadRequest, err.Error()) | ||
return | ||
} | ||
n.Name = strings.TrimSpace(n.Name) | ||
n.Addr = strings.TrimSpace(n.Addr) | ||
if n.Name == "" || n.Addr == "" { | ||
rt.Error(w, http.StatusBadRequest, "name and addr cannot be empty") | ||
return | ||
} | ||
if _, err := netip.ParseAddrPort(n.Addr); err != nil { | ||
rt.Error(w, http.StatusBadRequest, "invalid address") | ||
return | ||
} | ||
|
||
if err := s.agent.Join(n.Name, n.Addr); err != nil { | ||
rt.Error(w, http.StatusInternalServerError, err.Error()) | ||
} else { | ||
rt.Ok(w, n) | ||
} | ||
} | ||
|
||
// leave local node gracefully exits the cluster | ||
// DELETE api/v1/node/{name} | ||
func (s *rest) leave(w http.ResponseWriter, r *http.Request) { | ||
name := r.PathValue("name") | ||
localName := s.agent.GetLocalName() | ||
if name != localName { | ||
rt.Error(w, http.StatusBadRequest, fmt.Sprintf("cannot remove not local node %s", localName)) | ||
return | ||
} | ||
|
||
if err := s.agent.Leave(); err != nil { | ||
rt.Error(w, http.StatusInternalServerError, err.Error()) | ||
return | ||
} else { | ||
rt.Ok(w, name) | ||
} | ||
} | ||
|
||
// addRaftPeer add peer to raft cluster | ||
// POST api/v1/cluster/peers | ||
func (s *rest) addRaftPeer(w http.ResponseWriter, r *http.Request) { | ||
var p node | ||
if err := json.NewDecoder(r.Body).Decode(&p); err != nil { | ||
rt.Error(w, http.StatusBadRequest, err.Error()) | ||
return | ||
} | ||
p.Name = strings.TrimSpace(p.Name) | ||
p.Addr = strings.TrimSpace(p.Addr) | ||
if p.Name == "" || p.Addr == "" { | ||
rt.Error(w, http.StatusBadRequest, "name and addr cannot be empty") | ||
return | ||
} | ||
if _, err := netip.ParseAddrPort(p.Addr); err != nil { | ||
rt.Error(w, http.StatusBadRequest, "invalid address") | ||
return | ||
} | ||
|
||
s.agent.AddRaftPeer(p.Name, p.Addr) | ||
rt.Ok(w, p) | ||
} | ||
|
||
// removeRaftPeer remove peer from raft cluster | ||
// DELETE api/v1/cluster/peers/{name} | ||
func (s *rest) removeRaftPeer(w http.ResponseWriter, r *http.Request) { | ||
name := r.PathValue("name") | ||
if strings.TrimSpace(name) == "" { | ||
rt.Error(w, http.StatusBadRequest, "name cannot be empty") | ||
return | ||
} | ||
|
||
s.agent.RemoveRaftPeer(name) | ||
rt.Ok(w, name) | ||
} | ||
|
||
// getOnlineCount return online number from all nodes in the cluster | ||
// GET api/v1/cluster/stat/online | ||
func (s *rest) getOnlineCount(w http.ResponseWriter, r *http.Request) { | ||
path := rt.MqttGetOnlinePath | ||
urls := genUrls(s.agent.GetMemberList(), path) | ||
rs := fetchM(HttpGet, urls, nil) | ||
rt.Ok(w, rs) | ||
} | ||
|
||
// getClient return a client information, search from all nodes in the cluster | ||
// GET api/v1/cluster/clients/{id} | ||
func (s *rest) getClient(w http.ResponseWriter, r *http.Request) { | ||
cid := r.PathValue("id") | ||
path := strings.Replace(rt.MqttGetClientPath, "{id}", cid, 1) | ||
urls := genUrls(s.agent.GetMemberList(), path) | ||
rs := fetchM(HttpGet, urls, nil) | ||
rt.Ok(w, rs) | ||
} | ||
|
||
// kickClient add it to the blacklist on all nodes in the cluster | ||
// POST api/v1/cluster/blacklist/{id} | ||
func (s *rest) kickClient(w http.ResponseWriter, r *http.Request) { | ||
cid := r.PathValue("id") | ||
path := strings.Replace(rt.MqttAddBlacklistPath, "{id}", cid, 1) | ||
urls := genUrls(s.agent.GetMemberList(), path) | ||
rs := fetchM(HttpPost, urls, nil) | ||
rt.Ok(w, rs) | ||
} | ||
|
||
// blanchClient remove from the blacklist on all nodes in the cluster | ||
// DELETE api/v1/cluster/blacklist/{id} | ||
func (s *rest) blanchClient(w http.ResponseWriter, r *http.Request) { | ||
cid := r.PathValue("id") | ||
path := strings.Replace(rt.MqttDelBlacklistPath, "{id}", cid, 1) | ||
urls := genUrls(s.agent.GetMemberList(), path) | ||
rs := fetchM(HttpDelete, urls, nil) | ||
rt.Ok(w, rs) | ||
} | ||
|
||
// genUrls generate urls | ||
func genUrls(ms []discovery.Member, path string) []string { | ||
urls := make([]string, len(ms)) | ||
for i, m := range ms { | ||
urls[i] = "http://" + m.Addr + ":8080" + path | ||
} | ||
return urls | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.