1

我是 PHP 世界的大部分新手。我目前正在尝试将 Silverpop 的 API 与我们页面上的表单进行交互。以下是我到目前为止的代码。

问题是,虽然身份验证工作得非常完美,但每当我尝试发布 API 所需的 XML 时,总是会返回一个错误消息:“会话已过期或无效”。我将在代码中标记出错误发生的区域。

我花了最后半天的时间进行研究,但我无法弄清楚我做错了什么或可能遗漏了什么。我被模糊地告知它与浏览器中的输出有关,因为我在一个空白(无 html)php 文件中测试它之前的 PHP 标头函数(如 cURL)。

<?php
// Vars
$firstname = 'a';
$lastname = 'a';
$email = 'a@b.com';

// cURL
function curl($url,$header,$postbody) {
    $ch = curl_init();
    curl_setopt($ch, CURLOPT_URL, $url);
    curl_setopt($ch, CURLOPT_HTTPHEADER, array($header));
    curl_setopt($ch, CURLOPT_POST, true);
    curl_setopt($ch, CURLOPT_POSTFIELDS, $postbody);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 300);
    $ret = curl_exec($ch);
    curl_close($ch);

    return $ret;
}

///# SILVERPOP API #///

// SILVERPOP API URLs
    // get_token uses oauth to retrieve an access token (works just fine)
$get_token = 'https://api1.silverpop.com/oauth/token?*inforemoved*';
$xmlapi = 'https://api1.silverpop.com/XMLAPI?Authorization=';

// XML STRINGS
$xml_post = '<Envelope><Body><AddRecipient><LIST_ID>database#</LIST_ID><CREATED_FROM>1</CREATED_FROM><SEND_AUTOREPLY>true</SEND_AUTOREPLY><UPDATE_IF_FOUND>true</UPDATE_IF_FOUND><COLUMN><NAME>Name</NAME><VALUE>'.$firstname.'</VALUE></COLUMN><COLUMN><NAME>LastName</NAME><VALUE>'.$lastname.'</VALUE></COLUMN><COLUMN><NAME>Email</NAME><VALUE>'.$email.'</VALUE></COLUMN><COLUMN><NAME>Lead Source</NAME><VALUE>Lead_SqueezePage_5Questions</VALUE></COLUMN></AddRecipient></Body></Envelope>';
$xml_done = '<Envelope><Body><Logout/></Body></Envelope>';

// HEADER VALUES
$h_access = 'Content-Type:x-www-form-urlencoded';
$h_api = 'Content-Type:text/xml;charset=UTF-8';

// Get API Access token
$auth = curl($get_token,$h_access,'');

// Pull access_token from the return string
$auth = explode('"', $auth);

for ($i=0; $i < count($auth); $i++) {
    if ( $auth[$i] == "access_token" ) {
        $access_token = $auth[$i + 2];
        break;
    } 
}

//Append token to URL unless auth failed, then die
if ( $access_token != NULL) {

    $xmlapi .= $access_token;

} else {
    // Logout API session - SESSION ERROR HERE
    $logout = curl($xmlapi,$h_api,$xml_done);
    //echo 'Authentication Failed!';
    die;
}

// Send Customer Data - SESSION ERROR HERE
$inject = curl($xmlapi,$h_api,$xml_post);

// Logout API Session - SESSION ERROR HERE
$logout = curl($xmlapi,$h_api,$xml_done);

///# END SILVERPOP API #///
?>
4

3 回答 3

5

我能够让 cURL 正常工作。我希望下面的代码可以帮助那些第一次使用 Silverpop API 并且遇到困难的人。

