最近为App开发消息推送功能,这里我们采用了友盟的消息推送服务,但其后台简陋,可定制化程度低,所以决定接入服务器端API,在自己的服务器上部署一套推送服务。

其中涉及到很多问题,首先要解决的就是与友盟服务器的加密验证问题。

官方示例

友盟官方的加密算法使用Python实现的,核心是MD5算法,如下:

import hashlib
import json
def md5(s):
print s
m = hashlib.md5(s)
return m.hexdigest() appkey = '你的appkey'
app_master_secret = '你的app_master_secret'
timestamp = '你的timestamp'
device_token=''
method = 'POST'
url = 'http://msg.umeng.com/api/send'
params = {'appkey': appkey,
'timestamp': timestamp,
'device_tokens': device_token,
'type': 'unicast',
'payload': {'body': {'ticker': 'Hello World',
'title':'你好',
'text':'来自友盟推送',
'after_open': 'go_app'},
'display_type': 'notification'
}
}
post_body = json.dumps(params)
print post_body
sign = md5('%s%s%s%s' % (method,url,post_body,app_master_secret))

Python程序输出结果为:

561128496d0f7a2acc098eb0ac263bd3

问题描述

简单一看发现很简单,于是使用C#实现了对应的一套算法,然而实际运行结果却大相径庭,让人困惑,如下:

private string GetSign()
{
var param = new
{
appkey = "你的appkey",
timestamp = "你的timestamp",
device_tokens = "",
type = "unicast",
payload = new
{
body = new
{
ticker = "Hello World",
title = "你好",
text = "来自友盟推送",
after_open = "go_app" },
display_type = "notification"
}
};
var app_master_secret = "你的app_master_secret";
var method = "POST";
var url = "http://msg.umeng.com/api/send";
var serializer = JsonSerializer.Create();
var jsonWriter = new System.IO.StringWriter();
serializer.Serialize(jsonWriter, param);
var post_body = jsonWriter.ToString();
var str = method + url + post_body + app_master_secret;
var sign = MD5(str);
return sign;
} private string MD5(string s)
{ var x = new MD5CryptoServiceProvider();
byte[] bs = System.Text.Encoding.UTF8.GetBytes(s);
bs = x.ComputeHash(bs);
var str = new System.Text.StringBuilder();
foreach (byte b in bs)
{
str.Append(b.ToString("x2").ToLower());
}
return str.ToString();
}

运行之后我们发现输出结果为:

244ba68e49004f7b4ccdaf9bae617296

显然这两个结果是不对的,是无法通过友盟服务器验证的,那到底是什么问题导致的呢?

问题分析

首先我们检查了自己的MD5加密算法和文本编码,确认了在使用同样输入字符串的情况下输出是一致的,效果与python中md5.hexdigest()、Java中DigestUtils.md5Hex(String)函数输出的结果是一致的,那么可以先确定我们的MD5算法是对的,如下:

private string MD5(string s)
{ var x = new MD5CryptoServiceProvider();
var bs = System.Text.Encoding.UTF8.GetBytes(s);
bs = x.ComputeHash(bs);
var str = new System.Text.StringBuilder();
foreach (byte b in bs)
{
str.Append(b.ToString("x2").ToLower());
}
return str.ToString();
}

那么剩余的问题就是,是否是我们的输入字符串有问题?于是我分别输出拼接出的输入字符串做了对比,如下:

//Python生成的拼接字符串

POSThttp://msg.umeng.com/api/send{"appkey": "\u4f60\u7684appkey", "timestamp": "\u4f60\u7684timestamp", "device_tokens": "", "type": "unicast", "payload": {"body": {"text": "\u6765\u81ea\u53cb\u76df\u63a8\u9001", "after_open": "go_app", "ticker": "Hello World", "title": "\u4f60\u597d"}, "display_type": "notification"}}你的app_master_secret

//c#生成的拼接字符串

POSThttp://msg.umeng.com/api/send{"appkey":"你的appkey","timestamp":"你的timestamp","device_tokens":"","type":"unicast","payload":{"body":{"ticker":"Hello World","title":"你好","text":"来自友盟推送","after_open":"go_app"},"display_type":"notification"}}你的app_master_secret

仔细对比之后,发现了3点区别:

1.Python生成的字符串post_body部分采用了Unicode编码,而c#则是默认编码

2.Python生成的字符串post_body部分“:”和“,”之后都有空格,格式与c#采用json序列化后的字符串格式不一致

3.Python生成的字符串post_body部分键值对的顺序与c#生成的不一致

问题解决

首先我们使用c#将字符串中post_body部分转成Unicode编码,再将相应键值的位置调整为和Python生成的一致,最后再将缺失的空格添加今日字符串中,完整的程序如下:

 private string GetSign()
{
var param = new
{
appkey = "你的appkey",
timestamp = "你的timestamp",
device_tokens = "",
type = "unicast",
payload = new
{
body = new
{
text = "来自友盟推送",
after_open = "go_app",
ticker = "Hello World",
title = "你好"
},
display_type = "notification"
}
};
var app_master_secret = "你的app_master_secret";
var method = "POST";
var url = "http://msg.umeng.com/api/send";
var serializer = JsonSerializer.Create();
var jsonWriter = new System.IO.StringWriter();
serializer.Serialize(jsonWriter, param);
var post_body = jsonWriter.ToString();
post_body = unicode_js_0(post_body).Replace(":",": ").Replace(",",", ");
var str = method + url + post_body + app_master_secret;
var sign = MD5(str);
return sign;
}
/// <summary>
/// MD5加密
/// </summary>
/// <param name="s"></param>
/// <returns></returns>
private string MD5(string s)
{ var x = new MD5CryptoServiceProvider();
var bs = System.Text.Encoding.UTF8.GetBytes(s);
bs = x.ComputeHash(bs);
var str = new System.Text.StringBuilder();
foreach (byte b in bs)
{
str.Append(b.ToString("x2").ToLower());
}
return str.ToString();
}
/// <summary>
/// 中文转unicode(符合js规则的)
/// </summary>
/// <returns></returns>
public string unicode_js_0(string str)
{
var outStr = string.Empty;
if (!string.IsNullOrEmpty(str))
{
for (int i = ; i < str.Length; i++)
{
if (Regex.IsMatch(str[i].ToString(), @"[\u4e00-\u9fa5]")) { outStr += "\\u" + ((int)str[i]).ToString("x"); }
else { outStr += str[i]; }
}
}
return outStr;
}

