import logging
import os
from pathlib import Path
from typing import List

import boto3
from botocore.exceptions import ClientError, NoCredentialsError

# --- 配置日志 ---
logging.basicConfig(level=logging.INFO, format='%(asctime)s - %(levelname)s - %(message)s')

# ##############################################################################
# ### 配置区域: 请根据您的实际情况修改以下变量 ###
# ##############################################################################

SCRIPT_DIR = Path(__file__).resolve().parent

# 您的云存储访问凭证，默认使用环境变量。
AWS_ACCESS_KEY_ID = os.environ.get("AWS_ACCESS_KEY_ID")
AWS_SECRET_ACCESS_KEY = os.environ.get("AWS_SECRET_ACCESS_KEY")

# 您的存储桶名称
BUCKET_NAME = os.environ.get("UPLOAD_BUCKET_NAME", "error_cot")


def _default_local_directories() -> List[Path]:
    env_dirs = os.environ.get("LOCAL_DIRECTORIES_TO_UPLOAD")
    if env_dirs:
        return [Path(p).expanduser() for p in env_dirs.split(os.pathsep) if p.strip()]
    # 提供几个相对于当前脚本的示例目录，使用者可自行调整
    return [
        SCRIPT_DIR / "demos",
        SCRIPT_DIR / "demo1",
        SCRIPT_DIR / "new",
    ]


LOCAL_DIRECTORIES_TO_UPLOAD = _default_local_directories()

# AWS 区域 (例如 'us-east-1', 'ap-southeast-2')
AWS_REGION = os.environ.get("AWS_REGION")

# S3 兼容存储的 Endpoint URL
ENDPOINT_URL = os.environ.get("S3_ENDPOINT_URL", "https://aoss-internal.cn-sh-01b.sensecoreapi-oss.cn")

# ##############################################################################
# ### 脚本主逻辑: 一般无需修改以下代码 ###
# ##############################################################################

def create_s3_client():
    """根据配置创建并返回一个 S3 客户端"""
    try:
        s3_client = boto3.client(
            's3',
            aws_access_key_id=AWS_ACCESS_KEY_ID,
            aws_secret_access_key=AWS_SECRET_ACCESS_KEY,
            region_name=AWS_REGION,
            endpoint_url=ENDPOINT_URL
        )
        # 尝试验证凭证是否有效
        s3_client.list_buckets()
        logging.info("S3 客户端创建成功，凭证有效。")
        return s3_client
    except NoCredentialsError:
        logging.error("凭证未找到。请配置 AWS_ACCESS_KEY_ID 和 AWS_SECRET_ACCESS_KEY。")
        return None
    except ClientError as e:
        if e.response['Error']['Code'] == 'InvalidAccessKeyId':
            logging.error("无效的 Access Key ID。请检查您的凭证。")
        elif e.response['Error']['Code'] == 'SignatureDoesNotMatch':
            logging.error("Secret Access Key 错误。请检查您的凭证。")
        else:
            logging.error(f"连接或认证时发生客户端错误: {e}")
        return None
    except Exception as e:
        logging.error(f"创建 S3 客户端时发生未知错误: {e}")
        return None

def upload_directory(s3_client, local_directory, bucket_name):
    """
    将本地单个目录完整上传到 S3 存储桶

    :param s3_client: Boto3 S3 客户端
    :param local_directory: 要上传的本地目录路径
    :param bucket_name: 目标 S3 存储桶名称
    """
    local_directory = Path(local_directory).expanduser()

    if not local_directory.is_dir():
        logging.error(f"错误: 目录 '{local_directory}' 不存在。")
        return

    logging.info(f"--- 开始上传目录: {local_directory} ---")

    # 获取目录的基本名称，它将作为 S3 中的顶级“文件夹”
    # 例如，上传 '/path/to/project_folder'，S3中的对象会以 'project_folder/...' 开头
    base_dir_name = local_directory.name

    total_files = sum([len(files) for r, d, files in os.walk(local_directory)])
    uploaded_count = 0

    try:
        for root, dirs, files in os.walk(local_directory):
            for filename in files:
                # 构造本地文件的完整路径
                local_path = os.path.join(root, filename)

                # 构造在 S3 中的对象键 (路径)
                # 保持原始的目录结构
                relative_path = os.path.relpath(local_path, local_directory)
                s3_key = os.path.join(base_dir_name, relative_path)

                # 确保 S3 路径使用正斜杠 '/'
                s3_key = s3_key.replace(os.path.sep, '/')
                
                uploaded_count += 1
                progress = f"({uploaded_count}/{total_files})"
                logging.info(f"{progress} 正在上传 {local_path} 到 s3://{bucket_name}/{s3_key}")

                # 执行上传
                s3_client.upload_file(local_path, bucket_name, s3_key)

        logging.info(f"--- 目录 '{local_directory}' 上传完成 ---")

    except FileNotFoundError:
        logging.error(f"错误: 文件 '{local_path}' 未找到。")
    except ClientError as e:
        logging.error(f"上传文件到 S3 时发生客户端错误: {e}")
    except Exception as e:
        logging.error(f"上传过程中发生未知错误: {e}")


if __name__ == '__main__':
    logging.info("脚本开始执行。")
    
    s3_client = create_s3_client()
    
    if s3_client:
        if not LOCAL_DIRECTORIES_TO_UPLOAD:
            logging.warning("LOCAL_DIRECTORIES_TO_UPLOAD 为空，请在运行前配置需要上传的目录。")
        for directory in LOCAL_DIRECTORIES_TO_UPLOAD:
            upload_directory(s3_client, directory, BUCKET_NAME)
        logging.info("目录上传流程执行完毕。")
    else:
        logging.error("无法初始化 S3 客户端，脚本已终止。")