Merge remote-tracking branch 'origin/develop' into feature/crop-params
This commit is contained in:
commit
797327802b
|
|
@ -55,3 +55,23 @@ class EstablishmentTypeTagFilter(filters.FilterSet):
|
|||
fields = (
|
||||
'type_id',
|
||||
)
|
||||
|
||||
|
||||
class EmployeeBackFilter(filters.FilterSet):
|
||||
"""Employee filter set."""
|
||||
|
||||
search = filters.CharFilter(method='search_by_name_or_last_name')
|
||||
|
||||
class Meta:
|
||||
"""Meta class."""
|
||||
|
||||
model = models.Employee
|
||||
fields = (
|
||||
'search',
|
||||
)
|
||||
|
||||
def search_by_name_or_last_name(self, queryset, name, value):
|
||||
"""Search by name or last name."""
|
||||
if value not in EMPTY_VALUES:
|
||||
return queryset.search_by_name_or_last_name(value)
|
||||
return queryset
|
||||
|
|
|
|||
28
apps/establishment/migrations/0066_auto_20191122_1144.py
Normal file
28
apps/establishment/migrations/0066_auto_20191122_1144.py
Normal file
|
|
@ -0,0 +1,28 @@
|
|||
# Generated by Django 2.2.7 on 2019-11-22 11:44
|
||||
|
||||
from django.db import migrations, models
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('establishment', '0065_establishment_purchased_products'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='employee',
|
||||
name='last_name',
|
||||
field=models.CharField(default=None, max_length=255, null=True, verbose_name='Last Name'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='establishmentemployee',
|
||||
name='status',
|
||||
field=models.CharField(choices=[('I', 'Idle'), ('A', 'Accepted'), ('D', 'Declined')], default='I', max_length=1),
|
||||
),
|
||||
migrations.AlterField(
|
||||
model_name='employee',
|
||||
name='name',
|
||||
field=models.CharField(max_length=255, verbose_name='Name'),
|
||||
),
|
||||
]
|
||||
39
apps/establishment/migrations/0067_auto_20191122_1244.py
Normal file
39
apps/establishment/migrations/0067_auto_20191122_1244.py
Normal file
|
|
@ -0,0 +1,39 @@
|
|||
# Generated by Django 2.2.7 on 2019-11-22 12:44
|
||||
|
||||
from django.db import migrations, models
|
||||
import phonenumber_field.modelfields
|
||||
|
||||
|
||||
class Migration(migrations.Migration):
|
||||
|
||||
dependencies = [
|
||||
('establishment', '0066_auto_20191122_1144'),
|
||||
]
|
||||
|
||||
operations = [
|
||||
migrations.AddField(
|
||||
model_name='employee',
|
||||
name='birth_date',
|
||||
field=models.DateTimeField(default=None, null=True, verbose_name='Birth date'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='employee',
|
||||
name='email',
|
||||
field=models.EmailField(blank=True, default=None, max_length=254, null=True, verbose_name='Email'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='employee',
|
||||
name='phone',
|
||||
field=phonenumber_field.modelfields.PhoneNumberField(default=None, max_length=128, null=True),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='employee',
|
||||
name='sex',
|
||||
field=models.PositiveSmallIntegerField(choices=[(0, 'Male'), (1, 'Female')], default=None, null=True, verbose_name='Sex'),
|
||||
),
|
||||
migrations.AddField(
|
||||
model_name='employee',
|
||||
name='toque_number',
|
||||
field=models.PositiveSmallIntegerField(default=None, null=True, verbose_name='Toque number'),
|
||||
),
|
||||
]
|
||||
|
|
@ -1,6 +1,7 @@
|
|||
"""Establishment models."""
|
||||
from datetime import datetime
|
||||
from functools import reduce
|
||||
from typing import List
|
||||
from operator import or_
|
||||
|
||||
import elasticsearch_dsl
|
||||
|
|
@ -397,6 +398,7 @@ class Establishment(GalleryModelMixin, ProjectBaseMixin, URLImageMixin,
|
|||
verbose_name=_('Tag'))
|
||||
reviews = generic.GenericRelation(to='review.Review')
|
||||
comments = generic.GenericRelation(to='comment.Comment')
|
||||
carousels = generic.GenericRelation(to='main.Carousel')
|
||||
favorites = generic.GenericRelation(to='favorites.Favorites')
|
||||
currency = models.ForeignKey(Currency, blank=True, null=True, default=None,
|
||||
on_delete=models.PROTECT,
|
||||
|
|
@ -436,11 +438,12 @@ class Establishment(GalleryModelMixin, ProjectBaseMixin, URLImageMixin,
|
|||
|
||||
@property
|
||||
def visible_tags(self):
|
||||
return super().visible_tags\
|
||||
.exclude(category__index_name__in=['guide', 'collection', 'purchased_item',
|
||||
'business_tag', 'business_tags_de'])\
|
||||
return super().visible_tags \
|
||||
.exclude(category__index_name__in=['guide', 'collection', 'purchased_item',
|
||||
'business_tag', 'business_tags_de']) \
|
||||
\
|
||||
# todo: recalculate toque_number
|
||||
|
||||
# todo: recalculate toque_number
|
||||
def recalculate_toque_number(self):
|
||||
toque_number = 0
|
||||
if self.address and self.public_mark:
|
||||
|
|
@ -613,7 +616,6 @@ class EstablishmentNote(ProjectBaseMixin):
|
|||
|
||||
|
||||
class EstablishmentGallery(IntermediateGalleryModelMixin):
|
||||
|
||||
establishment = models.ForeignKey(Establishment, null=True,
|
||||
related_name='establishment_gallery',
|
||||
on_delete=models.CASCADE,
|
||||
|
|
@ -664,6 +666,16 @@ class EstablishmentEmployeeQuerySet(models.QuerySet):
|
|||
class EstablishmentEmployee(BaseAttributes):
|
||||
"""EstablishmentEmployee model."""
|
||||
|
||||
IDLE = 'I'
|
||||
ACCEPTED = 'A'
|
||||
DECLINED = 'D'
|
||||
|
||||
STATUS_CHOICES = (
|
||||
(IDLE, 'Idle'),
|
||||
(ACCEPTED, 'Accepted'),
|
||||
(DECLINED, 'Declined'),
|
||||
)
|
||||
|
||||
establishment = models.ForeignKey(Establishment, on_delete=models.PROTECT,
|
||||
verbose_name=_('Establishment'))
|
||||
employee = models.ForeignKey('establishment.Employee', on_delete=models.PROTECT,
|
||||
|
|
@ -674,19 +686,53 @@ class EstablishmentEmployee(BaseAttributes):
|
|||
verbose_name=_('To date'))
|
||||
position = models.ForeignKey(Position, on_delete=models.PROTECT,
|
||||
verbose_name=_('Position'))
|
||||
|
||||
status = models.CharField(max_length=1, choices=STATUS_CHOICES, default=IDLE)
|
||||
|
||||
# old_id = affiliations_id
|
||||
old_id = models.IntegerField(verbose_name=_('Old id'), null=True, blank=True)
|
||||
|
||||
objects = EstablishmentEmployeeQuerySet.as_manager()
|
||||
|
||||
|
||||
class EmployeeQuerySet(models.QuerySet):
|
||||
|
||||
def _generic_search(self, value, filter_fields_names: List[str]):
|
||||
"""Generic method for searching value in specified fields"""
|
||||
filters = [
|
||||
{f'{field}__icontains': value}
|
||||
for field in filter_fields_names
|
||||
]
|
||||
return self.filter(reduce(lambda x, y: x | y, [models.Q(**i) for i in filters]))
|
||||
|
||||
def search_by_name_or_last_name(self, value):
|
||||
"""Search by name or last_name."""
|
||||
return self._generic_search(value, ['name', 'last_name'])
|
||||
|
||||
|
||||
class Employee(BaseAttributes):
|
||||
"""Employee model."""
|
||||
|
||||
user = models.OneToOneField('account.User', on_delete=models.PROTECT,
|
||||
null=True, blank=True, default=None,
|
||||
verbose_name=_('User'))
|
||||
name = models.CharField(max_length=255, verbose_name=_('Last name'))
|
||||
name = models.CharField(max_length=255, verbose_name=_('Name'))
|
||||
last_name = models.CharField(max_length=255, verbose_name=_('Last Name'), null=True, default=None)
|
||||
|
||||
# SEX CHOICES
|
||||
MALE = 0
|
||||
FEMALE = 1
|
||||
|
||||
SEX_CHOICES = (
|
||||
(MALE, _('Male')),
|
||||
(FEMALE, _('Female'))
|
||||
)
|
||||
sex = models.PositiveSmallIntegerField(choices=SEX_CHOICES, verbose_name=_('Sex'), null=True, default=None)
|
||||
birth_date = models.DateTimeField(editable=True, verbose_name=_('Birth date'), null=True, default=None)
|
||||
email = models.EmailField(blank=True, null=True, default=None, verbose_name=_('Email'))
|
||||
phone = PhoneNumberField(null=True, default=None)
|
||||
toque_number = models.PositiveSmallIntegerField(verbose_name=_('Toque number'), null=True, default=None)
|
||||
|
||||
establishments = models.ManyToManyField(Establishment, related_name='employees',
|
||||
through=EstablishmentEmployee, )
|
||||
awards = generic.GenericRelation(to='main.Award', related_query_name='employees')
|
||||
|
|
@ -695,6 +741,8 @@ class Employee(BaseAttributes):
|
|||
# old_id = profile_id
|
||||
old_id = models.IntegerField(verbose_name=_('Old id'), null=True, blank=True)
|
||||
|
||||
objects = EmployeeQuerySet.as_manager()
|
||||
|
||||
class Meta:
|
||||
"""Meta class."""
|
||||
|
||||
|
|
|
|||
|
|
@ -2,8 +2,9 @@ from rest_framework import serializers
|
|||
|
||||
from establishment import models
|
||||
from establishment import serializers as model_serializers
|
||||
from location.serializers import AddressDetailSerializer
|
||||
from location.serializers import AddressDetailSerializer, TranslatedField
|
||||
from main.models import Currency
|
||||
from main.serializers import AwardSerializer
|
||||
from utils.decorators import with_base_attributes
|
||||
from utils.serializers import TimeZoneChoiceField
|
||||
from gallery.models import Image
|
||||
|
|
@ -161,12 +162,53 @@ class ContactEmailBackSerializers(model_serializers.PlateSerializer):
|
|||
class EmployeeBackSerializers(serializers.ModelSerializer):
|
||||
"""Employee serializers."""
|
||||
|
||||
awards = AwardSerializer(many=True)
|
||||
|
||||
class Meta:
|
||||
model = models.Employee
|
||||
fields = [
|
||||
'id',
|
||||
'user',
|
||||
'name'
|
||||
'name',
|
||||
'last_name',
|
||||
'sex',
|
||||
'birth_date',
|
||||
'email',
|
||||
'phone',
|
||||
'toque_number',
|
||||
'awards',
|
||||
]
|
||||
|
||||
|
||||
class PositionBackSerializer(serializers.ModelSerializer):
|
||||
"""Position Back serializer."""
|
||||
|
||||
name_translated = TranslatedField()
|
||||
|
||||
class Meta:
|
||||
model = models.Position
|
||||
fields = [
|
||||
'id',
|
||||
'name_translated',
|
||||
'priority',
|
||||
'index_name',
|
||||
]
|
||||
|
||||
|
||||
class EstablishmentEmployeeBackSerializer(serializers.ModelSerializer):
|
||||
"""Establishment Employee serializer."""
|
||||
|
||||
employee = EmployeeBackSerializers()
|
||||
position = PositionBackSerializer()
|
||||
|
||||
class Meta:
|
||||
model = models.EstablishmentEmployee
|
||||
fields = [
|
||||
'id',
|
||||
'employee',
|
||||
'from_date',
|
||||
'to_date',
|
||||
'position',
|
||||
]
|
||||
|
||||
|
||||
|
|
|
|||
|
|
@ -13,7 +13,7 @@ from review.serializers import ReviewShortSerializer
|
|||
from tag.serializers import TagBaseSerializer
|
||||
from timetable.serialziers import ScheduleRUDSerializer
|
||||
from utils import exceptions as utils_exceptions
|
||||
from utils.serializers import ImageBaseSerializer
|
||||
from utils.serializers import ImageBaseSerializer, CarouselCreateSerializer
|
||||
from utils.serializers import (ProjectModelSerializer, TranslatedField,
|
||||
FavoritesCreateSerializer)
|
||||
|
||||
|
|
@ -168,12 +168,51 @@ class EstablishmentEmployeeSerializer(serializers.ModelSerializer):
|
|||
awards = AwardSerializer(source='employee.awards', many=True)
|
||||
priority = serializers.IntegerField(source='position.priority')
|
||||
position_index_name = serializers.CharField(source='position.index_name')
|
||||
status = serializers.CharField()
|
||||
|
||||
class Meta:
|
||||
"""Meta class."""
|
||||
|
||||
model = models.Employee
|
||||
fields = ('id', 'name', 'position_translated', 'awards', 'priority', 'position_index_name')
|
||||
fields = ('id', 'name', 'position_translated', 'awards', 'priority', 'position_index_name', 'status')
|
||||
|
||||
|
||||
class EstablishmentEmployeeCreateSerializer(serializers.ModelSerializer):
|
||||
"""Serializer for establishment employee relation."""
|
||||
|
||||
class Meta:
|
||||
"""Meta class."""
|
||||
|
||||
model = models.EstablishmentEmployee
|
||||
fields = ('id',)
|
||||
|
||||
def _validate_entity(self, entity_id_param: str, entity_class):
|
||||
entity_id = self.context.get('request').parser_context.get('kwargs').get(entity_id_param)
|
||||
entity_qs = entity_class.objects.filter(id=entity_id)
|
||||
if not entity_qs.exists():
|
||||
raise serializers.ValidationError({'detail': _(f'{entity_class.__name__} not found.')})
|
||||
return entity_qs.first()
|
||||
|
||||
def validate(self, attrs):
|
||||
"""Override validate method"""
|
||||
establishment = self._validate_entity("establishment_id", models.Establishment)
|
||||
employee = self._validate_entity("employee_id", models.Employee)
|
||||
position = self._validate_entity("position_id", models.Position)
|
||||
|
||||
attrs['establishment'] = establishment
|
||||
attrs['employee'] = employee
|
||||
attrs['position'] = position
|
||||
|
||||
return attrs
|
||||
|
||||
def create(self, validated_data, *args, **kwargs):
|
||||
"""Override create method"""
|
||||
validated_data.update({
|
||||
'employee': validated_data.pop('employee'),
|
||||
'establishment': validated_data.pop('establishment'),
|
||||
'position': validated_data.pop("position")
|
||||
})
|
||||
return super().create(validated_data)
|
||||
|
||||
|
||||
class EstablishmentShortSerializer(serializers.ModelSerializer):
|
||||
|
|
@ -396,6 +435,22 @@ class EstablishmentCommentCreateSerializer(comment_serializers.CommentSerializer
|
|||
return super().create(validated_data)
|
||||
|
||||
|
||||
class EstablishmentCommentRUDSerializer(comment_serializers.CommentSerializer):
|
||||
"""Retrieve/Update/Destroy comment serializer."""
|
||||
|
||||
class Meta:
|
||||
"""Meta class."""
|
||||
model = comment_models.Comment
|
||||
fields = [
|
||||
'id',
|
||||
'created',
|
||||
'text',
|
||||
'mark',
|
||||
'nickname',
|
||||
'profile_pic',
|
||||
]
|
||||
|
||||
|
||||
class EstablishmentFavoritesCreateSerializer(FavoritesCreateSerializer):
|
||||
"""Serializer to favorite object w/ model Establishment."""
|
||||
|
||||
|
|
@ -426,6 +481,27 @@ class EstablishmentFavoritesCreateSerializer(FavoritesCreateSerializer):
|
|||
return super().create(validated_data)
|
||||
|
||||
|
||||
class EstablishmentCarouselCreateSerializer(CarouselCreateSerializer):
|
||||
"""Serializer to carousel object w/ model News."""
|
||||
|
||||
def validate(self, attrs):
|
||||
establishment = models.Establishment.objects.filter(slug=self.slug).first()
|
||||
if not establishment:
|
||||
raise serializers.ValidationError({'detail': _('Object not found.')})
|
||||
|
||||
if establishment.carousels.exists():
|
||||
raise utils_exceptions.CarouselError()
|
||||
|
||||
attrs['establishment'] = establishment
|
||||
return attrs
|
||||
|
||||
def create(self, validated_data, *args, **kwargs):
|
||||
validated_data.update({
|
||||
'content_object': validated_data.pop('establishment')
|
||||
})
|
||||
return super().create(validated_data)
|
||||
|
||||
|
||||
class CompanyBaseSerializer(serializers.ModelSerializer):
|
||||
"""Company base serializer"""
|
||||
phone_list = serializers.SerializerMethodField(source='phones', read_only=True)
|
||||
|
|
|
|||
|
|
@ -38,8 +38,16 @@ urlpatterns = [
|
|||
path('phones/<int:pk>/', views.PhonesRUDView.as_view(), name='phones-rud'),
|
||||
path('emails/', views.EmailListCreateView.as_view(), name='emails'),
|
||||
path('emails/<int:pk>/', views.EmailRUDView.as_view(), name='emails-rud'),
|
||||
path('<int:establishment_id>/employees/', views.EstablishmentEmployeeListView.as_view(),
|
||||
name='establishment-employees'),
|
||||
path('employees/', views.EmployeeListCreateView.as_view(), name='employees'),
|
||||
path('employees/<int:pk>/', views.EmployeeRUDView.as_view(), name='employees-rud'),
|
||||
path('<int:establishment_id>/employee/<int:employee_id>/position/<int:position_id>',
|
||||
views.EstablishmentEmployeeCreateView.as_view(),
|
||||
name='employees-establishment-create'),
|
||||
path('<int:establishment_id>/employee/<int:employee_id>',
|
||||
views.EstablishmentEmployeeDeleteView.as_view(),
|
||||
name='employees-establishment-delete'),
|
||||
path('types/', views.EstablishmentTypeListCreateView.as_view(), name='type-list'),
|
||||
path('types/<int:pk>/', views.EstablishmentTypeRUDView.as_view(), name='type-rud'),
|
||||
path('subtypes/', views.EstablishmentSubtypeListCreateView.as_view(), name='subtype-list'),
|
||||
|
|
|
|||
|
|
@ -17,5 +17,7 @@ urlpatterns = [
|
|||
path('slug/<slug:slug>/comments/<int:comment_id>/', views.EstablishmentCommentRUDView.as_view(),
|
||||
name='rud-comment'),
|
||||
path('slug/<slug:slug>/favorites/', views.EstablishmentFavoritesCreateDestroyView.as_view(),
|
||||
name='create-destroy-favorites')
|
||||
name='create-destroy-favorites'),
|
||||
path('slug/<slug:slug>/carousels/', views.EstablishmentCarouselCreateDestroyView.as_view(),
|
||||
name='create-destroy-carousels')
|
||||
]
|
||||
|
|
|
|||
|
|
@ -1,7 +1,9 @@
|
|||
"""Establishment app views."""
|
||||
from django.http import Http404, HttpResponse
|
||||
from django.shortcuts import get_object_or_404
|
||||
from rest_framework import generics, permissions
|
||||
from rest_framework import generics, permissions, status
|
||||
|
||||
from utils.permissions import IsCountryAdmin, IsEstablishmentManager
|
||||
from establishment import filters, models, serializers
|
||||
from timetable.serialziers import ScheduleRUDSerializer, ScheduleCreateSerializer
|
||||
from utils.permissions import IsCountryAdmin, IsEstablishmentManager
|
||||
|
|
@ -43,8 +45,8 @@ class EstablishmentScheduleRUDView(generics.RetrieveUpdateDestroyAPIView):
|
|||
"""
|
||||
Returns the object the view is displaying.
|
||||
"""
|
||||
establishment_pk = self.kwargs.get('pk')
|
||||
schedule_id = self.kwargs.get('schedule_id')
|
||||
establishment_pk = self.kwargs['pk']
|
||||
schedule_id = self.kwargs['schedule_id']
|
||||
|
||||
establishment = get_object_or_404(klass=models.Establishment.objects.all(),
|
||||
pk=establishment_pk)
|
||||
|
|
@ -156,11 +158,23 @@ class EmailRUDView(generics.RetrieveUpdateDestroyAPIView):
|
|||
|
||||
class EmployeeListCreateView(generics.ListCreateAPIView):
|
||||
"""Emplyoee list create view."""
|
||||
permission_classes = (permissions.AllowAny, )
|
||||
filter_class = filters.EmployeeBackFilter
|
||||
serializer_class = serializers.EmployeeBackSerializers
|
||||
queryset = models.Employee.objects.all()
|
||||
pagination_class = None
|
||||
|
||||
|
||||
class EstablishmentEmployeeListView(generics.ListAPIView):
|
||||
"""Establishment emplyoees list view."""
|
||||
permission_classes = (permissions.AllowAny, )
|
||||
serializer_class = serializers.EstablishmentEmployeeBackSerializer
|
||||
|
||||
def get_queryset(self):
|
||||
establishment_id = self.kwargs['establishment_id']
|
||||
return models.EstablishmentEmployee.objects.filter(establishment__id=establishment_id)
|
||||
|
||||
|
||||
class EmployeeRUDView(generics.RetrieveUpdateDestroyAPIView):
|
||||
"""Employee RUD view."""
|
||||
serializer_class = serializers.EmployeeBackSerializers
|
||||
|
|
@ -318,3 +332,27 @@ class EstablishmentNoteRUDView(EstablishmentMixinViews,
|
|||
self.check_object_permissions(self.request, note)
|
||||
|
||||
return note
|
||||
|
||||
|
||||
class EstablishmentEmployeeCreateView(generics.CreateAPIView):
|
||||
serializer_class = serializers.EstablishmentEmployeeCreateSerializer
|
||||
queryset = models.EstablishmentEmployee.objects.all()
|
||||
# TODO send email to all admins and add endpoint for changing status
|
||||
|
||||
|
||||
class EstablishmentEmployeeDeleteView(generics.DestroyAPIView):
|
||||
|
||||
def _get_object_to_delete(self, establishment_id, employee_id):
|
||||
result_qs = models.EstablishmentEmployee\
|
||||
.objects\
|
||||
.filter(establishment_id=establishment_id, employee_id=employee_id)
|
||||
if not result_qs.exists():
|
||||
raise Http404
|
||||
return result_qs.first()
|
||||
|
||||
def delete(self, request, *args, **kwargs):
|
||||
establishment_id = self.kwargs["establishment_id"]
|
||||
employee_id = self.kwargs["employee_id"]
|
||||
object_to_delete = self._get_object_to_delete(establishment_id, employee_id)
|
||||
object_to_delete.delete()
|
||||
return HttpResponse(status=status.HTTP_204_NO_CONTENT)
|
||||
|
|
|
|||
|
|
@ -9,7 +9,7 @@ from comment.serializers import CommentRUDSerializer
|
|||
from establishment import filters, models, serializers
|
||||
from main import methods
|
||||
from utils.pagination import EstablishmentPortionPagination
|
||||
from utils.views import FavoritesCreateDestroyMixinView
|
||||
from utils.views import FavoritesCreateDestroyMixinView, CarouselCreateDestroyMixinView
|
||||
|
||||
|
||||
class EstablishmentMixinView:
|
||||
|
|
@ -34,7 +34,7 @@ class EstablishmentListView(EstablishmentMixinView, generics.ListAPIView):
|
|||
serializer_class = serializers.EstablishmentListRetrieveSerializer
|
||||
|
||||
def get_queryset(self):
|
||||
return super().get_queryset().with_schedule()\
|
||||
return super().get_queryset().with_schedule() \
|
||||
.with_extended_address_related().with_currency_related()
|
||||
|
||||
|
||||
|
|
@ -105,9 +105,9 @@ class EstablishmentCommentListView(generics.ListAPIView):
|
|||
|
||||
establishment = get_object_or_404(models.Establishment, slug=self.kwargs['slug'])
|
||||
return comment_models.Comment.objects.by_content_type(app_label='establishment',
|
||||
model='establishment')\
|
||||
.by_object_id(object_id=establishment.pk)\
|
||||
.order_by('-created')
|
||||
model='establishment') \
|
||||
.by_object_id(object_id=establishment.pk) \
|
||||
.order_by('-created')
|
||||
|
||||
|
||||
class EstablishmentCommentRUDView(generics.RetrieveUpdateDestroyAPIView):
|
||||
|
|
@ -139,6 +139,13 @@ class EstablishmentFavoritesCreateDestroyView(FavoritesCreateDestroyMixinView):
|
|||
serializer_class = serializers.EstablishmentFavoritesCreateSerializer
|
||||
|
||||
|
||||
class EstablishmentCarouselCreateDestroyView(CarouselCreateDestroyMixinView):
|
||||
"""View for create/destroy establishment from carousel."""
|
||||
|
||||
_model = models.Establishment
|
||||
serializer_class = serializers.EstablishmentCarouselCreateSerializer
|
||||
|
||||
|
||||
class EstablishmentNearestRetrieveView(EstablishmentListView, generics.ListAPIView):
|
||||
"""Resource for getting list of nearest establishments."""
|
||||
|
||||
|
|
|
|||
|
|
@ -3,9 +3,15 @@ from django.contrib import admin
|
|||
from main import models
|
||||
|
||||
|
||||
class SiteSettingsInline(admin.TabularInline):
|
||||
model = models.SiteFeature
|
||||
extra = 1
|
||||
|
||||
|
||||
@admin.register(models.SiteSettings)
|
||||
class SiteSettingsAdmin(admin.ModelAdmin):
|
||||
"""Site settings admin conf."""
|
||||
inlines = [SiteSettingsInline,]
|
||||
|
||||
|
||||
@admin.register(models.Feature)
|
||||
|
|
|
|||
|
|
@ -11,7 +11,7 @@ from news import models
|
|||
from tag.serializers import TagBaseSerializer
|
||||
from utils import exceptions as utils_exceptions
|
||||
from utils.serializers import (TranslatedField, ProjectModelSerializer,
|
||||
FavoritesCreateSerializer, ImageBaseSerializer)
|
||||
FavoritesCreateSerializer, ImageBaseSerializer, CarouselCreateSerializer)
|
||||
|
||||
|
||||
class AgendaSerializer(ProjectModelSerializer):
|
||||
|
|
@ -269,3 +269,24 @@ class NewsFavoritesCreateSerializer(FavoritesCreateSerializer):
|
|||
'content_object': validated_data.pop('news')
|
||||
})
|
||||
return super().create(validated_data)
|
||||
|
||||
|
||||
class NewsCarouselCreateSerializer(CarouselCreateSerializer):
|
||||
"""Serializer to carousel object w/ model News."""
|
||||
|
||||
def validate(self, attrs):
|
||||
news = models.News.objects.filter(slug=self.slug).first()
|
||||
if not news:
|
||||
raise serializers.ValidationError({'detail': _('Object not found.')})
|
||||
|
||||
if news.carousels.exists():
|
||||
raise utils_exceptions.CarouselError()
|
||||
|
||||
attrs['news'] = news
|
||||
return attrs
|
||||
|
||||
def create(self, validated_data, *args, **kwargs):
|
||||
validated_data.update({
|
||||
'content_object': validated_data.pop('news')
|
||||
})
|
||||
return super().create(validated_data)
|
||||
|
|
|
|||
|
|
@ -13,4 +13,6 @@ urlpatterns = [
|
|||
name='gallery-list'),
|
||||
path('<int:pk>/gallery/<int:image_id>/', views.NewsBackOfficeGalleryCreateDestroyView.as_view(),
|
||||
name='gallery-create-destroy'),
|
||||
path('slug/<slug:slug>/carousels/', views.NewsCarouselCreateDestroyView.as_view(),
|
||||
name='create-destroy-carousels'),
|
||||
]
|
||||
|
|
|
|||
|
|
@ -5,5 +5,8 @@ common_urlpatterns = [
|
|||
path('', views.NewsListView.as_view(), name='list'),
|
||||
path('types/', views.NewsTypeListView.as_view(), name='type'),
|
||||
path('slug/<slug:slug>/', views.NewsDetailView.as_view(), name='rud'),
|
||||
path('slug/<slug:slug>/favorites/', views.NewsFavoritesCreateDestroyView.as_view(), name='create-destroy-favorites')
|
||||
path('slug/<slug:slug>/favorites/', views.NewsFavoritesCreateDestroyView.as_view(),
|
||||
name='create-destroy-favorites'),
|
||||
path('slug/<slug:slug>/carousels/', views.NewsCarouselCreateDestroyView.as_view(),
|
||||
name='create-destroy-carousels'),
|
||||
]
|
||||
|
|
|
|||
|
|
@ -6,7 +6,7 @@ from rest_framework import generics, permissions
|
|||
from news import filters, models, serializers
|
||||
from rating.tasks import add_rating
|
||||
from utils.permissions import IsCountryAdmin, IsContentPageManager
|
||||
from utils.views import CreateDestroyGalleryViewMixin, FavoritesCreateDestroyMixinView
|
||||
from utils.views import CreateDestroyGalleryViewMixin, FavoritesCreateDestroyMixinView, CarouselCreateDestroyMixinView
|
||||
from utils.serializers import ImageBaseSerializer
|
||||
|
||||
|
||||
|
|
@ -155,3 +155,10 @@ class NewsFavoritesCreateDestroyView(FavoritesCreateDestroyMixinView):
|
|||
|
||||
_model = models.News
|
||||
serializer_class = serializers.NewsFavoritesCreateSerializer
|
||||
|
||||
|
||||
class NewsCarouselCreateDestroyView(CarouselCreateDestroyMixinView):
|
||||
"""View for create/destroy news from carousel."""
|
||||
|
||||
_model = models.News
|
||||
serializer_class = serializers.NewsCarouselCreateSerializer
|
||||
|
|
|
|||
|
|
@ -135,6 +135,14 @@ class FavoritesError(exceptions.APIException):
|
|||
default_detail = _('Item is already in favorites.')
|
||||
|
||||
|
||||
class CarouselError(exceptions.APIException):
|
||||
"""
|
||||
The exception should be thrown when the object is already in carousels.
|
||||
"""
|
||||
status_code = status.HTTP_400_BAD_REQUEST
|
||||
default_detail = _('Item is already in carousels.')
|
||||
|
||||
|
||||
class PasswordResetRequestExistedError(exceptions.APIException):
|
||||
"""
|
||||
The exception should be thrown when password reset request
|
||||
|
|
|
|||
|
|
@ -2,10 +2,11 @@
|
|||
import pytz
|
||||
from django.core import exceptions
|
||||
from rest_framework import serializers
|
||||
from utils import models
|
||||
from translation.models import Language
|
||||
|
||||
from favorites.models import Favorites
|
||||
from gallery.models import Image
|
||||
from main.models import Carousel
|
||||
from translation.models import Language
|
||||
from utils import models
|
||||
|
||||
|
||||
class EmptySerializer(serializers.Serializer):
|
||||
|
|
@ -80,7 +81,6 @@ class FavoritesCreateSerializer(serializers.ModelSerializer):
|
|||
"""Serializer to favorite object."""
|
||||
|
||||
class Meta:
|
||||
"""Serializer for model Comment."""
|
||||
model = Favorites
|
||||
fields = [
|
||||
'id',
|
||||
|
|
@ -101,6 +101,24 @@ class FavoritesCreateSerializer(serializers.ModelSerializer):
|
|||
return self.request.parser_context.get('kwargs').get('slug')
|
||||
|
||||
|
||||
class CarouselCreateSerializer(serializers.ModelSerializer):
|
||||
"""Carousel to favorite object."""
|
||||
|
||||
class Meta:
|
||||
model = Carousel
|
||||
fields = [
|
||||
'id',
|
||||
]
|
||||
|
||||
@property
|
||||
def request(self):
|
||||
return self.context.get('request')
|
||||
|
||||
@property
|
||||
def slug(self):
|
||||
return self.request.parser_context.get('kwargs').get('slug')
|
||||
|
||||
|
||||
class RecursiveFieldSerializer(serializers.Serializer):
|
||||
def to_representation(self, value):
|
||||
serializer = self.parent.parent.__class__(value, context=self.context)
|
||||
|
|
|
|||
|
|
@ -170,9 +170,10 @@ class CarouselCreateDestroyMixinView(BaseCreateDestroyMixinView):
|
|||
Returns the object the view is displaying.
|
||||
"""
|
||||
obj = self.get_base_object()
|
||||
carousels = get_object_or_404(obj.carousels.filter(user=self.request.user))
|
||||
carousels = get_object_or_404(obj.carousels.all())
|
||||
# May raise a permission denied
|
||||
self.check_object_permissions(self.request, carousels)
|
||||
# TODO: возможно нужны пермишены
|
||||
# self.check_object_permissions(self.request, carousels)
|
||||
return carousels
|
||||
|
||||
|
||||
|
|
|
|||
14
make_data_migration.sh
Executable file
14
make_data_migration.sh
Executable file
|
|
@ -0,0 +1,14 @@
|
|||
#!/usr/bin/env bash
|
||||
./manage.py transfer -a
|
||||
#./manage.py transfer -d
|
||||
./manage.py transfer -e
|
||||
./manage.py transfer --fill_city_gallery
|
||||
./manage.py transfer -l
|
||||
./manage.py transfer --product
|
||||
./manage.py transfer --souvenir
|
||||
./manage.py transfer --establishment_note
|
||||
./manage.py transfer --product_note
|
||||
./manage.py transfer --wine_characteristics
|
||||
./manage.py transfer --inquiries
|
||||
./manage.py transfer --assemblage
|
||||
./manage.py transfer --purchased_plaques
|
||||
|
|
@ -80,11 +80,11 @@ LOGGING = {
|
|||
'py.warnings': {
|
||||
'handlers': ['console'],
|
||||
},
|
||||
'django.db.backends': {
|
||||
'handlers': ['console', ],
|
||||
'level': 'DEBUG',
|
||||
'propagate': False,
|
||||
},
|
||||
# 'django.db.backends': {
|
||||
# 'handlers': ['console', ],
|
||||
# 'level': 'DEBUG',
|
||||
# 'propagate': False,
|
||||
# },
|
||||
}
|
||||
}
|
||||
|
||||
|
|
@ -106,3 +106,4 @@ ELASTICSEARCH_INDEX_NAMES = {
|
|||
TESTING = sys.argv[1:2] == ['test']
|
||||
if TESTING:
|
||||
ELASTICSEARCH_INDEX_NAMES = {}
|
||||
ELASTICSEARCH_DSL_AUTOSYNC = False
|
||||
Loading…
Reference in New Issue
Block a user