Java эквивалент PHP HMAC-SHA1 - PullRequest
       40

Java эквивалент PHP HMAC-SHA1

48 голосов
/ 23 октября 2009

Я ищу Java-эквивалент этого вызова php:

hash_hmac('sha1', "test", "secret")

Я попробовал это, используя java.crypto.Mac , но оба не согласны:

String mykey = "secret";
String test = "test";
try {
    Mac mac = Mac.getInstance("HmacSHA1");
    SecretKeySpec secret = new SecretKeySpec(mykey.getBytes(),"HmacSHA1");
    mac.init(secret);
    byte[] digest = mac.doFinal(test.getBytes());
    String enc = new String(digest);
    System.out.println(enc);  
} catch (Exception e) {
    System.out.println(e.getMessage());
}

Выводы с key = "secret" и test = "test" не совпадают.

Ответы [ 7 ]

38 голосов
/ 23 октября 2009

На самом деле они согласны.
Как уже отмечал Ганс Догген, PHP выводит дайджест сообщения в шестнадцатеричном формате, если вы не установили для параметра raw output значение true.
Если вы хотите использовать те же обозначения в Java, вы можете использовать что-то вроде

for (byte b : digest) {
    System.out.format("%02x", b);
}
System.out.println();

для форматирования вывода соответственно.

16 голосов
/ 25 мая 2011

Вы можете попробовать это на Java:

private static String computeSignature(String baseString, String keyString) throws GeneralSecurityException, UnsupportedEncodingException {

    SecretKey secretKey = null;

    byte[] keyBytes = keyString.getBytes();
    secretKey = new SecretKeySpec(keyBytes, "HmacSHA1");

    Mac mac = Mac.getInstance("HmacSHA1");

    mac.init(secretKey);

    byte[] text = baseString.getBytes();

    return new String(Base64.encodeBase64(mac.doFinal(text))).trim();
}
5 голосов
/ 11 мая 2011

Это моя реализация:

        String hmac = "";

    Mac mac = Mac.getInstance("HmacSHA1");
    SecretKeySpec secret = new SecretKeySpec(llave.getBytes(), "HmacSHA1");
    mac.init(secret);
    byte[] digest = mac.doFinal(cadena.getBytes());
    BigInteger hash = new BigInteger(1, digest);
    hmac = hash.toString(16);

    if (hmac.length() % 2 != 0) {
        hmac = "0" + hmac;
    }

    return hmac;
3 голосов
/ 13 мая 2013

Моя реализация для HmacMD5 - просто измените алгоритм на HmacSHA1:

SecretKeySpec keySpec = new SecretKeySpec("secretkey".getBytes(), "HmacMD5");
Mac mac = Mac.getInstance("HmacMD5");
mac.init(keySpec);
byte[] hashBytes = mac.doFinal("text2crypt".getBytes());
return Hex.encodeHexString(hashBytes);
3 голосов
/ 23 октября 2009

Мне кажется, что PHP использует HEX-нотацию для байтов, которые создает Java (1a = 26) - но я не проверял все выражение.

Что произойдет, если вы запустите байтовый массив с помощью метода на этой странице?

1 голос
/ 10 октября 2014

Таким образом, я мог получить ту же строку, что и с hash_hmac в php

String result;

try {
        String data = "mydata";
        String key = "myKey";
        // Get an hmac_sha1 key from the raw key bytes
        byte[] keyBytes = key.getBytes();
        SecretKeySpec signingKey = new SecretKeySpec(keyBytes, "HmacSHA1");

        // Get an hmac_sha1 Mac instance and initialize with the signing key
        Mac mac = Mac.getInstance("HmacSHA1");
        mac.init(signingKey);

        // Compute the hmac on input data bytes
        byte[] rawHmac = mac.doFinal(data.getBytes());

        // Convert raw bytes to Hex
        byte[] hexBytes = new Hex().encode(rawHmac);

        //  Covert array of Hex bytes to a String
        result = new String(hexBytes, "ISO-8859-1");
        out.println("MAC : " + result);
}
catch (Exception e) {

}
1 голос
/ 23 октября 2009

Не проверял, но попробуйте это:

        BigInteger hash = new BigInteger(1, digest);
        String enc = hash.toString(16);
        if ((enc.length() % 2) != 0) {
            enc = "0" + enc;
        }

Это снимок из моего метода, который делает md5 и sha1 java равными php.

...