原文链接:http://blog.csdn.net/hjun01/article/details/42032841      

 OAuth 2.0 for Web Server Applications, verifying a user's Android in-app subscription

在写本文之前先说些题外话。

前段时间游戏急于在GoolePlay上线,明知道如果不加Auth2.0的校验是不安全的还是暂时略过了这一步,果然没几天就发现后台记录与玩家实际付费不太一致,怀疑有玩家盗刷游戏元宝等,并且真实的走过了GooglePlay的所有支付流程完成道具兑换,时间一长严重性可想而知。经过查阅大量google官方文档后把代码补上,并在这里记录下OAuth 2.0 的使用,Google提供了OAuth2.0的好几种使用用途,每种使用方法都有些不同,具体可以看下这篇博客。在这里只写OAuth 2.0 for Web Server Applications的使用,涉及refresh_token, access_token等的获取和使用,以及如何向google发送GET和POST请求等,最终完成用户在安卓应用内支付购买信息的校验。

先贴下关于Using OAuth 2.0 for Web Server Applications的解释的谷歌官方原文

The authorization sequence begins when your application redirects a browser to a Google URL; the URL includes query parameters that indicate the type of access being requested. As in other scenarios, Google handles user authentication, session selection, and user consent. The result is an authorization code, which Google returns to your application in a query string.
        After receiving the authorization code, your application can exchange the code (along with a client ID and client secret) for an access token and, in some cases, a refresh token.
        The application can then use the access token to access a Google API.
        If a refresh token is present in the authorization code exchange, then it can be used to obtain new access tokens at any time. This is called offline access, because the user does not have to be present at the browser when the application obtains a new access token.

通过原文和图解我们可以知道这样一个流程(下文会详细说明):

一. 在Google Developer Console中创建一个 Web Application账户,得到client_id,client_secret 和 redirect_uri,这3个参数后边步骤常用到(此为前提)

二. 获取Authorization code

三. 利用code 获取access_token,refresh_token

四. 进一步可利用refresh_token获取新的access_token

五. 使用access_token 调用Google API 达到最终目的(如果access_token过时,回到第四步)

需注意的是:在第三步操作,当我们第一次利用code获取access_token时,谷歌会同时返回给你一个refresh_token,以后再次用code获取access_token操作将不会再看到refresh_token,所以一定要保存下来。这个refresh_token是长期有效的,如果没有明确的被应用管理者撤销是不会过期的,而access_token则只有3600秒的时效,即1个小时,那么问题来了,access_token和refresh_token是什么关系?很明显的,我们最终是要使用access_token 去调用Google API,而access_token又有时效限制,所以当access_token过期后,我们可以用长效的refresh_token去再次获取access_token,并且可以可以在任何时间多次获取,没有次数限制。其实当我们得到refresh_token后,就是一个转折点。

下面详细分解步骤:

一、在Google Developer Console中创建一个Web application账户

(这里使用的是新版的Google Developer Console页面,其实可在Account settings中设置为中文显示~)

其中4.可以随意填写。创建完成后可以看下下图所示:

在这里我们拿到3个关键参数: client_id,client_secret,redirect_uris,,于下边步骤。

可能会有人有疑问,怎么就能确定在google developer console 建立的project就于Googleplay上线的安卓应用有关联呢?为什么可以用这些参数得来的access_token去调用谷歌API?其实在Googleplay发布应用时就有关联project的操作,之后创建project的人可以给其他谷歌账户授权,这样其他谷歌账户可以在自己的developer console页面直接看到该project和以下的web application等, 并且可在下一步操作中登录自己的谷歌账户获取code。

二. 获取Authorization code

https://accounts.google.com/o/oauth2/auth?scope=https://www.googleapis.com/auth/androidpublisher
&response_type=code
&access_type=offline
&redirect_uri={REDIRECT_URIS}
&client_id={CLIENT_ID}
 

我们需要将这个URL以浏览器的形式打开,这时会跳出提示你Sign in with your Google Account,然后在用有project授权的谷歌账户登录,地址栏会出现我们所需的code。例如:https://www.example.com/oauth2callback?code=4/CpVOd8CljO_gxTRE1M5jtwEFwf8gRD44vrmKNDi4GSS.kr-GHuseD-oZEnp6UADFXm0E0MD3FlAI

三. 利用code 获取access_token,refresh_token

https://accounts.google.com/o/oauth2/token?
code={CODE}
&client_id={CLIENT_ID}
&client_secret={CLIENT_SECRET}
&redirect_uri={REDIRECT}
&grant_type=authorization_code
 

我们这一步的目的是获取refresh_token,只要有了这个长效token,access_token是随时可以获取的,第一次发起请求得到的JSON字符串如下所示,以后再请求将不再出现refresh_token,要保存好。expires_in是指access_token的时效,为3600秒。

