| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172 | from collections.abc import Generatorfrom contextlib import closingimport boto3from botocore.client import Configfrom botocore.exceptions import ClientErrorfrom flask import Flaskfrom extensions.storage.base_storage import BaseStorageclass S3Storage(BaseStorage):    """Implementation for s3 storage.    """    def __init__(self, app: Flask):        super().__init__(app)        app_config = self.app.config        self.bucket_name = app_config.get('S3_BUCKET_NAME')        if app_config.get('S3_USE_AWS_MANAGED_IAM'):            session = boto3.Session()            self.client = session.client('s3')        else:            self.client = boto3.client(                        's3',                        aws_secret_access_key=app_config.get('S3_SECRET_KEY'),                        aws_access_key_id=app_config.get('S3_ACCESS_KEY'),                        endpoint_url=app_config.get('S3_ENDPOINT'),                        region_name=app_config.get('S3_REGION'),                        config=Config(s3={'addressing_style': app_config.get('S3_ADDRESS_STYLE')})                    )    def save(self, filename, data):        self.client.put_object(Bucket=self.bucket_name, Key=filename, Body=data)    def load_once(self, filename: str) -> bytes:        try:            with closing(self.client) as client:                data = client.get_object(Bucket=self.bucket_name, Key=filename)['Body'].read()        except ClientError as ex:            if ex.response['Error']['Code'] == 'NoSuchKey':                raise FileNotFoundError("File not found")            else:                raise        return data    def load_stream(self, filename: str) -> Generator:        def generate(filename: str = filename) -> Generator:            try:                with closing(self.client) as client:                    response = client.get_object(Bucket=self.bucket_name, Key=filename)                    yield from response['Body'].iter_chunks()            except ClientError as ex:                if ex.response['Error']['Code'] == 'NoSuchKey':                    raise FileNotFoundError("File not found")                else:                    raise        return generate()    def download(self, filename, target_filepath):        with closing(self.client) as client:            client.download_file(self.bucket_name, filename, target_filepath)    def exists(self, filename):        with closing(self.client) as client:            try:                client.head_object(Bucket=self.bucket_name, Key=filename)                return True            except:                return False    def delete(self, filename):        self.client.delete_object(Bucket=self.bucket_name, Key=filename)
 |