App开放接口api安全性—Token签名sign的设计与实现

前言

在app开放接口api的设计中,避免不了的就是安全性问题,因为大多数接口涉及到用户的个人信息以及一些敏感的数据,所以对这些接口需要进行身份的认证,那么这就需要用户提供一些信息,比如用户名密码等,但是为了安全起见让用户暴露的明文密码次数越少越好,我们一般在web项目中,大多数采用保存的session中,然后在存一份到cookie中,来保持用户的回话有效性。但是在app提供的开放接口中,后端服务器在用户登录后如何去验证和维护用户的登陆有效性呢,以下是参考项目中设计的解决方案,其原理和大多数开放接口安全验证一样,如淘宝的开放接口token验证,微信开发平台token验证都是同理。

签名设计

对于敏感的api接口,需使用https协议

https是在http超文本传输协议加入SSL层,它在网络间通信是加密的,所以需要加密证书。

https协议需要ca证书,一般需要交费。

签名的设计

原理:用户登录后向服务器提供用户认证信息(如账户和密码),服务器认证完后给客户端返回一个Token令牌,用户再次获取信息时,带上此令牌,如果令牌正取,则返回数据。对于获取Token信息后,访问用户相关接口,客户端请求的url需要带上如下参数:

时间戳:timestamp

Token令牌:token

然后将所有用户请求的参数按照字母排序(包括timestamp,token),然后更具MD5加密(可以加点盐),全部大写,生成sign签名,这就是所说的url签名算法。然后登陆后每次调用用户信息时,带上sign,timestamp,token参数。

例如:原请求https://www.andy.cn/api/user/update/info.shtml?city=北京 (post和get都一样,对所有参数排序加密)

加上时间戳和token

https://www.andy.cn/api/user/update/info.shtml?city=北京×tamp=12445323134&token=wefkfjdskfjewfjkjfdfnc  
然后更具url参数生成sign

最终的请求如

https://www.andy.cn/api/user/update/info.shtml?city=北京×tamp=12445323134&token=wefkfjdskfjewfjkjfdfnc&sign=FDK2434JKJFD334FDF2  
其最终的原理是减小明文的暴露次数;保证数据安全的访问。

具体实现如下:

1. api请求客户端想服务器端一次发送用用户认证信息(用户名和密码),服务器端请求到改请求后,验证用户信息是否正确。

如果正确:则返回一个唯一不重复的字符串(一般为UUID),然后在Redis(任意缓存服务器)中维护Token----Uid的用户信息关系,以便其他api对token的校验。

如果错误:则返回错误码。



2.服务器设计一个url请求拦截规则

(1)判断是否包含timestamp,token,sign参数,如果不含有返回错误码。

(2)判断服务器接到请求的时间和参数中的时间戳是否相差很长一段时间(时间自定义如半个小时),如果超过则说明该 url已经过期(如果url被盗,他改变了时间戳,但是会导致sign签名不相等)。

(3)判断token是否有效,根据请求过来的token,查询redis缓存中的uid,如果获取不到这说明该token已过期。

(4)根据用户请求的url参数,服务器端按照同样的规则生成sign签名,对比签名看是否相等,相等则放行。(自然url签名 也无法100%保证其安全,也可以通过公钥AES对数据和url加密,但这样如果无法确保公钥丢失,所以签名只是很大程 度上保证安全)。

(5)此url拦截只需对获取身份认证的url放行(如登陆url),剩余所有的url都需拦截。

3.Token和Uid关系维护

对于用户登录我们需要创建token--uid的关系,用户退出时需要需删除token--uid的关系。

签名实现

获取全部请求参数



[java]  view plain copy
  1. String sign = request.getParameter("sign");  
  2.         Enumeration<?> pNames =  request.getParameterNames();  
  3.         Map<String, Object> params = new HashMap<String, Object>();  
  4.         while (pNames.hasMoreElements()) {  
  5.             String pName = (String) pNames.nextElement();  
  6.             if("sign".equals(pName))continue;  
  7.             Object pValue = request.getParameter(pName);  
  8.             params.put(pName, pValue);  
  9.         }  


