AliyunIotSubscribeClient.java 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. package com.coffee.aliyun;
  2. import lombok.Getter;
  3. import lombok.extern.slf4j.Slf4j;
  4. import org.apache.commons.codec.binary.Base64;
  5. import org.apache.qpid.jms.JmsConnection;
  6. import org.apache.qpid.jms.JmsConnectionListener;
  7. import org.apache.qpid.jms.message.JmsInboundMessageDispatch;
  8. import org.springframework.boot.context.properties.EnableConfigurationProperties;
  9. import org.springframework.context.annotation.Configuration;
  10. import org.springframework.stereotype.Component;
  11. import javax.crypto.Mac;
  12. import javax.crypto.spec.SecretKeySpec;
  13. import javax.jms.*;
  14. import javax.naming.Context;
  15. import javax.naming.InitialContext;
  16. import java.net.InetAddress;
  17. import java.net.URI;
  18. import java.net.UnknownHostException;
  19. import java.util.ArrayList;
  20. import java.util.Hashtable;
  21. import java.util.List;
  22. /**
  23. * @Author XX
  24. * @Date 2021-06-17 10:53:30
  25. * @Version 1.0
  26. * @Description XXX
  27. */
  28. @Configuration
  29. @EnableConfigurationProperties(PlatformAccount.class)
  30. @Slf4j
  31. public class AliyunIotSubscribeClient {
  32. private final PlatformAccount platformAccount;
  33. public AliyunIotSubscribeClient(PlatformAccount platformAccount) {
  34. this.platformAccount = platformAccount;
  35. this.accessKey = platformAccount.getAccessKey();
  36. this.accessSecret = platformAccount.getAccessSecret();
  37. this.regionId = platformAccount.getRegionId();
  38. this.consumerGroupId = platformAccount.getConsumerGroupId();
  39. this.aliyunUid = platformAccount.getAliyunUid();
  40. this.iotInstanceId = platformAccount.getIotInstanceId();
  41. //
  42. setClientIdAndHost();
  43. }
  44. // 设置客户端id和host
  45. private void setClientIdAndHost(){
  46. // 获取ip地址
  47. InetAddress addr = null;
  48. try {
  49. addr = InetAddress.getLocalHost();
  50. } catch (UnknownHostException e) {
  51. e.printStackTrace();
  52. }
  53. String ip = addr.getHostAddress();
  54. this.clientId = consumerGroupId + ip;
  55. this.host = aliyunUid+".iot-amqp."+ regionId +".aliyuncs.com";
  56. }
  57. private String accessKey = "LTAI4FhB19MgQuviGxwA3aod";
  58. private String accessSecret = "cQQVkATR0yv2G9CEtfjAhEGBepPDRs";
  59. private String consumerGroupId = "DEFAULT_GROUP";
  60. private String aliyunUid = "1177450762772738";
  61. private String regionId = "cn-shanghai";
  62. //iotInstanceId:企业版实例请填写实例ID,公共实例请填空字符串""。
  63. private String iotInstanceId = "";
  64. //控制台服务端订阅中消费组状态页客户端ID一栏将显示clientId参数。
  65. //建议使用机器UUID、MAC地址、IP等唯一标识等作为clientId。便于您区分识别不同的客户端。
  66. private String clientId = "123456789";
  67. //${YourHost}为接入域名,请参见AMQP客户端接入说明文档。
  68. private String host = "1177450762772738.iot-amqp.cn-shanghai.aliyuncs.com";
  69. // 指定单个进程启动的连接数
  70. // 单个连接消费速率有限,请参考使用限制,最大64个连接
  71. // 连接数和消费速率及rebalance相关,建议每500QPS增加一个连接
  72. private static int connectionCount = 4;
  73. @Getter
  74. private List<Connection> connections = null;
  75. public void start(MessageListener messageListener) throws Exception {
  76. // 先关闭一下
  77. close();
  78. connections = new ArrayList<>();
  79. //参数说明,请参见AMQP客户端接入说明文档。
  80. for (int i = 0; i < connectionCount; i++) {
  81. long timeStamp = System.currentTimeMillis();
  82. //签名方法:支持hmacmd5、hmacsha1和hmacsha256。
  83. String signMethod = "hmacsha1";
  84. //userName组装方法,请参见AMQP客户端接入说明文档。
  85. String userName = clientId + "-" + i + "|authMode=aksign"
  86. + ",signMethod=" + signMethod
  87. + ",timestamp=" + timeStamp
  88. + ",authId=" + accessKey
  89. + ",iotInstanceId=" + iotInstanceId
  90. + ",consumerGroupId=" + consumerGroupId
  91. + "|";
  92. //计算签名,password组装方法,请参见AMQP客户端接入说明文档。
  93. String signContent = "authId=" + accessKey + "&timestamp=" + timeStamp;
  94. String password = doSign(signContent, accessSecret, signMethod);
  95. String connectionUrl = "failover:(amqps://" + host + ":5671?amqp.idleTimeout=80000)"
  96. + "?failover.reconnectDelay=30";
  97. Hashtable<String, String> hashtable = new Hashtable<>();
  98. hashtable.put("connectionfactory.SBCF", connectionUrl);
  99. hashtable.put("queue.QUEUE", "default");
  100. hashtable.put(Context.INITIAL_CONTEXT_FACTORY, "org.apache.qpid.jms.jndi.JmsInitialContextFactory");
  101. Context context = new InitialContext(hashtable);
  102. ConnectionFactory cf = (ConnectionFactory)context.lookup("SBCF");
  103. Destination queue = (Destination)context.lookup("QUEUE");
  104. // 创建连接。
  105. Connection connection = cf.createConnection(userName, password);
  106. connections.add(connection);
  107. ((JmsConnection)connection).addConnectionListener(myJmsConnectionListener);
  108. // 创建会话。
  109. // Session.CLIENT_ACKNOWLEDGE: 收到消息后,需要手动调用message.acknowledge()。
  110. // Session.AUTO_ACKNOWLEDGE: SDK自动ACK(推荐)。
  111. Session session = connection.createSession(false, Session.AUTO_ACKNOWLEDGE);
  112. connection.start();
  113. // 创建Receiver连接。
  114. MessageConsumer consumer = session.createConsumer(queue);
  115. consumer.setMessageListener(messageListener);
  116. }
  117. }
  118. public void close() throws Exception {
  119. if (connections == null){
  120. return;
  121. }
  122. connections.forEach(c-> {
  123. try {
  124. c.close();
  125. } catch (JMSException e) {
  126. log.error("连接关闭失败", e);
  127. }
  128. });
  129. }
  130. private static JmsConnectionListener myJmsConnectionListener = new JmsConnectionListener() {
  131. /**
  132. * 连接成功建立。
  133. */
  134. @Override
  135. public void onConnectionEstablished(URI remoteURI) {
  136. log.info("连接成功, remoteUri:{}", remoteURI);
  137. }
  138. /**
  139. * 尝试过最大重试次数之后,最终连接失败。
  140. */
  141. @Override
  142. public void onConnectionFailure(Throwable error) {
  143. log.error("连接失败, {}", error.getMessage());
  144. }
  145. /**
  146. * 连接中断。
  147. */
  148. @Override
  149. public void onConnectionInterrupted(URI remoteURI) {
  150. log.info("连接中断, remoteUri:{}", remoteURI);
  151. }
  152. /**
  153. * 连接中断后又自动重连上。
  154. */
  155. @Override
  156. public void onConnectionRestored(URI remoteURI) {
  157. log.info("自动重连, remoteUri:{}", remoteURI);
  158. }
  159. @Override
  160. public void onInboundMessage(JmsInboundMessageDispatch envelope) {}
  161. @Override
  162. public void onSessionClosed(Session session, Throwable cause) {}
  163. @Override
  164. public void onConsumerClosed(MessageConsumer consumer, Throwable cause) {}
  165. @Override
  166. public void onProducerClosed(MessageProducer producer, Throwable cause) {}
  167. };
  168. /**
  169. * 计算签名,password组装方法,请参见AMQP客户端接入说明文档。
  170. */
  171. private static String doSign(String toSignString, String secret, String signMethod) throws Exception {
  172. SecretKeySpec signingKey = new SecretKeySpec(secret.getBytes(), signMethod);
  173. Mac mac = Mac.getInstance(signMethod);
  174. mac.init(signingKey);
  175. byte[] rawHmac = mac.doFinal(toSignString.getBytes());
  176. return Base64.encodeBase64String(rawHmac);
  177. }
  178. }