refactor project structure
Run Tests / Run Tests (push) Failing after 31s
Details
Run Tests / Run Tests (push) Failing after 31s
Details
This commit is contained in:
parent
ee6d7c26a1
commit
c82cb00013
|
@ -0,0 +1,3 @@
|
|||
from .celery import app as celery_app
|
||||
|
||||
__all__ = ('celery_app',)
|
|
@ -42,7 +42,7 @@ INSTALLED_APPS = [
|
|||
"django.contrib.sessions",
|
||||
"django.contrib.messages",
|
||||
"django.contrib.staticfiles",
|
||||
"webui",
|
||||
"penparse",
|
||||
"markdown_deux",
|
||||
"markdownify.apps.MarkdownifyConfig",
|
||||
"rest_framework",
|
||||
|
@ -129,9 +129,9 @@ AUTH_PASSWORD_VALIDATORS = [
|
|||
|
||||
LOGIN_REDIRECT_URL = "/dashboard"
|
||||
|
||||
AUTH_USER_MODEL = "webui.User"
|
||||
AUTH_USER_MODEL = "penparse.User"
|
||||
|
||||
AUTHENTICATION_BACKENDS = ["webui.auth.EmailBackend"]
|
||||
AUTHENTICATION_BACKENDS = ["penparse.auth.EmailBackend"]
|
||||
|
||||
# Internationalization
|
||||
# https://docs.djangoproject.com/en/4.2/topics/i18n/
|
||||
|
@ -156,7 +156,7 @@ STATIC_URL = "static/"
|
|||
DEFAULT_AUTO_FIELD = "django.db.models.BigAutoField"
|
||||
|
||||
|
||||
CELERY_BROKER_URL = "amqp://guest:guest@localhost/"
|
||||
CELERY_BROKER_URL = os.getenv("CELERY_BROKER_URL")
|
||||
|
||||
|
||||
OPENAI_API_BASE = os.getenv("OPENAI_API_BASE")
|
|
@ -37,7 +37,7 @@ INSTALLED_APPS = [
|
|||
"django.contrib.sessions",
|
||||
"django.contrib.messages",
|
||||
"django.contrib.staticfiles",
|
||||
"webui",
|
||||
"penparse",
|
||||
"markdown_deux",
|
||||
"markdownify.apps.MarkdownifyConfig",
|
||||
"rest_framework",
|
||||
|
@ -115,9 +115,9 @@ AUTH_PASSWORD_VALIDATORS = [
|
|||
|
||||
LOGIN_REDIRECT_URL = "/dashboard"
|
||||
|
||||
AUTH_USER_MODEL = "webui.User"
|
||||
AUTH_USER_MODEL = "penparse.User"
|
||||
|
||||
AUTHENTICATION_BACKENDS = ["webui.auth.EmailBackend"]
|
||||
AUTHENTICATION_BACKENDS = ["penparse.auth.EmailBackend"]
|
||||
|
||||
# Internationalization
|
||||
# https://docs.djangoproject.com/en/4.2/topics/i18n/
|
|
@ -0,0 +1,26 @@
|
|||
"""
|
||||
URL configuration for penparse project.
|
||||
|
||||
The `urlpatterns` list routes URLs to views. For more information please see:
|
||||
https://docs.djangoproject.com/en/4.2/topics/http/urls/
|
||||
Examples:
|
||||
Function views
|
||||
1. Add an import: from my_app import views
|
||||
2. Add a URL to urlpatterns: path('', views.home, name='home')
|
||||
Class-based views
|
||||
1. Add an import: from other_app.views import Home
|
||||
2. Add a URL to urlpatterns: path('', Home.as_view(), name='home')
|
||||
Including another URLconf
|
||||
1. Import the include() function: from django.urls import include, path
|
||||
2. Add a URL to urlpatterns: path('blog/', include('blog.urls'))
|
||||
"""
|
||||
from django.urls import include, path
|
||||
from django.contrib import admin
|
||||
from django.urls import path, include
|
||||
|
||||
urlpatterns = [
|
||||
path("", include("penparse.urls")),
|
||||
path('admin/', admin.site.urls),
|
||||
path("accounts/", include("django.contrib.auth.urls")),
|
||||
|
||||
]
|
|
@ -6,7 +6,7 @@ import sys
|
|||
|
||||
def main():
|
||||
"""Run administrative tasks."""
|
||||
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'penparse.settings')
|
||||
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'config.settings')
|
||||
try:
|
||||
from django.core.management import execute_from_command_line
|
||||
except ImportError as exc:
|
||||
|
|
|
@ -1,3 +0,0 @@
|
|||
from .celery import app as celery_app
|
||||
|
||||
__all__ = ('celery_app',)
|
|
@ -1,6 +1,6 @@
|
|||
from django.apps import AppConfig
|
||||
|
||||
|
||||
class WebuiConfig(AppConfig):
|
||||
class PenParseConfig(AppConfig):
|
||||
default_auto_field = 'django.db.models.BigAutoField'
|
||||
name = 'webui'
|
||||
name = 'penparse'
|
|
@ -0,0 +1,51 @@
|
|||
# Generated by Django 4.2.16 on 2024-11-30 06:31
|
||||
|
||||
from django.db import migrations, models
|
||||
import django.utils.timezone
|
||||
import penparse.models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
('auth', '0012_alter_user_first_name_max_length'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='User',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True,
|
||||
primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('password', models.CharField(
|
||||
max_length=128, verbose_name='password')),
|
||||
('last_login', models.DateTimeField(
|
||||
blank=True, null=True, verbose_name='last login')),
|
||||
('is_superuser', models.BooleanField(default=False,
|
||||
help_text='Designates that this user has all permissions without explicitly assigning them.', verbose_name='superuser status')),
|
||||
('is_staff', models.BooleanField(default=False,
|
||||
help_text='Designates whether the user can log into this admin site.', verbose_name='staff status')),
|
||||
('is_active', models.BooleanField(
|
||||
default=True, help_text='Designates whether this user should be treated as active. Unselect this instead of deleting accounts.', verbose_name='active')),
|
||||
('date_joined', models.DateTimeField(
|
||||
default=django.utils.timezone.now, verbose_name='date joined')),
|
||||
('email', models.EmailField(max_length=254, unique=True)),
|
||||
('first_name', models.CharField(max_length=150)),
|
||||
('last_name', models.CharField(max_length=150)),
|
||||
('groups', models.ManyToManyField(blank=True, help_text='The groups this user belongs to. A user will get all permissions granted to each of their groups.',
|
||||
related_name='user_set', related_query_name='user', to='auth.group', verbose_name='groups')),
|
||||
('user_permissions', models.ManyToManyField(blank=True, help_text='Specific permissions for this user.',
|
||||
related_name='user_set', related_query_name='user', to='auth.permission', verbose_name='user permissions')),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'user',
|
||||
'verbose_name_plural': 'users',
|
||||
'abstract': False,
|
||||
},
|
||||
managers=[
|
||||
('objects', penparse.models.UserManager()),
|
||||
],
|
||||
),
|
||||
]
|
|
@ -9,19 +9,21 @@ import uuid
|
|||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('webui', '0001_initial'),
|
||||
('penparse', '0001_initial'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='ImageMemo',
|
||||
fields=[
|
||||
('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)),
|
||||
('id', models.UUIDField(default=uuid.uuid4,
|
||||
editable=False, primary_key=True, serialize=False)),
|
||||
('image', models.ImageField(upload_to='')),
|
||||
('content', models.TextField()),
|
||||
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||
('updated_at', models.DateTimeField(auto_now=True)),
|
||||
('author', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='memos', to=settings.AUTH_USER_MODEL)),
|
||||
('author', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE,
|
||||
related_name='memos', to=settings.AUTH_USER_MODEL)),
|
||||
],
|
||||
options={
|
||||
'ordering': ['-created_at'],
|
|
@ -6,7 +6,7 @@ from django.db import migrations, models
|
|||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('webui', '0002_imagememo'),
|
||||
('penparse', '0002_imagememo'),
|
||||
]
|
||||
|
||||
operations = [
|
|
@ -6,13 +6,14 @@ from django.db import migrations, models
|
|||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('webui', '0003_imagememo_image_mimetype_alter_imagememo_image'),
|
||||
('penparse', '0003_imagememo_image_mimetype_alter_imagememo_image'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='imagememo',
|
||||
name='status',
|
||||
field=models.CharField(choices=[('pending', 'Pending'), ('processing', 'Processing'), ('done', 'Done'), ('error', 'Error')], default='pending', max_length=10),
|
||||
field=models.CharField(choices=[('pending', 'Pending'), ('processing', 'Processing'), (
|
||||
'done', 'Done'), ('error', 'Error')], default='pending', max_length=10),
|
||||
),
|
||||
]
|
|
@ -6,7 +6,7 @@ from django.db import migrations, models
|
|||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('webui', '0004_imagememo_status'),
|
||||
('penparse', '0004_imagememo_status'),
|
||||
]
|
||||
|
||||
operations = [
|
|
@ -6,7 +6,7 @@ from django.db import migrations, models
|
|||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('webui', '0005_imagememo_error_message'),
|
||||
('penparse', '0005_imagememo_error_message'),
|
||||
]
|
||||
|
||||
operations = [
|
Before Width: | Height: | Size: 27 KiB After Width: | Height: | Size: 27 KiB |
Before Width: | Height: | Size: 751 KiB After Width: | Height: | Size: 751 KiB |
Before Width: | Height: | Size: 145 KiB After Width: | Height: | Size: 145 KiB |
|
@ -13,7 +13,8 @@ class TestProcessMemo:
|
|||
|
||||
@pytest.fixture
|
||||
def sample_image_memo(self, db):
|
||||
user1 = User.objects.create_user(email="user1@test.com", password="password1")
|
||||
user1 = User.objects.create_user(
|
||||
email="user1@test.com", password="password1")
|
||||
memo = ImageMemo.objects.create(
|
||||
author=user1,
|
||||
status=MemoStatus.Pending,
|
||||
|
@ -23,9 +24,10 @@ class TestProcessMemo:
|
|||
return memo
|
||||
|
||||
def test_process_memo_success(self, sample_image_memo):
|
||||
with patch("webui.tasks.litellm") as mock_litellm:
|
||||
with patch("penparse.tasks.litellm") as mock_litellm:
|
||||
mock_response = MagicMock()
|
||||
mock_response.choices[0].message = {"content": "Transcribed content"}
|
||||
mock_response.choices[0].message = {
|
||||
"content": "Transcribed content"}
|
||||
mock_litellm.completion.return_value = mock_response
|
||||
|
||||
process_memo(sample_image_memo.id)
|
||||
|
@ -45,10 +47,12 @@ class TestProcessMemo:
|
|||
assert "Image file" in processed_memo.error_message
|
||||
|
||||
def test_process_memo_api_error(self, sample_image_memo):
|
||||
with patch("webui.tasks.litellm") as mock_litellm:
|
||||
with patch("penparse.tasks.litellm") as mock_litellm:
|
||||
mock_response = MagicMock()
|
||||
mock_response.choices[0].message = {"content": "Transcribed content"}
|
||||
mock_litellm.completion.side_effect = litellm.APIError(400, "API Error", "openai", "any")
|
||||
mock_response.choices[0].message = {
|
||||
"content": "Transcribed content"}
|
||||
mock_litellm.completion.side_effect = litellm.APIError(
|
||||
400, "API Error", "openai", "any")
|
||||
|
||||
process_memo(sample_image_memo.id)
|
||||
|
||||
|
@ -58,11 +62,12 @@ class TestProcessMemo:
|
|||
|
||||
def test_process_memo_sets_model_name(self, sample_image_memo):
|
||||
with (
|
||||
patch("webui.tasks.litellm") as mock_litellm,
|
||||
patch("webui.tasks.settings") as mock_settings,
|
||||
patch("penparse.tasks.litellm") as mock_litellm,
|
||||
patch("penparse.tasks.settings") as mock_settings,
|
||||
):
|
||||
mock_response = MagicMock()
|
||||
mock_response.choices[0].message = {"content": "Transcribed content"}
|
||||
mock_response.choices[0].message = {
|
||||
"content": "Transcribed content"}
|
||||
mock_litellm.completion.return_value = mock_response
|
||||
mock_settings.OPENAI_MODEL = "test-model"
|
||||
|
||||
|
@ -73,11 +78,12 @@ class TestProcessMemo:
|
|||
|
||||
def test_process_memo_uses_correct_api_settings(self, sample_image_memo):
|
||||
with (
|
||||
patch("webui.tasks.litellm") as mock_litellm,
|
||||
patch("webui.tasks.settings") as mock_settings,
|
||||
patch("penparse.tasks.litellm") as mock_litellm,
|
||||
patch("penparse.tasks.settings") as mock_settings,
|
||||
):
|
||||
mock_response = MagicMock()
|
||||
mock_response.choices[0].message = {"content": "Transcribed content"}
|
||||
mock_response.choices[0].message = {
|
||||
"content": "Transcribed content"}
|
||||
mock_litellm.completion.return_value = mock_response
|
||||
mock_settings.OPENAI_API_BASE = "https://test-api-base.com"
|
||||
mock_settings.OPENAI_API_KEY = "test-api-key"
|
|
@ -1,26 +1,26 @@
|
|||
"""
|
||||
URL configuration for penparse project.
|
||||
from django.urls import path
|
||||
|
||||
The `urlpatterns` list routes URLs to views. For more information please see:
|
||||
https://docs.djangoproject.com/en/4.2/topics/http/urls/
|
||||
Examples:
|
||||
Function views
|
||||
1. Add an import: from my_app import views
|
||||
2. Add a URL to urlpatterns: path('', views.home, name='home')
|
||||
Class-based views
|
||||
1. Add an import: from other_app.views import Home
|
||||
2. Add a URL to urlpatterns: path('', Home.as_view(), name='home')
|
||||
Including another URLconf
|
||||
1. Import the include() function: from django.urls import include, path
|
||||
2. Add a URL to urlpatterns: path('blog/', include('blog.urls'))
|
||||
"""
|
||||
from django.urls import include, path
|
||||
from django.contrib import admin
|
||||
from django.urls import path, include
|
||||
from . import views
|
||||
|
||||
urlpatterns = [
|
||||
path("", include("webui.urls")),
|
||||
path('admin/', admin.site.urls),
|
||||
path("accounts/", include("django.contrib.auth.urls")),
|
||||
|
||||
path("", views.index, name="index"),
|
||||
path("dashboard", views.dashboard, name="dashboard"),
|
||||
path("settings", views.settings, name="settings"),
|
||||
path("documents/upload", views.upload_document, name="upload_document"),
|
||||
path("documents/<str:pk>", views.view_document, name="view_document"),
|
||||
path(
|
||||
"documents/<str:pk>/thumbnail",
|
||||
views.document_thumbnail,
|
||||
name="document_thumbnail",
|
||||
),
|
||||
path(
|
||||
"documents/<str:pk>/image",
|
||||
views.document_image,
|
||||
name="document_image",
|
||||
),
|
||||
path(
|
||||
"documents/<str:pk>/download", views.download_document, name="download_document"
|
||||
),
|
||||
path("documents/<str:pk>/delete", views.delete_document, name="delete_document"),
|
||||
path("auth/register", views.register, name="register"),
|
||||
]
|
||||
|
|
|
@ -1,42 +0,0 @@
|
|||
# Generated by Django 4.2.16 on 2024-11-30 06:31
|
||||
|
||||
from django.db import migrations, models
|
||||
import django.utils.timezone
|
||||
import webui.models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
initial = True
|
||||
|
||||
dependencies = [
|
||||
('auth', '0012_alter_user_first_name_max_length'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.CreateModel(
|
||||
name='User',
|
||||
fields=[
|
||||
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||
('password', models.CharField(max_length=128, verbose_name='password')),
|
||||
('last_login', models.DateTimeField(blank=True, null=True, verbose_name='last login')),
|
||||
('is_superuser', models.BooleanField(default=False, help_text='Designates that this user has all permissions without explicitly assigning them.', verbose_name='superuser status')),
|
||||
('is_staff', models.BooleanField(default=False, help_text='Designates whether the user can log into this admin site.', verbose_name='staff status')),
|
||||
('is_active', models.BooleanField(default=True, help_text='Designates whether this user should be treated as active. Unselect this instead of deleting accounts.', verbose_name='active')),
|
||||
('date_joined', models.DateTimeField(default=django.utils.timezone.now, verbose_name='date joined')),
|
||||
('email', models.EmailField(max_length=254, unique=True)),
|
||||
('first_name', models.CharField(max_length=150)),
|
||||
('last_name', models.CharField(max_length=150)),
|
||||
('groups', models.ManyToManyField(blank=True, help_text='The groups this user belongs to. A user will get all permissions granted to each of their groups.', related_name='user_set', related_query_name='user', to='auth.group', verbose_name='groups')),
|
||||
('user_permissions', models.ManyToManyField(blank=True, help_text='Specific permissions for this user.', related_name='user_set', related_query_name='user', to='auth.permission', verbose_name='user permissions')),
|
||||
],
|
||||
options={
|
||||
'verbose_name': 'user',
|
||||
'verbose_name_plural': 'users',
|
||||
'abstract': False,
|
||||
},
|
||||
managers=[
|
||||
('objects', webui.models.UserManager()),
|
||||
],
|
||||
),
|
||||
]
|
|
@ -1,26 +0,0 @@
|
|||
from django.urls import path
|
||||
|
||||
from . import views
|
||||
|
||||
urlpatterns = [
|
||||
path("", views.index, name="index"),
|
||||
path("dashboard", views.dashboard, name="dashboard"),
|
||||
path("settings", views.settings, name="settings"),
|
||||
path("documents/upload", views.upload_document, name="upload_document"),
|
||||
path("documents/<str:pk>", views.view_document, name="view_document"),
|
||||
path(
|
||||
"documents/<str:pk>/thumbnail",
|
||||
views.document_thumbnail,
|
||||
name="document_thumbnail",
|
||||
),
|
||||
path(
|
||||
"documents/<str:pk>/image",
|
||||
views.document_image,
|
||||
name="document_image",
|
||||
),
|
||||
path(
|
||||
"documents/<str:pk>/download", views.download_document, name="download_document"
|
||||
),
|
||||
path("documents/<str:pk>/delete", views.delete_document, name="delete_document"),
|
||||
path("auth/register", views.register, name="register"),
|
||||
]
|
Loading…
Reference in New Issue