Compare commits
10 Commits
add_swager
...
tests
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9152cc9772 | ||
| fdd40d4592 | |||
| 0ee81d4b42 | |||
|
|
bba4be6b27 | ||
|
|
4ab5f11dd2 | ||
|
|
e3b93f3524 | ||
|
|
a6a31c8c10 | ||
|
|
889acf6c45 | ||
|
|
7e02dff184 | ||
|
|
61bb90540e |
1
.gitignore
vendored
1
.gitignore
vendored
@@ -65,6 +65,5 @@ target/
|
|||||||
*.jsl
|
*.jsl
|
||||||
*.db
|
*.db
|
||||||
tmp*
|
tmp*
|
||||||
test_*
|
|
||||||
.env*
|
.env*
|
||||||
venv*
|
venv*
|
||||||
|
|||||||
@@ -53,3 +53,8 @@ docker-compose up
|
|||||||
```bash
|
```bash
|
||||||
docker-compose exec web python manage.py makemigrations backend
|
docker-compose exec web python manage.py makemigrations backend
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Для запуска тестов использовать
|
||||||
|
```bash
|
||||||
|
docker-compose run -e DJANGO_SETTINGS_MODULE=backend.settings web pytest --cov=backend
|
||||||
|
```
|
||||||
|
|||||||
3
backend/.coveragerc
Normal file
3
backend/.coveragerc
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
[run]
|
||||||
|
omit =
|
||||||
|
backend/migrations/*
|
||||||
@@ -28,7 +28,7 @@ DEBUG = True
|
|||||||
|
|
||||||
ALLOWED_HOSTS = []
|
ALLOWED_HOSTS = []
|
||||||
if DEBUG:
|
if DEBUG:
|
||||||
ALLOWED_HOSTS = ["0.0.0.0", "localhost", "127.0.0.1"]
|
ALLOWED_HOSTS = ["0.0.0.0", "localhost", "127.0.0.1", "ALLOWED_HOSTS", "testserver"]
|
||||||
|
|
||||||
# Application definition
|
# Application definition
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ from .api import router
|
|||||||
schema_view = get_schema_view(
|
schema_view = get_schema_view(
|
||||||
openapi.Info(
|
openapi.Info(
|
||||||
title="ToDo List",
|
title="ToDo List",
|
||||||
default_version='v1',
|
default_version="v1",
|
||||||
description="Swagger Interface for ToDo List",
|
description="Swagger Interface for ToDo List",
|
||||||
),
|
),
|
||||||
public=True,
|
public=True,
|
||||||
@@ -28,5 +28,5 @@ urlpatterns = [
|
|||||||
path("api/token/refresh/", TokenRefreshView.as_view(), name="token_refresh"),
|
path("api/token/refresh/", TokenRefreshView.as_view(), name="token_refresh"),
|
||||||
path("api/", include(router.urls)),
|
path("api/", include(router.urls)),
|
||||||
path("api-auth/", include("rest_framework.urls", namespace="rest_framework")),
|
path("api-auth/", include("rest_framework.urls", namespace="rest_framework")),
|
||||||
path('swagger/', schema_view.with_ui('swagger', cache_timeout=0), name='schema-swagger-ui'),
|
path("swagger/", schema_view.with_ui("swagger", cache_timeout=0), name="schema-swagger-ui"),
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
pytest==6.2.3
|
||||||
|
pytest-cov==2.11.1
|
||||||
djangorestframework==3.12.4
|
djangorestframework==3.12.4
|
||||||
django-filter==2.4.0
|
django-filter==2.4.0
|
||||||
markdown==3.3.4
|
markdown==3.3.4
|
||||||
|
|||||||
3
backend/tests/__init__.py
Normal file
3
backend/tests/__init__.py
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
from django.core.wsgi import get_wsgi_application
|
||||||
|
|
||||||
|
application = get_wsgi_application()
|
||||||
154
backend/tests/test_item.py
Normal file
154
backend/tests/test_item.py
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
from django.urls import reverse
|
||||||
|
from rest_framework import status
|
||||||
|
from rest_framework.test import APITestCase
|
||||||
|
from django.contrib.auth.models import User
|
||||||
|
from .test_todo import create_todo
|
||||||
|
|
||||||
|
|
||||||
|
class ItemTest(APITestCase):
|
||||||
|
"""Tests API for items."""
|
||||||
|
|
||||||
|
def prepare(self):
|
||||||
|
user = User.objects.create_user("test_user4", "test@test.com", "test_password")
|
||||||
|
self.client.force_authenticate(user=user)
|
||||||
|
to_do_id_1 = create_todo(self.client, "ToDoList1").data["id"]
|
||||||
|
to_do_id_2 = create_todo(self.client, "ToDoList2").data["id"]
|
||||||
|
return to_do_id_1, to_do_id_2
|
||||||
|
|
||||||
|
def get(self, expected_titles, todo_id=None, finished=None):
|
||||||
|
url = reverse("ToDoItems-list")
|
||||||
|
data = {}
|
||||||
|
if finished is not None:
|
||||||
|
data["finished"] = finished
|
||||||
|
if todo_id is not None:
|
||||||
|
data["parent"] = todo_id
|
||||||
|
|
||||||
|
response = self.client.get(url, data, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
real_titles = [(d["text"], d["parent"]) for d in response.data["results"]]
|
||||||
|
self.assertEqual(real_titles, expected_titles)
|
||||||
|
|
||||||
|
if finished is not None:
|
||||||
|
item_status = [data["finished"] for data in response.data["results"]]
|
||||||
|
self.assertEqual(finished, all(item_status))
|
||||||
|
|
||||||
|
def post(self, item_text, todo_id, finished=None):
|
||||||
|
url = reverse("ToDoItems-list")
|
||||||
|
if finished is not None:
|
||||||
|
data = {"text": item_text, "parent": todo_id, "finished": finished}
|
||||||
|
else:
|
||||||
|
data = {"text": item_text, "parent": todo_id}
|
||||||
|
response = self.client.post(url, data, format="json")
|
||||||
|
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
||||||
|
check_finished = False if (finished is None) else finished
|
||||||
|
self.assertEqual(response.data["text"], item_text)
|
||||||
|
self.assertEqual(response.data["parent"], todo_id)
|
||||||
|
self.assertEqual(response.data["finished"], check_finished)
|
||||||
|
|
||||||
|
return response.data["id"], response.data["finished"]
|
||||||
|
|
||||||
|
def get_by_id(self, id, text, finished, parent):
|
||||||
|
url_with_id = reverse("ToDoItems-detail", args=(id,))
|
||||||
|
response = self.client.get(url_with_id, {id: id}, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
self.assertEqual(response.data["text"], text)
|
||||||
|
self.assertEqual(response.data["finished"], finished)
|
||||||
|
self.assertEqual(response.data["parent"], parent)
|
||||||
|
|
||||||
|
def put(self, id, text, parent, finished=None):
|
||||||
|
url_with_id = reverse("ToDoItems-detail", args=(id,))
|
||||||
|
data = {"text": text, "parent": parent}
|
||||||
|
if finished is not None:
|
||||||
|
data["finished"] = finished
|
||||||
|
response = self.client.put(url_with_id, data, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
self.assertEqual(response.data["text"], text)
|
||||||
|
self.assertEqual(response.data["parent"], parent)
|
||||||
|
if finished is not None:
|
||||||
|
self.assertEqual(response.data["finished"], finished)
|
||||||
|
|
||||||
|
def patch(self, id, text=None, finished=None, parent=None):
|
||||||
|
url_with_id = reverse("ToDoItems-detail", args=(id,))
|
||||||
|
data = {}
|
||||||
|
if text is not None:
|
||||||
|
data["text"] = text
|
||||||
|
if finished is not None:
|
||||||
|
data["finished"] = finished
|
||||||
|
if parent is not None:
|
||||||
|
data["parent"] = parent
|
||||||
|
response = self.client.patch(url_with_id, data, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
if text is not None:
|
||||||
|
self.assertEqual(response.data["text"], text)
|
||||||
|
if finished is not None:
|
||||||
|
self.assertEqual(response.data["finished"], finished)
|
||||||
|
if parent is not None:
|
||||||
|
self.assertEqual(response.data["parent"], parent)
|
||||||
|
|
||||||
|
def delete(self, id, title, finished, to_do_id):
|
||||||
|
self.get_by_id(id, title, finished, to_do_id)
|
||||||
|
url_with_id = reverse("ToDoItems-detail", args=(id,))
|
||||||
|
response = self.client.delete(url_with_id, {}, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_204_NO_CONTENT)
|
||||||
|
|
||||||
|
def test_create_delete(self):
|
||||||
|
"""
|
||||||
|
/todo_items/: get, post (create)
|
||||||
|
/todo_items/{id}/: get (read), delete
|
||||||
|
"""
|
||||||
|
to_do_id_1, to_do_id_2 = self.prepare()
|
||||||
|
self.get([], to_do_id_1)
|
||||||
|
item_text_1, item_text_2, item_text_3, item_text_4 = "Item1", "Item2", "Item3", "Item4"
|
||||||
|
item_id_1, item_finished_1 = self.post(item_text_1, to_do_id_1)
|
||||||
|
self.get([(item_text_1, to_do_id_1)], to_do_id_1)
|
||||||
|
item_id_2, item_finished_2 = self.post(item_text_2, to_do_id_1, finished=False)
|
||||||
|
self.get([(item_text_1, to_do_id_1), (item_text_2, to_do_id_1)], to_do_id_1)
|
||||||
|
item_id_3, item_finished_3 = self.post(item_text_3, to_do_id_1, finished=True)
|
||||||
|
self.get(
|
||||||
|
[(item_text_1, to_do_id_1), (item_text_2, to_do_id_1), (item_text_3, to_do_id_1)],
|
||||||
|
to_do_id_1,
|
||||||
|
)
|
||||||
|
item_id_4, item_finished_4 = self.post(item_text_4, to_do_id_2, finished=False)
|
||||||
|
self.get(
|
||||||
|
[
|
||||||
|
(item_text_1, to_do_id_1),
|
||||||
|
(item_text_2, to_do_id_1),
|
||||||
|
(item_text_3, to_do_id_1),
|
||||||
|
(item_text_4, to_do_id_2),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
self.get(
|
||||||
|
[(item_text_1, to_do_id_1), (item_text_2, to_do_id_1), (item_text_3, to_do_id_1)],
|
||||||
|
to_do_id_1,
|
||||||
|
)
|
||||||
|
self.get([(item_text_1, to_do_id_1), (item_text_2, to_do_id_1)], to_do_id_1, finished=False)
|
||||||
|
self.get([(item_text_3, to_do_id_1)], to_do_id_1, finished=True)
|
||||||
|
|
||||||
|
self.get_by_id(item_id_1, item_text_1, item_finished_1, to_do_id_1)
|
||||||
|
self.get_by_id(item_id_2, item_text_2, item_finished_2, to_do_id_1)
|
||||||
|
self.get_by_id(item_id_3, item_text_3, item_finished_3, to_do_id_1)
|
||||||
|
|
||||||
|
self.delete(item_id_3, item_text_3, item_finished_3, to_do_id_1)
|
||||||
|
self.get([(item_text_1, to_do_id_1), (item_text_2, to_do_id_1)], to_do_id_1)
|
||||||
|
|
||||||
|
def test_update(self):
|
||||||
|
"""
|
||||||
|
/todo_items/{id}/: put (update), patch (partial_update)
|
||||||
|
"""
|
||||||
|
|
||||||
|
to_do_id_1, to_do_id_2 = self.prepare()
|
||||||
|
|
||||||
|
item_text_1 = "Item1"
|
||||||
|
item_id_1, item_finished_1 = self.post(item_text_1, to_do_id_1)
|
||||||
|
|
||||||
|
item_text_1_2 = "Item5"
|
||||||
|
self.put(item_id_1, item_text_1_2, to_do_id_2)
|
||||||
|
self.put(item_id_1, item_text_1_2, to_do_id_2, finished=False)
|
||||||
|
self.put(item_id_1, item_text_1_2, to_do_id_2, finished=True)
|
||||||
|
|
||||||
|
item_text_1_3 = "Item6"
|
||||||
|
self.patch(item_id_1, parent=to_do_id_1)
|
||||||
|
self.patch(item_id_1, finished=True)
|
||||||
|
self.patch(item_id_1, text=item_text_1_3)
|
||||||
85
backend/tests/test_todo.py
Normal file
85
backend/tests/test_todo.py
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
from django.urls import reverse
|
||||||
|
from rest_framework import status
|
||||||
|
from rest_framework.test import APITestCase
|
||||||
|
from django.contrib.auth.models import User
|
||||||
|
|
||||||
|
|
||||||
|
def create_todo(client, title):
|
||||||
|
url = reverse("ToDoLists-list")
|
||||||
|
response = client.post(url, {"title": title}, format="json")
|
||||||
|
return response
|
||||||
|
|
||||||
|
|
||||||
|
class ToDoTest(APITestCase):
|
||||||
|
"""Tests API for todo."""
|
||||||
|
|
||||||
|
def get(self, expected_titles):
|
||||||
|
url = reverse("ToDoLists-list")
|
||||||
|
response = self.client.get(url, {}, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
real_titles = [data["title"] for data in response.data["results"]]
|
||||||
|
self.assertEqual(real_titles, expected_titles)
|
||||||
|
|
||||||
|
def post(self, to_do_title):
|
||||||
|
response = create_todo(self.client, to_do_title)
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_201_CREATED)
|
||||||
|
self.assertEqual(response.data["title"], to_do_title)
|
||||||
|
return response.data["id"]
|
||||||
|
|
||||||
|
def get_by_id(self, id, expected_title):
|
||||||
|
url_with_id = reverse("ToDoLists-detail", args=(id,))
|
||||||
|
response = self.client.get(url_with_id, {id: id}, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
self.assertEqual(response.data["title"], expected_title)
|
||||||
|
|
||||||
|
def put(self, id, new_title):
|
||||||
|
url_with_id = reverse("ToDoLists-detail", args=(id,))
|
||||||
|
response = self.client.put(url_with_id, {"title": new_title}, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
self.assertEqual(response.data["title"], new_title)
|
||||||
|
|
||||||
|
def patch(self, id, new_title):
|
||||||
|
url_with_id = reverse("ToDoLists-detail", args=(id,))
|
||||||
|
response = self.client.patch(url_with_id, {"title": new_title}, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_200_OK)
|
||||||
|
self.assertEqual(response.data["title"], new_title)
|
||||||
|
|
||||||
|
def delete(self, id, title):
|
||||||
|
self.get_by_id(id, title)
|
||||||
|
url_with_id = reverse("ToDoLists-detail", args=(id,))
|
||||||
|
response = self.client.delete(url_with_id, {}, format="json")
|
||||||
|
self.assertEqual(response.status_code, status.HTTP_204_NO_CONTENT)
|
||||||
|
|
||||||
|
def prepare(self):
|
||||||
|
user = User.objects.create_user("test_user", "test@test.com", "test_password")
|
||||||
|
self.client.force_authenticate(user=user)
|
||||||
|
|
||||||
|
def test_create_delete(self):
|
||||||
|
"""
|
||||||
|
lists/{id}/: put (update), patch (partial_update)
|
||||||
|
"""
|
||||||
|
self.prepare()
|
||||||
|
to_do_title_1 = "ToDoList1"
|
||||||
|
to_do_id1 = self.post(to_do_title_1)
|
||||||
|
self.put(to_do_id1, "ToDoList11")
|
||||||
|
self.patch(to_do_id1, "ToDoList12")
|
||||||
|
|
||||||
|
def test_todo(self):
|
||||||
|
"""
|
||||||
|
lists/: get, post
|
||||||
|
lists/{id}/: get, delete
|
||||||
|
"""
|
||||||
|
|
||||||
|
self.prepare()
|
||||||
|
self.get([])
|
||||||
|
to_do_title_1, to_do_title_2 = "ToDoList1", "ToDoList2"
|
||||||
|
to_do_id1 = self.post(to_do_title_1)
|
||||||
|
self.get([to_do_title_1])
|
||||||
|
to_do_id2 = self.post(to_do_title_2)
|
||||||
|
self.get([to_do_title_1, to_do_title_2])
|
||||||
|
|
||||||
|
self.get_by_id(to_do_id1, to_do_title_1)
|
||||||
|
self.get_by_id(to_do_id2, to_do_title_2)
|
||||||
|
|
||||||
|
self.delete(to_do_id2, to_do_title_2)
|
||||||
|
self.get([to_do_title_1])
|
||||||
@@ -1,113 +1,14 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
import sys
|
|
||||||
import tkinter as tk
|
import tkinter as tk
|
||||||
from tkinter import messagebox as mb
|
|
||||||
from user import User
|
|
||||||
|
|
||||||
if 'win' in sys.platform.lower():
|
|
||||||
DEFAULT_URL = "http://localhost:8000"
|
|
||||||
else:
|
|
||||||
DEFAULT_URL = "http://0.0.0.0:8000"
|
|
||||||
|
|
||||||
BASE_W = 580
|
|
||||||
BASE_H = 400
|
|
||||||
|
|
||||||
|
|
||||||
class Application(tk.Frame):
|
class Application(tk.Frame):
|
||||||
def __init__(self,
|
def __init__(self, master=None):
|
||||||
master=None
|
|
||||||
) -> None:
|
|
||||||
"""
|
|
||||||
Функция инициаизации класса
|
|
||||||
"""
|
|
||||||
super().__init__(master)
|
super().__init__(master)
|
||||||
|
|
||||||
#Иницализируем параметры окна
|
|
||||||
self.master = master
|
|
||||||
self.pack(fill=tk.BOTH, expand=1)
|
|
||||||
|
|
||||||
self.grid(row=0, column=0, sticky=tk.N + tk.S + tk.E + tk.W)
|
|
||||||
tk.Grid.rowconfigure(master, 0, weight=1)
|
|
||||||
tk.Grid.columnconfigure(master, 0, weight=1)
|
|
||||||
|
|
||||||
# Иницализируем параметры пользователя
|
|
||||||
self.user = User(url=DEFAULT_URL)
|
|
||||||
|
|
||||||
# Настраиваем размеры и включаем иницализацию
|
|
||||||
self.centerWindow()
|
|
||||||
self.initAUTH()
|
|
||||||
|
|
||||||
def centerWindow(self,
|
|
||||||
width: str = BASE_W,
|
|
||||||
heigh: str = BASE_H
|
|
||||||
) -> None:
|
|
||||||
"""
|
|
||||||
Центрирует приложение по центру экрана
|
|
||||||
|
|
||||||
:param width: ширина окна
|
|
||||||
:param heigh: высота окна
|
|
||||||
"""
|
|
||||||
sw = self.master.winfo_screenwidth()
|
|
||||||
sh = self.master.winfo_screenheight()
|
|
||||||
|
|
||||||
x = (sw - width) / 2
|
|
||||||
y = (sh - heigh) / 2
|
|
||||||
self.master.geometry('%dx%d+%d+%d' % (width, heigh, x, y))
|
|
||||||
|
|
||||||
def login_clicked(self) -> None:
|
|
||||||
"""
|
|
||||||
Функция авторизации
|
|
||||||
"""
|
|
||||||
try:
|
|
||||||
self.user.auth(self.login.get(), self.password.get())
|
|
||||||
except Exception as ex:
|
|
||||||
print(ex)
|
|
||||||
self.show_info()
|
|
||||||
|
|
||||||
def show_info(selfб,
|
|
||||||
msg: str = None
|
|
||||||
) -> None:
|
|
||||||
"""
|
|
||||||
Показывает передаваемое сообщение в messagebox
|
|
||||||
|
|
||||||
:param msg: передаваемое сообщение
|
|
||||||
"""
|
|
||||||
if msg is None:
|
|
||||||
msg = "Неправильный логин или пароль"
|
|
||||||
mb.showinfo("Информация", msg)
|
|
||||||
|
|
||||||
def initAUTH(self) -> None:
|
|
||||||
"""
|
|
||||||
Создает окно авторизации программы
|
|
||||||
"""
|
|
||||||
#Конфигурируем сетку
|
|
||||||
for rows in range(25):
|
|
||||||
tk.Grid.rowconfigure(self, rows, weight=1)
|
|
||||||
|
|
||||||
for columns in range(25):
|
|
||||||
tk.Grid.columnconfigure(self, columns, weight=1)
|
|
||||||
|
|
||||||
# Подпись и поле ввода для логина
|
|
||||||
login_label = tk.Label(self, text="Введите логин")
|
|
||||||
login_label.grid(row=9, column=12, columnspan=3, rowspan=1, sticky="nsew")
|
|
||||||
|
|
||||||
self.login = tk.Entry(self)
|
|
||||||
self.login.grid(row=10, column=12, columnspan=3, rowspan=1, sticky="nsew")
|
|
||||||
|
|
||||||
# Подпись и поле ввода для пароля
|
|
||||||
password_label = tk.Label(self, text="Введите пароль")
|
|
||||||
password_label.grid(row=11, column=12, columnspan=3, rowspan=1, sticky="nsew")
|
|
||||||
|
|
||||||
self.password = tk.Entry(self, show="*")
|
|
||||||
self.password.grid(row=12, column=12, columnspan=3, rowspan=1, sticky="nsew")
|
|
||||||
|
|
||||||
# Кнопка авториазции
|
|
||||||
btn = tk.Button(self, text="Войти", command=self.login_clicked)
|
|
||||||
btn.grid(row=14, column=12, columnspan=3, rowspan=1, sticky="nsew")
|
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
master = tk.Tk()
|
app = Application()
|
||||||
app = Application(master)
|
|
||||||
app.master.title("ToDo")
|
app.master.title("ToDo")
|
||||||
app.mainloop()
|
app.mainloop()
|
||||||
|
|||||||
Reference in New Issue
Block a user