生成签名



[java]  view plain copy
  1. public static String createSign(Map<String, String> params, boolean encode)  
  2.             throws UnsupportedEncodingException {  
  3.         Set<String> keysSet = params.keySet();  
  4.         Object[] keys = keysSet.toArray();  
  5.         Arrays.sort(keys);  
  6.         StringBuffer temp = new StringBuffer();  
  7.         boolean first = true;  
  8.         for (Object key : keys) {  
  9.             if (first) {  
  10.                 first = false;  
  11.             } else {  
  12.                 temp.append("&");  
  13.             }  
  14.             temp.append(key).append("=");  
  15.             Object value = params.get(key);  
  16.             String valueString = "";  
  17.             if (null != value) {  
  18.                 valueString = String.valueOf(value);  
  19.             }  
  20.             if (encode) {  
  21.                 temp.append(URLEncoder.encode(valueString, "UTF-8"));  
  22.             } else {  
  23.                 temp.append(valueString);  
  24.             }  
  25.         }  
  26.   
  27.         return MD5Utils.getMD5(temp.toString()).toUpperCase();  
  28.     }  


http://www.lai18.com/content/944366.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在 Java 中实现 API 接口token 验证可以通过以下步骤进行: 1. 创建一个 TokenUtil 类,用于生成和验证 token。 ```java import io.jsonwebtoken.Claims; import io.jsonwebtoken.Jwts; import io.jsonwebtoken.security.Keys; import javax.crypto.SecretKey; import java.util.Date; public class TokenUtil { private static final String SECRET_KEY = "your_secret_key_here"; // 生成 token public static String generateToken(String userId) { SecretKey key = Keys.hmacShaKeyFor(SECRET_KEY.getBytes()); return Jwts.builder() .setSubject(userId) .setExpiration(new Date(System.currentTimeMillis() + 86400000)) // 设置过期时间为一天 .signWith(key) .compact(); } // 验证 token public static boolean validateToken(String token) { try { SecretKey key = Keys.hmacShaKeyFor(SECRET_KEY.getBytes()); Jwts.parserBuilder().setSigningKey(key).build().parseClaimsJws(token); return true; } catch (Exception e) { return false; } } // 获取 token 中的用户ID public static String getUserIdFromToken(String token) { SecretKey key = Keys.hmacShaKeyFor(SECRET_KEY.getBytes()); Claims claims = Jwts.parserBuilder().setSigningKey(key).build().parseClaimsJws(token).getBody(); return claims.getSubject(); } } ``` 2. 在需要验证 tokenAPI 接口中,使用 TokenUtil 类进行验证。 ```java import javax.ws.rs.GET; import javax.ws.rs.Path; import javax.ws.rs.QueryParam; import javax.ws.rs.core.Response; @Path("/api") public class MyApi { @GET @Path("/someEndpoint") public Response someEndpoint(@QueryParam("token") String token) { if (TokenUtil.validateToken(token)) { String userId = TokenUtil.getUserIdFromToken(token); // 验证通过,执行相应的逻辑 return Response.ok().build(); } else { // 验证失败,返回错误信息 return Response.status(Response.Status.UNAUTHORIZED).build(); } } } ``` 在上述代码中,`generateToken` 方法用于生成 token,`validateToken` 方法用于验证 token 的有效性,`getUserIdFromToken` 方法用于从 token 中获取用户ID。在 API 接口中,通过调用 `validateToken` 方法来验证传递的 token,如果验证通过,则可以根据需要执行相应的逻辑。 请注意,上述代码中的 SECRET_KEY 是用于签名和验证 token 的密钥,应该保持安全,并且不应该直接暴露在代码中。可以将其存储在配置文件或环境变量中,并通过相应的方式获取。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值