我正在尝试实现 PayPal IPN 功能。基本协议是这样的:
- 客户从我的网站重定向到 PayPal 的网站以完成付款。他登录自己的帐户,授权付款。
- PayPal 调用我服务器上的一个页面,以 POST 形式传递详细信息。详细信息包括个人姓名、地址和付款信息等。
- 我需要从处理页面内部调用 PayPal 网站上的 URL,传回上面传递的所有参数以及另一个名为“cmd”、值为“_notify-validate”的参数。
当我尝试对 PayPal 发送给我的参数进行 urllib.urlencode 时,我得到:
While calling send_response_to_paypal. Traceback (most recent call last):
File "<snip>/account/paypal/views.py", line 108, in process_paypal_ipn
verify_result = send_response_to_paypal(params)
File "<snip>/account/paypal/views.py", line 41, in send_response_to_paypal
params = urllib.urlencode(params)
File "/usr/local/lib/python2.6/urllib.py", line 1261, in urlencode
v = quote_plus(str(v))
UnicodeEncodeError: 'ascii' codec can't encode character u'\ufffd' in position 9: ordinal not in range(128)
据我所知,urlencode 进行 ASCII 编码,并且在某些情况下,用户的联系信息可以包含非 ASCII 字符。这是可以理解的。我的问题是,如何使用 urllib2.urlopen(req) (或其他方法)对非 ASCII 字符进行编码以发布到 URL
Details:
我读取PayPal原始请求中的参数如下(GET用于测试):
def read_ipn_params(request):
if request.POST:
params= request.POST.copy()
if "ipn_auth" in request.GET:
params["ipn_auth"]=request.GET["ipn_auth"]
return params
else:
return request.GET.copy()
我用于从处理页面将请求发送回 PayPal 的代码是:
def send_response_to_paypal(params):
params['cmd']='_notify-validate'
params = urllib.urlencode(params)
req = urllib2.Request(PAYPAL_API_WEBSITE, params)
req.add_header("Content-type", "application/x-www-form-urlencoded")
response = urllib2.urlopen(req)
status = response.read()
if not status == "VERIFIED":
logging.warn("PayPal cannot verify IPN responses: " + status)
return False
return True
显然,只有当某人的姓名、地址或用于 PayPal 付款的其他字段不属于 ASCII 范围时,才会出现问题。