server.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435
  1. // mgo - MongoDB driver for Go
  2. //
  3. // Copyright (c) 2010-2012 - Gustavo Niemeyer <gustavo@niemeyer.net>
  4. //
  5. // All rights reserved.
  6. //
  7. // Redistribution and use in source and binary forms, with or without
  8. // modification, are permitted provided that the following conditions are met:
  9. //
  10. // 1. Redistributions of source code must retain the above copyright notice, this
  11. // list of conditions and the following disclaimer.
  12. // 2. Redistributions in binary form must reproduce the above copyright notice,
  13. // this list of conditions and the following disclaimer in the documentation
  14. // and/or other materials provided with the distribution.
  15. //
  16. // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
  17. // ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  18. // WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  19. // DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
  20. // ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  21. // (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  22. // LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  23. // ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  24. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  25. // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  26. package mgo
  27. import (
  28. "errors"
  29. "labix.org/v2/mgo/bson"
  30. "net"
  31. "sort"
  32. "sync"
  33. "time"
  34. )
  35. // ---------------------------------------------------------------------------
  36. // Mongo server encapsulation.
  37. type mongoServer struct {
  38. sync.RWMutex
  39. Addr string
  40. ResolvedAddr string
  41. tcpaddr *net.TCPAddr
  42. unusedSockets []*mongoSocket
  43. liveSockets []*mongoSocket
  44. closed bool
  45. abended bool
  46. sync chan bool
  47. dial dialer
  48. pingValue time.Duration
  49. pingIndex int
  50. pingCount uint32
  51. pingWindow [6]time.Duration
  52. info *mongoServerInfo
  53. }
  54. type dialer struct {
  55. old func(addr net.Addr) (net.Conn, error)
  56. new func(addr *ServerAddr) (net.Conn, error)
  57. }
  58. func (dial dialer) isSet() bool {
  59. return dial.old != nil || dial.new != nil
  60. }
  61. type mongoServerInfo struct {
  62. Master bool
  63. Mongos bool
  64. Tags bson.D
  65. }
  66. var defaultServerInfo mongoServerInfo
  67. func newServer(addr string, tcpaddr *net.TCPAddr, sync chan bool, dial dialer) *mongoServer {
  68. server := &mongoServer{
  69. Addr: addr,
  70. ResolvedAddr: tcpaddr.String(),
  71. tcpaddr: tcpaddr,
  72. sync: sync,
  73. dial: dial,
  74. info: &defaultServerInfo,
  75. }
  76. // Once so the server gets a ping value, then loop in background.
  77. server.pinger(false)
  78. go server.pinger(true)
  79. return server
  80. }
  81. var errSocketLimit = errors.New("per-server connection limit reached")
  82. var errServerClosed = errors.New("server was closed")
  83. // AcquireSocket returns a socket for communicating with the server.
  84. // This will attempt to reuse an old connection, if one is available. Otherwise,
  85. // it will establish a new one. The returned socket is owned by the call site,
  86. // and will return to the cache when the socket has its Release method called
  87. // the same number of times as AcquireSocket + Acquire were called for it.
  88. // If the limit argument is not zero, a socket will only be returned if the
  89. // number of sockets in use for this server is under the provided limit.
  90. func (server *mongoServer) AcquireSocket(limit int, timeout time.Duration) (socket *mongoSocket, abended bool, err error) {
  91. for {
  92. server.Lock()
  93. abended = server.abended
  94. if server.closed {
  95. server.Unlock()
  96. return nil, abended, errServerClosed
  97. }
  98. n := len(server.unusedSockets)
  99. if limit > 0 && len(server.liveSockets)-n >= limit {
  100. server.Unlock()
  101. return nil, false, errSocketLimit
  102. }
  103. if n > 0 {
  104. socket = server.unusedSockets[n-1]
  105. server.unusedSockets[n-1] = nil // Help GC.
  106. server.unusedSockets = server.unusedSockets[:n-1]
  107. info := server.info
  108. server.Unlock()
  109. err = socket.InitialAcquire(info, timeout)
  110. if err != nil {
  111. continue
  112. }
  113. } else {
  114. server.Unlock()
  115. socket, err = server.Connect(timeout)
  116. if err == nil {
  117. server.Lock()
  118. // We've waited for the Connect, see if we got
  119. // closed in the meantime
  120. if server.closed {
  121. server.Unlock()
  122. socket.Release()
  123. socket.Close()
  124. return nil, abended, errServerClosed
  125. }
  126. server.liveSockets = append(server.liveSockets, socket)
  127. server.Unlock()
  128. }
  129. }
  130. return
  131. }
  132. panic("unreachable")
  133. }
  134. // Connect establishes a new connection to the server. This should
  135. // generally be done through server.AcquireSocket().
  136. func (server *mongoServer) Connect(timeout time.Duration) (*mongoSocket, error) {
  137. server.RLock()
  138. master := server.info.Master
  139. dial := server.dial
  140. server.RUnlock()
  141. logf("Establishing new connection to %s (timeout=%s)...", server.Addr, timeout)
  142. var conn net.Conn
  143. var err error
  144. switch {
  145. case !dial.isSet():
  146. // Cannot do this because it lacks timeout support. :-(
  147. //conn, err = net.DialTCP("tcp", nil, server.tcpaddr)
  148. conn, err = net.DialTimeout("tcp", server.ResolvedAddr, timeout)
  149. case dial.old != nil:
  150. conn, err = dial.old(server.tcpaddr)
  151. case dial.new != nil:
  152. conn, err = dial.new(&ServerAddr{server.Addr, server.tcpaddr})
  153. default:
  154. panic("dialer is set, but both dial.old and dial.new are nil")
  155. }
  156. if err != nil {
  157. logf("Connection to %s failed: %v", server.Addr, err.Error())
  158. return nil, err
  159. }
  160. logf("Connection to %s established.", server.Addr)
  161. stats.conn(+1, master)
  162. return newSocket(server, conn, timeout), nil
  163. }
  164. // Close forces closing all sockets that are alive, whether
  165. // they're currently in use or not.
  166. func (server *mongoServer) Close() {
  167. server.Lock()
  168. server.closed = true
  169. liveSockets := server.liveSockets
  170. unusedSockets := server.unusedSockets
  171. server.liveSockets = nil
  172. server.unusedSockets = nil
  173. server.Unlock()
  174. logf("Connections to %s closing (%d live sockets).", server.Addr, len(liveSockets))
  175. for i, s := range liveSockets {
  176. s.Close()
  177. liveSockets[i] = nil
  178. }
  179. for i := range unusedSockets {
  180. unusedSockets[i] = nil
  181. }
  182. }
  183. // RecycleSocket puts socket back into the unused cache.
  184. func (server *mongoServer) RecycleSocket(socket *mongoSocket) {
  185. server.Lock()
  186. if !server.closed {
  187. server.unusedSockets = append(server.unusedSockets, socket)
  188. }
  189. server.Unlock()
  190. }
  191. func removeSocket(sockets []*mongoSocket, socket *mongoSocket) []*mongoSocket {
  192. for i, s := range sockets {
  193. if s == socket {
  194. copy(sockets[i:], sockets[i+1:])
  195. n := len(sockets) - 1
  196. sockets[n] = nil
  197. sockets = sockets[:n]
  198. break
  199. }
  200. }
  201. return sockets
  202. }
  203. // AbendSocket notifies the server that the given socket has terminated
  204. // abnormally, and thus should be discarded rather than cached.
  205. func (server *mongoServer) AbendSocket(socket *mongoSocket) {
  206. server.Lock()
  207. server.abended = true
  208. if server.closed {
  209. server.Unlock()
  210. return
  211. }
  212. server.liveSockets = removeSocket(server.liveSockets, socket)
  213. server.unusedSockets = removeSocket(server.unusedSockets, socket)
  214. server.Unlock()
  215. // Maybe just a timeout, but suggest a cluster sync up just in case.
  216. select {
  217. case server.sync <- true:
  218. default:
  219. }
  220. }
  221. func (server *mongoServer) SetInfo(info *mongoServerInfo) {
  222. server.Lock()
  223. server.info = info
  224. server.Unlock()
  225. }
  226. func (server *mongoServer) Info() *mongoServerInfo {
  227. server.Lock()
  228. info := server.info
  229. server.Unlock()
  230. return info
  231. }
  232. func (server *mongoServer) hasTags(serverTags []bson.D) bool {
  233. NextTagSet:
  234. for _, tags := range serverTags {
  235. NextReqTag:
  236. for _, req := range tags {
  237. for _, has := range server.info.Tags {
  238. if req.Name == has.Name {
  239. if req.Value == has.Value {
  240. continue NextReqTag
  241. }
  242. continue NextTagSet
  243. }
  244. }
  245. continue NextTagSet
  246. }
  247. return true
  248. }
  249. return false
  250. }
  251. var pingDelay = 5 * time.Second
  252. func (server *mongoServer) pinger(loop bool) {
  253. op := queryOp{
  254. collection: "admin.$cmd",
  255. query: bson.D{{"ping", 1}},
  256. flags: flagSlaveOk,
  257. limit: -1,
  258. }
  259. for {
  260. if loop {
  261. time.Sleep(pingDelay)
  262. }
  263. op := op
  264. socket, _, err := server.AcquireSocket(0, 3 * pingDelay)
  265. if err == nil {
  266. start := time.Now()
  267. _, _ = socket.SimpleQuery(&op)
  268. delay := time.Now().Sub(start)
  269. server.pingWindow[server.pingIndex] = delay
  270. server.pingIndex = (server.pingIndex + 1) % len(server.pingWindow)
  271. server.pingCount++
  272. var max time.Duration
  273. for i := 0; i < len(server.pingWindow) && uint32(i) < server.pingCount; i++ {
  274. if server.pingWindow[i] > max {
  275. max = server.pingWindow[i]
  276. }
  277. }
  278. socket.Release()
  279. server.Lock()
  280. if server.closed {
  281. loop = false
  282. }
  283. server.pingValue = max
  284. server.Unlock()
  285. logf("Ping for %s is %d ms", server.Addr, max/time.Millisecond)
  286. } else if err == errServerClosed {
  287. return
  288. }
  289. if !loop {
  290. return
  291. }
  292. }
  293. }
  294. type mongoServerSlice []*mongoServer
  295. func (s mongoServerSlice) Len() int {
  296. return len(s)
  297. }
  298. func (s mongoServerSlice) Less(i, j int) bool {
  299. return s[i].ResolvedAddr < s[j].ResolvedAddr
  300. }
  301. func (s mongoServerSlice) Swap(i, j int) {
  302. s[i], s[j] = s[j], s[i]
  303. }
  304. func (s mongoServerSlice) Sort() {
  305. sort.Sort(s)
  306. }
  307. func (s mongoServerSlice) Search(resolvedAddr string) (i int, ok bool) {
  308. n := len(s)
  309. i = sort.Search(n, func(i int) bool {
  310. return s[i].ResolvedAddr >= resolvedAddr
  311. })
  312. return i, i != n && s[i].ResolvedAddr == resolvedAddr
  313. }
  314. type mongoServers struct {
  315. slice mongoServerSlice
  316. }
  317. func (servers *mongoServers) Search(resolvedAddr string) (server *mongoServer) {
  318. if i, ok := servers.slice.Search(resolvedAddr); ok {
  319. return servers.slice[i]
  320. }
  321. return nil
  322. }
  323. func (servers *mongoServers) Add(server *mongoServer) {
  324. servers.slice = append(servers.slice, server)
  325. servers.slice.Sort()
  326. }
  327. func (servers *mongoServers) Remove(other *mongoServer) (server *mongoServer) {
  328. if i, found := servers.slice.Search(other.ResolvedAddr); found {
  329. server = servers.slice[i]
  330. copy(servers.slice[i:], servers.slice[i+1:])
  331. n := len(servers.slice) - 1
  332. servers.slice[n] = nil // Help GC.
  333. servers.slice = servers.slice[:n]
  334. }
  335. return
  336. }
  337. func (servers *mongoServers) Slice() []*mongoServer {
  338. return ([]*mongoServer)(servers.slice)
  339. }
  340. func (servers *mongoServers) Get(i int) *mongoServer {
  341. return servers.slice[i]
  342. }
  343. func (servers *mongoServers) Len() int {
  344. return len(servers.slice)
  345. }
  346. func (servers *mongoServers) Empty() bool {
  347. return len(servers.slice) == 0
  348. }
  349. // BestFit returns the best guess of what would be the most interesting
  350. // server to perform operations on at this point in time.
  351. func (servers *mongoServers) BestFit(serverTags []bson.D) *mongoServer {
  352. var best *mongoServer
  353. for _, next := range servers.slice {
  354. if best == nil {
  355. best = next
  356. best.RLock()
  357. if serverTags != nil && !next.info.Mongos && !best.hasTags(serverTags) {
  358. best.RUnlock()
  359. best = nil
  360. }
  361. continue
  362. }
  363. next.RLock()
  364. swap := false
  365. switch {
  366. case serverTags != nil && !next.info.Mongos && !next.hasTags(serverTags):
  367. // Must have requested tags.
  368. case next.info.Master != best.info.Master:
  369. // Prefer slaves.
  370. swap = best.info.Master
  371. case absDuration(next.pingValue-best.pingValue) > 15*time.Millisecond:
  372. // Prefer nearest server.
  373. swap = next.pingValue < best.pingValue
  374. case len(next.liveSockets)-len(next.unusedSockets) < len(best.liveSockets)-len(best.unusedSockets):
  375. // Prefer servers with less connections.
  376. swap = true
  377. }
  378. if swap {
  379. best.RUnlock()
  380. best = next
  381. } else {
  382. next.RUnlock()
  383. }
  384. }
  385. if best != nil {
  386. best.RUnlock()
  387. }
  388. return best
  389. }
  390. func absDuration(d time.Duration) time.Duration {
  391. if d < 0 {
  392. return -d
  393. }
  394. return d
  395. }