Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[HP-1539] Add AWS functionality to cirrus #96

Merged
merged 21 commits into from
Aug 26, 2024
Merged
Show file tree
Hide file tree
Changes from 9 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion .pre-commit-config.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,6 @@ repos:
- id: no-commit-to-branch
args: [--branch, develop, --branch, master, --branch, main, --pattern, release/.*]
- repo: https://github.com/psf/black
rev: 20.8b1
rev: 22.3.0
hooks:
- id: black
25 changes: 25 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,23 @@ using the library like above.

So... you should at least read how to set up your environment.

For AWS functionality you can use an example like

```
import boto3
from gen3cirrus import AwsService

client = boto3.client()

aws = AwsService(client)

object = "test.txt"
bucket = "testBucket"
expiration = 3600

url = aws.requestorPaysDownloadPresignedURL( bucket, object, expiration)
mfshao marked this conversation as resolved.
Show resolved Hide resolved
```

## Setting up Environment for `cirrus`
`cirrus`'s wispy clouds must dwell in the great blue expanse with other Clouds.
Thus, you'll need to configure `cirrus` with necessary information about those Clouds
Expand Down Expand Up @@ -185,6 +202,14 @@ cirrus_config.update(**settings)

*Still uses Google libraries for auth*

## AWS Specific Implentation Details

### Method for communication with AWS's API(s)

For AWS you must bring your own Boto3 client that you have configured.

You can then setup the AWS service and your client will be passed as an argument to the AWS API.

## Building the Documentation
- `pipenv install --dev`
- `pipenv run python docs/create_docs.py`
Expand Down
2 changes: 2 additions & 0 deletions gen3cirrus/__init__.py
Original file line number Diff line number Diff line change
@@ -1,2 +1,4 @@
# Expose public API from each cloud provider
from .google_cloud import GoogleCloudManager

from .aws import AwsCloudManager, AwsService
2 changes: 2 additions & 0 deletions gen3cirrus/aws/__init__.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,2 @@
from .manager import AwsCloudManager
from .services import AwsService
7 changes: 7 additions & 0 deletions gen3cirrus/aws/manager.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
class AwsCloudManager:
mfshao marked this conversation as resolved.
Show resolved Hide resolved
"""
Manager for AWS cloud clients. Currently this is not in use but has been added in case it is needed in the future
"""

def __init__(self):
pass
61 changes: 61 additions & 0 deletions gen3cirrus/aws/services.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,61 @@
"""
Amazon service for interacting with APIs
"""

import backoff
import boto3

from botocore.exceptions import ClientError
from gen3cirrus.config import config
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'd prefer if we break up the imports by newlines for 3 sections:

  • built in libs
  • external deps
  • internal imports

from gen3cirrus.aws.utils import (
generatePresignedURL,
generatePresignedURLRequesterPays,
generateMultipartUploadURL,
)

from cdislogging import get_logger

logger = get_logger(__name__, log_level="info")


class AwsService(object):
"""
Generic Amazon servicing using Boto3
"""

def __init__(self, client):
Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

maybe just so it's super clear we should call this boto3_client?

Copy link
Collaborator

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

bump

self.client = client

def download_presigned_url(self, bucket, key, expiration, additonal_info={}):
"""
Wrapper function for generating a presigned URL for downloading an object
"""
return generatePresignedURL(
mfshao marked this conversation as resolved.
Show resolved Hide resolved
self.client, "get", bucket, key, expiration, additonal_info
)

def upload_presigned_url(self, bucket, key, expiration, additonal_info={}):
"""
Wrapper function for generating a presigned URL for uploading an object
"""
return generatePresignedURL(
self.client, "put", bucket, key, expiration, additonal_info
)

def multipart_upload_presigned_url(self, bucket, key, expiration, upload_id, part):
"""
Wrapper function for generating a presigned URL for uploading an object
"""
return generateMultipartUploadURL(
self.client, bucket, key, expiration, upload_id, part
)

