alydns.py 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. # coding:utf-8
  2. import base64
  3. import urllib
  4. import hmac
  5. import pytz
  6. import datetime
  7. import random
  8. import string
  9. import json
  10. import sys
  11. pv = "python2"
  12. #python2
  13. if sys.version_info[0] < 3:
  14. from urllib import quote
  15. from urllib import urlencode
  16. import hashlib
  17. else:
  18. from urllib.parse import quote
  19. from urllib.parse import urlencode
  20. from urllib import request
  21. pv = "python3"
  22. class AliDns:
  23. def __init__(self, access_key_id, access_key_secret, domain_name):
  24. self.access_key_id = access_key_id
  25. self.access_key_secret = access_key_secret
  26. self.domain_name = domain_name
  27. @staticmethod
  28. def generate_random_str(length=14):
  29. """
  30. 生成一个指定长度(默认14位)的随机数值,其中
  31. string.digits = "0123456789'
  32. """
  33. str_list = [random.choice(string.digits) for i in range(length)]
  34. random_str = ''.join(str_list)
  35. return random_str
  36. @staticmethod
  37. def percent_encode(str):
  38. res = quote(str.encode('utf-8'), '')
  39. res = res.replace('+', '%20')
  40. res = res.replace('*', '%2A')
  41. res = res.replace('%7E', '~')
  42. return res
  43. @staticmethod
  44. def utc_time():
  45. """
  46. 请求的时间戳。日期格式按照ISO8601标准表示,
  47. 并需要使用UTC时间。格式为YYYY-MM-DDThh:mm:ssZ
  48. 例如,2015-01-09T12:00:00Z(为UTC时间2015年1月9日12点0分0秒)
  49. :return:
  50. """
  51. utc_tz = pytz.timezone('UTC')
  52. time = datetime.datetime.now(tz=utc_tz).strftime('%Y-%m-%dT%H:%M:%SZ')
  53. return time
  54. @staticmethod
  55. def sign_string(url_param):
  56. percent_encode = AliDns.percent_encode
  57. sorted_url_param = sorted(url_param.items(), key=lambda x: x[0])
  58. can_string = ''
  59. for k, v in sorted_url_param:
  60. can_string += '&' + percent_encode(k) + '=' + percent_encode(v)
  61. string_to_sign = 'GET' + '&' + '%2F' + '&' + percent_encode(can_string[1:])
  62. return string_to_sign
  63. @staticmethod
  64. def access_url(url):
  65. if pv == "python2" :
  66. f = urllib.urlopen(url)
  67. result = f.read().decode('utf-8')
  68. #print(result)
  69. return json.loads(result)
  70. else :
  71. req = request.Request(url)
  72. with request.urlopen(req) as f:
  73. result = f.read().decode('utf-8')
  74. #print(result)
  75. return json.loads(result)
  76. def visit_url(self, action_param):
  77. common_param = {
  78. 'Format': 'json',
  79. 'Version': '2015-01-09',
  80. 'AccessKeyId': self.access_key_id,
  81. 'SignatureMethod': 'HMAC-SHA1',
  82. 'Timestamp': AliDns.utc_time(),
  83. 'SignatureVersion': '1.0',
  84. 'SignatureNonce': AliDns.generate_random_str(),
  85. 'DomainName': self.domain_name,
  86. }
  87. url_param = dict(common_param, **action_param)
  88. string_to_sign = AliDns.sign_string(url_param)
  89. hash_bytes = self.access_key_secret + "&"
  90. if pv == "python2":
  91. h = hmac.new(hash_bytes, string_to_sign, digestmod=hashlib.sha1)
  92. else :
  93. h = hmac.new(hash_bytes.encode('utf-8'), string_to_sign.encode('utf-8'), digestmod='SHA1')
  94. signature = base64.encodestring(h.digest()).strip()
  95. url_param.setdefault('Signature', signature)
  96. url = 'https://alidns.aliyuncs.com/?' + urlencode(url_param)
  97. #print(url)
  98. return AliDns.access_url(url)
  99. # 显示所有
  100. def describe_domain_records(self):
  101. """
  102. 最多只能查询此域名的 500条解析记录
  103. PageNumber 当前页数,起始值为1,默认为1
  104. PageSize 分页查询时设置的每页行数,最大值500,默认为20
  105. :return:
  106. """
  107. action_param = dict(
  108. Action='DescribeDomainRecords',
  109. PageNumber='1',
  110. PageSize='500',
  111. )
  112. result = self.visit_url(action_param)
  113. return result
  114. # 增加解析
  115. def add_domain_record(self, type, rr, value):
  116. action_param = dict(
  117. Action='AddDomainRecord',
  118. RR=rr,
  119. Type=type,
  120. Value=value,
  121. )
  122. result = self.visit_url(action_param)
  123. return result
  124. # 修改解析
  125. def update_domain_record(self, id, type, rr, value):
  126. action_param = dict(
  127. Action="UpdateDomainRecord",
  128. RecordId=id,
  129. RR=rr,
  130. Type=type,
  131. Value=value,
  132. )
  133. result = self.visit_url(action_param)
  134. return result
  135. # 删除解析
  136. def delete_domain_record(self, id):
  137. action_param = dict(
  138. Action="DeleteDomainRecord",
  139. RecordId=id,
  140. )
  141. result = self.visit_url(action_param)
  142. return result
  143. if __name__ == '__main__':
  144. #filename,ACCESS_KEY_ID, ACCESS_KEY_SECRET = sys.argv
  145. #domain = AliDns(ACCESS_KEY_ID, ACCESS_KEY_SECRET, 'simplehttps.com')
  146. #domain.describe_domain_records()
  147. #增加记录
  148. #print(domain.add_domain_record("TXT", "test", "test"))
  149. # 修改解析
  150. #domain.update_domain_record('4011918010876928', 'TXT', 'test2', 'text2')
  151. # 删除解析记录
  152. # data = domain.describe_domain_records()
  153. # record_list = data["DomainRecords"]["Record"]
  154. # for item in record_list:
  155. # if 'test' in item['RR']:
  156. # domain.delete_domain_record(item['RecordId'])
  157. # 第一个参数是 action,代表 (add/clean)
  158. # 第二个参数是域名
  159. # 第三个参数是主机名(第三个参数+第二个参数组合起来就是要添加的 TXT 记录)
  160. # 第四个参数是 TXT 记录值
  161. # 第五个参数是 APPKEY
  162. # 第六个参数是 APPTOKEN
  163. #sys.exit(0)
  164. print(sys.argv)
  165. file_name, cmd ,certbot_domain, acme_challenge, certbot_validation,ACCESS_KEY_ID, ACCESS_KEY_SECRET = sys.argv
  166. domain = AliDns(ACCESS_KEY_ID, ACCESS_KEY_SECRET, certbot_domain)
  167. if cmd == "add":
  168. print(domain.add_domain_record("TXT", acme_challenge, certbot_validation))
  169. elif cmd == "clean":
  170. data = domain.describe_domain_records()
  171. record_list = data["DomainRecords"]["Record"]
  172. if record_list:
  173. for item in record_list:
  174. if (item['RR'] == acme_challenge and item['Value'] == certbot_validation):
  175. domain.delete_domain_record(item['RecordId'])