This repository has been archived by the owner on Sep 18, 2024. It is now read-only.
forked from HHS/simpler-grants-gov
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Browse files
Browse the repository at this point in the history
Fixes HHS#2046 Setup S3 localstack for having a local version of S3 to use (for future work) Script / utils for interacting with S3 Localstack is a tool that creates a mock version of AWS locally. While the ability to mock out certain features varies, S3 being just a file storage system is pretty simple and fully featured even when mocked. Note that localstack has a paid version as well that adds more features, but all of S3's features are [supported in the free community tier](https://docs.localstack.cloud/references/coverage/coverage_s3/). We've used localstack for s3 and a few other AWS services on other projects. The script creates the S3 bucket in localstack. You can actually interact with the localstack instance of s3 with the AWS cli like so: ```sh aws --endpoint-url http://localhost:4566 s3 ls > 2024-07-12 13:10:24 local-opportunities ``` I created a tmp file in it succesfully: ```sh aws --endpoint-url http://localhost:4566 s3 cp tmp.txt s3://local-opportunities/path/to/tmp.txt ``` I can see the tmp file: ```sh aws --endpoint-url http://localhost:4566 s3 ls s3://local-opportunities/path/to/ > 2024-07-12 13:23:22 15 tmp.txt ``` And I can download it: ```sh aws --endpoint-url http://localhost:4566 s3 cp s3://local-opportunities/path/to/tmp.txt local_tmp.txt ```
- Loading branch information
Showing
9 changed files
with
128 additions
and
17 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -29,3 +29,6 @@ coverage.* | |
|
||
#e2e | ||
/test-results/ | ||
|
||
# localstack | ||
/volume |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,49 @@ | ||
import logging | ||
|
||
import botocore.client | ||
import botocore.exceptions | ||
|
||
import src.logging | ||
from src.adapters.aws import S3Config, get_s3_client | ||
from src.util.local import error_if_not_local | ||
|
||
logger = logging.getLogger(__name__) | ||
|
||
|
||
def does_s3_bucket_exist(s3_client: botocore.client.BaseClient, bucket_name: str) -> bool: | ||
try: | ||
s3_client.head_bucket(Bucket=bucket_name) | ||
return True | ||
except botocore.exceptions.ClientError as e: | ||
# We'll assume that if the error code is a 404 that means | ||
# it could not find the bucket and thus it needs to be created | ||
# as there are not more specific errors than this available | ||
error_code = e.response.get("Error", {}).get("Code") | ||
if error_code != "404": | ||
raise e | ||
|
||
return False | ||
|
||
|
||
def setup_s3() -> None: | ||
s3_config = S3Config() | ||
s3_client = get_s3_client(s3_config) | ||
|
||
if s3_config.s3_opportunity_bucket is None: | ||
raise Exception("S3_OPPORTUNITY_BUCKET env var must be set") | ||
|
||
if not does_s3_bucket_exist(s3_client, s3_config.s3_opportunity_bucket): | ||
logger.info("Creating S3 bucket %s", s3_config.s3_opportunity_bucket) | ||
s3_client.create_bucket(Bucket=s3_config.s3_opportunity_bucket) | ||
else: | ||
logger.info("S3 bucket %s already exists - skipping", s3_config.s3_opportunity_bucket) | ||
|
||
|
||
def main() -> None: | ||
with src.logging.init("setup_localstack"): | ||
error_if_not_local() | ||
setup_s3() | ||
|
||
|
||
if __name__ == "__main__": | ||
main() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
from .s3_adapter import S3Config, get_s3_client | ||
|
||
__all__ = ["get_s3_client", "S3Config"] |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,30 @@ | ||
import boto3 | ||
import botocore.client | ||
|
||
from src.util.env_config import PydanticBaseEnvConfig | ||
|
||
|
||
class S3Config(PydanticBaseEnvConfig): | ||
# We should generally not need to set this except | ||
# locally to use localstack | ||
s3_endpoint_url: str | None = None | ||
|
||
### S3 Buckets | ||
# note that we default these to None | ||
# so that we don't need to set all of these for every | ||
# process that uses S3 | ||
|
||
# TODO - I'm not sure how we want to organize our | ||
# s3 buckets so this will likely change in the future | ||
s3_opportunity_bucket: str | None = None | ||
|
||
|
||
def get_s3_client(s3_config: S3Config | None = None) -> botocore.client.BaseClient: | ||
if s3_config is None: | ||
s3_config = S3Config() | ||
|
||
params = {} | ||
if s3_config.s3_endpoint_url is not None: | ||
params["endpoint_url"] = s3_config.s3_endpoint_url | ||
|
||
return boto3.client("s3", **params) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters