commit
351c733d82
24 changed files with 627 additions and 0 deletions
@ -0,0 +1,138 @@ |
|||||
|
# Django # |
||||
|
*.log |
||||
|
*.pot |
||||
|
*.pyc |
||||
|
__pycache__ |
||||
|
db.sqlite3 |
||||
|
media |
||||
|
|
||||
|
# Backup files # |
||||
|
*.bak |
||||
|
|
||||
|
# If you are using PyCharm # |
||||
|
# User-specific stuff |
||||
|
.idea/**/workspace.xml |
||||
|
.idea/**/tasks.xml |
||||
|
.idea/**/usage.statistics.xml |
||||
|
.idea/**/dictionaries |
||||
|
.idea/**/shelf |
||||
|
|
||||
|
# AWS User-specific |
||||
|
.idea/**/aws.xml |
||||
|
|
||||
|
# Generated files |
||||
|
.idea/**/contentModel.xml |
||||
|
|
||||
|
# Sensitive or high-churn files |
||||
|
.idea/**/dataSources/ |
||||
|
.idea/**/dataSources.ids |
||||
|
.idea/**/dataSources.local.xml |
||||
|
.idea/**/sqlDataSources.xml |
||||
|
.idea/**/dynamic.xml |
||||
|
.idea/**/uiDesigner.xml |
||||
|
.idea/**/dbnavigator.xml |
||||
|
|
||||
|
# Gradle |
||||
|
.idea/**/gradle.xml |
||||
|
.idea/**/libraries |
||||
|
|
||||
|
# File-based project format |
||||
|
*.iws |
||||
|
|
||||
|
# IntelliJ |
||||
|
out/ |
||||
|
|
||||
|
# JIRA plugin |
||||
|
atlassian-ide-plugin.xml |
||||
|
|
||||
|
# Python # |
||||
|
*.py[cod] |
||||
|
*$py.class |
||||
|
|
||||
|
# Distribution / packaging |
||||
|
.Python build/ |
||||
|
develop-eggs/ |
||||
|
dist/ |
||||
|
downloads/ |
||||
|
eggs/ |
||||
|
.eggs/ |
||||
|
lib/ |
||||
|
lib64/ |
||||
|
parts/ |
||||
|
sdist/ |
||||
|
var/ |
||||
|
wheels/ |
||||
|
*.whl |
||||
|
*.egg-info/ |
||||
|
.installed.cfg |
||||
|
*.egg |
||||
|
*.manifest |
||||
|
*.spec |
||||
|
|
||||
|
# Installer logs |
||||
|
pip-log.txt |
||||
|
pip-delete-this-directory.txt |
||||
|
|
||||
|
# Unit test / coverage reports |
||||
|
htmlcov/ |
||||
|
.tox/ |
||||
|
.coverage |
||||
|
.coverage.* |
||||
|
.cache |
||||
|
.pytest_cache/ |
||||
|
nosetests.xml |
||||
|
coverage.xml |
||||
|
*.cover |
||||
|
.hypothesis/ |
||||
|
|
||||
|
# Jupyter Notebook |
||||
|
.ipynb_checkpoints |
||||
|
|
||||
|
# pyenv |
||||
|
.python-version |
||||
|
|
||||
|
# celery |
||||
|
celerybeat-schedule.* |
||||
|
|
||||
|
# SageMath parsed files |
||||
|
*.sage.py |
||||
|
|
||||
|
# Environments |
||||
|
.env |
||||
|
.venv |
||||
|
env/ |
||||
|
venv/ |
||||
|
ENV/ |
||||
|
env.bak/ |
||||
|
venv.bak/ |
||||
|
|
||||
|
# mkdocs documentation |
||||
|
/site |
||||
|
|
||||
|
# mypy |
||||
|
.mypy_cache/ |
||||
|
|
||||
|
# Sublime Text # |
||||
|
*.tmlanguage.cache |
||||
|
*.tmPreferences.cache |
||||
|
*.stTheme.cache |
||||
|
*.sublime-workspace |
||||
|
*.sublime-project |
||||
|
|
||||
|
# sftp configuration file |
||||
|
sftp-config.json |
||||
|
|
||||
|
# Package control specific files Package |
||||
|
Control.last-run |
||||
|
Control.ca-list |
||||
|
Control.ca-bundle |
||||
|
Control.system-ca-bundle |
||||
|
GitHub.sublime-settings |
||||
|
|
||||
|
# Visual Studio Code # |
||||
|
.vscode/* |
||||
|
!.vscode/settings.json |
||||
|
!.vscode/tasks.json |
||||
|
!.vscode/launch.json |
||||
|
!.vscode/extensions.json |
||||
|
.history |
||||
@ -0,0 +1,20 @@ |
|||||
|
### ```python3 -m venv back_venv``` |
||||
|
Creates python virutal env |
||||
|
|
||||
|
### ```. back_venv/bin/activate``` |
||||
|
uses venv |
||||
|
|
||||
|
### ```python manage.py runserver``` |
||||
|
launches the development server at ```127.0.0.1:8000``` |
||||
|
|
||||
|
### ```python manage.py runserver <your_port>``` |
||||
|
launches the development server at a custom port |
||||
|
|
||||
|
### ```python manage.py startapp <app_name>``` |
||||
|
creates a new app in the current directory |
||||
|
|
||||
|
### ```python manage.py createsuperuser``` |
||||
|
starts admin user creation process in terminal |
||||
|
|
||||
|
### ```pip install -r requirments.txt``` |
||||
|
installs requirements filed |
||||
@ -0,0 +1,16 @@ |
|||||
|
""" |
||||
|
ASGI config for backend project. |
||||
|
|
||||
|
It exposes the ASGI callable as a module-level variable named ``application``. |
||||
|
|
||||
|
For more information on this file, see |
||||
|
https://docs.djangoproject.com/en/4.1/howto/deployment/asgi/ |
||||
|
""" |
||||
|
|
||||
|
import os |
||||
|
|
||||
|
from django.core.asgi import get_asgi_application |
||||
|
|
||||
|
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "backend.settings") |
||||
|
|
||||
|
application = get_asgi_application() |
||||
@ -0,0 +1,177 @@ |
|||||
|
""" |
||||
|
Django settings for backend project. |
||||
|
|
||||
|
Generated by 'django-admin startproject' using Django 4.1.1. |
||||
|
|
||||
|
For more information on this file, see |
||||
|
https://docs.djangoproject.com/en/4.1/topics/settings/ |
||||
|
|
||||
|
For the full list of settings and their values, see |
||||
|
https://docs.djangoproject.com/en/4.1/ref/settings/ |
||||
|
""" |
||||
|
|
||||
|
from pathlib import Path |
||||
|
from datetime import timedelta |
||||
|
|
||||
|
# Build paths inside the project like this: BASE_DIR / 'subdir'. |
||||
|
BASE_DIR = Path(__file__).resolve().parent.parent |
||||
|
|
||||
|
|
||||
|
# Quick-start development settings - unsuitable for production |
||||
|
# See https://docs.djangoproject.com/en/4.1/howto/deployment/checklist/ |
||||
|
|
||||
|
# SECURITY WARNING: keep the secret key used in production secret! |
||||
|
SECRET_KEY = "django-insecure-o&qo$m$13v0*=0t+#l&$j^vnvpz@vitmabf6h#b@!@4zy1w!qx" |
||||
|
|
||||
|
# SECURITY WARNING: don't run with debug turned on in production! |
||||
|
DEBUG = True |
||||
|
|
||||
|
ALLOWED_HOSTS = [] |
||||
|
|
||||
|
|
||||
|
# Application definition |
||||
|
|
||||
|
INSTALLED_APPS = [ |
||||
|
"django.contrib.admin", |
||||
|
"django.contrib.auth", |
||||
|
"django.contrib.contenttypes", |
||||
|
"django.contrib.sessions", |
||||
|
"django.contrib.messages", |
||||
|
"django.contrib.staticfiles", |
||||
|
"rest_framework", |
||||
|
'rest_framework_simplejwt.token_blacklist', |
||||
|
'corsheaders', |
||||
|
'base', |
||||
|
] |
||||
|
|
||||
|
CORS_ALLOW_ALL_ORIGINS = True |
||||
|
|
||||
|
REST_FRAMEWORK = { |
||||
|
'DEFAULT_AUTHENTICATION_CLASSES': ( |
||||
|
'rest_framework_simplejwt.authentication.JWTAuthentication', |
||||
|
) |
||||
|
} |
||||
|
|
||||
|
SIMPLE_JWT = { |
||||
|
"ACCESS_TOKEN_LIFETIME": timedelta(minutes=5), |
||||
|
"REFRESH_TOKEN_LIFETIME": timedelta(days=15), |
||||
|
"ROTATE_REFRESH_TOKENS": True, |
||||
|
"BLACKLIST_AFTER_ROTATION": True, |
||||
|
"UPDATE_LAST_LOGIN": False, |
||||
|
|
||||
|
"ALGORITHM": "HS256", |
||||
|
"SIGNING_KEY": SECRET_KEY, |
||||
|
"VERIFYING_KEY": "", |
||||
|
"AUDIENCE": None, |
||||
|
"ISSUER": None, |
||||
|
"JSON_ENCODER": None, |
||||
|
"JWK_URL": None, |
||||
|
"LEEWAY": 0, |
||||
|
|
||||
|
"AUTH_HEADER_TYPES": ("Bearer",), |
||||
|
"AUTH_HEADER_NAME": "HTTP_AUTHORIZATION", |
||||
|
"USER_ID_FIELD": "id", |
||||
|
"USER_ID_CLAIM": "user_id", |
||||
|
"USER_AUTHENTICATION_RULE": "rest_framework_simplejwt.authentication.default_user_authentication_rule", |
||||
|
|
||||
|
"AUTH_TOKEN_CLASSES": ("rest_framework_simplejwt.tokens.AccessToken",), |
||||
|
"TOKEN_TYPE_CLAIM": "token_type", |
||||
|
"TOKEN_USER_CLASS": "rest_framework_simplejwt.models.TokenUser", |
||||
|
|
||||
|
"JTI_CLAIM": "jti", |
||||
|
|
||||
|
"SLIDING_TOKEN_REFRESH_EXP_CLAIM": "refresh_exp", |
||||
|
"SLIDING_TOKEN_LIFETIME": timedelta(minutes=5), |
||||
|
"SLIDING_TOKEN_REFRESH_LIFETIME": timedelta(days=1), |
||||
|
|
||||
|
"TOKEN_OBTAIN_SERIALIZER": "rest_framework_simplejwt.serializers.MyTokenObtainPairSerializer", |
||||
|
"TOKEN_REFRESH_SERIALIZER": "rest_framework_simplejwt.serializers.TokenRefreshSerializer", |
||||
|
"TOKEN_VERIFY_SERIALIZER": "rest_framework_simplejwt.serializers.TokenVerifySerializer", |
||||
|
"TOKEN_BLACKLIST_SERIALIZER": "rest_framework_simplejwt.serializers.TokenBlacklistSerializer", |
||||
|
"SLIDING_TOKEN_OBTAIN_SERIALIZER": "rest_framework_simplejwt.serializers.TokenObtainSlidingSerializer", |
||||
|
"SLIDING_TOKEN_REFRESH_SERIALIZER": "rest_framework_simplejwt.serializers.TokenRefreshSlidingSerializer", |
||||
|
} |
||||
|
|
||||
|
MIDDLEWARE = [ |
||||
|
"corsheaders.middleware.CorsMiddleware", |
||||
|
"django.middleware.common.CommonMiddleware", |
||||
|
"django.middleware.security.SecurityMiddleware", |
||||
|
"django.contrib.sessions.middleware.SessionMiddleware", |
||||
|
"django.middleware.common.CommonMiddleware", |
||||
|
"django.middleware.csrf.CsrfViewMiddleware", |
||||
|
"django.contrib.auth.middleware.AuthenticationMiddleware", |
||||
|
"django.contrib.messages.middleware.MessageMiddleware", |
||||
|
"django.middleware.clickjacking.XFrameOptionsMiddleware", |
||||
|
] |
||||
|
|
||||
|
ROOT_URLCONF = "backend.urls" |
||||
|
|
||||
|
TEMPLATES = [ |
||||
|
{ |
||||
|
"BACKEND": "django.template.backends.django.DjangoTemplates", |
||||
|
"DIRS": [], |
||||
|
"APP_DIRS": True, |
||||
|
"OPTIONS": { |
||||
|
"context_processors": [ |
||||
|
"django.template.context_processors.debug", |
||||
|
"django.template.context_processors.request", |
||||
|
"django.contrib.auth.context_processors.auth", |
||||
|
"django.contrib.messages.context_processors.messages", |
||||
|
], |
||||
|
}, |
||||
|
}, |
||||
|
] |
||||
|
|
||||
|
WSGI_APPLICATION = "backend.wsgi.application" |
||||
|
|
||||
|
# Database |
||||
|
# https://docs.djangoproject.com/en/4.1/ref/settings/#databases |
||||
|
|
||||
|
DATABASES = { |
||||
|
"default": { |
||||
|
"ENGINE": "django.db.backends.sqlite3", |
||||
|
"NAME": BASE_DIR / "db.sqlite3", |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
|
||||
|
# Password validation |
||||
|
# https://docs.djangoproject.com/en/4.1/ref/settings/#auth-password-validators |
||||
|
|
||||
|
AUTH_PASSWORD_VALIDATORS = [ |
||||
|
{ |
||||
|
"NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator", |
||||
|
}, |
||||
|
{ |
||||
|
"NAME": "django.contrib.auth.password_validation.MinimumLengthValidator", |
||||
|
}, |
||||
|
{ |
||||
|
"NAME": "django.contrib.auth.password_validation.CommonPasswordValidator", |
||||
|
}, |
||||
|
{ |
||||
|
"NAME": "django.contrib.auth.password_validation.NumericPasswordValidator", |
||||
|
}, |
||||
|
] |
||||
|
|
||||
|
|
||||
|
# Internationalization |
||||
|
# https://docs.djangoproject.com/en/4.1/topics/i18n/ |
||||
|
|
||||
|
LANGUAGE_CODE = "en-us" |
||||
|
|
||||
|
TIME_ZONE = "UTC" |
||||
|
|
||||
|
USE_I18N = True |
||||
|
|
||||
|
USE_TZ = True |
||||
|
|
||||
|
|
||||
|
# Static files (CSS, JavaScript, Images) |
||||
|
# https://docs.djangoproject.com/en/4.1/howto/static-files/ |
||||
|
|
||||
|
STATIC_URL = "static/" |
||||
|
|
||||
|
# Default primary key field type |
||||
|
# https://docs.djangoproject.com/en/4.1/ref/settings/#default-auto-field |
||||
|
|
||||
|
DEFAULT_AUTO_FIELD = "django.db.models.BigAutoField" |
||||
@ -0,0 +1,23 @@ |
|||||
|
"""backend URL Configuration |
||||
|
|
||||
|
The `urlpatterns` list routes URLs to views. For more information please see: |
||||
|
https://docs.djangoproject.com/en/4.1/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.contrib import admin |
||||
|
from django.urls import path, include |
||||
|
|
||||
|
urlpatterns = [ |
||||
|
path('admin/', admin.site.urls), |
||||
|
path('api/', include('base.api.urls')) |
||||
|
] |
||||
@ -0,0 +1,16 @@ |
|||||
|
""" |
||||
|
WSGI config for backend project. |
||||
|
|
||||
|
It exposes the WSGI callable as a module-level variable named ``application``. |
||||
|
|
||||
|
For more information on this file, see |
||||
|
https://docs.djangoproject.com/en/4.1/howto/deployment/wsgi/ |
||||
|
""" |
||||
|
|
||||
|
import os |
||||
|
|
||||
|
from django.core.wsgi import get_wsgi_application |
||||
|
|
||||
|
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "backend.settings") |
||||
|
|
||||
|
application = get_wsgi_application() |
||||
@ -0,0 +1,6 @@ |
|||||
|
from django.contrib import admin |
||||
|
from .models import * |
||||
|
|
||||
|
# Register your models here. |
||||
|
|
||||
|
admin.site.register(Profile) |
||||
@ -0,0 +1,14 @@ |
|||||
|
|
||||
|
from django.urls import path |
||||
|
from . import views |
||||
|
from .views import MyTokenObtainPairView |
||||
|
|
||||
|
from rest_framework_simplejwt.views import ( |
||||
|
TokenRefreshView, |
||||
|
) |
||||
|
|
||||
|
urlpatterns = [ |
||||
|
path('profile/', views.get_profile), |
||||
|
path('token/', MyTokenObtainPairView.as_view(), name='token_obtain_pair'), |
||||
|
path('token/refresh/', TokenRefreshView.as_view(), name='token_refresh'), |
||||
|
] |
||||
@ -0,0 +1,28 @@ |
|||||
|
from rest_framework.response import Response |
||||
|
from rest_framework.decorators import api_view, permission_classes |
||||
|
from rest_framework.permissions import IsAuthenticated |
||||
|
|
||||
|
from rest_framework_simplejwt.serializers import TokenObtainPairSerializer |
||||
|
from rest_framework_simplejwt.views import TokenObtainPairView |
||||
|
|
||||
|
from base.serializer import ProfileSerializer |
||||
|
|
||||
|
class MyTokenObtainPairSerializer(TokenObtainPairSerializer): |
||||
|
@classmethod |
||||
|
def get_token(cls, user): |
||||
|
token = super().get_token(user) |
||||
|
|
||||
|
token['username'] = user.username |
||||
|
|
||||
|
return token |
||||
|
|
||||
|
class MyTokenObtainPairView(TokenObtainPairView): |
||||
|
serializer_class = MyTokenObtainPairSerializer |
||||
|
|
||||
|
@api_view(['GET']) |
||||
|
@permission_classes([IsAuthenticated]) |
||||
|
def get_profile(request): |
||||
|
user = request.user |
||||
|
profile = user.profile |
||||
|
serializer = ProfileSerializer(profile, many=False) |
||||
|
return Response(serializer.data) |
||||
@ -0,0 +1,8 @@ |
|||||
|
from django.apps import AppConfig |
||||
|
|
||||
|
class BaseConfig(AppConfig): |
||||
|
default_auto_field = "django.db.models.BigAutoField" |
||||
|
name = "base" |
||||
|
|
||||
|
def ready(self): |
||||
|
from .signals import create_profile, save_profile |
||||
@ -0,0 +1,39 @@ |
|||||
|
# Generated by Django 4.1.7 on 2023-02-16 18:11 |
||||
|
|
||||
|
from django.conf import settings |
||||
|
from django.db import migrations, models |
||||
|
import django.db.models.deletion |
||||
|
|
||||
|
|
||||
|
class Migration(migrations.Migration): |
||||
|
|
||||
|
initial = True |
||||
|
|
||||
|
dependencies = [ |
||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL), |
||||
|
] |
||||
|
|
||||
|
operations = [ |
||||
|
migrations.CreateModel( |
||||
|
name="Profile", |
||||
|
fields=[ |
||||
|
( |
||||
|
"id", |
||||
|
models.BigAutoField( |
||||
|
auto_created=True, |
||||
|
primary_key=True, |
||||
|
serialize=False, |
||||
|
verbose_name="ID", |
||||
|
), |
||||
|
), |
||||
|
("first_name", models.CharField(max_length=100)), |
||||
|
( |
||||
|
"user", |
||||
|
models.OneToOneField( |
||||
|
on_delete=django.db.models.deletion.CASCADE, |
||||
|
to=settings.AUTH_USER_MODEL, |
||||
|
), |
||||
|
), |
||||
|
], |
||||
|
), |
||||
|
] |
||||
@ -0,0 +1,28 @@ |
|||||
|
# Generated by Django 4.1.7 on 2023-02-16 18:20 |
||||
|
|
||||
|
from django.db import migrations, models |
||||
|
|
||||
|
|
||||
|
class Migration(migrations.Migration): |
||||
|
|
||||
|
dependencies = [ |
||||
|
("base", "0001_initial"), |
||||
|
] |
||||
|
|
||||
|
operations = [ |
||||
|
migrations.AddField( |
||||
|
model_name="profile", |
||||
|
name="email", |
||||
|
field=models.EmailField(max_length=254, null=True), |
||||
|
), |
||||
|
migrations.AddField( |
||||
|
model_name="profile", |
||||
|
name="last_name", |
||||
|
field=models.CharField(max_length=100, null=True), |
||||
|
), |
||||
|
migrations.AlterField( |
||||
|
model_name="profile", |
||||
|
name="first_name", |
||||
|
field=models.CharField(max_length=100, null=True), |
||||
|
), |
||||
|
] |
||||
@ -0,0 +1,38 @@ |
|||||
|
# Generated by Django 5.0.2 on 2024-02-08 21:41 |
||||
|
|
||||
|
import django.db.models.deletion |
||||
|
import django.utils.timezone |
||||
|
from django.conf import settings |
||||
|
from django.db import migrations, models |
||||
|
|
||||
|
|
||||
|
class Migration(migrations.Migration): |
||||
|
|
||||
|
dependencies = [ |
||||
|
('base', '0002_profile_email_profile_last_name_and_more'), |
||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL), |
||||
|
] |
||||
|
|
||||
|
operations = [ |
||||
|
migrations.AlterField( |
||||
|
model_name='profile', |
||||
|
name='email', |
||||
|
field=models.EmailField(default=django.utils.timezone.now, max_length=254), |
||||
|
preserve_default=False, |
||||
|
), |
||||
|
migrations.AlterField( |
||||
|
model_name='profile', |
||||
|
name='first_name', |
||||
|
field=models.CharField(max_length=100), |
||||
|
), |
||||
|
migrations.AlterField( |
||||
|
model_name='profile', |
||||
|
name='last_name', |
||||
|
field=models.CharField(max_length=100), |
||||
|
), |
||||
|
migrations.AlterField( |
||||
|
model_name='profile', |
||||
|
name='user', |
||||
|
field=models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, related_name='profile', to=settings.AUTH_USER_MODEL), |
||||
|
), |
||||
|
] |
||||
@ -0,0 +1,11 @@ |
|||||
|
from django.db import models |
||||
|
from django.contrib.auth.models import User |
||||
|
|
||||
|
class Profile(models.Model): |
||||
|
user = models.OneToOneField(User, on_delete=models.CASCADE, related_name='profile') |
||||
|
first_name = models.CharField(max_length=100) |
||||
|
last_name = models.CharField(max_length=100) |
||||
|
email = models.EmailField() |
||||
|
|
||||
|
def __str__(self): |
||||
|
return self.user.username |
||||
@ -0,0 +1,13 @@ |
|||||
|
from rest_framework import serializers |
||||
|
from base.models import * |
||||
|
|
||||
|
class UserSerializer(serializers.ModelSerializer): |
||||
|
class Meta: |
||||
|
model = User |
||||
|
fields = '__all__' |
||||
|
|
||||
|
class ProfileSerializer(serializers.ModelSerializer): |
||||
|
user = UserSerializer(many=False, read_only=True) |
||||
|
class Meta: |
||||
|
model = Profile |
||||
|
fields = ('user', 'first_name', 'last_name', 'email') |
||||
@ -0,0 +1,15 @@ |
|||||
|
from django.db.models.signals import post_save #Import a post_save signal when a user is created |
||||
|
from django.contrib.auth.models import User # Import the built-in User model, which is a sender |
||||
|
from django.dispatch import receiver # Import the receiver |
||||
|
from .models import Profile |
||||
|
|
||||
|
|
||||
|
@receiver(post_save, sender=User) |
||||
|
def create_profile(sender, instance, created, **kwargs): |
||||
|
if created: |
||||
|
Profile.objects.create(user=instance) |
||||
|
|
||||
|
|
||||
|
@receiver(post_save, sender=User) |
||||
|
def save_profile(sender, instance, **kwargs): |
||||
|
instance.profile.save() |
||||
@ -0,0 +1,3 @@ |
|||||
|
from django.test import TestCase |
||||
|
|
||||
|
# Create your tests here. |
||||
@ -0,0 +1,3 @@ |
|||||
|
from django.shortcuts import render |
||||
|
|
||||
|
# Create your views here. |
||||
@ -0,0 +1,22 @@ |
|||||
|
#!/usr/bin/env python |
||||
|
"""Django's command-line utility for administrative tasks.""" |
||||
|
import os |
||||
|
import sys |
||||
|
|
||||
|
|
||||
|
def main(): |
||||
|
"""Run administrative tasks.""" |
||||
|
os.environ.setdefault("DJANGO_SETTINGS_MODULE", "backend.settings") |
||||
|
try: |
||||
|
from django.core.management import execute_from_command_line |
||||
|
except ImportError as exc: |
||||
|
raise ImportError( |
||||
|
"Couldn't import Django. Are you sure it's installed and " |
||||
|
"available on your PYTHONPATH environment variable? Did you " |
||||
|
"forget to activate a virtual environment?" |
||||
|
) from exc |
||||
|
execute_from_command_line(sys.argv) |
||||
|
|
||||
|
|
||||
|
if __name__ == "__main__": |
||||
|
main() |
||||
@ -0,0 +1,9 @@ |
|||||
|
asgiref==3.7.2 |
||||
|
Django==5.0.2 |
||||
|
django-cors-headers==4.3.1 |
||||
|
djangorestframework==3.14.0 |
||||
|
djangorestframework-simplejwt==5.3.1 |
||||
|
PyJWT==2.8.0 |
||||
|
pytz==2024.1 |
||||
|
sqlparse==0.4.4 |
||||
|
typing_extensions==4.9.0 |
||||
Loading…
Reference in new issue