{
+ private static final long serialVersionUID = -1185015143654744140L;
+ /**
+ * 此UUID的最高64有效位
+ */
+ private final long mostSigBits;
+ /**
+ * 此UUID的最低64有效位
+ */
+ private final long leastSigBits;
+
+ /**
+ * 私有构造
+ *
+ * @param data 数据
+ */
+ private UUID(byte[] data) {
+ long msb = 0;
+ long lsb = 0;
+ assert data.length == 16 : "data must be 16 bytes in length";
+ for (int i = 0; i < 8; i++) {
+ msb = (msb << 8) | (data[i] & 0xff);
+ }
+ for (int i = 8; i < 16; i++) {
+ lsb = (lsb << 8) | (data[i] & 0xff);
+ }
+ this.mostSigBits = msb;
+ this.leastSigBits = lsb;
+ }
+
+ /**
+ * 使用指定的数据构造新的 UUID。
+ *
+ * @param mostSigBits 用于 {@code UUID} 的最高有效 64 位
+ * @param leastSigBits 用于 {@code UUID} 的最低有效 64 位
+ */
+ public UUID(long mostSigBits, long leastSigBits) {
+ this.mostSigBits = mostSigBits;
+ this.leastSigBits = leastSigBits;
+ }
+
+ /**
+ * 获取类型 4(伪随机生成的)UUID 的静态工厂。
+ *
+ * @return 随机生成的 {@code UUID}
+ */
+ public static UUID fastUUID() {
+ return randomUUID(false);
+ }
+
+ /**
+ * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的强伪随机数生成器生成该 UUID。
+ *
+ * @return 随机生成的 {@code UUID}
+ */
+ public static UUID randomUUID() {
+ return randomUUID(true);
+ }
+
+ /**
+ * 获取类型 4(伪随机生成的)UUID 的静态工厂。 使用加密的强伪随机数生成器生成该 UUID。
+ *
+ * @param isSecure 是否使用{@link SecureRandom}如果是可以获得更安全的随机码,否则可以得到更好的性能
+ * @return 随机生成的 {@code UUID}
+ */
+ public static UUID randomUUID(boolean isSecure) {
+ final Random ng = isSecure ? Holder.numberGenerator : getRandom();
+
+ byte[] randomBytes = new byte[16];
+ ng.nextBytes(randomBytes);
+ randomBytes[6] &= 0x0f; /* clear version */
+ randomBytes[6] |= 0x40; /* set to version 4 */
+ randomBytes[8] &= 0x3f; /* clear variant */
+ randomBytes[8] |= 0x80; /* set to IETF variant */
+ return new UUID(randomBytes);
+ }
+
+ /**
+ * 根据指定的字节数组获取类型 3(基于名称的)UUID 的静态工厂。
+ *
+ * @param name 用于构造 UUID 的字节数组。
+ * @return 根据指定数组生成的 {@code UUID}
+ */
+ public static UUID nameUUIDFromBytes(byte[] name) {
+ MessageDigest md;
+ try {
+ md = MessageDigest.getInstance("MD5");
+ } catch (NoSuchAlgorithmException nsae) {
+ throw new InternalError("MD5 not supported");
+ }
+ byte[] md5Bytes = md.digest(name);
+ md5Bytes[6] &= 0x0f; /* clear version */
+ md5Bytes[6] |= 0x30; /* set to version 3 */
+ md5Bytes[8] &= 0x3f; /* clear variant */
+ md5Bytes[8] |= 0x80; /* set to IETF variant */
+ return new UUID(md5Bytes);
+ }
+
+ /**
+ * 根据 {@link #toString()} 方法中描述的字符串标准表示形式创建{@code UUID}。
+ *
+ * @param name 指定 {@code UUID} 字符串
+ * @return 具有指定值的 {@code UUID}
+ * @throws IllegalArgumentException 如果 name 与 {@link #toString} 中描述的字符串表示形式不符抛出此异常
+ */
+ public static UUID fromString(String name) {
+ String[] components = name.split("-");
+ if (components.length != 5) {
+ throw new IllegalArgumentException("Invalid UUID string: " + name);
+ }
+ for (int i = 0; i < 5; i++) {
+ components[i] = "0x" + components[i];
+ }
+
+ long mostSigBits = Long.decode(components[0]).longValue();
+ mostSigBits <<= 16;
+ mostSigBits |= Long.decode(components[1]).longValue();
+ mostSigBits <<= 16;
+ mostSigBits |= Long.decode(components[2]).longValue();
+
+ long leastSigBits = Long.decode(components[3]).longValue();
+ leastSigBits <<= 48;
+ leastSigBits |= Long.decode(components[4]).longValue();
+
+ return new UUID(mostSigBits, leastSigBits);
+ }
+
+ /**
+ * 返回指定数字对应的hex值
+ *
+ * @param val 值
+ * @param digits 位
+ * @return 值
+ */
+ private static String digits(long val, int digits) {
+ long hi = 1L << (digits * 4);
+ return Long.toHexString(hi | (val & (hi - 1))).substring(1);
+ }
+
+ /**
+ * 获取{@link SecureRandom},类提供加密的强随机数生成器 (RNG)
+ *
+ * @return {@link SecureRandom}
+ */
+ public static SecureRandom getSecureRandom() {
+ try {
+ return SecureRandom.getInstance("SHA1PRNG");
+ } catch (NoSuchAlgorithmException e) {
+ throw new RuntimeException(e);
+ }
+ }
+
+ /**
+ * 获取随机数生成器对象
+ * ThreadLocalRandom是JDK 7之后提供并发产生随机数,能够解决多个线程发生的竞争争夺。
+ *
+ * @return {@link ThreadLocalRandom}
+ */
+ public static ThreadLocalRandom getRandom() {
+ return ThreadLocalRandom.current();
+ }
+
+ /**
+ * 返回此 UUID 的 128 位值中的最低有效 64 位。
+ *
+ * @return 此 UUID 的 128 位值中的最低有效 64 位。
+ */
+ public long getLeastSignificantBits() {
+ return leastSigBits;
+ }
+
+ /**
+ * 返回此 UUID 的 128 位值中的最高有效 64 位。
+ *
+ * @return 此 UUID 的 128 位值中最高有效 64 位。
+ */
+ public long getMostSignificantBits() {
+ return mostSigBits;
+ }
+
+ /**
+ * 与此 {@code UUID} 相关联的版本号. 版本号描述此 {@code UUID} 是如何生成的。
+ *
+ * 版本号具有以下含意:
+ *
+ * - 1 基于时间的 UUID
+ *
- 2 DCE 安全 UUID
+ *
- 3 基于名称的 UUID
+ *
- 4 随机生成的 UUID
+ *
+ *
+ * @return 此 {@code UUID} 的版本号
+ */
+ public int version() {
+ // Version is bits masked by 0x000000000000F000 in MS long
+ return (int) ((mostSigBits >> 12) & 0x0f);
+ }
+
+ /**
+ * 与此 {@code UUID} 相关联的变体号。变体号描述 {@code UUID} 的布局。
+ *
+ * 变体号具有以下含意:
+ *
+ * - 0 为 NCS 向后兼容保留
+ *
- 2 IETF RFC 4122(Leach-Salz), 用于此类
+ *
- 6 保留,微软向后兼容
+ *
- 7 保留供以后定义使用
+ *
+ *
+ * @return 此 {@code UUID} 相关联的变体号
+ */
+ public int variant() {
+ // This field is composed of a varying number of bits.
+ // 0 - - Reserved for NCS backward compatibility
+ // 1 0 - The IETF aka Leach-Salz variant (used by this class)
+ // 1 1 0 Reserved, Microsoft backward compatibility
+ // 1 1 1 Reserved for future definition.
+ return (int) ((leastSigBits >>> (64 - (leastSigBits >>> 62))) & (leastSigBits >> 63));
+ }
+
+ /**
+ * 与此 UUID 相关联的时间戳值。
+ *
+ *
+ * 60 位的时间戳值根据此 {@code UUID} 的 time_low、time_mid 和 time_hi 字段构造。
+ * 所得到的时间戳以 100 毫微秒为单位,从 UTC(通用协调时间) 1582 年 10 月 15 日零时开始。
+ *
+ *
+ * 时间戳值仅在在基于时间的 UUID(其 version 类型为 1)中才有意义。
+ * 如果此 {@code UUID} 不是基于时间的 UUID,则此方法抛出 UnsupportedOperationException。
+ *
+ * @throws UnsupportedOperationException 如果此 {@code UUID} 不是 version 为 1 的 UUID。
+ */
+ public long timestamp() throws UnsupportedOperationException {
+ checkTimeBase();
+ return (mostSigBits & 0x0FFFL) << 48//
+ | ((mostSigBits >> 16) & 0x0FFFFL) << 32//
+ | mostSigBits >>> 32;
+ }
+
+ /**
+ * 与此 UUID 相关联的时钟序列值。
+ *
+ *
+ * 14 位的时钟序列值根据此 UUID 的 clock_seq 字段构造。clock_seq 字段用于保证在基于时间的 UUID 中的时间唯一性。
+ *
+ * {@code clockSequence} 值仅在基于时间的 UUID(其 version 类型为 1)中才有意义。 如果此 UUID 不是基于时间的 UUID,则此方法抛出
+ * UnsupportedOperationException。
+ *
+ * @return 此 {@code UUID} 的时钟序列
+ * @throws UnsupportedOperationException 如果此 UUID 的 version 不为 1
+ */
+ public int clockSequence() throws UnsupportedOperationException {
+ checkTimeBase();
+ return (int) ((leastSigBits & 0x3FFF000000000000L) >>> 48);
+ }
+
+ /**
+ * 与此 UUID 相关的节点值。
+ *
+ *
+ * 48 位的节点值根据此 UUID 的 node 字段构造。此字段旨在用于保存机器的 IEEE 802 地址,该地址用于生成此 UUID 以保证空间唯一性。
+ *
+ * 节点值仅在基于时间的 UUID(其 version 类型为 1)中才有意义。
+ * 如果此 UUID 不是基于时间的 UUID,则此方法抛出 UnsupportedOperationException。
+ *
+ * @return 此 {@code UUID} 的节点值
+ * @throws UnsupportedOperationException 如果此 UUID 的 version 不为 1
+ */
+ public long node() throws UnsupportedOperationException {
+ checkTimeBase();
+ return leastSigBits & 0x0000FFFFFFFFFFFFL;
+ }
+
+ /**
+ * 返回此{@code UUID} 的字符串表现形式。
+ *
+ *
+ * UUID 的字符串表示形式由此 BNF 描述:
+ *
+ *
+ * {@code
+ * UUID = ----
+ * time_low = 4*
+ * time_mid = 2*
+ * time_high_and_version = 2*
+ * variant_and_sequence = 2*
+ * node = 6*
+ * hexOctet =
+ * hexDigit = [0-9a-fA-F]
+ * }
+ *
+ *
+ *
+ *
+ * @return 此{@code UUID} 的字符串表现形式
+ * @see #toString(boolean)
+ */
+ @Override
+ public String toString() {
+ return toString(false);
+ }
+
+ /**
+ * 返回此{@code UUID} 的字符串表现形式。
+ *
+ *
+ * UUID 的字符串表示形式由此 BNF 描述:
+ *
+ *
+ * {@code
+ * UUID = ----
+ * time_low = 4*
+ * time_mid = 2*
+ * time_high_and_version = 2*
+ * variant_and_sequence = 2*
+ * node = 6*
+ * hexOctet =
+ * hexDigit = [0-9a-fA-F]
+ * }
+ *
+ *
+ *
+ *
+ * @param isSimple 是否简单模式,简单模式为不带'-'的UUID字符串
+ * @return 此{@code UUID} 的字符串表现形式
+ */
+ public String toString(boolean isSimple) {
+ final StringBuilder builder = new StringBuilder(isSimple ? 32 : 36);
+ // time_low
+ builder.append(digits(mostSigBits >> 32, 8));
+ if (false == isSimple) {
+ builder.append('-');
+ }
+ // time_mid
+ builder.append(digits(mostSigBits >> 16, 4));
+ if (false == isSimple) {
+ builder.append('-');
+ }
+ // time_high_and_version
+ builder.append(digits(mostSigBits, 4));
+ if (false == isSimple) {
+ builder.append('-');
+ }
+ // variant_and_sequence
+ builder.append(digits(leastSigBits >> 48, 4));
+ if (false == isSimple) {
+ builder.append('-');
+ }
+ // node
+ builder.append(digits(leastSigBits, 12));
+
+ return builder.toString();
+ }
+
+ // Comparison Operations
+
+ /**
+ * 返回此 UUID 的哈希码。
+ *
+ * @return UUID 的哈希码值。
+ */
+ @Override
+ public int hashCode() {
+ long hilo = mostSigBits ^ leastSigBits;
+ return ((int) (hilo >> 32)) ^ (int) hilo;
+ }
+
+ // -------------------------------------------------------------------------------------------------------------------
+ // Private method start
+
+ /**
+ * 将此对象与指定对象比较。
+ *
+ * 当且仅当参数不为 {@code null}、而是一个 UUID 对象、具有与此 UUID 相同的 varriant、包含相同的值(每一位均相同)时,结果才为 {@code true}。
+ *
+ * @param obj 要与之比较的对象
+ * @return 如果对象相同,则返回 {@code true};否则返回 {@code false}
+ */
+ @Override
+ public boolean equals(Object obj) {
+ if ((null == obj) || (obj.getClass() != UUID.class)) {
+ return false;
+ }
+ UUID id = (UUID) obj;
+ return (mostSigBits == id.mostSigBits && leastSigBits == id.leastSigBits);
+ }
+
+ /**
+ * 将此 UUID 与指定的 UUID 比较。
+ *
+ *
+ * 如果两个 UUID 不同,且第一个 UUID 的最高有效字段大于第二个 UUID 的对应字段,则第一个 UUID 大于第二个 UUID。
+ *
+ * @param val 与此 UUID 比较的 UUID
+ * @return 在此 UUID 小于、等于或大于 val 时,分别返回 -1、0 或 1。
+ */
+ @Override
+ public int compareTo(UUID val) {
+ // The ordering is intentionally set up so that the UUIDs
+ // can simply be numerically compared as two numbers
+ return (this.mostSigBits < val.mostSigBits ? -1 : //
+ (this.mostSigBits > val.mostSigBits ? 1 : //
+ (this.leastSigBits < val.leastSigBits ? -1 : //
+ (this.leastSigBits > val.leastSigBits ? 1 : //
+ 0))));
+ }
+
+ /**
+ * 检查是否为time-based版本UUID
+ */
+ private void checkTimeBase() {
+ if (version() != 1) {
+ throw new UnsupportedOperationException("Not a time-based UUID");
+ }
+ }
+
+ /**
+ * SecureRandom 的单例
+ */
+ private static class Holder {
+ static final SecureRandom numberGenerator = getSecureRandom();
+ }
+}
diff --git a/src/main/java/com/lyh/controller/LoadCenterController.java b/src/main/java/com/lyh/controller/LoadCenterController.java
new file mode 100644
index 0000000..2e23fc2
--- /dev/null
+++ b/src/main/java/com/lyh/controller/LoadCenterController.java
@@ -0,0 +1,37 @@
+package com.lyh.controller;
+
+import com.lyh.common.domain.resp.Result;
+import com.lyh.service.LoadCenterService;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.web.bind.annotation.PostMapping;
+import org.springframework.web.bind.annotation.RequestMapping;
+import org.springframework.web.bind.annotation.RestController;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/15
+ * @Description: 负载中心控制层
+ */
+
+@RestController
+@RequestMapping("/loadCenter")
+@Slf4j
+public class LoadCenterController {
+ @Autowired
+ private LoadCenterService loadCenterService;
+
+
+ /*
+ * @Author: LiuYunHu
+ * @Date: 2024/4/15 14:57
+ * @Description: 获取指定的服务器IP
+ * @Param: []
+ * @Return: com.lyh.common.domain.resp.Result
+ **/
+ @PostMapping("/getAssignedServer")
+ public Result getAssignedServer() {
+ return loadCenterService.getAssignedServer();
+ }
+}
diff --git a/src/main/java/com/lyh/job/Timer.java b/src/main/java/com/lyh/job/Timer.java
new file mode 100644
index 0000000..5c5910a
--- /dev/null
+++ b/src/main/java/com/lyh/job/Timer.java
@@ -0,0 +1,70 @@
+package com.lyh.job;
+
+import com.alibaba.fastjson2.JSONArray;
+import com.alibaba.fastjson2.JSONObject;
+import com.lyh.openAPI.WorkCenter.CreateAndRunInstance;
+import lombok.extern.slf4j.Slf4j;
+import okhttp3.OkHttpClient;
+import okhttp3.Request;
+import okhttp3.Response;
+import org.springframework.stereotype.Component;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/13
+ * @Description:
+ */
+@Component
+@Slf4j
+public class Timer {
+ // @Scheduled(cron = "0/10 * * * * ?")
+ public void test() {
+ String ip = "47.102.123.209";
+ //请求路径
+ String URL = "http://" + ip + ":8080/public/cluster";
+
+ OkHttpClient client = new OkHttpClient();
+
+ Request req = new Request.Builder()
+ .url(URL)
+ .get()
+ .addHeader("User-Agent", "Apifox/1.0.0 (https://apifox.com)")
+ .addHeader("Accesstoken", "")
+ .build();
+
+ try {
+ Response response = client.newCall(req).execute();
+
+ log.info(String.valueOf(response));
+
+
+ JSONArray jsonArray = JSONArray.parseArray(response.body().string());
+ JSONObject jsonObject = jsonArray.getJSONObject(0);
+ //获取mqttInfo对象的值
+ JSONObject mqttInfo = jsonObject.getJSONObject("mqttInfo");
+ //获取连接数
+ int connectSize = mqttInfo.getIntValue("connectSize");
+
+ log.info(ip + " 的fluxmq连接数为:" + connectSize);
+
+
+ if (connectSize >= 80) {
+ //执行节点扩容
+
+ //返回实例的ID
+ String instanceId = CreateAndRunInstance.startCreate();
+
+ if (!instanceId.isEmpty()) {
+ log.info("扩容 成功!");
+ log.info("扩容的节点ip为:" + instanceId);
+ }
+ }
+
+ } catch (Exception e) {
+ e.printStackTrace();
+ }
+
+
+ }
+}
diff --git a/src/main/java/com/lyh/openAPI/WorkCenter/CreateAndRunInstance.java b/src/main/java/com/lyh/openAPI/WorkCenter/CreateAndRunInstance.java
new file mode 100644
index 0000000..042651c
--- /dev/null
+++ b/src/main/java/com/lyh/openAPI/WorkCenter/CreateAndRunInstance.java
@@ -0,0 +1,105 @@
+// This file is auto-generated, don't edit it. Thanks.
+package com.lyh.openAPI.WorkCenter;
+
+import com.aliyun.ecs20140526.models.RunInstancesRequest;
+import com.aliyun.ecs20140526.models.RunInstancesResponse;
+import com.aliyun.tea.TeaException;
+import com.lyh.common.config.ClientService;
+import com.lyh.common.utils.UserUtil;
+import org.springframework.stereotype.Component;
+
+import java.util.Date;
+
+/*
+ * 根据镜像创建并运行实例
+ * */
+@Component
+public class CreateAndRunInstance {
+
+
+ /*
+ * @Author: LiuYunHu
+ * @Date: 2024/4/13 9:28
+ * @Description: 开始创建实例
+ * @Param: []
+ * @Return: java.lang.String
+ **/
+ public static String startCreate() throws Exception {
+ // 地域Id
+ String regionId = "cn-shanghai";
+ // 镜像 ID,启动实例时选择的镜像资源。
+ String imageId = "m-uf63t8i01vefni8piwen";
+ // 实例规格
+ String instanceType = "ecs.e-c1m1.large";
+ // 新创建实例所属于的安全组 ID。
+ String securityGroupId = "sg-uf6fvbhimn7xa54i36wc";
+ // 虚拟交换机 ID。
+ String vSwitchId = "vsw-uf62esxo4f5dfs43aqfu6";
+ // 公网出带宽最大值,单位为 Mbit/s。取值范围:0~100。 默认值:0。
+ Integer internetMaxBandwidthOut = com.aliyun.darabonbanumber.Client.parseInt("5");
+ // 网络计费类型。取值范围:
+ // PayByBandwidth: 按固定带宽计费。
+ // PayByTraffic: 按使用流量计费。
+ // 默认值:PayByTraffic。
+ String internetChargeType = "PayByTraffic";
+ // 系统盘大小
+ String size = "20";
+ // 系统盘的云盘种类
+ String category = "cloud_essd_entry";
+ // ECS实例的计费方式
+ // PrePaid:包年包月
+ // PostPaid:按量付费
+ String instanceChargeType = "PostPaid";
+
+
+ com.aliyun.ecs20140526.Client client = ClientService.createEcsClient(regionId);
+ // 批量创建实例
+ return CreateAndRunInstance.RunInstances(client, regionId, imageId, instanceType, securityGroupId, vSwitchId, internetMaxBandwidthOut, internetChargeType, size, category, instanceChargeType);
+ }
+
+ /**
+ * RunInstances 通过备选实例规格创建ECS实例最佳实践
+ * 该场景中,在调用RunInstances创建ECS实例时判断是否发生库存不足等错误,如果发生错误,将调用DescribeRecommendInstanceType查询备选实例,然后通过备选实例规格重新创建ECS实例。
+ */
+ public static String RunInstances(com.aliyun.ecs20140526.Client client, String regionId, String imageId, String instanceType, String securityGroupId, String vSwitchId, Integer internetMaxBandwidthOut, String internetChargeType, String size, String category, String instanceChargeType) throws Exception {
+ RunInstancesRequest request1 = new RunInstancesRequest()
+ .setRegionId(regionId)
+ .setImageId(imageId)
+ .setInstanceType(instanceType)
+ .setSecurityGroupId(securityGroupId)
+ .setVSwitchId(vSwitchId)
+ .setInstanceName("自动创建的实例节点")
+ .setDescription(new Date().toLocaleString() + " 创建的实例节点")
+ .setInternetMaxBandwidthOut(internetMaxBandwidthOut)
+ .setInternetChargeType(internetChargeType)
+ .setInstanceChargeType(instanceChargeType)
+ // 批量创建五台ECS实例,如果不设置该参数,默认创建一台ECS实例。
+ // amount = 5,
+ // 如果缺少库存可以接受的最低创建数量。
+ // minAmount = 2,
+ // 打开预检参数功能,不会实际创建ECS实例,只检查参数正确性、用户权限或者ECS库存等问题。
+ // 实际情况下,设置了DryRun参数后,Amount必须为1,MinAmount必须为空,您可以根据实际需求修改代码。
+// .setDryRun(true)
+ .setDryRun(false)
+ .setSystemDisk(new RunInstancesRequest.RunInstancesRequestSystemDisk()
+ .setSize(size)
+ .setCategory(category));
+
+
+ String result = "";
+ try {
+ com.aliyun.teaconsole.Client.log("--------------------批量创建实例开始--------------------");
+ RunInstancesResponse responces = client.runInstances(request1);
+ com.aliyun.teaconsole.Client.log("--------------------创建实例成功,实例ID:" + com.aliyun.teautil.Common.toJSONString(responces.body.instanceIdSets.instanceIdSet) + "--------------------");
+ //返回实例ID
+ result = responces.body.instanceIdSets.instanceIdSet + "";//前后带 []
+ result = UserUtil.removeBrackets(result);//前后不带[]
+ } catch (TeaException error) {
+ com.aliyun.teaconsole.Client.log("--------------------创建实例失败:" + com.aliyun.teautil.Common.toJSONString(error.code) + "--------------------" + error.message);
+ } catch (Exception _error) {
+ TeaException error = new TeaException(_error.getMessage(), _error);
+ com.aliyun.teaconsole.Client.log("--------------------创建实例失败:" + com.aliyun.teautil.Common.toJSONString(error.code) + "--------------------" + error.message);
+ }
+ return result;
+ }
+}
diff --git a/src/main/java/com/lyh/openAPI/WorkCenter/GetInstanceIDList.java b/src/main/java/com/lyh/openAPI/WorkCenter/GetInstanceIDList.java
new file mode 100644
index 0000000..d26fed8
--- /dev/null
+++ b/src/main/java/com/lyh/openAPI/WorkCenter/GetInstanceIDList.java
@@ -0,0 +1,68 @@
+package com.lyh.openAPI.WorkCenter;
+
+import com.aliyun.ecs20140526.Client;
+import com.aliyun.ecs20140526.models.DescribeInstancesRequest;
+import com.aliyun.ecs20140526.models.DescribeInstancesResponse;
+import com.aliyun.ecs20140526.models.DescribeInstancesResponseBody;
+import com.lyh.common.config.ClientService;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.stereotype.Component;
+
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/15
+ * @Description: 获取指定区域的实例ID列表
+ */
+@Component
+@Slf4j
+public class GetInstanceIDList {
+ public static List getIDList() throws Exception {
+// // 1. 初始化配置
+// Config config = new Config();
+// // 您的AccessKey ID
+// config.accessKeyId = "LTAI5tEBb3cjLf8nRBJZmfD9";
+// // 您的AccessKey Secret
+// config.accessKeySecret = "Emzjl8SlqwlHP7xeFOrhwtXiJNfbkF";
+// //设置请求地址
+// config.endpoint = "ecs.aliyuncs.com";
+// // 设置连接超时为5000毫秒
+// config.connectTimeout = 5000;
+// // 设置读超时为5000毫秒
+// config.readTimeout = 5000;
+// // 2. 初始化客户端
+// com.aliyun.ecs20140526.Client client = new com.aliyun.ecs20140526.Client(config);
+
+ // 地域Id
+ String addressId = "cn-shanghai";
+
+ Client client = ClientService.createEcsClient(addressId);
+ java.util.List regionIds = com.aliyun.darabonbastring.Client.split(addressId, ",", 50);
+
+ String regionId = regionIds.get(0);
+
+ DescribeInstancesRequest describeInstancesRequest = new DescribeInstancesRequest()
+ .setPageSize(100)
+ .setRegionId(regionId);
+ DescribeInstancesResponse resp = client.describeInstances(describeInstancesRequest);
+ java.util.List instances = resp.body.instances.instance;
+ com.aliyun.teaconsole.Client.log("" + regionId + " 下 ECS 实例列表:");
+
+ //存储结果的List
+ ArrayList result = new ArrayList<>();
+ for (DescribeInstancesResponseBody.DescribeInstancesResponseBodyInstancesInstance instance : instances) {
+ com.aliyun.teaconsole.Client.log("主机名:" + instance.hostName + " 实例ID:" + instance.instanceId + " CPU:" + instance.cpu + " 内存:" + instance.memory + " MB 规格:" + instance.instanceType + " 系统:" + instance.OSType + "(" + instance.OSName + ") 状态:" + instance.status + "");
+
+ result.add(instance.instanceId);
+ }
+
+ log.info(result);
+
+ //返回结果
+ return result;
+ }
+
+}
diff --git a/src/main/java/com/lyh/openAPI/WorkCenter/GetInstanceProperties.java b/src/main/java/com/lyh/openAPI/WorkCenter/GetInstanceProperties.java
new file mode 100644
index 0000000..8206fa5
--- /dev/null
+++ b/src/main/java/com/lyh/openAPI/WorkCenter/GetInstanceProperties.java
@@ -0,0 +1,98 @@
+package com.lyh.openAPI.WorkCenter;
+
+
+import com.aliyun.ecs20140526.models.DescribeInstancesResponse;
+import com.aliyun.ecs20140526.models.DescribeInstancesResponseBody;
+import com.aliyun.tea.TeaException;
+import com.lyh.common.config.ClientService;
+import com.lyh.common.utils.UserUtil;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.stereotype.Component;
+
+import java.util.List;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/13
+ * @Description: 查询一台或多台ECS实例的详细信息
+ */
+@Component
+@Slf4j
+public class GetInstanceProperties {
+ static int i = 1;
+
+ /*
+ * @Author: LiuYunHu
+ * @Date: 2024/4/13 11:01
+ * @Description: 实例ID 用 英文逗号拼接
+ * @Param: [instanceIds] i-uf6chlqotgoc9h173alu
+ * @Return: void
+ **/
+ public static List startGet(String instanceIds) throws Exception {
+ // 地域Id
+ String regionId = "cn-shanghai";
+
+ com.aliyun.ecs20140526.Client client = ClientService.createEcsClient(regionId);
+ com.aliyun.ecs20140526.models.DescribeInstancesRequest describeInstancesRequest = new com.aliyun.ecs20140526.models.DescribeInstancesRequest()
+ .setRegionId(regionId)
+ .setInstanceName("*")
+ .setInstanceIds(com.aliyun.teautil.Common.toJSONString(com.aliyun.darabonbastring.Client.split(instanceIds, ",", 50)))
+ .setPageSize(10);
+
+
+ com.aliyun.teautil.models.RuntimeOptions runtime = new com.aliyun.teautil.models.RuntimeOptions();
+
+ //初始化返回值
+ List instance = null;
+
+ try {
+ // 复制代码运行请自行打印 API 的返回值
+ DescribeInstancesResponse describeInstancesResponse = client.describeInstancesWithOptions(describeInstancesRequest, runtime);
+ DescribeInstancesResponseBody body = describeInstancesResponse.getBody();
+ DescribeInstancesResponseBody.DescribeInstancesResponseBodyInstances instances = body.getInstances();
+
+ //能返回的值
+ instance = instances.getInstance();
+ instance.forEach(item -> {
+ log.info("实例{" + i + "}的ID:" + item.getInstanceId());
+ log.info("名称:" + item.getInstanceName());
+ log.info("地域ID:" + item.getRegionId());
+ log.info("状态:" + item.getStatus());
+ log.info("类型:" + item.getInstanceType());
+ log.info("CPU核心数:" + item.getCpu());
+ log.info("内存大小:" + item.getMemory() + "MB");
+ log.info("磁盘大小:" + item.getLocalStorageCapacity() + "G");
+ log.info("操作系统:" + item.getOSName());
+ log.info("网络类型:" + item.getInstanceNetworkType());
+ log.info("公网出带宽值:" + item.getInternetMaxBandwidthOut() + "Mbit/s");
+ log.info("公网入带宽值:" + item.getInternetMaxBandwidthIn() + "Mbit/s");
+ log.info("公网IP:" + UserUtil.removeBrackets(item.getPublicIpAddress().getIpAddress().toString()));
+ log.info("私网IP:" + UserUtil.removeBrackets(item.getVpcAttributes().getPrivateIpAddress().ipAddress.toString()));
+ log.info("专有网络VPCID:" + item.getVpcAttributes().getVpcId());
+ log.info("安全组ID:" + UserUtil.removeBrackets(item.getSecurityGroupIds().getSecurityGroupId().toString()));
+ log.info("创建时间:" + item.getCreationTime());
+ log.info("到期时间:" + item.getExpiredTime());
+ log.info("是否可以回收:" + (item.getRecyclable() ? "是" : "否") + "\n\n");
+ log.info("---------------------");
+ i++;
+ });
+ } catch (TeaException error) {
+ // 此处仅做打印展示,请谨慎对待异常处理,在工程项目中切勿直接忽略异常。
+ // 错误 message
+ log.error(error.getMessage());
+ // 诊断地址
+ log.error(error.getData().get("Recommend").toString());
+ com.aliyun.teautil.Common.assertAsString(error.message);
+ } catch (Exception _error) {
+ TeaException error = new TeaException(_error.getMessage(), _error);
+ // 此处仅做打印展示,请谨慎对待异常处理,在工程项目中切勿直接忽略异常。
+ // 错误 message
+ log.error(error.getMessage());
+ // 诊断地址
+ log.error(error.getData().get("Recommend").toString());
+ com.aliyun.teautil.Common.assertAsString(error.message);
+ }
+ return instance;
+ }
+}
diff --git a/src/main/java/com/lyh/openAPI/WorkCenter/ReleaseInstance.java b/src/main/java/com/lyh/openAPI/WorkCenter/ReleaseInstance.java
new file mode 100644
index 0000000..e535f16
--- /dev/null
+++ b/src/main/java/com/lyh/openAPI/WorkCenter/ReleaseInstance.java
@@ -0,0 +1,85 @@
+// This file is auto-generated, don't edit it. Thanks.
+package com.lyh.openAPI.WorkCenter;
+
+import com.aliyun.ecs20140526.models.*;
+import com.lyh.common.config.ClientService;
+import org.springframework.stereotype.Component;
+
+/**
+ * @ProjectName: five-groups-couplet
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/12
+ * @Description: 释放实例
+ */
+
+@Component
+public class ReleaseInstance {
+ public static DescribeInstancesResponse DescribeInstances(com.aliyun.ecs20140526.Client client, String regionId, String instanceIds, String instanceName) throws Exception {
+ DescribeInstancesRequest req = new DescribeInstancesRequest()
+ .setRegionId(regionId)
+ .setInstanceName(instanceName);
+ if (!com.aliyun.teautil.Common.empty(instanceIds)) {
+ req.instanceIds = com.aliyun.teautil.Common.toJSONString(com.aliyun.darabonbastring.Client.split(instanceIds, ",", 50));
+ }
+
+ DescribeInstancesResponse resp = client.describeInstances(req);
+ com.aliyun.teaconsole.Client.log("--------------------查询需要删除的实例--------------------");
+ return resp;
+ }
+
+ public static void ModifyInstanceAttribute(com.aliyun.ecs20140526.Client client, String instatnceId) throws Exception {
+ ModifyInstanceAttributeRequest req = new ModifyInstanceAttributeRequest()
+ .setInstanceId(instatnceId)
+ .setDeletionProtection(false);
+ client.modifyInstanceAttribute(req);
+ com.aliyun.teaconsole.Client.log("--------------------" + instatnceId + "释放保护取消成功--------------------");
+ }
+
+ public static void DeleteInstances(com.aliyun.ecs20140526.Client client, String regionId, String instanceIds, String force) throws Exception {
+ DeleteInstancesRequest req = new DeleteInstancesRequest()
+ .setRegionId(regionId)
+ .setInstanceId(com.aliyun.darabonbastring.Client.split(instanceIds, ",", 50))
+ .setForce(com.aliyun.teautil.Common.equalString(force, "true"));
+ DeleteInstancesResponse resp = client.deleteInstances(req);
+ com.aliyun.teaconsole.Client.log("--------------------实例释放成功--------------------");
+ com.aliyun.teaconsole.Client.log(com.aliyun.teautil.Common.toJSONString(com.aliyun.teautil.Common.toMap(resp)));
+ }
+
+ /*
+ * @Author: LiuYunHu
+ * @Date: 2024/4/13 11:02
+ * @Description: 多个实例ID,用英文逗号分隔
+ * @Param: [instanceIds] i-uf6chlqotgoc9h173alu
+ * @Return: void
+ **/
+ public static void startRelease(String instanceIds) throws Exception {
+ // 区域ID
+ String regionId = "cn-shanghai";
+ // 实例名称,支持使用通配符*进行模糊搜索
+ String instanceName = "*";
+ // 强制删除有删除保护的机器
+ String deleteProtected = "true";
+ // 强制删除运行中的机器
+ String force = "true";
+ com.aliyun.ecs20140526.Client client = ClientService.createEcsClient(regionId);
+ if (com.aliyun.teautil.Common.equalString(deleteProtected, "true")) {
+ DescribeInstancesResponse describeInstancesResp = ReleaseInstance.DescribeInstances(client, regionId, instanceIds, instanceName);
+ instanceIds = "";
+ for (DescribeInstancesResponseBody.DescribeInstancesResponseBodyInstancesInstance instance : describeInstancesResp.body.instances.instance) {
+ instanceIds = "" + instance.instanceId + "," + instanceIds + "";
+ if (instance.deletionProtection) {
+ ReleaseInstance.ModifyInstanceAttribute(client, instance.instanceId);
+ }
+
+ }
+ instanceIds = com.aliyun.darabonbastring.Client.subString(instanceIds, 0, -1);
+ }
+
+ if (com.aliyun.teautil.Common.empty(instanceIds)) {
+ com.aliyun.teaconsole.Client.log("--------------------无有效实例可删除--------------------");
+ return;
+ }
+
+ ReleaseInstance.DeleteInstances(client, regionId, instanceIds, force);
+ }
+}
diff --git a/src/main/java/com/lyh/service/LoadCenterService.java b/src/main/java/com/lyh/service/LoadCenterService.java
new file mode 100644
index 0000000..1baa94b
--- /dev/null
+++ b/src/main/java/com/lyh/service/LoadCenterService.java
@@ -0,0 +1,15 @@
+package com.lyh.service;
+
+import com.lyh.common.domain.resp.Result;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/15
+ * @Description:
+ */
+
+public interface LoadCenterService {
+ Result getAssignedServer();
+
+}
diff --git a/src/main/java/com/lyh/service/impl/LoadCenterServiceImpl.java b/src/main/java/com/lyh/service/impl/LoadCenterServiceImpl.java
new file mode 100644
index 0000000..d0f111b
--- /dev/null
+++ b/src/main/java/com/lyh/service/impl/LoadCenterServiceImpl.java
@@ -0,0 +1,157 @@
+package com.lyh.service.impl;
+
+import com.aliyun.ecs20140526.models.DescribeInstancesResponseBody;
+import com.lyh.common.domain.IpAndLoadCount;
+import com.lyh.common.domain.IpAndWeight;
+import com.lyh.common.domain.resp.Result;
+import com.lyh.common.redis.service.RedisService;
+import com.lyh.common.utils.UserUtil;
+import com.lyh.common.utils.mqtt.MqttUtil;
+import com.lyh.openAPI.WorkCenter.GetInstanceIDList;
+import com.lyh.openAPI.WorkCenter.GetInstanceProperties;
+import com.lyh.service.LoadCenterService;
+import lombok.extern.log4j.Log4j2;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.stereotype.Service;
+
+import java.util.ArrayList;
+import java.util.List;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/15
+ * @Description:
+ */
+@Service
+@Log4j2
+public class LoadCenterServiceImpl implements LoadCenterService {
+ @Autowired
+ private MqttUtil mqttUtil;
+
+ @Autowired
+ private RedisService redis;
+
+ /*
+ * @Author: LiuYunHu
+ * @Date: 2024/4/15 21:49
+ * @Description: 获取 分配的一个服务器IP
+ * @Param: []
+ * @Return: com.lyh.common.domain.resp.Result
+ **/
+ @Override
+ public Result getAssignedServer() {
+
+ //存IP的List
+ ArrayList ipList = new ArrayList<>();
+
+ //获取上海区的实例ID列表
+ try {
+ List idList = GetInstanceIDList.getIDList();
+ idList.forEach(id -> {
+ //调用方法,获取对应ID实例的IP
+ try {
+ List result = GetInstanceProperties.startGet(id);
+
+ //获取集合第一个的属性
+ DescribeInstancesResponseBody.DescribeInstancesResponseBodyInstancesInstance response = result.get(0);
+
+ //获取添加的实例的公网ip
+ String instanceIp = UserUtil.removeBrackets(response.getPublicIpAddress().getIpAddress().toString());
+ ipList.add(instanceIp);
+
+ } catch (Exception e) {
+ log.error(e.getMessage());
+ }
+ });
+ } catch (Exception e) {
+ log.error(e.getMessage());
+ }
+
+ log.info(ipList);
+
+ //将IP列表存入redis
+ redis.deleteObject("服务器列表:");
+ redis.setCacheList("服务器列表:", ipList);
+
+ //存各个 服务器的负载量
+ ArrayList ipAndLoadCounts = new ArrayList<>();
+
+
+ ipList.forEach(ip -> {
+
+ //拿到IP后,获取各个IP的负载量
+ int fetchLoad = mqttUtil.getFetchLoad(ip);
+ ipAndLoadCounts.add(new IpAndLoadCount(ip, fetchLoad));
+ });
+
+
+ //求出空负载的总量
+ Integer emptyLoadCount = 0;
+ for (IpAndLoadCount ipAndLoadCount : ipAndLoadCounts) {
+
+ //假设使用2/8原则
+ emptyLoadCount += (80 - ipAndLoadCount.getLoadCount());
+ }
+
+
+ //存储IP和对应的权重
+ ArrayList ipAndWeights = new ArrayList<>();
+ for (IpAndLoadCount ipAndLoadCount : ipAndLoadCounts) {
+ IpAndWeight ipAndWeight = new IpAndWeight(ipAndLoadCount.getIp(), (80 - ipAndLoadCount.getLoadCount()) * 100 / emptyLoadCount);
+
+ ipAndWeights.add(ipAndWeight);
+ }
+
+ log.info(ipAndWeights);//[IpAndWeight(nodeIp=47.102.158.233, weight=55), IpAndWeight(nodeIp=47.102.123.209, weight=44)]
+
+
+ //*******************************************以下为改老师的代码
+ ArrayList loadNodeList = new ArrayList<>();
+
+ int sum = ipAndWeights.stream().mapToInt(IpAndWeight::getWeight).sum();
+ if (sum < 100) {
+ List list = ipAndWeights.stream().sorted(((o1, o2) -> o2.getWeight() - o1.getWeight())).toList();
+
+ //给权重高的节点 权重再加一个
+ int countWeight = 0;
+ for (int i = sum; i < 100; i++) {
+ IpAndWeight ipAndWeight = list.get(countWeight++ % list.size());
+ ipAndWeight.setWeight(ipAndWeight.getWeight() + 1);
+ }
+ }
+
+ whFor:
+ while (true) {
+ for (IpAndWeight ipAndWeight : ipAndWeights) {
+ Integer weight = ipAndWeight.getWeight();
+ if (weight > 0) {
+ loadNodeList.add(ipAndWeight.getNodeIp());
+ }
+ ipAndWeight.setWeight(weight - 1);
+ }
+
+ int sum1 = ipAndWeights.stream()
+ .mapToInt(IpAndWeight::getWeight)
+ .sum();
+ if (sum1 <= 0) {
+ break whFor;
+ }
+ }
+
+ //节点IP列表存入缓存
+ redis.deleteObject("work:node:gateway");
+ redis.setCacheList("work:node:gateway", loadNodeList);
+
+
+
+ //获取缓存里最后一个IP进行返回
+ //最后一个IP进行返回
+ String result = loadNodeList.get(loadNodeList.size() - 1);
+
+ return Result.success(result);
+ }
+
+}
+
+
diff --git a/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports b/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports
new file mode 100644
index 0000000..3746c28
--- /dev/null
+++ b/src/main/resources/META-INF/spring/org.springframework.boot.autoconfigure.AutoConfiguration.imports
@@ -0,0 +1,2 @@
+com.lyh.common.redis.configure.RedisConfig
+com.lyh.common.redis.service.RedisService
diff --git a/src/main/resources/application.yml b/src/main/resources/application.yml
new file mode 100644
index 0000000..06e942a
--- /dev/null
+++ b/src/main/resources/application.yml
@@ -0,0 +1,13 @@
+server:
+ port: 9958
+
+
+spring:
+ redis:
+ # host: 39.103.132.68
+ host: 127.0.0.1
+ port: 6379
+
+
+
+
diff --git a/src/main/resources/banner.txt b/src/main/resources/banner.txt
new file mode 100644
index 0000000..9e37143
--- /dev/null
+++ b/src/main/resources/banner.txt
@@ -0,0 +1,15 @@
+ ,--,
+,---.'| ,--,
+| | : ,--.'|
+: : | ,--, ,---, ,--, | :
+| ' : ,--.'| ,--, /_ ./| ,--, ,---, ,---.'| : ' ,--,
+; ; ' | |, ,'_ /| ,---, | ' : ,'_ /| ,-+-. / || | : _' | ,'_ /|
+' | |__ `--'_ .--. | | :/___/ \. : | .--. | | : ,--.'|' |: : |.' | .--. | | :
+| | :.'|,' ,'| ,'_ /| : . | . \ \ ,' ','_ /| : . || | ,"' || ' ' ; :,'_ /| : . |
+' : ;' | | | ' | | . . \ ; ` ,'| ' | | . .| | / | |' | .'. || ' | | . .
+| | ./ | | : | | ' | | | \ \ ' | | ' | | || | | | || | : | '| | ' | | |
+; : ; ' : |__ : | : ; ; | ' \ | : | : ; ; || | | |/ ' : | : ;: | : ; ; |
+| ,/ | | '.'|' : `--' \ \ ; ; ' : `--' \ | |--' | | ' ,/ ' : `--' \
+'---' ; : ;: , .-./ : \ \: , .-./ |/ ; : ;--' : , .-./
+ | , / `--`----' \ ' ; `--`----' '---' | ,/ `--`----'
+ ---`-' `--` '---'
diff --git a/src/test/java/LoadTest.java b/src/test/java/LoadTest.java
new file mode 100644
index 0000000..7b88ca0
--- /dev/null
+++ b/src/test/java/LoadTest.java
@@ -0,0 +1,172 @@
+import com.lyh.LoadCenterApplication;
+import com.lyh.common.redis.service.RedisService;
+import lombok.AllArgsConstructor;
+import lombok.Data;
+import lombok.NoArgsConstructor;
+import lombok.extern.log4j.Log4j2;
+import org.junit.jupiter.api.Test;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.boot.test.context.SpringBootTest;
+
+import java.util.ArrayList;
+import java.util.HashMap;
+import java.util.List;
+import java.util.Map;
+import java.util.concurrent.CountDownLatch;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/14
+ * @Description: 负载测试
+ */
+
+@SpringBootTest(classes = LoadCenterApplication.class)
+@Log4j2
+public class LoadTest {
+ @Autowired
+ private RedisService redisService;
+
+ @Test
+ public void load() {
+ //初始化序列
+ redisService.setCacheObject("cursor", 0);
+ //初始化缓存
+ redisService.deleteObject("work:node:gateway");
+
+ //模拟三台节点服务器 的ID(或IP) 权重值
+ ArrayList nodeIdList = new ArrayList<>() {{
+ add(new WorkGatewayNode("node-A", 8));
+ add(new WorkGatewayNode("node-B", 12));
+ add(new WorkGatewayNode("node-C", 2));
+ add(new WorkGatewayNode("node-D", 39));
+ add(new WorkGatewayNode("node-E", 39));
+ }};
+ //100
+ ArrayList loadNodeList = new ArrayList<>();
+
+ long count = nodeIdList.stream().mapToInt(WorkGatewayNode::getWeight).sum();//求和
+ if (count < 100) {
+ List list = nodeIdList.stream().sorted((o1, o2) -> o2.getWeight() - o1.getWeight()).toList();
+
+ int countWeight = 0;//给权重高的节点 权重再加一个
+ for (long i = count; i < 100; i++) {
+ WorkGatewayNode workGatewayNode = list.get(countWeight++ % list.size());
+ workGatewayNode.setWeight(workGatewayNode.getWeight() + 1);
+ }
+
+ }
+
+
+ whFor:
+ while (true) {
+ for (WorkGatewayNode workGatewayNode : nodeIdList) {
+ int weight = workGatewayNode.getWeight();
+ if (weight > 0) {
+ loadNodeList.add(
+ workGatewayNode.getNodeId()
+ );
+ workGatewayNode.setWeight(weight - 1);
+
+ }
+
+ }
+
+ int sum = nodeIdList.stream()
+ .mapToInt(WorkGatewayNode::getWeight)
+ .sum();
+ if (sum <= 0) {
+ break whFor;
+ }
+
+ }
+
+ redisService.setCacheList("work:node:gateway", loadNodeList);
+
+ CountDownLatch countDownLatch = new CountDownLatch(3000);
+ new Thread(() -> {
+ for (int i = 0; i < 1000; i++) {
+ Long cursor = redisService.increment("cursor", 1L);//游标
+ String nodeId = redisService.getCacheList("work:node:gateway", cursor % 100);
+ log.info(Thread.currentThread().getName() + ":" + cursor + "---" + nodeId);
+
+ //每次调通,进行统计
+ stiNode.sti(nodeId);
+
+ countDownLatch.countDown();
+ }
+
+ }).start();
+ new Thread(() -> {
+ for (int i = 0; i < 1000; i++) {
+ Long cursor = redisService.increment("cursor", 1L);//游标
+ String nodeId = redisService.getCacheList("work:node:gateway", cursor % 100);
+ log.info(Thread.currentThread().getName() + ":" + cursor + "---" + nodeId);
+
+ //每次调通,进行统计
+ stiNode.sti(nodeId);
+
+ countDownLatch.countDown();
+ }
+ }).start();
+ new Thread(() -> {
+ for (int i = 0; i < 1000; i++) {
+ Long cursor = redisService.increment("cursor", 1L);//游标
+ String nodeId = redisService.getCacheList("work:node:gateway", cursor % 100);
+ log.info(Thread.currentThread().getName() + ":" + cursor + "---" + nodeId);
+
+ //每次调通,进行统计
+ stiNode.sti(nodeId);
+
+ countDownLatch.countDown();
+ }
+ }).start();
+
+ try {
+ countDownLatch.await();
+ stiNode.show();
+ } catch (InterruptedException e) {
+ throw new RuntimeException(e);
+ }
+
+// for (int i = 0; i < 1000; i++) {
+// Long cursor = redisService.increment("cursor", 1L);//游标
+// String nodeId = redisService.getCacheList("work:node:gateway", cursor % 100);
+// log.info(cursor + "---" + nodeId);
+// }
+
+ log.info(nodeIdList);
+
+ log.info(loadNodeList);
+
+ }
+
+
+}
+
+
+@Log4j2
+class stiNode {
+ private static Map stiNodeMap = new HashMap<>();
+
+ public synchronized static void sti(String nodeId) {
+ Integer stiCount = stiNodeMap.getOrDefault(nodeId, 0);
+ stiNodeMap.put(nodeId, stiCount + 1);
+ }
+
+ public static void show() {
+ stiNodeMap.forEach((key,val)->{
+ log.info(key+"----"+val);
+ });
+ }
+}
+
+
+
+@Data
+@AllArgsConstructor
+@NoArgsConstructor
+class WorkGatewayNode {
+ private String nodeId;
+ private int weight;
+}
diff --git a/src/test/java/Test.java b/src/test/java/Test.java
new file mode 100644
index 0000000..08aaf63
--- /dev/null
+++ b/src/test/java/Test.java
@@ -0,0 +1,65 @@
+import com.aliyun.ecs20140526.models.DescribeInstancesResponseBody;
+import com.lyh.LoadCenterApplication;
+import com.lyh.common.redis.service.RedisService;
+import com.lyh.common.utils.UserUtil;
+import com.lyh.openAPI.WorkCenter.GetInstanceProperties;
+import com.lyh.openAPI.WorkCenter.ReleaseInstance;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.beans.factory.annotation.Autowired;
+import org.springframework.boot.test.context.SpringBootTest;
+
+import java.util.List;
+
+/**
+ * @ProjectName: LoadCenter
+ * @Author: LiuYunHu
+ * @CreateTime: 2024/4/13
+ * @Description: 测试
+ */
+
+//@SpringBootTest
+@SpringBootTest(classes = LoadCenterApplication.class)
+@Slf4j
+public class Test {
+ @Autowired
+ private RedisService redisService;
+
+
+ @org.junit.jupiter.api.Test
+ public void load() throws Exception {
+// //返回实例的ID
+// String instanceId = CreateAndRunInstance.startCreate();
+//
+// Thread.sleep(2000);
+
+ String instanceId = "i-uf6chlqotgoc9h173alu";
+
+ //通过实例ID获取实例的详细属性
+ List describeInstancesResponseBodyInstancesInstances = GetInstanceProperties.startGet(instanceId);
+
+ //获取集合第一个的属性
+ DescribeInstancesResponseBody.DescribeInstancesResponseBodyInstancesInstance response = describeInstancesResponseBodyInstancesInstances.get(0);
+
+ //获取添加的实例的公网ip
+ String instanceIp = UserUtil.removeBrackets(response.getPublicIpAddress().getIpAddress().toString());
+ System.out.println("-----------------------");
+ log.info(instanceIp);
+ System.out.println("-----------------------");
+
+ }
+
+
+ //获取指定节点的负载数
+
+
+ /*
+ * 释放实例
+ * */
+ @org.junit.jupiter.api.Test
+ public void aaa() throws Exception {
+ ReleaseInstance.startRelease("i-uf660jbffjg2mrrs4h91,i-uf6dac58ernhfgpjn6nn");
+
+ }
+
+
+}