Невозможно получить OAuth "Request Token" при работе с API Tumblr с использованием Python - PullRequest
5 голосов
/ 05 сентября 2011

До сих пор я использовал библиотеки для обработки OAuth, но в последнее время я копаю глубже, пытаясь понять основной процесс OAuth.В настоящее время я пытаюсь подключиться к Tumblr API v2 , используя OAuth 1.0a с этим простым кодом:

import urllib, urllib2, time, random, hmac, base64, hashlib

def makenonce():
    random_number = ''.join( str( random.randint( 0, 9 ) ) for _ in range( 40 ) )
    m = hashlib.md5( str( time.time() ) + str( random_number ) )
    return m.hexdigest()

def encodeparams(s):
    return urllib.quote( str( s ), safe='~' )

# Actual key and secret from a test app created using a dummy Tumblr account
consumer_key = '97oAujQhSaQNv4XDXzCjdZlOxwNyhobmDwmueJBCHWsFFsW7Ly'
consumer_secret = '5q1dpF659SOgSUb0Eo52aAyoud8N8QOuJu6enCG92aDR6WoMlf'

#oauth URLs
request_tokenURL = 'http://www.tumblr.com/oauth/request_token'

#oauth params
oauth_parameters = {
            'oauth_consumer_key'     : consumer_key,
            'oauth_nonce'            : makenonce(),
            'oauth_timestamp'        : str(int(time.time())),
            'oauth_signature_method' : "HMAC-SHA1",
            'oauth_version'          : "1.0"
            }

normalized_parameters = encodeparams( '&'.join( ['%s=%s' % ( encodeparams( str( k ) ), encodeparams( str( oauth_parameters[k] ) ) ) for k in sorted( oauth_parameters )] ) )
# Since I'm focusing only on getting the request token for now, I set this to POST.
normalized_http_method = 'POST'
normalized_http_url = encodeparams( request_tokenURL )
signature_base_string = '&'.join( [normalized_http_method, normalized_http_url, normalized_parameters] )
oauth_key = consumer_secret + '&'
hashed = hmac.new( oauth_key, signature_base_string, hashlib.sha1 )
oauth_parameters['oauth_signature'] = base64.b64encode( hashed.digest() )
oauth_header = 'Authorization: OAuth realm="http://www.tumblr.com",' + 'oauth_nonce="' + oauth_parameters['oauth_nonce'] + '",' + 'oauth_timestamp="' + oauth_parameters['oauth_timestamp'] + '",' + 'oauth_consumer_key="' + oauth_parameters['oauth_consumer_key'] + '",' + 'oauth_signature_method="HMAC-SHA1",oauth_version="1.0",oauth_signature="' + oauth_parameters['oauth_signature'] +'"'

# sample oauth_header generated by the code above:
# Authorization: OAuth realm="http://www.tumblr.com",oauth_nonce="c200a0e06f30b84b851ac3e99a71054b",oauth_timestamp="1315231855",oauth_consumer_key="97oAujQhSaQNv4XDXzCjdZlOxwNyhobmDwmueJBCHWsFFsW7Ly",oauth_signature_method="HMAC-SHA1",oauth_version="1.0",oauth_signature="kVAlmwolCX0WJIvTF9MB2UV5rnU="


req = urllib2.Request( request_tokenURL )
req.add_header( 'Authorization', oauth_header )
# If all goes well, Tumblr should send me the oauth request token.
print urllib2.urlopen( req ).read()

Вместо токена OAuth-запроса Tumblr возвращает Ошибка HTTP401: Несанкционированный .

Вещи, которые я пытался безуспешно:

  1. Изменил oauth_version с "1.0" на "1.0a" и снова вернул обратно.
  2. Руководство по OAuth обязало добавить '&' в конце consumer_secret, чтобы получить oauth_key.Позже я попытался удалить '&', чтобы понять, имеет ли это какое-то значение.
  3. Проверено, были ли отсортированы параметры OAuth, и они были.
  4. Не добавлена ​​строка "Авторизация:" вoauth_header, затем добавил его позже.Ничего не изменилось.

Где я ошибся?

1 Ответ

4 голосов
/ 10 сентября 2011

Решено после двух простых изменений в приведенном выше коде:

  1. normalized_http_method = 'GET' # не POST
  2. oauth_header = 'OAuth realm="http://www...' # Слово «Авторизация» не нужно. Я снял это ранее, как указано в «Вещи, которые я пытался без успеха», но ошибка, перечисленная в (1), сбила меня с толку. Решив (1), я понял, что «Авторизация» действительно не нужна.

Токен OAuth Request Tumblr прислал мне, когда я, наконец, понял это правильно: oauth_token=mbRUgyDkPePfkEztiLELMqUl1kyNXEcaTCCwpb7SoXDF9mhiTF&oauth_token_secret=5pXllXGKA8orAaUat1G7ckIfMfYup8juMBAgEELUkeMZoC3pv6&oauth_callback_confirmed=true

Это одноразовый токен, и я перечислил его здесь только для полноты картины.

...