""" Text Asset Tracker Utility Helper utility for saving and tracking text content as files in the asset library. """ from typing import Dict, Any, Optional from pathlib import Path from sqlalchemy.orm import Session from utils.asset_tracker import save_asset_to_library from utils.file_storage import save_text_file_safely, generate_unique_filename, sanitize_filename import logging logger = logging.getLogger(__name__) def save_and_track_text_content( db: Session, user_id: str, content: str, source_module: str, title: str, description: Optional[str] = None, prompt: Optional[str] = None, tags: Optional[list] = None, asset_metadata: Optional[Dict[str, Any]] = None, base_dir: Optional[Path] = None, subdirectory: Optional[str] = None, file_extension: str = ".txt" ) -> Optional[int]: """ Save text content to disk and track it in the asset library. Args: db: Database session user_id: Clerk user ID content: Text content to save source_module: Source module name (e.g., "linkedin_writer", "facebook_writer") title: Title for the asset description: Description of the content prompt: Original prompt used for generation tags: List of tags for search/filtering asset_metadata: Additional metadata base_dir: Base directory for file storage (defaults to backend/{module}_text) subdirectory: Optional subdirectory (e.g., "posts", "articles") file_extension: File extension (.txt, .md, etc.) Returns: Asset ID if successful, None otherwise """ try: if not content or not isinstance(content, str) or len(content.strip()) == 0: logger.warning("Empty or invalid content provided") return None if not user_id or not isinstance(user_id, str): logger.error("Invalid user_id provided") return None # Determine output directory if base_dir is None: # Default to backend/{module}_text base_dir = Path(__file__).parent.parent module_name = source_module.replace('_', '') output_dir = base_dir / f"{module_name}_text" else: output_dir = base_dir # Add subdirectory if specified if subdirectory: output_dir = output_dir / subdirectory # Generate safe filename from title safe_title = sanitize_filename(title, max_length=80) filename = generate_unique_filename( prefix=safe_title, extension=file_extension, include_uuid=True ) # Save text file file_path, save_error = save_text_file_safely( content=content, directory=output_dir, filename=filename, encoding='utf-8', max_file_size=10 * 1024 * 1024 # 10MB for text ) if not file_path or save_error: logger.error(f"Failed to save text file: {save_error}") return None # Generate file URL relative_path = file_path.relative_to(base_dir) file_url = f"/api/text-assets/{relative_path.as_posix()}" # Prepare metadata final_metadata = asset_metadata or {} final_metadata.update({ "status": "completed", "character_count": len(content), "word_count": len(content.split()) }) # Save to asset library asset_id = save_asset_to_library( db=db, user_id=user_id, asset_type="text", source_module=source_module, filename=filename, file_url=file_url, file_path=str(file_path), file_size=len(content.encode('utf-8')), mime_type="text/plain" if file_extension == ".txt" else "text/markdown", title=title, description=description or f"Generated {source_module.replace('_', ' ')} content", prompt=prompt, tags=tags or [source_module, "text"], asset_metadata=final_metadata ) if asset_id: logger.info(f"✅ Text asset saved to library: ID={asset_id}, filename={filename}") else: logger.warning(f"Asset tracking returned None for {filename}") return asset_id except Exception as e: logger.error(f"❌ Error saving and tracking text content: {str(e)}", exc_info=True) return None