这样就能使得c#程序输出和官方python demo一样的结果了。

总结

下面会进一步的进行与友盟消息推送服务的集成,进一步验证此方案的有效性,若有较好方案,欢迎指教,不甚感激

.NET手记-友盟消息推送服务器端加密算法的实现的更多相关文章

  1. 友盟消息推送api、python sdk问题、测试demo代码

    一,友盟消息推送python服务端sdk地址和文档地址 1.sdk地址:http://dev.umeng.com/system/resources/W1siZiIsIjIwMTYvMDgvMTkvMT ...

  2. 为友盟消息推送开发的PHP SDK(composer版):可以按省发Android push

    一直以来APP希望按省市县推送Android push,只能自己分析用户经纬度,打tag发送. 现在终于有服务商提供了. 友盟消息推送 可以“按省推送”,很方便. 我为友盟做了PHP SDK(comp ...

  3. umeng友盟消息推送功能集成

    umeng友盟消息推送功能集成(本人使用的是eclipse开发) 1.首先请自行观看友盟消息推送集成的API文档. 观看地址如下: http://dev.umeng.com/push/android/ ...

  4. 【转载自友盟消息推送iOS文档】在appDelegate中注册推送

    1.2   基本功能集成指南 提示 请先在友盟的消息推送管理后台中创建App,获得AppKey和AppSecret 导入SDK 下载 UMessage_Sdk_All_x.x.x.zip并解压缩 导入 ...

  5. 友盟消息推送UPush

    第一步:把下载的SDK里面的PushSDK当做Module导入自己的项目 第二步:在自己项目的build.gradle里面一定要配置applicationId defaultConfig { appl ...

  6. 友盟消息推送和更新XML配置

    <receiver android:name="com.umeng.message.NotificationProxyBroadcastReceiver" android:e ...

  7. 友盟iOS推送配置(从真机调试到推送)

    下面我来讲解一下友盟iOS的推送配置,其实友盟只是一个示例,换做其余的第三方推送服务也会适用,只是第三方的后面服务变了而已. iOS推送(包括真机调试)所需要的步骤和文件如下: 备注:这里我将省略掉一 ...

  8. atitit.极光消息推送服务器端开发实现推送  jpush v3. 总结o7p

    atitit.极光消息推送服务器端开发实现推送  jpush v3. 总结o7p 1. 推送所设计到底功能1 1.1. 内容压缩1 1.2. 多引擎1 2. reg  ,设置appkey and pw ...

  9. 使用PoolingHttpClientConnectionManager解决友盟(umeng)推送在多线程环境推送失败的问题

    在友盟(umeng)提供的服务端推送的sdk中,使用的是apache提供的httpclient.在单线程化境下,httpclient工作没有问题.但是由于umeng的sdk中并未考虑并发的情况,因此很 ...

随机推荐

  1. python的int方法实现数据类型转换

    int方法默认以十进制来实现数据类型的转换: 举例: str1=" #给定的内容最好是纯数字,当然也可以是数字再掺杂点别的,最好别掺杂,因为会报错 print(type(str1),str) ...

  2. HDU 5828 Rikka with Sequence(线段树区间加开根求和)

    Problem DescriptionAs we know, Rikka is poor at math. Yuta is worrying about this situation, so he g ...

  3. vm ware虚拟机ping不通解决办法

    本人是linux菜鸟,在使用vm ware的时候,在多台电脑上安装了多个虚拟机,这多台电脑是同一网段的,并且能够互相ping通,但是vm ware下的虚拟机就无法ping通 通过自己的各种才是,发现在 ...

  4. 515. Find Largest Value in Each Tree Row查找一行中的最大值

    [抄题]: You need to find the largest value in each row of a binary tree. Example: Input: 1 / \ 3 2 / \ ...

  5. springboot + @KafkaListener 手动提交及消费能力优化

    转载 https://blog.csdn.net/asd5629626/article/details/82776450  https://blog.csdn.net/asd5629626/artic ...

  6. SSD性能测试第一神器:FIO

    SSD性能测试第一神器:FIO  2017年8月12日 syswift 0 对于SSD性能测试来说,最好的工具莫过于FIO了. 上面这个可爱的小伙子名字叫Jens Axboe,他是丹麦哥本哈根大学计算 ...

  7. Linux 只列出目录的方法

    1. ls -d 2. find -type d -maxdepth 1 3. ls -F | grep "/$" 4. ls -l | grep "^d"

  8. slf4j 日志组件

    slf4j:Simple Logging Facade for Java 官网:https://www.slf4j.org/

  9. HTML中调用JavaScript的几种情况和规范写法

    JavaScript执行在html中,引用有几种方式? 我知道的方法有3种: 第一种:外部引用远程JavaScript文件.如<script type="text/javascript ...

  10. 接口测试3A原则

    手工的功能测试用例也可以用3A原则来编写. Arrange: 准备被测功能相关的测试数据,比如往系统里录入一批工单以便测试工单的分页功能 Act : 调用被测的功能,实际上这就是我们一直讲的测试步骤 ...