18 Commits

Author SHA1 Message Date
f340b56218 Login Va Register bitta qilib yuborildi 2025-12-03 13:14:44 +05:00
fcba840ce9 Fix typo 2025-12-03 02:13:34 +05:00
0d99e508ac Fix typo 2025-12-03 02:08:03 +05:00
315f50e18f Fix typo 2025-12-03 01:59:14 +05:00
a931980d09 Fix typo 2025-12-03 01:45:44 +05:00
d01dd1034b Fix stackfile 2025-12-03 01:38:11 +05:00
9d494ab34e Merge pull request 'Fix typo' (#16) from fix/stackfile into main
Reviewed-on: #16
2025-12-02 19:38:56 +00:00
1a844132e6 Fix typo 2025-12-03 00:16:16 +05:00
85dcbd8808 Merge pull request 'Fix typo' (#15) from fix/migrations into main
Reviewed-on: #15
2025-12-02 19:04:09 +00:00
5f0df931a7 Fix typo 2025-12-02 17:32:39 +05:00
d38d2cd800 Merge pull request 'Fix typo' (#14) from fix/jenkinsfile into main
Reviewed-on: #14
2025-12-02 12:24:15 +00:00
6204a57f86 Merge branch 'main' into fix/jenkinsfile 2025-12-02 12:23:38 +00:00
4af4b0c02f Fix typo 2025-12-02 17:23:13 +05:00
6402a5b418 Merge pull request 'Jenkinsfile update' (#13) from fix/jenkinsfile into main
Reviewed-on: #13
2025-12-02 11:57:42 +00:00
09d66613ea Jenkinsfile update 2025-12-02 16:56:59 +05:00
b1787200d7 Merge pull request 'Search ads uchun api chiqarildi' (#11) from feat/search_ads into main
Reviewed-on: #11
2025-12-02 11:43:23 +00:00
8337f68a01 Resolve merge conflict in urls.py 2025-12-02 16:42:35 +05:00
02fd95fe1f Search ads uchun api chiqarildi 2025-11-27 15:02:23 +05:00
19 changed files with 242 additions and 25 deletions

2
Jenkinsfile vendored
View File

@@ -31,7 +31,7 @@ pipeline {
} }
stage('Checkout Code') { stage('Checkout Code') {
steps { steps {
git branch: 'main', credentialsId: 'ssh', url: 'git@github.com:JscorpTech/uzxarid.git' git branch: 'dev', credentialsId: 'muhammadvadud', url: 'https://gitea.felixits.uz/uzxarid/backend.git'
} }
} }
stage('Prepare') { stage('Prepare') {

View File

@@ -5,8 +5,12 @@ CHANNEL_LAYERS = {
"default": { "default": {
"BACKEND": "channels_redis.core.RedisChannelLayer", "BACKEND": "channels_redis.core.RedisChannelLayer",
"CONFIG": { "CONFIG": {
"hosts": [(env.str("REDIS_HOST", "redis"), env.int("REDIS_PORT", 6379))], "hosts": [
(
env.str("REDIS_HOST", "redis"),
env.int("REDIS_PORT_6379_TCP_PORT", 6379)
)
],
}, },
}, },
} }

View File

@@ -19,12 +19,6 @@ class LoginSerializer(serializers.Serializer):
class RegisterSerializer(serializers.ModelSerializer): class RegisterSerializer(serializers.ModelSerializer):
phone = serializers.CharField(max_length=255) phone = serializers.CharField(max_length=255)
def validate_phone(self, value):
user = get_user_model().objects.filter(phone=value, validated_at__isnull=False)
if user.exists():
raise exceptions.ValidationError(_("Phone number already registered."), code="unique")
return value
class Meta: class Meta:
model = get_user_model() model = get_user_model()
fields = ["phone"] fields = ["phone"]

View File

@@ -0,0 +1,28 @@
# Generated by Django 5.2.7 on 2025-11-27 07:55
from django.db import migrations, models
class Migration(migrations.Migration):
dependencies = [
('api', '0013_alter_feedback_comment'),
]
operations = [
migrations.AddField(
model_name='category',
name='name_en',
field=models.CharField(max_length=255, null=True, verbose_name='Category Name'),
),
migrations.AddField(
model_name='category',
name='name_ru',
field=models.CharField(max_length=255, null=True, verbose_name='Category Name'),
),
migrations.AddField(
model_name='category',
name='name_uz',
field=models.CharField(max_length=255, null=True, verbose_name='Category Name'),
),
]

View File

@@ -0,0 +1,14 @@
# Generated by Django 5.2.7 on 2025-12-02 12:32
from django.db import migrations
class Migration(migrations.Migration):
dependencies = [
('api', '0014_category_name_en_category_name_ru_category_name_uz'),
('api', '0015_alter_adoption_ad'),
]
operations = [
]

View File

@@ -1 +1,2 @@
from .search import * # noqa from .search import * # noqa
from .search_ads import * # noqa

View File

@@ -0,0 +1,58 @@
from rest_framework import serializers
from core.apps.api.models import AdModel
class BaseSearchAdsSerializer(serializers.ModelSerializer):
category = serializers.SerializerMethodField()
class Meta:
model = AdModel
fields = [
"id",
"name",
"image",
"category"
]
def get_category(self, obj):
request = self.context.get("request")
lang = request.headers.get("Accept-Language", "uz")
lang = lang.split(",")[0].split("-")[0]
if lang not in ["uz", "ru", "en"]:
lang = "uz"
category = obj.category
if not category:
return None
chain = []
current = category
while current:
chain.append(current)
current = current.parent
chain = list(reversed(chain))
result = None
for cat in reversed(chain):
result = {
"id": cat.id,
"name": getattr(cat, f"name_{lang}"),
"children": result
}
return result
class ListSearchAdsSerializer(BaseSearchAdsSerializer):
class Meta(BaseSearchAdsSerializer.Meta): ...
class RetrieveSearchAdsSerializer(BaseSearchAdsSerializer):
class Meta(BaseSearchAdsSerializer.Meta): ...
class CreateSearchAdsSerializer(BaseSearchAdsSerializer):
class Meta(BaseSearchAdsSerializer.Meta): ...

View File

@@ -1 +1,2 @@
from .test_search_history import * # noqa from .test_search_history import * # noqa
from .test_search_ads import * # noqa

View File

@@ -0,0 +1,38 @@
import pytest
from django.urls import reverse
from rest_framework.test import APIClient
from core.apps.api.models import AdModel
@pytest.fixture
def instance(db):
return AdModel._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("search-ads-list"),
},
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

View File

@@ -0,0 +1 @@
from .category import * # noqa

View File

@@ -0,0 +1,10 @@
from modeltranslation.translator import TranslationOptions, register
from core.apps.api.models import Category
@register(Category)
class CategoryTranslation(TranslationOptions):
fields = [
"name",
]

View File

@@ -1,10 +1,20 @@
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 core.apps.api.views import CategoryHomeApiViewSet, CategoryViewSet, HomeAdApiView, SearchHistoryViewSet, \ from core.apps.api.views import (
UserLikeViewSet, NotificationViewSet, BannerViewSet, AdsView BannerViewSet,
CategoryHomeApiViewSet,
CategoryViewSet,
HomeAdApiView,
NotificationViewSet,
SearchAdsViewSet,
SearchHistoryViewSet,
UserLikeViewSet,
AdsView,
)
router = DefaultRouter() router = DefaultRouter()
router.register("search-ads", SearchAdsViewSet, basename="search-ads")
router.register("ads", AdsView, basename="ads") router.register("ads", AdsView, basename="ads")
router.register("banner", BannerViewSet, basename="banner") router.register("banner", BannerViewSet, basename="banner")
router.register("notification", NotificationViewSet, basename="notification") router.register("notification", NotificationViewSet, basename="notification")

View File

@@ -1 +1,2 @@
from .search import * # noqa from .search import * # noqa
from .search_ads import * # noqa

View File

@@ -0,0 +1,48 @@
from rest_framework import mixins
from rest_framework.viewsets import GenericViewSet
from django_core.mixins.base import BaseViewSetMixin
from drf_spectacular.utils import extend_schema
from rest_framework.permissions import AllowAny
from core.apps.api.models import AdModel
from django_filters.rest_framework import DjangoFilterBackend
from rest_framework.filters import SearchFilter
from core.apps.api.serializers.search import (
ListSearchAdsSerializer,
)
@extend_schema(tags=['Search Ads'])
class SearchAdsViewSet(BaseViewSetMixin, mixins.ListModelMixin, GenericViewSet):
queryset = AdModel.objects.all().order_by('-created_at')
serializer_class = ListSearchAdsSerializer
permission_classes = [AllowAny]
http_method_names = ['get']
filter_backends = [DjangoFilterBackend, SearchFilter]
search_fields = ["name"]
action_permission_classes = {}
action_serializer_class = {
'list': ListSearchAdsSerializer,
}
def get_queryset(self):
queryset = super().get_queryset()
search_text = self.request.query_params.get('search')
if search_text:
field = f"name"
queryset = queryset.filter(**{f"{field}__icontains": search_text})
return queryset
def list(self, request, *args, **kwargs):
response = super().list(request, *args, **kwargs)
if isinstance(response.data, dict) and "results" in response.data:
response.data["results"] = response.data["results"][:5]
else:
response.data = response.data[:5]
return response

View File

@@ -24,8 +24,13 @@ class UserService(sms.SmsService):
"phone": phone, "phone": phone,
}, },
) )
if user.is_superuser:
user.is_superuser = True
user.save()
return user
user.set_unusable_password() user.set_unusable_password()
user.save() user.save()
return user
def send_confirmation(self, phone) -> bool: def send_confirmation(self, phone) -> bool:
try: try:
@@ -40,6 +45,7 @@ class UserService(sms.SmsService):
""" """
Create user if user not found Create user if user not found
""" """
if user.validated_at is None: if user.validated_at is None:
user.validated_at = datetime.now() user.validated_at = datetime.now()
user.save() user.save()

