update blog performance and fix blog view

This commit is contained in:
Parsa Nazer
2026-05-06 10:26:14 +03:30
parent 908535da35
commit 74554a664a
3 changed files with 52 additions and 34 deletions
+12 -5
View File
@@ -14,18 +14,25 @@ class AuthorSerializer(serializers.ModelSerializer):
else:
return 'ادمین وبسایت'
class BlogCategorySerilizer(serializers.ModelSerializer):
name = serializers.SerializerMethodField()
class Meta:
model = BlogCategoryModel
fields = ['id', 'name']
def get_name(self, obj):
return obj.title
class BlogSerilizer(serializers.ModelSerializer):
category = SubCategorySerializer()
category = BlogCategorySerilizer()
author = AuthorSerializer()
class Meta:
model = BlogModel
exclude = ('is_published',)
class BlogCategorySerilizer(serializers.ModelSerializer):
class Meta:
model = BlogCategoryModel
fields = '__all__'
class AllBlogSerilizer(serializers.ModelSerializer):
+2 -2
View File
@@ -1,8 +1,8 @@
from django.urls import path
from django.urls import path, re_path
from . import views
urlpatterns = [
path('all', views.AllBlogView.as_view(), name='product-chat-view'),
path('categories', views.AllBlogCategoryView.as_view()),
path('<int:slug>', views.BlogView.as_view(), name='product-chat-view'),
re_path(r'^(?P<slug>[\w\u0600-\u06FF\-]+)$', views.BlogView.as_view(), name='blog-view'),
]
+38 -27
View File
@@ -6,15 +6,19 @@ from .serializers import AllBlogSerilizer, BlogSerilizer, AllBlogCategorySeriliz
from django.shortcuts import get_object_or_404
from utils.pagination import StructurePagination
from drf_spectacular.utils import extend_schema, OpenApiParameter, OpenApiTypes
from django.db.models import Q
from django.db.models import Q, F
from django.views.decorators.cache import cache_page
from django.utils.decorators import method_decorator
from django.core.cache import cache
class AllBlogCategoryView(APIView):
serializer_class = AllBlogCategorySerilizer
authentication_classes = []
@method_decorator(cache_page(60 * 60)) # Cache for 1 hour
def get(self, request):
categories = BlogCategoryModel.objects.all()
categories = BlogCategoryModel.objects.all().only('id', 'name')
category_ser = self.serializer_class(instance=categories, many=True, context={'request': request})
return Response(category_ser.data, status=status.HTTP_200_OK)
@@ -23,29 +27,30 @@ class AllBlogView(APIView):
serializer_class = AllBlogSerilizer
pagination_class = StructurePagination
authentication_classes = []
@extend_schema(
parameters=[
OpenApiParameter(
name="search",
description="بگرددددد تو بلاااااگگگووووو",
description="بگردش در بلاگ",
required=False,
type=OpenApiTypes.STR,
),
OpenApiParameter(
name="category_id",
description="",
description="فیلتر بر اساس دسته بندی",
required=False,
type=OpenApiTypes.STR,
),
OpenApiParameter(
name="limit",
description="لیمیتش",
description="تعداد نتایج",
required=False,
type=OpenApiTypes.INT,
),
OpenApiParameter(
name="offset",
description="افستش",
description="شروع از",
required=False,
type=OpenApiTypes.INT,
)
@@ -56,25 +61,30 @@ class AllBlogView(APIView):
},
)
def get(self, request):
blogs = BlogModel.objects.filter(is_published=True)
# Use only() to fetch required fields
blogs = BlogModel.objects.filter(is_published=True).select_related('category').only(
'id', 'title', 'slug', 'summery', 'cover_image', 'views', 'created_at', 'category'
)
search_query = request.query_params.get('search', None)
category_id = request.query_params.get('category_id', None)
if search_query:
blogs = blogs.filter(Q(title__icontains=search_query) | Q(content__icontains=search_query))
blogs = blogs.filter(Q(title__icontains=search_query) | Q(summery__icontains=search_query))
if category_id:
category_obj = get_object_or_404(BlogCategoryModel, pk=category_id)
blogs.filter(category=category_obj)
blogs = blogs.filter(category_id=category_id) # Use category_id directly
paginator = self.pagination_class()
paginated_blogs = paginator.paginate_queryset(blogs, request)
blog_ser = self.serializer_class(instance=paginated_blogs, many=True, context={'request': request})
return paginator.get_paginated_response(blog_ser.data)
class BlogView(APIView):
serializer_class = BlogSerilizer
authentication_classes = []
def get_client_ip(self, request):
"""Helper function to get the client IP from request headers."""
x_forwarded_for = request.META.get('HTTP_X_FORWARDED_FOR')
@@ -84,24 +94,25 @@ class BlogView(APIView):
ip = request.META.get('REMOTE_ADDR')
return ip
def get(self, request, slug):
blog = get_object_or_404(BlogModel, slug=slug)
if blog.is_published:
# Track views using session
client_ip = self.get_client_ip(request)
session_key = f'viewed_blog_{slug}_{client_ip}'
# Use only() to fetch required fields
blog = get_object_or_404(BlogModel.objects.select_related('category', 'author'), slug=slug)
if not request.session.get(session_key):
blog.views += 1
blog.save()
print(f'views {blog.views}')
print(session_key)
request.session[session_key] = True
request.session.set_expiry(3600)
if blog.is_published:
# Track views using cache instead of session (more efficient)
client_ip = self.get_client_ip(request)
cache_key = f'viewed_blog_{slug}_{client_ip}'
if not cache.get(cache_key):
# Use F() to avoid race conditions
BlogModel.objects.filter(pk=blog.pk).update(views=F('views') + 1)
blog.refresh_from_db(fields=['views'])
cache.set(cache_key, True, 3600) # Cache for 1 hour
blog_ser = self.serializer_class(instance=blog, context={'request': request})
return Response(blog_ser.data, status=status.HTTP_200_OK)
else:
return Response({'detail': 'object with the given id does not exist or is not published yet'},
status=status.HTTP_404_NOT_FOUND)
return Response(
{'detail': 'object with the given id does not exist or is not published yet'},
status=status.HTTP_404_NOT_FOUND
)