GlobalAuthUtil.java 10.1 KB
Newer Older
智布道's avatar
智布道 已提交
1 2 3
package me.zhyd.oauth.utils;

import cn.hutool.core.codec.Base64;
4 5
import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.util.ObjectUtil;
H
Hongwei Peng 已提交
6
import cn.hutool.http.HttpUtil;
7
import com.alibaba.fastjson.JSON;
智布道's avatar
智布道 已提交
8 9 10 11 12
import me.zhyd.oauth.exception.AuthException;

import javax.crypto.Mac;
import javax.crypto.spec.SecretKeySpec;
import java.io.UnsupportedEncodingException;
智布道's avatar
智布道 已提交
13
import java.net.URLDecoder;
智布道's avatar
智布道 已提交
14
import java.net.URLEncoder;
15 16
import java.nio.charset.Charset;
import java.nio.charset.StandardCharsets;
智布道's avatar
智布道 已提交
17
import java.security.InvalidKeyException;
18
import java.security.MessageDigest;
智布道's avatar
智布道 已提交
19
import java.security.NoSuchAlgorithmException;
H
Hongwei Peng 已提交
20 21 22 23 24 25 26
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import java.util.Random;
import java.util.TreeMap;
智布道's avatar
智布道 已提交
27

智布道's avatar
智布道 已提交
28 29 30 31
/**
 * 全局的工具类
 *
 * @author yadong.zhang (yadong.zhang0415(a)gmail.com)
智布道's avatar
智布道 已提交
32
 * @since 1.0.0
智布道's avatar
智布道 已提交
33
 */
智布道's avatar
智布道 已提交
34
public class GlobalAuthUtil {
35
    private static final Charset DEFAULT_ENCODING = StandardCharsets.UTF_8;
H
Hongwei Peng 已提交
36 37
    private static final String HMAC_SHA1 = "HmacSHA1";
    private static final String HMAC_SHA_256 = "HmacSHA256";
智布道's avatar
智布道 已提交
38

39 40 41 42 43 44 45
    /**
     * 生成钉钉请求的Signature
     *
     * @param secretKey 平台应用的授权密钥
     * @param timestamp 时间戳
     * @return Signature
     */
46
    public static String generateDingTalkSignature(String secretKey, String timestamp) {
H
Hongwei Peng 已提交
47
        byte[] signData = sign(secretKey.getBytes(DEFAULT_ENCODING), timestamp.getBytes(DEFAULT_ENCODING), HMAC_SHA_256);
48
        return urlEncode(new String(Base64.encode(signData, false)));
智布道's avatar
智布道 已提交
49 50
    }

51 52 53
    /**
     * 签名
     *
H
Hongwei Peng 已提交
54 55 56
     * @param key       key
     * @param data      data
     * @param algorithm algorithm
57 58
     * @return byte[]
     */
H
Hongwei Peng 已提交
59
    private static byte[] sign(byte[] key, byte[] data, String algorithm) {
智布道's avatar
智布道 已提交
60
        try {
H
Hongwei Peng 已提交
61 62
            Mac mac = Mac.getInstance(algorithm);
            mac.init(new SecretKeySpec(key, algorithm));
智布道's avatar
智布道 已提交
63 64
            return mac.doFinal(data);
        } catch (NoSuchAlgorithmException ex) {
H
Hongwei Peng 已提交
65
            throw new AuthException("Unsupported algorithm: " + algorithm, ex);
智布道's avatar
智布道 已提交
66 67 68 69 70
        } catch (InvalidKeyException ex) {
            throw new AuthException("Invalid key: " + Arrays.toString(key), ex);
        }
    }

71 72 73 74 75 76
    /**
     * 编码
     *
     * @param value str
     * @return encode str
     */
77
    public static String urlEncode(String value) {
智布道's avatar
智布道 已提交
78 79 80 81
        if (value == null) {
            return "";
        }
        try {
82 83
            String encoded = URLEncoder.encode(value, GlobalAuthUtil.DEFAULT_ENCODING.displayName());
            return encoded.replace("+", "%20").replace("*", "%2A").replace("~", "%7E").replace("/", "%2F");
智布道's avatar
智布道 已提交
84
        } catch (UnsupportedEncodingException e) {
智布道's avatar
智布道 已提交
85 86 87 88
            throw new AuthException("Failed To Encode Uri", e);
        }
    }

89 90 91 92 93 94 95

