Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Mocking boto3 S3 client method Python

I'm trying to mock a singluar method from the boto3 s3 client object to throw an exception. But I need all other methods for this class to work as normal.

This is so I can test a singular Exception test when and error occurs performing a upload_part_copy

1st Attempt

import boto3 from mock import patch  with patch('botocore.client.S3.upload_part_copy', side_effect=Exception('Error Uploading')) as mock:     client = boto3.client('s3')     # Should return actual result     o = client.get_object(Bucket='my-bucket', Key='my-key')     # Should return mocked exception     e = client.upload_part_copy() 

However this gives the following error:

ImportError: No module named S3 

2nd Attempt

After looking at the botocore.client.py source code I found that it is doing something clever and the method upload_part_copy does not exist. I found that it seems to call BaseClient._make_api_call instead so I tried to mock that

import boto3 from mock import patch  with patch('botocore.client.BaseClient._make_api_call', side_effect=Exception('Error Uploading')) as mock:     client = boto3.client('s3')     # Should return actual result     o = client.get_object(Bucket='my-bucket', Key='my-key')     # Should return mocked exception     e = client.upload_part_copy() 

This throws an exception... but on the get_object which I want to avoid.

Any ideas about how I can only throw the exception on the upload_part_copy method?

like image 536
ptimson Avatar asked May 10 '16 15:05

ptimson


People also ask

Should I use Boto3 resource or client?

To summarize, resources are higher-level abstractions of AWS services compared to clients. Resources are the recommended pattern to use boto3 as you don't have to worry about a lot of the underlying details when interacting with AWS services. As a result, code written with Resources tends to be simpler.

What is Boto3 client Python?

Boto3 is the name of the Python SDK for AWS. It allows you to directly create, update, and delete AWS resources from your Python scripts.


2 Answers

Botocore has a client stubber you can use for just this purpose: docs.

Here's an example of putting an error in:

import boto3 from botocore.stub import Stubber  client = boto3.client('s3') stubber = Stubber(client) stubber.add_client_error('upload_part_copy') stubber.activate()  # Will raise a ClientError client.upload_part_copy() 

Here's an example of putting a normal response in. Additionally, the stubber can now be used in a context. It's important to note that the stubber will verify, so far as it is able, that your provided response matches what the service will actually return. This isn't perfect, but it will protect you from inserting total nonsense responses.

import boto3 from botocore.stub import Stubber  client = boto3.client('s3') stubber = Stubber(client) list_buckets_response = {     "Owner": {         "DisplayName": "name",         "ID": "EXAMPLE123"     },     "Buckets": [{         "CreationDate": "2016-05-25T16:55:48.000Z",         "Name": "foo"     }] } expected_params = {} stubber.add_response('list_buckets', list_buckets_response, expected_params)  with stubber:     response = client.list_buckets()  assert response == list_buckets_response 
like image 62
Jordon Phillips Avatar answered Sep 19 '22 22:09

Jordon Phillips


As soon as I posted on here I managed to come up with a solution. Here it is hope it helps :)

import botocore from botocore.exceptions import ClientError from mock import patch import boto3  orig = botocore.client.BaseClient._make_api_call  def mock_make_api_call(self, operation_name, kwarg):     if operation_name == 'UploadPartCopy':         parsed_response = {'Error': {'Code': '500', 'Message': 'Error Uploading'}}         raise ClientError(parsed_response, operation_name)     return orig(self, operation_name, kwarg)  with patch('botocore.client.BaseClient._make_api_call', new=mock_make_api_call):     client = boto3.client('s3')     # Should return actual result     o = client.get_object(Bucket='my-bucket', Key='my-key')     # Should return mocked exception     e = client.upload_part_copy() 

Jordan Philips also posted a great solution using the the botocore.stub.Stubber class. Whilst a cleaner solution I was un-able to mock specific operations.

like image 23
ptimson Avatar answered Sep 20 '22 22:09

ptimson