diff --git a/.vscode/settings.json b/.vscode/settings.json index 0d04735..2697427 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -1,4 +1,14 @@ { + "python.envFile": "${workspaceFolder}/backend/.env", "prettier.configPath": "./frontend/.prettierrc.js", - "prettier.bracketSpacing": false + "prettier.bracketSpacing": false, + "python.testing.unittestArgs": [ + "-v", + "-s", + "./backend", + "-p", + "test_*.py" + ], + "python.testing.pytestEnabled": false, + "python.testing.unittestEnabled": true } diff --git a/backend/.env b/backend/.env.development similarity index 84% rename from backend/.env rename to backend/.env.development index b917f06..2d63c5b 100644 --- a/backend/.env +++ b/backend/.env.development @@ -1,6 +1,7 @@ DEBUG=True DEVELOPMENT_MODE=True DJANGO_ALLOWED_HOSTS='localhost,0.0.0.0,127.0.0.1,192.168.1.220' +DJANGO_SETTINGS_MODULE=backend.settings DB_NAME = crumpet_db DB_USERNAME = head_baker diff --git a/backend/.env.production b/backend/.env.production new file mode 100644 index 0000000..69d4e37 --- /dev/null +++ b/backend/.env.production @@ -0,0 +1,11 @@ +DEBUG=False +DEVELOPMENT_MODE=False +DJANGO_ALLOWED_HOSTS='localhost,0.0.0.0,127.0.0.1,192.168.1.220' +DJANGO_SETTINGS_MODULE=backend.settings + +DB_NAME = crumpet_db +DB_USERNAME = head_baker +DB_PASSWORD = Crumpet2023 +DB_HOST = db +DB_PORT = 5432 +DB_SSL_MODE = require diff --git a/backend/app/__init__.py b/backend/app/__init__.py index e69de29..583cbcb 100644 --- a/backend/app/__init__.py +++ b/backend/app/__init__.py @@ -0,0 +1 @@ +default_app_config = 'app.apps.AppConfig' \ No newline at end of file diff --git a/backend/app/apps.py b/backend/app/apps.py index ed327d2..9557717 100644 --- a/backend/app/apps.py +++ b/backend/app/apps.py @@ -4,3 +4,6 @@ class AppConfig(AppConfig): default_auto_field = 'django.db.models.BigAutoField' name = 'app' + + def ready(self): + import app.signals diff --git a/backend/app/migrations/0007_environment.py b/backend/app/migrations/0007_environment.py new file mode 100644 index 0000000..d98cc66 --- /dev/null +++ b/backend/app/migrations/0007_environment.py @@ -0,0 +1,42 @@ +# Generated by Django 4.1.5 on 2023-10-29 21:24 + +from django.db import migrations, models +import django.db.models.deletion +import uuid + + +class Migration(migrations.Migration): + dependencies = [ + ("app", "0006_alter_project_api_key"), + ] + + operations = [ + migrations.CreateModel( + name="Environment", + fields=[ + ( + "id", + models.BigAutoField( + auto_created=True, + primary_key=True, + serialize=False, + verbose_name="ID", + ), + ), + ("name", models.CharField(max_length=100)), + ("identifier", models.CharField(default=uuid.uuid4, max_length=255)), + ("is_default", models.BooleanField(default=False)), + ( + "project", + models.ForeignKey( + on_delete=django.db.models.deletion.CASCADE, + related_name="environments", + to="app.project", + ), + ), + ], + options={ + "unique_together": {("identifier", "project")}, + }, + ), + ] diff --git a/backend/app/models/__init__.py b/backend/app/models/__init__.py index 4705c4f..8604388 100644 --- a/backend/app/models/__init__.py +++ b/backend/app/models/__init__.py @@ -13,3 +13,4 @@ from .user import User from .project import Project, ProjectMembership from .selector import Selector +from .environment import Environment diff --git a/backend/app/models/environment.py b/backend/app/models/environment.py new file mode 100644 index 0000000..5643f40 --- /dev/null +++ b/backend/app/models/environment.py @@ -0,0 +1,24 @@ +from django.db import models + +import uuid + + +class Environment(models.Model): + name = models.CharField(max_length=100, blank=False, null=False) + identifier = models.CharField(max_length=255, default=uuid.uuid4, blank=False, null=False) + project = models.ForeignKey( + "Project", on_delete=models.CASCADE, related_name="environments" + ) + is_default = models.BooleanField(default=False, blank=False, null=False) + + class Meta: + unique_together = ["identifier", "project"] + + def save(self, *args, **kwargs): + if not self.identifier: + # If identifier is not provided, assign a GUID. + self.identifier = str(uuid.uuid4()) + super(Environment, self).save(*args, **kwargs) + + def __str__(self): + return self.name diff --git a/backend/app/serializers/environment_serializer.py b/backend/app/serializers/environment_serializer.py new file mode 100644 index 0000000..e094ce5 --- /dev/null +++ b/backend/app/serializers/environment_serializer.py @@ -0,0 +1,9 @@ +from app.models import Environment +from rest_framework import serializers + + +class EnvironmentSerializer(serializers.ModelSerializer): + class Meta: + model = Environment + fields = ["id", "name", "identifier", "is_default"] + extra_kwargs = {'id': {'required': True}, 'name': {'required': True}, 'identifier': {'required': True}, 'is_default': {'required': True} } diff --git a/backend/app/serializers/project_serializer.py b/backend/app/serializers/project_serializer.py index f8c2348..a3fa6a0 100644 --- a/backend/app/serializers/project_serializer.py +++ b/backend/app/serializers/project_serializer.py @@ -1,9 +1,11 @@ -from app.models import Project +from app.models import Project, Environment from app.models.project import ProjectMembership from .user_serializer import UserSummarySerializer +from .environment_serializer import EnvironmentSerializer from rest_framework import serializers + class ProjectMembershipSerializer(serializers.ModelSerializer): """Used as a nested serializer by ApplicationSerializer.""" @@ -21,10 +23,11 @@ class ProjectSerializer(serializers.ModelSerializer): source="projectmembership_set.all", # default related name for ProjectMembership. read_only=True, ) + environments = EnvironmentSerializer(many=True, read_only=True) class Meta: model = Project - fields = ["id", "name", "api_key", "members"] + fields = ["id", "name", "api_key", "members", "environments"] def to_representation(self, instance): """ @@ -37,5 +40,8 @@ def to_representation(self, instance): # Serialize the `members` using `ProjectMembershipSerializer`. membership_qs = ProjectMembership.objects.filter(project=instance) representation["members"] = ProjectMembershipSerializer(membership_qs, many=True).data + # Serialize the 'environments' using 'EnvironmentSerializer' + environment_qs = Environment.objects.filter(project=instance) + representation["environments"] = EnvironmentSerializer(environment_qs, many=True).data return representation diff --git a/backend/app/signals.py b/backend/app/signals.py new file mode 100644 index 0000000..7840bcf --- /dev/null +++ b/backend/app/signals.py @@ -0,0 +1,11 @@ +from django.db.models.signals import post_save +from django.dispatch import receiver +from .models.environment import Environment +from .models.project import Project + + +@receiver(post_save, sender=Project) +def create_default_environments(sender, instance, created, **kwargs): + if created: + Environment.objects.create(name="Development", identifier="development", project=instance, is_default=True) + Environment.objects.create(name="Production", identifier="production", project=instance, is_default=True) \ No newline at end of file diff --git a/backend/app/tests/__init__.py b/backend/app/tests/__init__.py index 9c4d0ae..d813816 100644 --- a/backend/app/tests/__init__.py +++ b/backend/app/tests/__init__.py @@ -1,4 +1,8 @@ +import django +django.setup() + from .test_openapi import * from .test_user_auth import * from .test_views import * from .test_projects import * +from .test_environments import * diff --git a/backend/app/tests/test_environments.py b/backend/app/tests/test_environments.py new file mode 100644 index 0000000..c9c0b1c --- /dev/null +++ b/backend/app/tests/test_environments.py @@ -0,0 +1,68 @@ +import uuid +from django.test import TestCase +from django.contrib.auth import get_user_model +from app.models import Environment, Project, ProjectMembership + +User = get_user_model() + + +class EnvironmentModelTestCase(TestCase): + def setUp(self): + self.project = Project.objects.create(name="Test Project") + self.environment = Environment.objects.create( + name="Staging", project=self.project + ) + + def test_environment_creation(self): + """ + Test if the environment is created with provided name and project. + """ + self.assertEqual(self.environment.name, "Staging") + self.assertEqual(self.environment.project, self.project) + + def test_environment_identifier(self): + """ + Test if the environment gets a UUID identifier if not provided. + """ + self.assertIsNotNone(self.environment.identifier) + self.assertTrue(isinstance(self.environment.identifier, uuid.UUID)) + + def test_unique_together_constraint(self): + """ + Test the unique together constraint for identifier and project. + """ + with self.assertRaises(Exception): + Environment.objects.create( + name="Duplicate", + identifier=self.environment.identifier, + project=self.project, + ) + + +class ProjectEnvironmentCreationTestCase(TestCase): + def setUp(self): + self.project = Project.objects.create(name="Another Test Project") + + def test_default_environments_created(self): + """ + Test if Development and Production environments are created for new projects. + """ + dev_env = Environment.objects.filter( + name="Development", project=self.project + ).first() + prod_env = Environment.objects.filter( + name="Production", project=self.project + ).first() + + self.assertIsNotNone(dev_env, "Development environment was not created.") + self.assertIsNotNone(prod_env, "Production environment was not created.") + + def test_default_environments_identifiers(self): + """ + Test if Development and Production environments have correct identifiers. + """ + dev_env = Environment.objects.get(name="Development", project=self.project) + prod_env = Environment.objects.get(name="Production", project=self.project) + + self.assertEqual(dev_env.identifier, "development") + self.assertEqual(prod_env.identifier, "production") diff --git a/backend/app/views/project_views.py b/backend/app/views/project_views.py index e5bec52..cc89dad 100644 --- a/backend/app/views/project_views.py +++ b/backend/app/views/project_views.py @@ -13,7 +13,7 @@ class ProjectsView(viewsets.ModelViewSet): permission_classes = [IsAuthenticated, ProjectMemberPermission] def get_queryset(self): - return self.queryset.filter(members__pk=self.request.user.pk) + return self.queryset.filter(members__pk=self.request.user.pk).prefetch_related('environments') def get_object(self): # Fetch the object and check if the request user has the necessary permissions. diff --git a/backend/backend/settings.py b/backend/backend/settings.py index da6a6a7..1e94ad4 100644 --- a/backend/backend/settings.py +++ b/backend/backend/settings.py @@ -17,13 +17,17 @@ from dotenv import load_dotenv import os -LOAD_DOTENV = os.getenv("LOAD_DOTENV", "True") == "True" -if LOAD_DOTENV: - load_dotenv() - # Build paths inside the project like this: BASE_DIR / 'subdir'. BASE_DIR = Path(__file__).resolve().parent.parent +ENVIRONMENT = os.getenv('ENVIRONMENT', 'development') # Assume development by default +if ENVIRONMENT == 'production': + dotenv_path = BASE_DIR / '.env.production' +else: + dotenv_path = BASE_DIR / '.env.development' + +if dotenv_path.exists(): + load_dotenv(dotenv_path=dotenv_path) # Quick-start development settings - unsuitable for production # See https://docs.djangoproject.com/en/4.1/howto/deployment/checklist/ @@ -84,20 +88,27 @@ WSGI_APPLICATION = "backend.wsgi.application" - # Database # https://docs.djangoproject.com/en/4.1/ref/settings/#databases +if DEVELOPMENT_MODE: + print("DEVELOPMENT MODE: ACTIVE") + host = "localhost" +else: + print("DEVELOPMENT MODE: INACTIVE") + host = os.environ.get("DB_HOST") + + DATABASES = { - "default": { - "ENGINE": "django.db.backends.postgresql", - "NAME": os.environ.get("DB_NAME"), - "USER": os.environ.get("DB_USERNAME"), - "PASSWORD": os.environ.get("DB_PASSWORD"), - "HOST": os.environ.get("DB_HOST"), - "PORT": os.environ.get("DB_PORT"), - } + "default": { + "ENGINE": "django.db.backends.postgresql", + "NAME": os.environ.get("DB_NAME"), + "USER": os.environ.get("DB_USERNAME"), + "PASSWORD": os.environ.get("DB_PASSWORD"), + "HOST": host, + "PORT": os.environ.get("DB_PORT"), } +} print("Connected to database") @@ -128,7 +139,7 @@ # } # print("Connected to database") -AUTH_USER_MODEL = 'app.User' +AUTH_USER_MODEL = "app.User" # Password validation # https://docs.djangoproject.com/en/4.1/ref/settings/#auth-password-validators @@ -184,7 +195,7 @@ "rest_framework_simplejwt.authentication.JWTAuthentication", "rest_framework.authentication.SessionAuthentication", "rest_framework.authentication.BasicAuthentication", - "app.authentication.ProjectAPIKeyAuthentication" + "app.authentication.ProjectAPIKeyAuthentication", ), "DEFAULT_PERMISSION_CLASSES": [ "rest_framework.permissions.IsAuthenticated", diff --git a/backend/docker-compose.yml b/backend/docker-compose.yml index 45a7a58..a4b2ff9 100644 --- a/backend/docker-compose.yml +++ b/backend/docker-compose.yml @@ -10,8 +10,10 @@ services: ports: - "8000:8000" tty: true + environment: + - ENVIRONMENT=production env_file: - - .env + - .env.production depends_on: - db diff --git a/frontend/openapi-schema.yml b/frontend/openapi-schema.yml index e462adb..40faf87 100644 --- a/frontend/openapi-schema.yml +++ b/frontend/openapi-schema.yml @@ -319,6 +319,27 @@ components: required: - user readOnly: true + environments: + type: array + items: + type: object + properties: + id: + type: integer + readOnly: true + name: + type: string + maxLength: 100 + identifier: + type: string + maxLength: 255 + is_default: + type: boolean + required: + - name + - identifier + - is_default + readOnly: true required: - name TokenObtainPair: diff --git a/frontend/src/api/schema/api.ts b/frontend/src/api/schema/api.ts index e06797f..2901b66 100644 --- a/frontend/src/api/schema/api.ts +++ b/frontend/src/api/schema/api.ts @@ -53,6 +53,43 @@ export interface Project { * @memberof Project */ 'members'?: Array; + /** + * + * @type {Array} + * @memberof Project + */ + 'environments'?: Array; +} +/** + * + * @export + * @interface ProjectEnvironmentsInner + */ +export interface ProjectEnvironmentsInner { + /** + * + * @type {number} + * @memberof ProjectEnvironmentsInner + */ + 'id'?: number; + /** + * + * @type {string} + * @memberof ProjectEnvironmentsInner + */ + 'name': string; + /** + * + * @type {string} + * @memberof ProjectEnvironmentsInner + */ + 'identifier': string; + /** + * + * @type {boolean} + * @memberof ProjectEnvironmentsInner + */ + 'is_default': boolean; } /** * diff --git a/frontend/src/components/picker/index.tsx b/frontend/src/components/picker/index.tsx index 72b78bb..adfbd46 100644 --- a/frontend/src/components/picker/index.tsx +++ b/frontend/src/components/picker/index.tsx @@ -3,7 +3,6 @@ import { Listbox, Transition } from '@headlessui/react'; import { MdUnfoldMore, MdOutlineCheck } from 'react-icons/md'; interface Item { - id: number; name: string; } diff --git a/frontend/src/components/sidebarMenu/index.tsx b/frontend/src/components/sidebarMenu/index.tsx index a1a0d66..71e41d9 100644 --- a/frontend/src/components/sidebarMenu/index.tsx +++ b/frontend/src/components/sidebarMenu/index.tsx @@ -22,11 +22,6 @@ import { useNavigate } from 'react-router'; import { isHasData } from 'api/utils'; import { usePopper } from 'react-popper'; -const environments = [ - { id: 1, name: 'Development' }, - { id: 2, name: 'Production' }, -]; - interface ProjectEntry { id: number; name: string; @@ -87,7 +82,13 @@ const SidebarMenu = ({ projects }: SidebarMenuProps) => {
Crumpet
- + {isHasData(selectedProject) && selectedProject.data.environments != undefined && ( + + )}
{ButtonList.map((button, index) => ( { ref={setPopperElement} style={styles.popper} {...attributes.popper} - className="w-full px-2"> + className="w-full px-2">
{projects.map((project, index) => (