    /**
     * 解码
     *
     * @param value str
     * @return decode str
     */
智布道's avatar
智布道 已提交
96 97 98 99 100
    public static String urlDecode(String value) {
        if (value == null) {
            return "";
        }
        try {
101
            return URLDecoder.decode(value, GlobalAuthUtil.DEFAULT_ENCODING.displayName());
智布道's avatar
智布道 已提交
102 103 104 105 106
        } catch (UnsupportedEncodingException e) {
            throw new AuthException("Failed To Decode Uri", e);
        }
    }

107 108 109 110 111 112
    /**
     * string字符串转map,str格式为 {@code xxx=xxx&xxx=xxx}
     *
     * @param accessTokenStr 待转换的字符串
     * @return map
     */
智布道's avatar
智布道 已提交
113 114 115 116 117 118 119 120 121 122
    public static Map<String, String> parseStringToMap(String accessTokenStr) {
        Map<String, String> res = new HashMap<>();
        if (accessTokenStr.contains("&")) {
            String[] fields = accessTokenStr.split("&");
            for (String field : fields) {
                if (field.contains("=")) {
                    String[] keyValue = field.split("=");
                    res.put(GlobalAuthUtil.urlDecode(keyValue[0]), keyValue.length == 2 ? GlobalAuthUtil.urlDecode(keyValue[1]) : null);
                }
            }
智布道's avatar
智布道 已提交
123
        }
智布道's avatar
智布道 已提交
124
        return res;
智布道's avatar
智布道 已提交
125
    }
126

127 128 129 130 131 132 133
    /**
     * map转字符串,转换后的字符串格式为 {@code xxx=xxx&xxx=xxx}
     *
     * @param params 待转换的map
     * @param encode 是否转码
     * @return str
     */
134 135 136 137 138 139 140 141 142 143 144 145
    public static String parseMapToString(Map<String, Object> params, boolean encode) {
        List<String> paramList = new ArrayList<>();
        params.forEach((k, v) -> {
            if (ObjectUtil.isNull(v)) {
                paramList.add(k + "=");
            } else {
                String valueString = v.toString();
                paramList.add(k + "=" + (encode ? urlEncode(valueString) : valueString));
            }
        });
        return CollUtil.join(paramList, "&");
    }
146 147 148 149 150 151 152

    /**
     * 将url的参数列表转换成map
     *
     * @param url 待转换的url
     * @return map
     */
H
Hongwei Peng 已提交
153 154 155 156 157
    public static Map<String, Object> parseQueryToMap(String url) {
        Map<String, Object> paramMap = new HashMap<>();
        HttpUtil.decodeParamMap(url, "UTF-8").forEach(paramMap::put);
        return paramMap;
    }
158

159 160 161 162 163 164
    /**
     * 是否为http协议
     *
     * @param url 待验证的url
     * @return true: http协议, false: 非http协议
     */
165 166 167 168 169 170 171
    public static boolean isHttpProtocol(String url) {
        if (StringUtils.isEmpty(url)) {
            return false;
        }
        return url.startsWith("http://");
    }

172 173 174 175 176 177
    /**
     * 是否为https协议
     *
     * @param url 待验证的url
     * @return true: https协议, false: 非https协议
     */
178 179 180 181 182 183
    public static boolean isHttpsProtocol(String url) {
        if (StringUtils.isEmpty(url)) {
            return false;
        }
        return url.startsWith("https://");
    }
184

185 186 187 188 189 190
    /**
     * 是否为本地主机(域名)
     *
     * @param url 待验证的url
     * @return true: 本地主机(域名), false: 非本地主机(域名)
     */
191 192 193 194
    public static boolean isLocalHost(String url) {
        return StringUtils.isEmpty(url) || url.contains("127.0.0.1") || url.contains("localhost");
    }

H
Hongwei Peng 已提交
195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245