从 Silverpop 检索访问密钥:

    // POST Fields
    $fields = array(
        'client_id' => CLIENT_KEY,
        'client_secret' => CLIENT_SECRET,
        'refresh_token' => REFRESH_TOKEN,
        'grant_type' => 'refresh_token'
    );

    // Init cURL
    $ch = curl_init();

    // Set Options
    curl_setopt($ch, CURLOPT_URL, ACCESS_KEY_URL);
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
    curl_setopt($ch, CURLOPT_POST, count($fields));
    curl_setopt($ch, CURLOPT_POSTFIELDS, http_build_query($fields));

    // Execute cURL
    $result = curl_exec($ch);

    // Check for HTTP and sever errors here if you wish
    // with curl_getinfo($ch, CURLINFO_HTTP_CODE)

    // Close Connection
    curl_close($ch);

    // Now you can work with the returned $result string

向 Silverpop 发送 XML 请求信封:

// Set POST header
        $header = array(
            'Content-Type:text/xml;charset=UTF-8','Authorization: Bearer '.$access_key
        );

        // init cURL
        $ch = curl_init();

        // set cURL options
        curl_setopt($ch, CURLOPT_URL, REQUEST_URL);
        curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
        curl_setopt($ch, CURLOPT_HTTPHEADER, $header);
        curl_setopt($ch, CURLOPT_POST, count($xml_post));
        curl_setopt($ch, CURLOPT_POSTFIELDS, $xml_post);
于 2014-02-21T17:30:16.810 回答
0

我认为您需要发回 cookie 以保持会话有效。尝试将此选项添加到您的 curl 请求中。

CURLOPT_COOKIEJAR

这个种类是 cookie 应该存储的文件名。如果服务器设置任何它们将被写入此文件,如果它不存在则将被创建。

CURLOPT_COOKIEFILE

cookie jar 设置是 curl 将 cookie 写入的位置,但 curl 需要单独设置才能将 cookie 发送回服务器。这是 CURLOPT_COOKIEFILE 设置。如果未设置,则不会向服务器发送任何 cookie。如果文件不存在,则不会发出错误。

curl_setopt($ch, CURLOPT_COOKIEJAR, '/tmp/cookies.txt');
curl_setopt($ch, CURLOPT_COOKIEFILE, '/tmp/cookies.txt');

您将需要/tmp/cookies.txt用自己的路径替换。

于 2013-11-08T14:32:01.143 回答
0

您在一小时内只能请求一定数量的访问令牌。
在您的代码中,您似乎正在混合使用 OAuth 授权和 jsession 登录/注销。

如果您有访问令牌(通过 OAuth 请求),请将您的访问令牌保存在会话中,因为它的有效期为 4 小时。curl 标头需要在请求中包含令牌:

$curlHeaders[] = "Authorization: Bearer {$accessToken}"; 

当您使用用户名/密码对 api 进行身份验证时,将使用 xml logout 命令。login 命令将为您提供一个 jsessionid,然后您可以在每个请求中使用它(而不是访问令牌)。

我当前使用的脚本将访问令牌与到期时间/日期一起保存在会话中。当 Silverpop 连接器触发一个 xml 命令时,它会检查访问令牌,随后会检查它是否即将过期或已经过期,如果是,则请求一个新的。

$getNewToken = true; //assume you need a new access token
//check if the session has an access token
if (!empty($_SESSION['access'])) {

//check access token is present and not expired
if ( ($expiry > (time() + 5)) && (!empty($token)) ) { 

//set current access token in the Connector, no new access token required
SilverpopConnector::getInstance()->setAccessToken($token,$expiry); 
$getNewToken = false;

//if the session wasn't set or token in session invalid/expired:
if ($getNewToken) { 
//get access token & expiry from the Connector (will authenticate if expired/invalid)
$accessToken = SilverpopConnector::getInstance()->getAccessToken();
$tokenExpiry = SilverpopConnector::getInstance()->getAccessTokenExpiry();
//2a. if successful, set in session
//2b. if not successful, check if you're connected/re-connect

我上面提到的 Silverpop 连接器是 mrmarkfrench 在 GitHub 上创建的脚本。请查看https://github.com/mrmarkfrench/silverpop-php-connector

希望这会有所帮助,
桑德拉。

于 2014-02-21T15:36:48.580 回答