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.

237 lines
5.8 KiB

8 years ago
8 years ago
7 years ago
7 years ago
8 years ago
8 years ago
8 years ago
  1. package server
  2. import (
  3. "bufio"
  4. "fmt"
  5. "io"
  6. "net"
  7. "sync"
  8. "github.com/tendermint/tendermint/abci/types"
  9. cmn "github.com/tendermint/tendermint/libs/common"
  10. )
  11. // var maxNumberConnections = 2
  12. type SocketServer struct {
  13. cmn.BaseService
  14. proto string
  15. addr string
  16. listener net.Listener
  17. connsMtx sync.Mutex
  18. conns map[int]net.Conn
  19. nextConnID int
  20. appMtx sync.Mutex
  21. app types.Application
  22. }
  23. func NewSocketServer(protoAddr string, app types.Application) cmn.Service {
  24. proto, addr := cmn.ProtocolAndAddress(protoAddr)
  25. s := &SocketServer{
  26. proto: proto,
  27. addr: addr,
  28. listener: nil,
  29. app: app,
  30. conns: make(map[int]net.Conn),
  31. }
  32. s.BaseService = *cmn.NewBaseService(nil, "ABCIServer", s)
  33. return s
  34. }
  35. func (s *SocketServer) OnStart() error {
  36. if err := s.BaseService.OnStart(); err != nil {
  37. return err
  38. }
  39. ln, err := net.Listen(s.proto, s.addr)
  40. if err != nil {
  41. return err
  42. }
  43. s.listener = ln
  44. go s.acceptConnectionsRoutine()
  45. return nil
  46. }
  47. func (s *SocketServer) OnStop() {
  48. s.BaseService.OnStop()
  49. if err := s.listener.Close(); err != nil {
  50. s.Logger.Error("Error closing listener", "err", err)
  51. }
  52. s.connsMtx.Lock()
  53. defer s.connsMtx.Unlock()
  54. for id, conn := range s.conns {
  55. delete(s.conns, id)
  56. if err := conn.Close(); err != nil {
  57. s.Logger.Error("Error closing connection", "id", id, "conn", conn, "err", err)
  58. }
  59. }
  60. }
  61. func (s *SocketServer) addConn(conn net.Conn) int {
  62. s.connsMtx.Lock()
  63. defer s.connsMtx.Unlock()
  64. connID := s.nextConnID
  65. s.nextConnID++
  66. s.conns[connID] = conn
  67. return connID
  68. }
  69. // deletes conn even if close errs
  70. func (s *SocketServer) rmConn(connID int) error {
  71. s.connsMtx.Lock()
  72. defer s.connsMtx.Unlock()
  73. conn, ok := s.conns[connID]
  74. if !ok {
  75. return fmt.Errorf("connection %d does not exist", connID)
  76. }
  77. delete(s.conns, connID)
  78. return conn.Close()
  79. }
  80. func (s *SocketServer) acceptConnectionsRoutine() {
  81. for {
  82. // Accept a connection
  83. s.Logger.Info("Waiting for new connection...")
  84. conn, err := s.listener.Accept()
  85. if err != nil {
  86. if !s.IsRunning() {
  87. return // Ignore error from listener closing.
  88. }
  89. s.Logger.Error("Failed to accept connection: " + err.Error())
  90. continue
  91. }
  92. s.Logger.Info("Accepted a new connection")
  93. connID := s.addConn(conn)
  94. closeConn := make(chan error, 2) // Push to signal connection closed
  95. responses := make(chan *types.Response, 1000) // A channel to buffer responses
  96. // Read requests from conn and deal with them
  97. go s.handleRequests(closeConn, conn, responses)
  98. // Pull responses from 'responses' and write them to conn.
  99. go s.handleResponses(closeConn, conn, responses)
  100. // Wait until signal to close connection
  101. go s.waitForClose(closeConn, connID)
  102. }
  103. }
  104. func (s *SocketServer) waitForClose(closeConn chan error, connID int) {
  105. err := <-closeConn
  106. switch {
  107. case err == io.EOF:
  108. s.Logger.Error("Connection was closed by client")
  109. case err != nil:
  110. s.Logger.Error("Connection error", "error", err)
  111. default:
  112. // never happens
  113. s.Logger.Error("Connection was closed.")
  114. }
  115. // Close the connection
  116. if err := s.rmConn(connID); err != nil {
  117. s.Logger.Error("Error in closing connection", "error", err)
  118. }
  119. }
  120. // Read requests from conn and deal with them
  121. func (s *SocketServer) handleRequests(closeConn chan error, conn io.Reader, responses chan<- *types.Response) {
  122. var count int
  123. var bufReader = bufio.NewReader(conn)
  124. defer func() {
  125. // make sure to recover from any app-related panics to allow proper socket cleanup
  126. r := recover()
  127. if r != nil {
  128. closeConn <- fmt.Errorf("recovered from panic: %v", r)
  129. s.appMtx.Unlock()
  130. }
  131. }()
  132. for {
  133. var req = &types.Request{}
  134. err := types.ReadMessage(bufReader, req)
  135. if err != nil {
  136. if err == io.EOF {
  137. closeConn <- err
  138. } else {
  139. closeConn <- fmt.Errorf("error reading message: %v", err)
  140. }
  141. return
  142. }
  143. s.appMtx.Lock()
  144. count++
  145. s.handleRequest(req, responses)
  146. s.appMtx.Unlock()
  147. }
  148. }
  149. func (s *SocketServer) handleRequest(req *types.Request, responses chan<- *types.Response) {
  150. switch r := req.Value.(type) {
  151. case *types.Request_Echo:
  152. responses <- types.ToResponseEcho(r.Echo.Message)
  153. case *types.Request_Flush:
  154. responses <- types.ToResponseFlush()
  155. case *types.Request_Info:
  156. res := s.app.Info(*r.Info)
  157. responses <- types.ToResponseInfo(res)
  158. case *types.Request_SetOption:
  159. res := s.app.SetOption(*r.SetOption)
  160. responses <- types.ToResponseSetOption(res)
  161. case *types.Request_DeliverTx:
  162. res := s.app.DeliverTx(*r.DeliverTx)
  163. responses <- types.ToResponseDeliverTx(res)
  164. case *types.Request_CheckTx:
  165. res := s.app.CheckTx(*r.CheckTx)
  166. responses <- types.ToResponseCheckTx(res)
  167. case *types.Request_Commit:
  168. res := s.app.Commit()
  169. responses <- types.ToResponseCommit(res)
  170. case *types.Request_Query:
  171. res := s.app.Query(*r.Query)
  172. responses <- types.ToResponseQuery(res)
  173. case *types.Request_InitChain:
  174. res := s.app.InitChain(*r.InitChain)
  175. responses <- types.ToResponseInitChain(res)
  176. case *types.Request_BeginBlock:
  177. res := s.app.BeginBlock(*r.BeginBlock)
  178. responses <- types.ToResponseBeginBlock(res)
  179. case *types.Request_EndBlock:
  180. res := s.app.EndBlock(*r.EndBlock)
  181. responses <- types.ToResponseEndBlock(res)
  182. default:
  183. responses <- types.ToResponseException("Unknown request")
  184. }
  185. }
  186. // Pull responses from 'responses' and write them to conn.
  187. func (s *SocketServer) handleResponses(closeConn chan error, conn io.Writer, responses <-chan *types.Response) {
  188. var count int
  189. var bufWriter = bufio.NewWriter(conn)
  190. for {
  191. var res = <-responses
  192. err := types.WriteMessage(res, bufWriter)
  193. if err != nil {
  194. closeConn <- fmt.Errorf("error writing message: %v", err.Error())
  195. return
  196. }
  197. if _, ok := res.Value.(*types.Response_Flush); ok {
  198. err = bufWriter.Flush()
  199. if err != nil {
  200. closeConn <- fmt.Errorf("error flushing write buffer: %v", err.Error())
  201. return
  202. }
  203. }
  204. count++
  205. }
  206. }