«Неизвестный заголовок авторизации» при извлечении контактов Google с использованием php, (zend) gdata и 2 legged oauth - PullRequest
2 голосов
/ 20 сентября 2011

Я пишу приложение 2 Legged OAuth (приложение Google Marketplace) с использованием библиотеки zend gdata.Мне нужно получить контакты Google.

Код, указанный ниже.

require_once 'Zend/Oauth/Consumer.php';
$oauthOptions = array(
    'requestScheme' => Zend_Oauth::REQUEST_SCHEME_HEADER,
    'version' => '1.0',
    'signatureMethod' => 'HMAC-SHA1',
    'consumerKey' => $CONSUMER_KEY,
    'consumerSecret' => $CONSUMER_SECRET
);

$consumer = new Zend_Oauth_Consumer($oauthOptions);
$token = new Zend_Oauth_Token_Access();
$httpClient = $token->getHttpClient($oauthOptions);

$url = 'http://www.google.com/m8/feeds/contacts/default/full';

require_once 'Zend/Gdata/Gapps.php';
$gdata = new Zend_Gdata($httpClient);

require_once 'Zend/Gdata/Query.php';
require_once 'Zend/Gdata/Feed.php';
require_once 'Zend/Gdata/App.php';

$query = new Zend_Gdata_Query($url);
try {
    $feed = $gdata->getFeed($query);
} catch(Zend_Gdata_App_Exception $ex){
    print_r($ex->getMessage());
}

Я получаю следующую ошибку:

Expected response code 200, got 401
Unknown authorization header
Error 401



Заголовки HTTP, отправленные при выполнении вышеуказанного кода:

GET /m8/feeds/contacts/default/full HTTP/1.1
Host: www.google.com
Connection: close
User-Agent: MyCompany-MyApp-1.0 Zend_Framework_Gdata/1.11.0dev
Accept-encoding: identity
Authorization: OAuth realm="",oauth_consumer_key="686518909188.apps.googleusercontent.com",oauth_nonce="fc99e10f42cdb01c7f3ce1ab2775e616",oauth_signature_method="HMAC-SHA1",oauth_timestamp="1316583946",oauth_version="1.0",oauth_signature="hlbTvPExy4r4%2FyY1ddEsy1AJhf4%3D"

HTTP-ответ получен:

HTTP/1.1 401 Unknown authorization header
Content-Type: text/html; charset=UTF-8
Date: Wed, 21 Sep 2011 05:45:47 GMT
Expires: Wed, 21 Sep 2011 05:45:47 GMT
Cache-Control: private, max-age=0
X-Content-Type-Options: nosniff
X-Frame-Options: SAMEORIGIN
X-XSS-Protection: 1; mode=block
Server: GSE
Connection: close

<HTML>
<HEAD>
<TITLE>Unknown authorization header</TITLE>
</HEAD>
<BODY BGCOLOR="#FFFFFF" TEXT="#000000">
<H1>Unknown authorization header</H1>
<H2>Error 401</H2>
</BODY>
</HTML>


Пожалуйста, дайте мне знать, что мне здесь не хватает.

Ответы [ 5 ]

2 голосов
/ 10 ноября 2011

Похоже, что параметр запроса "xoauth_requestor_id" не устанавливается при использовании двухстороннего OAuth (xoauth).

Я попытался установить этот параметр запроса для объекта HttpClient, который передается конструктору Zend_Gdata_Calendar:

$httpClient->setParameterGet('xoauth_requestor_id', $userEmail);

Но, к сожалению, этот чистый подход не сработал. Похоже, что класс Zend_GData_Calendar не полностью использует введенный объект HttpClient.

Итак, я прибег к расширению Zend_GData_Calendar и создаю вместо него свою собственную версию. Я не полностью протестировал всю функциональность, но вот что у меня есть, и на нее легко можно построить:

/**
 * Slightly modified class which extends Zend_Gdata_Calendar. Fixes a problem with the request URI when using 2-legged
 * OAuth (xoauth). Adds on the xoauth_requestor_id query parameter.
 */
class Xoauth_Gdata_Calendar extends Zend_Gdata_Calendar {
    protected $requestorId;

