x.go 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204
  1. /*
  2. * Copyright (c) 2000-2018, 达梦数据库有限公司.
  3. * All rights reserved.
  4. */
  5. package dm
  6. import (
  7. "context"
  8. "strconv"
  9. "strings"
  10. "sync"
  11. "time"
  12. )
  13. const (
  14. STATUS_VALID_TIME = 20 * time.Second // ms
  15. // sort 值
  16. SORT_SERVER_MODE_INVALID = -1 // 不允许连接的模式
  17. SORT_SERVER_NOT_ALIVE = -2 // 站点无法连接
  18. SORT_UNKNOWN = INT32_MAX // 站点还未连接过,模式未知
  19. SORT_NORMAL = 30
  20. SORT_PRIMARY = 20
  21. SORT_STANDBY = 10
  22. // OPEN>MOUNT>SUSPEND
  23. SORT_OPEN = 3
  24. SORT_MOUNT = 2
  25. SORT_SUSPEND = 1
  26. )
  27. type ep struct {
  28. host string
  29. port int32
  30. alive bool
  31. statusRefreshTs int64 // 状态更新的时间点
  32. serverMode int32
  33. serverStatus int32
  34. dscControl bool
  35. sort int32
  36. epSeqno int32
  37. epStatus int32
  38. lock sync.Mutex
  39. }
  40. func newEP(host string, port int32) *ep {
  41. ep := new(ep)
  42. ep.host = host
  43. ep.port = port
  44. ep.serverMode = -1
  45. ep.serverStatus = -1
  46. ep.sort = SORT_UNKNOWN
  47. return ep
  48. }
  49. func (ep *ep) getSort(checkTime bool) int32 {
  50. if checkTime {
  51. if time.Now().UnixNano()-ep.statusRefreshTs < int64(STATUS_VALID_TIME) {
  52. return ep.sort
  53. } else {
  54. return SORT_UNKNOWN
  55. }
  56. }
  57. return ep.sort
  58. }
  59. func (ep *ep) calcSort(loginMode int32) int32 {
  60. var sort int32 = 0
  61. switch loginMode {
  62. case LOGIN_MODE_PRIMARY_FIRST:
  63. {
  64. // 主机优先:PRIMARY>NORMAL>STANDBY
  65. switch ep.serverMode {
  66. case SERVER_MODE_NORMAL:
  67. sort += SORT_NORMAL * 10
  68. case SERVER_MODE_PRIMARY:
  69. sort += SORT_PRIMARY * 100
  70. case SERVER_MODE_STANDBY:
  71. sort += SORT_STANDBY
  72. }
  73. }
  74. case LOGIN_MODE_STANDBY_FIRST:
  75. {
  76. // STANDBY优先: STANDBY>PRIMARY>NORMAL
  77. switch ep.serverMode {
  78. case SERVER_MODE_NORMAL:
  79. sort += SORT_NORMAL
  80. case SERVER_MODE_PRIMARY:
  81. sort += SORT_PRIMARY * 10
  82. case SERVER_MODE_STANDBY:
  83. sort += SORT_STANDBY * 100
  84. }
  85. }
  86. case LOGIN_MODE_NORMAL_FIRST:
  87. {
  88. // NORMAL优先: NORMAL>PRIMARY>STANDBY
  89. switch ep.serverMode {
  90. case SERVER_MODE_STANDBY:
  91. sort += SORT_STANDBY
  92. case SERVER_MODE_PRIMARY:
  93. sort += SORT_PRIMARY * 10
  94. case SERVER_MODE_NORMAL:
  95. sort += SORT_NORMAL * 100
  96. }
  97. }
  98. case LOGIN_MODE_PRIMARY_ONLY:
  99. if ep.serverMode != SERVER_MODE_PRIMARY {
  100. return SORT_SERVER_MODE_INVALID
  101. }
  102. sort += SORT_PRIMARY
  103. case LOGIN_MODE_STANDBY_ONLY:
  104. if ep.serverMode != SERVER_MODE_STANDBY {
  105. return SORT_SERVER_MODE_INVALID
  106. }
  107. sort += SORT_STANDBY
  108. }
  109. switch ep.serverStatus {
  110. case SERVER_STATUS_MOUNT:
  111. sort += SORT_MOUNT
  112. case SERVER_STATUS_OPEN:
  113. sort += SORT_OPEN
  114. case SERVER_STATUS_SUSPEND:
  115. sort += SORT_SUSPEND
  116. }
  117. return sort
  118. }
  119. func (ep *ep) refreshStatus(alive bool, conn *DmConnection) {
  120. ep.lock.Lock()
  121. defer ep.lock.Unlock()
  122. ep.alive = alive
  123. ep.statusRefreshTs = time.Now().UnixNano()
  124. if alive {
  125. ep.serverMode = conn.SvrMode
  126. ep.serverStatus = conn.SvrStat
  127. ep.dscControl = conn.dscControl
  128. ep.sort = ep.calcSort(int32(conn.dmConnector.loginMode))
  129. } else {
  130. ep.serverMode = -1
  131. ep.serverStatus = -1
  132. ep.dscControl = false
  133. ep.sort = SORT_SERVER_NOT_ALIVE
  134. }
  135. }
  136. func (ep *ep) connect(connector *DmConnector) (*DmConnection, error) {
  137. connector.host = ep.host
  138. connector.port = ep.port
  139. conn, err := connector.connectSingle(context.Background())
  140. if err != nil {
  141. ep.refreshStatus(false, conn)
  142. return nil, err
  143. }
  144. ep.refreshStatus(true, conn)
  145. return conn, nil
  146. }
  147. func (ep *ep) getServerStatusDesc(serverStatus int32) string {
  148. ret := ""
  149. switch ep.serverStatus {
  150. case SERVER_STATUS_OPEN:
  151. ret = "OPEN"
  152. case SERVER_STATUS_MOUNT:
  153. ret = "MOUNT"
  154. case SERVER_STATUS_SUSPEND:
  155. ret = "SUSPEND"
  156. default:
  157. ret = "UNKNOWN"
  158. }
  159. return ret
  160. }
  161. func (ep *ep) getServerModeDesc(serverMode int32) string {
  162. ret := ""
  163. switch ep.serverMode {
  164. case SERVER_MODE_NORMAL:
  165. ret = "NORMAL"
  166. case SERVER_MODE_PRIMARY:
  167. ret = "PRIMARY"
  168. case SERVER_MODE_STANDBY:
  169. ret = "STANDBY"
  170. default:
  171. ret = "UNKNOWN"
  172. }
  173. return ret
  174. }
  175. func (ep *ep) String() string {
  176. dscControl := ")"
  177. if ep.dscControl {
  178. dscControl = ", DSC CONTROL)"
  179. }
  180. return strings.TrimSpace(ep.host) + ":" + strconv.Itoa(int(ep.port)) +
  181. " (" + ep.getServerModeDesc(ep.serverMode) + ", " + ep.getServerStatusDesc(ep.serverStatus) + dscControl
  182. }