    /**
     * Generate nonce with given length
     *
     * @param len length
     * @return nonce string
     */
    public static String generateNonce(int len) {
        String s = "0123456789QWERTYUIOPLKJHGFDSAZXCVBNMqwertyuioplkjhgfdsazxcvbnm";
        Random rng = new Random();
        StringBuilder sb = new StringBuilder();
        for (int i = 0; i < len; i++) {
            int index = rng.nextInt(62);
            sb.append(s, index, index + 1);
        }
        return sb.toString();
    }

    /**
     * Get current timestamp
     *
     * @return timestamp string
     */
    public static String getTimestamp() {
        return String.valueOf(System.currentTimeMillis() / 1000);
    }

    /**
     * Generate Twitter signature
     * https://developer.twitter.com/en/docs/basics/authentication/guides/creating-a-signature
     *
     * @param params      parameters including: oauth headers, query params, body params
     * @param method      HTTP method
     * @param baseUrl     base url
     * @param apiSecret   api key secret can be found in the developer portal by viewing the app details page
     * @param tokenSecret oauth token secret
     * @return BASE64 encoded signature string
     */
    public static String generateTwitterSignature(Map<String, Object> params, String method, String baseUrl, String apiSecret, String tokenSecret) {
        TreeMap<String, Object> map = new TreeMap<>();
        for (Map.Entry<String, Object> e : params.entrySet()) {
            map.put(urlEncode(e.getKey()), e.getValue());
        }
        String str = parseMapToString(map, true);
        String baseStr = method.toUpperCase() + "&" + urlEncode(baseUrl) + "&" + urlEncode(str);
        String signKey = apiSecret + "&" + (StringUtils.isEmpty(tokenSecret) ? "" : tokenSecret);
        byte[] signature = sign(signKey.getBytes(DEFAULT_ENCODING), baseStr.getBytes(DEFAULT_ENCODING), HMAC_SHA1);

        return new String(Base64.encode(signature, false));
    }

246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299
    /**
     * 生成饿了么请求的Signature
     * <p>
     * 代码copy并修改自:https://coding.net/u/napos_openapi/p/eleme-openapi-java-sdk/git/blob/master/src/main/java/eleme/openapi/sdk/utils/SignatureUtil.java
     *
     * @param appKey     平台应用的授权key
     * @param secret     平台应用的授权密钥
     * @param timestamp  时间戳,单位秒。API服务端允许客户端请求最大时间误差为正负5分钟。
     * @param action     饿了么请求的api方法
     * @param token      用户授权的token
     * @param parameters 加密参数
     * @return Signature
     */
    public static String generateElemeSignature(String appKey, String secret, long timestamp, String action, String token, Map<String, Object> parameters) {
        final Map<String, Object> sorted = new TreeMap<>();
        for (Map.Entry<String, Object> entry : parameters.entrySet()) {
            sorted.put(entry.getKey(), entry.getValue());
        }
        sorted.put("app_key", appKey);
        sorted.put("timestamp", timestamp);
        StringBuffer string = new StringBuffer();
        for (Map.Entry<String, Object> entry : sorted.entrySet()) {
            string.append(entry.getKey()).append("=").append(JSON.toJSONString(entry.getValue()));
        }
        String splice = String.format("%s%s%s%s", action, token, string, secret);
        String calculatedSignature = md5(splice);
        return calculatedSignature.toUpperCase();
    }

    /**
     * MD5加密饿了么请求的Signature
     * <p>
     * 代码copy并修改自:https://coding.net/u/napos_openapi/p/eleme-openapi-java-sdk/git/blob/master/src/main/java/eleme/openapi/sdk/utils/SignatureUtil.java
     *
     * @param str 饿了么请求的Signature
     * @return md5 str
     */
    private static String md5(String str) {
        MessageDigest md = null;
        StringBuilder buffer = null;
        try {
            md = MessageDigest.getInstance("MD5");
            md.update(str.getBytes(StandardCharsets.UTF_8));
            byte[] byteData = md.digest();
            buffer = new StringBuilder();
            for (byte byteDatum : byteData) {
                buffer.append(Integer.toString((byteDatum & 0xff) + 0x100, 16).substring(1));
            }
        } catch (Exception ignored) {
        }

        return null == buffer ? "" : buffer.toString();
    }

智布道's avatar
智布道 已提交
300
}