from collections.abc import Generator

from flask import Flask
from qcloud_cos import CosConfig, CosS3Client

from extensions.storage.base_storage import BaseStorage


class TencentStorage(BaseStorage):
    """Implementation for tencent cos storage."""

    def __init__(self, app: Flask):
        super().__init__(app)
        app_config = self.app.config
        self.bucket_name = app_config.get("TENCENT_COS_BUCKET_NAME")
        config = CosConfig(
            Region=app_config.get("TENCENT_COS_REGION"),
            SecretId=app_config.get("TENCENT_COS_SECRET_ID"),
            SecretKey=app_config.get("TENCENT_COS_SECRET_KEY"),
            Scheme=app_config.get("TENCENT_COS_SCHEME"),
        )
        self.client = CosS3Client(config)

    def save(self, filename, data):
        self.client.put_object(Bucket=self.bucket_name, Body=data, Key=filename)

    def load_once(self, filename: str) -> bytes:
        data = self.client.get_object(Bucket=self.bucket_name, Key=filename)["Body"].get_raw_stream().read()
        return data

    def load_stream(self, filename: str) -> Generator:
        def generate(filename: str = filename) -> Generator:
            response = self.client.get_object(Bucket=self.bucket_name, Key=filename)
            yield from response["Body"].get_stream(chunk_size=4096)

        return generate()

    def download(self, filename, target_filepath):
        response = self.client.get_object(Bucket=self.bucket_name, Key=filename)
        response["Body"].get_stream_to_file(target_filepath)

    def exists(self, filename):
        return self.client.object_exists(Bucket=self.bucket_name, Key=filename)

    def delete(self, filename):
        self.client.delete_object(Bucket=self.bucket_name, Key=filename)