websoket qoshildi

This commit is contained in:
behruz-dev
2025-11-28 17:51:00 +05:00
parent b14062e7b9
commit aa38215428
9 changed files with 215 additions and 6 deletions

View File

@@ -1,7 +1,23 @@
import os import os
import django
from django.core.asgi import get_asgi_application
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'config.settings.base') os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'config.settings.base')
application = get_asgi_application() django.setup()
from django.core.asgi import get_asgi_application
from channels.routing import ProtocolTypeRouter, URLRouter
from channels.auth import AuthMiddlewareStack
from django.urls import re_path
from core.apps.authentication.websocket.consumers import UserActivationConsumer
django_asgi_app = get_asgi_application()
application = ProtocolTypeRouter({
'http': django_asgi_app,
'websocket': AuthMiddlewareStack(
URLRouter([
re_path(r'^ws/user_activation/(?P<user_id>\d+)/$', UserActivationConsumer.as_asgi()),
])
),
})

View File

@@ -2,3 +2,4 @@ from .rest_framework import *
from .corsheaders import * from .corsheaders import *
from .simple_jwt import * from .simple_jwt import *
from .drf_yasg import * from .drf_yasg import *
from .redis import *

10
config/conf/redis.py Normal file
View File

@@ -0,0 +1,10 @@
from config.env import env
CHANNEL_LAYERS = {
'default': {
'BACKEND': 'channels_redis.core.RedisChannelLayer',
'CONFIG': {
"hosts": [(env.str('REDIS_HOST'), env.str('REDIS_PORT'))],
},
},
}

View File

@@ -61,6 +61,7 @@ TEMPLATES = [
] ]
WSGI_APPLICATION = 'config.wsgi.application' WSGI_APPLICATION = 'config.wsgi.application'
ASGI_APPLICATION = 'config.asgi.application'
DATABASES = { DATABASES = {

View File

@@ -0,0 +1,55 @@
import json
# channels
from channels.generic.websocket import AsyncJsonWebsocketConsumer
from channels.db import database_sync_to_async
# accounts
from core.apps.accounts.models import User
class UserActivationConsumer(AsyncJsonWebsocketConsumer):
async def connect(self):
self.user_id = self.scope['url_route']['kwargs'].get('user_id')
if self.user_id:
await self.channel_layer.group_add(
f"user_{self.user_id}",
self.channel_name
)
await self.accept()
print(f"User {self.user_id} ulandi")
else:
await self.close()
async def disconnect(self, code):
if self.user_id:
await self.channel_layer.group_discard(
f"user_{self.user_id}",
self.channel_name,
)
print(f'User {self.user_id} uzilib ketti')
async def receive(self, text_data):
data = json.loads(text_data)
message_type = data.get('type')
if message_type == 'ping':
await self.send(text_data=json.dumps(
{
'type': 'pong',
'message': "boglanish faol"
}
))
async def user_activated(self, event):
await self.send(text_data=json.dumps({
'type': 'user_activated',
'message': 'user aktive qilindi',
'timestamp': event.get('timestamp'),
'status': "success",
"status_code": 200,
"status_message": "user_activated",
"token": event.get('token'),
}))

View File

@@ -22,6 +22,7 @@ urlpatterns = [
path('create/', user_views.UserCreateApiView.as_view(), name='user-create-api'), path('create/', user_views.UserCreateApiView.as_view(), name='user-create-api'),
path('<int:id>/delete/', user_views.UserDeleteApiView.as_view(), name='user-delete-api'), path('<int:id>/delete/', user_views.UserDeleteApiView.as_view(), name='user-delete-api'),
path('<int:id>/update/', user_views.UserUpdateApiView.as_view(), name='user-update-api'), path('<int:id>/update/', user_views.UserUpdateApiView.as_view(), name='user-update-api'),
path('<int:id>/activate/', user_views.UserActivateApiView.as_view(), name='user-activate-api'),
], ],
)), )),
# -------------- district -------------- # -------------- district --------------

View File

