ConsulServiceRegister.cs 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. using Consul;
  2. using Grpc.Extension.Abstract;
  3. using Grpc.Extension.Abstract.Discovery;
  4. using Grpc.Extension.Abstract.Model;
  5. using System;
  6. using System.Threading;
  7. namespace Grpc.Extension.Discovery.Consul
  8. {
  9. /// <summary>
  10. /// Consul服务注册
  11. /// </summary>
  12. public class ConsulServiceRegister : IServiceRegister
  13. {
  14. private Timer _timerTTL;
  15. private string _guid;
  16. private ConsulClient _client;
  17. private ServiceRegisterModel _model;
  18. /// <summary>
  19. /// Consul服务注册
  20. /// </summary>
  21. public ConsulServiceRegister()
  22. {
  23. this._guid = Guid.NewGuid().ToString();
  24. }
  25. /// <summary>
  26. /// 注册服务到consul
  27. /// </summary>
  28. /// <param name="ip"></param>
  29. /// <param name="port"></param>
  30. public void RegisterService(ServiceRegisterModel model)
  31. {
  32. this._model = model;
  33. this._client = CreateConsulClient();
  34. RegisterServiceCore();
  35. //因为公司的consul不支持consul主动检查服务状态,所以启动定时器主动去检测
  36. _timerTTL = new Timer(state => DoTTL(), null, Timeout.Infinite, Timeout.Infinite);
  37. DoTTL();
  38. }
  39. private void RegisterServiceCore()
  40. {
  41. var registration = new AgentServiceRegistration()
  42. {
  43. ID = GetServiceId(),
  44. Name = _model.DiscoveryServiceName,
  45. Tags = _model.DiscoveryServiceTags?.Split(','),
  46. EnableTagOverride = true,
  47. Address = _model.ServiceIp,
  48. Port = _model.ServicePort,
  49. //因为公司的consul不支持consul主动检查服务状态,所以注释掉
  50. //Check = new AgentServiceCheck
  51. //{
  52. // TCP = $"{_model.ServiceIp}:{_model.ServicePort}",
  53. // Interval = TimeSpan.FromSeconds(15),
  54. // Status = HealthStatus.Passing,
  55. // DeregisterCriticalServiceAfter = TimeSpan.FromMinutes(1)
  56. //}
  57. //因为公司的consul不支持consul主动检查服务状态,所以主动去TTL consul
  58. Check = new AgentCheckRegistration
  59. {
  60. ID = GetTTLCheckId(),
  61. Name = "ttlcheck",
  62. TTL = TimeSpan.FromSeconds(15),
  63. Status = HealthStatus.Passing,
  64. DeregisterCriticalServiceAfter = TimeSpan.FromMinutes(1),
  65. }
  66. };
  67. _client.Agent.ServiceRegister(registration).Wait();
  68. }
  69. /// <summary>
  70. /// 从consul反注册
  71. /// </summary>
  72. public void DeregisterService()
  73. {
  74. _client.Agent.ServiceDeregister(GetServiceId()).Wait();
  75. }
  76. private string GetServiceId()
  77. {
  78. return $"{_model.DiscoveryServiceName}-{(_model.ServiceIp)}-{(_model.ServicePort)}-{_guid}";
  79. }
  80. private string GetTTLCheckId()
  81. {
  82. return $"service:{GetServiceId()}";
  83. }
  84. private void DoTTL()
  85. {
  86. _timerTTL.Change(Timeout.Infinite, Timeout.Infinite);
  87. try
  88. {
  89. _client.Agent.PassTTL(GetTTLCheckId(), "timer:" + DateTime.Now).Wait();
  90. }
  91. catch (Exception ex)
  92. {
  93. LoggerAccessor.Instance.OnLoggerError(new InternalException(GrpcErrorCode.Internal, "DoTTL", ex));
  94. /*
  95. * passTTL会出现如下几种情况:
  96. * 1. consul服务重启中,ex会显示 connection refused by ip:port
  97. * 这种情况下,不去处理,等consul服务重启之后就好了
  98. * 2. consul服务重启之后,会丢失之前的service,check,会有如下的错误:
  99. * Unexpected response, status code InternalServerError: CheckID "followme.srv.sms-192.168.3.10-10086-07f21040-0be9-4a73-b0a1-71755c6d6d46:ttlcheck" does not have associated TTL
  100. * 在这种情况下,需要处理,重新注册服务,check;
  101. */
  102. if (ex.ToString().Contains($"CheckID \"{GetTTLCheckId()}\" does not have associated TTL"))
  103. {
  104. RegisterServiceCore();
  105. }
  106. }
  107. finally
  108. {
  109. _timerTTL.Change(TimeSpan.FromSeconds(_model.DiscoveryTTLInterval), TimeSpan.FromSeconds(_model.DiscoveryTTLInterval));
  110. }
  111. }
  112. private ConsulClient CreateConsulClient()
  113. {
  114. return new ConsulClient(conf => conf.Address = new Uri(_model.DiscoveryUrl));
  115. }
  116. }
  117. }