utils.py 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. import logging
  2. import json
  3. import sys
  4. import time
  5. import requests
  6. from aliyunsdkcore.client import AcsClient
  7. from aliyunsdkecs.request.v20140526.RunInstancesRequest import RunInstancesRequest
  8. from aliyunsdkecs.request.v20140526.DescribeInstancesRequest import DescribeInstancesRequest
  9. from aliyunsdkecs.request.v20140526.DescribeNetworkInterfacesRequest import DescribeNetworkInterfacesRequest
  10. from aliyunsdkslb.request.v20140515.DescribeLoadBalancerAttributeRequest import DescribeLoadBalancerAttributeRequest
  11. from aliyunsdkecs.request.v20140526.RunCommandRequest import RunCommandRequest
  12. from aliyunsdkecs.request.v20140526.SendFileRequest import SendFileRequest
  13. from aliyunsdkecs.request.v20140526.StopInstancesRequest import StopInstancesRequest
  14. from aliyunsdkecs.request.v20140526.DeleteInstancesRequest import DeleteInstancesRequest
  15. from aliyunsdkecs.request.v20140526.DescribeInstanceStatusRequest import DescribeInstanceStatusRequest
  16. from aliyunsdkcore.request import CommonRequest
  17. logging.basicConfig(level=logging.INFO,
  18. format='%(asctime)s %(filename)s[line:%(lineno)d] %(levelname)s %(message)s',
  19. datefmt='%a, %d %b %Y %H:%M:%S')
  20. def send_msg_to_feishu(webhook, key_word, msg_text):
  21. """发送消息到飞书"""
  22. headers = {'Content-Type': 'application/json'}
  23. payload_message = {
  24. "msg_type": "text",
  25. "content": {
  26. "text": '{}: {}'.format(key_word, msg_text)
  27. }
  28. }
  29. response = requests.request('POST', url=webhook, headers=headers, data=json.dumps(payload_message))
  30. logging.info(response.text)
  31. def connect_client(access_key_id, access_key_secret, region_id):
  32. """
  33. 初始化账号,连接客户端
  34. :param access_key_id: access key Id, type-string
  35. :param access_key_secret: access key secret, type-string
  36. :param region_id: region_id
  37. :return: clt
  38. """
  39. try:
  40. clt = AcsClient(ak=access_key_id, secret=access_key_secret, region_id=region_id)
  41. return clt
  42. except Exception as e:
  43. # 失败,记录报错信息,发送通知,停止并退出
  44. logging.error(e)
  45. sys.exit()
  46. def build_create_instances_request(image_id, vswitch_id, security_group_id, zone_id, instance_type, instance_name,
  47. disk_size, disk_category, key_pair_name, tags):
  48. """
  49. 购买服务器参数配置
  50. :param image_id: 使用的镜像信息 type-string
  51. :param vswitch_id: 选择的交换机 type-string
  52. :param security_group_id: 当前vpc类型的安全组 type-string
  53. :param zone_id: 服务器所在区域 type-string
  54. :param instance_type: 实例规格 type-string
  55. :param instance_name: 实例命名 type-string
  56. :param disk_size: 磁盘大小,单位:G,type-string
  57. :param disk_category: 磁盘类型 type-string
  58. :param key_pair_name: 密钥对名称 type-string
  59. :param tags: 标签 type-list, eg: [{"Key": "ecs", "Value": "rov-server.prod"}, ...]
  60. :return: request
  61. """
  62. request = RunInstancesRequest()
  63. request.set_ImageId(image_id)
  64. request.set_VSwitchId(vswitch_id)
  65. request.set_SecurityGroupId(security_group_id)
  66. request.set_ZoneId(zone_id)
  67. request.set_InstanceType(instance_type)
  68. request.set_InstanceName(instance_name)
  69. request.set_SystemDiskSize(disk_size)
  70. request.set_SystemDiskCategory(disk_category)
  71. request.set_KeyPairName(key_pair_name)
  72. request.set_Tags(tags)
  73. return request
  74. def send_request(client, request):
  75. """
  76. 发送API请求
  77. :param client: 客户端连接
  78. :param request: 请求配置
  79. :return: response
  80. """
  81. request.set_accept_format('json')
  82. try:
  83. response = client.do_action_with_exception(request)
  84. response = json.loads(response)
  85. logging.info(response)
  86. return response
  87. except Exception as e:
  88. # 失败,记录报错信息,发送通知,停止并退出
  89. logging.error(e)
  90. sys.exit()
  91. def check_instance_running(client, instance_ids):
  92. """
  93. 检查服务器运行状态
  94. :param client: 客户端连接
  95. :param instance_ids: 实例id列表, type-list
  96. :return: running_count,Status为Running的实例数
  97. """
  98. try:
  99. request = DescribeInstancesRequest()
  100. request.set_InstanceIds(json.dumps(instance_ids))
  101. response = send_request(client=client, request=request)
  102. if response.get('Code') is None:
  103. instances_list = response.get('Instances').get('Instance')
  104. running_count = 0
  105. running_instances = []
  106. for instance_detail in instances_list:
  107. if instance_detail.get('Status') == "Running":
  108. running_count += 1
  109. running_instances.append(instance_detail.get('InstanceId'))
  110. return running_count, running_instances
  111. else:
  112. # 失败,记录报错信息,发送通知,停止并退出
  113. logging.error(response)
  114. sys.exit()
  115. except Exception as e:
  116. # 失败,记录报错信息,发送通知,停止并退出
  117. logging.error(e)
  118. sys.exit()
  119. def create_multiple_instances(amount, client,
  120. image_id, vswitch_id, security_group_id, zone_id, instance_type, instance_name,
  121. disk_size, disk_category, key_pair_name, tags):
  122. """
  123. 创建多个ECS实例
  124. :param amount: 创建实例数 type-int 取值范围:[1, 100]
  125. :param client: 购买机器客户端连接
  126. :param image_id: 使用的镜像信息 type-string
  127. :param vswitch_id: 选择的交换机 type-string
  128. :param security_group_id: 当前vpc类型的安全组 type-string
  129. :param zone_id: 服务器所在区域 type-string
  130. :param instance_type: 实例规格 type-string
  131. :param instance_name: 实例命名 type-string
  132. :param disk_size: 磁盘大小,单位:G,type-string
  133. :param disk_category: 磁盘类型 type-string
  134. :param key_pair_name: 密钥对名称 type-string
  135. :param tags: 标签 type-list, eg: [{"Key": "ecs", "Value": "rov-server.prod"}, ...]
  136. :return:
  137. """
  138. logging.info(f"create instances start, request amount: {amount}.")
  139. # 1. 连接客户端
  140. # create_instances_clt = connect_client(
  141. # access_key_id=access_key_id, access_key_secret=access_key_secret, region_id=region_id
  142. # )
  143. # 2. 请求参数配置
  144. request = build_create_instances_request(
  145. image_id=image_id, vswitch_id=vswitch_id, security_group_id=security_group_id, zone_id=zone_id,
  146. instance_type=instance_type, instance_name=instance_name, disk_size=disk_size, disk_category=disk_category,
  147. key_pair_name=key_pair_name, tags=tags
  148. )
  149. request.set_Amount(amount)
  150. # 3. 发送API请求,购买机器并启动
  151. response = send_request(client=client, request=request)
  152. if response.get('Code') is None:
  153. instance_ids = response.get('InstanceIdSets').get('InstanceIdSet')
  154. logging.info(f"success amount: {len(instance_ids)}, instance ids: {instance_ids}.")
  155. # 获取机器运行状态
  156. running_amount = 0
  157. while running_amount < amount:
  158. time.sleep(10)
  159. running_amount, running_instances = check_instance_running(client=client, instance_ids=instance_ids)
  160. logging.info(f"running amount: {running_amount}, running instances: {running_instances}.")
  161. return instance_ids
  162. else:
  163. # 失败,记录报错信息,发送通知,停止并退出
  164. logging.error(response)
  165. sys.exit()
  166. def run_command(client, instance_ids, command):
  167. """
  168. 批量执行命令
  169. :param client: 客户端连接
  170. :param instance_ids: 实例id列表, type-list
  171. :param command: 命令 type-string
  172. :return:
  173. """
  174. request = RunCommandRequest()
  175. request.set_accept_format('json')
  176. request.set_Type("RunShellScript")
  177. request.set_CommandContent(command)
  178. request.set_InstanceIds(instance_ids)
  179. response = send_request(client=client, request=request)
  180. logging.info(response)
  181. def get_instance_ids(client, slb_id):
  182. """
  183. 获取slb下所有服务器instanceId
  184. :param client: 客户端连接
  185. :param slb_id: 负载均衡id type-string
  186. :return: instance_ids type-list
  187. """
  188. request = DescribeLoadBalancerAttributeRequest()
  189. request.set_accept_format('json')
  190. request.set_LoadBalancerId(slb_id)
  191. response = send_request(client=client, request=request)
  192. instance_ids = [instance["ServerId"] for instance in response["BackendServers"]["BackendServer"]]
  193. return instance_ids
  194. def get_ip_address(client, instance_id):
  195. """
  196. 获取实例IP地址
  197. :param client: 客户端连接
  198. :param instance_id: 实例id, type-string
  199. :return: ip_address, type-string
  200. """
  201. request = DescribeNetworkInterfacesRequest()
  202. request.set_accept_format('json')
  203. request.set_InstanceId(instance_id)
  204. response = send_request(client=client, request=request)
  205. ip_address = response['NetworkInterfaceSets']['NetworkInterfaceSet'][0]['PrivateIpAddress']
  206. return ip_address
  207. def set_weight_for_instances(client, slb_id, instance_id_list, weight):
  208. """
  209. 同时设置多台服务器的slb权重,权重一样
  210. :param client: 客户端连接
  211. :param slb_id: slb_id
  212. :param instance_id_list: 服务器id list
  213. :param weight: 权重值
  214. :return: None
  215. """
  216. BackendServers = [{"ServerId": instance_id, "Weight": weight} for instance_id in instance_id_list]
  217. request = CommonRequest()
  218. request.set_accept_format('json')
  219. request.set_domain('slb.aliyuncs.com')
  220. request.set_version('2014-05-15')
  221. request.set_method('POST')
  222. request.set_action_name('SetBackendServers')
  223. request.add_query_param('BackendServers', BackendServers)
  224. request.add_query_param('LoadBalancerId', slb_id)
  225. response = send_request(client=client, request=request)
  226. return response
  227. def send_file_to_ecs(client, instance_id_list, target_dir, name, content):
  228. """
  229. 发送文件到ecs
  230. :param client:
  231. :param instance_id_list:
  232. :param target_dir: 文件存放目录 type-string
  233. :param name: 文件名 type-string
  234. :param content: 文件内容 type-string
  235. :return:
  236. """
  237. request = SendFileRequest()
  238. request.set_Content(content)
  239. request.set_TargetDir(target_dir)
  240. request.set_Name(name)
  241. request.set_Overwrite(True)
  242. request.set_InstanceIds(instance_id_list)
  243. response = send_request(client=client, request=request)
  244. return response
  245. def stop_instances(client, instance_ids, force_stop=False):
  246. """
  247. 停止实例
  248. :param client:
  249. :param instance_ids: 实例ID, type-list
  250. :param force_stop: 是否强制关机, True-强制关机, False-正常关机, type-bool
  251. :return:
  252. """
  253. request = StopInstancesRequest()
  254. request.set_InstanceIds(instance_ids)
  255. request.set_ForceStop(force_stop)
  256. response = send_request(client=client, request=request)
  257. return response
  258. def release_instances(client, instance_ids, force=False):
  259. """
  260. 释放实例
  261. :param client:
  262. :param instance_ids: instance_id, type-list
  263. :param force: 是否强制释放, True-强制释放, False-正常释放, type-bool
  264. :return:
  265. """
  266. request = DeleteInstancesRequest()
  267. request.set_InstanceIds(instance_ids)
  268. request.set_Force(force)
  269. response = send_request(client=client, request=request)
  270. return response
  271. def get_instances_status(client, instance_ids):
  272. """
  273. 获取实例运行状态
  274. :param client:
  275. :param instance_ids: instance_id, type-liist
  276. :return:
  277. """
  278. request = DescribeInstanceStatusRequest()
  279. request.set_InstanceIds(instance_ids)
  280. response = send_request(client=client, request=request)
  281. return response