{

    "access_token": "ya29.3gC2jw5vm77YPkylq0H5sPJeJJDHX93Kq8qZHRJaMlknwJ85595eMogL300XKDOEI7zIsdeFEPY6zg",
"token_type": "Bearer",
"expires_in": 3600,
"refresh_token": "1/FbQD448CdDPfDEDpCy4gj_m3WDr_M0U5WupquXL_o"
}

四. 进一步可利用refresh_token获取新的access_token

https://accounts.google.com/o/oauth2/token?
grant_type=refresh_token
&client_id={CLIENT_ID}
&client_secret={CLIENT_SECRET}
&refresh_token={REFRESH_TOKEN}

这里我们要向谷歌发起POST请求,Java代码如下:

  1. /** 获取access_token **/
  2. private static Map<String,String> getAccessToken(){
  3. final String CLIENT_ID = "填入你的client_id";
  4. final String CLIENT_SECRET = "填入你的client_secret";
  5. final String REFRESH_TOKEN = "填入上一步获取的refresh_token";
  6. Map<String,String> map = null;
  7. try {
  8. /**
  9. * https://accounts.google.com/o/oauth2/token?refresh_token={REFRESH_TOKEN}
  10. * &client_id={CLIENT_ID}&client_secret={CLIENT_SECRET}&grant_type=refresh_token
  11. */
  12. URL urlGetToken = new URL("https://accounts.google.com/o/oauth2/token");
  13. HttpURLConnection connectionGetToken = (HttpURLConnection) urlGetToken.openConnection();
  14. connectionGetToken.setRequestMethod("POST");
  15. connectionGetToken.setDoOutput(true);
  16. // 开始传送参数
  17. OutputStreamWriter writer  = new OutputStreamWriter(connectionGetToken.getOutputStream());
  18. writer.write("refresh_token="+REFRESH_TOKEN+"&");
  19. writer.write("client_id="+CLIENT_ID+"&");
  20. writer.write("client_secret="+CLIENT_SECRET+"&");
  21. writer.write("grant_type=refresh_token");
  22. writer.close();
  23. //若响应码为200则表示请求成功
  24. if(connectionGetToken.getResponseCode() == HttpURLConnection.HTTP_OK){
  25. StringBuilder sb = new StringBuilder();
  26. BufferedReader reader = new BufferedReader(
  27. new InputStreamReader(connectionGetToken.getInputStream(), "utf-8"));
  28. String strLine = "";
  29. while((strLine = reader.readLine()) != null){
  30. sb.append(strLine);
  31. }
  32. // 取得谷歌回传的信息(JSON格式)
  33. JSONObject jo = JSONObject.fromObject(sb.toString());
  34. String ACCESS_TOKEN = jo.getString("access_token");
  35. Integer EXPIRES_IN = jo.getInt("expires_in");
  36. map = new HashMap<String,String>();
  37. map.put("access_token", ACCESS_TOKEN);
  38. map.put("expires_in", String.valueOf(EXPIRES_IN));
  39. // 带入access_token的创建时间,用于之后判断是否失效
  40. map.put("create_time",String.valueOf((new Date().getTime()) / 1000));
  41. logger.info("包含access_token的JSON信息为: "+jo);
  42. }
  43. } catch (MalformedURLException e) {
  44. logger.error("获取access_token失败,原因是:"+e);
  45. e.printStackTrace();
  46. } catch (IOException e) {
  47. logger.error("获取access_token失败,原因是:"+e);
  48. e.printStackTrace();
  49. }
  50. return map;
  51. }

五. 使用access_token 调用Google API 达到最终目的(如果access_token过时,回到第四步)

