使用PHP OpenSSL验证Android应用内购买服务器签名

22

为了遵循应用内购买的一些安全指南,我正在尝试遵循以下链接中的建议:http://developer.android.com/guide/market/billing/billing_best_practices.html

我正在尝试在服务器上进行签名验证,而不是在应用程序中进行。我希望使用php openssl库,看起来以下代码应该可以工作:

<?php
// $data and $signature are assumed to contain the data and the signature

// fetch public key from certificate and ready it
$fp = fopen("/src/openssl-0.9.6/demos/sign/cert.pem", "r");
$cert = fread($fp, 8192);
fclose($fp);
$pubkeyid = openssl_get_publickey($cert);

// state whether signature is okay or not
$ok = openssl_verify($data, $signature, $pubkeyid);
if ($ok == 1) {
    echo "good";
} elseif ($ok == 0) {
    echo "bad";
} else {
    echo "ugly, error checking signature";
}
// free the key from memory
openssl_free_key($pubkeyid);
?>

我在应用购买包中使用base64解码后的签名字符串替换签名,并使用同一包中的数据。公钥需要采用PEM格式,我添加了BEGIN和END标记以及一些换行符。
我的问题是,我无法使这段PHP代码成功验证数据/签名,并且我不知道需要什么更改才能使其正常工作。
如果我使用openssl创建私钥和公钥,为相同数据创建sha1签名并将其通过上述php代码运行,则可以正常工作并成功验证。
以下是我如何使用OpenSSL:
openssl genrsa -out private.pem
openssl rsa -in private.pem -pubout -out public.pem

然后我使用private.pem和一些php代码生成签名:

...
openssl_sign($data, $signature, $pkeyid);
...

有人有带有服务器端应用内签名验证的工作示例php代码吗?

如果可能的话,我想直接使用php,虽然我可以运行示例应用程序中的等效java代码,但似乎也可以正常工作。


从Android Market发布网站获取公钥并添加BEGIN/END和换行符是否足以使其成为有效的PEM格式公钥? - Nathan Totura
解决了,这个代码是正确的,唯一不对的是我从文件中读取的数据末尾添加了换行符,即使我已经删除了所有额外字符。修复上面的代码最简单的方法是添加 $data=trim($data);。这样之后,一切都按预期工作。 - Nathan Totura
你是如何获取PEM格式的Base64编码RSA密钥的? - Sylvain
这是我使用的openssl命令:openssl enc -base64 -d -in publickey.raw -A | openssl rsa -inform DER -pubin > out。publickey.raw是直接从Android Market粘贴的密钥。 - Nathan Totura
1
使用$signature时要小心:http://crazyviraj.blogspot.com/2011/06/some-notes-on-implementing-in-app.html - psychotik
2个回答

3
我已经编写了一个用于验证Android市场许可证响应的库,并且它可以在Google Code上获得。
只需要几行PHP代码即可验证许可证,密钥和OpenSSL相关内容的格式化处理已经为您处理好。

你的库似乎不再适用于v3谷歌Play API了?你能否更新一下呢? - Pentium10
我一直在计划更新库,并支持应用内购买。 - David Snabel-Caunt
你有计划什么时候准备好吗?我最感兴趣的是PHP部分。 - Pentium10
@DavidCaunt 你更新了你的优秀库以适应v3了吗? - Dr.jacky

0

在您的PHP脚本中,是否可以使用cURL而不是内置于PHP流中的东西?我以前用过它们,并发现问题更少,错误消息更详细。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接