1 Star 0 Fork 64

Bill_Ho/ATX

forked from King/ATX 
加入 Gitee
与超过 1200万 开发者一起发现、参与优秀开源项目,私有仓库也完全免费 :)
免费加入
文件
克隆/下载
httpserver.go 14.04 KB
一键复制 编辑 原始数据 按行查看 历史
King 提交于 2019-02-14 17:04 . 安卓设备集群管理平台
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537
package main
import (
"bytes"
"encoding/json"
"errors"
"fmt"
"html/template"
"io"
"io/ioutil"
"net/http"
"net/http/httputil"
"net/url"
"os"
"strconv"
"strings"
"time"
"github.com/codeskyblue/heartbeat"
"github.com/codeskyblue/realip"
"github.com/gorilla/mux"
"github.com/gorilla/websocket"
"github.com/koding/websocketproxy"
hb2 "github.com/openatx/atx-server/heartbeat"
"github.com/openatx/atx-server/proto"
"github.com/qiniu/log"
)
var (
upgrader = websocket.Upgrader{
CheckOrigin: func(r *http.Request) bool {
return true
},
}
// Time allowed to write message to the client
wsWriteWait = 10 * time.Second
// Send pings to client with this period. Must be less than pongWait.
wsPingPeriod = 10 * time.Second
// Time allowed to read the next pong message from client
wsPongWait = wsPingPeriod * 3
funcMap template.FuncMap
)
func init() {
funcMap = template.FuncMap{
"title": strings.Title,
"urlhash": func(s string) string {
path := strings.TrimPrefix(s, "/")
info, err := os.Stat(path)
if err != nil {
return s + "#no-such-file"
}
return fmt.Sprintf("%s?t=%d", s, info.ModTime().Unix())
},
}
}
func renderHTML(w http.ResponseWriter, filename string, value interface{}) {
tmpl := template.Must(template.New("").Funcs(funcMap).Delims("[[", "]]").ParseGlob("templates/*.html"))
tmpl.ExecuteTemplate(w, filename, value)
// content, _ := ioutil.ReadFile("templates/" + filename)
// template.Must(template.New(filename).Parse(string(content))).Execute(w, nil)
}
func renderJSON(w http.ResponseWriter, data interface{}) {
js, err := json.Marshal(data)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
w.Header().Set("Content-Type", "application/json; charset=UTF-8")
w.Header().Set("Content-Length", fmt.Sprintf("%d", len(js)))
w.Write(js)
}
func newHandler() http.Handler {
r := mux.NewRouter()
r.HandleFunc("/version", func(w http.ResponseWriter, r *http.Request) {
renderJSON(w, map[string]string{
"server": version,
"atx-agent": atxAgentVersion,
})
})
// 设备远程控制
r.HandleFunc("/devices/ip:{ip}/remote", func(w http.ResponseWriter, r *http.Request) {
ip := mux.Vars(r)["ip"]
renderHTML(w, "remote.html", ip)
}).Methods("GET")
r.HandleFunc("/devices/{udid}/remote", func(w http.ResponseWriter, r *http.Request) {
udid := mux.Vars(r)["udid"]
info, err := db.DeviceGet(udid)
if err != nil {
http.Error(w, err.Error(), 404)
return
}
renderHTML(w, "remote.html", map[string]interface{}{
"IP": info.IP,
"Port": info.Port,
"Udid": udid})
}).Methods("GET")
// 设备信息修改
r.HandleFunc("/devices/{udid}/edit", func(w http.ResponseWriter, r *http.Request) {
udid := mux.Vars(r)["udid"]
renderHTML(w, "edit.html", udid)
}).Methods("GET")
// Video-backend starts
videoProxyURL, _ := url.Parse(*videoBackend)
wsProxyURL, _ := url.Parse(*videoBackend)
wsProxyURL.Scheme = "ws"
videoProxy := httputil.NewSingleHostReverseProxy(videoProxyURL)
wsVideoProxy := websocketproxy.NewProxy(wsProxyURL)
r.PathPrefix("/videos").Handler(videoProxy).Methods("GET", "DELETE")
r.Handle("/video/images2video", videoProxy) // not working with POST proxy
r.PathPrefix("/static/videos/").Handler(videoProxy)
r.Handle("/video/convert", wsVideoProxy)
// End of video-backend
r.HandleFunc("/products/{brand}/{model}", func(w http.ResponseWriter, r *http.Request) {
vars := mux.Vars(r)
brand, model := vars["brand"], vars["model"]
products, err := db.ProductsFindAll(brand, model)
if err != nil {
http.Error(w, err.Error(), 500)
return
}
renderJSON(w, products)
})
r.HandleFunc("/devices/{udid}/product", func(w http.ResponseWriter, r *http.Request) {
var product proto.Product
err := json.NewDecoder(r.Body).Decode(&product)
if err != nil {
http.Error(w, err.Error(), http.StatusForbidden)
return
}
if product.Id == "" {
http.Error(w, "product id is required", http.StatusForbidden)
return
}
if err := db.ProductUpdate(product.Id, product); err != nil {
http.Error(w, err.Error(), http.StatusForbidden)
return
}
err = db.DeviceUpdate(mux.Vars(r)["udid"], proto.DeviceInfo{
Product: &proto.Product{
Id: product.Id,
},
})
if err != nil {
http.Error(w, err.Error(), http.StatusForbidden)
return
}
renderJSON(w, map[string]interface{}{
"success": true,
})
}).Methods("PUT")
r.HandleFunc("/echo", echo)
r.HandleFunc("/feeds", func(w http.ResponseWriter, r *http.Request) {
ws, err := upgrader.Upgrade(w, r, nil)
if err != nil {
http.Error(w, err.Error(), 500)
return
}
defer ws.Close()
feeds, cancel, err := db.WatchDeviceChanges()
if err != nil {
ws.WriteJSON(map[string]string{
"error": err.Error(),
})
return
}
go func() {
defer cancel()
for {
_, _, err := ws.ReadMessage()
if err != nil {
break
}
}
log.Debug("ws read closed")
}()
for change := range feeds {
buf := bytes.NewBuffer(nil)
json.NewEncoder(buf).Encode(map[string]interface{}{
"new": change.NewValue,
"old": change.OldValue,
})
err := ws.WriteMessage(websocket.TextMessage, buf.Bytes()) // []byte(`{"new": "haha", "old": "wowo"}`))
if err != nil {
break
}
}
log.Debug("ws write closed")
})
r.HandleFunc("/providers", func(w http.ResponseWriter, r *http.Request) {
values := r.URL.Query()
if _, ok := values["json"]; ok {
providers, err := db.ProvidersAll()
if err != nil {
renderJSON(w, map[string]interface{}{
"success": false,
"description": err.Error(),
})
return
}
renderJSON(w, providers)
return
}
renderHTML(w, "providers.html", nil)
})
r.HandleFunc("/providers/{id}", func(w http.ResponseWriter, r *http.Request) {
var p proto.Provider
data, _ := ioutil.ReadAll(r.Body)
if err := json.Unmarshal(data, &p); err != nil {
renderJSON(w, map[string]interface{}{
"success": false,
"description": err.Error(),
})
return
}
id := mux.Vars(r)["id"]
db.ProviderUpdate(id, p)
renderJSON(w, map[string]interface{}{
"success": true,
})
}).Methods("PUT")
r.HandleFunc("/api/v1/batch/unlock", func(w http.ResponseWriter, r *http.Request) {
batchRunCommand("am start -W --user 0 -a com.github.uiautomator.ACTION_IDENTIFY; input keyevent HOME")
io.WriteString(w, "Success")
})
r.HandleFunc("/api/v1/batch/lock", func(w http.ResponseWriter, r *http.Request) {
batchRunCommand("input keyevent POWER")
io.WriteString(w, "Success")
})
r.HandleFunc("/api/v1/batch/shell", func(w http.ResponseWriter, r *http.Request) {
command := r.FormValue("command")
batchRunCommand(command)
io.WriteString(w, "Success")
})
// r.HandleFunc("/api/v1/phones/identify")
r.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
renderHTML(w, "index.html", nil)
})
r.PathPrefix("/assets").Handler(http.StripPrefix("/assets/", http.FileServer(http.Dir("./assets"))))
r.HandleFunc("/favicon.ico", func(w http.ResponseWriter, r *http.Request) {
http.ServeFile(w, r, "assets/favicon.ico")
})
r.HandleFunc("/list", func(w http.ResponseWriter, r *http.Request) {
devices, err := db.DeviceList()
if err != nil {
http.Error(w, err.Error(), 500)
return
}
renderJSON(w, devices)
})
r.HandleFunc("/devices/{query}/info", func(w http.ResponseWriter, r *http.Request) {
query := mux.Vars(r)["query"]
udid, err := deviceQueryToUdid(query)
if err != nil {
io.WriteString(w, "Failure, device "+query+" not found")
return
}
if r.Method == "GET" {
info, _ := db.DeviceGet(udid)
renderJSON(w, info)
return
}
// POST
var info proto.DeviceInfo
if err := json.NewDecoder(r.Body).Decode(&info); err != nil {
io.WriteString(w, err.Error())
return
}
db.DeviceUpdate(udid, info)
io.WriteString(w, "Success")
}).Methods("GET", "POST")
r.HandleFunc("/property", func(w http.ResponseWriter, r *http.Request) {
clientIp := realip.FromRequest(r)
udid, err := deviceQueryToUdid("ip:" + clientIp)
if err != nil {
io.WriteString(w, "init with uiautomator2")
return
}
info, err := db.DeviceGet(udid)
if err != nil {
http.Error(w, err.Error(), 500)
return
}
if r.Method == "POST" {
var id string = r.FormValue("id")
if id == "" && r.FormValue("id_number") != "" {
id = "HIH-PHO-" + r.FormValue("id_number")
}
db.DeviceUpdate(info.Udid, proto.DeviceInfo{
PropertyId: id,
})
info.PropertyId = id
io.WriteString(w, "<h1>Updated to "+id+"</h1>")
return
}
renderHTML(w, "property.html", info.PropertyId)
}).Methods("GET", "POST")
r.HandleFunc("/devices/{query}/reserved", func(w http.ResponseWriter, r *http.Request) {
query := mux.Vars(r)["query"]
udid, err := deviceQueryToUdid(query)
if err != nil {
http.Error(w, "Device not found "+err.Error(), http.StatusGone)
return
}
info, err := db.DeviceGet(udid)
if err != nil {
http.Error(w, "Device get error "+err.Error(), http.StatusGone)
return
}
// create websocket connection
ws, err := upgrader.Upgrade(w, r, nil)
if err != nil {
log.Println(err)
return
}
defer ws.Close()
if toBool(info.Using) {
log.Printf("Device %s is using", udid)
return
}
db.DeviceUpdate(info.Udid, proto.DeviceInfo{
Using: newBool(true),
UsingBeganAt: time.Now(),
Owner: &proto.OwnerInfo{
IP: realip.FromRequest(r),
},
})
defer func() {
db.DeviceUpdate(udid, proto.DeviceInfo{
Using: newBool(false),
})
go func() {
port := info.Port
if port == 0 {
port = 7912
}
reqURL := "http://" + info.IP + ":" + strconv.Itoa(port) + "/shell"
req, _ := http.NewRequest("GET", reqURL, nil)
q := req.URL.Query()
q.Add("command", "am start -n com.github.uiautomator/.IdentifyActivity")
req.URL.RawQuery = q.Encode()
resp, err := http.DefaultClient.Do(req)
if err == nil {
resp.Body.Close()
}
}()
}()
// wait until ws disconnected
for {
if _, _, err := ws.ReadMessage(); err != nil {
return
}
}
}).Methods("GET")
r.HandleFunc("/devices/{query}/reserved", func(w http.ResponseWriter, r *http.Request) {
query := mux.Vars(r)["query"]
udid, err := deviceQueryToUdid(query)
// info := hostsManager.Lookup(query)
if err != nil {
http.Error(w, "Device not found "+err.Error(), http.StatusGone)
return
}
if r.Method == "POST" {
info, err := db.DeviceGet(udid)
if err != nil {
http.Error(w, "Device get error "+err.Error(), http.StatusGone)
return
}
if !toBool(info.Present) {
http.Error(w, "Device offline", http.StatusGone)
return
}
if toBool(info.Using) {
http.Error(w, "Device is using", http.StatusForbidden)
return
}
db.DeviceUpdate(info.Udid, proto.DeviceInfo{
Using: newBool(true),
UsingBeganAt: time.Now(),
Owner: &proto.OwnerInfo{
IP: realip.FromRequest(r),
},
})
io.WriteString(w, "Success")
return
}
// DELETE
db.DeviceUpdate(udid, proto.DeviceInfo{
Using: newBool(false),
})
io.WriteString(w, "Release success")
}).Methods("POST", "DELETE")
r.HandleFunc("/devices/{query}/shell", func(w http.ResponseWriter, r *http.Request) {
query := mux.Vars(r)["query"]
udid, err := deviceQueryToUdid(query)
if err != nil {
http.Error(w, "Device not found", 410)
return
}
info, err := db.DeviceGet(udid)
if err != nil {
http.Error(w, "Device get error "+err.Error(), 500)
return
}
command := r.FormValue("command")
output, err := runAndroidShell(info.IP, command)
if err != nil {
renderJSON(w, map[string]string{
"error": err.Error(),
})
} else {
w.Header().Set("Content-Type", "application/json; charset=UTF-8")
io.WriteString(w, output) // the output is already json
}
}).Methods("POST")
// heartbeat for reverse proxies (adb forward device 7912 port)
hbs := heartbeat.NewServer("hello kitty", 15*time.Second)
hbs.OnConnect = func(identifier string, r *http.Request) {
host := realip.FromRequest(r)
db.DeviceUpdateOrInsert(proto.DeviceInfo{
Udid: identifier,
IP: host,
})
log.Println(identifier, "is online")
}
// Called when client ip changes
hbs.OnReconnect = func(identifier string, r *http.Request) {
host := realip.FromRequest(r)
db.DeviceUpdateOrInsert(proto.DeviceInfo{
Udid: identifier,
IP: host,
})
log.Println(identifier, "is reconnected")
}
hbs.OnDisconnect = func(identifier string) {
db.SetDeviceAbsent(identifier)
log.Println(identifier, "is offline")
}
r.Handle("/heartbeat", hbs)
providerhbs := hb2.NewServer(&ProviderReceiver{})
r.Handle("/provider/heartbeat", providerhbs)
return r
}
type ProviderReceiver struct{}
func (p *ProviderReceiver) OnConnect(ctx hb2.Context) error {
port, _ := strconv.Atoi(ctx.Request.FormValue("port"))
if port == 0 {
return errors.New("Request port is required")
}
log.Printf("Provider id:%s ip:%s port:%d is connected", ctx.ID, ctx.IP, port)
return db.ProviderUpdateOrInsert(ctx.ID, ctx.IP, port)
}
func (p *ProviderReceiver) OnDisconnect(id string) {
log.Printf("Provider %s disconnected", id)
db.ProviderOffline(id)
}
/*
POST udid, status=<online|offline>
*/
func (p *ProviderReceiver) OnRequest(ctx hb2.Context) error {
id, req := ctx.ID, ctx.Request
data := req.FormValue("data")
log.Println("Receive data:", data)
var v struct {
Status string `json:"status"`
Udid string `json:"udid"`
ProviderForwardedPort int `json:"providerForwardedPort"`
}
if err := json.Unmarshal([]byte(data), &v); err != nil {
return err
}
status, udid := v.Status, v.Udid
if status == "" || udid == "" {
return errors.New("status or udid is empty")
}
provider, err := db.ProviderGet(id)
if err != nil {
log.Println("Unexpect err:", err)
return err
}
var providerId = &provider.Id
if status == "online" {
log.Printf("Device: %s is plugged-in", udid)
} else if status == "offline" {
log.Printf("Device: %s is plugged-off", udid)
providerId = nil
} else {
log.Printf("Invalid status: %s, only <offline|online> is allowed.", status)
return errors.New("status is required")
}
return db.DeviceUpdate(udid, map[string]interface{}{
"provider_id": providerId,
"providerForwardedPort": v.ProviderForwardedPort,
})
}
Loading...
马建仓 AI 助手
尝试更多
代码解读
代码找茬
代码优化
Go
1
https://gitee.com/billho/ATX.git
git@gitee.com:billho/ATX.git
billho
ATX
ATX
master

搜索帮助