Compare commits
278 Commits
40a68ff47d
...
shaxob
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
1ff23af8bf | ||
|
|
feecb580c1 | ||
|
|
f53125cfdc | ||
| 65ab51e652 | |||
| 2997810fae | |||
|
|
d014f5a2fb | ||
|
|
7d49929772 | ||
|
|
c29546a04b | ||
|
|
b39c080de3 | ||
|
|
7ad385af94 | ||
| 3781ce29e5 | |||
|
|
db7e34c1c2 | ||
|
|
1cb9551e81 | ||
|
|
51b30c2cc4 | ||
|
|
dc4c98bfc9 | ||
|
|
abed9e59b4 | ||
| f238c92a09 | |||
|
|
113f2da120 | ||
|
|
99b265f68f | ||
| c5d60e799c | |||
|
|
7829c9c625 | ||
|
|
7f462674a8 | ||
| f7be3be5d2 | |||
|
|
557f9f821d | ||
|
|
5f70d69896 | ||
|
|
4ea7070a8f | ||
| 8b02f3a3a3 | |||
|
|
f0d93b10ac | ||
|
|
172ddf4da4 | ||
| 435dd56334 | |||
|
|
779c9db924 | ||
| eaaba123b0 | |||
|
|
63c4ad81eb | ||
|
|
d065891ad5 | ||
| 94c4d03925 | |||
|
|
4a958f064b | ||
|
|
d1f0a5a9ae | ||
| 0084d11c62 | |||
|
|
d1340cdd52 | ||
|
|
d7ea1acba6 | ||
| 560cbe8000 | |||
|
|
37d6a93529 | ||
|
|
e1b445d515 | ||
| ef87112c79 | |||
|
|
8c01c1dc2d | ||
|
|
921b54ab7c | ||
|
|
a74c348187 | ||
| 52fab30588 | |||
|
|
0de50ec328 | ||
|
|
e346546d24 | ||
| e97c6c7ab2 | |||
|
|
f7706e77ee | ||
|
|
e351ed5303 | ||
| affd3e1221 | |||
|
|
59ed3d23ac | ||
|
|
3ac6263035 | ||
| 2c6d7dd2f7 | |||
|
|
a6e0fca165 | ||
|
|
b64073e1ad | ||
| e3ffdddc46 | |||
|
|
6f48632e2d | ||
| faea9bdb89 | |||
|
|
e3e7f18d7f | ||
|
|
1dd1a132e4 | ||
| 04e193bae6 | |||
|
|
7134b2c185 | ||
|
|
306aecc956 | ||
| 3ede209e52 | |||
|
|
05857a227a | ||
| 10b25b5228 | |||
|
|
fcbfa94dd4 | ||
| 7e778d3a3e | |||
|
|
81a4287db1 | ||
|
|
e560fdaf2d | ||
|
|
0d96167a7b | ||
|
|
ae65d9d793 | ||
|
|
5249f7e6f7 | ||
|
|
e1b771e166 | ||
|
|
eded642bd7 | ||
| f830235813 | |||
|
|
a62cf3a1ee | ||
|
|
2f471173c3 | ||
| 3838fbaa47 | |||
|
|
b02078e618 | ||
|
|
1f59347d87 | ||
|
|
fe40057d95 | ||
| 550da049b9 | |||
|
|
bdd5aa9ce2 | ||
|
|
bd27205252 | ||
| 34aba90ebd | |||
|
|
260bc9101e | ||
|
|
f46dac515a | ||
|
|
c6fc150162 | ||
| 7272ef3fce | |||
|
|
1db936126d | ||
| 5e1b02064e | |||
|
|
4e242a4358 | ||
| bcb453d52a | |||
| 039ca92031 | |||
|
|
21bb61e51c | ||
|
|
c9d60acfc9 | ||
|
|
127a2073f8 | ||
| 1ad2016790 | |||
|
|
3fd0f9959b | ||
|
|
84cc11fe2a | ||
|
|
deebae384c | ||
|
|
2b5238f3c8 | ||
|
|
988d07f4b6 | ||
|
|
e27a9b7f11 | ||
|
|
c89f2b32af | ||
|
|
82489cf64c | ||
| 88dedd85c7 | |||
|
|
fb275a091a | ||
|
|
af559dadda | ||
|
|
190480b6f7 | ||
| 1a985ffa4b | |||
|
|
3b62c5a7bf | ||
|
|
d2f8d73cdd | ||
|
|
c4b2a80b2e | ||
|
|
07f8d55966 | ||
| b0b4ccfeee | |||
|
|
ccefe9c119 | ||
|
|
6456283f72 | ||
|
|
6eed2d998e | ||
| 2c82691166 | |||
|
|
7a88e39b96 | ||
|
|
7961fd76de | ||
|
|
dc622ce305 | ||
|
|
6e0718c5db | ||
|
|
32d3bea234 | ||
|
|
129827b495 | ||
|
|
50cc555783 | ||
|
|
76563b3ef0 | ||
|
|
207363dc6a | ||
| 2a08ad9662 | |||
|
|
049cd6ff25 | ||
|
|
5cf4b950fb | ||
|
|
3a08c81ff3 | ||
|
|
4fee037467 | ||
|
|
320f490d23 | ||
| d4e6d80c86 | |||
| 1f0e942be8 | |||
|
|
b8021c7728 | ||
|
|
2af67333e2 | ||
| 81c689e7e9 | |||
|
|
09d2e0954c | ||
| 84a5afb2ee | |||
|
|
cc8fc345d9 | ||
| 03e4387e4d | |||
|
|
64993805af | ||
|
|
01711b5927 | ||
|
|
33aa06f80b | ||
|
|
cb2d83b00d | ||
| 46c9621457 | |||
|
|
8d73fa09a7 | ||
| 3367063707 | |||
|
|
ae926914a5 | ||
|
|
229676be5c | ||
| d2f517baf4 | |||
|
|
6df20f35c2 | ||
|
|
fa676bfa96 | ||
| ca9f12ae32 | |||
|
|
56f693abfd | ||
|
|
5049919865 | ||
| ca3c9bfe4a | |||
|
|
8d8744731a | ||
|
|
7ab1e0b1e0 | ||
| f66a35ec80 | |||
|
|
f71dfa50c1 | ||
|
|
f68f34178e | ||
| ab68a6a463 | |||
|
|
d246406384 | ||
| 0a74f71efa | |||
|
|
575364e0ef | ||
|
|
406477fc33 | ||
|
|
2aa73e15a0 | ||
| 372a289447 | |||
|
|
7343f6191d | ||
|
|
5c4d5fbb71 | ||
| 20043db5b3 | |||
|
|
7d6618155f | ||
|
|
11605e6e6c | ||
|
|
8a1a66a05d | ||
|
|
70555fa93a | ||
| 674eafe65b | |||
|
|
47833176e2 | ||
| b03d2e1c5e | |||
|
|
7be8999a39 | ||
|
|
c5624e361d | ||
| c44b08bb28 | |||
|
|
d448324f89 | ||
|
|
0508a49c1d | ||
| 8b97ca53bb | |||
|
|
2d64777041 | ||
|
|
9f7b29fa13 | ||
| b4a3243e9c | |||
|
|
63359e69f7 | ||
|
|
3c9438aff6 | ||
| a3c67c5bfb | |||
|
|
e768c73d54 | ||
|
|
a2684ff749 | ||
| 91bbb6b6e5 | |||
|
|
01a70debaf | ||
|
|
ae9f10aa4d | ||
| f2dc2e1d52 | |||
|
|
7081a03a98 | ||
|
|
10646e2ebf | ||
| 93f6e79f58 | |||
|
|
3b95734ea4 | ||
| d1aad04ab8 | |||
|
|
adf3c9fff9 | ||
|
|
7e4a90fe2a | ||
| 503c16bdb8 | |||
| 701a375e3c | |||
|
|
e64d7d037f | ||
| b7ecf0e47c | |||
|
|
c7c76e9c79 | ||
|
|
afdc6aa0ac | ||
| f991e3e4ec | |||
|
|
0f142be77b | ||
|
|
182b2483bf | ||
| 469659baa8 | |||
|
|
965328edd4 | ||
|
|
11aa88a2b0 | ||
| 225565bc41 | |||
|
|
75365f8e7e | ||
|
|
3e9675a4be | ||
| c3cf7ac9d5 | |||
|
|
5dcafb9cc1 | ||
|
|
62bc6c17b8 | ||
| 22284f228f | |||
|
|
aa487c1557 | ||
|
|
73d5222a5b | ||
| 707ea1e1fd | |||
|
|
bfbc14db82 | ||
|
|
4c1f1d7104 | ||
| 38e7aac505 | |||
|
|
a973da91f5 | ||
|
|
ec94e4cdd0 | ||
| 87701c31f1 | |||
|
|
35739ddc27 | ||
| e8feafca0b | |||
|
|
505c33a554 | ||
|
|
f24afaf55c | ||
| 665db58bbc | |||
|
|
57e22e04db | ||
|
|
528dc57ea2 | ||
| 774be87f45 | |||
|
|
3d0141197f | ||
| 9dc093e244 | |||
|
|
aa1e4ca6fc | ||
|
|
04ef567d5c | ||
| b604fff1c2 | |||
|
|
277bc2874e | ||
|
|
37459e8485 | ||
| f231d889a5 | |||
|
|
3fa70d9436 | ||
|
|
6b1a9439ce | ||
| d26e2b0147 | |||
|
|
da71ce6d28 | ||
|
|
bc157da00d | ||
| 9c40908cc7 | |||
|
|
7ee2eee437 | ||
|
|
31a08c810d | ||
| 264548c3b1 | |||
|
|
9a18dda657 | ||
|
|
c7612221b8 | ||
| a89dbafb62 | |||
|
|
be0c96b28a | ||
|
|
9d39f0b339 | ||
| 797286bcdb | |||
|
|
091ddb39ed | ||
|
|
12b19290d6 | ||
|
|
36ac4af4cd | ||
| 4784b58abc | |||
|
|
3664f8f66d | ||
|
|
17e63ee2e4 | ||
|
|
8207b750b8 |
@@ -73,6 +73,5 @@ STORAGE_BUCKET_STATIC=name
|
|||||||
STORAGE_PATH=127.0.0.1:8081/bucket/
|
STORAGE_PATH=127.0.0.1:8081/bucket/
|
||||||
STORAGE_PROTOCOL=http:
|
STORAGE_PROTOCOL=http:
|
||||||
|
|
||||||
|
# Didox configs
|
||||||
|
DIDOX_PARTNER_TOKEN=...
|
||||||
# Celery configs
|
|
||||||
|
|||||||
39
.github/workflows/deploy.yaml
vendored
39
.github/workflows/deploy.yaml
vendored
@@ -8,7 +8,6 @@ on:
|
|||||||
env:
|
env:
|
||||||
PROJECT_NAME: sifatbaho
|
PROJECT_NAME: sifatbaho
|
||||||
|
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
contents: write
|
contents: write
|
||||||
|
|
||||||
@@ -48,6 +47,24 @@ jobs:
|
|||||||
- name: Copy env
|
- name: Copy env
|
||||||
run: |
|
run: |
|
||||||
cp .env.example .env
|
cp .env.example .env
|
||||||
|
update_env() {
|
||||||
|
local env_file=".env"
|
||||||
|
for kv in "$@"; do
|
||||||
|
local key="${kv%%=*}"
|
||||||
|
local value="${kv#*=}"
|
||||||
|
if grep -q "^$key=" "$env_file"; then
|
||||||
|
sed -i "s|^$key=.*|$key=$value|" "$env_file"
|
||||||
|
else
|
||||||
|
echo "$key=$value" >> "$env_file"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
}
|
||||||
|
|
||||||
|
update_env \
|
||||||
|
"DB_HOST=postgres" \
|
||||||
|
"DB_NAME=sifatbahodb" \
|
||||||
|
"DB_PORT=5432" \
|
||||||
|
"DIDOX_PARTNER_TOKEN=${{ secrets.DIDOX_TOKEN }}"
|
||||||
|
|
||||||
- name: Set up Docker Buildx
|
- name: Set up Docker Buildx
|
||||||
uses: docker/setup-buildx-action@v3
|
uses: docker/setup-buildx-action@v3
|
||||||
@@ -95,7 +112,6 @@ jobs:
|
|||||||
sed -i "s|image: .*/${{ env.PROJECT_NAME }}:.*|image: ${{ secrets.DOCKER_USERNAME }}/${{ env.PROJECT_NAME }}:${{ github.run_number }}|g" stack.yaml
|
sed -i "s|image: .*/${{ env.PROJECT_NAME }}:.*|image: ${{ secrets.DOCKER_USERNAME }}/${{ env.PROJECT_NAME }}:${{ github.run_number }}|g" stack.yaml
|
||||||
sed -i 's/return HttpResponse("OK.*"/return HttpResponse("OK: #${{ github.sha }}"/' config/urls.py
|
sed -i 's/return HttpResponse("OK.*"/return HttpResponse("OK: #${{ github.sha }}"/' config/urls.py
|
||||||
|
|
||||||
|
|
||||||
- name: Commit and push updated version
|
- name: Commit and push updated version
|
||||||
run: |
|
run: |
|
||||||
git config user.name "github-actions[bot]"
|
git config user.name "github-actions[bot]"
|
||||||
@@ -136,24 +152,5 @@ jobs:
|
|||||||
git reset --hard origin/main
|
git reset --hard origin/main
|
||||||
|
|
||||||
cp .env.example .env
|
cp .env.example .env
|
||||||
|
|
||||||
update_env() {
|
|
||||||
local env_file=".env"
|
|
||||||
for kv in "$@"; do
|
|
||||||
local key="${kv%%=*}"
|
|
||||||
local value="${kv#*=}"
|
|
||||||
if grep -q "^$key=" "$env_file"; then
|
|
||||||
sed -i "s|^$key=.*|$key=$value|" "$env_file"
|
|
||||||
else
|
|
||||||
echo "$key=$value" >> "$env_file"
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
}
|
|
||||||
|
|
||||||
update_env \
|
|
||||||
"DB_HOST=postgres" \
|
|
||||||
"DB_NAME=sifatbahodb" \
|
|
||||||
"DB_PORT=5432"
|
|
||||||
|
|
||||||
export PORT=8085
|
export PORT=8085
|
||||||
docker stack deploy -c stack.yaml ${{ env.PROJECT_NAME }} --with-registry-auth
|
docker stack deploy -c stack.yaml ${{ env.PROJECT_NAME }} --with-registry-auth
|
||||||
|
|||||||
@@ -14,6 +14,7 @@ APPS = [
|
|||||||
"rest_framework_simplejwt",
|
"rest_framework_simplejwt",
|
||||||
"django_core",
|
"django_core",
|
||||||
"core.apps.accounts.apps.AccountsConfig",
|
"core.apps.accounts.apps.AccountsConfig",
|
||||||
|
'core.apps.tasks.apps.TasksConfig',
|
||||||
]
|
]
|
||||||
|
|
||||||
if env.bool("SILK_ENABLED", False):
|
if env.bool("SILK_ENABLED", False):
|
||||||
|
|||||||
@@ -150,4 +150,93 @@ PAGES = [
|
|||||||
},
|
},
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"title": _("Auto baholash"),
|
||||||
|
"separator": True,
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"title": _("Hujjatlar"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:evaluation_documentmodel_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Kategoriyalar"),
|
||||||
|
"icon": "category",
|
||||||
|
"link": reverse_lazy("admin:evaluation_documentcategorymodel_changelist"),
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Joylashuvlar"),
|
||||||
|
"separator": True,
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"title": _("Shaharlar"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:shared_regionmodel_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Tumanlar"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:shared_districtmodel_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Mahallalar"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:shared_villagemodel_changelist"),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Ruxsatlar"),
|
||||||
|
"separator": True,
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"title": _("Ruxsatlar"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:accounts_permission_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Sahifa uchun ruxsatlar"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:accounts_permissiontotab_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Actionlar uchun ruxsatlar"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:accounts_permissiontoaction_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Role"),
|
||||||
|
"icon": "attach_file",
|
||||||
|
"link": reverse_lazy("admin:accounts_role_changelist"),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Task Management"),
|
||||||
|
"separator": True,
|
||||||
|
"items": [
|
||||||
|
{
|
||||||
|
"title": _("Task"),
|
||||||
|
"icon": "task",
|
||||||
|
"link": reverse_lazy("admin:tasks_task_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Column"),
|
||||||
|
"icon": "tag",
|
||||||
|
"link": reverse_lazy("admin:tasks_column_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Comment"),
|
||||||
|
"icon": "message",
|
||||||
|
"link": reverse_lazy("admin:tasks_comment_changelist"),
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"title": _("Label"),
|
||||||
|
"icon": "tag",
|
||||||
|
"link": reverse_lazy("admin:tasks_label_changelist"),
|
||||||
|
},
|
||||||
|
]
|
||||||
|
}
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import environ
|
|||||||
environ.Env.read_env(os.path.join(".env"))
|
environ.Env.read_env(os.path.join(".env"))
|
||||||
|
|
||||||
env = environ.Env(
|
env = environ.Env(
|
||||||
DEBUG=(bool, False),
|
DEBUG=(bool, True),
|
||||||
CACHE_TIME=(int, 180),
|
CACHE_TIME=(int, 180),
|
||||||
OTP_EXPIRE_TIME=(int, 2),
|
OTP_EXPIRE_TIME=(int, 2),
|
||||||
VITE_LIVE=(bool, False),
|
VITE_LIVE=(bool, False),
|
||||||
@@ -26,4 +26,5 @@ env = environ.Env(
|
|||||||
OTP_SERVICE="EskizService",
|
OTP_SERVICE="EskizService",
|
||||||
PROJECT_ENV=(str, "prod"),
|
PROJECT_ENV=(str, "prod"),
|
||||||
SILK_ENABLED=(bool, False),
|
SILK_ENABLED=(bool, False),
|
||||||
|
DIDOX_PARTNER_TOKEN=(str),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -49,6 +49,7 @@ INSTALLED_APPS = [
|
|||||||
"django.contrib.sessions",
|
"django.contrib.sessions",
|
||||||
"django.contrib.messages",
|
"django.contrib.messages",
|
||||||
"django.contrib.staticfiles",
|
"django.contrib.staticfiles",
|
||||||
|
"django.contrib.postgres",
|
||||||
] + APPS
|
] + APPS
|
||||||
|
|
||||||
MODULES = [app for app in MODULES if isinstance(app, str)]
|
MODULES = [app for app in MODULES if isinstance(app, str)]
|
||||||
@@ -165,6 +166,8 @@ SITE_URL = env.str("SITE_URL", default="http://localhost:8055")
|
|||||||
MODELTRANSLATION_LANGUAGES = ("uz", "ru", "en")
|
MODELTRANSLATION_LANGUAGES = ("uz", "ru", "en")
|
||||||
MODELTRANSLATION_DEFAULT_LANGUAGE = "uz"
|
MODELTRANSLATION_DEFAULT_LANGUAGE = "uz"
|
||||||
|
|
||||||
|
DIDOX_PARTNER_TOKEN = env.str("DIDOX_PARTNER_TOKEN")
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
JST_LANGUAGES = [
|
JST_LANGUAGES = [
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ from config.env import env
|
|||||||
|
|
||||||
|
|
||||||
def home(request):
|
def home(request):
|
||||||
return HttpResponse("OK: #ea3f14058e4a7edbb94999d6287f396cd60e7305")
|
return HttpResponse("OK: #65ab51e65224a92a4b6d488d3e8f9b21d3256876")
|
||||||
|
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
@@ -23,6 +23,7 @@ urlpatterns = [
|
|||||||
path("api/v1/", include("core.apps.evaluation.urls")),
|
path("api/v1/", include("core.apps.evaluation.urls")),
|
||||||
path("api/v1/", include("core.apps.payment.urls")),
|
path("api/v1/", include("core.apps.payment.urls")),
|
||||||
path("api/v1/", include("core.apps.chat.urls")),
|
path("api/v1/", include("core.apps.chat.urls")),
|
||||||
|
path("api/v1/tasks/", include("core.apps.tasks.urls")),
|
||||||
]
|
]
|
||||||
urlpatterns += [
|
urlpatterns += [
|
||||||
path("admin/", admin.site.urls),
|
path("admin/", admin.site.urls),
|
||||||
|
|||||||
@@ -1,2 +1,3 @@
|
|||||||
from .core import * # noqa
|
from .core import * # noqa
|
||||||
from .user import * # noqa
|
from .user import * # noqa
|
||||||
|
from .permission import *
|
||||||
|
|||||||
77
core/apps/accounts/admin/permission.py
Normal file
77
core/apps/accounts/admin/permission.py
Normal file
@@ -0,0 +1,77 @@
|
|||||||
|
from django.contrib import admin
|
||||||
|
from core.apps.accounts.models.permission import (
|
||||||
|
PermissionToAction,
|
||||||
|
PermissionToTab,
|
||||||
|
Permission,
|
||||||
|
Role,
|
||||||
|
)
|
||||||
|
|
||||||
|
@admin.register(PermissionToAction)
|
||||||
|
class PermissionToActionAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ("id", "name", "code", "created_at")
|
||||||
|
search_fields = ("name", "code")
|
||||||
|
|
||||||
|
fieldsets = (
|
||||||
|
("Asosiy", {
|
||||||
|
"fields": ("name", "code"),
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(PermissionToTab)
|
||||||
|
class PermissionToTabAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ("id", "name", "code", "created_at")
|
||||||
|
search_fields = ("name", "code")
|
||||||
|
filter_horizontal = ("permission_to_actions",)
|
||||||
|
|
||||||
|
fieldsets = (
|
||||||
|
("Asosiy", {
|
||||||
|
"fields": ("name", "code"),
|
||||||
|
}),
|
||||||
|
("Harakatlar", {
|
||||||
|
"fields": ("permission_to_actions",),
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(Permission)
|
||||||
|
class PermissionAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ("id", "name", "code", "created_at")
|
||||||
|
search_fields = ("name", "code")
|
||||||
|
filter_horizontal = ("permission_tabs",)
|
||||||
|
|
||||||
|
fieldsets = (
|
||||||
|
("Asosiy", {
|
||||||
|
"fields": ("name", "code"),
|
||||||
|
}),
|
||||||
|
("Bog‘lanishlar", {
|
||||||
|
"fields": ("permission_tabs",),
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(Role)
|
||||||
|
class RoleAdmin(admin.ModelAdmin):
|
||||||
|
list_display = ("id", "name")
|
||||||
|
search_fields = ("name",)
|
||||||
|
|
||||||
|
filter_horizontal = (
|
||||||
|
"permissions",
|
||||||
|
"permission_to_tabs",
|
||||||
|
"permission_to_actions",
|
||||||
|
)
|
||||||
|
|
||||||
|
fieldsets = (
|
||||||
|
("Asosiy ma'lumotlar", {
|
||||||
|
"fields": ("name", "comment"),
|
||||||
|
}),
|
||||||
|
("Sahifa ruxsatlari", {
|
||||||
|
"fields": ("permissions",),
|
||||||
|
}),
|
||||||
|
("Bo‘lim ruxsatlari", {
|
||||||
|
"fields": ("permission_to_tabs",),
|
||||||
|
}),
|
||||||
|
("Harakat ruxsatlari", {
|
||||||
|
"fields": ("permission_to_actions",),
|
||||||
|
}),
|
||||||
|
)
|
||||||
@@ -1,4 +1,5 @@
|
|||||||
from django.contrib.auth import admin
|
from django.contrib.auth import admin
|
||||||
|
from django.contrib.auth.admin import UserAdmin
|
||||||
from django.utils.translation import gettext_lazy as _
|
from django.utils.translation import gettext_lazy as _
|
||||||
from django.utils.safestring import mark_safe
|
from django.utils.safestring import mark_safe
|
||||||
from unfold.admin import ModelAdmin
|
from unfold.admin import ModelAdmin
|
||||||
|
|||||||
@@ -0,0 +1,78 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-27 09:33
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('accounts', '0003_user_avatar'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='PermissionToAction',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('name', models.CharField(max_length=200)),
|
||||||
|
('code', models.CharField(max_length=100, unique=True)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Harakatlar uchun ruxsatnoma',
|
||||||
|
'verbose_name_plural': 'Harakatlar uchun ruxsatnomalar',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='PermissionToTab',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('name', models.CharField(max_length=200)),
|
||||||
|
('code', models.CharField(max_length=100, unique=True)),
|
||||||
|
('permission_to_actions', models.ManyToManyField(related_name='permission_to_tabs', to='accounts.permissiontoaction')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': "Bo'lim uchun ruxsatnoma",
|
||||||
|
'verbose_name_plural': "Bo'lim uchun ruxsatnomalar",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Permission',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('name', models.CharField(max_length=200)),
|
||||||
|
('code', models.CharField(max_length=100, unique=True)),
|
||||||
|
('permission_tab', models.ManyToManyField(related_name='permissions', to='accounts.permissiontotab')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Sahifa uchun ruxsatnoma',
|
||||||
|
'verbose_name_plural': 'Sahifa uchun ruxsatnomalar',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Role',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('name', models.CharField(max_length=200, unique=True)),
|
||||||
|
('comment', models.CharField(blank=True, max_length=200, null=True)),
|
||||||
|
('permission_to_actions', models.ManyToManyField(blank=True, related_name='roles', to='accounts.permissiontoaction')),
|
||||||
|
('permission_to_tabs', models.ManyToManyField(blank=True, related_name='roles', to='accounts.permissiontotab')),
|
||||||
|
('permissions', models.ManyToManyField(blank=True, related_name='roles', to='accounts.permission')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Rol',
|
||||||
|
'verbose_name_plural': 'Rollar',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='user',
|
||||||
|
name='permission',
|
||||||
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='users', to='accounts.role'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-28 11:02
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('accounts', '0004_permissiontoaction_permissiontotab_permission_role_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RenameField(
|
||||||
|
model_name='permission',
|
||||||
|
old_name='permission_tab',
|
||||||
|
new_name='permission_tabs',
|
||||||
|
),
|
||||||
|
]
|
||||||
60
core/apps/accounts/models/permission.py
Normal file
60
core/apps/accounts/models/permission.py
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
from django.db import models
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionToAction(AbstractBaseModel):
|
||||||
|
name = models.CharField(max_length=200)
|
||||||
|
code = models.CharField(max_length=100, unique=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"{self.name} - {self.code}"
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _('Harakatlar uchun ruxsatnoma')
|
||||||
|
verbose_name_plural = _('Harakatlar uchun ruxsatnomalar')
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionToTab(AbstractBaseModel):
|
||||||
|
name = models.CharField(max_length=200)
|
||||||
|
code = models.CharField(max_length=100, unique=True)
|
||||||
|
permission_to_actions = models.ManyToManyField(
|
||||||
|
PermissionToAction, related_name='permission_to_tabs'
|
||||||
|
)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f'{self.name} - {self.code}'
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _("Bo'lim uchun ruxsatnoma")
|
||||||
|
verbose_name_plural = _("Bo'lim uchun ruxsatnomalar")
|
||||||
|
|
||||||
|
|
||||||
|
class Permission(AbstractBaseModel):
|
||||||
|
name = models.CharField(max_length=200)
|
||||||
|
code = models.CharField(max_length=100, unique=True)
|
||||||
|
permission_tabs = models.ManyToManyField(PermissionToTab, related_name='permissions')
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f'{self.name} - {self.code}'
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _('Sahifa uchun ruxsatnoma')
|
||||||
|
verbose_name_plural = _('Sahifa uchun ruxsatnomalar')
|
||||||
|
|
||||||
|
class Role(models.Model):
|
||||||
|
name = models.CharField(max_length=200, unique=True)
|
||||||
|
permissions = models.ManyToManyField(Permission, related_name='roles', blank=True)
|
||||||
|
permission_to_tabs = models.ManyToManyField(PermissionToTab, related_name='roles', blank=True)
|
||||||
|
permission_to_actions = models.ManyToManyField(
|
||||||
|
PermissionToAction, related_name='roles', blank=True
|
||||||
|
)
|
||||||
|
comment = models.CharField(max_length=200, null=True, blank=True)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _('Rol')
|
||||||
|
verbose_name_plural = _('Rollar')
|
||||||
|
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from django.contrib.auth import models as auth_models
|
from django.contrib.auth import models as auth_models
|
||||||
from django.db import models
|
from django.db import models
|
||||||
|
|
||||||
|
from .permission import Role
|
||||||
from ..choices import RoleChoice
|
from ..choices import RoleChoice
|
||||||
from ..managers import UserManager
|
from ..managers import UserManager
|
||||||
|
|
||||||
@@ -17,6 +18,7 @@ class User(auth_models.AbstractUser):
|
|||||||
default=RoleChoice.USER,
|
default=RoleChoice.USER,
|
||||||
)
|
)
|
||||||
avatar = models.ImageField(upload_to="avatars/", null=True, blank=True)
|
avatar = models.ImageField(upload_to="avatars/", null=True, blank=True)
|
||||||
|
permission = models.ForeignKey(Role, on_delete=models.SET_NULL, null=True, blank=True, related_name='users')
|
||||||
|
|
||||||
USERNAME_FIELD = "phone"
|
USERNAME_FIELD = "phone"
|
||||||
objects = UserManager()
|
objects = UserManager()
|
||||||
|
|||||||
15
core/apps/accounts/permissions.py
Normal file
15
core/apps/accounts/permissions.py
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
from rest_framework.exceptions import PermissionDenied
|
||||||
|
from rest_framework.permissions import BasePermission
|
||||||
|
|
||||||
|
from core.apps.accounts.choices import RoleChoice
|
||||||
|
|
||||||
|
|
||||||
|
class IsAdminRole(BasePermission):
|
||||||
|
def has_permission(self, request, view):
|
||||||
|
if not request.user.is_authenticated:
|
||||||
|
return False
|
||||||
|
|
||||||
|
if request.user.role != RoleChoice.ADMIN:
|
||||||
|
raise PermissionDenied("Only admin can access this")
|
||||||
|
|
||||||
|
return True
|
||||||
55
core/apps/accounts/serializers/permission.py
Normal file
55
core/apps/accounts/serializers/permission.py
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.accounts.models.permission import PermissionToAction, PermissionToTab, Permission, Role
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionToActionSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = PermissionToAction
|
||||||
|
fields = ['id', 'name']
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionToTabSerializer(serializers.ModelSerializer):
|
||||||
|
permission_to_actions = PermissionToActionSerializer(many=True, read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = PermissionToTab
|
||||||
|
fields = ['id', 'name', 'permission_to_actions']
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionSerializer(serializers.ModelSerializer):
|
||||||
|
permission_tabs = PermissionToTabSerializer(many=True, read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Permission
|
||||||
|
fields = ['id', 'name', 'permission_tabs']
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionToActionListSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = PermissionToAction
|
||||||
|
fields = ['id', 'name']
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionToTabListSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = PermissionToTab
|
||||||
|
fields = ['id', 'name']
|
||||||
|
|
||||||
|
|
||||||
|
class PermissionListSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = Permission
|
||||||
|
fields = ['id', 'name']
|
||||||
|
|
||||||
|
|
||||||
|
class RoleListSerializer(serializers.ModelSerializer):
|
||||||
|
permissions = PermissionListSerializer(many=True)
|
||||||
|
permission_to_tabs = PermissionToTabListSerializer(many=True)
|
||||||
|
permission_to_actions = PermissionToActionListSerializer(many=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = Role
|
||||||
|
fields = [
|
||||||
|
'id', 'name', 'comment', 'permissions', 'permission_to_tabs', 'permission_to_actions',
|
||||||
|
]
|
||||||
@@ -3,16 +3,24 @@ from rest_framework import serializers
|
|||||||
|
|
||||||
|
|
||||||
class UserSerializer(serializers.ModelSerializer):
|
class UserSerializer(serializers.ModelSerializer):
|
||||||
|
avatar = serializers.SerializerMethodField(method_name='get_avatar')
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
exclude = [
|
exclude = [
|
||||||
"created_at",
|
"created_at",
|
||||||
"updated_at",
|
"updated_at",
|
||||||
"password",
|
"password",
|
||||||
"groups",
|
"groups",
|
||||||
"user_permissions"
|
"user_permissions",
|
||||||
]
|
]
|
||||||
model = get_user_model()
|
model = get_user_model()
|
||||||
|
|
||||||
|
def get_avatar(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
if obj.avatar:
|
||||||
|
return request.build_absolute_uri(obj.avatar.url)
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
class UserUpdateSerializer(serializers.ModelSerializer):
|
class UserUpdateSerializer(serializers.ModelSerializer):
|
||||||
class Meta:
|
class Meta:
|
||||||
@@ -22,3 +30,46 @@ class UserUpdateSerializer(serializers.ModelSerializer):
|
|||||||
"last_name",
|
"last_name",
|
||||||
"avatar"
|
"avatar"
|
||||||
]
|
]
|
||||||
|
|
||||||
|
class AdminUserSerializer(serializers.ModelSerializer):
|
||||||
|
avatar = serializers.SerializerMethodField(method_name='get_avatar')
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = get_user_model()
|
||||||
|
fields = "__all__"
|
||||||
|
|
||||||
|
def get_avatar(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
if obj.avatar:
|
||||||
|
return request.build_absolute_uri(obj.avatar.url)
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class UserCreateSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = get_user_model()
|
||||||
|
fields = [
|
||||||
|
"phone",
|
||||||
|
"first_name",
|
||||||
|
"last_name",
|
||||||
|
"password",
|
||||||
|
"role"
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ShortUserSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = get_user_model()
|
||||||
|
fields = [
|
||||||
|
'id',
|
||||||
|
'first_name',
|
||||||
|
'last_name',
|
||||||
|
'avatar',
|
||||||
|
]
|
||||||
|
|
||||||
|
def get_avatar(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
if obj.avatar:
|
||||||
|
return request.build_absolute_uri(obj.avatar.url)
|
||||||
|
return None
|
||||||
@@ -4,23 +4,34 @@ Accounts app urls
|
|||||||
|
|
||||||
from django.urls import path, include
|
from django.urls import path, include
|
||||||
from rest_framework_simplejwt import views as jwt_views
|
from rest_framework_simplejwt import views as jwt_views
|
||||||
from .views import RegisterView, ResetPasswordView, MeView, ChangePasswordView
|
from .views import RegisterView, ResetPasswordView, MeView, ChangePasswordView, UserListApiView, AdminUserListApiView, \
|
||||||
|
AdminUserView, AdminCreateAPIView, AdminUpdateAPIView
|
||||||
from rest_framework.routers import DefaultRouter
|
from rest_framework.routers import DefaultRouter
|
||||||
|
|
||||||
|
from .views.permission import PermissionToActionViewSet, PermissionToTabViewSet, PermissionViewSet, RoleViewSet
|
||||||
|
from core.apps.accounts.views.user import DeleteAdminUserApiView, UserDetailAPIView
|
||||||
|
|
||||||
router = DefaultRouter()
|
router = DefaultRouter()
|
||||||
router.register("auth", RegisterView, basename="auth")
|
router.register("auth", RegisterView, basename="auth")
|
||||||
router.register("auth", ResetPasswordView, basename="reset-password")
|
router.register("auth", ResetPasswordView, basename="reset-password")
|
||||||
router.register("auth", MeView, basename="me")
|
router.register("auth", MeView, basename="me")
|
||||||
router.register("auth", ChangePasswordView, basename="change-password")
|
router.register("auth", ChangePasswordView, basename="change-password")
|
||||||
|
router.register("user", AdminUserView, basename="user-crud")
|
||||||
|
router.register("action", PermissionToActionViewSet, basename="action")
|
||||||
|
router.register("permission-to-tab", PermissionToTabViewSet, basename="permission-to-tab")
|
||||||
|
router.register("permission", PermissionViewSet, basename="permission")
|
||||||
|
router.register("permission-role", RoleViewSet, basename="permission-role")
|
||||||
|
|
||||||
|
|
||||||
urlpatterns = [
|
urlpatterns = [
|
||||||
path("", include(router.urls)),
|
path("", include(router.urls)),
|
||||||
path("auth/token/", jwt_views.TokenObtainPairView.as_view(), name="token_obtain_pair"),
|
path("auth/token/", jwt_views.TokenObtainPairView.as_view(), name="token_obtain_pair"),
|
||||||
path("auth/token/verify/", jwt_views.TokenVerifyView.as_view(), name="token_verify"),
|
path("auth/token/verify/", jwt_views.TokenVerifyView.as_view(), name="token_verify"),
|
||||||
path(
|
path("auth/token/refresh/", jwt_views.TokenRefreshView.as_view()),
|
||||||
"auth/token/refresh/",
|
path("user/list/", UserListApiView.as_view(), name="user-list"),
|
||||||
jwt_views.TokenRefreshView.as_view(),
|
path("admin-user/list/", AdminUserListApiView.as_view(), name="admin-user-list"),
|
||||||
name="token_refresh",
|
path("admin/create/", AdminCreateAPIView.as_view(), name="user-create"),
|
||||||
),
|
path("admin/update/<int:pk>/", AdminUpdateAPIView.as_view(), name="user-update"),
|
||||||
|
path('user/admin/<int:pk>/delete/', DeleteAdminUserApiView.as_view(), name='user-delete'),
|
||||||
|
path('user/<int:pk>/', UserDetailAPIView.as_view(), name='user-detail'),
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -1 +1,2 @@
|
|||||||
from .auth import * # noqa
|
from .auth import * # noqa
|
||||||
|
from .user import * # noqa
|
||||||
@@ -177,11 +177,11 @@ class MeView(BaseViewSetMixin, GenericViewSet, UserService):
|
|||||||
|
|
||||||
@action(methods=["GET", "OPTIONS"], detail=False, url_path="me")
|
@action(methods=["GET", "OPTIONS"], detail=False, url_path="me")
|
||||||
def me(self, request):
|
def me(self, request):
|
||||||
return Response(self.get_serializer(request.user).data)
|
return Response(UserSerializer(request.user, context={"request": request}).data)
|
||||||
|
|
||||||
@action(methods=["PATCH", "PUT"], detail=False, url_path="user-update")
|
@action(methods=["PATCH", "PUT"], detail=False, url_path="user-update")
|
||||||
def user_update(self, request):
|
def user_update(self, request):
|
||||||
ser = self.get_serializer(instance=request.user, data=request.data, partial=True)
|
ser = self.get_serializer(instance=request.user, data=request.data, partial=True, context={"request": request})
|
||||||
ser.is_valid(raise_exception=True)
|
ser.is_valid(raise_exception=True)
|
||||||
data = ser.save()
|
data = ser.save()
|
||||||
return Response(UserSerializer(data).data)
|
return Response(UserSerializer(data).data)
|
||||||
|
|||||||
41
core/apps/accounts/views/permission.py
Normal file
41
core/apps/accounts/views/permission.py
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
from django_core.mixins import BaseViewSetMixin
|
||||||
|
from drf_spectacular.utils import extend_schema
|
||||||
|
from rest_framework.permissions import AllowAny, IsAdminUser
|
||||||
|
from rest_framework.viewsets import ModelViewSet
|
||||||
|
|
||||||
|
from core.apps.accounts.models.permission import PermissionToAction, PermissionToTab, Permission, Role
|
||||||
|
from core.apps.accounts.serializers.permission import PermissionToActionSerializer, PermissionToTabSerializer, \
|
||||||
|
PermissionSerializer, RoleListSerializer
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["permission"])
|
||||||
|
class PermissionToActionViewSet(BaseViewSetMixin, ModelViewSet):
|
||||||
|
queryset = PermissionToAction.objects.all()
|
||||||
|
serializer_class = PermissionToActionSerializer
|
||||||
|
|
||||||
|
action_serializer_class = {
|
||||||
|
'create': PermissionToActionSerializer,
|
||||||
|
'update': PermissionToActionSerializer,
|
||||||
|
}
|
||||||
|
|
||||||
|
action_permission_classes = {
|
||||||
|
'create': [AllowAny],
|
||||||
|
'destroy': [IsAdminUser],
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["permission"])
|
||||||
|
class PermissionToTabViewSet(BaseViewSetMixin, ModelViewSet):
|
||||||
|
queryset = PermissionToTab.objects.all()
|
||||||
|
serializer_class = PermissionToTabSerializer
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["permission"])
|
||||||
|
class PermissionViewSet(BaseViewSetMixin, ModelViewSet):
|
||||||
|
queryset = Permission.objects.all()
|
||||||
|
serializer_class = PermissionSerializer
|
||||||
|
|
||||||
|
|
||||||
|
class RoleViewSet(BaseViewSetMixin, ModelViewSet):
|
||||||
|
queryset = Role.objects.all()
|
||||||
|
serializer_class = RoleListSerializer
|
||||||
118
core/apps/accounts/views/user.py
Normal file
118
core/apps/accounts/views/user.py
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
from django.contrib.auth import get_user_model
|
||||||
|
from django.shortcuts import get_object_or_404
|
||||||
|
from django_core.mixins import BaseViewSetMixin
|
||||||
|
from drf_spectacular.utils import extend_schema
|
||||||
|
from rest_framework import generics, filters
|
||||||
|
from rest_framework.permissions import IsAuthenticated
|
||||||
|
from rest_framework.response import Response
|
||||||
|
from rest_framework.views import APIView
|
||||||
|
from rest_framework.viewsets import ModelViewSet
|
||||||
|
|
||||||
|
from core.apps.accounts.choices.user import RoleChoice
|
||||||
|
from core.apps.accounts.models import Role
|
||||||
|
from core.apps.accounts.serializers.permission import RoleListSerializer
|
||||||
|
from core.apps.accounts.serializers.user import UserSerializer, AdminUserSerializer, UserCreateSerializer
|
||||||
|
|
||||||
|
User = get_user_model()
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['User'])
|
||||||
|
class UserListApiView(generics.ListAPIView):
|
||||||
|
queryset = User.objects.filter(role=RoleChoice.USER)
|
||||||
|
serializer_class = UserSerializer
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
filter_backends = [filters.SearchFilter]
|
||||||
|
search_fields = ['phone', 'first_name', 'last_name']
|
||||||
|
|
||||||
|
def serializer_context(self):
|
||||||
|
return self.serializer_class(context={"request": self.request})
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['User'])
|
||||||
|
class AdminUserListApiView(generics.ListAPIView):
|
||||||
|
queryset = User.objects.exclude(role=RoleChoice.USER)
|
||||||
|
serializer_class = UserSerializer
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
filter_backends = [filters.SearchFilter]
|
||||||
|
search_fields = ['phone', 'first_name', 'last_name']
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=["User"], request=AdminUserSerializer)
|
||||||
|
class AdminUserView(BaseViewSetMixin, ModelViewSet):
|
||||||
|
queryset = User.objects.filter(role=RoleChoice.USER)
|
||||||
|
serializer_class = AdminUserSerializer
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
filter_backends = [filters.SearchFilter]
|
||||||
|
search_fields = ['phone', 'first_name', 'last_name']
|
||||||
|
|
||||||
|
def serializer_context(self):
|
||||||
|
return self.serializer_class(context={"request": self.request})
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['User'],
|
||||||
|
responses={200: UserSerializer},
|
||||||
|
request=UserCreateSerializer)
|
||||||
|
class AdminCreateAPIView(APIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
|
||||||
|
def post(self, request):
|
||||||
|
if request.user.role not in (RoleChoice.SUPERUSER, RoleChoice.ADMIN):
|
||||||
|
return Response({'detail': 'Forbidden'}, status=403)
|
||||||
|
|
||||||
|
serializer = UserCreateSerializer(data=request.data)
|
||||||
|
serializer.is_valid(raise_exception=True)
|
||||||
|
serializer.save()
|
||||||
|
|
||||||
|
return Response(serializer.data, status=201)
|
||||||
|
|
||||||
|
|
||||||
|
@extend_schema(tags=['User'], )
|
||||||
|
class AdminUpdateAPIView(generics.GenericAPIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
serializer_class = UserCreateSerializer
|
||||||
|
|
||||||
|
def put(self, request, pk):
|
||||||
|
if request.user.role not in (RoleChoice.SUPERUSER, RoleChoice.ADMIN):
|
||||||
|
return Response({'detail': 'Forbidden'}, status=403)
|
||||||
|
|
||||||
|
user = get_object_or_404(User, pk=pk)
|
||||||
|
serializer = UserCreateSerializer(user, data=request.data)
|
||||||
|
serializer.is_valid(raise_exception=True)
|
||||||
|
serializer.save()
|
||||||
|
|
||||||
|
return Response(serializer.data, status=200)
|
||||||
|
|
||||||
|
|
||||||
|
class DeleteAdminUserApiView(APIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
|
||||||
|
def delete(self, request, pk):
|
||||||
|
if request.user.role != RoleChoice.SUPERUSER:
|
||||||
|
return Response({'detail': 'Forbidden'}, status=403)
|
||||||
|
|
||||||
|
user = get_object_or_404(User, pk=pk)
|
||||||
|
if user.role != RoleChoice.ADMIN:
|
||||||
|
return Response({'detail': 'This user is not an admin'}, status=400)
|
||||||
|
user.delete()
|
||||||
|
|
||||||
|
return Response(status=204)
|
||||||
|
|
||||||
|
|
||||||
|
class UserDetailAPIView(generics.RetrieveAPIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
serializer_class = UserSerializer
|
||||||
|
lookup_field = 'id'
|
||||||
|
|
||||||
|
|
||||||
|
class AdminPermissionsAPIView(generics.GenericAPIView):
|
||||||
|
permission_classes = [IsAuthenticated]
|
||||||
|
queryset = User.objects.all()
|
||||||
|
|
||||||
|
def get(self, request):
|
||||||
|
if request.user.role.name != RoleChoice.ADMIN:
|
||||||
|
return Response({'detail': 'Forbidden'}, status=403)
|
||||||
|
|
||||||
|
admin_role = Role.objects.get(name=RoleChoice.ADMIN)
|
||||||
|
|
||||||
|
serializer = RoleListSerializer(admin_role)
|
||||||
|
return Response(serializer.data)
|
||||||
@@ -4,6 +4,11 @@ from channels.db import database_sync_to_async
|
|||||||
from channels.generic.websocket import AsyncWebsocketConsumer
|
from channels.generic.websocket import AsyncWebsocketConsumer
|
||||||
from django.contrib.auth.models import AnonymousUser
|
from django.contrib.auth.models import AnonymousUser
|
||||||
|
|
||||||
|
def get_base_url(scope):
|
||||||
|
headers = dict(scope["headers"])
|
||||||
|
host = headers.get(b"host", b"").decode()
|
||||||
|
scheme = "https" if scope.get("scheme") == "https" else "http"
|
||||||
|
return f"{scheme}://{host}"
|
||||||
|
|
||||||
class ChatConsumer(AsyncWebsocketConsumer):
|
class ChatConsumer(AsyncWebsocketConsumer):
|
||||||
"""
|
"""
|
||||||
@@ -36,7 +41,7 @@ class ChatConsumer(AsyncWebsocketConsumer):
|
|||||||
async def receive(self, text_data):
|
async def receive(self, text_data):
|
||||||
user = self.scope.get("user")
|
user = self.scope.get("user")
|
||||||
if not user or isinstance(user, AnonymousUser):
|
if not user or isinstance(user, AnonymousUser):
|
||||||
await self.close(code=4001)
|
await self.close(code=401)
|
||||||
return
|
return
|
||||||
|
|
||||||
try:
|
try:
|
||||||
@@ -48,13 +53,10 @@ class ChatConsumer(AsyncWebsocketConsumer):
|
|||||||
message_type = data.get("message_type", "text")
|
message_type = data.get("message_type", "text")
|
||||||
text = (data.get("text") or "").strip()
|
text = (data.get("text") or "").strip()
|
||||||
|
|
||||||
# Matn xabari uchun text majburiy
|
|
||||||
if message_type == "text" and not text:
|
if message_type == "text" and not text:
|
||||||
await self.send(text_data=json.dumps({"error": "Matn bo'sh bo'lishi mumkin emas."}))
|
await self.send(text_data=json.dumps({"error": "Matn bo'sh bo'lishi mumkin emas."}))
|
||||||
return
|
return
|
||||||
|
|
||||||
# WS orqali faqat matn + caption saqlanadi.
|
|
||||||
# Fayl yuklash uchun REST /chat/messages/ POST ishlatiladi.
|
|
||||||
if message_type != "text":
|
if message_type != "text":
|
||||||
await self.send(
|
await self.send(
|
||||||
text_data=json.dumps(
|
text_data=json.dumps(
|
||||||
@@ -63,9 +65,9 @@ class ChatConsumer(AsyncWebsocketConsumer):
|
|||||||
)
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
# DB ga saqlash — post_save signal WS ga broadcast qiladi
|
|
||||||
await self._save_message(user, text)
|
await self._save_message(user, text)
|
||||||
|
|
||||||
|
|
||||||
async def chat_message(self, event):
|
async def chat_message(self, event):
|
||||||
await self.send(
|
await self.send(
|
||||||
text_data=json.dumps(
|
text_data=json.dumps(
|
||||||
@@ -91,6 +93,12 @@ class ChatConsumer(AsyncWebsocketConsumer):
|
|||||||
text=text,
|
text=text,
|
||||||
)
|
)
|
||||||
full_name = user.get_full_name().strip() or str(user.phone)
|
full_name = user.get_full_name().strip() or str(user.phone)
|
||||||
|
base_url = get_base_url(self.scope)
|
||||||
|
|
||||||
|
avatar_url = (
|
||||||
|
base_url + user.avatar.url
|
||||||
|
if user.avatar else None
|
||||||
|
)
|
||||||
return {
|
return {
|
||||||
"id": msg.id,
|
"id": msg.id,
|
||||||
"message_type": msg.message_type,
|
"message_type": msg.message_type,
|
||||||
@@ -100,6 +108,8 @@ class ChatConsumer(AsyncWebsocketConsumer):
|
|||||||
"id": user.id,
|
"id": user.id,
|
||||||
"full_name": full_name,
|
"full_name": full_name,
|
||||||
"role": user.role,
|
"role": user.role,
|
||||||
|
"phone": user.phone,
|
||||||
|
"avatar": avatar_url,
|
||||||
},
|
},
|
||||||
"created_at": msg.created_at.isoformat(),
|
"created_at": msg.created_at.isoformat(),
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,22 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-03 10:58
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('chat', '0002_chatroom_and_message_media'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AlterModelOptions(
|
||||||
|
name='chatmessagemodel',
|
||||||
|
options={'ordering': ['created_at'], 'verbose_name': 'Chat Xabar', 'verbose_name_plural': 'Chat Xabarlar'},
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='chatroommodel',
|
||||||
|
name='type',
|
||||||
|
field=models.CharField(choices=[('auto_evaluation', 'AutoEvaluation xonasi'), ('direct', "To'g'ridan-to'g'ri")], db_index=True, default='auto_evaluation', max_length=20, verbose_name='type'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
from core.apps.chat.models import ChatmessageModel
|
from core.apps.chat.models import ChatmessageModel
|
||||||
|
from core.apps.chat.tasks.message import send_message_to_chat
|
||||||
|
|
||||||
class BaseChatmessageSerializer(serializers.ModelSerializer):
|
class BaseChatmessageSerializer(serializers.ModelSerializer):
|
||||||
sender = serializers.SerializerMethodField()
|
sender = serializers.SerializerMethodField()
|
||||||
@@ -13,10 +13,13 @@ class BaseChatmessageSerializer(serializers.ModelSerializer):
|
|||||||
full_name = obj.sender.get_full_name().strip()
|
full_name = obj.sender.get_full_name().strip()
|
||||||
if not full_name:
|
if not full_name:
|
||||||
full_name = str(obj.sender.phone)
|
full_name = str(obj.sender.phone)
|
||||||
|
request = self.context.get("request")
|
||||||
return {
|
return {
|
||||||
"id": obj.sender.id,
|
"id": obj.sender.id,
|
||||||
"full_name": full_name,
|
"full_name": full_name,
|
||||||
"role": obj.sender.role,
|
"role": obj.sender.role,
|
||||||
|
"phone": obj.sender.phone,
|
||||||
|
"avatar": request.build_absolute_uri(obj.sender.avatar.url) if obj.sender.avatar else None,
|
||||||
}
|
}
|
||||||
|
|
||||||
def get_file_url(self, obj):
|
def get_file_url(self, obj):
|
||||||
@@ -72,4 +75,9 @@ class CreateChatmessageSerializer(serializers.ModelSerializer):
|
|||||||
|
|
||||||
def create(self, validated_data):
|
def create(self, validated_data):
|
||||||
validated_data["sender"] = self.context["request"].user
|
validated_data["sender"] = self.context["request"].user
|
||||||
return super().create(validated_data)
|
request = self.context["request"]
|
||||||
|
message = super().create(validated_data)
|
||||||
|
file_url = request.build_absolute_uri(message.file.url) if message.file else None
|
||||||
|
avatar_url = request.build_absolute_uri(self.context['request'].user.avatar.url) if self.context['request'].user.avatar else None
|
||||||
|
send_message_to_chat.delay(message.id, file_url, avatar_url)
|
||||||
|
return message
|
||||||
|
|||||||
@@ -7,42 +7,42 @@ from django.dispatch import receiver
|
|||||||
from core.apps.chat.models import ChatmessageModel, ChatroomModel
|
from core.apps.chat.models import ChatmessageModel, ChatroomModel
|
||||||
|
|
||||||
|
|
||||||
@receiver(post_save, sender=ChatmessageModel)
|
# @receiver(post_save, sender=ChatmessageModel)
|
||||||
def broadcast_new_message(sender, instance, created, **kwargs):
|
# def broadcast_new_message(sender, instance, created, **kwargs):
|
||||||
"""Yangi xabar saqlanganda xonadagi barcha WS ulanishlariga yuboradi."""
|
# """Yangi xabar saqlanganda xonadagi barcha WS ulanishlariga yuboradi."""
|
||||||
if not created:
|
# if not created:
|
||||||
return
|
# return
|
||||||
|
|
||||||
channel_layer = get_channel_layer()
|
# channel_layer = get_channel_layer()
|
||||||
if channel_layer is None:
|
# if channel_layer is None:
|
||||||
return
|
# return
|
||||||
|
|
||||||
sender_obj = instance.sender
|
# sender_obj = instance.sender
|
||||||
if sender_obj:
|
# if sender_obj:
|
||||||
full_name = sender_obj.get_full_name().strip() or str(sender_obj.phone)
|
# full_name = sender_obj.get_full_name().strip() or str(sender_obj.phone)
|
||||||
sender_data = {
|
# sender_data = {
|
||||||
"id": sender_obj.id,
|
# "id": sender_obj.id,
|
||||||
"full_name": full_name,
|
# "full_name": full_name,
|
||||||
"role": sender_obj.role,
|
# "role": sender_obj.role,
|
||||||
}
|
# }
|
||||||
else:
|
# else:
|
||||||
sender_data = None
|
# sender_data = None
|
||||||
|
|
||||||
site_url = getattr(settings, "SITE_URL", "").rstrip("/")
|
# site_url = getattr(settings, "SITE_URL", "").rstrip("/")
|
||||||
file_url = (site_url + instance.file.url) if instance.file else None
|
# file_url = (site_url + instance.file.url) if instance.file else None
|
||||||
|
|
||||||
async_to_sync(channel_layer.group_send)(
|
# async_to_sync(channel_layer.group_send)(
|
||||||
f"chat_room_{instance.room_id}",
|
# f"chat_room_{instance.room_id}",
|
||||||
{
|
# {
|
||||||
"type": "chat_message",
|
# "type": "chat_message",
|
||||||
"id": instance.id,
|
# "id": instance.id,
|
||||||
"message_type": instance.message_type,
|
# "message_type": instance.message_type,
|
||||||
"text": instance.text,
|
# "text": instance.text,
|
||||||
"file_url": file_url,
|
# "file_url": file_url,
|
||||||
"sender": sender_data,
|
# "sender": sender_data,
|
||||||
"created_at": instance.created_at.isoformat(),
|
# "created_at": instance.created_at.isoformat(),
|
||||||
},
|
# },
|
||||||
)
|
# )
|
||||||
|
|
||||||
|
|
||||||
@receiver(post_save, sender="evaluation.AutoEvaluationModel")
|
@receiver(post_save, sender="evaluation.AutoEvaluationModel")
|
||||||
|
|||||||
1
core/apps/chat/tasks/__init__.py
Normal file
1
core/apps/chat/tasks/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
from .message import send_message_to_chat
|
||||||
44
core/apps/chat/tasks/message.py
Normal file
44
core/apps/chat/tasks/message.py
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
from asgiref.sync import async_to_sync
|
||||||
|
from channels.layers import get_channel_layer
|
||||||
|
|
||||||
|
from celery import shared_task
|
||||||
|
|
||||||
|
from core.apps.chat.models import ChatmessageModel
|
||||||
|
|
||||||
|
|
||||||
|
@shared_task
|
||||||
|
def send_message_to_chat(message_id, file_url, avatar_url):
|
||||||
|
try:
|
||||||
|
message = ChatmessageModel.objects.get(id=message_id)
|
||||||
|
except ChatmessageModel.DoesNotExist:
|
||||||
|
return "Not found"
|
||||||
|
|
||||||
|
channel_layer = get_channel_layer()
|
||||||
|
if channel_layer is None:
|
||||||
|
return
|
||||||
|
|
||||||
|
sender_obj = message.sender
|
||||||
|
if sender_obj:
|
||||||
|
full_name = sender_obj.get_full_name().strip() or str(sender_obj.phone)
|
||||||
|
sender_data = {
|
||||||
|
"id": sender_obj.id,
|
||||||
|
"full_name": full_name,
|
||||||
|
"role": sender_obj.role,
|
||||||
|
"phone": sender_obj.phone,
|
||||||
|
"avatar": avatar_url,
|
||||||
|
}
|
||||||
|
else:
|
||||||
|
sender_data = None
|
||||||
|
|
||||||
|
async_to_sync(channel_layer.group_send)(
|
||||||
|
f"chat_room_{message.room_id}",
|
||||||
|
{
|
||||||
|
"type": "chat_message",
|
||||||
|
"id": message.id,
|
||||||
|
"message_type": message.message_type,
|
||||||
|
"text": message.text,
|
||||||
|
"file_url": file_url,
|
||||||
|
"sender": sender_data,
|
||||||
|
"created_at": message.created_at.isoformat(),
|
||||||
|
},
|
||||||
|
)
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -50,15 +50,8 @@ class AutoEvaluationAdmin(ModelAdmin):
|
|||||||
("value_determined", "rate_type"),
|
("value_determined", "rate_type"),
|
||||||
),
|
),
|
||||||
}),
|
}),
|
||||||
("Step 3 — Manzil ma'lumotlari", {
|
|
||||||
"fields": (
|
("Step 3 — Avtomobil ma'lumotlari", {
|
||||||
("object_location_province", "object_location_district"),
|
|
||||||
("object_location_city", "object_location_neighborhood"),
|
|
||||||
("object_location_street", "object_location_home"),
|
|
||||||
("object_location_highways", "object_location_covenience"),
|
|
||||||
),
|
|
||||||
}),
|
|
||||||
("Step 4 — Avtomobil ma'lumotlari", {
|
|
||||||
"fields": (
|
"fields": (
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
("tex_passport_gived_date", "tex_passport_gived_location"),
|
("tex_passport_gived_date", "tex_passport_gived_location"),
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
from django.contrib import admin
|
from django.contrib import admin
|
||||||
from unfold.admin import ModelAdmin
|
from unfold.admin import ModelAdmin
|
||||||
|
|
||||||
from core.apps.evaluation.models import ValuationDocumentModel
|
from core.apps.evaluation.models import DocumentModel, ValuationDocumentModel
|
||||||
|
|
||||||
|
|
||||||
@admin.register(ValuationDocumentModel)
|
@admin.register(ValuationDocumentModel)
|
||||||
@@ -23,20 +23,37 @@ class ValuationDocumentAdmin(ModelAdmin):
|
|||||||
readonly_fields = ("created_at", "updated_at")
|
readonly_fields = ("created_at", "updated_at")
|
||||||
autocomplete_fields = ("valuation", "uploaded_by")
|
autocomplete_fields = ("valuation", "uploaded_by")
|
||||||
fieldsets = (
|
fieldsets = (
|
||||||
("Hujjat", {
|
(
|
||||||
"fields": (
|
"Hujjat",
|
||||||
"valuation",
|
{
|
||||||
"document_type",
|
"fields": (
|
||||||
"title",
|
"valuation",
|
||||||
"file",
|
"document_type",
|
||||||
"uploaded_by",
|
"title",
|
||||||
),
|
"file",
|
||||||
}),
|
"uploaded_by",
|
||||||
("Qo'shimcha", {
|
),
|
||||||
"fields": ("description",),
|
},
|
||||||
}),
|
),
|
||||||
("Tizim", {
|
(
|
||||||
"classes": ("collapse",),
|
"Qo'shimcha",
|
||||||
"fields": ("created_at", "updated_at"),
|
{
|
||||||
}),
|
"fields": ("description",),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
(
|
||||||
|
"Tizim",
|
||||||
|
{
|
||||||
|
"classes": ("collapse",),
|
||||||
|
"fields": ("created_at", "updated_at"),
|
||||||
|
},
|
||||||
|
),
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(DocumentModel)
|
||||||
|
class DocumentAdmin(ModelAdmin):
|
||||||
|
list_display = (
|
||||||
|
"id",
|
||||||
|
"__str__",
|
||||||
)
|
)
|
||||||
|
|||||||
57
core/apps/evaluation/admin/documentcategory.py
Normal file
57
core/apps/evaluation/admin/documentcategory.py
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
from django.contrib import admin
|
||||||
|
from unfold.admin import ModelAdmin
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentcategoryModel
|
||||||
|
|
||||||
|
|
||||||
|
@admin.register(DocumentcategoryModel)
|
||||||
|
class DocumentcategoryAdmin(ModelAdmin):
|
||||||
|
list_display = (
|
||||||
|
"id",
|
||||||
|
"label",
|
||||||
|
"value",
|
||||||
|
"created_at",
|
||||||
|
"updated_at"
|
||||||
|
)
|
||||||
|
search_fields = (
|
||||||
|
"label",
|
||||||
|
"value",
|
||||||
|
)
|
||||||
|
list_filter = (
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
readonly_fields = (
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
fieldsets = (
|
||||||
|
(
|
||||||
|
("Uzbekcha"),
|
||||||
|
{
|
||||||
|
"fields": (
|
||||||
|
"label_uz",
|
||||||
|
"value_uz",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
),
|
||||||
|
(
|
||||||
|
("Ruscha"),
|
||||||
|
{
|
||||||
|
"fields": (
|
||||||
|
"label_ru",
|
||||||
|
"value_ru",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
),
|
||||||
|
(
|
||||||
|
("Inglizcha"),
|
||||||
|
{
|
||||||
|
"fields": (
|
||||||
|
"label_en",
|
||||||
|
"value_en",
|
||||||
|
)
|
||||||
|
}
|
||||||
|
),
|
||||||
|
)
|
||||||
|
list_display_links =list_display
|
||||||
8
core/apps/evaluation/choices/bonus.py
Normal file
8
core/apps/evaluation/choices/bonus.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
from django.db import models
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
|
||||||
|
|
||||||
|
class EvaluationCategory(models.TextChoices):
|
||||||
|
AUTO = "auto_transport", _("Avtotransport")
|
||||||
|
REAL_ESTATE = "real estate", _("ko'chmas mulk")
|
||||||
|
EQUIPMENT = "equipment", _("uskuna va jihozlar")
|
||||||
@@ -19,3 +19,8 @@ class RequestStatus(models.TextChoices):
|
|||||||
IN_PROGRESS = "in_progress", _("Jarayonda")
|
IN_PROGRESS = "in_progress", _("Jarayonda")
|
||||||
COMPLETED = "completed", _("Bajarildi")
|
COMPLETED = "completed", _("Bajarildi")
|
||||||
REJECTED = "rejected", _("Rad etildi")
|
REJECTED = "rejected", _("Rad etildi")
|
||||||
|
|
||||||
|
|
||||||
|
class RequestPersonType(models.TextChoices):
|
||||||
|
INDIVIDUAL_PERSON = "individual_person", "Jismoniy shaxs"
|
||||||
|
LEGAL_PERSON = "legal_person", 'Yuridik shaxs',
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -1,11 +1,18 @@
|
|||||||
from django_filters import rest_framework as filters
|
from django_filters import rest_framework as filters
|
||||||
|
|
||||||
from core.apps.evaluation.models import ValuationDocumentModel
|
from core.apps.evaluation.models import DocumentModel, ValuationDocumentModel
|
||||||
|
|
||||||
|
|
||||||
class ValuationdocumentFilter(filters.FilterSet):
|
class ValuationdocumentFilter(filters.FilterSet):
|
||||||
# name = filters.CharFilter(field_name="name", lookup_expr="icontains")
|
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = ValuationDocumentModel
|
model = ValuationDocumentModel
|
||||||
fields = []
|
fields = []
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentFilter(filters.FilterSet):
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = DocumentModel
|
||||||
|
fields = [
|
||||||
|
]
|
||||||
|
|||||||
10
core/apps/evaluation/filters/documentcategory.py
Normal file
10
core/apps/evaluation/filters/documentcategory.py
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
from django_filters import rest_framework as filters
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentcategoryModel
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentcategoryFilter(filters.FilterSet):
|
||||||
|
class Meta:
|
||||||
|
model = DocumentcategoryModel
|
||||||
|
fields = [
|
||||||
|
]
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
from django import forms
|
from django import forms
|
||||||
|
|
||||||
from core.apps.evaluation.models import ValuationDocumentModel
|
from core.apps.evaluation.models import DocumentModel, ValuationDocumentModel
|
||||||
|
|
||||||
|
|
||||||
class ValuationdocumentForm(forms.ModelForm):
|
class ValuationdocumentForm(forms.ModelForm):
|
||||||
@@ -8,3 +8,10 @@ class ValuationdocumentForm(forms.ModelForm):
|
|||||||
class Meta:
|
class Meta:
|
||||||
model = ValuationDocumentModel
|
model = ValuationDocumentModel
|
||||||
fields = "__all__"
|
fields = "__all__"
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentForm(forms.ModelForm):
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = DocumentModel
|
||||||
|
fields = "__all__"
|
||||||
|
|||||||
10
core/apps/evaluation/forms/documentcategory.py
Normal file
10
core/apps/evaluation/forms/documentcategory.py
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
from django import forms
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentcategoryModel
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentcategoryForm(forms.ModelForm):
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = DocumentcategoryModel
|
||||||
|
fields = "__all__"
|
||||||
@@ -0,0 +1,52 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-03 10:58
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0026_alter_autoevaluationmodel_form_ownership_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='DocumentcategoryModel',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('label', models.CharField(max_length=255, verbose_name='label')),
|
||||||
|
('label_uz', models.CharField(max_length=255, null=True, verbose_name='label')),
|
||||||
|
('label_ru', models.CharField(max_length=255, null=True, verbose_name='label')),
|
||||||
|
('label_en', models.CharField(max_length=255, null=True, verbose_name='label')),
|
||||||
|
('value', models.CharField(max_length=255, verbose_name='value')),
|
||||||
|
('value_uz', models.CharField(max_length=255, null=True, verbose_name='value')),
|
||||||
|
('value_ru', models.CharField(max_length=255, null=True, verbose_name='value')),
|
||||||
|
('value_en', models.CharField(max_length=255, null=True, verbose_name='value')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'DocumentcategoryModel',
|
||||||
|
'verbose_name_plural': 'DocumentcategoryModels',
|
||||||
|
'db_table': 'DocumentCategory',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='DocumentModel',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('title', models.CharField(max_length=255, verbose_name='title')),
|
||||||
|
('document', models.FileField(upload_to='evaluation/documents/%Y/%m/', verbose_name='document')),
|
||||||
|
('auto_evaluation', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='documents', to='evaluation.autoevaluationmodel', verbose_name='auto evaluation')),
|
||||||
|
('category', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='documents', to='evaluation.documentcategorymodel', verbose_name='category')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'DocumentModel',
|
||||||
|
'verbose_name_plural': 'DocumentModels',
|
||||||
|
'db_table': 'Document',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,20 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-03 11:51
|
||||||
|
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0027_documentcategorymodel_documentmodel'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='appraisers',
|
||||||
|
field=models.ManyToManyField(blank=True, null=True, related_name='auto_evaluations', to=settings.AUTH_USER_MODEL, verbose_name='appraisers'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-18 11:28
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0028_autoevaluationmodel_appraisers'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='user',
|
||||||
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='auto_evaluations_user', to=settings.AUTH_USER_MODEL, verbose_name='user'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-20 13:08
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0029_autoevaluationmodel_user'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='evaluation_request',
|
||||||
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='auto_evaluations_request', to='evaluation.evaluationrequestmodel', verbose_name='evaluation request'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,54 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-21 10:22
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0030_autoevaluationmodel_evaluation_request'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_city',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_covenience',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_district',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_highways',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_home',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_neighborhood',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_province',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='object_location_street',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='quickevaluationmodel',
|
||||||
|
name='tex_passport_file',
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='tex_passport_file',
|
||||||
|
field=models.FileField(blank=True, null=True, upload_to='quick_evaluation/tech_passports/%Y/%m/', verbose_name='tech passport file'),
|
||||||
|
),
|
||||||
|
]
|
||||||
28
core/apps/evaluation/migrations/0032_certificatemodel.py
Normal file
28
core/apps/evaluation/migrations/0032_certificatemodel.py
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
# Generated by Django 6.0.4 on 2026-04-23 11:07
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0031_remove_autoevaluationmodel_object_location_city_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='CertificateModel',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('title', models.CharField(max_length=255, verbose_name='title')),
|
||||||
|
('file_url', models.URLField(max_length=255, verbose_name='file url')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Certificate',
|
||||||
|
'verbose_name_plural': 'Certificates',
|
||||||
|
'db_table': 'certificate',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 6.0.4 on 2026-04-23 07:22
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0031_remove_autoevaluationmodel_object_location_city_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='is_archive',
|
||||||
|
field=models.BooleanField(default=False, verbose_name='is archive'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 6.0.4 on 2026-04-23 07:24
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0031_remove_autoevaluationmodel_object_location_city_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='quickevaluationmodel',
|
||||||
|
name='is_archive',
|
||||||
|
field=models.BooleanField(default=False, verbose_name='is archive'),
|
||||||
|
),
|
||||||
|
]
|
||||||
15
core/apps/evaluation/migrations/0033_merge_20260423_1622.py
Normal file
15
core/apps/evaluation/migrations/0033_merge_20260423_1622.py
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
# Generated by Django 6.0.4 on 2026-04-23 11:22
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0032_certificatemodel'),
|
||||||
|
('evaluation', '0032_evaluationrequestmodel_is_archive'),
|
||||||
|
('evaluation', '0032_quickevaluationmodel_is_archive'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
]
|
||||||
@@ -0,0 +1,22 @@
|
|||||||
|
# Generated by Django 6.0.4 on 2026-04-23 13:42
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0033_merge_20260423_1622'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='certificatemodel',
|
||||||
|
name='file_url',
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='certificatemodel',
|
||||||
|
name='file',
|
||||||
|
field=models.FileField(blank=True, null=True, upload_to='certificates/', verbose_name='file'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-27 09:33
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0034_remove_certificatemodel_file_url_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='is_archived',
|
||||||
|
field=models.BooleanField(default=False, verbose_name='is archived'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,53 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-28 11:07
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0035_autoevaluationmodel_is_archived'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='form_ownership',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='property_rights',
|
||||||
|
),
|
||||||
|
migrations.RemoveField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='rate_object_name',
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='appraisers',
|
||||||
|
field=models.ManyToManyField(blank=True, null=True, to=settings.AUTH_USER_MODEL, verbose_name='appraisers'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='evaluation_request',
|
||||||
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='auto_evaluations_request', to='evaluation.evaluationrequestmodel'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='user',
|
||||||
|
field=models.ForeignKey(blank=True, null=True, on_delete=django.db.models.deletion.SET_NULL, related_name='auto_evaluations_user', to=settings.AUTH_USER_MODEL),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='valuation',
|
||||||
|
field=models.OneToOneField(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='auto_detail', to='evaluation.valuationmodel'),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='autoevaluationmodel',
|
||||||
|
name='vehicle',
|
||||||
|
field=models.OneToOneField(blank=True, null=True, on_delete=django.db.models.deletion.CASCADE, related_name='evaluation', to='evaluation.vehiclemodel'),
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,49 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-28 11:41
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0036_remove_autoevaluationmodel_form_ownership_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='customer_and_owner_same',
|
||||||
|
field=models.BooleanField(default=False),
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='EvaluationRequestCustomerModel',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('type', models.CharField(choices=[('individual_person', 'Jismoniy shaxs'), ('legal_person', 'Yuridik shaxs')], max_length=100)),
|
||||||
|
('jshshir', models.CharField(max_length=100)),
|
||||||
|
('evaluation_request', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, related_name='customer', to='evaluation.evaluationrequestmodel')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Evaluation Request Customer',
|
||||||
|
'verbose_name_plural': 'Evaluation Request Customers',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='EvaluationRequestOwnerModel',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('type', models.CharField(choices=[('individual_person', 'Jismoniy shaxs'), ('legal_person', 'Yuridik shaxs')], max_length=100)),
|
||||||
|
('jshshir', models.CharField(max_length=100)),
|
||||||
|
('evaluation_request', models.OneToOneField(on_delete=django.db.models.deletion.CASCADE, related_name='owner', to='evaluation.evaluationrequestmodel')),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'verbose_name': 'Evaluation Request Owner',
|
||||||
|
'verbose_name_plural': 'Evaluation Request Owners',
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,88 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-04-28 11:47
|
||||||
|
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0037_evaluationrequestmodel_customer_and_owner_same_and_more'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='distance_covered',
|
||||||
|
field=models.FloatField(blank=True, default=0.0, null=True),
|
||||||
|
),
|
||||||
|
migrations.AddField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='gov_number',
|
||||||
|
field=models.CharField(blank=True, max_length=100, null=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='chassi',
|
||||||
|
field=models.CharField(blank=True, max_length=100, null=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='customer_inn_number',
|
||||||
|
field=models.CharField(max_length=20),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='is_archive',
|
||||||
|
field=models.BooleanField(default=False),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='location_lat',
|
||||||
|
field=models.DecimalField(blank=True, decimal_places=6, max_digits=9, null=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='location_lng',
|
||||||
|
field=models.DecimalField(blank=True, decimal_places=6, max_digits=9, null=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='location_name',
|
||||||
|
field=models.CharField(blank=True, max_length=255, null=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='need_delivering',
|
||||||
|
field=models.BooleanField(default=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='object_type',
|
||||||
|
field=models.CharField(blank=True, choices=[('lightweight_auto', 'Yengil automobil'), ('truck_car', 'Yuk automobil'), ('special_tech', 'Maxsus texnika')], max_length=50, null=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='owner_inn_number',
|
||||||
|
field=models.CharField(max_length=20),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='rate_type',
|
||||||
|
field=models.CharField(choices=[('auto', 'Automobil'), ('real_estate', "Ko'chmas mulk"), ('equipment', 'Uskuna')], max_length=50),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='status',
|
||||||
|
field=models.CharField(choices=[('pending', 'Kutilmoqda'), ('in_progress', 'Jarayonda'), ('completed', 'Bajarildi'), ('rejected', 'Rad etildi')], default='pending', max_length=50),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='tex_passport',
|
||||||
|
field=models.CharField(blank=True, max_length=20, null=True),
|
||||||
|
),
|
||||||
|
migrations.AlterField(
|
||||||
|
model_name='evaluationrequestmodel',
|
||||||
|
name='worked_hours',
|
||||||
|
field=models.IntegerField(blank=True, null=True),
|
||||||
|
),
|
||||||
|
]
|
||||||
31
core/apps/evaluation/migrations/0039_bonus.py
Normal file
31
core/apps/evaluation/migrations/0039_bonus.py
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-05-01 06:45
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0038_evaluationrequestmodel_distance_covered_and_more'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='Bonus',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('bonus_type', models.CharField(choices=[('lightweight_auto', 'Yengil automobil'), ('truck_car', 'Yuk automobil'), ('special_tech', 'Maxsus texnika')], max_length=50)),
|
||||||
|
('percentage', models.FloatField()),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('price', models.FloatField()),
|
||||||
|
('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='bonuses', to=settings.AUTH_USER_MODEL)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,59 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-05-01 11:43
|
||||||
|
|
||||||
|
import django.db.models.deletion
|
||||||
|
from django.conf import settings
|
||||||
|
from django.db import migrations, models
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0039_bonus'),
|
||||||
|
migrations.swappable_dependency(settings.AUTH_USER_MODEL),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='BaseValueBonus',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('base_price', models.DecimalField(decimal_places=2, max_digits=12)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='BonusType',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('name', models.CharField(max_length=255)),
|
||||||
|
('category', models.CharField(choices=[('auto_transport', 'Avtotransport'), ('real estate', "ko'chmas mulk"), ('equipment', 'uskuna va jihozlar')], max_length=50)),
|
||||||
|
('percentage', models.PositiveIntegerField()),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'abstract': False,
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.CreateModel(
|
||||||
|
name='EmployeeBonus',
|
||||||
|
fields=[
|
||||||
|
('id', models.BigAutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
|
||||||
|
('created_at', models.DateTimeField(auto_now_add=True)),
|
||||||
|
('updated_at', models.DateTimeField(auto_now=True)),
|
||||||
|
('percentage', models.PositiveIntegerField()),
|
||||||
|
('bonus_type', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, to='evaluation.bonustype')),
|
||||||
|
('user', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='bonuses', to=settings.AUTH_USER_MODEL)),
|
||||||
|
],
|
||||||
|
options={
|
||||||
|
'unique_together': {('user', 'bonus_type')},
|
||||||
|
},
|
||||||
|
),
|
||||||
|
migrations.DeleteModel(
|
||||||
|
name='Bonus',
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
# Generated by Django 5.2.7 on 2026-05-01 12:06
|
||||||
|
|
||||||
|
from django.db import migrations
|
||||||
|
|
||||||
|
|
||||||
|
class Migration(migrations.Migration):
|
||||||
|
|
||||||
|
dependencies = [
|
||||||
|
('evaluation', '0040_basevaluebonus_bonustype_employeebonus_delete_bonus'),
|
||||||
|
]
|
||||||
|
|
||||||
|
operations = [
|
||||||
|
migrations.RenameModel(
|
||||||
|
old_name='BonusType',
|
||||||
|
new_name='BonusCategory',
|
||||||
|
),
|
||||||
|
]
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
@@ -10,3 +11,4 @@ from .report import * # noqa
|
|||||||
from .request import * # noqa
|
from .request import * # noqa
|
||||||
from .valuation import * # noqa
|
from .valuation import * # noqa
|
||||||
from .vehicle import * # noqa
|
from .vehicle import * # noqa
|
||||||
|
from .certificate import * # noqa
|
||||||
|
|||||||
@@ -9,24 +9,34 @@ from core.apps.evaluation.choices.auto import (
|
|||||||
AutoEvaluationStatus,
|
AutoEvaluationStatus,
|
||||||
AutoObjectType,
|
AutoObjectType,
|
||||||
# FormOwnership,
|
# FormOwnership,
|
||||||
LocationConvenience,
|
|
||||||
LocationHighways,
|
|
||||||
ObjectOwnerType,
|
ObjectOwnerType,
|
||||||
# PropertyRights,
|
# PropertyRights,
|
||||||
# RateType,
|
# RateType,
|
||||||
# ValueDetermined,
|
# ValueDetermined,
|
||||||
)
|
)
|
||||||
|
|
||||||
from .valuation import ValuationModel
|
from .valuation import ValuationModel
|
||||||
from .vehicle import VehicleModel
|
from .vehicle import VehicleModel
|
||||||
|
|
||||||
|
|
||||||
class AutoEvaluationModel(AbstractBaseModel):
|
class AutoEvaluationModel(AbstractBaseModel):
|
||||||
|
user = models.ForeignKey(
|
||||||
|
"accounts.User",
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name="auto_evaluations_user",
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
|
evaluation_request = models.ForeignKey(
|
||||||
|
"evaluation.EvaluationRequestModel",
|
||||||
|
on_delete=models.SET_NULL,
|
||||||
|
related_name="auto_evaluations_request",
|
||||||
|
null=True,
|
||||||
|
blank=True,
|
||||||
|
)
|
||||||
valuation = models.OneToOneField(
|
valuation = models.OneToOneField(
|
||||||
ValuationModel,
|
ValuationModel,
|
||||||
on_delete=models.CASCADE,
|
on_delete=models.CASCADE,
|
||||||
related_name="auto_detail",
|
related_name="auto_detail",
|
||||||
verbose_name=_("valuation"),
|
|
||||||
null=True,
|
null=True,
|
||||||
blank=True,
|
blank=True,
|
||||||
)
|
)
|
||||||
@@ -34,10 +44,22 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
VehicleModel,
|
VehicleModel,
|
||||||
on_delete=models.CASCADE,
|
on_delete=models.CASCADE,
|
||||||
related_name="evaluation",
|
related_name="evaluation",
|
||||||
verbose_name=_("vehicle"),
|
|
||||||
null=True,
|
null=True,
|
||||||
blank=True,
|
blank=True,
|
||||||
)
|
)
|
||||||
|
appraisers = models.ManyToManyField(
|
||||||
|
"accounts.User",
|
||||||
|
verbose_name=_("appraisers"),
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
tex_passport_file = models.FileField(
|
||||||
|
verbose_name=_("tech passport file"),
|
||||||
|
upload_to="quick_evaluation/tech_passports/%Y/%m/",
|
||||||
|
blank=True,
|
||||||
|
null=True,
|
||||||
|
)
|
||||||
|
|
||||||
# ── Step 1 — Umumiy ma'lumotlar ──────────────────────────────────
|
# ── Step 1 — Umumiy ma'lumotlar ──────────────────────────────────
|
||||||
registration_number = models.CharField(
|
registration_number = models.CharField(
|
||||||
@@ -66,12 +88,6 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
rate_object_name = models.CharField(
|
|
||||||
verbose_name=_("rate object name"),
|
|
||||||
max_length=255,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_type = models.CharField(
|
object_type = models.CharField(
|
||||||
verbose_name=_("object type"),
|
verbose_name=_("object type"),
|
||||||
max_length=50,
|
max_length=50,
|
||||||
@@ -123,23 +139,6 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
property_rights = models.ForeignKey(
|
|
||||||
'evaluation.ReferenceitemModel',
|
|
||||||
verbose_name=_("property rights"),
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
on_delete=models.SET_NULL,
|
|
||||||
related_name='evaluation_auto_property_rights'
|
|
||||||
)
|
|
||||||
form_ownership = models.ForeignKey(
|
|
||||||
'evaluation.ReferenceitemModel',
|
|
||||||
verbose_name=_("form of ownership"),
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
on_delete=models.SET_NULL,
|
|
||||||
related_name='evaluation_auto_form_ownership'
|
|
||||||
)
|
|
||||||
value_determined = models.ForeignKey(
|
value_determined = models.ForeignKey(
|
||||||
'evaluation.ReferenceitemModel',
|
'evaluation.ReferenceitemModel',
|
||||||
verbose_name=_("value determined"),
|
verbose_name=_("value determined"),
|
||||||
@@ -157,56 +156,6 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
related_name='evaluation_auto_rate_type'
|
related_name='evaluation_auto_rate_type'
|
||||||
)
|
)
|
||||||
|
|
||||||
# ── Step 3 — Manzil ma'lumotlari ────────────────────────────────
|
|
||||||
object_location_province = models.CharField(
|
|
||||||
verbose_name=_("object location province"),
|
|
||||||
max_length=100,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_location_district = models.CharField(
|
|
||||||
verbose_name=_("object location district"),
|
|
||||||
max_length=100,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_location_city = models.CharField(
|
|
||||||
verbose_name=_("object location city"),
|
|
||||||
max_length=100,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_location_neighborhood = models.CharField(
|
|
||||||
verbose_name=_("object location neighborhood"),
|
|
||||||
max_length=100,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_location_street = models.CharField(
|
|
||||||
verbose_name=_("object location street"),
|
|
||||||
max_length=100,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_location_home = models.CharField(
|
|
||||||
verbose_name=_("object location home"),
|
|
||||||
max_length=50,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_location_highways = models.IntegerField(
|
|
||||||
verbose_name=_("location highways"),
|
|
||||||
choices=LocationHighways.choices,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
object_location_covenience = models.IntegerField(
|
|
||||||
verbose_name=_("location convenience"),
|
|
||||||
choices=LocationConvenience.choices,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
# ── Step 4 — Avtomobil ma'lumotlari ─────────────────────────────
|
# ── Step 4 — Avtomobil ma'lumotlari ─────────────────────────────
|
||||||
tex_passport_serie_num = models.CharField(
|
tex_passport_serie_num = models.CharField(
|
||||||
verbose_name=_("tech passport series and number"),
|
verbose_name=_("tech passport series and number"),
|
||||||
@@ -287,6 +236,10 @@ class AutoEvaluationModel(AbstractBaseModel):
|
|||||||
choices=AutoEvaluationStatus.choices,
|
choices=AutoEvaluationStatus.choices,
|
||||||
default=AutoEvaluationStatus.CREATED,
|
default=AutoEvaluationStatus.CREATED,
|
||||||
)
|
)
|
||||||
|
is_archived = models.BooleanField(
|
||||||
|
verbose_name=_("is archived"),
|
||||||
|
default=False,
|
||||||
|
)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"Auto Evaluation {self.registration_number or self.pk}"
|
return f"Auto Evaluation {self.registration_number or self.pk}"
|
||||||
|
|||||||
33
core/apps/evaluation/models/bonus.py
Normal file
33
core/apps/evaluation/models/bonus.py
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
from django.db import models
|
||||||
|
from django.db.models.fields import PositiveIntegerField
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
|
||||||
|
from core.apps.evaluation.choices.bonus import EvaluationCategory
|
||||||
|
|
||||||
|
|
||||||
|
class BaseValueBonus(AbstractBaseModel):
|
||||||
|
base_price = models.DecimalField(max_digits=12, decimal_places=2)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"Base: {self.base_price}"
|
||||||
|
|
||||||
|
|
||||||
|
class BonusCategory(AbstractBaseModel):
|
||||||
|
name = models.CharField(max_length=255)
|
||||||
|
category = models.CharField(
|
||||||
|
max_length=50,
|
||||||
|
choices=EvaluationCategory.choices
|
||||||
|
)
|
||||||
|
percentage = PositiveIntegerField()
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.name
|
||||||
|
|
||||||
|
|
||||||
|
class EmployeeBonus(AbstractBaseModel):
|
||||||
|
user = models.ForeignKey("accounts.User", on_delete=models.CASCADE, related_name="bonuses", )
|
||||||
|
bonus_type = models.ForeignKey(BonusCategory, on_delete=models.CASCADE)
|
||||||
|
percentage = models.PositiveIntegerField()
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
unique_together = ("user", "bonus_type")
|
||||||
30
core/apps/evaluation/models/certificate.py
Normal file
30
core/apps/evaluation/models/certificate.py
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
from django.db import models
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from model_bakery import baker
|
||||||
|
|
||||||
|
|
||||||
|
class CertificateModel(AbstractBaseModel):
|
||||||
|
title = models.CharField(
|
||||||
|
verbose_name=_("title"),
|
||||||
|
max_length=255
|
||||||
|
)
|
||||||
|
|
||||||
|
file = models.FileField(
|
||||||
|
verbose_name=_("file"),
|
||||||
|
upload_to="certificates/",
|
||||||
|
blank=True,
|
||||||
|
null=True
|
||||||
|
)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return self.title
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _baker(cls):
|
||||||
|
return baker.make(cls)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
db_table = "certificate"
|
||||||
|
verbose_name = _("Certificate")
|
||||||
|
verbose_name_plural = _("Certificates")
|
||||||
@@ -3,9 +3,10 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
from django_core.models import AbstractBaseModel
|
from django_core.models import AbstractBaseModel
|
||||||
from model_bakery import baker
|
from model_bakery import baker
|
||||||
|
|
||||||
from .valuation import ValuationModel
|
|
||||||
from core.apps.evaluation.choices.document import DocumentType
|
from core.apps.evaluation.choices.document import DocumentType
|
||||||
|
|
||||||
|
from .valuation import ValuationModel
|
||||||
|
|
||||||
|
|
||||||
class ValuationDocumentModel(AbstractBaseModel):
|
class ValuationDocumentModel(AbstractBaseModel):
|
||||||
valuation = models.ForeignKey(
|
valuation = models.ForeignKey(
|
||||||
@@ -54,3 +55,35 @@ class ValuationDocumentModel(AbstractBaseModel):
|
|||||||
verbose_name = _("Valuation Document")
|
verbose_name = _("Valuation Document")
|
||||||
verbose_name_plural = _("Valuation Documents")
|
verbose_name_plural = _("Valuation Documents")
|
||||||
ordering = ["-created_at"]
|
ordering = ["-created_at"]
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentModel(AbstractBaseModel):
|
||||||
|
auto_evaluation = models.ForeignKey(
|
||||||
|
"evaluation.AutoEvaluationModel",
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="documents",
|
||||||
|
verbose_name=_("auto evaluation"),
|
||||||
|
)
|
||||||
|
title = models.CharField(verbose_name=_("title"), max_length=255)
|
||||||
|
document = models.FileField(
|
||||||
|
verbose_name=_("document"),
|
||||||
|
upload_to="evaluation/documents/%Y/%m/",
|
||||||
|
)
|
||||||
|
category = models.ForeignKey(
|
||||||
|
"evaluation.DocumentCategoryModel",
|
||||||
|
on_delete=models.CASCADE,
|
||||||
|
related_name="documents",
|
||||||
|
verbose_name=_("category"),
|
||||||
|
)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return str(self.pk)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _baker(cls):
|
||||||
|
return baker.make(cls)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
db_table = "Document"
|
||||||
|
verbose_name = _("DocumentModel")
|
||||||
|
verbose_name_plural = _("DocumentModels")
|
||||||
|
|||||||
21
core/apps/evaluation/models/documentcategory.py
Normal file
21
core/apps/evaluation/models/documentcategory.py
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
from django.db import models
|
||||||
|
from django.utils.translation import gettext_lazy as _
|
||||||
|
from django_core.models import AbstractBaseModel
|
||||||
|
from model_bakery import baker
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentcategoryModel(AbstractBaseModel):
|
||||||
|
label = models.CharField(verbose_name=_("label"), max_length=255)
|
||||||
|
value = models.CharField(verbose_name=_("value"), max_length=255)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return str(self.pk)
|
||||||
|
|
||||||
|
@classmethod
|
||||||
|
def _baker(cls):
|
||||||
|
return baker.make(cls)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
db_table = "DocumentCategory"
|
||||||
|
verbose_name = _("DocumentcategoryModel")
|
||||||
|
verbose_name_plural = _("DocumentcategoryModels")
|
||||||
@@ -3,12 +3,11 @@ from django.utils.translation import gettext_lazy as _
|
|||||||
from django_core.models import AbstractBaseModel
|
from django_core.models import AbstractBaseModel
|
||||||
from model_bakery import baker
|
from model_bakery import baker
|
||||||
|
|
||||||
|
|
||||||
from .valuation import ValuationModel
|
|
||||||
from core.apps.evaluation.choices.movable import (
|
from core.apps.evaluation.choices.movable import (
|
||||||
MovablePropertyCategory,
|
MovablePropertyCategory,
|
||||||
MovablePropertyCondition,
|
MovablePropertyCondition,
|
||||||
)
|
)
|
||||||
|
from .valuation import ValuationModel
|
||||||
|
|
||||||
|
|
||||||
class MovablePropertyEvaluationModel(AbstractBaseModel):
|
class MovablePropertyEvaluationModel(AbstractBaseModel):
|
||||||
@@ -51,4 +50,3 @@ class MovablePropertyEvaluationModel(AbstractBaseModel):
|
|||||||
db_table = "MovablePropertyEvaluation"
|
db_table = "MovablePropertyEvaluation"
|
||||||
verbose_name = _("Movable Property Evaluation")
|
verbose_name = _("Movable Property Evaluation")
|
||||||
verbose_name_plural = _("Movable Property Evaluations")
|
verbose_name_plural = _("Movable Property Evaluations")
|
||||||
|
|
||||||
|
|||||||
@@ -34,12 +34,6 @@ class QuickEvaluationModel(AbstractBaseModel):
|
|||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
tex_passport_file = models.FileField(
|
|
||||||
verbose_name=_("tech passport file"),
|
|
||||||
upload_to="quick_evaluation/tech_passports/%Y/%m/",
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
|
|
||||||
# Car info
|
# Car info
|
||||||
car_type = models.CharField(
|
car_type = models.CharField(
|
||||||
@@ -138,6 +132,11 @@ class QuickEvaluationModel(AbstractBaseModel):
|
|||||||
default=QuickEvaluationStatus.CREATED,
|
default=QuickEvaluationStatus.CREATED,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
is_archive = models.BooleanField(
|
||||||
|
verbose_name=_("is archive"),
|
||||||
|
default=False,
|
||||||
|
)
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"Quick Evaluation {self.pk} by {self.created_by}"
|
return f"Quick Evaluation {self.pk} by {self.created_by}"
|
||||||
|
|
||||||
|
|||||||
@@ -8,43 +8,41 @@ from core.apps.evaluation.choices.request import (
|
|||||||
EvaluationRateType,
|
EvaluationRateType,
|
||||||
RequestObjectType,
|
RequestObjectType,
|
||||||
RequestStatus,
|
RequestStatus,
|
||||||
|
RequestPersonType,
|
||||||
)
|
)
|
||||||
from core.apps.evaluation.models import ReferenceitemModel
|
from core.apps.evaluation.models import ReferenceitemModel
|
||||||
|
|
||||||
|
|
||||||
class EvaluationrequestModel(AbstractBaseModel):
|
class EvaluationrequestModel(AbstractBaseModel):
|
||||||
|
rate_type = models.CharField(max_length=50,choices=EvaluationRateType.choices)
|
||||||
|
object_type = models.CharField(max_length=50,choices=RequestObjectType.choices,blank=True,null=True)
|
||||||
|
status = models.CharField(max_length=50, choices=RequestStatus.choices, default=RequestStatus.PENDING)
|
||||||
|
|
||||||
|
distance_covered = models.FloatField(default=0.0, null=True, blank=True)
|
||||||
|
worked_hours = models.IntegerField(blank=True,null=True)
|
||||||
|
customer_inn_number = models.CharField(max_length=20)
|
||||||
|
owner_inn_number = models.CharField(max_length=20)
|
||||||
|
tex_passport = models.CharField(max_length=20,blank=True,null=True)
|
||||||
|
chassi = models.CharField(max_length=100,blank=True,null=True)
|
||||||
|
gov_number = models.CharField(max_length=100, null=True, blank=True)
|
||||||
|
|
||||||
|
location_name = models.CharField(max_length=255,blank=True,null=True)
|
||||||
|
location_lat = models.DecimalField(max_digits=9,decimal_places=6,blank=True, null=True)
|
||||||
|
location_lng = models.DecimalField(max_digits=9,decimal_places=6,blank=True,null=True)
|
||||||
|
|
||||||
|
need_delivering = models.BooleanField(default=True)
|
||||||
|
is_archive = models.BooleanField(default=False)
|
||||||
|
customer_and_owner_same = models.BooleanField(default=False)
|
||||||
|
|
||||||
|
###################
|
||||||
|
# Foreign Keys
|
||||||
|
###################
|
||||||
user = models.ForeignKey(
|
user = models.ForeignKey(
|
||||||
"accounts.User",
|
"accounts.User",
|
||||||
on_delete=models.CASCADE,
|
on_delete=models.CASCADE,
|
||||||
related_name="evaluation_requests",
|
related_name="evaluation_requests",
|
||||||
verbose_name=_("user"),
|
verbose_name=_("user"),
|
||||||
)
|
)
|
||||||
rate_type = models.CharField(
|
|
||||||
verbose_name=_("rate type"),
|
|
||||||
max_length=50,
|
|
||||||
choices=EvaluationRateType.choices,
|
|
||||||
)
|
|
||||||
object_type = models.CharField(
|
|
||||||
verbose_name=_("object type"),
|
|
||||||
max_length=50,
|
|
||||||
choices=RequestObjectType.choices,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
customer_inn_number = models.CharField(
|
|
||||||
verbose_name=_("customer INN number"),
|
|
||||||
max_length=20,
|
|
||||||
)
|
|
||||||
owner_inn_number = models.CharField(
|
|
||||||
verbose_name=_("owner INN number"),
|
|
||||||
max_length=20,
|
|
||||||
)
|
|
||||||
tex_passport = models.CharField(
|
|
||||||
verbose_name=_("tex passport"),
|
|
||||||
max_length=20,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
value_determined = models.ForeignKey(
|
value_determined = models.ForeignKey(
|
||||||
"evaluation.ReferenceitemModel",
|
"evaluation.ReferenceitemModel",
|
||||||
verbose_name=_("value determined"),
|
verbose_name=_("value determined"),
|
||||||
@@ -77,50 +75,10 @@ class EvaluationrequestModel(AbstractBaseModel):
|
|||||||
blank=True,
|
blank=True,
|
||||||
null=True,
|
null=True,
|
||||||
)
|
)
|
||||||
worked_hours = models.IntegerField(
|
|
||||||
verbose_name=_("worked hours"),
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
chassi = models.CharField(
|
|
||||||
verbose_name=_("chassi"),
|
|
||||||
max_length=100,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
need_delivering = models.BooleanField(
|
|
||||||
verbose_name=_("need delivering"),
|
|
||||||
default=True,
|
|
||||||
)
|
|
||||||
location_name = models.CharField(
|
|
||||||
verbose_name=_("location name"),
|
|
||||||
max_length=255,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
location_lat = models.DecimalField(
|
|
||||||
verbose_name=_("location latitude"),
|
|
||||||
max_digits=9,
|
|
||||||
decimal_places=6,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
location_lng = models.DecimalField(
|
|
||||||
verbose_name=_("location longitude"),
|
|
||||||
max_digits=9,
|
|
||||||
decimal_places=6,
|
|
||||||
blank=True,
|
|
||||||
null=True,
|
|
||||||
)
|
|
||||||
status = models.CharField(
|
|
||||||
verbose_name=_("status"),
|
|
||||||
max_length=50,
|
|
||||||
choices=RequestStatus.choices,
|
|
||||||
default=RequestStatus.PENDING,
|
|
||||||
)
|
|
||||||
|
|
||||||
def __str__(self):
|
def __str__(self):
|
||||||
return f"Request #{self.pk} — {self.get_rate_type_display()}"
|
return f"Requests #{self.pk} — {self.get_rate_type_display()}"
|
||||||
|
|
||||||
@classmethod
|
@classmethod
|
||||||
def _baker(cls):
|
def _baker(cls):
|
||||||
@@ -144,3 +102,29 @@ class EvaluationrequestModel(AbstractBaseModel):
|
|||||||
db_table = "EvaluationRequest"
|
db_table = "EvaluationRequest"
|
||||||
verbose_name = _("Evaluation Request")
|
verbose_name = _("Evaluation Request")
|
||||||
verbose_name_plural = _("Evaluation Requests")
|
verbose_name_plural = _("Evaluation Requests")
|
||||||
|
|
||||||
|
|
||||||
|
class EvaluationRequestOwnerModel(AbstractBaseModel):
|
||||||
|
evaluation_request = models.OneToOneField(EvaluationrequestModel, on_delete=models.CASCADE, related_name='owner')
|
||||||
|
type = models.CharField(max_length=100, choices=RequestPersonType.choices)
|
||||||
|
jshshir = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"Owner #{self.pk} — {self.type}"
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _("Evaluation Request Owner")
|
||||||
|
verbose_name_plural = _("Evaluation Request Owners")
|
||||||
|
|
||||||
|
|
||||||
|
class EvaluationRequestCustomerModel(AbstractBaseModel):
|
||||||
|
evaluation_request = models.OneToOneField(EvaluationrequestModel, on_delete=models.CASCADE, related_name='customer')
|
||||||
|
type = models.CharField(max_length=100, choices=RequestPersonType.choices)
|
||||||
|
jshshir = models.CharField(max_length=100)
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return f"Customer #{self.pk} — {self.type}"
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
verbose_name = _("Evaluation Request Customer")
|
||||||
|
verbose_name_plural = _("Evaluation Request Customers")
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -10,3 +10,14 @@ class ValuationdocumentPermission(permissions.BasePermission):
|
|||||||
|
|
||||||
def has_permission(self, request, view):
|
def has_permission(self, request, view):
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentPermission(permissions.BasePermission):
|
||||||
|
|
||||||
|
def __init__(self) -> None: ...
|
||||||
|
|
||||||
|
def __call__(self, *args, **kwargs):
|
||||||
|
return self
|
||||||
|
|
||||||
|
def has_permission(self, request, view):
|
||||||
|
return True
|
||||||
|
|||||||
12
core/apps/evaluation/permissions/documentcategory.py
Normal file
12
core/apps/evaluation/permissions/documentcategory.py
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
from rest_framework import permissions
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentcategoryPermission(permissions.BasePermission):
|
||||||
|
|
||||||
|
def __init__(self) -> None: ...
|
||||||
|
|
||||||
|
def __call__(self, *args, **kwargs):
|
||||||
|
return self
|
||||||
|
|
||||||
|
def has_permission(self, request, view):
|
||||||
|
return True
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
@@ -10,3 +11,5 @@ from .report import * # noqa
|
|||||||
from .request import * # noqa
|
from .request import * # noqa
|
||||||
from .valuation import * # noqa
|
from .valuation import * # noqa
|
||||||
from .vehicle import * # noqa
|
from .vehicle import * # noqa
|
||||||
|
from .tech_passport import * # noqa
|
||||||
|
from .certificate import * # noqa
|
||||||
|
|||||||
@@ -1,24 +1,39 @@
|
|||||||
import re
|
import re
|
||||||
|
|
||||||
|
from django.contrib.auth import get_user_model
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
from core.apps.evaluation.models import AutoEvaluationModel,ReferenceitemModel
|
from core.apps.evaluation.choices.request import RequestStatus
|
||||||
|
from core.apps.evaluation.models import AutoEvaluationModel, ReferenceitemModel, EvaluationrequestModel
|
||||||
from core.apps.evaluation.serializers.reference import ListReferenceitemSerializer
|
from core.apps.evaluation.serializers.reference import ListReferenceitemSerializer
|
||||||
|
|
||||||
|
User = get_user_model()
|
||||||
|
|
||||||
|
|
||||||
class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
status_display = serializers.CharField(source="get_status_display", read_only=True)
|
status_display = serializers.CharField(source="get_status_display", read_only=True)
|
||||||
object_type_display = serializers.CharField(source="get_object_type_display", read_only=True, default=None)
|
object_type_display = serializers.CharField(source="get_object_type_display", read_only=True, default=None)
|
||||||
object_owner_type_display = serializers.CharField(source="get_object_owner_type_display", read_only=True, default=None)
|
object_owner_type_display = serializers.CharField(source="get_object_owner_type_display", read_only=True,
|
||||||
|
default=None)
|
||||||
rate_type = ListReferenceitemSerializer(read_only=True)
|
rate_type = ListReferenceitemSerializer(read_only=True)
|
||||||
value_determined = ListReferenceitemSerializer(read_only=True)
|
value_determined = ListReferenceitemSerializer(read_only=True)
|
||||||
property_rights = ListReferenceitemSerializer(read_only=True)
|
user = serializers.SerializerMethodField(method_name="get_user", read_only=True)
|
||||||
form_ownership = ListReferenceitemSerializer(read_only=True)
|
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = AutoEvaluationModel
|
model = AutoEvaluationModel
|
||||||
fields = [
|
fields = [
|
||||||
"id",
|
"id",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"rating_goal",
|
||||||
"registration_number",
|
"registration_number",
|
||||||
"object_type",
|
"object_type",
|
||||||
"object_type_display",
|
"object_type_display",
|
||||||
@@ -32,10 +47,20 @@ class BaseAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"created_at",
|
"created_at",
|
||||||
"value_determined",
|
"value_determined",
|
||||||
"rate_type",
|
"rate_type",
|
||||||
"property_rights",
|
"user",
|
||||||
"form_ownership",
|
"evaluation_request",
|
||||||
]
|
]
|
||||||
|
|
||||||
|
def get_user(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
return {
|
||||||
|
"id": obj.user.id,
|
||||||
|
"phone": obj.user.phone,
|
||||||
|
"first_name": obj.user.first_name,
|
||||||
|
"last_name": obj.user.last_name,
|
||||||
|
"avatar": request.build_absolute_uri(obj.user.avatar.url) if obj.user.avatar else None
|
||||||
|
} if obj.user else None
|
||||||
|
|
||||||
|
|
||||||
class ListAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
class ListAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
||||||
class Meta(BaseAutoevaluationSerializer.Meta):
|
class Meta(BaseAutoevaluationSerializer.Meta):
|
||||||
@@ -45,12 +70,6 @@ class ListAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
|||||||
class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
||||||
car_type_display = serializers.CharField(source="get_car_type_display", read_only=True, default=None)
|
car_type_display = serializers.CharField(source="get_car_type_display", read_only=True, default=None)
|
||||||
car_wheel_display = serializers.CharField(source="get_car_wheel_display", read_only=True, default=None)
|
car_wheel_display = serializers.CharField(source="get_car_wheel_display", read_only=True, default=None)
|
||||||
object_location_highways_display = serializers.CharField(
|
|
||||||
source="get_object_location_highways_display", read_only=True, default=None
|
|
||||||
)
|
|
||||||
object_location_covenience_display = serializers.CharField(
|
|
||||||
source="get_object_location_covenience_display", read_only=True, default=None
|
|
||||||
)
|
|
||||||
|
|
||||||
class Meta(BaseAutoevaluationSerializer.Meta):
|
class Meta(BaseAutoevaluationSerializer.Meta):
|
||||||
fields = BaseAutoevaluationSerializer.Meta.fields + [
|
fields = BaseAutoevaluationSerializer.Meta.fields + [
|
||||||
@@ -59,7 +78,6 @@ class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
|||||||
"object_inspection_date",
|
"object_inspection_date",
|
||||||
"rate_date",
|
"rate_date",
|
||||||
"rate_report_date",
|
"rate_report_date",
|
||||||
"rate_object_name",
|
|
||||||
# Step 2
|
# Step 2
|
||||||
"object_owner_type",
|
"object_owner_type",
|
||||||
"object_owner_type_display",
|
"object_owner_type_display",
|
||||||
@@ -69,19 +87,9 @@ class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
|||||||
"object_owner_individual_person_passport_num",
|
"object_owner_individual_person_passport_num",
|
||||||
"object_owner_legal_entity",
|
"object_owner_legal_entity",
|
||||||
"object_owner_legal_inn",
|
"object_owner_legal_inn",
|
||||||
# Step 3
|
|
||||||
"object_location_province",
|
|
||||||
"object_location_district",
|
|
||||||
"object_location_city",
|
|
||||||
"object_location_neighborhood",
|
|
||||||
"object_location_street",
|
|
||||||
"object_location_home",
|
|
||||||
"object_location_highways",
|
|
||||||
"object_location_highways_display",
|
|
||||||
"object_location_covenience",
|
|
||||||
"object_location_covenience_display",
|
|
||||||
# Step 4
|
# Step 4
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_file",
|
||||||
"tex_passport_gived_date",
|
"tex_passport_gived_date",
|
||||||
"tex_passport_gived_location",
|
"tex_passport_gived_location",
|
||||||
"car_type",
|
"car_type",
|
||||||
@@ -97,22 +105,12 @@ class RetrieveAutoevaluationSerializer(BaseAutoevaluationSerializer):
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
class UpdateAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
property_rights = serializers.PrimaryKeyRelatedField(
|
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
|
||||||
required=False,
|
|
||||||
allow_null=True,
|
|
||||||
)
|
|
||||||
value_determined = serializers.PrimaryKeyRelatedField(
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
required=False,
|
required=False,
|
||||||
allow_null=True,
|
allow_null=True,
|
||||||
)
|
)
|
||||||
form_ownership = serializers.PrimaryKeyRelatedField(
|
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
|
||||||
required=False,
|
|
||||||
allow_null=True,
|
|
||||||
)
|
|
||||||
value_determined = serializers.PrimaryKeyRelatedField(
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
queryset=ReferenceitemModel.objects.all(),
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
required=False,
|
required=False,
|
||||||
@@ -124,7 +122,6 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
allow_null=True,
|
allow_null=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = AutoEvaluationModel
|
model = AutoEvaluationModel
|
||||||
fields = [
|
fields = [
|
||||||
@@ -134,7 +131,6 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"object_inspection_date",
|
"object_inspection_date",
|
||||||
"rate_date",
|
"rate_date",
|
||||||
"rate_report_date",
|
"rate_report_date",
|
||||||
"rate_object_name",
|
|
||||||
"object_type",
|
"object_type",
|
||||||
# Step 2
|
# Step 2
|
||||||
"object_owner_type",
|
"object_owner_type",
|
||||||
@@ -144,20 +140,10 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"object_owner_individual_person_passport_num",
|
"object_owner_individual_person_passport_num",
|
||||||
"object_owner_legal_entity",
|
"object_owner_legal_entity",
|
||||||
"object_owner_legal_inn",
|
"object_owner_legal_inn",
|
||||||
"property_rights",
|
|
||||||
"form_ownership",
|
|
||||||
"value_determined",
|
"value_determined",
|
||||||
"rate_type",
|
"rate_type",
|
||||||
# Step 3
|
|
||||||
"object_location_province",
|
|
||||||
"object_location_district",
|
|
||||||
"object_location_city",
|
|
||||||
"object_location_neighborhood",
|
|
||||||
"object_location_street",
|
|
||||||
"object_location_home",
|
|
||||||
"object_location_highways",
|
|
||||||
"object_location_covenience",
|
|
||||||
# Step 4
|
# Step 4
|
||||||
|
"tex_passport_file",
|
||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
"tex_passport_gived_date",
|
"tex_passport_gived_date",
|
||||||
"tex_passport_gived_location",
|
"tex_passport_gived_location",
|
||||||
@@ -212,3 +198,192 @@ class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
|||||||
)
|
)
|
||||||
|
|
||||||
return attrs
|
return attrs
|
||||||
|
|
||||||
|
|
||||||
|
class CreateAutoevaluationSerializer(serializers.ModelSerializer):
|
||||||
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
value_determined = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
rate_type = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=ReferenceitemModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
evaluation_request = serializers.PrimaryKeyRelatedField(
|
||||||
|
queryset=EvaluationrequestModel.objects.all(),
|
||||||
|
required=False,
|
||||||
|
allow_null=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = AutoEvaluationModel
|
||||||
|
fields = [
|
||||||
|
# Step 1
|
||||||
|
"registration_number",
|
||||||
|
"evaluation_request",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"rate_date",
|
||||||
|
"rate_report_date",
|
||||||
|
"object_type",
|
||||||
|
# Step 2
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
# Step 4
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_file",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_type",
|
||||||
|
"car_wheel",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"car_color",
|
||||||
|
]
|
||||||
|
|
||||||
|
def validate_tex_passport_serie_num(self, value):
|
||||||
|
if value and not re.match(r"^[A-Z]{3}\s?\d{7}$", value):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
"Format: AAA 1234567 (3 harf + 7 raqam)"
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def validate_object_owner_individual_person_passport_num(self, value):
|
||||||
|
if value and not re.match(r"^[A-Z]{2}\s?\d{7}$", value):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
"Format: AA 1234567 (2 harf + 7 raqam)"
|
||||||
|
)
|
||||||
|
return value
|
||||||
|
|
||||||
|
def validate(self, attrs):
|
||||||
|
owner_type = attrs.get("object_owner_type")
|
||||||
|
if owner_type == 1:
|
||||||
|
required_fields = {
|
||||||
|
"object_owner_individual_person_f_name": "Ismi",
|
||||||
|
"object_owner_individual_person_l_name": "Familiyasi",
|
||||||
|
"object_owner_individual_person_p_name": "Sharifi",
|
||||||
|
"object_owner_individual_person_passport_num": "Passport raqami",
|
||||||
|
}
|
||||||
|
for field, label in required_fields.items():
|
||||||
|
if not attrs.get(field):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{field: f"Jismoniy shaxs uchun {label} majburiy."}
|
||||||
|
)
|
||||||
|
|
||||||
|
elif owner_type == 2:
|
||||||
|
if not attrs.get("object_owner_legal_entity"):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{"object_owner_legal_entity": "Yuridik shaxs nomi majburiy."}
|
||||||
|
)
|
||||||
|
if not attrs.get("object_owner_legal_inn"):
|
||||||
|
raise serializers.ValidationError(
|
||||||
|
{"object_owner_legal_inn": "INN raqami majburiy."}
|
||||||
|
)
|
||||||
|
|
||||||
|
return attrs
|
||||||
|
|
||||||
|
def create(self, validated_data):
|
||||||
|
user = self.context.get('request').user
|
||||||
|
validated_data['user'] = user
|
||||||
|
evaluation_req = validated_data.get("evaluation_request")
|
||||||
|
if evaluation_req:
|
||||||
|
evaluation_req.status = RequestStatus.IN_PROGRESS
|
||||||
|
evaluation_req.save()
|
||||||
|
return super().create(validated_data)
|
||||||
|
|
||||||
|
|
||||||
|
class AutoEvaluationAppraisersSerializer(serializers.Serializer):
|
||||||
|
ids = serializers.ListField(child=serializers.IntegerField())
|
||||||
|
|
||||||
|
def validate(self, data):
|
||||||
|
if not data.get("ids"):
|
||||||
|
raise serializers.ValidationError("Appraisers IDs are required.")
|
||||||
|
users = User.objects.filter(id__in=data["ids"])
|
||||||
|
if not users:
|
||||||
|
raise serializers.ValidationError("Invalid appraisers IDs.")
|
||||||
|
data['users'] = users
|
||||||
|
return data
|
||||||
|
|
||||||
|
|
||||||
|
class AutoEvaluationSerializer(serializers.Serializer):
|
||||||
|
brand = serializers.CharField()
|
||||||
|
brand_model = serializers.CharField()
|
||||||
|
year = serializers.CharField()
|
||||||
|
color = serializers.CharField()
|
||||||
|
transmission = serializers.CharField()
|
||||||
|
condition = serializers.CharField()
|
||||||
|
fuel_type = serializers.CharField()
|
||||||
|
mileage = serializers.CharField()
|
||||||
|
|
||||||
|
|
||||||
|
class AutoEvaluationModelSerializer(serializers.ModelSerializer):
|
||||||
|
user = serializers.StringRelatedField(read_only=True)
|
||||||
|
appraisers = serializers.PrimaryKeyRelatedField(
|
||||||
|
many=True,
|
||||||
|
queryset=User.objects.all(),
|
||||||
|
required=False
|
||||||
|
)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = AutoEvaluationModel
|
||||||
|
fields = ("tex_passport_file",
|
||||||
|
|
||||||
|
"registration_number",
|
||||||
|
"contract_date",
|
||||||
|
"object_inspection_date",
|
||||||
|
"rate_date",
|
||||||
|
"rate_report_date",
|
||||||
|
"object_type",
|
||||||
|
|
||||||
|
"object_owner_type",
|
||||||
|
"object_owner_individual_person_f_name",
|
||||||
|
"object_owner_individual_person_l_name",
|
||||||
|
"object_owner_individual_person_p_name",
|
||||||
|
"object_owner_individual_person_passport_num",
|
||||||
|
"object_owner_legal_entity",
|
||||||
|
"object_owner_legal_inn",
|
||||||
|
"value_determined",
|
||||||
|
"rate_type",
|
||||||
|
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tex_passport_gived_date",
|
||||||
|
"tex_passport_gived_location",
|
||||||
|
"car_type",
|
||||||
|
"car_wheel",
|
||||||
|
"car_brand",
|
||||||
|
"car_model",
|
||||||
|
"car_number",
|
||||||
|
"manufacture_year",
|
||||||
|
"car_dvigatel_number",
|
||||||
|
"car_color",
|
||||||
|
|
||||||
|
"rating_goal",
|
||||||
|
"status",
|
||||||
|
"is_archived",
|
||||||
|
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|
||||||
|
read_only_fields = (
|
||||||
|
"id",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|||||||
11
core/apps/evaluation/serializers/auto/AvgCost.py
Normal file
11
core/apps/evaluation/serializers/auto/AvgCost.py
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
|
||||||
|
class AvgCostSerializer(serializers.Serializer):
|
||||||
|
brand = serializers.CharField(max_length=100)
|
||||||
|
condition = serializers.CharField(max_length=100)
|
||||||
|
model = serializers.CharField(max_length=100)
|
||||||
|
complication = serializers.CharField(max_length=100)
|
||||||
|
manufacture_date = serializers.DateField()
|
||||||
|
distance_covered = serializers.IntegerField()
|
||||||
|
color = serializers.CharField(max_length=100)
|
||||||
56
core/apps/evaluation/serializers/bonus/Bonus.py
Normal file
56
core/apps/evaluation/serializers/bonus/Bonus.py
Normal file
@@ -0,0 +1,56 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.evaluation.models.bonus import BonusCategory, EmployeeBonus, BaseValueBonus
|
||||||
|
|
||||||
|
|
||||||
|
class BaseBonusSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = BaseValueBonus
|
||||||
|
fields = 'id', 'base_price'
|
||||||
|
|
||||||
|
def create(self, validated_data):
|
||||||
|
if BaseValueBonus.objects.exists():
|
||||||
|
raise serializers.ValidationError("Base bonus already exists")
|
||||||
|
|
||||||
|
return super().create(validated_data)
|
||||||
|
|
||||||
|
|
||||||
|
class BonusCategorySerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = BonusCategory
|
||||||
|
fields = 'name', 'category', 'percentage'
|
||||||
|
|
||||||
|
|
||||||
|
class BonusCategoryListSerializer(serializers.ModelSerializer):
|
||||||
|
price = serializers.DecimalField(max_digits=12, decimal_places=2)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = BonusCategory
|
||||||
|
fields = 'id', 'name', 'category', 'percentage' , 'price'
|
||||||
|
|
||||||
|
def get_price(self, obj):
|
||||||
|
base_obj = BaseValueBonus.objects.first()
|
||||||
|
if not base_obj:
|
||||||
|
return 0
|
||||||
|
|
||||||
|
return (base_obj.base_price * obj.percentage) / 100
|
||||||
|
|
||||||
|
|
||||||
|
class BonusEmployeeBonusSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = EmployeeBonus
|
||||||
|
fields = 'user', 'bonus_type', 'percentage'
|
||||||
|
|
||||||
|
|
||||||
|
class EmployeeBonusListSerializer(serializers.ModelSerializer):
|
||||||
|
price = serializers.DecimalField(max_digits=12, decimal_places=2)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = EmployeeBonus
|
||||||
|
fields = 'id', 'user', 'bonus_type', 'percentage' , 'price'
|
||||||
|
|
||||||
|
def get_price(self, obj):
|
||||||
|
base_obj = BaseValueBonus.objects.first()
|
||||||
|
if not base_obj:
|
||||||
|
return 0
|
||||||
|
return (base_obj.base_price * obj.percentage) / 100
|
||||||
0
core/apps/evaluation/serializers/bonus/__init__.py
Normal file
0
core/apps/evaluation/serializers/bonus/__init__.py
Normal file
1
core/apps/evaluation/serializers/certificate/__init__.py
Normal file
1
core/apps/evaluation/serializers/certificate/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
from .certificate import * # noqa
|
||||||
30
core/apps/evaluation/serializers/certificate/certificate.py
Normal file
30
core/apps/evaluation/serializers/certificate/certificate.py
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
from core.apps.evaluation.models import CertificateModel
|
||||||
|
|
||||||
|
|
||||||
|
class BaseCertificateSerializer(serializers.ModelSerializer):
|
||||||
|
file = serializers.SerializerMethodField(method_name='get_file', read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = CertificateModel
|
||||||
|
fields = [
|
||||||
|
"id",
|
||||||
|
"title",
|
||||||
|
"file",
|
||||||
|
]
|
||||||
|
|
||||||
|
def get_file(self, obj):
|
||||||
|
if obj.file:
|
||||||
|
request = self.context.get('request')
|
||||||
|
return request.build_absolute_uri(obj.file.url)
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class CreateCertificateSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = CertificateModel
|
||||||
|
fields = [
|
||||||
|
"id",
|
||||||
|
"title",
|
||||||
|
"file",
|
||||||
|
]
|
||||||
57
core/apps/evaluation/serializers/document/Document.py
Normal file
57
core/apps/evaluation/serializers/document/Document.py
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentModel, AutoEvaluationModel, DocumentcategoryModel
|
||||||
|
from core.apps.evaluation.serializers.documentcategory import ListDocumentcategorySerializer
|
||||||
|
|
||||||
|
class BaseDocumentSerializer(serializers.ModelSerializer):
|
||||||
|
category = ListDocumentcategorySerializer(read_only=True)
|
||||||
|
|
||||||
|
class Meta:
|
||||||
|
model = DocumentModel
|
||||||
|
fields = [
|
||||||
|
"id",
|
||||||
|
"title",
|
||||||
|
"document",
|
||||||
|
"category",
|
||||||
|
"created_at",
|
||||||
|
"updated_at"
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ListDocumentSerializer(BaseDocumentSerializer):
|
||||||
|
class Meta(BaseDocumentSerializer.Meta): ...
|
||||||
|
|
||||||
|
def get_document(self, obj):
|
||||||
|
request = self.context.get("request")
|
||||||
|
if obj.document:
|
||||||
|
if request:
|
||||||
|
return request.build_absolute_uri(obj.document.url)
|
||||||
|
return obj.document.url
|
||||||
|
return None
|
||||||
|
|
||||||
|
class RetrieveDocumentSerializer(BaseDocumentSerializer):
|
||||||
|
class Meta(BaseDocumentSerializer.Meta): ...
|
||||||
|
|
||||||
|
def get_document(self, obj):
|
||||||
|
request = self.context.get("request")
|
||||||
|
if obj.document:
|
||||||
|
if request:
|
||||||
|
return request.build_absolute_uri(obj.document.url)
|
||||||
|
return obj.document.url
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
class CreateDocumentSerializer(BaseDocumentSerializer):
|
||||||
|
auto_evaluation = serializers.PrimaryKeyRelatedField(queryset=AutoEvaluationModel.objects.all())
|
||||||
|
category = serializers.PrimaryKeyRelatedField(queryset=DocumentcategoryModel.objects.all())
|
||||||
|
|
||||||
|
class Meta(BaseDocumentSerializer.Meta):
|
||||||
|
fields = [
|
||||||
|
"id",
|
||||||
|
"title",
|
||||||
|
"auto_evaluation",
|
||||||
|
"document",
|
||||||
|
"category",
|
||||||
|
"created_at",
|
||||||
|
"updated_at"
|
||||||
|
]
|
||||||
@@ -1 +1,2 @@
|
|||||||
|
from .Document import * # noqa
|
||||||
from .ValuationDocument import * # noqa
|
from .ValuationDocument import * # noqa
|
||||||
|
|||||||
@@ -0,0 +1,34 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentcategoryModel
|
||||||
|
|
||||||
|
|
||||||
|
class BaseDocumentcategorySerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = DocumentcategoryModel
|
||||||
|
fields = [
|
||||||
|
"id",
|
||||||
|
"label",
|
||||||
|
"value",
|
||||||
|
"created_at",
|
||||||
|
"updated_at"
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
class ListDocumentcategorySerializer(BaseDocumentcategorySerializer):
|
||||||
|
class Meta(BaseDocumentcategorySerializer.Meta): ...
|
||||||
|
|
||||||
|
|
||||||
|
class RetrieveDocumentcategorySerializer(BaseDocumentcategorySerializer):
|
||||||
|
class Meta(BaseDocumentcategorySerializer.Meta): ...
|
||||||
|
|
||||||
|
|
||||||
|
# class CreateDocumentcategorySerializer(BaseDocumentcategorySerializer):
|
||||||
|
# class Meta(BaseDocumentcategorySerializer.Meta):
|
||||||
|
# fields = [
|
||||||
|
# "id",
|
||||||
|
# "label",
|
||||||
|
# "value",
|
||||||
|
# "created_at",
|
||||||
|
# "updated_at"
|
||||||
|
# ]
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
from .DocumentCategory import * # noqa
|
||||||
@@ -37,6 +37,9 @@ class BaseQuickevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"state_car",
|
"state_car",
|
||||||
"state_car_name",
|
"state_car_name",
|
||||||
"created_at",
|
"created_at",
|
||||||
|
"distance_covered",
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"is_archive"
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
@@ -51,7 +54,6 @@ class RetrieveQuickevaluationSerializer(BaseQuickevaluationSerializer):
|
|||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
"tech_passport_issued_date",
|
"tech_passport_issued_date",
|
||||||
"tech_passport_issued_place",
|
"tech_passport_issued_place",
|
||||||
"tex_passport_file",
|
|
||||||
"car_position",
|
"car_position",
|
||||||
"car_position_name",
|
"car_position_name",
|
||||||
"distance_covered",
|
"distance_covered",
|
||||||
@@ -74,7 +76,6 @@ class CreateQuickevaluationSerializer(serializers.ModelSerializer):
|
|||||||
"tex_passport_serie_num",
|
"tex_passport_serie_num",
|
||||||
"tech_passport_issued_date",
|
"tech_passport_issued_date",
|
||||||
"tech_passport_issued_place",
|
"tech_passport_issued_place",
|
||||||
"tex_passport_file",
|
|
||||||
"car_type",
|
"car_type",
|
||||||
"brand",
|
"brand",
|
||||||
"marka",
|
"marka",
|
||||||
@@ -125,3 +126,44 @@ class CreateQuickevaluationSerializer(serializers.ModelSerializer):
|
|||||||
if request and request.user and request.user.is_authenticated:
|
if request and request.user and request.user.is_authenticated:
|
||||||
validated_data["created_by"] = request.user
|
validated_data["created_by"] = request.user
|
||||||
return super().create(validated_data)
|
return super().create(validated_data)
|
||||||
|
|
||||||
|
|
||||||
|
class QuickEvaluationModelSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = QuickEvaluationModel
|
||||||
|
fields = (
|
||||||
|
"id",
|
||||||
|
|
||||||
|
"created_by",
|
||||||
|
"brand",
|
||||||
|
"marka",
|
||||||
|
"car_position",
|
||||||
|
"body_type",
|
||||||
|
"color",
|
||||||
|
"fuel_type",
|
||||||
|
"state_car",
|
||||||
|
|
||||||
|
"tex_passport_serie_num",
|
||||||
|
"tech_passport_issued_date",
|
||||||
|
"tech_passport_issued_place",
|
||||||
|
|
||||||
|
"car_type",
|
||||||
|
"distance_covered",
|
||||||
|
"vin_number",
|
||||||
|
"car_number",
|
||||||
|
"car_manufactured_date",
|
||||||
|
"engine_number",
|
||||||
|
|
||||||
|
"estimated_price",
|
||||||
|
"status",
|
||||||
|
"is_archive",
|
||||||
|
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
|
|
||||||
|
read_only_fields = (
|
||||||
|
"id",
|
||||||
|
"created_at",
|
||||||
|
"updated_at",
|
||||||
|
)
|
||||||
@@ -1,9 +1,17 @@
|
|||||||
import re
|
import re
|
||||||
|
|
||||||
|
from django.contrib.auth import get_user_model
|
||||||
|
|
||||||
from rest_framework import serializers
|
from rest_framework import serializers
|
||||||
|
|
||||||
from core.apps.evaluation.models import EvaluationrequestModel, ReferenceitemModel
|
from core.apps.evaluation.models import EvaluationrequestModel, ReferenceitemModel, EvaluationRequestOwnerModel, EvaluationRequestCustomerModel
|
||||||
from core.apps.evaluation.serializers.reference import ListReferenceitemSerializer
|
from core.apps.evaluation.serializers.reference import ListReferenceitemSerializer
|
||||||
|
from core.apps.evaluation.serializers.request.owner import EvaluationRequestOwnerSerializer
|
||||||
|
from core.apps.evaluation.serializers.request.req_customer import EvaluationRequestCustomerSerializer
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
User = get_user_model()
|
||||||
|
|
||||||
|
|
||||||
class BaseEvaluationrequestSerializer(serializers.ModelSerializer):
|
class BaseEvaluationrequestSerializer(serializers.ModelSerializer):
|
||||||
@@ -23,6 +31,9 @@ class BaseEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
rate_goal = ListReferenceitemSerializer(read_only=True)
|
rate_goal = ListReferenceitemSerializer(read_only=True)
|
||||||
property_rights = ListReferenceitemSerializer(read_only=True)
|
property_rights = ListReferenceitemSerializer(read_only=True)
|
||||||
form_ownership = ListReferenceitemSerializer(read_only=True)
|
form_ownership = ListReferenceitemSerializer(read_only=True)
|
||||||
|
user = serializers.SerializerMethodField(method_name="get_user")
|
||||||
|
customer = EvaluationRequestCustomerSerializer(read_only=True)
|
||||||
|
owner = EvaluationRequestOwnerSerializer(read_only=True)
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = EvaluationrequestModel
|
model = EvaluationrequestModel
|
||||||
@@ -46,8 +57,12 @@ class BaseEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
"location_name",
|
"location_name",
|
||||||
"status",
|
"status",
|
||||||
"status_display",
|
"status_display",
|
||||||
|
"user",
|
||||||
"created_at",
|
"created_at",
|
||||||
"updated_at",
|
"updated_at",
|
||||||
|
"is_archive",
|
||||||
|
"customer",
|
||||||
|
"owner",
|
||||||
]
|
]
|
||||||
|
|
||||||
def get_location(self, obj):
|
def get_location(self, obj):
|
||||||
@@ -59,6 +74,17 @@ class BaseEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
}
|
}
|
||||||
return None
|
return None
|
||||||
|
|
||||||
|
def get_user(self, obj):
|
||||||
|
request = self.context.get('request')
|
||||||
|
return {
|
||||||
|
"id": obj.user.id,
|
||||||
|
"phone": obj.user.phone,
|
||||||
|
"first_name": obj.user.first_name,
|
||||||
|
"last_name": obj.user.last_name,
|
||||||
|
"role": obj.user.role,
|
||||||
|
"avatar": request.build_absolute_uri(obj.user.avatar.url) if obj.user.avatar else None
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
class ListEvaluationrequestSerializer(BaseEvaluationrequestSerializer):
|
class ListEvaluationrequestSerializer(BaseEvaluationrequestSerializer):
|
||||||
class Meta(BaseEvaluationrequestSerializer.Meta):
|
class Meta(BaseEvaluationrequestSerializer.Meta):
|
||||||
@@ -73,7 +99,6 @@ class RetrieveEvaluationrequestSerializer(BaseEvaluationrequestSerializer):
|
|||||||
class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
||||||
location = serializers.DictField(required=False, allow_empty=True)
|
location = serializers.DictField(required=False, allow_empty=True)
|
||||||
|
|
||||||
# locationName — string qabul qiladi, location ichida yoki tashqarida yuborsa bo'ladi
|
|
||||||
locationName = serializers.CharField(
|
locationName = serializers.CharField(
|
||||||
write_only=True,
|
write_only=True,
|
||||||
required=False,
|
required=False,
|
||||||
@@ -81,8 +106,6 @@ class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
allow_blank=True,
|
allow_blank=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
# Faqat truck_car uchun majburiy — field darajasida optional qilib belgilaymiz,
|
|
||||||
# validate() da object_type ga qarab tekshiramiz
|
|
||||||
worked_hours = serializers.IntegerField(
|
worked_hours = serializers.IntegerField(
|
||||||
required=False,
|
required=False,
|
||||||
allow_null=True,
|
allow_null=True,
|
||||||
@@ -93,10 +116,12 @@ class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
allow_blank=True,
|
allow_blank=True,
|
||||||
)
|
)
|
||||||
|
|
||||||
value_determined = serializers.IntegerField(required=False, allow_null=True)
|
value_determined = serializers.PrimaryKeyRelatedField(required=False, queryset=ReferenceitemModel.objects.all())
|
||||||
rate_goal = serializers.IntegerField(required=False, allow_null=True)
|
rate_goal = serializers.PrimaryKeyRelatedField(required=False, queryset=ReferenceitemModel.objects.all())
|
||||||
property_rights = serializers.IntegerField(required=False, allow_null=True)
|
property_rights = serializers.PrimaryKeyRelatedField(required=False, queryset=ReferenceitemModel.objects.all())
|
||||||
form_ownership = serializers.IntegerField(required=False, allow_null=True)
|
form_ownership = serializers.PrimaryKeyRelatedField(required=False, queryset=ReferenceitemModel.objects.all())
|
||||||
|
customer = EvaluationRequestCustomerSerializer()
|
||||||
|
owner = EvaluationRequestOwnerSerializer()
|
||||||
|
|
||||||
class Meta:
|
class Meta:
|
||||||
model = EvaluationrequestModel
|
model = EvaluationrequestModel
|
||||||
@@ -115,6 +140,11 @@ class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
"need_delivering",
|
"need_delivering",
|
||||||
"location",
|
"location",
|
||||||
"locationName",
|
"locationName",
|
||||||
|
"customer",
|
||||||
|
"owner",
|
||||||
|
"customer_and_owner_same",
|
||||||
|
"distance_covered",
|
||||||
|
"gov_number"
|
||||||
]
|
]
|
||||||
|
|
||||||
def validate_tex_passport(self, value):
|
def validate_tex_passport(self, value):
|
||||||
@@ -138,7 +168,6 @@ class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
{"tex_passport": "rate_type 'auto' bo'lganda tex_passport majburiy."}
|
{"tex_passport": "rate_type 'auto' bo'lganda tex_passport majburiy."}
|
||||||
)
|
)
|
||||||
|
|
||||||
# worked_hours va chassi FAQAT yuk avtomobil (truck_car) uchun majburiy
|
|
||||||
if object_type == "truck_car":
|
if object_type == "truck_car":
|
||||||
if attrs.get("worked_hours") is None:
|
if attrs.get("worked_hours") is None:
|
||||||
raise serializers.ValidationError(
|
raise serializers.ValidationError(
|
||||||
@@ -149,30 +178,6 @@ class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
{"chassi": "Yuk avtomobil uchun shassi raqami majburiy."}
|
{"chassi": "Yuk avtomobil uchun shassi raqami majburiy."}
|
||||||
)
|
)
|
||||||
|
|
||||||
if attrs.get("value_determined"):
|
|
||||||
if attrs.get("value_determined") not in ReferenceitemModel.objects.values_list("id", flat=True):
|
|
||||||
raise serializers.ValidationError(
|
|
||||||
{"value_determined": "Noto'g'ri qiymat aniqlandi."}
|
|
||||||
)
|
|
||||||
|
|
||||||
if attrs.get("rate_goal"):
|
|
||||||
if attrs.get("rate_goal") not in ReferenceitemModel.objects.values_list("id", flat=True):
|
|
||||||
raise serializers.ValidationError(
|
|
||||||
{"rate_goal": "Noto'g'ri qiymat aniqlandi."}
|
|
||||||
)
|
|
||||||
|
|
||||||
if attrs.get("property_rights"):
|
|
||||||
if attrs.get("property_rights") not in ReferenceitemModel.objects.values_list("id", flat=True):
|
|
||||||
raise serializers.ValidationError(
|
|
||||||
{"property_rights": "Noto'g'ri qiymat aniqlandi."}
|
|
||||||
)
|
|
||||||
|
|
||||||
if attrs.get("form_ownership"):
|
|
||||||
if attrs.get("form_ownership") not in ReferenceitemModel.objects.values_list("id", flat=True):
|
|
||||||
raise serializers.ValidationError(
|
|
||||||
{"form_ownership": "Noto'g'ri qiymat aniqlandi."}
|
|
||||||
)
|
|
||||||
|
|
||||||
return attrs
|
return attrs
|
||||||
|
|
||||||
def create(self, validated_data):
|
def create(self, validated_data):
|
||||||
@@ -187,6 +192,33 @@ class CreateEvaluationrequestSerializer(serializers.ModelSerializer):
|
|||||||
|
|
||||||
if location_name:
|
if location_name:
|
||||||
validated_data["location_name"] = str(location_name)
|
validated_data["location_name"] = str(location_name)
|
||||||
|
|
||||||
validated_data["user"] = self.context["request"].user
|
validated_data["user"] = self.context["request"].user
|
||||||
return super().create(validated_data)
|
|
||||||
|
instance = super().create(validated_data)
|
||||||
|
|
||||||
|
customer = validated_data.pop("customer", None)
|
||||||
|
owner = validated_data.pop("owner", None)
|
||||||
|
EvaluationRequestCustomerModel.objects.create(
|
||||||
|
evaluation_request=instance,
|
||||||
|
type=customer.get("type"),
|
||||||
|
jshshir=customer.get("jshshir")
|
||||||
|
)
|
||||||
|
if not instance.customer_and_owner_same:
|
||||||
|
EvaluationRequestOwnerModel.objects.create(
|
||||||
|
evaluation_request=instance,
|
||||||
|
type=owner.get("type"),
|
||||||
|
jshshir=owner.get("jshshir")
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
EvaluationRequestOwnerModel.objects.create(
|
||||||
|
evaluation_request=instance,
|
||||||
|
type=customer.get("type"),
|
||||||
|
jshshir=customer.get("jshshir")
|
||||||
|
)
|
||||||
|
|
||||||
|
return instance
|
||||||
|
|
||||||
|
|
||||||
|
class ArchiveEvaluationrequestSerializer(serializers.Serializer):
|
||||||
|
id = serializers.IntegerField(required=True)
|
||||||
|
is_archive = serializers.BooleanField(required=True)
|
||||||
|
|||||||
9
core/apps/evaluation/serializers/request/owner.py
Normal file
9
core/apps/evaluation/serializers/request/owner.py
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.evaluation.models.request import EvaluationRequestOwnerModel
|
||||||
|
|
||||||
|
|
||||||
|
class EvaluationRequestOwnerSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = EvaluationRequestOwnerModel
|
||||||
|
fields = ["id", "evaluation_request", "type", "jshshir"]
|
||||||
9
core/apps/evaluation/serializers/request/req_customer.py
Normal file
9
core/apps/evaluation/serializers/request/req_customer.py
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
from core.apps.evaluation.models.request import EvaluationRequestCustomerModel
|
||||||
|
|
||||||
|
|
||||||
|
class EvaluationRequestCustomerSerializer(serializers.ModelSerializer):
|
||||||
|
class Meta:
|
||||||
|
model = EvaluationRequestCustomerModel
|
||||||
|
fields = ["id", "evaluation_request", "type", "jshshir"]
|
||||||
@@ -0,0 +1 @@
|
|||||||
|
from .tech_passport import * # noqa
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
from rest_framework import serializers
|
||||||
|
|
||||||
|
class TechPassportSerializer(serializers.Serializer):
|
||||||
|
autonumber = serializers.CharField(required=True, max_length=20)
|
||||||
|
tech_pass_number = serializers.CharField(required=True, max_length=20)
|
||||||
|
tech_pass_series = serializers.CharField(required=True, max_length=20)
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -1,8 +1,12 @@
|
|||||||
from django.db.models.signals import post_save
|
from django.db.models.signals import post_save
|
||||||
from django.dispatch import receiver
|
from django.dispatch import receiver
|
||||||
|
|
||||||
from core.apps.evaluation.models import ValuationDocumentModel
|
from core.apps.evaluation.models import DocumentModel, ValuationDocumentModel
|
||||||
|
|
||||||
|
|
||||||
@receiver(post_save, sender=ValuationDocumentModel)
|
@receiver(post_save, sender=ValuationDocumentModel)
|
||||||
def ValuationdocumentSignal(sender, instance, created, **kwargs): ...
|
def ValuationdocumentSignal(sender, instance, created, **kwargs): ...
|
||||||
|
|
||||||
|
|
||||||
|
@receiver(post_save, sender=DocumentModel)
|
||||||
|
def DocumentSignal(sender, instance, created, **kwargs): ...
|
||||||
|
|||||||
8
core/apps/evaluation/signals/documentcategory.py
Normal file
8
core/apps/evaluation/signals/documentcategory.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
from django.db.models.signals import post_save
|
||||||
|
from django.dispatch import receiver
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentcategoryModel
|
||||||
|
|
||||||
|
|
||||||
|
@receiver(post_save, sender=DocumentcategoryModel)
|
||||||
|
def DocumentcategorySignal(sender, instance, created, **kwargs): ...
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -1 +1,2 @@
|
|||||||
|
from .test_Document import * # noqa
|
||||||
from .test_ValuationDocument import * # noqa
|
from .test_ValuationDocument import * # noqa
|
||||||
|
|||||||
101
core/apps/evaluation/tests/document/test_Document.py
Normal file
101
core/apps/evaluation/tests/document/test_Document.py
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
import pytest
|
||||||
|
from django.urls import reverse
|
||||||
|
from rest_framework.test import APIClient
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentModel
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def instance(db):
|
||||||
|
return DocumentModel._baker()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def api_client(instance):
|
||||||
|
client = APIClient()
|
||||||
|
##client.force_authenticate(user=instance.user)
|
||||||
|
return client, instance
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def data(api_client):
|
||||||
|
client, instance = api_client
|
||||||
|
return (
|
||||||
|
{
|
||||||
|
"list": reverse("Document-list"),
|
||||||
|
"retrieve": reverse("Document-detail", kwargs={"pk": instance.pk}),
|
||||||
|
"retrieve-not-found": reverse("Document-detail", kwargs={"pk": 1000}),
|
||||||
|
},
|
||||||
|
client,
|
||||||
|
instance,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_list(data):
|
||||||
|
urls, client, _ = data
|
||||||
|
response = client.get(urls["list"])
|
||||||
|
data_resp = response.json()
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert data_resp["status"] is True
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_retrieve(data):
|
||||||
|
urls, client, _ = data
|
||||||
|
response = client.get(urls["retrieve"])
|
||||||
|
data_resp = response.json()
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert data_resp["status"] is True
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_retrieve_not_found(data):
|
||||||
|
urls, client, _ = data
|
||||||
|
response = client.get(urls["retrieve-not-found"])
|
||||||
|
data_resp = response.json()
|
||||||
|
assert response.status_code == 404
|
||||||
|
assert data_resp["status"] is False
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_create(data):
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.post(urls["list"], data={"name": "test"})
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 201
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_update(data):
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.patch(urls["retrieve"], data={"name": "updated"})
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
#
|
||||||
|
# # verify updated value
|
||||||
|
# response = client.get(urls["retrieve"])
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
# assert response.json()["data"]["name"] == "updated"
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_partial_update():
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.patch(urls["retrieve"], data={"name": "updated"})
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
#
|
||||||
|
# # verify updated value
|
||||||
|
# response = client.get(urls["retrieve"])
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
# assert response.json()["data"]["name"] == "updated"
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_destroy(data):
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.delete(urls["retrieve"])
|
||||||
|
# assert response.status_code == 204
|
||||||
1
core/apps/evaluation/tests/documentcategory/__init__.py
Normal file
1
core/apps/evaluation/tests/documentcategory/__init__.py
Normal file
@@ -0,0 +1 @@
|
|||||||
|
from .test_DocumentCategory import * # noqa
|
||||||
@@ -0,0 +1,101 @@
|
|||||||
|
import pytest
|
||||||
|
from django.urls import reverse
|
||||||
|
from rest_framework.test import APIClient
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentcategoryModel
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def instance(db):
|
||||||
|
return DocumentcategoryModel._baker()
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def api_client(instance):
|
||||||
|
client = APIClient()
|
||||||
|
##client.force_authenticate(user=instance.user)
|
||||||
|
return client, instance
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def data(api_client):
|
||||||
|
client, instance = api_client
|
||||||
|
return (
|
||||||
|
{
|
||||||
|
"list": reverse("DocumentCategory-list"),
|
||||||
|
"retrieve": reverse("DocumentCategory-detail", kwargs={"pk": instance.pk}),
|
||||||
|
"retrieve-not-found": reverse("DocumentCategory-detail", kwargs={"pk": 1000}),
|
||||||
|
},
|
||||||
|
client,
|
||||||
|
instance,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_list(data):
|
||||||
|
urls, client, _ = data
|
||||||
|
response = client.get(urls["list"])
|
||||||
|
data_resp = response.json()
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert data_resp["status"] is True
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_retrieve(data):
|
||||||
|
urls, client, _ = data
|
||||||
|
response = client.get(urls["retrieve"])
|
||||||
|
data_resp = response.json()
|
||||||
|
assert response.status_code == 200
|
||||||
|
assert data_resp["status"] is True
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_retrieve_not_found(data):
|
||||||
|
urls, client, _ = data
|
||||||
|
response = client.get(urls["retrieve-not-found"])
|
||||||
|
data_resp = response.json()
|
||||||
|
assert response.status_code == 404
|
||||||
|
assert data_resp["status"] is False
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_create(data):
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.post(urls["list"], data={"name": "test"})
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 201
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_update(data):
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.patch(urls["retrieve"], data={"name": "updated"})
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
#
|
||||||
|
# # verify updated value
|
||||||
|
# response = client.get(urls["retrieve"])
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
# assert response.json()["data"]["name"] == "updated"
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_partial_update():
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.patch(urls["retrieve"], data={"name": "updated"})
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
#
|
||||||
|
# # verify updated value
|
||||||
|
# response = client.get(urls["retrieve"])
|
||||||
|
# assert response.json()["status"] is True
|
||||||
|
# assert response.status_code == 200
|
||||||
|
# assert response.json()["data"]["name"] == "updated"
|
||||||
|
|
||||||
|
|
||||||
|
# @pytest.mark.django_db
|
||||||
|
# def test_destroy(data):
|
||||||
|
# urls, client, _ = data
|
||||||
|
# response = client.delete(urls["retrieve"])
|
||||||
|
# assert response.status_code == 204
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -1,8 +1,13 @@
|
|||||||
from modeltranslation.translator import TranslationOptions, register
|
from modeltranslation.translator import TranslationOptions, register
|
||||||
|
|
||||||
from core.apps.evaluation.models import ValuationDocumentModel
|
from core.apps.evaluation.models import DocumentModel, ValuationDocumentModel
|
||||||
|
|
||||||
|
|
||||||
@register(ValuationDocumentModel)
|
@register(ValuationDocumentModel)
|
||||||
class ValuationdocumentTranslation(TranslationOptions):
|
class ValuationdocumentTranslation(TranslationOptions):
|
||||||
fields = []
|
fields = []
|
||||||
|
|
||||||
|
|
||||||
|
@register(DocumentModel)
|
||||||
|
class DocumentTranslation(TranslationOptions):
|
||||||
|
fields = []
|
||||||
|
|||||||
8
core/apps/evaluation/translation/documentcategory.py
Normal file
8
core/apps/evaluation/translation/documentcategory.py
Normal file
@@ -0,0 +1,8 @@
|
|||||||
|
from modeltranslation.translator import TranslationOptions, register
|
||||||
|
|
||||||
|
from core.apps.evaluation.models import DocumentcategoryModel
|
||||||
|
|
||||||
|
|
||||||
|
@register(DocumentcategoryModel)
|
||||||
|
class DocumentcategoryTranslation(TranslationOptions):
|
||||||
|
fields = ["label", "value"]
|
||||||
@@ -1,44 +1,91 @@
|
|||||||
from django.urls import include, path
|
from django.urls import include, path
|
||||||
from rest_framework.routers import DefaultRouter
|
from rest_framework.routers import DefaultRouter
|
||||||
|
|
||||||
from .views import (
|
from core.apps.evaluation import views
|
||||||
AutoEvaluationHistoryView,
|
|
||||||
AutoEvaluationView,
|
|
||||||
CustomerView,
|
|
||||||
DeterminedValueView,
|
|
||||||
EvaluationPurposeView,
|
|
||||||
EvaluationReportView,
|
|
||||||
EvaluationrequestView,
|
|
||||||
MovablePropertyEvaluationView,
|
|
||||||
OwnershipFormView,
|
|
||||||
PropertyOwnerView,
|
|
||||||
PropertyRightsView,
|
|
||||||
QuickEvaluationHistoryView,
|
|
||||||
QuickEvaluationView,
|
|
||||||
RealEstateEvaluationView,
|
|
||||||
ReferenceitemView,
|
|
||||||
ValuationDocumentView,
|
|
||||||
ValuationView,
|
|
||||||
VehicleView,
|
|
||||||
)
|
|
||||||
|
|
||||||
router = DefaultRouter()
|
router = DefaultRouter()
|
||||||
router.register("auto-evaluation-history", AutoEvaluationHistoryView, basename="auto-evaluation-history")
|
router.register("document-category", views.DocumentCategoryView, basename="DocumentCategory")
|
||||||
router.register("quick-evaluation-history", QuickEvaluationHistoryView, basename="quick-evaluation-history")
|
router.register("document", views.DocumentView, basename="Document")
|
||||||
router.register("determined-value", DeterminedValueView, basename="determined-value")
|
router.register("auto-evaluation-history", views.AutoEvaluationHistoryView, basename="auto-evaluation-history")
|
||||||
router.register("evaluation-purpose", EvaluationPurposeView, basename="evaluation-purpose")
|
router.register("quick-evaluation-history", views.QuickEvaluationHistoryView, basename="quick-evaluation-history")
|
||||||
router.register("property-rights", PropertyRightsView, basename="property-rights")
|
router.register("determined-value", views.DeterminedValueView, basename="determined-value")
|
||||||
router.register("ownership-form", OwnershipFormView, basename="ownership-form")
|
router.register("evaluation-purpose", views.EvaluationPurposeView, basename="evaluation-purpose")
|
||||||
router.register("evaluation-request", EvaluationrequestView, basename="evaluation-request")
|
router.register("property-rights", views.PropertyRightsView, basename="property-rights")
|
||||||
router.register("reference-item", ReferenceitemView, basename="reference-item")
|
router.register("ownership-form", views.OwnershipFormView, basename="ownership-form")
|
||||||
router.register("valuation-document", ValuationDocumentView, basename="valuation-document")
|
router.register("evaluation-request", views.EvaluationrequestView, basename="evaluation-request")
|
||||||
router.register("evaluation-report", EvaluationReportView, basename="evaluation-report")
|
router.register("admin-evaluation-request", views.AdminEvaluationrequestView, basename="admin-evaluation-request")
|
||||||
router.register("quick-evaluation", QuickEvaluationView, basename="quick-evaluation")
|
router.register("reference-item", views.ReferenceitemView, basename="reference-item")
|
||||||
router.register("movable-property-evaluation", MovablePropertyEvaluationView, basename="movable-property-evaluation")
|
router.register("valuation-document", views.ValuationDocumentView, basename="valuation-document")
|
||||||
router.register("real-estate-evaluation", RealEstateEvaluationView, basename="real-estate-evaluation")
|
router.register("evaluation-report", views.EvaluationReportView, basename="evaluation-report")
|
||||||
router.register("auto-evaluation", AutoEvaluationView, basename="auto-evaluation")
|
router.register("quick-evaluation", views.QuickEvaluationView, basename="quick-evaluation")
|
||||||
router.register("vehicle", VehicleView, basename="vehicle")
|
router.register("movable-property-evaluation", views.MovablePropertyEvaluationView, basename="movable-property-evaluation")
|
||||||
router.register("valuation", ValuationView, basename="valuation")
|
router.register("real-estate-evaluation", views.RealEstateEvaluationView, basename="real-estate-evaluation")
|
||||||
router.register("property-owner", PropertyOwnerView, basename="property-owner")
|
router.register("auto-evaluation", views.AutoEvaluationView, basename="auto-evaluation")
|
||||||
router.register("customer", CustomerView, basename="customer")
|
router.register("vehicle", views.VehicleView, basename="vehicle")
|
||||||
urlpatterns = [path("", include(router.urls))]
|
router.register("valuation", views.ValuationView, basename="valuation")
|
||||||
|
router.register("property-owner", views.PropertyOwnerView, basename="property-owner")
|
||||||
|
router.register("customer", views.CustomerView, basename="customer")
|
||||||
|
router.register("certificate", views.CertificateView, basename="certificate")
|
||||||
|
router.register("bonus-type", views.BonusTypeView, basename="bonus-type")
|
||||||
|
router.register("bonus-employee", views.BonusEmployeeViewSet, basename="bonus-employee")
|
||||||
|
router.register("bonus-base", views.BaseBonusViewSet, basename="bonus-base")
|
||||||
|
|
||||||
|
urlpatterns = [
|
||||||
|
path("", include(router.urls)),
|
||||||
|
|
||||||
|
path("didox/info/<int:tin>/", views.DidoxCompanyInfoAPIView.as_view()),
|
||||||
|
path("tech-passport/", views.TechPassportAPIView.as_view()),
|
||||||
|
|
||||||
|
# Quick Evaluation
|
||||||
|
path('quick-evaluation/', include(
|
||||||
|
[
|
||||||
|
path("admin/", views.AdminQuickEvalAPIView.as_view(), name="quick-evaluation"),
|
||||||
|
path(
|
||||||
|
'archive/', include(
|
||||||
|
[
|
||||||
|
path("list/", views.QuickEvaluationArchivedListAPIView.as_view()),
|
||||||
|
path("<int:pk>/", views.QuickEvaluationArchiveAPIView.as_view()),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
),
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
|
||||||
|
# Auto Evaluation
|
||||||
|
path("auto-evaluation/", include(
|
||||||
|
[
|
||||||
|
path("admin/", views.AdminEvaluationsAPIView.as_view(), name="admin-evaluations"),
|
||||||
|
path('archive/', include(
|
||||||
|
[
|
||||||
|
path('<int:pk>/', views.AutoEvaluationArchiveAPIView.as_view()),
|
||||||
|
path('list/', views.AutoEvaluationArchivedListAPIView.as_view())
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
path('appraisers/', include(
|
||||||
|
[
|
||||||
|
path("<int:id>/list/", views.AutoEvaluationListAppraisersView.as_view()),
|
||||||
|
path("<int:id>/set/", views.AutoEvaluationSetAppraisersView.as_view()),
|
||||||
|
path("<int:id>/remove/", views.AutoEvaluationRemoveAppraisersView.as_view()),
|
||||||
|
]
|
||||||
|
))
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
|
||||||
|
# Evaluation Request
|
||||||
|
path("evaluation-request/", include(
|
||||||
|
[
|
||||||
|
path("<int:pk>/change-status/", views.EvaluationStatusChange.as_view()),
|
||||||
|
path(
|
||||||
|
'archive/', include(
|
||||||
|
[
|
||||||
|
path('list/', views.RequestEvaluationArchivedListAPIView.as_view()),
|
||||||
|
path('<int:pk>/', views.RequestEvaluationArchiveAPIView.as_view()),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)),
|
||||||
|
|
||||||
|
path("calculate_avg_cost/", views.AvgCostAPIView.as_view()),
|
||||||
|
]
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
from .auto import * # noqa
|
from .auto import * # noqa
|
||||||
from .customer import * # noqa
|
from .customer import * # noqa
|
||||||
from .document import * # noqa
|
from .document import * # noqa
|
||||||
|
from .documentcategory import * # noqa
|
||||||
from .history import * # noqa
|
from .history import * # noqa
|
||||||
from .movable import * # noqa
|
from .movable import * # noqa
|
||||||
from .quick import * # noqa
|
from .quick import * # noqa
|
||||||
|
|||||||
@@ -6,3 +6,10 @@ class ValuationdocumentValidator:
|
|||||||
|
|
||||||
def __call__(self):
|
def __call__(self):
|
||||||
return True
|
return True
|
||||||
|
|
||||||
|
|
||||||
|
class DocumentValidator:
|
||||||
|
def __init__(self): ...
|
||||||
|
|
||||||
|
def __call__(self):
|
||||||
|
return True
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user