199

In boto 2, you can write to an S3 object using these methods:

Is there a boto 3 equivalent? What is the boto3 method for saving data to an object stored on S3?

jkdev
  • 11,360
  • 15
  • 54
  • 77

8 Answers8

371

In boto 3, the 'Key.set_contents_from_' methods were replaced by

For example:

import boto3

some_binary_data = b'Here we have some data'
more_binary_data = b'Here we have some more data'

# Method 1: Object.put()
s3 = boto3.resource('s3')
object = s3.Object('my_bucket_name', 'my/key/including/filename.txt')
object.put(Body=some_binary_data)

# Method 2: Client.put_object()
client = boto3.client('s3')
client.put_object(Body=more_binary_data, Bucket='my_bucket_name', Key='my/key/including/anotherfilename.txt')

Alternatively, the binary data can come from reading a file, as described in the official docs comparing boto 2 and boto 3:

Storing Data

Storing data from a file, stream, or string is easy:

# Boto 2.x
from boto.s3.key import Key
key = Key('hello.txt')
key.set_contents_from_file('/tmp/hello.txt')

# Boto 3
s3.Object('mybucket', 'hello.txt').put(Body=open('/tmp/hello.txt', 'rb'))
Community
  • 1
  • 1
jkdev
  • 11,360
  • 15
  • 54
  • 77
  • botocore.exceptions.NoCredentialsError: Unable to locate credentials how to fix this ? – deepak murthy Oct 16 '17 at 07:10
  • 2
    @deepakmurthy I'm not sure why you're getting that error... You'd need to [ask a new Stack Overflow question](https://stackoverflow.com/questions/ask) and provide more details about the issue. – jkdev Oct 16 '17 at 16:48
  • 2
    When I try `s3.Object().put()` I end up with an object with zero `content-length`. For me `put()` only accepts string data, but `put(str(binarydata))` seems to have some sort of encoding issues. I end up with an object roughly 3times the size of the original data, which makes it useless for me. – user1129682 Feb 28 '18 at 16:05
  • @user1129682 I'm not sure why that is. Could you please [ask a new question](https://stackoverflow.com/questions/ask) and provide more details? – jkdev Feb 28 '18 at 17:28
  • @jkdev It'd be great if you could [take a look](https://stackoverflow.com/q/49038039/1129682). – user1129682 Feb 28 '18 at 20:33
  • I always wonder which is the most optimal i.e. boto3.resource v/s boto3.client – SanjoS30 Mar 14 '22 at 19:03
101

boto3 also has a method for uploading a file directly:

s3 = boto3.resource('s3')    
s3.Bucket('bucketname').upload_file('/local/file/here.txt','folder/sub/path/to/s3key')

http://boto3.readthedocs.io/en/latest/reference/services/s3.html#S3.Bucket.upload_file

mathetes
  • 11,766
  • 7
  • 25
  • 32
EM Bee
  • 1,229
  • 1
  • 8
  • 11
75

You no longer have to convert the contents to binary before writing to the file in S3. The following example creates a new text file (called newfile.txt) in an S3 bucket with string contents:

import boto3