在这里我所需要获取的是我在应用内给GooglePlay支付的购买信息,此类信息包含以下几个属性:(可参考Google Play Developer API下的Purchases.products

A ProductPurchase resource indicates the status of a user's inapp product purchase.

{
  "kind": "androidpublisher#productPurchase",
  "purchaseTimeMillis": long,
  "purchaseState": integer, (purchased:0 cancelled:1,我们就是依靠这个判断购买信息)
  "consumptionState": integer,
  "developerPayload": string
}

带着access_token参数向GoogleApi发起GET请求,Java代码如下:

  1. private static Map<String,String> cacheToken = null;//设置静态变量,用于判断access_token是否过期
  2. public static GooglePlayBuyEntity getInfoFromGooglePlayServer(String packageName,String productId, String purchaseToken) {
  3. if(null != cacheToken){
  4. Long expires_in = Long.valueOf(cacheToken.get("expires_in")); // 有效时长
  5. Long create_time = Long.valueOf(cacheToken.get("create_time")); // access_token的创建时间
  6. Long now_time = (new Date().getTime()) / 1000;
  7. if(now_time > (create_time + expires_in - 300)){ // 提前五分钟重新获取access_token
  8. cacheToken = getAccessToken();
  9. }
  10. }else{
  11. cacheToken = getAccessToken();
  12. }
  13. String access_token = cacheToken.get("access_token");
  14. GooglePlayBuyEntity buyEntity = null;
  15. try {
  16. /**这是写这篇博客时间时的最新API,v2版本。
  17. * https://www.googleapis.com/androidpublisher/v2/applications/{packageName}
  18. * /purchases/products/{productId}/tokens/{purchaseToken}?access_token={access_token}
  19. */
  20. String url = "https://www.googleapis.com/androidpublisher/v2/applications";
  21. StringBuffer getURL = new StringBuffer();
  22. getURL.append(url);
  23. getURL.append("/" + packageName);
  24. getURL.append("/purchases/products");
  25. getURL.append("/" + productId   );
  26. getURL.append("/tokens/" + purchaseToken);
  27. getURL.append("?access_token=" + access_token);
  28. URL urlObtainOrder = new URL(getURL.toString());
  29. HttpURLConnection connectionObtainOrder = (HttpURLConnection) urlObtainOrder.openConnection();
  30. connectionObtainOrder.setRequestMethod("GET");
  31. connectionObtainOrder.setDoOutput(true);
  32. // 如果认证成功
  33. if (connectionObtainOrder.getResponseCode() == HttpURLConnection.HTTP_OK) {
  34. StringBuilder sbLines = new StringBuilder("");
  35. BufferedReader reader = new BufferedReader(new InputStreamReader(
  36. connectionObtainOrder.getInputStream(), "utf-8"));
  37. String strLine = "";
  38. while ((strLine = reader.readLine()) != null) {
  39. sbLines.append(strLine);
  40. }
  41. // 把上面取回來的資料,放進JSONObject中,以方便我們直接存取到想要的參數
  42. JSONObject jo = JSONObject.fromObject(sbLines.toString());
  43. Integer status = jo.getInt("purchaseState");
  44. if(status == 0){ //验证成功
  45. buyEntity = new GooglePlayBuyEntity();
  46. buyEntity.setConsumptionState(jo.getInt("consumptionState"));
  47. buyEntity.setDeveloperPayload(jo.getString("developerPayload"));
  48. buyEntity.setKind(jo.getString("kind"));
  49. buyEntity.setPurchaseState(status);
  50. buyEntity.setPurchaseTimeMillis(jo.getLong("purchaseTimeMillis"));
  51. }else{
  52. // 购买无效
  53. buyEntity = new GooglePlayBuyEntity();
  54. buyEntity.setPurchaseState(status);
  55. logger.info("从GooglePlay账单校验失败,原因是purchaseStatus为" + status);
  56. }
  57. }
  58. } catch (Exception e) {
  59. e.printStackTrace();
  60. buyEntity = new GooglePlayBuyEntity();
  61. buyEntity.setPurchaseState(-1);
  62. }
  63. return buyEntity;
  64. }

到这里就写完了,如果有什么疑问可以留言。

另外,iOS应用内支付,苹果商店AppStore购买信息校验的博客在这里:http://blog.csdn.net/hjun01/article/details/44039939

【实例图文详解】OAuth 2.0 for Web Server Applications的更多相关文章

  1. Windows-007-进程相关命令(netstat、tasklist、taskkill、tskill)实战实例图文详解

    本节主要讲述 Windows 系统下,nestat.tasklist.tskill 三个 CMD 命令的参数,及使用方法:以及如何利用三者结合查看进程信息和结束进程.敬请亲们参阅,希望能对亲们有所帮助 ...

  2. 访问Storm ui界面,出现Nimbus Summary或Supervisor Summary时有时无的问题解决(图文详解)

    不多说,直接上干货! 前期博客 apache-storm-0.9.6.tar.gz的集群搭建(3节点)(图文详解) apache-storm-1.0.2.tar.gz的集群搭建(3节点)(图文详解)( ...

  3. 访问Storm ui界面,出现org.apache.storm.utils.NimbusLeaderNotFoundException: Could not find leader nimbus from seed hosts ["master"]. Did you specify a valid list of nimbus hosts for confi的问题解决(图文详解)

    不多说,直接上干货! 前期博客 apache-storm-0.9.6.tar.gz的集群搭建(3节点)(图文详解) apache-storm-1.0.2.tar.gz的集群搭建(3节点)(图文详解)( ...

  4. Android EventBus 3.0 实例使用详解

    EventBus的使用和原理在网上有很多的博客了,其中泓洋大哥和启舰写的非常非常棒,我也是跟着他们的博客学会的EventBus,因为是第一次接触并使用EventBus,所以我写的更多是如何使用,源码解 ...

  5. CentOS7+CDH5.14.0安装全流程记录,图文详解全程实测-总目录

    CentOS7+CDH5.14.0安装全流程记录,图文详解全程实测-总目录: 0.Windows 10本机下载Xshell,以方便往Linux主机上上传大文件 1.CentOS7+CDH5.14.0安 ...

  6. spark最新源码下载并导入到开发环境下助推高质量代码(Scala IDEA for Eclipse和IntelliJ IDEA皆适用)(以spark2.2.0源码包为例)(图文详解)

    不多说,直接上干货! 前言   其实啊,无论你是初学者还是具备了有一定spark编程经验,都需要对spark源码足够重视起来. 本人,肺腑之己见,想要成为大数据的大牛和顶尖专家,多结合源码和操练编程. ...

  7. hadoop-2.7.3.tar.gz + spark-2.0.2-bin-hadoop2.7.tgz + zeppelin-0.6.2-incubating-bin-all.tgz(master、slave1和slave2)(博主推荐)(图文详解)

    不多说,直接上干货! 我这里,采取的是ubuntu 16.04系统,当然大家也可以在CentOS6.5里,这些都是小事 CentOS 6.5的安装详解 hadoop-2.6.0.tar.gz + sp ...

  8. hadoop-2.6.0.tar.gz + spark-1.6.1-bin-hadoop2.6.tgz + zeppelin-0.5.6-incubating-bin-all.tgz(master、slave1和slave2)(博主推荐)(图文详解)

    不多说,直接上干货! 我这里,采取的是CentOS6.5,当然大家也可以在ubuntu 16.04系统里,这些都是小事 CentOS 6.5的安装详解 hadoop-2.6.0.tar.gz + sp ...

  9. VMware下OSSIM 4.1.0的下载、安装和初步使用(图文详解)

    不多说,直接上干货! 为什么,我写了一篇OSSIM 5.2.0的,还要再来写OSSIM 4.1.0呢,是因为,OSSIM 5.2.0所需内存较大,8G甚至16G,但是,肯定性能和里面集成组件越高级.也 ...

随机推荐

  1. HDU 1255 覆盖的面积(线段树:扫描线求面积并)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1255 题目大意:给你若干个矩形,让你求这些矩形重叠两次及以上的部分的面积. 解题思路:模板题,跟HDU ...

  2. Codeforces 351D Jeff and Removing Periods(莫队+区间等差数列更新)

    题目链接:http://codeforces.com/problemset/problem/351/D 题目大意:有n个数,每次可以删除掉数值相同并且所在位置成等差数列的数(只删2个数或者只删1个数应 ...

  3. django orm如何作一个优雅一点的filter?

    如果有N多fitler条件, 单独放在一个长语句里显然不好看. 还好, django支持字典方式的过滤条件, 写法大约与单独的长语里差不多. 如下: def get_queryset(self): f ...

  4. day1 str字符串常用方法

    字符串是编程中常用的类型,字符型在内存中是以单个形式存储的,比如name = "alex",在内存中存储的形式为["a","l"," ...

  5. poj1562 Oil Deposits(DFS)

    题目链接 http://poj.org/problem?id=1562 题意 输入一个m行n列的棋盘,棋盘上每个位置为'*'或者'@',求'@'的连通块有几个(连通为8连通,即上下左右,两条对角线). ...

  6. HTML5 canvas绘制arcTo、translate和rotate的画法探索

      arcTo(x1,y1,x2,y2,radius) ; 还要加上moveTo的点(x0,y0) ; 第一步:找到切点 过点 (x1,y1), (x0,y0)引射线与点(x1,y1),(x2,y2) ...

  7. 使用Nginx实现TCP反向代理

    Nginx 在1.9.0版本发布以前如果要想做到基于TCP的代理及负载均衡需要通过打名为 nginx_tcp_proxy_module 的第三方patch来实现,该模块的代码托管在github上网址: ...

  8. java反射,代码优化

    java的反射机制属实强大,能解决好些问题 在接手别人写的代码的时候,有一个bean类的get方法特别low,我都看不下去 重复代码写五遍,我都觉得太不合理.之后将其中代码抽取出来修改了下. publ ...

  9. java基础小测试

    1.JDK,JRE,JVM三者的区别 jdk:java 开发工具包 jre:运行环境 jvm:虚拟机 2.javac的作用 ,反编译工具的作用 javac:将java文件编译成class文件 反编译: ...

  10. Xamarin 2017.9.19更新

     Xamarin 2017.9.19更新   本次更新是添加Xamarin.iOS对iOS 11和Xcode 9的支持.Visual Studio 2017升级到15.3.5获得更新功能.Visual ...