【问题标题】:How to send and receive data via HTTP header using cURL?如何使用 cURL 通过 HTTP 标头发送和接收数据?
【发布时间】:2014-05-13 00:28:07
【问题描述】:

我正在尝试构建一个简单的 API 以允许客户端通过 HTTPS 向我发送数据。 我创建了一个使用用户名/密码的类,然后它会查找数据库。如果找到用户,则它会发出一个令牌。然后令牌将通过 HTTP 标头发送回请求者。

一旦发回用户名、密码和令牌,脚本就会读取客户端通过$_POST请求发送的数据并进行处理。

我遇到的挑战是通过 cURL 将令牌发送给请求者,并从 HTTP 标头正确接收 USERNAMEPASSWORDTOKEN

我的问题是如何通过 generateToken() 方法中的 HTTP 标头正确发送令牌?另外,一旦发出请求,如何读取 HTTP 标头?

下面是我的类:api.php文件

<?php
require('../classes/connection.php');

class api {

    private $user_name;
    private $user_password;
    private $user_token;
    private $db;
    private $keepAlive = 120; //2 minutes = 120 seconds
    private $authorizes = false;
    private $token = '';
    private $ch;
    private $user_ready = false;

    function api($database, $server){

        //establish a database connection
        $this->db = new connection($database, $server);
        $this->ch = curl_init();

        //read user_name, password, token from the header and set it
        if(isset($_SERVER['API-User-Name']))
            $this->user_name = $_SERVER['API-User-Name'];

        if(isset($_SERVER['API-User-Password']))
            $this->user_password = $_SERVER['API-User-Password'];

        if(isset($_SERVER['API-User-Token']))
            $this->user_token = $_SERVER['API-User-Token'];

        //check if the user is allowed
        if(  $this->authenticateAccess() === true  ){
            $this->authorizes = true;

            //ensure the token is valid otherwise generate a new token
            if(     $this->isValidToken()   )
                $this->user_ready = true;
            else 
                $this->generateToken();
        }
    }

    //return weather to process the send data
    public function isUserReady(){
        return $this->user_ready;
    }

    //return weather the user is authorized
    private function isAutherized(){
        return $this->authorizes;
    }

    //return the set token
    private function getToken(){
        return $this->token;
    }