    /**
     * Set the xoauth_requestor_id for requests
     *
     * @param string $requestorId
     */
    public function setRequestorId($requestorId) {
        $this->requestorId = $requestorId;
    }

    /**
     * Get the currently set xoauth_requestor_id
     *
     * @return string
     */
    public function getRequestorId() {
        return $this->requestorId;
    }

    /**
     * {@inheritdoc} extended to also include the xoauth_requestor_id query parameter in the request URI. This fixes
     * an authentication issue when using 2-legged OAuth (xoauth).
     *
     * @return string|Zend_Gdata_App_Feed
     */
    public function getCalendarListFeed() {
        $uri = self::CALENDAR_FEED_URI . '/default';
        if($this->requestorId) {
            $uri .= '?xoauth_requestor_id=' . urlencode($this->requestorId);
        }

        return parent::getFeed($uri,'Zend_Gdata_Calendar_ListFeed');
    }
}

Затем, чтобы использовать это, вы создаете экземпляр этого класса, а не базовый класс Zend_Gdata_Calendar. Вам просто нужно вызвать setRequestorId ($ usersEmail), например:

$calendarClient = new Xoauth_Gdata_Calendar($httpClient);
$calendarClient->setRequestorId($userEmail);
0 голосов
/ 11 мая 2012

Исправление Xoauth_Gdata_Calendar работает.
И если вы хотите опубликовать событие, вам также необходимо переопределить функцию insertEvent().

/**
 * {@inheritdoc} extended to also include the xoauth_requestor_id query
 * parameter in the request URI. This fixes an authentication issue 
 * when using 2-legged OAuth (xoauth).
 *
 * @return string|Zend_Gdata_Calendar_EventEntry
 */

public function insertEvent($event, $uri=null) {
    if ($uri == null) {
      $uri = $this->_defaultPostUri;
      if($this->requestorId) {
          $uri .= '?xoauth_requestor_id=' . urlencode($this->requestorId);
      }
    }
    return parent::insertEntry($event, $uri, 'Zend_Gdata_Calendar_EventEntry');
}
0 голосов
/ 02 марта 2012

Существует причина, по которой следующий чистый подход не работает

$httpClient->setParameterGet('xoauth_requestor_id', $userEmail);

В строке 638 Zend / Gdata / App.php он делает следующее

// Make sure the HTTP client object is 'clean' before making a request
// In addition to standard headers to reset via resetParameters(),
// also reset the Slug and If-Match headers
$this->_httpClient->resetParameters();

Поскольку это вызывается для каждого запроса, то, что вы устанавливаете в клиенте, стирается.

0 голосов
/ 10 ноября 2011

Я отправил Zend сообщение об ошибке по этому вопросу.См. ZF-11880 .Пожалуйста, проголосуйте за эту ошибку, чтобы попытаться повысить ее приоритет.

0 голосов
/ 21 сентября 2011

Я столкнулся с той же проблемой при попытке получить доступ к Календарю, поэтому расстраивает. Я использую последнюю загрузку с Zen (1.11.10).

require_once 'Zend/Oauth/Consumer.php';
    require_once 'Zend/Gdata/Calendar.php';

    $CONSUMER_KEY = 'mydomain.com';
    $CONSUMER_SECRET = 'mysecret';
    $USER = 'user@mydomain.com';

    $oauthOptions = array(
        'requestScheme' => Zend_Oauth::REQUEST_SCHEME_HEADER,
        'version' => '1.0',
        'signatureMethod' => 'HMAC-SHA1',
        'consumerKey' => $CONSUMER_KEY,
        'consumerSecret' => $CONSUMER_SECRET
    );

    $consumer = new Zend_Oauth_Consumer($oauthOptions);
    $token = new Zend_Oauth_Token_Access();
    $httpClient = $token->getHttpClient($oauthOptions);


    // Create an instance of the Calendar service
    $service = new Zend_Gdata_Calendar($httpClient);

    try {
        $listFeed= $service->getCalendarListFeed();
    } catch (Zend_Gdata_App_Exception $e) {
        echo "Error: " . $e->getMessage();
    }
...