我正在为网站创建一个PHP API,我想限制对我们服务器上注册的域的API访问(以防止滥用API)。所以,这是我现在的方法,好吧,它在纸面上应该看起来很不错。
api.example.com
。mcrypt
)并通过cURL
将其发送至api.example.com
。mcrypt
解密请求,然后使用相同的方法加密并发送结果。我坚持第4步。最初,我打算使用HTTP_REFERER来检查它,但是因为cURL默认情况下没有发送一个,所以它很容易在用户端代码中伪造(CURLOPT_REFERER,就我而言)记得),我被困在这里。
是否有方法可以知道此API请求来自哪个域?我看到它可以用一些流行的API来完成,比如reCAPTCHA。由于共享主机(它们具有相同的IP),检查_SERVER [“REMOTE_HOST”]实际上不是一个选项,因此这将无法防止滥用(无论如何主要来自共享服务器)。
有没有这样的方法来检查它?谢谢!
答案 0 :(得分:7)
因此,当用户注册一个密钥时,您实际上是在为用户分配两个不同的密钥
API_KEY - 将您连接到您的域的公钥。系统会查找提供的域和密钥,以便找到下一个密钥
MCRYPT_KEY - 这是用于通过Mcrypt实际加密该数据的密钥。由于它是加密数据,因此只有请求者和服务器才知道它是什么。您使用密钥加密数据并将带有API密钥的加密输入发送到服务器,服务器找到通过API密钥和已提供的域(和IP)解密该输入所需的密钥。如果他们没有使用正确的密钥加密数据,那么使用正确的密钥进行解密将返回乱码并且json_decode()
调用将返回NULL,从而允许脚本只返回'invalid_input'响应。
最终使用这种方法,我们甚至需要检查请求来自哪里(域/ IP)?使用这种方法实际上归结为API用户没有向其他用户泄露他们的API / MCRYPT密钥对,类似于不泄露您的用户名/密码。即便如此,任何网站都可以轻松注册以获取自己的密钥对并使用API。另外需要注意的是,除非最终用户使用正确的用户名和密码登录,否则API甚至不会向其服务器返回任何有用的内容,因此他们的结尾将已经拥有该信息。我们的服务器真正返回的唯一新功能是成功验证用户后的电子邮件地址。话虽如此,我们甚至需要使用cURL吗?我们不能简单地使用file_get_contents('http://api.example.com/{$API_KEY}/{$MCRYPT_DATA}')
吗?我意识到我在回答中提出了更多问题......
答案 1 :(得分:3)
你可以改变请求来自哪个ip,并且你可以进行ptr搜索以获取该ip的域名,但是探测到ip地址有多个域名,你最终会使用错误的域名,所以我推荐客户端在请求中发送他的域名,也许是HTTP_REFERER,并且你做了一个dns检查该域是否指向ip要求它,但请注意域名,如google.com,可以指向更多然后一个ip。 (知识产权可能会被伪装成一个好的黑客技能,但这是我所知道的)
答案 2 :(得分:3)
如何引入第二个变量,比如说app id
。当用户注册其域时,将此id与域关联。用户需要在没有加密的情况下为每个请求提交app id
以及加密的api调用。然后,您可以查找app id
获取app secret
并尝试解密?
答案 3 :(得分:2)
为了最好地防止滥用您的API,请限制请求的速度,或限制他们可以提出的请求数量。如果有人是愚蠢的并且共享他们的API密钥,他们只会限制他们自己的API使用,这使得那些打算滥用API获取他们自己的密钥的人更经济。
另外,如果有人决定使用您的API实施桌面应用程序,该怎么办?当然他们不会要求他们的用户向他们发送他们的IP地址,以便他们可以将他们列入白名单?
此外,您可以结合限制速度/限制请求,并根据请求数量限制速度,例如,如果您通过一定数量的数据使用,Verizon将如何限制其3G网络的速度。