@@ -1,3 +1,6 @@
from asgiref.sync import async_to_sync
import datetime
# django # django
from django.shortcuts import get_object_or_404 from django.shortcuts import get_object_or_404
from django.db.models import Q from django.db.models import Q
@@ -10,6 +13,12 @@ from rest_framework.permissions import IsAdminUser
from drf_yasg import openapi from drf_yasg import openapi
from drf_yasg.utils import swagger_auto_schema from drf_yasg.utils import swagger_auto_schema
# rest framework simple jwt
from rest_framework_simplejwt.tokens import RefreshToken
# channels
from channels.layers import get_channel_layer
# dashboard # dashboard
from core.apps.dashboard.serializers import user as serializers from core.apps.dashboard.serializers import user as serializers
# accounts # accounts
@@ -350,3 +359,88 @@ class UserDeleteApiView(views.APIView, ResponseMixin):
except Exception as e: except Exception as e:
return self.error_response(data=str(e), message="xatolik") return self.error_response(data=str(e), message="xatolik")
class UserActivateApiView(views.APIView, ResponseMixin):
permission_classes = [IsAdminUser]
@swagger_auto_schema(
responses={
200: openapi.Response(
schema=None,
description="Success",
examples={
"application/json": {
"status_code": 204,
"status": "success",
"message": "user tasdiqlandi",
"data": {
"id": 0,
"first_name": "string",
"last_name": "string",
"region": {
"id": 0,
"name": "string"
},
"is_active": True,
"created_at": "string"
}
}
}
),
404: openapi.Response(
schema=None,
description="Not Found",
examples={
"application/json": {
"status_code": 404,
"success": "failure",
"message": "User not found",
"data": {},
}
}
),
500: openapi.Response(
schema=None,
description="Server Error",
examples={
"application/json": {
"status_code": 500,
"status": "error",
"message": "xatolik",
"data": "string"
}
}
),
}
)
def post(self, request, id):
try:
channel_layer = get_channel_layer()
user = User.objects.filter(id=id).first()
if not user:
return self.failure_response(data={}, message='User not found', status_code=404)
user.is_active = True
user.save()
token = RefreshToken.for_user(user)
async_to_sync(channel_layer.group_send)(
f"user_{user.id}",
{
'type': 'user_activated',
'message': 'user aktive qilindi',
'status': "success",
"status_code": 200,
"status_message": "user_activated",
"token": str(token.access_token),
}
)
return self.success_response(
data=serializers.UserListSerializer(user).data,
message="user tasdiqlandi",
)
except Exception as e:
return self.error_response(
data=str(e),
message='xatolik'
)

View File

@@ -34,6 +34,7 @@ services:
- '.:/code' - '.:/code'
depends_on: depends_on:
- db - db
- redis
restart: always restart: always
db: db:
@@ -47,3 +48,8 @@ services:
volumes: volumes:
- pg_data:/var/lib/postgresql/data - pg_data:/var/lib/postgresql/data
restart: always restart: always
redis:
image: redis
networks:
- meridyn_pharma

View File

@@ -1,38 +1,63 @@
asgiref==3.11.0 asgiref==3.11.0
attrs==25.4.0
autobahn==25.11.1
Automat==25.4.16
brotli==1.2.0 brotli==1.2.0
cbor2==5.7.1
certifi==2025.11.12 certifi==2025.11.12
cffi==2.0.0 cffi==2.0.0
channels==4.3.2
channels_redis==4.3.0
charset-normalizer==3.4.4 charset-normalizer==3.4.4
click==8.3.1 click==8.3.1
constantly==23.10.4
cryptography==46.0.3
cssselect2==0.8.0 cssselect2==0.8.0
daphne==4.2.1
Django==5.2 Django==5.2
django-cors-headers==4.9.0 django-cors-headers==4.9.0
django-environ==0.12.0 django-environ==0.12.0
django-redis==6.0.0
djangorestframework==3.16.1 djangorestframework==3.16.1
djangorestframework_simplejwt==5.5.1 djangorestframework_simplejwt==5.5.1
drf-yasg==1.21.11 drf-yasg==1.21.11
fonttools==4.60.1 fonttools==4.60.1
gunicorn==23.0.0 gunicorn==23.0.0
h11==0.16.0 h11==0.16.0
hyperlink==21.0.0
idna==3.11 idna==3.11
Incremental==24.11.0
inflection==0.5.1 inflection==0.5.1
msgpack==1.1.2
packaging==25.0 packaging==25.0
pillow==12.0.0 pillow==12.0.0
psycopg2-binary==2.9.11 psycopg2-binary==2.9.11
py-ubjson==0.16.1
pyasn1==0.6.1
pyasn1_modules==0.4.2
pycparser==2.23 pycparser==2.23
pydyf==0.11.0 pydyf==0.11.0
PyJWT==2.10.1 PyJWT==2.10.1
pyOpenSSL==25.3.0
pyphen==0.17.2 pyphen==0.17.2
pytz==2025.2 pytz==2025.2
PyYAML==6.0.3 PyYAML==6.0.3
redis==7.1.0
reportlab==4.4.5 reportlab==4.4.5
requests==2.32.5 requests==2.32.5
service-identity==24.2.0
sqlparse==0.5.3 sqlparse==0.5.3
tinycss2==1.5.1 tinycss2==1.5.1
tinyhtml5==2.0.0 tinyhtml5==2.0.0
Twisted==25.5.0
txaio==25.9.2
typing_extensions==4.15.0
ujson==5.11.0
uritemplate==4.2.0 uritemplate==4.2.0
urllib3==2.5.0 urllib3==2.5.0
uvicorn==0.38.0 uvicorn==0.38.0
weasyprint==66.0 weasyprint==66.0
webencodings==0.5.1 webencodings==0.5.1
zope.interface==8.1.1
zopfli==0.4.0 zopfli==0.4.0
websockets