feat: CONAI Phase 1 MVP 초기 구현
소형 건설업체(100억 미만)를 위한 AI 기반 토목공사 통합관리 플랫폼 Backend (FastAPI): - SQLAlchemy 모델 13개 (users, projects, wbs, tasks, daily_reports, reports, inspections, quality, weather, permits, rag, settings) - API 라우터 11개 (auth, projects, tasks, daily_reports, reports, inspections, weather, rag, kakao, permits, settings) - Services: Claude AI 래퍼, CPM Gantt 계산, 기상청 API, RAG(pgvector), 카카오 Skill API - Alembic 마이그레이션 (pgvector 포함) - pytest 테스트 (CPM, 날씨 경보) Frontend (Next.js 15): - 11개 페이지 (대시보드, 프로젝트, Gantt, 일보, 검측, 품질, 날씨, 인허가, RAG, 설정) - TanStack Query + Zustand + Tailwind CSS 인프라: - Docker Compose (PostgreSQL pgvector + backend + frontend) - 한국어 README 및 설치 가이드 Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
This commit is contained in:
@@ -0,0 +1,44 @@
|
||||
import uuid
|
||||
from pathlib import Path
|
||||
from typing import Optional
|
||||
from supabase import create_client, Client
|
||||
from app.config import settings
|
||||
|
||||
|
||||
def get_supabase() -> Client:
|
||||
return create_client(settings.SUPABASE_URL, settings.SUPABASE_SERVICE_KEY)
|
||||
|
||||
|
||||
def upload_file(
|
||||
file_bytes: bytes,
|
||||
project_id: str,
|
||||
file_type: str,
|
||||
filename: str,
|
||||
content_type: str = "application/octet-stream",
|
||||
) -> str:
|
||||
"""Upload file to Supabase Storage. Returns storage path (s3_key)."""
|
||||
client = get_supabase()
|
||||
ext = Path(filename).suffix
|
||||
unique_name = f"{uuid.uuid4()}{ext}"
|
||||
path = f"{project_id}/{file_type}/{unique_name}"
|
||||
|
||||
client.storage.from_(settings.SUPABASE_STORAGE_BUCKET).upload(
|
||||
path,
|
||||
file_bytes,
|
||||
file_options={"content-type": content_type},
|
||||
)
|
||||
return path
|
||||
|
||||
|
||||
def get_download_url(s3_key: str, expires_in: int = 3600) -> str:
|
||||
"""Get a presigned download URL."""
|
||||
client = get_supabase()
|
||||
response = client.storage.from_(settings.SUPABASE_STORAGE_BUCKET).create_signed_url(
|
||||
s3_key, expires_in
|
||||
)
|
||||
return response["signedURL"]
|
||||
|
||||
|
||||
def delete_file(s3_key: str) -> None:
|
||||
client = get_supabase()
|
||||
client.storage.from_(settings.SUPABASE_STORAGE_BUCKET).remove([s3_key])
|
||||
Reference in New Issue
Block a user