led.go 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. package main
  2. import (
  3. "bufio"
  4. "encoding/json"
  5. "flag"
  6. "fmt"
  7. "net"
  8. "os"
  9. "os/signal"
  10. "strconv"
  11. "strings"
  12. "sync"
  13. "time"
  14. "github.com/davecheney/gpio"
  15. "github.com/pkg/errors"
  16. "git.lattuga.net/boyska/direttoforo"
  17. )
  18. // appicc consumes a channel representing a stream of direttoforo.State updates
  19. // for each message, GPIOs are set accordingly
  20. // if the pin cannot be opened, this function will fail early. However, caller has no way to receive this
  21. // information nor informations about the error
  22. // this function will live as long as the chan "s" will
  23. func appicc(gpios gpioMapFlag, s <-chan direttoforo.State) error {
  24. unassigned := make([]string, 0)
  25. pins := make(map[string]gpio.Pin)
  26. for name, pinnumber := range gpios {
  27. pin, err := gpio.OpenPin(pinnumber, gpio.ModeOutput)
  28. if err != nil {
  29. return errors.Wrap(err, "error opening pin")
  30. }
  31. pins[name] = pin
  32. }
  33. for state := range s {
  34. fmt.Println("led", state)
  35. for sname, sstate := range state.Streams {
  36. pin, ok := pins[sname]
  37. if ok {
  38. if sstate.Up {
  39. pin.Set()
  40. } else {
  41. pin.Clear()
  42. }
  43. } else {
  44. // warn about it?
  45. alreadywarned := false
  46. for _, v := range unassigned {
  47. if v == sname {
  48. alreadywarned = true
  49. }
  50. }
  51. if !alreadywarned {
  52. unassigned = append(unassigned, sname)
  53. fmt.Fprintf(os.Stderr, "INFO: stream '%s' unassigned\n", sname)
  54. }
  55. }
  56. }
  57. }
  58. for _, pin := range pins {
  59. pin.Clear()
  60. pin.Close()
  61. }
  62. fmt.Println("led poweroff")
  63. return nil
  64. }
  65. func connect(bindpath string, s chan<- direttoforo.State, quit <-chan interface{}) {
  66. defer close(s)
  67. OuterLoop:
  68. for {
  69. select {
  70. case <-quit:
  71. return
  72. default:
  73. time.Sleep(time.Second)
  74. }
  75. conn, err := net.Dial("unix", bindpath)
  76. if err != nil {
  77. fmt.Fprintln(os.Stderr, "error connecting", err)
  78. continue OuterLoop
  79. }
  80. defer conn.Close()
  81. hello := []byte("DIRETTOFORO V1\n") // hello size is specified by protocol to be constant
  82. b := bufio.NewReader(conn)
  83. nread, err := b.Read(hello)
  84. if err != nil {
  85. fmt.Fprintln(os.Stderr, "error reading protocol version", err)
  86. continue OuterLoop
  87. }
  88. if nread != 15 || string(hello) != "DIRETTOFORO V1\n" {
  89. fmt.Fprintln(os.Stderr, "error: protocol version mismatch")
  90. continue OuterLoop
  91. }
  92. // read data from connection and put in a chan
  93. msgs := make(chan direttoforo.State)
  94. go func() {
  95. decoder := json.NewDecoder(conn)
  96. var newState direttoforo.State
  97. for {
  98. err = decoder.Decode(&newState)
  99. if err != nil {
  100. fmt.Fprintln(os.Stderr, "error decoding", err)
  101. time.Sleep(time.Second)
  102. close(msgs)
  103. break
  104. }
  105. msgs <- newState
  106. }
  107. }()
  108. MsgLoop: // copy msgs to s, but handles quit signal, too
  109. for {
  110. select {
  111. case <-quit:
  112. return
  113. case msg := <-msgs:
  114. if msg.Streams == nil { // msgs closed
  115. break MsgLoop
  116. }
  117. s <- msg
  118. }
  119. }
  120. }
  121. }
  122. // TODO: maybe create a map[string]int instead?
  123. type gpioMapFlag map[string]int
  124. func (i *gpioMapFlag) String() string {
  125. return fmt.Sprint(*i)
  126. }
  127. func (i *gpioMapFlag) Set(value string) error {
  128. for _, keyvalue := range strings.Split(value, ",") {
  129. kv := strings.SplitN(keyvalue, ":", 2)
  130. if len(kv) != 2 {
  131. return fmt.Errorf("Invalid keyvalue: %s", keyvalue)
  132. }
  133. value, err := strconv.Atoi(kv[1])
  134. if err != nil {
  135. return err
  136. }
  137. for k, v := range *i {
  138. if v == value {
  139. return fmt.Errorf("Can't assign both '%s' and '%s' to %d", k, kv[0], value)
  140. }
  141. }
  142. (*i)[kv[0]] = value
  143. }
  144. return nil
  145. }
  146. func main() {
  147. gpios := make(gpioMapFlag)
  148. bindpath := flag.String("bindpath", "/var/lib/direttoforo/ui.sock", "UNIX domain socket path for UIs")
  149. flag.Var(&gpios, "gpios", "Comma-separated list of stream-GPIOnumber to be used. Example: rec:11,stream:7")
  150. flag.Parse()
  151. fmt.Println(gpios)
  152. states := make(chan direttoforo.State)
  153. connectcloser := make(chan interface{})
  154. allDone := make(chan interface{})
  155. ledDone := make(chan error)
  156. var wg sync.WaitGroup
  157. wg.Add(2)
  158. go func() {
  159. wg.Wait()
  160. close(allDone)
  161. }()
  162. go func() {
  163. err := appicc(gpios, states)
  164. if err != nil {
  165. fmt.Fprintln(os.Stderr, err)
  166. }
  167. ledDone <- err // appicc has no termination notification. This provides a way to have one
  168. wg.Done()
  169. }()
  170. go func() {
  171. connect(*bindpath, states, connectcloser)
  172. wg.Done()
  173. }()
  174. killed := make(chan os.Signal, 1)
  175. signal.Notify(killed, os.Interrupt) // ctrl-c
  176. for {
  177. select {
  178. case sig := <-killed:
  179. fmt.Println("killed by", sig)
  180. close(connectcloser)
  181. case err := <-ledDone:
  182. // led controller failed early (error on OpenPin)
  183. if err != nil {
  184. close(connectcloser)
  185. }
  186. // else: it was closed because we are shutting
  187. // down
  188. case <-allDone:
  189. os.Exit(0)
  190. }
  191. }
  192. }