    //check if the requester is authorized to access the system
    private function authenticateAccess(){

        //unauthorized old session
        $this->unautherizeExpiredTokens();

        if( $this->ch === false) 
            return false;

        if( empty($this->user_name)  || empty($this->user_password)  )
            return false;

        //ensure HTTPS is used  
        if( !isset($_SERVER['HTTPS']) || $_SERVER['HTTPS'] != 'on') 
            return false;
         //read the user information                
         $get_user = $this->db->getDataSet('SELECT ip_addreses, user_password, token_expires_on, current_token
                                            FROM api_users
                                            WHERE user_name = ?
                                            LIMIT 1', array($this->user_name));

         if( count($get_user) != 1)
            return false;

         $data = $get_user[0];

         //remove bad values if any
         $ip_addreses = preg_replace("/[^0-9,.]/", "", $data['ip_addreses']);

         $allowed_ips = explode(',', $ip_addreses);

        //ensure the IP address is allowed
        if( !isset($_SERVER['REMOTE_ADDR']) || !in_array($_SERVER['REMOTE_ADDR'], $allowed_ips) )
            return false;

        //check if the password is valid    
        if( password_verify($this->password, $data['user_password'] ) )
            return true;
        else        
            return false;
    }


    //check if the token is valid
    private function isValidToken(){

        if(     !$this->isAutherized()  )
            return false;

        //unauthorized old session
        $this->unautherizeExpiredTokens();


        if( empty($this->user_token) )
            return false;

         $get_user = $this->db->getDataSet('SELECT token_expires_on, current_token
                                            FROM api_users
                                            WHERE user_name = ? AND current_token = ?
                                            LIMIT 1', array($this->user_name, $this->user_token ));

         if( count($get_user) != 1)
            return false;

        $data = $get_user[0];

        if( empty($data['token_expires_on']) || $data['current_token'] != $this->user_token )
            return false;

        //make sure that the token is not expired
        if( !empty($data['token_expires_on']) && time() > $data['token_expires_on'])
            return false;   

    }

    //generate a new token
    private function generateToken(){
            //generate a token          
            $token = md5(uniqid(mt_rand(), true));
            //set expiration date for this token
            $expire_on = time() + $this->keepAlive;

            //Save the new token in the database with expiration time = $this->keepAlive seconds
            $update = $this->db->processQuery('UPDATE api_users
                                               SET current_ip = ?,
                                               current_token = ?,
                                               token_expites_on = ?
                                               WHERE user_name = ?', array($_SERVER['REMOTE_ADDR'], $token, $expire_on ));
            //if the token is saved in the database then send the new token via cURL header.                                   
            if($update){
                //set the token as a header value and then sent it back to the requester.
                $this->token = $token;
                $curl_header = array();
                $curl_header[] = 'API-User-Token: ' . $token;
                curl_setopt($this->ch, CURLOPT_HEADER, true);
                curl_setopt($this->ch, CURLOPT_RETURNTRANSFER, true);
                curl_setopt($this->ch, CURLOPT_HTTPHEADER, $curl_header);

                return $token;
            } else
                return false;   
    }


    //remove old tokens
    private function unautherizeExpiredTokens(){

        $this->db->processQuery('UPDATE api_users
                                 SET current_ip = NULL,
                                 current_token = NULL,
                                 token_expites_on = NULL
                                 WHERE token_expites_on IS NOT NULL AND token_expites_on <= ?', array( time() ) );  

    }

}

?>

要使用这个类,我将通过 API 访问链接执行以下操作。一旦我弄清楚如何准备好 http 数据,就不需要将 $username$password 传递给类,而是从标题中的类中准备好。

因此,access.php 文件将如下所示

include('api.php');


$request = new api('database_name','serverIPaddress');


if( $request->isUserReady() ){
    //process transaction all transactions
    $_POST['notes'];  //// take the data validated it and then insert into the database

    echo 'Bingo!';

} else {
    echo 'You are not authorized to use this API';
}

?>

要使用此 API,客户端必须像这样调用它,client.php 文件将如下所示:

<?php

$curl_header = array();
$curl_header[] = 'API-User-Name: test';
$curl_header[] = 'API-User-Password: password';

$ch = curl_init();
curl_setopt($ch, CURLOPT_URL, "https://mydomainname.com/api/access.php");
//curl_setopt($ch, CURLOPT_POST, 1);
curl_setopt($ch, CURLOPT_HTTPHEADER, $curl_header);
$result = curl_exec($ch);


print_r($result);
curl_close($ch);


?>

【问题讨论】:

标签: php api curl token


【解决方案1】:

只需使用CURLOPT_HTTPHEADER:

curl_setopt($ch, CURLOPT_HTTPHEADER, array(
  'X-THING_ONE: abcdefghijklmnopqrstuvwxyz',
  'X-THING_TWO: 12345678910'
));

我更喜欢将标头设置为curl_setopt 之外的数组,如下所示:

$curl_headers = array();

$curl_headers[] = 'X-THING_ONE: abcdefghijklmnopqrstuvwxyz';
$curl_headers[] = 'X-THING_TWO: 12345678910';

curl_setopt($ch, CURLOPT_HTTPHEADER, $curl_headers);

编辑:好的,看来您知道CURLOPT_HEADER 的工作原理。但是看看你的代码,这里似乎有一个错字。

curl_setopt($this->ch, CURLOPT_HEADER, true);
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($this->ch, CURLOPT_HTTPHEADER, array('API-Token: ' . $this->getToken() ));

为什么CURLOPT_HEADERCURLOPT_HTTPHEADER 中有$this-&gt;chCURLOPT_RETURNTRANSFER 中只有$ch?不应该是这样吗?

curl_setopt($this->ch, CURLOPT_HEADER, true);
curl_setopt($this->ch, CURLOPT_RETURNTRANSFER, true);
curl_setopt($this->ch, CURLOPT_HTTPHEADER, array('API-Token: ' . $this->getToken() ));

编辑:除了那个错字,看来原帖者需要知道如何获取接收方的参数:

另外,一旦发出请求,我如何读取 HTTP 标头?

简单。它们是通过 PHP 中的 $_SERVER 预定义变量访问的。所以你会像这样抓住它们:

$_SERVER['X-API-User-Name'];
$_SERVER['X-API-User-Password'];
$_SERVER['X-API-User-Token'];

您可以通过以下方式检查调试时传递的内容:

echo '<pre>';
print_r($_SERVER);
echo '</pre>';

【讨论】:

  • 感谢您找到错字。我已经纠正了。所以看来我的设置是正确的。我如何准备标题?如何获取 API-Token、API-username、API-password 的值?
  • @Mike 检查我的编辑。密钥在$_SERVER 预定义变量中。
  • $_SERVER 没有返回用户名和密码。我通过展示我如何尝试从不同的服务器访问此 api 但它不起作用来更新问题。我只是得到一个空白页。我究竟做错了什么?我应该用我更新的代码更新问题吗?
  • @Mike:一个想法是您的自定义标头需要更加自定义,以避免与实际的服务器/系统值发生冲突。传统上,在值前添加 X-,如下所示:X-API-User-NameX-API-User-PasswordX-API-User-Token
  • 我按照您的建议添加了 X- 但我无法通过 client.php 页面连接到 API
猜你喜欢
  • 1970-01-01
  • 2010-09-26
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2014-11-02
  • 2015-07-12
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多