diff --git a/requirements.txt b/requirements.txt
new file mode 100644
index 0000000..c4ee82c
--- /dev/null
+++ b/requirements.txt
@@ -0,0 +1,74 @@
+amqp==5.2.0
+annotated-types==0.7.0
+anyio==4.4.0
+asgiref==3.8.1
+attrs==24.1.0
+beautifulsoup4==4.12.3
+billiard==4.2.0
+bs4==0.0.2
+celery==5.4.0
+certifi==2024.7.4
+cffi==1.17.0
+charset-normalizer==3.3.2
+click==8.1.7
+click-didyoumean==0.3.1
+click-plugins==1.1.1
+click-repl==0.3.0
+cron-descriptor==1.4.3
+cryptography==43.0.0
+distro==1.9.0
+Django==5.0.8
+django-admin-interface==0.28.8
+django-celery-beat==2.6.0
+django-ckeditor==6.7.1
+django-ckeditor-5==0.2.13
+django-colorfield==0.11.0
+django-cors-headers==4.4.0
+django-filter==24.2
+django-jazzmin==3.0.0
+django-js-asset==2.2.0
+django-shortcuts==1.6
+django-sortedm2m==4.0.0
+django-timezone-field==7.0
+django-tinymce==4.1.0
+djangorestframework==3.15.2
+drf-spectacular==0.27.2
+h11==0.14.0
+httpcore==1.0.5
+httpx==0.27.0
+idna==3.7
+inflection==0.5.1
+jiter==0.8.2
+jsonschema==4.23.0
+jsonschema-specifications==2023.12.1
+kombu==5.4.0
+openai==1.57.2
+pillow==10.4.0
+prompt_toolkit==3.0.47
+pycparser==2.22
+pydantic==2.10.3
+pydantic_core==2.27.1
+pyOpenSSL==24.2.1
+python-crontab==3.2.0
+python-dateutil==2.9.0.post0
+python-decouple==3.8
+python-dotenv==1.0.1
+python-slugify==8.0.4
+PyYAML==6.0.1
+redis==5.0.8
+referencing==0.35.1
+requests==2.32.3
+rpds-py==0.19.1
+six==1.16.0
+sniffio==1.3.1
+soupsieve==2.6
+sqlparse==0.5.1
+text-unidecode==1.3
+tqdm==4.67.1
+typing_extensions==4.12.2
+tzdata==2024.1
+uritemplate==4.1.1
+urllib3==2.2.2
+vine==5.1.0
+wcwidth==0.2.13
+whitenoise==6.7.0
diff --git a/server/apps/research/admin/article_admin.py b/server/apps/research/admin/article_admin.py
index 7dc624c..1a29c07 100644
--- a/server/apps/research/admin/article_admin.py
+++ b/server/apps/research/admin/article_admin.py
@@ -4,6 +4,11 @@
from apps.research.models import Article, ArticleSlugHistory
from tinymce.widgets import TinyMCE
from .slug_history import current_slug_history
+from django.conf import settings
+from django.http import JsonResponse
+from django.urls import path
+from ..services.gpt_service import GPTService
+import asyncio
class ArticleForm(forms.ModelForm):
class Meta:
@@ -21,13 +26,64 @@ def __init__(self, *args, **kwargs):
Q(pk=self.instance.pk) | Q(status='draft')
).order_by('-scheduled_publish_time')
- self.fields['acknowledgement'].widget = TinyMCE(attrs={'cols': 80, 'rows': 30, 'id': "acknowledgement_richtext_field", 'placeholder': "Enter Acknowledgement here"})
- self.fields['content'].widget = TinyMCE(attrs={'cols': 80, 'rows': 30, 'id': "content_richtext_field", 'placeholder': "Enter Article Content here"})
+ # Configure TinyMCE widgets
+ self.fields['acknowledgement'].widget = TinyMCE(attrs={
+ 'cols': 80,
+ 'rows': 30,
+ 'id': "acknowledgement_richtext_field",
+ 'placeholder': "Enter Acknowledgement here"
+ })
+ self.fields['content'].widget = TinyMCE(attrs={
+ 'cols': 80,
+ 'rows': 30,
+ 'id': "content_richtext_field",
+ 'placeholder': "Enter Article Content here"
+ })
+ self.fields['gpt_summary'].widget = TinyMCE(attrs={
+ 'cols': 80,
+ 'rows': 15,
+ 'id': "gpt_summary_richtext_field",
+ 'placeholder': "GPT-generated summary will appear here"
+ })
class ArticleAdmin(admin.ModelAdmin):
"""Admin interface for the Article model."""
form = ArticleForm
+ def __init__(self, model, admin_site):
+ super().__init__(model, admin_site)
+ self.gpt_service = GPTService()
+
+ def get_urls(self):
+ urls = super().get_urls()
+ custom_urls = [
+ path('generate-summary/', self.generate_summary_view, name='generate-summary'),
+ ]
+ return custom_urls + urls
+
+ async def _generate_summary(self, content: str) -> str:
+ system_prompt = (
+ "You are a professional summarizer at 2077 Research. Below is an article on Ethereum technical aspects. "
+ "Your goal is to produce a summary that is shorter than the original content, yet detailed enough for readers "
+ "to fully understand the piece without needing to read the original. Your summary should:\n"
+ "- Provide enough depth and detail so the user gets a complete understanding of the core ideas.\n"
+ "- Be in HTML format, use
tags for headings if needed. Avoid other heading levels.\n"
+ "- Minimize the use of bullet points. If you need to list items, you can, but prefer concise paragraph formatting.\n\n"
+ )
+ return await self.gpt_service.prompt(system_prompt, content)
+
+ def generate_summary_view(self, request):
+ if request.method == 'POST':
+ content = request.POST.get('content')
+ try:
+ gpt_summary = asyncio.run(self._generate_summary(content))
+ return JsonResponse({'summary': gpt_summary})
+ except Exception as e:
+ import logging
+ logging.error("An error occurred while generating the summary", exc_info=True)
+ return JsonResponse({'error': 'An internal error has occurred!'}, status=500)
+ return JsonResponse({'error': 'Invalid request method'}, status=400)
+
def current_slug_history(self, obj):
return current_slug_history(obj)
current_slug_history.short_description = 'Slug Change History'
@@ -36,7 +92,7 @@ def current_slug_history(self, obj):
('Article Details', {
'fields': [
'title', 'slug', 'authors', 'acknowledgement', 'categories',
- 'thumb', 'content', 'summary', 'status', 'scheduled_publish_time'
+ 'thumb', 'content', 'summary', 'gpt_summary', 'status', 'scheduled_publish_time'
]
}),
('Related Content', {
@@ -60,6 +116,12 @@ def current_slug_history(self, obj):
readonly_fields = ('views', 'current_slug_history',)
list_editable = ('status',)
+ class Media:
+ css = {
+ 'all': ('css/article_admin.css',)
+ }
+ js = ('js/article_admin.js',)
+
def display_authors(self, obj):
"""Return a comma-separated list of authors for the article."""
return ", ".join(author.user.username for author in obj.authors.all())
diff --git a/server/apps/research/migrations/0017_article_gpt_summary_alter_article_summary.py b/server/apps/research/migrations/0017_article_gpt_summary_alter_article_summary.py
new file mode 100644
index 0000000..939a188
--- /dev/null
+++ b/server/apps/research/migrations/0017_article_gpt_summary_alter_article_summary.py
@@ -0,0 +1,18 @@
+# Generated by Django 5.0.8 on 2024-12-10 18:18
+
+import tinymce.models
+from django.db import migrations
+
+class Migration(migrations.Migration):
+
+ dependencies = [
+ ('research', '0016_article_related_articles'),
+ ]
+
+ operations = [
+ migrations.AddField(
+ model_name='article',
+ name='gpt_summary',
+ field=tinymce.models.HTMLField(blank=True, null=True),
+ ),
+ ]
diff --git a/server/apps/research/models/article.py b/server/apps/research/models/article.py
index 2c3f4f8..d46e504 100644
--- a/server/apps/research/models/article.py
+++ b/server/apps/research/models/article.py
@@ -27,6 +27,7 @@ class Article(BaseModel):
title = models.TextField()
content = HTMLField(blank=True, null=True)
summary = models.TextField(blank=True)
+ gpt_summary = HTMLField(blank=True, null=True)
acknowledgement = HTMLField(blank=True, null=True)
authors = models.ManyToManyField(Author, blank=True, related_name='articles')
slug = models.SlugField(max_length=255, blank=True, db_index=True)
diff --git a/server/apps/research/serializers/article_serializer.py b/server/apps/research/serializers/article_serializer.py
index f8b02b2..ff04ebf 100644
--- a/server/apps/research/serializers/article_serializer.py
+++ b/server/apps/research/serializers/article_serializer.py
@@ -47,7 +47,7 @@ class Meta:
model = Article
fields = [
'id', 'slug', 'title', 'authors', 'thumb', 'categories', 'summary',
- 'acknowledgement', 'content', 'min_read', 'status', 'views',
+ 'acknowledgement', 'content', 'min_read', 'status', 'views', 'gpt_summary',
'created_at', 'updated_at', 'scheduled_publish_time', 'table_of_contents',
'is_sponsored', 'sponsor_color', 'sponsor_text_color', 'related_articles'
]
@@ -66,8 +66,9 @@ class Meta:
model = Article
fields = [
'title', 'slug', 'categories', 'thumb', 'content', 'summary',
- 'acknowledgement', 'status', 'authors', 'scheduled_publish_time',
- 'is_sponsored', 'sponsor_color', 'sponsor_text_color', 'related_articles'
+ 'gpt_summary', 'acknowledgement', 'status', 'authors',
+ 'scheduled_publish_time', 'is_sponsored', 'sponsor_color',
+ 'sponsor_text_color', 'related_articles'
]
def validate_related_articles(self, value):
diff --git a/server/apps/research/services/__init__.py b/server/apps/research/services/__init__.py
new file mode 100644
index 0000000..5aaea07
--- /dev/null
+++ b/server/apps/research/services/__init__.py
@@ -0,0 +1,4 @@
+"""
+Services package for the research app.
+This package contains service classes that handle business logic and external API interactions.
+"""
\ No newline at end of file
diff --git a/server/apps/research/services/gpt_service.py b/server/apps/research/services/gpt_service.py
new file mode 100644
index 0000000..4ea1010
--- /dev/null
+++ b/server/apps/research/services/gpt_service.py
@@ -0,0 +1,42 @@
+from django.conf import settings
+from openai import AsyncOpenAI
+
+class GPTService:
+ """Service for handling OpenAI GPT API interactions."""
+
+ def __init__(self):
+ self.client = AsyncOpenAI(api_key=settings.OPENAI_API_KEY)
+ self.model = "gpt-3.5-turbo"
+ self.max_tokens = 500
+
+ async def prompt(self, system: str, user: str) -> str:
+ """
+ Send a prompt to GPT and get the response.
+
+ Args:
+ system (str): The system message that sets the behavior of the assistant
+ user (str): The user's input/question
+
+ Returns:
+ str: The generated response from GPT
+
+ Raises:
+ Exception: If there's an error in the API call or if the API key is not set
+ """
+ if not settings.OPENAI_API_KEY:
+ raise Exception("OpenAI API key is not configured")
+
+ try:
+ completion = await self.client.chat.completions.create(
+ model=self.model,
+ messages=[
+ {"role": "system", "content": system},
+ {"role": "user", "content": user}
+ ],
+ max_tokens=self.max_tokens
+ )
+ # Access the response content directly from the completion object
+ return completion.choices[0].message.content
+ except Exception as e:
+ print(e)
+ raise Exception(f"Error calling OpenAI API: {str(e)}")
\ No newline at end of file
diff --git a/server/core/config/base.py b/server/core/config/base.py
index ae1ed98..a618bf0 100644
--- a/server/core/config/base.py
+++ b/server/core/config/base.py
@@ -196,4 +196,7 @@
SILENCED_SYSTEM_CHECKS = ["security.W019"]
# Tinymce API Config
-TINYMCE_API_KEY = config('TINYMCE_API_KEY')
\ No newline at end of file
+TINYMCE_API_KEY = config('TINYMCE_API_KEY')
+
+# OpenAI API Config
+OPENAI_API_KEY = config('OPENAI_API_KEY', default=None)
\ No newline at end of file
diff --git a/server/requirements.txt b/server/requirements.txt
index a9ee425..3b32972 100644
--- a/server/requirements.txt
+++ b/server/requirements.txt
@@ -1,4 +1,5 @@
amqp==5.2.0
+annotated-types==0.7.0
anyio==4.4.0
asgiref==3.8.1
attrs==24.1.0
@@ -15,6 +16,7 @@ click-plugins==1.1.1
click-repl==0.3.0
cron-descriptor==1.4.3
cryptography==43.0.0
+distro==1.9.0
Django==5.0.8
django-admin-interface==0.28.8
django-celery-beat==2.6.0
@@ -33,12 +35,16 @@ httpcore==1.0.5
httpx==0.27.0
idna==3.7
inflection==0.5.1
+jiter==0.8.2
jsonschema==4.23.0
jsonschema-specifications==2023.12.1
kombu==5.4.0
+openai==1.57.4
pillow==10.4.0
prompt_toolkit==3.0.47
pycparser==2.22
+pydantic==2.10.3
+pydantic_core==2.27.1
pyOpenSSL==24.2.1
python-crontab==3.2.0
python-dateutil==2.9.0.post0
@@ -55,6 +61,8 @@ sniffio==1.3.1
soupsieve==2.6
sqlparse==0.5.1
text-unidecode==1.3
+tqdm==4.67.1
+typing_extensions==4.12.2
tzdata==2024.1
uritemplate==4.1.1
urllib3==2.2.2
diff --git a/server/static/css/article_admin.css b/server/static/css/article_admin.css
new file mode 100644
index 0000000..fef5266
--- /dev/null
+++ b/server/static/css/article_admin.css
@@ -0,0 +1,28 @@
+.generate-summary-btn {
+ background-color: #0C4B33;
+ color: white;
+ padding: 10px 15px;
+ border: none;
+ border-radius: 4px;
+ cursor: pointer;
+ font-weight: bold;
+ margin-left: 10px;
+ align-self: baseline;
+}
+
+.generate-summary-btn:disabled {
+ background-color: #cccccc;
+ cursor: not-allowed;
+}
+
+.summary-status {
+ margin-left: 10px;
+ font-style: italic;
+ color: #666;
+}
+
+.generate-summary-container-btn {
+ display: flex;
+ flex-direction: column;
+ gap: 10px;
+}
\ No newline at end of file
diff --git a/server/static/js/article_admin.js b/server/static/js/article_admin.js
new file mode 100644
index 0000000..4e28844
--- /dev/null
+++ b/server/static/js/article_admin.js
@@ -0,0 +1,101 @@
+document.addEventListener('DOMContentLoaded', function() {
+ // Wait for TinyMCE to initialize
+ if (typeof tinymce !== 'undefined') {
+ const gptSummaryContainer = document.getElementById('gpt_summary_richtext_field');
+ if (gptSummaryContainer) {
+ const buttonContainer = document.createElement('div');
+ buttonContainer.id = 'generate-summary-container-btn';
+
+ const button = document.createElement('button');
+ button.type = 'button';
+ button.id = 'generate-summary-btn';
+ button.className = 'generate-summary-btn';
+ button.textContent = 'Generate Summary';
+
+ const statusSpan = document.createElement('p');
+ statusSpan.className = 'summary-status';
+ statusSpan.id = 'summary-status';
+
+ buttonContainer.appendChild(button);
+ buttonContainer.appendChild(statusSpan);
+
+ gptSummaryContainer.parentNode.insertBefore(buttonContainer, gptSummaryContainer.nextSibling);
+ button.parentNode.insertBefore(statusSpan, button.nextSibling);
+
+ button.addEventListener('click', function() {
+ const contentEditor = tinymce.get('content_richtext_field');
+ const gptSummaryContainer = document.getElementById('gpt_summary_richtext_field');
+ statusSpan.textContent = ' Generating summary...';
+
+ if (contentEditor && gptSummaryContainer) {
+ const content = contentEditor.getContent();
+ if (!content.trim()) {
+ alert('Please enter some content before generating a summary.');
+ return;
+ }
+
+ // Call the GPT API
+ generateSummary(content)
+ .then(() => {
+ statusSpan.textContent = ' Summary generated successfully!';
+ })
+ .catch(error => {
+ statusSpan.textContent = ' Error generating summary. Please try again.';
+ })
+ .finally(() => {
+ // Re-enable editors and button
+ contentEditor.setMode('design');
+ gptSummaryEditor.setMode('design');
+ button.disabled = false;
+ });
+ }
+ });
+ }
+ }
+});
+
+async function generateSummary(content) {
+ try {
+ const response = await fetch('/admin/research/article/generate-summary/', {
+ method: 'POST',
+ headers: {
+ 'Content-Type': 'application/x-www-form-urlencoded',
+ 'X-CSRFToken': getCookie('csrftoken')
+ },
+ body: 'content=' + encodeURIComponent(content)
+ });
+
+ if (!response.ok) {
+ throw new Error('Network response was not ok');
+ }
+
+ const data = await response.json();
+
+ // Update the summary field
+ if (typeof tinymce !== 'undefined') {
+ const summaryEditor = tinymce.get('gpt_summary_richtext_field');
+ if (summaryEditor) {
+ summaryEditor.setContent(data.summary);
+ }
+ }
+ } catch (error) {
+ console.error('Error:', error);
+ throw error;
+ }
+}
+
+// Helper function to get CSRF token
+function getCookie(name) {
+ let cookieValue = null;
+ if (document.cookie && document.cookie !== '') {
+ const cookies = document.cookie.split(';');
+ for (let i = 0; i < cookies.length; i++) {
+ const cookie = cookies[i].trim();
+ if (cookie.substring(0, name.length + 1) === (name + '=')) {
+ cookieValue = decodeURIComponent(cookie.substring(name.length + 1));
+ break;
+ }
+ }
+ }
+ return cookieValue;
+}
\ No newline at end of file