x.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192
  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_PRIMARY_ONLY:
  87. if ep.serverMode != SERVER_MODE_PRIMARY {
  88. return SORT_SERVER_MODE_INVALID
  89. }
  90. sort += SORT_PRIMARY
  91. case LOGIN_MODE_STANDBY_ONLY:
  92. if ep.serverMode != SERVER_MODE_STANDBY {
  93. return SORT_SERVER_MODE_INVALID
  94. }
  95. sort += SORT_STANDBY
  96. }
  97. switch ep.serverStatus {
  98. case SERVER_STATUS_MOUNT:
  99. sort += SORT_MOUNT
  100. case SERVER_STATUS_OPEN:
  101. sort += SORT_OPEN
  102. case SERVER_STATUS_SUSPEND:
  103. sort += SORT_SUSPEND
  104. }
  105. return sort
  106. }
  107. func (ep *ep) refreshStatus(alive bool, conn *DmConnection) {
  108. ep.lock.Lock()
  109. defer ep.lock.Unlock()
  110. ep.alive = alive
  111. ep.statusRefreshTs = time.Now().UnixNano()
  112. if alive {
  113. ep.serverMode = conn.SvrMode
  114. ep.serverStatus = conn.SvrStat
  115. ep.dscControl = conn.dscControl
  116. ep.sort = ep.calcSort(int32(conn.dmConnector.loginMode))
  117. } else {
  118. ep.serverMode = -1
  119. ep.serverStatus = -1
  120. ep.dscControl = false
  121. ep.sort = SORT_SERVER_NOT_ALIVE
  122. }
  123. }
  124. func (ep *ep) connect(connector *DmConnector) (*DmConnection, error) {
  125. connector.host = ep.host
  126. connector.port = ep.port
  127. conn, err := connector.connectSingle(context.Background())
  128. if err != nil {
  129. ep.refreshStatus(false, conn)
  130. return nil, err
  131. }
  132. ep.refreshStatus(true, conn)
  133. return conn, nil
  134. }
  135. func (ep *ep) getServerStatusDesc(serverStatus int32) string {
  136. ret := ""
  137. switch ep.serverStatus {
  138. case SERVER_STATUS_OPEN:
  139. ret = "OPEN"
  140. case SERVER_STATUS_MOUNT:
  141. ret = "MOUNT"
  142. case SERVER_STATUS_SUSPEND:
  143. ret = "SUSPEND"
  144. default:
  145. ret = "UNKNOWN"
  146. }
  147. return ret
  148. }
  149. func (ep *ep) getServerModeDesc(serverMode int32) string {
  150. ret := ""
  151. switch ep.serverMode {
  152. case SERVER_MODE_NORMAL:
  153. ret = "NORMAL"
  154. case SERVER_MODE_PRIMARY:
  155. ret = "PRIMARY"
  156. case SERVER_MODE_STANDBY:
  157. ret = "STANDBY"
  158. default:
  159. ret = "UNKNOWN"
  160. }
  161. return ret
  162. }
  163. func (ep *ep) String() string {
  164. dscControl := ")"
  165. if ep.dscControl {
  166. dscControl = ", DSC CONTROL)"
  167. }
  168. return strings.TrimSpace(ep.host) + ":" + strconv.Itoa(int(ep.port)) +
  169. " (" + ep.getServerModeDesc(ep.serverMode) + ", " + ep.getServerStatusDesc(ep.serverStatus) + dscControl
  170. }