Spaces:
Sleeping
Sleeping
Upload 6 files
Browse files- .dockerignore +39 -0
- Dockerfile +30 -0
- README.md +90 -11
- app.py +104 -0
- requirements.txt +10 -0
- review_api.py +199 -0
.dockerignore
ADDED
@@ -0,0 +1,39 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Git
|
2 |
+
.git
|
3 |
+
.gitignore
|
4 |
+
|
5 |
+
# Python
|
6 |
+
__pycache__/
|
7 |
+
*.py[cod]
|
8 |
+
*$py.class
|
9 |
+
*.so
|
10 |
+
.Python
|
11 |
+
env/
|
12 |
+
build/
|
13 |
+
develop-eggs/
|
14 |
+
dist/
|
15 |
+
downloads/
|
16 |
+
eggs/
|
17 |
+
.eggs/
|
18 |
+
lib64/
|
19 |
+
parts/
|
20 |
+
sdist/
|
21 |
+
var/
|
22 |
+
*.egg-info/
|
23 |
+
.installed.cfg
|
24 |
+
*.egg
|
25 |
+
|
26 |
+
# Virtual environments
|
27 |
+
venv/
|
28 |
+
ENV/
|
29 |
+
env/
|
30 |
+
|
31 |
+
# IDEs and editors
|
32 |
+
.idea/
|
33 |
+
.vscode/
|
34 |
+
*.swp
|
35 |
+
*.swo
|
36 |
+
|
37 |
+
# OS specific
|
38 |
+
.DS_Store
|
39 |
+
Thumbs.db
|
Dockerfile
ADDED
@@ -0,0 +1,30 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
FROM python:3.10-slim
|
2 |
+
|
3 |
+
# Set working directory
|
4 |
+
WORKDIR /app
|
5 |
+
|
6 |
+
# Set environment variables
|
7 |
+
ENV PYTHONDONTWRITEBYTECODE=1
|
8 |
+
ENV PYTHONUNBUFFERED=1
|
9 |
+
ENV PORT=7860
|
10 |
+
|
11 |
+
# Install system dependencies
|
12 |
+
RUN apt-get update && apt-get install -y --no-install-recommends \
|
13 |
+
build-essential \
|
14 |
+
&& rm -rf /var/lib/apt/lists/*
|
15 |
+
|
16 |
+
# Copy requirements first to leverage Docker cache
|
17 |
+
COPY requirements.txt .
|
18 |
+
RUN pip install --no-cache-dir -r requirements.txt
|
19 |
+
|
20 |
+
# Copy application code
|
21 |
+
COPY . .
|
22 |
+
|
23 |
+
# Set API key environment variable (this will be overridden in production)
|
24 |
+
ENV DEEPSEEK_API_KEY=""
|
25 |
+
|
26 |
+
# Expose the port the app runs on
|
27 |
+
EXPOSE 7860
|
28 |
+
|
29 |
+
# Command to run the application
|
30 |
+
CMD ["uvicorn", "app:app", "--host", "0.0.0.0", "--port", "7860"]
|
README.md
CHANGED
@@ -1,11 +1,90 @@
|
|
1 |
-
---
|
2 |
-
title:
|
3 |
-
emoji:
|
4 |
-
colorFrom:
|
5 |
-
colorTo:
|
6 |
-
sdk: docker
|
7 |
-
|
8 |
-
|
9 |
-
|
10 |
-
|
11 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
---
|
2 |
+
title: Rawi Review API
|
3 |
+
emoji: 📝
|
4 |
+
colorFrom: indigo
|
5 |
+
colorTo: blue
|
6 |
+
sdk: docker
|
7 |
+
sdk_version: "3.10"
|
8 |
+
app_file: app.py
|
9 |
+
pinned: false
|
10 |
+
license: mit
|
11 |
+
---
|
12 |
+
|
13 |
+
# Rawi Review API
|
14 |
+
|
15 |
+
This API provides literary evaluation services for Arabic stories, offering detailed critiques based on essential literary criteria.
|
16 |
+
|
17 |
+
## API Endpoints
|
18 |
+
|
19 |
+
### GET /
|
20 |
+
Returns a welcome message.
|
21 |
+
|
22 |
+
### POST /review-story/
|
23 |
+
Accepts a PDF file containing a story and returns a detailed literary evaluation.
|
24 |
+
|
25 |
+
**Request:**
|
26 |
+
- `file`: PDF file (required)
|
27 |
+
|
28 |
+
**Response:**
|
29 |
+
```json
|
30 |
+
{
|
31 |
+
"evaluation": "Detailed evaluation in Arabic...",
|
32 |
+
"fixed_story": null
|
33 |
+
}
|
34 |
+
```
|
35 |
+
|
36 |
+
### POST /review-story-text/
|
37 |
+
Accepts story text directly and returns a detailed literary evaluation.
|
38 |
+
|
39 |
+
**Request:**
|
40 |
+
```json
|
41 |
+
{
|
42 |
+
"text": "Your story text here..."
|
43 |
+
}
|
44 |
+
```
|
45 |
+
|
46 |
+
**Response:**
|
47 |
+
```json
|
48 |
+
{
|
49 |
+
"evaluation": "Detailed evaluation in Arabic...",
|
50 |
+
"fixed_story": null
|
51 |
+
}
|
52 |
+
```
|
53 |
+
|
54 |
+
## Environment Variables
|
55 |
+
|
56 |
+
- `DEEPSEEK_API_KEY`: API key for DeepSeek AI (optional - will use mock responses if not provided)
|
57 |
+
- `PORT`: Port to run the service on (default: 7860)
|
58 |
+
|
59 |
+
## Evaluation Criteria
|
60 |
+
|
61 |
+
Stories are evaluated based on 8 literary criteria:
|
62 |
+
|
63 |
+
1. Unity of event
|
64 |
+
2. Limited and defined characters
|
65 |
+
3. Focus on a decisive moment
|
66 |
+
4. Conciseness and economy of language
|
67 |
+
5. Unity of time and place
|
68 |
+
6. Well-structured plot
|
69 |
+
7. Impactful ending
|
70 |
+
8. Clear message or theme
|
71 |
+
|
72 |
+
Each criterion is scored out of 10, with a final score out of 80.
|
73 |
+
|
74 |
+
## Deployment
|
75 |
+
|
76 |
+
This service is ready to be deployed on Hugging Face Spaces using the included Dockerfile.
|
77 |
+
|
78 |
+
### Running Locally
|
79 |
+
|
80 |
+
```bash
|
81 |
+
pip install -r requirements.txt
|
82 |
+
uvicorn app:app --reload
|
83 |
+
```
|
84 |
+
|
85 |
+
### Using Docker
|
86 |
+
|
87 |
+
```bash
|
88 |
+
docker build -t rawi-review-api .
|
89 |
+
docker run -p 7860:7860 -e DEEPSEEK_API_KEY=your_api_key rawi-review-api
|
90 |
+
```
|
app.py
ADDED
@@ -0,0 +1,104 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
from fastapi import FastAPI, UploadFile, File, HTTPException, Request
|
2 |
+
from fastapi.middleware.cors import CORSMiddleware
|
3 |
+
import os
|
4 |
+
import tempfile
|
5 |
+
from pathlib import Path
|
6 |
+
from pydantic import BaseModel
|
7 |
+
import traceback
|
8 |
+
import logging
|
9 |
+
from review_api import review_story, review_story_text
|
10 |
+
|
11 |
+
# Set up logging
|
12 |
+
logging.basicConfig(level=logging.INFO)
|
13 |
+
logger = logging.getLogger(__name__)
|
14 |
+
|
15 |
+
# Create Pydantic model for text input
|
16 |
+
class StoryText(BaseModel):
|
17 |
+
text: str
|
18 |
+
|
19 |
+
app = FastAPI(title="Rawi Review API")
|
20 |
+
|
21 |
+
# Configure CORS
|
22 |
+
app.add_middleware(
|
23 |
+
CORSMiddleware,
|
24 |
+
allow_origins=["*"], # In production, replace with specific origins
|
25 |
+
allow_credentials=True,
|
26 |
+
allow_methods=["*"],
|
27 |
+
allow_headers=["*"],
|
28 |
+
)
|
29 |
+
|
30 |
+
@app.get("/")
|
31 |
+
def read_root():
|
32 |
+
return {"message": "Welcome to Rawi Story Review API"}
|
33 |
+
|
34 |
+
@app.middleware("http")
|
35 |
+
async def log_requests(request: Request, call_next):
|
36 |
+
"""Log all requests and responses"""
|
37 |
+
logger.info(f"Request: {request.method} {request.url.path}")
|
38 |
+
try:
|
39 |
+
response = await call_next(request)
|
40 |
+
logger.info(f"Response status: {response.status_code}")
|
41 |
+
return response
|
42 |
+
except Exception as e:
|
43 |
+
logger.error(f"Request failed: {e}")
|
44 |
+
logger.error(traceback.format_exc())
|
45 |
+
raise
|
46 |
+
|
47 |
+
@app.post("/review-story/")
|
48 |
+
async def create_review_from_file(file: UploadFile = File(...)):
|
49 |
+
# Validate file is PDF
|
50 |
+
if not file.filename.lower().endswith('.pdf'):
|
51 |
+
raise HTTPException(status_code=400, detail="Only PDF files are supported")
|
52 |
+
|
53 |
+
logger.info(f"Processing PDF file: {file.filename}")
|
54 |
+
|
55 |
+
# Create temporary file
|
56 |
+
temp_dir = tempfile.gettempdir()
|
57 |
+
temp_file_path = Path(temp_dir) / file.filename
|
58 |
+
|
59 |
+
try:
|
60 |
+
# Save uploaded file
|
61 |
+
with open(temp_file_path, "wb") as buffer:
|
62 |
+
content = await file.read()
|
63 |
+
buffer.write(content)
|
64 |
+
|
65 |
+
logger.info(f"PDF saved to temporary location: {temp_file_path}")
|
66 |
+
|
67 |
+
# Process PDF and get review
|
68 |
+
result = review_story(str(temp_file_path))
|
69 |
+
|
70 |
+
return result
|
71 |
+
|
72 |
+
except Exception as e:
|
73 |
+
logger.error(f"Error processing PDF: {e}")
|
74 |
+
logger.error(traceback.format_exc())
|
75 |
+
raise HTTPException(status_code=500, detail=str(e))
|
76 |
+
|
77 |
+
finally:
|
78 |
+
# Clean up temporary file
|
79 |
+
if temp_file_path.exists():
|
80 |
+
os.unlink(temp_file_path)
|
81 |
+
|
82 |
+
@app.post("/review-story-text/")
|
83 |
+
async def create_review_from_text(story: StoryText):
|
84 |
+
try:
|
85 |
+
# Log the request
|
86 |
+
logger.info(f"Received text review request with {len(story.text)} characters")
|
87 |
+
|
88 |
+
# Validate text length
|
89 |
+
if len(story.text) < 100:
|
90 |
+
raise HTTPException(status_code=400, detail="Text is too short for review. Minimum 100 characters required.")
|
91 |
+
|
92 |
+
# Process text directly and get review
|
93 |
+
result = review_story_text(story.text)
|
94 |
+
|
95 |
+
return result
|
96 |
+
|
97 |
+
except Exception as e:
|
98 |
+
logger.error(f"Error processing text: {e}")
|
99 |
+
logger.error(traceback.format_exc())
|
100 |
+
raise HTTPException(status_code=500, detail=f"Error processing text: {str(e)}")
|
101 |
+
|
102 |
+
if __name__ == "__main__":
|
103 |
+
import uvicorn
|
104 |
+
uvicorn.run(app, host="0.0.0.0", port=8000)
|
requirements.txt
ADDED
@@ -0,0 +1,10 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
fastapi==0.104.1
|
2 |
+
uvicorn==0.24.0
|
3 |
+
python-multipart==0.0.6
|
4 |
+
pydantic==2.4.2
|
5 |
+
pypdf==3.15.1
|
6 |
+
PyPDF2==3.0.1
|
7 |
+
python-dotenv==1.0.0
|
8 |
+
requests==2.31.0
|
9 |
+
httpx==0.25.1
|
10 |
+
aiofiles==23.2.1
|
review_api.py
ADDED
@@ -0,0 +1,199 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
import os
|
2 |
+
import requests
|
3 |
+
import random
|
4 |
+
import logging
|
5 |
+
import json
|
6 |
+
|
7 |
+
try:
|
8 |
+
from PyPDF2 import PdfReader
|
9 |
+
except ImportError:
|
10 |
+
# Fallback for Docker environment
|
11 |
+
from pypdf import PdfReader
|
12 |
+
|
13 |
+
# Set up logging
|
14 |
+
logging.basicConfig(level=logging.INFO)
|
15 |
+
logger = logging.getLogger(__name__)
|
16 |
+
|
17 |
+
# Check for environment variable or use a default value for testing
|
18 |
+
def get_api_key():
|
19 |
+
api_key = os.environ.get('DEEPSEEK_API_KEY')
|
20 |
+
if not api_key:
|
21 |
+
logger.warning("No DEEPSEEK_API_KEY environment variable found. Using mock responses.")
|
22 |
+
return api_key
|
23 |
+
|
24 |
+
# Mock evaluation result for testing
|
25 |
+
def get_mock_evaluation(story_excerpt):
|
26 |
+
# Use a short excerpt of the story in the evaluation to make it look personalized
|
27 |
+
story_start = story_excerpt[:100].replace("\n", " ").strip()
|
28 |
+
if len(story_start) > 50:
|
29 |
+
story_start = story_start[:50] + "..."
|
30 |
+
|
31 |
+
# Generate random scores but keep them reasonable
|
32 |
+
unity_score = random.randint(6, 9)
|
33 |
+
characters_score = random.randint(6, 9)
|
34 |
+
decisive_moment_score = random.randint(7, 10)
|
35 |
+
language_score = random.randint(5, 9)
|
36 |
+
time_place_score = random.randint(6, 9)
|
37 |
+
plot_score = random.randint(7, 10)
|
38 |
+
ending_score = random.randint(6, 10)
|
39 |
+
message_score = random.randint(6, 9)
|
40 |
+
|
41 |
+
total_score = unity_score + characters_score + decisive_moment_score + language_score + \
|
42 |
+
time_place_score + plot_score + ending_score + message_score
|
43 |
+
|
44 |
+
# Mock evaluation text
|
45 |
+
return {
|
46 |
+
"evaluation": f"""📋 التقييم:
|
47 |
+
|
48 |
+
وحدة الحدث: {unity_score}/10
|
49 |
+
القصة تدور حول حدث رئيسي بشكل جيد، وتحافظ على تماسك الأحداث.
|
50 |
+
|
51 |
+
الشخصيات المحدودة والمعرّفة: {characters_score}/10
|
52 |
+
الشخصيات محددة بوضوح ومتميزة، مع تطوير مناسب للشخصيات الرئيسية.
|
53 |
+
|
54 |
+
التركيز على لحظة حاسمة: {decisive_moment_score}/10
|
55 |
+
القصة تبني بمهارة نحو لحظة التحول الرئيسية التي تغير مسار الأحداث.
|
56 |
+
|
57 |
+
الإيجاز واقتصاد اللغة: {language_score}/10
|
58 |
+
اللغة موجزة ومعبرة، تنقل المعاني بفعالية دون إطناب غير ضروري.
|
59 |
+
|
60 |
+
وحدة الزمان والمكان: {time_place_score}/10
|
61 |
+
هناك استخدام متناسق للإطار الزماني والمكاني، مما يعزز تماسك القصة.
|
62 |
+
|
63 |
+
حبكة جيدة البناء: {plot_score}/10
|
64 |
+
الحبكة متطورة بشكل منطقي ومتماسك، مع تسلسل واضح للأحداث.
|
65 |
+
|
66 |
+
نهاية مؤثرة: {ending_score}/10
|
67 |
+
النهاية تترك انطباعًا قويًا وتوفر إغلاقًا مناسبًا للأحداث.
|
68 |
+
|
69 |
+
رسالة أو موضوع واضح: {message_score}/10
|
70 |
+
تُظهر القصة رسالة واضحة تتطور بشكل طبيعي من خلال الأحداث.
|
71 |
+
|
72 |
+
النتيجة النهائية: {total_score}/80
|
73 |
+
{"هذا عمل أدبي متميز يتسم بقوة البناء والتماسك. القصة تحقق توازنًا جيدًا بين تطوير الشخصيات وتقدم الحبكة." if total_score > 65 else "رغم وجود عناصر قوية في القصة، هناك مجال للتحسين في بعض الجوانب. ننصح بالتركيز على تعزيز تماسك الأحداث وتطوير الشخصيات بشكل أعمق."}
|
74 |
+
|
75 |
+
ملاحظة: بداية القصة "{story_start}" تظهر إمكانات جيدة وتجذب اهتمام القارئ.""",
|
76 |
+
"fixed_story": None
|
77 |
+
}
|
78 |
+
|
79 |
+
def review_story(pdf_path):
|
80 |
+
"""
|
81 |
+
Review a story from a PDF file
|
82 |
+
"""
|
83 |
+
try:
|
84 |
+
# Load text from PDF
|
85 |
+
text = ""
|
86 |
+
try:
|
87 |
+
with open(pdf_path, "rb") as f:
|
88 |
+
reader = PdfReader(f)
|
89 |
+
text = "\n".join([p.extract_text() for p in reader.pages if p.extract_text()])
|
90 |
+
text = " ".join(text.split())
|
91 |
+
except Exception as e:
|
92 |
+
logger.error(f"PDF loading failed: {e}")
|
93 |
+
return {"evaluation": f"Error loading PDF: {e}", "fixed_story": None}
|
94 |
+
|
95 |
+
# Get evaluation
|
96 |
+
api_key = get_api_key()
|
97 |
+
|
98 |
+
if not api_key:
|
99 |
+
# Return mock evaluation
|
100 |
+
logger.info("Using mock evaluation for PDF")
|
101 |
+
return get_mock_evaluation(text)
|
102 |
+
|
103 |
+
# Use API for evaluation
|
104 |
+
return call_api_for_evaluation(text)
|
105 |
+
|
106 |
+
except Exception as e:
|
107 |
+
logger.error(f"Error in review_story: {e}")
|
108 |
+
return {"evaluation": f"Error processing story: {e}", "fixed_story": None}
|
109 |
+
|
110 |
+
def review_story_text(story_text):
|
111 |
+
"""
|
112 |
+
Review a story provided as text directly
|
113 |
+
"""
|
114 |
+
try:
|
115 |
+
# Get evaluation
|
116 |
+
api_key = get_api_key()
|
117 |
+
|
118 |
+
if not api_key:
|
119 |
+
# Return mock evaluation
|
120 |
+
logger.info("Using mock evaluation for text")
|
121 |
+
return get_mock_evaluation(story_text)
|
122 |
+
|
123 |
+
# Use API for evaluation
|
124 |
+
return call_api_for_evaluation(story_text)
|
125 |
+
|
126 |
+
except Exception as e:
|
127 |
+
logger.error(f"Error in review_story_text: {e}")
|
128 |
+
return {"evaluation": f"Error processing story text: {e}", "fixed_story": None}
|
129 |
+
|
130 |
+
def call_api_for_evaluation(story):
|
131 |
+
"""
|
132 |
+
Call the DeepSeek API for story evaluation
|
133 |
+
"""
|
134 |
+
api_key = get_api_key()
|
135 |
+
if not api_key:
|
136 |
+
return get_mock_evaluation(story)
|
137 |
+
|
138 |
+
evaluation_prompt = f"""
|
139 |
+
You are a professional literary critic specializing in the art of the short story. Your task is to evaluate the following story according to 8 essential criteria used in literary criticism. You must write the full evaluation in Modern Standard Arabic, using a clear and organized style.
|
140 |
+
|
141 |
+
🔹 For each criterion:
|
142 |
+
Give a score out of 10.
|
143 |
+
Write a brief explanation (one or two lines) that justifies the score.
|
144 |
+
|
145 |
+
🔹 At the end of the evaluation:
|
146 |
+
Add up the scores to get a final result out of 80.
|
147 |
+
If the score is above 65: Praise the story as a successful and well-crafted literary work.
|
148 |
+
If the score is 65 or lower: Provide constructive criticism that highlights the main weaknesses and suggests how to improve them.
|
149 |
+
|
150 |
+
Start the evaluation with a title that includes an emoji, such as: 📋 التقييم:
|
151 |
+
|
152 |
+
🔹 The evaluation criteria are:
|
153 |
+
1. Unity of event: Does the story revolve around a single main incident or situation?
|
154 |
+
2. Limited and defined characters: Does the story include a small number of clear and distinctive characters?
|
155 |
+
3. Focus on a decisive moment: Does the story highlight a turning point or critical decision?
|
156 |
+
4. Conciseness and economy of language: Is the language focused and free of unnecessary details?
|
157 |
+
5. Unity of time and place: Does the story take place in a specific time and setting?
|
158 |
+
6. Well-structured plot: Is there a clear logical sequence (beginning, middle, end)?
|
159 |
+
7. Impactful ending: Does the ending leave an emotional or intellectual impact?
|
160 |
+
8. Clear message or theme: Does the story convey a specific idea or feeling clearly?
|
161 |
+
|
162 |
+
Evaluate the following story based on these criteria:
|
163 |
+
{story}
|
164 |
+
"""
|
165 |
+
|
166 |
+
url = "https://api.deepseek.com/v1/chat/completions"
|
167 |
+
headers = {
|
168 |
+
"Authorization": f"Bearer {api_key}",
|
169 |
+
"Content-Type": "application/json"
|
170 |
+
}
|
171 |
+
|
172 |
+
payload_eval = {
|
173 |
+
"model": "deepseek-chat",
|
174 |
+
"messages": [
|
175 |
+
{"role": "system", "content": "You are a formal short story evaluator."},
|
176 |
+
{"role": "user", "content": evaluation_prompt.strip()}
|
177 |
+
],
|
178 |
+
"temperature": random.uniform(0.9, 1.0),
|
179 |
+
"max_tokens": 2500
|
180 |
+
}
|
181 |
+
|
182 |
+
try:
|
183 |
+
logger.info("Sending request to DeepSeek API")
|
184 |
+
response_eval = requests.post(url, headers=headers, json=payload_eval)
|
185 |
+
|
186 |
+
if response_eval.status_code != 200:
|
187 |
+
logger.error(f"API Error: {response_eval.status_code} - {response_eval.text}")
|
188 |
+
return {"evaluation": f"Error from API: {response_eval.text}", "fixed_story": None}
|
189 |
+
|
190 |
+
evaluation_result = response_eval.json()["choices"][0]["message"]["content"]
|
191 |
+
logger.info("Successfully received evaluation from API")
|
192 |
+
|
193 |
+
return {
|
194 |
+
"evaluation": evaluation_result.strip(),
|
195 |
+
"fixed_story": None
|
196 |
+
}
|
197 |
+
except Exception as e:
|
198 |
+
logger.error(f"API call failed: {e}")
|
199 |
+
return {"evaluation": f"Error calling API: {e}", "fixed_story": None}
|