Twitter API请求需要401授权

时间:2018-10-30 10:54:44

标签: python twitter request signature http-status-code-401

我正在构建一个Python脚本来自动在Twitter上发布状态更新。我已经仔细阅读并遵循了用于创建签名的API文档,并提出了更新状态的请求。但是在执行请求时出现错误。

HTTP Error 401: Authorization Required

我在Stackoverflow和其他资源上看到了更多类似的问题,但是还没有找到解决该问题的方法。

 def sendTweet(self):

      url = 'https://api.twitter.com/1.1/statuses/update.json'
      message = 'Testing'

      oauths = [
            ['oauth_consumer_key', '1234567890'],
            ['oauth_nonce', self.generate_nonce()],
            ['oauth_signature', ''],
            ['oauth_signature_method', 'HMAC-SHA1'],
            ['oauth_timestamp', str(int(time.time()))],
            ['oauth_token', '1234567890'],
            ['oauth_version', '1.0']
        ]

        signature = self.sign(oauths, message)
        oauths[2][1] = signature

        count = 0
        auth = 'OAuth '
        for oauth in oauths:
            count = count + 1
            if oauth[1]:
                auth = auth + self.quote(oauth[0]) + '="' + self.quote(oauth[1]) + '"'
            if count < len(oauths):
                auth = auth + ', '

        headers = {
            'Authorization': auth,
            'Content-Type': 'application/x-www-form-urlencoded'
        }

        payload = {'status' : message.encode('utf-8')}

        req = urllib2.Request(url, data=urllib.urlencode(payload), headers=headers)

        try:
            response = urllib2.urlopen(req)
            data = response.read()
            print data
        except (urllib2.HTTPError, urllib2.URLError) as err:
            print err

这就是self.sign函数的作用

def sign(self, oauths, message):

        signatureParams = ''
        count = 0

        values = [
            ['oauth_consumer_key', oauths[0][1]],
            ['oauth_nonce', oauths[1][1]],
            ['oauth_signature_method', oauths[3][1]],
            ['oauth_timestamp', oauths[4][1]],
            ['oauth_token', oauths[5][1]],
            ['oauth_version', oauths[6][1]],
            ['status', message]
        ]

        customerSecret = '1234567890'
        tokenSecret = '1234567890'

        for value in oauths:
            count = count + 1
            signatureParams = signatureParams + self.quote(value[0]) + '=' + self.quote(value[1])
            if count < len(oauths):
                signatureParams = signatureParams + '&'

        signature = 'POST&' + self.quote(self.endPoint) + '&' + self.quote(signatureParams)
        signinKey = self.quote(customerSecret) + '&' + self.quote(tokenSecret)

        signed = base64.b64encode(self.signRequest(signinKey, signature))

        return signed

以下是助手功能。

def generate_nonce(self):
    return ''.join([str(random.randint(0, 9)) for i in range(8)])

def quote(self, param):
    return urllib.quote(param.encode('utf8'), safe='')

def signRequest(self, key, raw):
    from hashlib import sha1
    import hmac
    hashed = hmac.new(key, raw, sha1)
    return hashed.digest().encode('base64').rstrip('\n')

这是我打印身份验证字符串时的样子。

OAuth oauth_consumer_key="1234567890",
oauth_nonce="78261149",
oauth_signature="akc2alpFWWdjbElZV2RCMmpGcDc0V1d1S1B3PQ%3D%3D",
oauth_signature_method="HMAC-SHA1", oauth_timestamp="1540896473",
oauth_token="1234567890",
oauth_version="1.0"

那与我阅读Twitter文档时应该的样子完全一样。但是我仍然收到401 Authorization Required错误。我只是看不到这里出了什么问题。

按键的访问级别为Read, write, and direct messages

所有键都替换为1234567890,但在实际代码中,我使用的是真实键。

有人知道我在做什么错吗?预先感谢!

0 个答案:

没有答案