s3 = boto3.resource(
    's3',
    region_name='us-east-1',
    aws_access_key_id=KEY_ID,
    aws_secret_access_key=ACCESS_KEY
)
content="String content to write to a new S3 file"
s3.Object('my-bucket-name', 'newfile.txt').put(Body=content)
Franke
  • 1,234
  • 12
  • 14
  • Have no idea my 'put' action has no access. I created this bucket and put my canonical id under the access list. – Drake .C Mar 05 '19 at 00:49
  • How do you give a `prefix` in this case? Meaning, what if you want to store the file in `my-bucket-name/subfolder/` ? – kev Apr 09 '19 at 22:47
  • 4
    @kev you can specify that along with the filename 'subfolder/newfile.txt' instead of 'newfile.txt' – Madhava Carrillo Apr 10 '19 at 10:08
  • Re "You no longer have to convert the contents to binary before writing to the file in S3.", is this documented somewhere? I was looking at https://boto3.amazonaws.com/v1/documentation/api/latest/reference/services/s3.html#S3.Client.put_object, and thought it only accepted bytes. I'm not sure what exactly constitutes a "seekable file-like object", but didn't think that included strings. – Emma Jun 11 '20 at 20:49
  • 1
    I may have comparing this with download_fileobj() which is for large multipart file uploads. The upload methods require [seekable file objects](https://stackoverflow.com/questions/11696472/seek-function#:~:text=The%20seek%20position%20is%20a,cannot%20seek%20to%20file's%20beginning.%20%E2%80%93), but put() lets you write strings directly to a file in the bucket, which is handy for lambda functions to dynamically create and write files to an S3 bucket. – Franke Jun 13 '20 at 13:11
  • How can I read this data later? Does it directly convert it to string? Can you provide code samples? – Bat Dec 04 '20 at 07:37
47

Here's a nice trick to read JSON from s3:

import json, boto3
s3 = boto3.resource("s3").Bucket("bucket")
json.load_s3 = lambda f: json.load(s3.Object(key=f).get()["Body"])
json.dump_s3 = lambda obj, f: s3.Object(key=f).put(Body=json.dumps(obj))

Now you can use json.load_s3 and json.dump_s3 with the same API as load and dump

data = {"test":0}
json.dump_s3(data, "key") # saves json to s3://bucket/key
data = json.load_s3("key") # read json from s3://bucket/key
Uri Goren
  • 13,386
  • 6
  • 58
  • 110
  • 4
    Excellent. To get it to work, I added this extra bit: `...["Body"].read().decode('utf-8')`. – sedeh Nov 19 '18 at 15:05
  • Great idea. Anyway, it provides some space for naming improvements. – Jan Vlcinsky Jun 14 '20 at 09:46
  • Proposed rewrite of this nice idea: https://gist.github.com/vlcinsky/bbeda4321208aa98745afc29b58e90ac – Jan Vlcinsky Jun 14 '20 at 10:47
  • Hmm. It doesn't seem like a good idea to monkeypatch core Python library modules. Too surprising. Better to use plain functions or your own module, then call `json` without monkeypatching, e.g. `def load_s3(key): json.load(s3.Object(key=key).get()["Body"])` and so forth. – ggorlen Mar 28 '23 at 04:56
31

A cleaner and concise version which I use to upload files on the fly to a given S3 bucket and sub-folder-

import boto3

BUCKET_NAME = 'sample_bucket_name'
PREFIX = 'sub-folder/'

s3 = boto3.resource('s3')

# Creating an empty file called "_DONE" and putting it in the S3 bucket
s3.Object(BUCKET_NAME, PREFIX + '_DONE').put(Body="")

Note: You should ALWAYS put your AWS credentials (aws_access_key_id and aws_secret_access_key) in a separate file, for example- ~/.aws/credentials

kev
  • 2,741
  • 5
  • 22
  • 48
11

After some research, I found this. It can be achieved using a simple csv writer. It is to write a dictionary to CSV directly to S3 bucket.

eg: data_dict = [{"Key1": "value1", "Key2": "value2"}, {"Key1": "value4", "Key2": "value3"}] assuming that the keys in all the dictionary are uniform.

import csv
import boto3

# Sample input dictionary
data_dict = [{"Key1": "value1", "Key2": "value2"}, {"Key1": "value4", "Key2": "value3"}]
data_dict_keys = data_dict[0].keys()

# creating a file buffer
file_buff = StringIO()
# writing csv data to file buffer
writer = csv.DictWriter(file_buff, fieldnames=data_dict_keys)
writer.writeheader()
for data in data_dict:
    writer.writerow(data)
# creating s3 client connection
client = boto3.client('s3')
# placing file to S3, file_buff.getvalue() is the CSV body for the file
client.put_object(Body=file_buff.getvalue(), Bucket='my_bucket_name', Key='my/key/including/anotherfilename.txt')
ouflak
  • 2,458
  • 10
  • 44
  • 49
Oves Khan
  • 111
  • 1
  • 2
5

it is worth mentioning smart-open that uses boto3 as a back-end.

smart-open is a drop-in replacement for python's open that can open files from s3, as well as ftp, http and many other protocols.

for example

from smart_open import open
import json
with open("s3://your_bucket/your_key.json", 'r') as f:
    data = json.load(f)

The aws credentials are loaded via boto3 credentials, usually a file in the ~/.aws/ dir or an environment variable.

Uri Goren
  • 13,386
  • 6
  • 58
  • 110
-1

You may use the below code to write, for example an image to S3 in 2019. To be able to connect to S3 you will have to install AWS CLI using command pip install awscli, then enter few credentials using command aws configure:

import urllib3
import uuid
from pathlib import Path
from io import BytesIO
from errors import custom_exceptions as cex

BUCKET_NAME = "xxx.yyy.zzz"
POSTERS_BASE_PATH = "assets/wallcontent"
CLOUDFRONT_BASE_URL = "https://xxx.cloudfront.net/"


class S3(object):
    def __init__(self):
        self.client = boto3.client('s3')
        self.bucket_name = BUCKET_NAME
        self.posters_base_path = POSTERS_BASE_PATH

    def __download_image(self, url):
        manager = urllib3.PoolManager()
        try:
            res = manager.request('GET', url)
        except Exception:
            print("Could not download the image from URL: ", url)
            raise cex.ImageDownloadFailed
        return BytesIO(res.data)  # any file-like object that implements read()

    def upload_image(self, url):
        try:
            image_file = self.__download_image(url)
        except cex.ImageDownloadFailed:
            raise cex.ImageUploadFailed

        extension = Path(url).suffix
        id = uuid.uuid1().hex + extension
        final_path = self.posters_base_path + "/" + id
        try:
            self.client.upload_fileobj(image_file,
                                       self.bucket_name,
                                       final_path
                                       )
        except Exception:
            print("Image Upload Error for URL: ", url)
            raise cex.ImageUploadFailed

        return CLOUDFRONT_BASE_URL + id
Prateek Bhuwania
  • 755
  • 1
  • 8
  • 19