Files
ds_task_ai_news_bolade/backend/main.py
T
boladeE 82fe3608d2 Refactor backend configuration and enhance news fetching functionality
- Introduced a Config dataclass in config.py to manage API keys, RSS feeds, and directory paths more effectively.
- Updated the NewsFetcher class to include retry logic for fetching articles from RSS feeds.
- Modified the EmbeddingGenerator and NewsRecommender classes to utilize the new configuration structure.
- Enhanced main.py to implement API token verification for secure access to news fetching and recommendations.
2025-04-16 17:55:36 +01:00

168 lines
5.9 KiB
Python

from fastapi import FastAPI, HTTPException, Request, Depends
from fastapi.middleware.cors import CORSMiddleware
from fastapi.templating import Jinja2Templates
from fastapi.responses import HTMLResponse
from typing import List, Dict, Any
import json
import os
from news_fetcher import NewsFetcher
from embeddings import EmbeddingGenerator
from vector_store import VectorStore
from recommender import NewsRecommender
from config import config
from fastapi import HTTPException
app = FastAPI(title="DS Task AI News API")
# Configure templates
templates = Jinja2Templates(directory="backend/templates")
def verify_api_token(token: str):
if token == config.api_token:
print(f"API key verified: {token}")
return token
return None
# Add custom filters
def from_json(value):
"""Parse a JSON string into a Python object."""
try:
return json.loads(value)
except (json.JSONDecodeError, TypeError):
return None
templates.env.filters["from_json"] = from_json
# Add CORS middleware
app.add_middleware(
CORSMiddleware,
allow_origins=["*"],
allow_credentials=True,
allow_methods=["*"],
allow_headers=["*"],
)
# Initialize components
news_fetcher = NewsFetcher()
embedding_generator = EmbeddingGenerator()
vector_store = VectorStore()
recommender = NewsRecommender()
@app.get("/", response_class=HTMLResponse)
async def root(request: Request):
"""Root endpoint returning the home page with links to other routes."""
return templates.TemplateResponse(
"home.html",
{"request": request}
)
@app.get("/fetch-news", response_class=HTMLResponse)
def fetch_news(request: Request, token: str = Depends(verify_api_token)):
# print(f"Fetching news with token: {token}")
"""Fetch news from RSS feeds and store in vector database."""
try:
result = news_fetcher.process()
if result["status"] == "error":
raise HTTPException(status_code=404, detail=result["message"])
# Get the latest processed articles
processed_files = sorted(os.listdir(config.processed_news_dir), reverse=True)
if not processed_files:
raise HTTPException(status_code=404, detail="No processed articles found")
latest_file = os.path.join(config.processed_news_dir, processed_files[0])
with open(latest_file, 'r', encoding='utf-8') as f:
articles = json.load(f)
# Ensure each article has a link
for article in articles:
if 'link' not in article or not article['link']:
# If no link is available, use the article ID as a fallback
article['link'] = f"/article/{article.get('id', '')}"
return templates.TemplateResponse(
"news.html",
{"request": request, "articles": articles}
)
except Exception as e:
raise HTTPException(status_code=500, detail=str(e))
@app.get("/recommend-news", response_class=HTMLResponse)
async def recommend_news(request: Request, article_id: str = None, query: str = None, token: str = Depends(verify_api_token)):
"""Get news recommendations based on article ID or search query."""
try:
if article_id:
# Get article from vector store
article = vector_store.search_similar([0] * 1024, top_k=1) # Placeholder vector with correct dimension
if not article:
raise HTTPException(status_code=404, detail="Article not found")
# Generate query embedding from article content
query_embedding = embedding_generator.get_query_embedding(
f"{article[0]['title']} {article[0]['content']}"
)
elif query:
# Generate query embedding from search query
query_embedding = embedding_generator.get_query_embedding(query)
else:
raise HTTPException(
status_code=400,
detail="Either article_id or query parameter is required"
)
# Search for similar articles
similar_articles = vector_store.search_similar(query_embedding)
if not similar_articles:
raise HTTPException(status_code=404, detail="No similar articles found")
# Ensure each article has a link
for article in similar_articles:
if 'link' not in article or not article['link']:
# If no link is available, use the article ID as a fallback
article['link'] = f"/article/{article.get('id', '')}"
# Generate insights for the articles
insights = recommender.analyze_articles(similar_articles)
return templates.TemplateResponse(
"recommendations.html",
{
"request": request,
"articles": similar_articles,
"insights": insights
}
)
except Exception as e:
raise HTTPException(status_code=500, detail=str(e))
@app.get("/article/{article_id}")
async def get_article(article_id: str, token: str = Depends(verify_api_token)):
"""Get a specific article and its summary."""
try:
# Search for the article
articles = vector_store.search_similar([0] * 1024, top_k=1) # Placeholder vector with correct dimension
if not articles:
raise HTTPException(status_code=404, detail="Article not found")
article = articles[0]
# Ensure the article has a link
if 'link' not in article or not article['link']:
# If no link is available, use the article ID as a fallback
article['link'] = f"/article/{article.get('id', '')}"
# Generate summary
summary = recommender.generate_summary(article)
return {
"article": article,
"summary": summary
}
except Exception as e:
raise HTTPException(status_code=500, detail=str(e))
if __name__ == "__main__":
import uvicorn
uvicorn.run(app, host="localhost", port=8000)