AuthorizationException: 请求中包含的安全令牌已过期

Posted

技术标签:

【中文标题】AuthorizationException: 请求中包含的安全令牌已过期【英文标题】:AuthorizationException: The security token included in the request is expired 【发布时间】:2020-01-11 09:51:36 【问题描述】:

我有一个需要查询 Elasticsearch 和 Dynamodb 的 AWS Fargate。与集群关联的角色有权访问这些服务。过了一会儿(大约 40 分钟),我开始收到此错误 AuthorizationException: TransportError(403, '"message":"The security token included in the request is expired"')。这是我尝试访问 Elasticsearch 时引发的错误,如果我尝试访问 Dynamodb,我不会收到该错误。 我正在使用 boto3 版本 1.9.160,并使用以下代码行获取凭据:

session = boto3.Session()
dynamodb_client_nvirginia = session.client(service_name='dynamodb')
aws_auth = AWS4Auth(
  session.get_credentials().access_key, session.get_credentials().secret_key, 'us-east-1', 'es', session_token=session.get_credentials().token)
elasticsearch_client = Elasticsearch(
  hosts=['host': 'my-elasticsearch-host', 'port': 443],
  http_auth=aws_auth, use_ssl=True, verify_certs=True,
  connection_class=RequestsHttpConnection, timeout=30, max_retries=10, retry_on_timeout=True)

我了解到 boto3 会自动刷新凭据。

作为尝试,我决定在 30 分钟后刷新凭据以连接到 Elasticsearch,但我仍然遇到同样的错误。

我做错了什么?

【问题讨论】:

【参考方案1】:

我决定使用 Singleton 来管理此错误并刷新客户端凭据。我创建了这个类:

class ElasticsearchClientInstanceGenerator:

    __instance = None

    def __init__(self):
        """ Virtually private constructor. """
        if self.__instance is not None:
            raise Exception("This class is a singleton!")
        else:
            ElasticsearchClientInstanceGenerator.__instance = self
        self.__elasticsearch_client = None
        self.__host: Optional[str] = None
        self.__port: Optional[int] = None
        self.__use_ssl: Optional[bool] = None
        self.__verify_certs: Optional[bool] = None

    def generate_elasticsearch_client(self):
        import boto3
        from requests_aws4auth import AWS4Auth
        from elasticsearch import Elasticsearch
        from elasticsearch import RequestsHttpConnection
        session = boto3.Session()
        credentials = session.get_credentials()
        aws_auth = AWS4Auth(
            credentials.access_key, credentials.secret_key, 'us-east-1', 'es', session_token=credentials.token)
        self.__elasticsearch_client = Elasticsearch(
            hosts=['host': self.__host, 'port': self.__port],
            http_auth=aws_auth, use_ssl=self.__use_ssl, verify_certs=self.__verify_certs,
            connection_class=RequestsHttpConnection, timeout=30, max_retries=10, retry_on_timeout=True)

    def setup(self, host: str, port: int, use_ssl: bool, verify_certs: bool):
        self.__host: str = host
        self.__port: int = port
        self.__use_ssl: bool = use_ssl
        self.__verify_certs: bool = verify_certs

    @staticmethod
    def get_instance():
        if ElasticsearchClientInstanceGenerator.__instance is None:
            ElasticsearchClientInstanceGenerator()
        return ElasticsearchClientInstanceGenerator.__instance

然后我创建了这个装饰器

def try_until_succeed(func):

    def catch_authorization_exception(*args, **kwargs):

        for i in range(0,10):
            try:
                data = func(*args, **kwargs)
                return data
            except AuthorizationException as ae:
               ElasticsearchClientInstanceGenerator.get_instance().generate_elasticsearch_client()
        raise Execption('Elasticsearch general exception')
    return catch_authorization_exception

我在每个需要与 Elasticcearch 连接的方法上都使用了装饰器。每当我收到 AuthorizationException 错误时,都会刷新客户端凭据,然后我可以连接到 Elasticsearch。

【讨论】:

以上是关于AuthorizationException: 请求中包含的安全令牌已过期的主要内容,如果未能解决你的问题,请参考以下文章

Facebook Marketing Api - 循环运行抛出“AuthorizationException 发生未知错误”。 (第二次)

Shiro报异常org.apache.shiro.authz.AuthorizationException: Not authorized to invoke method:

自定义 Laravel FormRequest 自动验证方法

使用 oauth2-client 作为隐式授权类型时解决 [authorization_request_not_found]

高手进,追加200分 机翻者请自重

python 中 请输入一个数字 请输入第二个数字 请输入进行的计算符合(1+ 2-):