alb_utils.py 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580
  1. import logging
  2. import json
  3. import sys
  4. import time
  5. from asyncio import wait_for
  6. import requests
  7. import asyncio
  8. import time
  9. from alibabacloud_tea_util.client import Client as UtilClient
  10. from aliyunsdkcore.client import AcsClient
  11. from aliyunsdkslb.request.v20140515.AddBackendServersRequest import AddBackendServersRequest
  12. from aliyunsdkslb.request.v20140515.RemoveBackendServersRequest import RemoveBackendServersRequest
  13. from aliyunsdkecs.request.v20140526.RunInstancesRequest import RunInstancesRequest
  14. from aliyunsdkecs.request.v20140526.DescribeInstancesRequest import DescribeInstancesRequest
  15. from aliyunsdkecs.request.v20140526.DescribeNetworkInterfacesRequest import DescribeNetworkInterfacesRequest
  16. from aliyunsdkslb.request.v20140515.DescribeLoadBalancerAttributeRequest import DescribeLoadBalancerAttributeRequest
  17. from aliyunsdkecs.request.v20140526.RunCommandRequest import RunCommandRequest
  18. from aliyunsdkecs.request.v20140526.SendFileRequest import SendFileRequest
  19. from aliyunsdkecs.request.v20140526.StopInstancesRequest import StopInstancesRequest
  20. from aliyunsdkecs.request.v20140526.DeleteInstancesRequest import DeleteInstancesRequest
  21. from aliyunsdkecs.request.v20140526.DescribeInstanceStatusRequest import DescribeInstanceStatusRequest
  22. from aliyunsdkcore.request import CommonRequest
  23. from alibabacloud_alb20200616.client import Client as Alb20200616Client
  24. from alibabacloud_tea_openapi import models as open_api_models
  25. from alibabacloud_alb20200616 import models as alb_models
  26. from alibabacloud_alb20200616 import models as alb_20200616_models
  27. from alibabacloud_tea_util import models as util_models
  28. from aliyunsdkalb.request.v20200616.ListServerGroupServersRequest import ListServerGroupServersRequest
  29. from aliyunsdkcore.acs_exception.exceptions import ClientException, ServerException
  30. logging.basicConfig(level=logging.INFO,
  31. format='%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s',
  32. datefmt='%a, %d %b %Y %H:%M:%S')
  33. def send_msg_to_feishu(webhook, key_word, msg_text):
  34. """发送消息到飞书"""
  35. headers = {'Content-Type': 'application/json'}
  36. payload_message = {
  37. "msg_type": "text",
  38. "content": {
  39. "text": '{}: {}'.format(key_word, msg_text)
  40. }
  41. }
  42. response = requests.request('POST', url=webhook, headers=headers, data=json.dumps(payload_message))
  43. logging.info(response.text)
  44. def connect_client(access_key_id, access_key_secret, region_id):
  45. """
  46. 初始化账号,连接客户端
  47. :param access_key_id: access key Id, type-string
  48. :param access_key_secret: access key secret, type-string
  49. :param region_id: region_id
  50. :return: clt
  51. """
  52. try:
  53. clt = AcsClient(ak=access_key_id, secret=access_key_secret, region_id=region_id)
  54. return clt
  55. except Exception as e:
  56. # 失败,记录报错信息,发送通知,停止并退出
  57. logging.error(e)
  58. sys.exit()
  59. def connect_alb_client(access_key_id, access_key_secret, endpoint):
  60. """
  61. 初始化ALB客户端
  62. :param access_key_id: access key Id, type-string
  63. :param access_key_secret: access key secret, type-string
  64. :return: alb_client
  65. """
  66. config = open_api_models.Config(
  67. access_key_id=access_key_id,
  68. access_key_secret=access_key_secret,
  69. endpoint=endpoint
  70. )
  71. alb_client = Alb20200616Client(config)
  72. return alb_client
  73. def build_create_instances_request(image_id, vswitch_id, security_group_id, zone_id, instance_type, instance_name,
  74. disk_size, disk_category, key_pair_name, tags):
  75. """
  76. 购买服务器参数配置
  77. :param image_id: 使用的镜像信息 type-string
  78. :param vswitch_id: 选择的交换机 type-string
  79. :param security_group_id: 当前vpc类型的安全组 type-string
  80. :param zone_id: 服务器所在区域 type-string
  81. :param instance_type: 实例规格 type-string
  82. :param instance_name: 实例命名 type-string
  83. :param disk_size: 磁盘大小,单位:G,type-string
  84. :param disk_category: 磁盘类型 type-string
  85. :param key_pair_name: 密钥对名称 type-string
  86. :param tags: 标签 type-list, eg: [{"Key": "ecs", "Value": "rov-server.prod"}, ...]
  87. :return: request
  88. """
  89. request = RunInstancesRequest()
  90. request.set_ImageId(image_id)
  91. request.set_VSwitchId(vswitch_id)
  92. request.set_SecurityGroupId(security_group_id)
  93. request.set_ZoneId(zone_id)
  94. request.set_InstanceType(instance_type)
  95. request.set_InstanceName(instance_name)
  96. request.set_SystemDiskSize(disk_size)
  97. request.set_SystemDiskCategory(disk_category)
  98. request.set_KeyPairName(key_pair_name)
  99. request.set_Tags(tags)
  100. return request
  101. def send_req(client, request):
  102. """
  103. 发送API请求
  104. :param client: 客户端连接
  105. :param request: 请求配置
  106. :return: response
  107. """
  108. request.set_accept_format('json')
  109. response = client.do_action_with_exception(request)
  110. #print(response)
  111. response = json.loads(response)
  112. print(response)
  113. # logging.info(response)
  114. print(response.get('Code'))
  115. return response
  116. #except Exception as e:
  117. # 失败,记录报错信息,发送通知,停止并退出
  118. #logging.error(e)
  119. #sys.exit()
  120. def check_instance_running(ecs_client, instance_ids):
  121. """
  122. 检查服务器运行状态
  123. :param ecs_client: 客户端连接
  124. :param instance_ids: 实例id列表, type-list
  125. :return: running_count,Status为Running的实例数
  126. """
  127. try:
  128. request = DescribeInstancesRequest()
  129. request.set_InstanceIds(json.dumps(instance_ids))
  130. request.set_PageSize(100)
  131. response = send_request(ecs_client=ecs_client, request=request)
  132. if response.get('Code') is None:
  133. instances_list = response.get('Instances').get('Instance')
  134. running_count = 0
  135. running_instances = []
  136. for instance_detail in instances_list:
  137. if instance_detail.get('Status') == "Running":
  138. running_count += 1
  139. running_instances.append(instance_detail.get('InstanceId'))
  140. return running_count, running_instances
  141. else:
  142. # 失败,记录报错信息,发送通知,停止并退出
  143. logging.error(response)
  144. sys.exit()
  145. except Exception as e:
  146. # 失败,记录报错信息,发送通知,停止并退出
  147. logging.error(e)
  148. sys.exit()
  149. def get_ip_address(ecs_client, instance_id):
  150. """
  151. 获取实例IP地址
  152. :param ecs_client: 客户端连接
  153. :param instance_id: 实例id, type-string
  154. :return: ip_address, type-string
  155. """
  156. request = DescribeNetworkInterfacesRequest()
  157. request.set_accept_format('json')
  158. request.set_InstanceId(instance_id)
  159. response = send_request(ecs_client=ecs_client, request=request)
  160. ip_address = response['NetworkInterfaceSets']['NetworkInterfaceSet'][0]['PrivateIpAddress']
  161. return ip_address
  162. def create_multiple_instances(amount, ecs_client,
  163. image_id, vswitch_id, security_group_id, zone_id, instance_type, instance_name,
  164. disk_size, disk_category, key_pair_name, tags):
  165. """
  166. 创建多个ECS实例
  167. :param amount: 创建实例数 type-int 取值范围:[1, 100]
  168. :param ecs_client: 购买机器客户端连接
  169. :param image_id: 使用的镜像信息 type-string
  170. :param vswitch_id: 选择的交换机 type-string
  171. :param security_group_id: 当前vpc类型的安全组 type-string
  172. :param zone_id: 服务器所在区域 type-string
  173. :param instance_type: 实例规格 type-string
  174. :param instance_name: 实例命名 type-string
  175. :param disk_size: 磁盘大小,单位:G,type-string
  176. :param disk_category: 磁盘类型 type-string
  177. :param key_pair_name: 密钥对名称 type-string
  178. :param tags: 标签 type-list, eg: [{"Key": "ecs", "Value": "rov-server.prod"}, ...]
  179. :return:
  180. """
  181. logging.info(f"create instances start, request amount: {amount}.")
  182. # 1. 连接客户端
  183. # create_instances_clt = connect_client(
  184. # access_key_id=access_key_id, access_key_secret=access_key_secret, region_id=region_id
  185. # )
  186. # 2. 请求参数配置
  187. request = build_create_instances_request(
  188. image_id=image_id, vswitch_id=vswitch_id, security_group_id=security_group_id, zone_id=zone_id,
  189. instance_type=instance_type, instance_name=instance_name, disk_size=disk_size, disk_category=disk_category,
  190. key_pair_name=key_pair_name, tags=tags
  191. )
  192. request.set_Amount(amount)
  193. # 3. 发送API请求,购买机器并启动
  194. response = send_request(ecs_client=ecs_client, request=request)
  195. if response.get('Code') is None:
  196. instance_ids = response.get('InstanceIdSets').get('InstanceIdSet')
  197. logging.info(f"success amount: {len(instance_ids)}, instance ids: {instance_ids}.")
  198. # 获取机器运行状态
  199. running_amount = 0
  200. while running_amount < amount:
  201. time.sleep(10)
  202. running_amount, running_instances = check_instance_running(ecs_client=ecs_client, instance_ids=instance_ids)
  203. logging.info(f"running amount: {running_amount}, running instances: {running_instances}.")
  204. return instance_ids
  205. else:
  206. # 失败,记录报错信息,发送通知,停止并退出
  207. logging.error(response)
  208. sys.exit()
  209. def release_instances(ecs_client, instance_ids, force=False):
  210. """
  211. 释放实例
  212. :param ecs_client:
  213. :param instance_ids: instance_id, type-list
  214. :param force: 是否强制释放, True-强制释放, False-正常释放, type-bool
  215. :return:
  216. """
  217. request = DeleteInstancesRequest()
  218. request.set_InstanceIds(instance_ids)
  219. request.set_Force(force)
  220. response = send_request(ecs_client=ecs_client, request=request)
  221. return response
  222. def get_instances_status(ecs_client, instance_ids):
  223. """
  224. 获取实例运行状态
  225. :param ecs_client:
  226. :param instance_ids: instance_id, type-liist
  227. :return:
  228. """
  229. request = DescribeInstanceStatusRequest()
  230. request.set_InstanceIds(instance_ids)
  231. request.set_PageSize(50)
  232. response = send_request(ecs_client=ecs_client, request=request)
  233. return response
  234. def stop_instances(ecs_client, instance_ids, force_stop=False):
  235. """
  236. 停止实例
  237. :param ecs_client:
  238. :param instance_ids: 实例ID, type-list
  239. :param force_stop: 是否强制关机, True-强制关机, False-正常关机, type-bool
  240. :return:
  241. """
  242. request = StopInstancesRequest()
  243. request.set_InstanceIds(instance_ids)
  244. request.set_ForceStop(force_stop)
  245. response = send_request(ecs_client=ecs_client, request=request)
  246. return response
  247. def send_request(ecs_client, request):
  248. """
  249. 发送API请求
  250. :param ecs_client: 客户端连接
  251. :param request: 请求配置
  252. :return: response
  253. """
  254. request.set_accept_format('json')
  255. try:
  256. response = ecs_client.do_action_with_exception(request)
  257. response = json.loads(response)
  258. # logging.info(response)
  259. return response
  260. except Exception as e:
  261. # 失败,记录报错信息,发送通知,停止并退出
  262. logging.error(e)
  263. sys.exit()
  264. def run_command(ecs_client, instance_ids, command):
  265. """
  266. 批量执行命令
  267. :param ecs_client: 客户端连接
  268. :param instance_ids: 实例id列表, type-list, 最多能指定50台ECS实例ID
  269. :param command: 命令 type-string
  270. :return:
  271. """
  272. for i in range(len(instance_ids) // 50 + 1):
  273. instance_id_list = instance_ids[i * 50:(i + 1) * 50]
  274. if len(instance_id_list) == 0:
  275. return
  276. request = RunCommandRequest()
  277. request.set_accept_format('json')
  278. request.set_Type("RunShellScript")
  279. request.set_CommandContent(command)
  280. request.set_InstanceIds(instance_id_list)
  281. response = send_request(ecs_client=ecs_client, request=request)
  282. logging.info(response)
  283. def send_file_to_ecs(ecs_client, instance_id_list, target_dir, name, content):
  284. """
  285. 发送文件到ecs;alb应用,区分上方clb
  286. :param ecs_client:
  287. :param instance_id_list: 最多能指定50台ECS实例ID
  288. :param target_dir: 文件存放目录 type-string
  289. :param name: 文件名 type-string
  290. :param content: 文件内容 type-string
  291. :return:
  292. """
  293. if not instance_id_list:
  294. logging.warning("实例ID列表为空,无法发送文件。")
  295. return
  296. for i in range(len(instance_id_list) // 50 + 1):
  297. instance_ids = instance_id_list[i * 50:(i + 1) * 50]
  298. if len(instance_ids) == 0:
  299. logging.info("没有更多的实例ID需要发送文件,退出。")
  300. return
  301. request = SendFileRequest()
  302. request.set_Content(content)
  303. request.set_TargetDir(target_dir)
  304. request.set_Name(name)
  305. request.set_Overwrite(True)
  306. request.set_InstanceIds(instance_ids)
  307. try:
  308. logging.info(f"正在向实例 {instance_ids} 发送文件 '{name}' 到目录 '{target_dir}'")
  309. response = send_request(ecs_client=ecs_client, request=request)
  310. logging.info(f"成功发送文件到实例 {instance_ids},响应: {response}")
  311. except Exception as e:
  312. logging.error(f"发送文件到实例 {instance_ids} 失败,错误: {str(e)}")
  313. def add_servers_to_server_group(alb_client, server_group_id, instance_id, weight):
  314. """
  315. 添加服务器到ALB服务器组
  316. :param alb_client: ALB客户端连接
  317. :param server_group_id: 服务器组ID
  318. :param instance_id: 实例ID
  319. :param weight: 权重
  320. """
  321. server = alb_models.AddServersToServerGroupRequestServers(
  322. port=80,
  323. server_id=instance_id,
  324. server_type='ecs',
  325. weight=weight
  326. )
  327. request = alb_models.AddServersToServerGroupRequest(
  328. server_group_id=server_group_id,
  329. servers=[server]
  330. )
  331. runtime = util_models.RuntimeOptions()
  332. try:
  333. alb_client.add_servers_to_server_group_with_options(request, runtime)
  334. logging.info(f"Successfully added server {instance_id} to server group {server_group_id} with weight {weight}.")
  335. except Exception as e:
  336. logging.error(f"Failed to add server {instance_id} to server group {server_group_id}: {str(e)}")
  337. def remove_servers_from_server_group(alb_client, server_group_id, instance_id):
  338. """
  339. 从ALB服务器组中移除服务器
  340. :param alb_client: ALB客户端连接
  341. :param server_group_id: 服务器组ID
  342. :param instance_id: 实例ID
  343. """
  344. server = alb_models.RemoveServersFromServerGroupRequestServers(
  345. port=80,
  346. server_id=instance_id,
  347. server_type='ecs'
  348. )
  349. request = alb_models.RemoveServersFromServerGroupRequest(
  350. server_group_id=server_group_id,
  351. servers=[server]
  352. )
  353. runtime = util_models.RuntimeOptions()
  354. try:
  355. alb_client.remove_servers_from_server_group_with_options(request, runtime)
  356. logging.info(f"Successfully removed server {instance_id} from server group {server_group_id}.")
  357. except Exception as e:
  358. logging.error(f"Failed to remove server {instance_id} from server group {server_group_id}: {str(e)}")
  359. #
  360. # def set_weight_for_instances_alb(alb_client, server_group_id, instance_id_list, weights):
  361. # """
  362. # 设置ALB服务器组中实例的权重
  363. # :param alb_client: ALB客户端连接
  364. # :param server_group_id: 服务器组ID
  365. # :param instance_id_list: 实例ID列表
  366. # :param weights: 权重列表
  367. # :return: None
  368. # """
  369. # servers = []
  370. # for instance_id, weight in zip(instance_id_list, weights):
  371. # server = alb_models.AddServersToServerGroupRequestServers( # todo update
  372. # port=80, # todo
  373. # server_id=instance_id,
  374. # server_type='ecs',
  375. # weight=weight
  376. # )
  377. # servers.append(server)
  378. #
  379. # request = alb_models.AddServersToServerGroupRequest(
  380. # server_group_id=server_group_id,
  381. # servers=servers
  382. # )
  383. #
  384. # runtime = util_models.RuntimeOptions()
  385. # try:
  386. # alb_client.add_servers_to_server_group_with_options(request, runtime)
  387. # logging.info(f"Successfully set weights for instances in server group {server_group_id}.")
  388. # except Exception as e:
  389. # logging.error(f"Failed to set weights for instances: {str(e)}")
  390. #
  391. #
  392. # def set_instance_weight_process_with_alb(alb_client, server_group_id_list, instance_id_list, weight_list):
  393. # """
  394. # 修改ALB服务器组中实例的权重值
  395. # :param alb_client: ALB客户端连接
  396. # :param server_group_id_list: 服务器组ID列表
  397. # :param instance_id_list: 实例ID列表
  398. # :param weight_list: 权重修改列表 type-list [(weight, sleep_time), ...]
  399. # :return:
  400. # """
  401. # for weight, sleep_time in weight_list:
  402. # logging.info(f"修改权重中: weight = {weight}")
  403. # for server_group_id in server_group_id_list:
  404. # flag = True
  405. # while flag:
  406. # try:
  407. # # 使用新的权重设置函数
  408. # set_weight_for_instances_alb(alb_client, server_group_id, instance_id_list, [weight] * len(instance_id_list))
  409. # logging.info(f"服务器组: {server_group_id} 权重设置完成!")
  410. # flag = False
  411. # except Exception as e:
  412. # logging.error(f"更新权重失败: {e}, 将重试...")
  413. # time.sleep(10)
  414. # continue
  415. # time.sleep(sleep_time)
  416. def list_server_group_servers(alb_client, server_group_id):
  417. """
  418. 列出服务器组中的服务器并返回实例ID列表
  419. @param alb_client: ALB客户端
  420. @param server_group_id: 服务器组ID
  421. @return: 实例ID列表
  422. """
  423. list_server_group_servers_request = alb_20200616_models.ListServerGroupServersRequest(
  424. server_group_id=server_group_id
  425. )
  426. runtime = util_models.RuntimeOptions()
  427. try:
  428. response = alb_client.list_server_group_servers_with_options(list_server_group_servers_request, runtime)
  429. instance_ids = [server.server_id for server in response.body.servers]
  430. if instance_ids:
  431. return instance_ids[0]
  432. else:
  433. return None
  434. # return instance_ids
  435. except Exception as error:
  436. print(error.message)
  437. print(error.data.get("Recommend"))
  438. UtilClient.assert_as_string(error.message)
  439. return []
  440. async def list_server_group_servers_async(alb_client, server_group_id):
  441. """
  442. 异步列出指定服务器组中的服务器并返回实例ID列表
  443. @param alb_client: ALB客户端
  444. @param server_group_id: 服务器组ID
  445. @return: 实例ID列表
  446. """
  447. list_server_group_servers_request = alb_20200616_models.ListServerGroupServersRequest(
  448. server_group_id=server_group_id
  449. )
  450. runtime = util_models.RuntimeOptions()
  451. try:
  452. response = await alb_client.list_server_group_servers_with_options_async(list_server_group_servers_request,
  453. runtime)
  454. instance_ids = [server.server_id for server in response.body.servers]
  455. return instance_ids
  456. except Exception as error:
  457. print(error.message)
  458. print(error.data.get("Recommend"))
  459. UtilClient.assert_as_string(error.message)
  460. return []
  461. def update_server_group_servers_attribute(alb_client, server_group_id_list, instance_id_list, weight_list):
  462. """
  463. 更新服务器组中的服务器权重
  464. :param alb_client: ALB客户端
  465. :param server_group_id_list: 服务器组ID列表
  466. :param instance_id_list: 实例ID列表
  467. :param weight_list: 权重修改列表 type-list [(weight, sleep_time), ...]
  468. """
  469. for server_group_id in server_group_id_list:
  470. for instance_id in instance_id_list:
  471. for weight, sleep_time in weight_list:
  472. server = alb_20200616_models.UpdateServerGroupServersAttributeRequestServers(
  473. server_type='Ecs',
  474. server_id=instance_id,
  475. weight=weight,
  476. port=80
  477. )
  478. request = alb_20200616_models.UpdateServerGroupServersAttributeRequest(
  479. servers=[server],
  480. server_group_id=server_group_id
  481. )
  482. runtime = util_models.RuntimeOptions()
  483. try:
  484. alb_client.update_server_group_servers_attribute_with_options(request, runtime)
  485. print(f"Successfully updated server {instance_id} in group {server_group_id} to weight {weight}.")
  486. except Exception as error:
  487. print(error.message)
  488. print(error.data.get("Recommend"))
  489. UtilClient.assert_as_string(error.message)
  490. time.sleep(sleep_time)
  491. async def update_server_group_servers_attribute_async(alb_client, server_group_id_list, instance_ids, weight_list):
  492. """
  493. 异步更新服务器组中的服务器属性
  494. :param alb_client: ALB客户端
  495. :param server_group_id_list: 服务器组ID列表
  496. :param instance_ids: 实例ID列表
  497. :param weight_list: 权重修改列表 type-list [(weight, sleep_time), ...]
  498. """
  499. for server_group_id in server_group_id_list:
  500. for instance_id in instance_ids:
  501. for weight, sleep_time in weight_list:
  502. server = alb_20200616_models.UpdateServerGroupServersAttributeRequestServers(
  503. server_type='Ecs',
  504. server_id=instance_id,
  505. weight=weight,
  506. port=80
  507. )
  508. request = alb_20200616_models.UpdateServerGroupServersAttributeRequest(
  509. servers=[server],
  510. server_group_id=server_group_id
  511. )
  512. runtime = util_models.RuntimeOptions()
  513. try:
  514. await alb_client.update_server_group_servers_attribute_with_options_async(request, runtime)
  515. print(f"Successfully updated server {instance_id} in group {server_group_id} to weight {weight} asynchronously.")
  516. except Exception as error:
  517. print(error.message)
  518. print(error.data.get("Recommend"))
  519. UtilClient.assert_as_string(error.message)
  520. await asyncio.sleep(sleep_time)