utils.py 20 KB

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