From 0958c2ad83f536b9d1282482e28ea256f082e0be Mon Sep 17 00:00:00 2001 From: Krishna-Baldwa Date: Sun, 25 Jun 2023 18:33:44 +0530 Subject: [PATCH 1/7] client id, secret changed in master --- backend/settings.py | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/backend/settings.py b/backend/settings.py index 5c7089d1..112e267c 100644 --- a/backend/settings.py +++ b/backend/settings.py @@ -17,8 +17,8 @@ # SSO Config SSO_TOKEN_URL = 'https://gymkhana.iitb.ac.in/sso/oauth/token/' SSO_PROFILE_URL = 'https://gymkhana.iitb.ac.in/sso/user/api/user/?fields=first_name,last_name,type,profile_picture,sex,username,email,program,contacts,insti_address,secondary_emails,mobile,roll_number' -SSO_CLIENT_ID = '5jyMJufq0Vk0aDlj9Hnudsj84UfbFZlYRUnn02Xd' -SSO_CLIENT_ID_SECRET_BASE64 = 'NWp5TUp1ZnEwVmswYURsajlIbnVkc2o4NFVmYkZabFlSVW5uMDJYZDo2RTN5S3g4OU5XWVFxcWFJaUdwcE9RdFF4TmZxSkxFYXhkcDVXb2s2VzZsdGxDQWpPWmM3SWJrVXVpUXdnaTZNaGVoUU1LWnVTelFUMU1EOHBockhhVThvVUk5S1hYZ2NHMG51UE03Wkk3V0xoWjYydG5OZFU4Uk02TTJXaXdJZw==' +SSO_CLIENT_ID = 'HeKlfCluQLxa5cG5c4yHYiAEFZynroiKwylpiwNV' +SSO_CLIENT_ID_SECRET_BASE64 = 'SGVLbGZDbHVRTHhhNWNHNWM0eUhZaUFFRlp5bnJvaUt3eWxwaXdOVjpYbDg4OHNaOWFhbVVmV1FMR0Y3SjI1MU5taUFYeHRtdzRtM3pHejJLUVQ1d3M0b3hTRGNCTnhSNWw4SXpYbHNDTVVWVHh3MUE0VXRnRU5YZ1FpWlFDZ1RUcERoVHFXeUZmckhLdDhadU5SQk9SY2Q3Z2ZmWjNXUzQ5bGxCMXNBUg==' # Password Login SSO_DEFAULT_REDIR = 'https://insti.app/login' From a0fc3e33af1bee453bf5f4b08d059c9e13109ba4 Mon Sep 17 00:00:00 2001 From: Krishna-Baldwa Date: Mon, 26 Jun 2023 20:34:03 +0530 Subject: [PATCH 2/7] Particular community posts comes in that particular community and not in others --- backend/settings.py | 2 ++ community/views.py | 8 ++++---- 2 files changed, 6 insertions(+), 4 deletions(-) diff --git a/backend/settings.py b/backend/settings.py index 112e267c..b9f47a4b 100644 --- a/backend/settings.py +++ b/backend/settings.py @@ -13,6 +13,8 @@ DEBUG = True ALLOWED_HOSTS = ['*'] +CORS_ORIGIN_WHITELIST = ['https://www.insti.app','https://api.insti.app', 'https://gymkhana.iitb.ac.in', 'http://10.105.177.175', 'http://localhost:4200', 'http://10.198.49.175'] +CORS_ALLOW_CREDENTIALS = True # SSO Config SSO_TOKEN_URL = 'https://gymkhana.iitb.ac.in/sso/oauth/token/' diff --git a/community/views.py b/community/views.py index c8021d9f..7b6dddea 100644 --- a/community/views.py +++ b/community/views.py @@ -73,20 +73,20 @@ def list(self, request): # Check for time and date filtered query params status = request.GET.get("status") - + community = request.GET.get("community") # If your posts if status is None: - queryset = CommunityPost.objects.filter(thread_rank=1, posted_by=request.user + queryset = CommunityPost.objects.filter(thread_rank=1, community=community, posted_by=request.user .profile).order_by("-time_of_modification") else: # If reported posts if status == "3": - queryset = CommunityPost.objects.filter(status=status, deleted=False).order_by("-time_of_modification") + queryset = CommunityPost.objects.filter(status=status, community=community, deleted=False).order_by("-time_of_modification") # queryset = CommunityPost.objects.all() else: queryset = CommunityPost.objects.filter( - status=status, deleted=False, thread_rank=1).order_by("-time_of_modification") + status=status, community=community, deleted=False, thread_rank=1).order_by("-time_of_modification") queryset = query_search(request, 3, queryset, ['content'], 'posts') queryset = query_from_num(request, 20, queryset) From 82d761c912958ea824c6a4de291451d2f8dfc743 Mon Sep 17 00:00:00 2001 From: srbh001 <22B3905@iitb.ac.in> Date: Wed, 12 Jul 2023 19:54:58 +0530 Subject: [PATCH 3/7] Add Community Roles for moderators, Fix some Bugs --- community/models.py | 8 +++++ community/urls.py | 1 + community/views.py | 23 ++++++++---- other/tasks.py | 5 +-- roles/admin.py | 7 +++- roles/helpers.py | 13 +++---- roles/migrations/0020_communityrole.py | 36 +++++++++++++++++++ roles/models.py | 21 +++++++++++ .../0041_userprofile_community_roles.py | 19 ++++++++++ .../0042_userprofile_followed_communities.py | 19 ++++++++++ users/models.py | 3 +- 11 files changed, 139 insertions(+), 16 deletions(-) create mode 100644 roles/migrations/0020_communityrole.py create mode 100644 users/migrations/0041_userprofile_community_roles.py create mode 100644 users/migrations/0042_userprofile_followed_communities.py diff --git a/community/models.py b/community/models.py index 51ac991c..362932ee 100644 --- a/community/models.py +++ b/community/models.py @@ -2,6 +2,7 @@ from uuid import uuid4 from django.db import models from helpers.misc import get_url_friendly +from users.models import UserProfile class Community(models.Model): id = models.UUIDField(primary_key=True, default=uuid4, editable=False) @@ -22,6 +23,13 @@ def __str__(self): def save(self, *args, **kwargs): # pylint: disable=W0222 self.str_id = get_url_friendly(self.name) + "-" + str(self.id)[:8] super().save(*args, **kwargs) + + @staticmethod + def pre_save(self): + followers = UserProfile.objects.filter(followed_communities__id=self.id) + if followers: + self.followers = followers.count() + class Meta: verbose_name = "Community" diff --git a/community/urls.py b/community/urls.py index c3407a9a..2e660f02 100644 --- a/community/urls.py +++ b/community/urls.py @@ -5,6 +5,7 @@ urlpatterns = [ path('communities', CommunityViewSet.as_view({ 'get': 'list', + 'post': 'create' })), # viewing the list of communities path('communities/', CommunityViewSet.as_view({ diff --git a/community/views.py b/community/views.py index 7b6dddea..21741ea2 100644 --- a/community/views.py +++ b/community/views.py @@ -6,12 +6,12 @@ from community.models import CommunityPost from community.serializer_min import CommunitySerializerMin, CommunityPostSerializerMin from community.serializers import CommunitySerializers, CommunityPostSerializers -from roles.helpers import user_has_privilege +from roles.helpers import user_has_privilege, user_has_community_privilege from roles.helpers import login_required_ajax from roles.helpers import forbidden_no_privileges from helpers.misc import query_from_num from helpers.misc import query_search - +from users.models import UserProfile class ModeratorViewSet(viewsets.ModelViewSet): queryset = CommunityPost.objects serializer_class = CommunityPostSerializers @@ -29,7 +29,7 @@ def change_status(self, request, pk): post = self.get_community_post(pk) if ((user_has_privilege(request.user.profile, post.community.body.id, 'AppP') and post.thread_rank == 1) - or (user_has_privilege(request.user.profile, post.community.body.id, 'ModC') and post.thread_rank > 1)): + or (user_has_privilege(request.user.profile, post.community.body.id, 'ModC') and post.thread_rank > 1)) or user_has_community_privilege(request.user.profile, post.community.id, 'ModC'): # Get query param status = request.data["status"] if status is None: @@ -102,7 +102,8 @@ def create(self, request): # Prevent posts without any community if 'community' not in request.data or not request.data['community']: return forbidden_no_privileges() - + + user, created = UserProfile.objects.get_or_create(user=request.user) return super().create(request) @login_required_ajax @@ -122,7 +123,7 @@ def perform_action(self, request, action, pk): post = self.get_community_post(pk) if action == "feature": - if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]): + if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]) or user_has_community_privilege(request.user.profile, post.community.id, 'ModC'): # Get query param is_featured = request.data["is_featured"] @@ -144,7 +145,7 @@ def perform_action(self, request, action, pk): post.save() return Response({"message": "Post deleted"}) - if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]): + if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]) or user_has_community_privilege(request.user.profile, post.community.id, 'ModC'): post.status = 2 post.featured = False post.save() @@ -203,3 +204,13 @@ def get_community(self, pk): return get_object_or_404(self.queryset, id=pk) except ValueError: return get_object_or_404(self.queryset, str_id=pk) + + @login_required_ajax + def create(self, request): + name = request.data['name'] + user, created = UserProfile.objects.get_or_create(user=request.user) + if not Community.objects.all().filter(name=name).exists(): + super().create(request) + return Response({"message": "Community created"}) + + return Response({"message": "Community already exists"}, status=400) diff --git a/other/tasks.py b/other/tasks.py index b5f58234..f59aaa34 100644 --- a/other/tasks.py +++ b/other/tasks.py @@ -78,8 +78,9 @@ def notify_new_commpost(pk): for interest in instance.interests.all(): users = User.objects.filter( - id__in=UserInterest.filter(title=interest.title).user.filter(active=True).values('user_id') - ) + + id__in=UserInterest.objects.filter(title=interest.title)).filter(is_active=True).values('id') + notify.send( instance, recipient=users, diff --git a/roles/admin.py b/roles/admin.py index 6feb8101..39dfb437 100644 --- a/roles/admin.py +++ b/roles/admin.py @@ -1,16 +1,21 @@ from django.contrib import admin from roles.models import BodyRole -from roles.models import InstituteRole +from roles.models import InstituteRole, CommunityRole class BodyRoleAdmin(admin.ModelAdmin): list_filter = ['body'] list_display = ('name', 'body', 'permissions') search_fields = ('body__name', 'name') +class CommunityRoleAdmin(admin.ModelAdmin): + list_filter = ['community'] + list_display = ('name', 'community', 'permissions') + search_fields = ('community__name', 'name') class InstittuteRoleAdmin(admin.ModelAdmin): list_display = ('name', 'permissions') search_fields = ['name'] admin.site.register(BodyRole, BodyRoleAdmin) +admin.site.register(CommunityRole, CommunityRoleAdmin) admin.site.register(InstituteRole, InstittuteRoleAdmin) diff --git a/roles/helpers.py b/roles/helpers.py index 725faa27..a5e58afa 100644 --- a/roles/helpers.py +++ b/roles/helpers.py @@ -2,6 +2,7 @@ from rest_framework.response import Response from bodies.models import Body from community.models import Community +from roles.models import CommunityRole def forbidden_no_privileges(): """Forbidden due to insufficient privileges.""" @@ -19,13 +20,13 @@ def user_has_privilege(profile, bodyid, privilege): return True return False def user_has_community_privilege(profile, communityid, privilege): - """Returns true if UserProfile has or has inherited the privilege.""" - community = Community.objects.get(pk=communityid) - parents = get_parents_recursive(community, []) - for role in profile.roles.all().filter(community__in=parents): - if (role.body == community or role.inheritable) and privilege in role.permissions: + """Returns true if UserProfile has particular previlege in a community.""" + coms = Community.objects.filter(pk=communityid) + for role in profile.community_roles.all().filter(community__in=coms): + if privilege in role.permissions: return True - return False + + def user_has_insti_privilege(profile, privilege): """Returns true if UserProfile has the institute privilege.""" diff --git a/roles/migrations/0020_communityrole.py b/roles/migrations/0020_communityrole.py new file mode 100644 index 00000000..be8a289a --- /dev/null +++ b/roles/migrations/0020_communityrole.py @@ -0,0 +1,36 @@ +# Generated by Django 3.2.16 on 2023-07-07 14:40 + +from django.db import migrations, models +import django.db.models.deletion +import multiselectfield.db.fields +import uuid + + +class Migration(migrations.Migration): + + dependencies = [ + ('community', '0028_auto_20221003_2130'), + ('roles', '0019_alter_bodyrole_permissions'), + ] + + operations = [ + migrations.CreateModel( + name='CommunityRole', + fields=[ + ('id', models.UUIDField(default=uuid.uuid4, editable=False, primary_key=True, serialize=False)), + ('time_of_creation', models.DateTimeField(auto_now_add=True)), + ('name', models.CharField(max_length=50)), + ('inheritable', models.BooleanField(default=False)), + ('permissions', multiselectfield.db.fields.MultiSelectField(choices=[('AddE', 'Add Event'), ('UpdE', 'Update Event'), ('DelE', 'Delete Event'), ('UpdB', 'Update Body'), ('Role', 'Modify Roles'), ('VerA', 'Verify Achievements'), ('AppP', 'Moderate Post'), ('ModC', 'Moderate Comment')], max_length=39)), + ('priority', models.IntegerField(default=0)), + ('official_post', models.BooleanField(default=True)), + ('permanent', models.BooleanField(default=False)), + ('community', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='roles', to='community.community')), + ], + options={ + 'verbose_name': 'Community Role', + 'verbose_name_plural': 'Community Roles', + 'ordering': ('community__name', 'priority'), + }, + ), + ] diff --git a/roles/models.py b/roles/models.py index a1267b24..677a7d1c 100644 --- a/roles/models.py +++ b/roles/models.py @@ -34,6 +34,27 @@ class Meta: def __str__(self): return self.body.name + " " + self.name + +class CommunityRole(models.Model): + """A role for a bodywhich can be granted to multiple users.""" + + id = models.UUIDField(primary_key=True, default=uuid4, editable=False) + time_of_creation = models.DateTimeField(auto_now_add=True) + name = models.CharField(max_length=50) + community = models.ForeignKey('community.Community', on_delete=models.CASCADE, related_name='roles') + inheritable = models.BooleanField(default=False) + permissions = MultiSelectField(choices=PERMISSION_CHOICES) + priority = models.IntegerField(default=0) + official_post = models.BooleanField(default=True) + permanent = models.BooleanField(default=False) + + class Meta: + verbose_name = "Community Role" + verbose_name_plural = "Community Roles" + ordering = ("community__name", "priority") + + def __str__(self): + return self.community.name + " " + self.name INSTITUTE_PERMISSION_CHOICES = ( diff --git a/users/migrations/0041_userprofile_community_roles.py b/users/migrations/0041_userprofile_community_roles.py new file mode 100644 index 00000000..b3e813c2 --- /dev/null +++ b/users/migrations/0041_userprofile_community_roles.py @@ -0,0 +1,19 @@ +# Generated by Django 3.2.16 on 2023-07-07 14:40 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('roles', '0020_communityrole'), + ('users', '0040_remove_userprofile_followed_communities'), + ] + + operations = [ + migrations.AddField( + model_name='userprofile', + name='community_roles', + field=models.ManyToManyField(blank=True, related_name='users', to='roles.CommunityRole'), + ), + ] diff --git a/users/migrations/0042_userprofile_followed_communities.py b/users/migrations/0042_userprofile_followed_communities.py new file mode 100644 index 00000000..279ccd87 --- /dev/null +++ b/users/migrations/0042_userprofile_followed_communities.py @@ -0,0 +1,19 @@ +# Generated by Django 3.2.16 on 2023-07-07 23:27 + +from django.db import migrations, models + + +class Migration(migrations.Migration): + + dependencies = [ + ('community', '0028_auto_20221003_2130'), + ('users', '0041_userprofile_community_roles'), + ] + + operations = [ + migrations.AddField( + model_name='userprofile', + name='followed_communities', + field=models.ManyToManyField(blank=True, related_name='community_followers', to='community.Community'), + ), + ] diff --git a/users/models.py b/users/models.py index 4e4b3ad5..54c751e6 100644 --- a/users/models.py +++ b/users/models.py @@ -48,13 +48,14 @@ class UserProfile(models.Model): # InstiApp feature fields active = models.BooleanField(default=True) followed_bodies = models.ManyToManyField('bodies.Body', related_name='followers', blank=True) + followed_communities = models.ManyToManyField('community.Community', related_name='community_followers', blank=True) # InstiApp roles roles = models.ManyToManyField('roles.BodyRole', related_name='users', blank=True) former_roles = models.ManyToManyField( 'roles.BodyRole', related_name='former_users', blank=True, through='UserFormerRole') institute_roles = models.ManyToManyField( 'roles.InstituteRole', related_name='users', blank=True) - + community_roles = models.ManyToManyField('roles.CommunityRole', related_name='users', blank=True) # User exposed fields show_contact_no = models.BooleanField(default=False) fcm_id = models.CharField(max_length=200, null=True, blank=True) From 3373508c515d047a8cc574180ad900b9a64fe2cb Mon Sep 17 00:00:00 2001 From: srbh001 <22B3905@iitb.ac.in> Date: Sat, 22 Jul 2023 19:15:47 +0530 Subject: [PATCH 4/7] (revert) : CommunityRole and assosciated changes --- community/views.py | 8 ++-- roles/admin.py | 8 +--- roles/helpers.py | 11 +++--- roles/migrations/0021_delete_communityrole.py | 17 +++++++++ roles/models.py | 38 ++++++++++--------- ...0043_remove_userprofile_community_roles.py | 17 +++++++++ users/models.py | 2 +- 7 files changed, 68 insertions(+), 33 deletions(-) create mode 100644 roles/migrations/0021_delete_communityrole.py create mode 100644 users/migrations/0043_remove_userprofile_community_roles.py diff --git a/community/views.py b/community/views.py index 21741ea2..07f3ff93 100644 --- a/community/views.py +++ b/community/views.py @@ -6,7 +6,7 @@ from community.models import CommunityPost from community.serializer_min import CommunitySerializerMin, CommunityPostSerializerMin from community.serializers import CommunitySerializers, CommunityPostSerializers -from roles.helpers import user_has_privilege, user_has_community_privilege +from roles.helpers import user_has_privilege from roles.helpers import login_required_ajax from roles.helpers import forbidden_no_privileges from helpers.misc import query_from_num @@ -29,7 +29,7 @@ def change_status(self, request, pk): post = self.get_community_post(pk) if ((user_has_privilege(request.user.profile, post.community.body.id, 'AppP') and post.thread_rank == 1) - or (user_has_privilege(request.user.profile, post.community.body.id, 'ModC') and post.thread_rank > 1)) or user_has_community_privilege(request.user.profile, post.community.id, 'ModC'): + or (user_has_privilege(request.user.profile, post.community.body.id, 'ModC') and post.thread_rank > 1)): # Get query param status = request.data["status"] if status is None: @@ -123,7 +123,7 @@ def perform_action(self, request, action, pk): post = self.get_community_post(pk) if action == "feature": - if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]) or user_has_community_privilege(request.user.profile, post.community.id, 'ModC'): + if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]): # Get query param is_featured = request.data["is_featured"] @@ -145,7 +145,7 @@ def perform_action(self, request, action, pk): post.save() return Response({"message": "Post deleted"}) - if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]) or user_has_community_privilege(request.user.profile, post.community.id, 'ModC'): + if all([user_has_privilege(request.user.profile, post.community.body.id, 'AppP')]): post.status = 2 post.featured = False post.save() diff --git a/roles/admin.py b/roles/admin.py index 39dfb437..1226a158 100644 --- a/roles/admin.py +++ b/roles/admin.py @@ -1,21 +1,17 @@ from django.contrib import admin from roles.models import BodyRole -from roles.models import InstituteRole, CommunityRole +from roles.models import InstituteRole class BodyRoleAdmin(admin.ModelAdmin): list_filter = ['body'] list_display = ('name', 'body', 'permissions') search_fields = ('body__name', 'name') -class CommunityRoleAdmin(admin.ModelAdmin): - list_filter = ['community'] - list_display = ('name', 'community', 'permissions') - search_fields = ('community__name', 'name') + class InstittuteRoleAdmin(admin.ModelAdmin): list_display = ('name', 'permissions') search_fields = ['name'] admin.site.register(BodyRole, BodyRoleAdmin) -admin.site.register(CommunityRole, CommunityRoleAdmin) admin.site.register(InstituteRole, InstittuteRoleAdmin) diff --git a/roles/helpers.py b/roles/helpers.py index a5e58afa..7e2ffbf4 100644 --- a/roles/helpers.py +++ b/roles/helpers.py @@ -2,7 +2,6 @@ from rest_framework.response import Response from bodies.models import Body from community.models import Community -from roles.models import CommunityRole def forbidden_no_privileges(): """Forbidden due to insufficient privileges.""" @@ -20,11 +19,13 @@ def user_has_privilege(profile, bodyid, privilege): return True return False def user_has_community_privilege(profile, communityid, privilege): - """Returns true if UserProfile has particular previlege in a community.""" - coms = Community.objects.filter(pk=communityid) - for role in profile.community_roles.all().filter(community__in=coms): - if privilege in role.permissions: + """Returns true if UserProfile has or has inherited the privilege.""" + community = Community.objects.get(pk=communityid) + parents = get_parents_recursive(community, []) + for role in profile.roles.all().filter(community__in=parents): + if (role.body == community or role.inheritable) and privilege in role.permissions: return True + return False diff --git a/roles/migrations/0021_delete_communityrole.py b/roles/migrations/0021_delete_communityrole.py new file mode 100644 index 00000000..8905be25 --- /dev/null +++ b/roles/migrations/0021_delete_communityrole.py @@ -0,0 +1,17 @@ +# Generated by Django 3.2.16 on 2023-07-22 13:35 + +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('users', '0043_remove_userprofile_community_roles'), + ('roles', '0020_communityrole'), + ] + + operations = [ + migrations.DeleteModel( + name='CommunityRole', + ), + ] diff --git a/roles/models.py b/roles/models.py index 677a7d1c..fb0bff80 100644 --- a/roles/models.py +++ b/roles/models.py @@ -35,26 +35,30 @@ class Meta: def __str__(self): return self.body.name + " " + self.name -class CommunityRole(models.Model): - """A role for a bodywhich can be granted to multiple users.""" +''' +Added Community Role Model : To allow to have various communities in a single body and have different roles for each community. +Ditched For Now +''' +# class CommunityRole(models.Model): +# """A role for a bodywhich can be granted to multiple users.""" - id = models.UUIDField(primary_key=True, default=uuid4, editable=False) - time_of_creation = models.DateTimeField(auto_now_add=True) - name = models.CharField(max_length=50) - community = models.ForeignKey('community.Community', on_delete=models.CASCADE, related_name='roles') - inheritable = models.BooleanField(default=False) - permissions = MultiSelectField(choices=PERMISSION_CHOICES) - priority = models.IntegerField(default=0) - official_post = models.BooleanField(default=True) - permanent = models.BooleanField(default=False) +# id = models.UUIDField(primary_key=True, default=uuid4, editable=False) +# time_of_creation = models.DateTimeField(auto_now_add=True) +# name = models.CharField(max_length=50) +# community = models.ForeignKey('community.Community', on_delete=models.CASCADE, related_name='roles') +# inheritable = models.BooleanField(default=False) +# permissions = MultiSelectField(choices=PERMISSION_CHOICES) +# priority = models.IntegerField(default=0) +# official_post = models.BooleanField(default=True) +# permanent = models.BooleanField(default=False) - class Meta: - verbose_name = "Community Role" - verbose_name_plural = "Community Roles" - ordering = ("community__name", "priority") +# class Meta: +# verbose_name = "Community Role" +# verbose_name_plural = "Community Roles" +# ordering = ("community__name", "priority") - def __str__(self): - return self.community.name + " " + self.name +# def __str__(self): +# return self.community.name + " " + self.name INSTITUTE_PERMISSION_CHOICES = ( diff --git a/users/migrations/0043_remove_userprofile_community_roles.py b/users/migrations/0043_remove_userprofile_community_roles.py new file mode 100644 index 00000000..a588305a --- /dev/null +++ b/users/migrations/0043_remove_userprofile_community_roles.py @@ -0,0 +1,17 @@ +# Generated by Django 3.2.16 on 2023-07-22 13:35 + +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('users', '0042_userprofile_followed_communities'), + ] + + operations = [ + migrations.RemoveField( + model_name='userprofile', + name='community_roles', + ), + ] diff --git a/users/models.py b/users/models.py index 54c751e6..49a234d8 100644 --- a/users/models.py +++ b/users/models.py @@ -55,7 +55,7 @@ class UserProfile(models.Model): 'roles.BodyRole', related_name='former_users', blank=True, through='UserFormerRole') institute_roles = models.ManyToManyField( 'roles.InstituteRole', related_name='users', blank=True) - community_roles = models.ManyToManyField('roles.CommunityRole', related_name='users', blank=True) + # community_roles = models.ManyToManyField('roles.CommunityRole', related_name='users', blank=True) # User exposed fields show_contact_no = models.BooleanField(default=False) fcm_id = models.CharField(max_length=200, null=True, blank=True) From e0f29427a77dbb8243c65c5395d488e96b3e89a9 Mon Sep 17 00:00:00 2001 From: Palash Mittal Date: Mon, 24 Jul 2023 23:30:54 +0530 Subject: [PATCH 5/7] fix(community): pass community id as query param in post list --- community/views.py | 13 +++++++++---- 1 file changed, 9 insertions(+), 4 deletions(-) diff --git a/community/views.py b/community/views.py index 07f3ff93..fadc5741 100644 --- a/community/views.py +++ b/community/views.py @@ -73,7 +73,11 @@ def list(self, request): # Check for time and date filtered query params status = request.GET.get("status") - community = request.GET.get("community") + comm_id = request.GET.get("community") + if comm_id is None: + return Response({"message": "comm_id is required"}, status=400) + community = get_object_or_404(Community.objects, id=comm_id) + # If your posts if status is None: queryset = CommunityPost.objects.filter(thread_rank=1, community=community, posted_by=request.user @@ -81,7 +85,8 @@ def list(self, request): else: # If reported posts if status == "3": - queryset = CommunityPost.objects.filter(status=status, community=community, deleted=False).order_by("-time_of_modification") + queryset = CommunityPost.objects.filter( + status=status, community=community, deleted=False).order_by("-time_of_modification") # queryset = CommunityPost.objects.all() else: @@ -102,7 +107,7 @@ def create(self, request): # Prevent posts without any community if 'community' not in request.data or not request.data['community']: return forbidden_no_privileges() - + user, created = UserProfile.objects.get_or_create(user=request.user) return super().create(request) @@ -204,7 +209,7 @@ def get_community(self, pk): return get_object_or_404(self.queryset, id=pk) except ValueError: return get_object_or_404(self.queryset, str_id=pk) - + @login_required_ajax def create(self, request): name = request.data['name'] From 92df7e38a3d691257b73900c1b5e15d2417fc15c Mon Sep 17 00:00:00 2001 From: Palash Mittal Date: Mon, 24 Jul 2023 23:40:36 +0530 Subject: [PATCH 6/7] fix(users): remove field followed_communities --- community/models.py | 7 ------- ...4_remove_userprofile_followed_communities.py | 17 +++++++++++++++++ users/models.py | 2 +- 3 files changed, 18 insertions(+), 8 deletions(-) create mode 100644 users/migrations/0044_remove_userprofile_followed_communities.py diff --git a/community/models.py b/community/models.py index 362932ee..67d31e5b 100644 --- a/community/models.py +++ b/community/models.py @@ -23,13 +23,6 @@ def __str__(self): def save(self, *args, **kwargs): # pylint: disable=W0222 self.str_id = get_url_friendly(self.name) + "-" + str(self.id)[:8] super().save(*args, **kwargs) - - @staticmethod - def pre_save(self): - followers = UserProfile.objects.filter(followed_communities__id=self.id) - if followers: - self.followers = followers.count() - class Meta: verbose_name = "Community" diff --git a/users/migrations/0044_remove_userprofile_followed_communities.py b/users/migrations/0044_remove_userprofile_followed_communities.py new file mode 100644 index 00000000..899f2f26 --- /dev/null +++ b/users/migrations/0044_remove_userprofile_followed_communities.py @@ -0,0 +1,17 @@ +# Generated by Django 3.2.16 on 2023-07-24 18:08 + +from django.db import migrations + + +class Migration(migrations.Migration): + + dependencies = [ + ('users', '0043_remove_userprofile_community_roles'), + ] + + operations = [ + migrations.RemoveField( + model_name='userprofile', + name='followed_communities', + ), + ] diff --git a/users/models.py b/users/models.py index 49a234d8..61a38c42 100644 --- a/users/models.py +++ b/users/models.py @@ -48,7 +48,7 @@ class UserProfile(models.Model): # InstiApp feature fields active = models.BooleanField(default=True) followed_bodies = models.ManyToManyField('bodies.Body', related_name='followers', blank=True) - followed_communities = models.ManyToManyField('community.Community', related_name='community_followers', blank=True) + # InstiApp roles roles = models.ManyToManyField('roles.BodyRole', related_name='users', blank=True) former_roles = models.ManyToManyField( From 4c9298aca9a6bcf1281f585e08246163dbfb1bb2 Mon Sep 17 00:00:00 2001 From: SanskarGosavi2003 <210100132@iitb.ac.in> Date: Thu, 30 Nov 2023 16:56:33 +0530 Subject: [PATCH 7/7] FIX: Anonymous name fetch from backend FIX: Anonymous name for moderators and users to be shown differently --- community/serializer_min.py | 15 ++++++++++++++- community/serializers.py | 13 +++++++++++++ community/views.py | 10 ++++++++-- 3 files changed, 35 insertions(+), 3 deletions(-) diff --git a/community/serializer_min.py b/community/serializer_min.py index 024ae70e..1576b123 100644 --- a/community/serializer_min.py +++ b/community/serializer_min.py @@ -3,6 +3,7 @@ from achievements.serializers import InterestSerializer from bodies.serializer_min import BodySerializerMin from community.models import Community, CommunityPost +from users.models import UserProfile from users.serializers import UserProfileSerializer class CommunitySerializerMin(serializers.ModelSerializer): @@ -36,7 +37,8 @@ class CommunityPostSerializerMin(serializers.ModelSerializer): reactions_count = serializers.SerializerMethodField() user_reaction = serializers.SerializerMethodField() comments_count = serializers.SerializerMethodField() - posted_by = UserProfileSerializer(read_only=True) + # posted_by = UserProfileSerializer(read_only=True) + posted_by = serializers.SerializerMethodField() image_url = serializers.SerializerMethodField() most_liked_comment = serializers.SerializerMethodField() community = CommunitySerializerMin(read_only=True) @@ -45,6 +47,17 @@ class CommunityPostSerializerMin(serializers.ModelSerializer): interests = InterestSerializer(read_only=True, many=True) has_user_reported = serializers.SerializerMethodField() + def get_posted_by(self, obj): + pb = UserProfile.objects.get(id=obj.posted_by.id) + if(obj.anonymous and "return_for_mod" in self.context and not self.context["return_for_mod"]): + pb.name = "Anonymous" + pb.id = "null" + pb.ldap_id = "null" + pb.profile_pic = "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcSM9q9XJKxlskry5gXTz1OXUyem5Ap59lcEGg&usqp=CAU" + elif(obj.anonymous and "return_for_mod" in self.context and self.context["return_for_mod"]): + pb.name += " (Anon)" + return UserProfileSerializer(pb).data + def get_most_liked_comment(self, obj): """Get the most liked comment of the community post """ queryset = obj.comments.filter(deleted=False, status=1) diff --git a/community/serializers.py b/community/serializers.py index 87b30bba..cc4529c8 100644 --- a/community/serializers.py +++ b/community/serializers.py @@ -5,6 +5,7 @@ from roles.serializers import RoleSerializerMin from users.models import UserProfile from bodies.models import Body +from users.serializers import UserProfileSerializer class CommunitySerializers(serializers.ModelSerializer): @@ -57,10 +58,22 @@ def setup_eager_loading(queryset, request): class CommunityPostSerializers(CommunityPostSerializerMin): comments = serializers.SerializerMethodField() + posted_by = serializers.SerializerMethodField() def get_comments(self, obj): comments = obj.comments.filter(deleted=False, status=1) return CommunityPostSerializerMin(comments, many=True).data + + def get_posted_by(self, obj): + pb = UserProfile.objects.get(id=obj.posted_by.id) + if(obj.anonymous and "return_for_mod" in self.context and not self.context["return_for_mod"]): + pb.name = "Anonymous" + pb.id = "null" + pb.ldap_id = "null" + pb.profile_pic = "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcSM9q9XJKxlskry5gXTz1OXUyem5Ap59lcEGg&usqp=CAU" + elif(obj.anonymous and "return_for_mod" in self.context and self.context["return_for_mod"]): + pb.name += " (Anon)" + return UserProfileSerializer(pb).data class Meta: model = CommunityPost diff --git a/community/views.py b/community/views.py index fadc5741..19eb3a82 100644 --- a/community/views.py +++ b/community/views.py @@ -62,7 +62,10 @@ def retrieve_full(self, request, pk): self.queryset = CommunityPostSerializers.setup_eager_loading(self.queryset, request) post = self.get_community_post(pk) - serialized = CommunityPostSerializers(post, context={'request': request}).data + return_for_mod = False + if(user_has_privilege(request.user.profile, post.community.body.id, "AppP")): + return_for_mod = True + serialized = CommunityPostSerializers(post, context={'return_for_mod': return_for_mod}).data return Response(serialized) @@ -94,8 +97,11 @@ def list(self, request): status=status, community=community, deleted=False, thread_rank=1).order_by("-time_of_modification") queryset = query_search(request, 3, queryset, ['content'], 'posts') queryset = query_from_num(request, 20, queryset) + return_for_mod = False + if(user_has_privilege(request.user.profile, community.body.id, "AppP")): + return_for_mod = True - serializer = CommunityPostSerializerMin(queryset, many=True, context={'request': request}) + serializer = CommunityPostSerializerMin(queryset, many=True, context={'return_for_mod': return_for_mod}) data = serializer.data return Response({'count': len(data), 'data': data})