You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

main.go 6.1KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. package main
  2. import "errors"
  3. import "log"
  4. import "os"
  5. import "fmt"
  6. import "path"
  7. import "net/http"
  8. import "io/ioutil"
  9. import "encoding/json"
  10. import "github.com/go-vgo/robotgo"
  11. import "github.com/BurntSushi/toml"
  12. import "coffee.mort.mediator/screencap"
  13. type Config struct {
  14. BasePath string `toml:"base_path"`
  15. }
  16. type EmptyData struct {}
  17. type KeyboardTypeData struct {
  18. Text string `json:"text"`
  19. }
  20. type KeyboardKeyData struct {
  21. Key string `json:"key"`
  22. Modifiers []string `json:"modifiers"`
  23. }
  24. type ScrollData struct {
  25. X int `json:"x"`
  26. Y int `json:"y"`
  27. }
  28. type MouseClickData struct {
  29. Button string `json:"button"`
  30. DoubleClick bool `json:"doubleClick"`
  31. }
  32. type MousePosData struct {
  33. X int `json:"x"`
  34. Y int `json:"y"`
  35. }
  36. type ScreenSizeData struct {
  37. Width int `json:"width"`
  38. Height int `json:"height"`
  39. }
  40. type DirEntryData struct {
  41. Name string `json:"name"`
  42. Type string `json:"type"`
  43. }
  44. type ListDirData struct {
  45. Entries []DirEntryData `json:"entries"`
  46. }
  47. type Error struct {
  48. Error string `json:"error"`
  49. }
  50. func readConfig() (*Config, error) {
  51. confFile, err := os.Open("config.toml")
  52. if err != nil {
  53. return nil, err
  54. }
  55. defer confFile.Close()
  56. var conf Config
  57. _, err = toml.NewDecoder(confFile).Decode(&conf)
  58. if err != nil {
  59. return nil, err
  60. }
  61. return &conf, nil
  62. }
  63. type RW http.ResponseWriter
  64. type Req http.Request
  65. func handler(h func(w RW, req *Req) error) (
  66. func(w http.ResponseWriter, req *http.Request)) {
  67. return func(w http.ResponseWriter, req *http.Request) {
  68. err := h(RW(w), (*Req)(req))
  69. if err != nil {
  70. w.WriteHeader(400)
  71. err = json.NewEncoder(w).Encode(&Error{err.Error()})
  72. if err != nil {
  73. w.Write([]byte("Oh no, failed to encode error struct"))
  74. }
  75. }
  76. }
  77. }
  78. func main() {
  79. conf, err := readConfig()
  80. if err != nil {
  81. log.Fatal(err)
  82. }
  83. log.Printf("Config: %#v", conf)
  84. fs := http.FileServer(http.Dir("./web"))
  85. http.Handle("/", fs)
  86. http.HandleFunc("/api/remote/screen-size", handler(func(w RW, req *Req) error {
  87. if req.Method == "GET" {
  88. var size ScreenSizeData
  89. size.Width, size.Height = robotgo.GetScreenSize()
  90. return json.NewEncoder(w).Encode(&size)
  91. } else {
  92. return errors.New("Invalid method: " + req.Method)
  93. }
  94. }))
  95. http.HandleFunc("/api/remote/mouse-pos", handler(func(w RW, req *Req) error {
  96. if req.Method == "GET" {
  97. var pos MousePosData
  98. pos.X, pos.Y = robotgo.GetMousePos()
  99. return json.NewEncoder(w).Encode(&pos)
  100. } else if req.Method == "PUT" {
  101. var pos MousePosData
  102. err := json.NewDecoder(req.Body).Decode(&pos)
  103. if err != nil {
  104. return err
  105. }
  106. robotgo.MoveMouse(pos.X, pos.Y)
  107. return json.NewEncoder(w).Encode(&EmptyData{})
  108. } else {
  109. return errors.New("Invalid method: " + req.Method)
  110. }
  111. }))
  112. http.HandleFunc("/api/remote/mouse-click", handler(func(w RW, req *Req) error {
  113. if req.Method == "POST" {
  114. var click MouseClickData
  115. err := json.NewDecoder(req.Body).Decode(&click)
  116. if err != nil {
  117. return err
  118. }
  119. robotgo.MouseClick(click.Button, click.DoubleClick)
  120. return json.NewEncoder(w).Encode(&EmptyData{})
  121. } else {
  122. return errors.New("Invalid method: " + req.Method)
  123. }
  124. }))
  125. http.HandleFunc("/api/remote/scroll", handler(func(w RW, req *Req) error {
  126. if req.Method == "POST" {
  127. var scroll ScrollData
  128. err := json.NewDecoder(req.Body).Decode(&scroll)
  129. if err != nil {
  130. return err
  131. }
  132. robotgo.Scroll(scroll.X, scroll.Y)
  133. return json.NewEncoder(w).Encode(&EmptyData{})
  134. } else {
  135. return errors.New("Invalid method: "+ req.Method)
  136. }
  137. }))
  138. http.HandleFunc("/api/remote/keyboard-type", handler(func(w RW, req *Req) error {
  139. if req.Method == "POST" {
  140. var text KeyboardTypeData
  141. err := json.NewDecoder(req.Body).Decode(&text)
  142. if err != nil {
  143. return err
  144. }
  145. robotgo.TypeStr(text.Text)
  146. return json.NewEncoder(w).Encode(&EmptyData{})
  147. } else {
  148. return errors.New("Invalid method: " + req.Method)
  149. }
  150. }))
  151. http.HandleFunc("/api/remote/keyboard-keys", handler(func(w RW, req *Req) error {
  152. if req.Method == "POST" {
  153. var key KeyboardKeyData
  154. err := json.NewDecoder(req.Body).Decode(&key)
  155. if err != nil {
  156. return err
  157. }
  158. var modifiers []interface{}
  159. for _, modifier := range key.Modifiers {
  160. modifiers = append(modifiers, modifier)
  161. }
  162. log.Printf("key: %s, modifiers: %#v", key.Key, modifiers)
  163. robotgo.KeyTap(key.Key, modifiers...)
  164. return json.NewEncoder(w).Encode(&EmptyData{})
  165. } else {
  166. return errors.New("Invalid method: " + req.Method)
  167. }
  168. }))
  169. http.HandleFunc("/api/remote/screencast", handler(func(w RW, req *Req) error {
  170. if req.Method == "GET" {
  171. w.Header().Add("Content-Type", "multipart/x-mixed-replace;boundary=MEDIATOR_FRAME_BOUNDARY")
  172. w.WriteHeader(200)
  173. for {
  174. img := <-screencap.Capture()
  175. log.Printf("Got image, %v bytes", img.Length)
  176. var err error
  177. _, err = w.Write([]byte(fmt.Sprintf(
  178. "--MEDIATOR_FRAME_BOUNDARY\r\n" +
  179. "Content-Type: image/jpeg\r\n" +
  180. "Content-Length: %d\r\n" +
  181. "\r\n", img.Length)))
  182. if err != nil {
  183. log.Printf("Write error: %v", err)
  184. return nil
  185. }
  186. _, err = w.Write(img.Data[0:img.Length])
  187. if err != nil {
  188. log.Printf("Write error: %v", err)
  189. return nil
  190. }
  191. _, err = w.Write([]byte("\r\n"))
  192. if err != nil {
  193. log.Printf("Write error: %v", err)
  194. return nil
  195. }
  196. }
  197. } else {
  198. return errors.New("Invalid method: " + req.Method)
  199. }
  200. }));
  201. http.HandleFunc("/api/dir/", handler(func(w RW, req *Req) error {
  202. if req.Method == "GET" {
  203. subPath := req.URL.Path[len("/api/dir/"):]
  204. dirEnts, err := ioutil.ReadDir(path.Join(conf.BasePath, subPath))
  205. if err != nil {
  206. return err
  207. }
  208. list := ListDirData{
  209. Entries: make([]DirEntryData, 0, len(dirEnts)),
  210. }
  211. for _, ent := range dirEnts {
  212. entType := "f"
  213. if ent.IsDir() {
  214. entType = "d"
  215. }
  216. list.Entries = append(list.Entries, DirEntryData{
  217. Name: ent.Name(),
  218. Type: entType,
  219. })
  220. }
  221. return json.NewEncoder(w).Encode(&list)
  222. } else {
  223. return errors.New("Invalid method: " + req.Method)
  224. }
  225. }))
  226. go screencap.Run()
  227. log.Println("Listening on :3000...")
  228. err = http.ListenAndServe(":3000", nil)
  229. if err != nil {
  230. log.Fatal(err)
  231. }
  232. }