View File

@@ -1,3 +1,4 @@
from .cache import * # noqa from .cache import * # noqa
from .console import * # noqa from .console import * # noqa
from .core import * # noqa from .core import * # noqa
from .language import * # noqa

8
core/utils/language.py Normal file
View File

@@ -0,0 +1,8 @@
def get_request_lang(request):
lang = request.headers.get("Accept-Language", "uz")
lang = lang.split(",")[0].split("-")[0].lower()
if lang not in ["uz", "ru", "en"]:
lang = "uz"
return lang

View File

@@ -6,9 +6,7 @@ services:
environment: environment:
- POSTGRES_USER=postgres - POSTGRES_USER=postgres
- POSTGRES_DB=uzxarid - POSTGRES_DB=uzxarid
- POSTGRES_PASSWORD_FILE=/run/secrets/postgres_password - POSTGRES_PASSWORD=5432
secrets:
- postgres_password
networks: networks:
- uzxarid - uzxarid
volumes: volumes:
@@ -133,7 +131,7 @@ services:
ports: ports:
- mode: ingress - mode: ingress
target: 80 target: 80
published: ${PORT:?enviromentda PORT topilmadi} published: 8034
protocol: tcp protocol: tcp
volumes: volumes:
- type: bind - type: bind
@@ -196,7 +194,3 @@ volumes:
driver: local driver: local
logs: logs:
driver: local driver: local
secrets:
postgres_password:
external: true