基于Python的阿里云API签名算法及邮件推送服务
更新时间:2018-10-06 分类:网络技术 浏览量:2237
因为TalkingCoder的服务都是部署在阿里云ECS上的,而之前一直用SMTP来发邮件遇到种种问题,正好最近阿里云在推邮件推送服务,就尝试把它迁移过去。阿里云的推送速度、数量和监控会更好一点。
邮件推送服务其实就是一个简单的API调用,但在和Celery集成过程中,却遇到几个很头疼的小问题。下面一一说明我遇到的一些坑。
签名算法
阿里云有提供SDK,但是签名算法目前只有JAVA、PHP、C#支持,其他需要自己写。在查了一些资料后, 基于python2.6/7 的Aliyun(阿里云) API的简单使用,最终实现了邮件推送。
先看一下这个核心类:
# coding=utf-8
import base64
import hmac
from hashlib import sha1
import urllib
import time
import uuid
from config.base import ALIYUN_ACCESS_KEY_ID, ALIYUN_ACCESS_KEY_SECRET
class AliyunMonitor:
def __init__(self, url):
self.access_id = ALIYUN_ACCESS_KEY_ID
self.access_secret = ALIYUN_ACCESS_KEY_SECRET
self.url = url
# 签名
def sign(self, accessKeySecret, parameters):
sortedParameters = sorted(parameters.items(), key=lambda parameters: parameters[0])
canonicalizedQueryString = ''
for (k, v) in sortedParameters:
canonicalizedQueryString += '&' + self.percent_encode(k) + '=' + self.percent_encode(v)
stringToSign = 'GET&%2F&' + self.percent_encode(canonicalizedQueryString[1:]) # 使用get请求方法
h = hmac.new(accessKeySecret + "&", stringToSign, sha1)
signature = base64.encodestring(h.digest()).strip()
return signature
def percent_encode(self, encodeStr):
encodeStr = str(encodeStr)
# 下面这行挺坑的,使用上面文章中的方法会在某些情况下报错,后面详细说明
res = urllib.quote(encodeStr.decode('utf-8').encode('utf-8'), '')
res = res.replace('+', '%20')
res = res.replace('*', '%2A')
res = res.replace('%7E', '~')
return res
def make_url(self, params):
timestamp = time.strftime("%Y-%m-%dT%H:%M:%SZ", time.gmtime())
parameters = {
'Format': 'JSON',
'Version': '2015-11-23',
'AccessKeyId': self.access_id,
'SignatureVersion': '1.0',
'SignatureMethod': 'HMAC-SHA1',
'SignatureNonce': str(uuid.uuid1()),
'Timestamp': timestamp,
}
for key in params.keys():
parameters[key] = params[key]
signature = self.sign(self.access_secret, parameters)
parameters['Signature'] = signature
# return parameters
url = self.url + "/?" + urllib.urlencode(parameters)
return url
然后,我们写一个send_email的方法来调用:
# coding=utf-8
import requests
from lib.aliyun_monitor import AliyunMonitor
def send_email(email_address, subject, text):
payload = {
'Action': 'SingleSendMail',
'AccountName': 'mail@mail.xxx.com',
'ReplyToAddress': 'true',
'AddressType': 0,
'ToAddress': email_address,
'FromAlias': 'TalkingCoder',
'Subject': subject,
'HtmlBody': text
}
aliyun = AliyunMonitor("http://dm.aliyuncs.com")
url = aliyun.make_url(payload)
request = requests.get(url)
print request.text
send_email('test@test.com', '标题', '内容')
这里使用了requests来get请求。
注意
代码写到这里,在本机环境测试都OK的,然后就往Celery上集成。这里要说明一下,因为之前是用SMTP来发邮件,平均发一封2秒左右吧,这样同步发的话是一件很恐怖的事情,所以就丢给Celery异步去执行任务了。现在集成了阿里的云邮件服务后,其实是可以同步发了,因为本身一个API请求也很快,但Celery闲着也是闲着,为什么不用起来呢,所以后面问题就来了。
就是第一段签名代码里写道的,原文中使用的是res = urllib.quote(encodeStr.decode(sys.stdin.encoding).encode('utf8'), ''),这样普通执行任务可以,但是nohup后和使用Celery都会报一个错误,大致意思是decode的第一个参数必须是String,因为当时nohup后就比较难看到Celery的日志,最后在本机模拟相同环境才知道原因,改为了res = urllib.quote(encodeStr.decode('utf-8').encode('utf-8'), '') 之后才可以。
最后,通过环境变量,在开发环境不使用Celery,在生成环境再使用:
# coding=utf-8
from worker.send_email_task import send_email_task
from config import ENV
def push_email(to_email_list, subject, text):
if ENV == 'development':
send_email_task(to_email_list, subject, text)
elif ENV == 'production':
send_email_task.delay(to_email_list, subject, text)
# coding=utf-8
from lib.send_mail_aliyun import send_email
from worker import celery, logger
@celery.task(name='worker.send_email_task', ignore_result=True)
def send_email_task(to_email_list, subject, text):
logger.info('START - SEND EMAIL')
send_email(to_email_list, subject, text)
logger.info('END - SEND EMAIL')
到这里就完了,不过还是希望阿里云尽早出来Python的SDK吧,以便接入更多的服务。
Via:方法博客