From 7314d90273245f897d78879702fc466dc809a48f Mon Sep 17 00:00:00 2001 From: hjpotter92 Date: Tue, 7 Jul 2020 01:38:29 +0530 Subject: [PATCH] Making use of service resource over the low level client for SQS. References #39. Signed-off-by: hjpotter92 --- README.rst | 12 +-- sqs_launcher/__init__.py | 25 ++--- sqs_listener/__init__.py | 198 ++++++++++++++++----------------------- 3 files changed, 100 insertions(+), 135 deletions(-) diff --git a/README.rst b/README.rst index 5918212..103e695 100644 --- a/README.rst +++ b/README.rst @@ -31,7 +31,7 @@ Here is a basic code sample: **Standard Listener** -:: +.. code:: python from sqs_listener import SqsListener @@ -44,7 +44,7 @@ Here is a basic code sample: **Error Listener** -:: +.. code:: python from sqs_listener import SqsListener class MyErrorListener(SqsListener): @@ -75,7 +75,7 @@ Running as a Daemon | Typically, in a production environment, you'll want to listen to an SQS queue with a daemonized process. The simplest way to do this is by running the listener in a detached process. On a typical Linux distribution it might look like this: -| +| ``nohup python my_listener.py > listener.log &`` | And saving the resulting process id for later (for stopping the listener via the ``kill`` command). | @@ -94,7 +94,7 @@ Logging | | For instance: -:: +.. code:: python logger = logging.getLogger('sqs_listener') logger.setLevel(logging.INFO) @@ -111,7 +111,7 @@ Logging | | Or to a log file: -:: +.. code:: python logger = logging.getLogger('sqs_listener') logger.setLevel(logging.INFO) @@ -140,7 +140,7 @@ Sending messages **Launcher Example** -:: +.. code:: python from sqs_launcher import SqsLauncher diff --git a/sqs_launcher/__init__.py b/sqs_launcher/__init__.py index 0496c35..4d2a027 100644 --- a/sqs_launcher/__init__.py +++ b/sqs_launcher/__init__.py @@ -45,32 +45,36 @@ def __init__(self, queue=None, queue_url=None, create_queue=False, visibility_ti raise EnvironmentError('Environment variable `AWS_ACCOUNT_ID` not set and no role found.') # new session for each instantiation self._session = boto3.session.Session() + self._resource = self._session.resource('sqs') self._client = self._session.client('sqs') self._queue_name = queue self._queue_url = queue_url + self._queue = None if not queue_url: - queues = self._client.list_queues(QueueNamePrefix=self._queue_name) + queues = self._resource.queues.filter(QueueNamePrefix=self._queue_name) exists = False - for q in queues.get('QueueUrls', []): - qname = q.split('/')[-1] + for q in queues: + qname = q.url.split('/')[-1] if qname == self._queue_name: exists = True - self._queue_url = q - + self._queue_url = q.url + self._queue = q if not exists: if create_queue: - q = self._client.create_queue( + q = self._resource.create_queue( QueueName=self._queue_name, Attributes={ - 'VisibilityTimeout': visibility_timeout # 10 minutes - } + 'VisibilityTimeout': visibility_timeout, # 10 minutes + }, ) - self._queue_url = q['QueueUrl'] + self._queue_url = q.url + self._queue = q else: raise ValueError('No queue found with name ' + self._queue_name) else: self._queue_name = self._get_queue_name_from_url(queue_url) + self._queue = self._resource.Queue(queue_url) def launch_message(self, message, **kwargs): """ @@ -81,8 +85,7 @@ def launch_message(self, message, **kwargs): :return: (dict) the message response from SQS """ sqs_logger.info("Sending message to queue " + self._queue_name) - return self._client.send_message( - QueueUrl=self._queue_url, + return self._queue.send_message( MessageBody=json.dumps(message), **kwargs, ) diff --git a/sqs_listener/__init__.py b/sqs_listener/__init__.py index 6620b90..23eb506 100644 --- a/sqs_listener/__init__.py +++ b/sqs_listener/__init__.py @@ -27,6 +27,7 @@ sqs_logger = logging.getLogger('sqs_listener') + class SqsListener(object): __metaclass__ = ABCMeta @@ -38,16 +39,16 @@ def __init__(self, queue, **kwargs): aws_access_key = kwargs.get('aws_access_key', '') aws_secret_key = kwargs.get('aws_secret_key', '') - if len(aws_access_key) != 0 and len(aws_secret_key) != 0: + if all(aws_access_key, aws_secret_key): boto3_session = boto3.Session( aws_access_key_id=aws_access_key, - aws_secret_access_key=aws_secret_key + aws_secret_access_key=aws_secret_key, ) else: if (not os.environ.get('AWS_ACCOUNT_ID', None) and not ('iam-role' == boto3.Session().get_credentials().method)): raise EnvironmentError('Environment variable `AWS_ACCOUNT_ID` not set and no role found.') - + self._queue = None # The SQS Queue resource self._queue_name = queue self._poll_interval = kwargs.get("interval", 60) self._queue_visibility_timeout = kwargs.get('visibility_timeout', '600') @@ -67,135 +68,109 @@ def __init__(self, queue, **kwargs): else: self._session = boto3.session.Session() self._region_name = kwargs.get('region_name', self._session.region_name) - self._client = self._initialize_client() - + self._resource = self._initialize_resource() - def _initialize_client(self): + def _initialize_resource(self): # new session for each instantiation ssl = True if self._region_name == 'elasticmq': ssl = False - sqs = self._session.client('sqs', region_name=self._region_name, endpoint_url=self._endpoint_name, use_ssl=ssl) - queues = sqs.list_queues(QueueNamePrefix=self._queue_name) - mainQueueExists = False - errorQueueExists = False - if 'QueueUrls' in queues: - for q in queues['QueueUrls']: - qname = q.split('/')[-1] - if qname == self._queue_name: - mainQueueExists = True - if self._error_queue_name and qname == self._error_queue_name: - errorQueueExists = True - - - # create queue if necessary. + sqs = self._session.resource('sqs', region_name=self._region_name, endpoint_url=self._endpoint_name, use_ssl=ssl) + queues = sqs.queues.filter(QueueNamePrefix=self._queue_name) + main_queue_exists = False + error_queue_exists = False + for q in queues: + qname = q.url.split('/')[-1] + if qname == self._queue_name: + main_queue_exists = True + if self._error_queue_name and qname == self._error_queue_name: + error_queue_exists = True + + # create queue if necessary. # creation is idempotent, no harm in calling on a queue if it already exists. if self._queue_url is None: - if not mainQueueExists: + if not main_queue_exists: sqs_logger.warning("main queue not found, creating now") - + queue_attributes = { + 'VisibilityTimeout': self._queue_visibility_timeout, # 10 minutes + } # is this a fifo queue? if self._queue_name.endswith(".fifo"): - fifoQueue="true" - q = sqs.create_queue( - QueueName=self._queue_name, - Attributes={ - 'VisibilityTimeout': self._queue_visibility_timeout, # 10 minutes - 'FifoQueue':fifoQueue - } - ) - else: - # need to avoid FifoQueue property for normal non-fifo queues - q = sqs.create_queue( - QueueName=self._queue_name, - Attributes={ - 'VisibilityTimeout': self._queue_visibility_timeout, # 10 minutes - } - ) - self._queue_url = q['QueueUrl'] - - if self._error_queue_name and not errorQueueExists: + queue_attributes["FifoQueue"] = "true" + q = sqs.create_queue( + QueueName=self._queue_name, + Attributes=queue_attributes, + ) + self._queue_url = q.url + + if self._error_queue_name and not error_queue_exists: sqs_logger.warning("error queue not found, creating now") q = sqs.create_queue( QueueName=self._error_queue_name, Attributes={ - 'VisibilityTimeout': self._queue_visibility_timeout # 10 minutes - } + 'VisibilityTimeout': self._queue_visibility_timeout, # 10 minutes + }, ) if self._queue_url is None: - if os.environ.get('AWS_ACCOUNT_ID', None): - qs = sqs.get_queue_url(QueueName=self._queue_name, - QueueOwnerAWSAccountId=os.environ.get('AWS_ACCOUNT_ID', None)) - else: - qs = sqs.get_queue_url(QueueName=self._queue_name) - self._queue_url = qs['QueueUrl'] + qs = sqs.get_queue_by_name( + QueueName=self._queue_name, + QueueOwnerAWSAccountId=os.environ.get('AWS_ACCOUNT_ID', None), + ) + self._queue_url = qs.url + self._queue = sqs.Queue(self._queue_url) return sqs def _start_listening(self): - # TODO consider incorporating output processing from here: https://github.com/debrouwere/sqs-antenna/blob/master/antenna/__init__.py + # TODO consider incorporating output processing from here: + # https://github.com/debrouwere/sqs-antenna/blob/master/antenna/__init__.py while True: - # calling with WaitTimeSecconds of zero show the same behavior as + # calling with `WaitTimeSecconds` of zero show the same behavior as # not specifiying a wait time, ie: short polling - messages = self._client.receive_message( - QueueUrl=self._queue_url, - MessageAttributeNames=self._message_attribute_names, + messages = self._queue.receive_message( AttributeNames=self._attribute_names, + MessageAttributeNames=self._message_attribute_names, WaitTimeSeconds=self._wait_time, - MaxNumberOfMessages=self._max_number_of_messages + MaxNumberOfMessages=self._max_number_of_messages, ) - if 'Messages' in messages: - - sqs_logger.debug(messages) - continue - sqs_logger.info("{} messages received".format(len(messages['Messages']))) - for m in messages['Messages']: - receipt_handle = m['ReceiptHandle'] - m_body = m['Body'] - message_attribs = None - attribs = None - - # catch problems with malformed JSON, usually a result of someone writing poor JSON directly in the AWS console - try: - params_dict = json.loads(m_body) - except: - sqs_logger.warning("Unable to parse message - JSON is not formatted properly") - continue - if 'MessageAttributes' in m: - message_attribs = m['MessageAttributes'] - if 'Attributes' in m: - attribs = m['Attributes'] - try: - if self._force_delete: - self._client.delete_message( - QueueUrl=self._queue_url, - ReceiptHandle=receipt_handle - ) - self.handle_message(params_dict, message_attribs, attribs) - else: - self.handle_message(params_dict, message_attribs, attribs) - self._client.delete_message( - QueueUrl=self._queue_url, - ReceiptHandle=receipt_handle - ) - except Exception as ex: - # need exception logtype to log stack trace - sqs_logger.exception(ex) - if self._error_queue_name: - exc_type, exc_obj, exc_tb = sys.exc_info() - - sqs_logger.info( "Pushing exception to error queue") - error_launcher = SqsLauncher(queue=self._error_queue_name, create_queue=True) - error_launcher.launch_message( - { - 'exception_type': str(exc_type), - 'error_message': str(ex.args) - } - ) - - else: + if not messages: time.sleep(self._poll_interval) + continue + sqs_logger.debug(messages) + sqs_logger.info("{} messages received".format(len(messages['Messages']))) + for m in messages: + receipt_handle = m.receipt_handle + m_body = m.body + message_attribs = m.message_attributes + attribs = m.attributes + # catch problems with malformed JSON, usually a result of someone writing poor JSON directly in the AWS console + try: + params_dict = json.loads(m_body) + except: + sqs_logger.warning("Unable to parse message - JSON is not formatted properly") + continue + try: + if self._force_delete: + m.delete() + self.handle_message(params_dict, message_attribs, attribs) + else: + self.handle_message(params_dict, message_attribs, attribs) + m.delete() + except Exception as ex: + # need exception logtype to log stack trace + sqs_logger.exception(ex) + if self._error_queue_name: + exc_type, exc_obj, exc_tb = sys.exc_info() + + sqs_logger.info( "Pushing exception to error queue") + error_launcher = SqsLauncher(queue=self._error_queue_name, create_queue=True) + error_launcher.launch_message( + { + 'exception_type': str(exc_type), + 'error_message': str(ex.args) + } + ) def listen(self): sqs_logger.info( "Listening to queue " + self._queue_name) @@ -204,19 +179,6 @@ def listen(self): self._start_listening() - def _prepare_logger(self): - logger = logging.getLogger('eg_daemon') - logger.setLevel(logging.INFO) - - sh = logging.StreamHandler(sys.stdout) - sh.setLevel(logging.INFO) - - formatstr = '[%(asctime)s - %(name)s - %(levelname)s] %(message)s' - formatter = logging.Formatter(formatstr) - - sh.setFormatter(formatter) - logger.addHandler(sh) - @abstractmethod def handle_message(self, body, attributes, messages_attributes): """