def requester_pays_download_presigned_url(
self, bucket, key, expiration, additonal_info={}
):
"""
Wrapper function for generating a presigned URL for downloading an object from a requester pays bucket
"""
return generatePresignedURLRequesterPays(
self.client, bucket, key, expiration, additonal_info
)
130 changes: 130 additions & 0 deletions gen3cirrus/aws/utils.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,130 @@
import datetime
import json

import boto3
from botocore.exceptions import ClientError

from cdislogging import get_logger

logger = get_logger(__name__, log_level="info")


def generatePresignedURL(
mfshao marked this conversation as resolved.
Show resolved Hide resolved
client, method, bucket_name, object_name, expires, additional_info={}
mfshao marked this conversation as resolved.
Show resolved Hide resolved
):
"""
Function for generating a presigned URL for upload or download

Args:
client: s3 boto client
method: ["get", "put"] "get" for download and "put" for upload
bucket_name: s3 bucket name
object_name: s3 bucket object key
expires: time for presigned URL to exist (in seconds)
additional_info: dict of additional parameters to pass to s3 for signing
"""

params = {}
params["Bucket"] = bucket_name
params["Key"] = object_name

for key in additional_info:
params[key] = additional_info[key]

s3_client = client

if method == "get":
m = "get_object"
mfshao marked this conversation as resolved.
Show resolved Hide resolved
elif method == "put":
m = "put_object"
else:
logger.error(
"method for generating presigned URL must be 'get' for download or 'put' for upload"
)
return None

try:
response = s3_client.generate_presigned_url(
m,
Params=params,
ExpiresIn=expires,
)

except ClientError as e:
logger.error(e)
return None

return response


def generateMultipartUploadURL(
client, bucket_name, object_name, expires, upload_id, part_no
):
"""
Function for generating a presigned URL only for one part of multipart upload

Args:
client: s3 boto client
method: ["get", "put"] "get" for download and "put" for upload
bucket_name: s3 bucket name
object_name: s3 bucket object key
expires: time for presigned URL to exist (in seconds)
upload_id: ID for upload to s3
part_no: part number of multipart upload
"""
s3_client = client
try:
response = s3_client.generate_presigned_url(
ClientMethod="upload_part",
Params={
"Bucket": bucket_name,
"Key": object_name,
"UploadId": upload_id,
"PartNumber": part_no,
},
ExpiresIn=expires,
)

except ClientError as e:
logger.error(e)
return None

return response


def generatePresignedURLRequesterPays(
client, bucket_name, object_name, expires, additional_info={}
):
"""
Function for generating a presigned URL only for requester pays buckets

Args:
client: s3 boto client
mfshao marked this conversation as resolved.
Show resolved Hide resolved
method: ["get", "put"] "get" for download and "put" for upload
bucket_name: s3 bucket name
object_name: s3 bucket object key
expires: time for presigned URL to exist (in seconds)
additional_info: dict of additional parameters to pass to s3 for signing
"""
params = {}
params["Bucket"] = bucket_name
params["Key"] = object_name
params["RequestPayer"] = "requester"

for key in additional_info:
params[key] = additional_info[key]

s3_client = client

try:
response = s3_client.generate_presigned_url(
"get_object",
Params=params,
ExpiresIn=expires,
)

except ClientError as e:
logger.error(e)
return None

return response
1 change: 1 addition & 0 deletions gen3cirrus/core.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
Current Capabilities:
- Manage Google resources, policies, and access (specific Google APIs
are abstracted through a Management class that exposes needed behavior)
- Manage AWS resources amd access S3 APIs
"""


Expand Down
4 changes: 3 additions & 1 deletion gen3cirrus/errors.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,7 @@
class CirrusError(Exception):
def __init__(self, message="There was an error within the gen3cirrus library.", *args):
def __init__(
self, message="There was an error within the gen3cirrus library.", *args
):
super(CirrusError, self).__init__(message)


Expand Down
Loading
Loading