Not a member of Pastebin yet?
Sign Up,
it unlocks many cool features!
- from __future__ import unicode_literals
- from django.db import models
- from django.contrib.auth.models import BaseUserManager
- from django.db.models import Q, QuerySet, Max, F, Count, Case, When, Exists, OuterRef
- from django.utils import timezone
- from django.forms.models import model_to_dict
- from django.utils.dateformat import format
- from django.core.validators import RegexValidator, URLValidator
- from django.core.exceptions import ValidationError
- from datetime import timedelta, datetime, date, time
- from passlib.hash import bcrypt_sha256
- from closedverse import settings
- from . import util
- from random import getrandbits
- import uuid, json, base64
- from django.core.mail import send_mail
- from django.template.loader import render_to_string
- from django.urls import reverse
- from django.db.models.signals import pre_delete
- from django.dispatch import receiver
- import re
- feelings = ((0, 'normal'), (1, 'happy'), (2, 'wink'), (3, 'surprised'), (4, 'frustrated'), (5, 'confused'), (38, 'japan'), (69, 'easter egg'), )
- post_status = ((0, 'ok'), (1, 'delete by user'), (2, 'delete by authority'), (3, 'delete by mod'), (4, 'delete by admin'), (5, 'account pruge'))
- visibility = ((0, 'show'), (1, 'friends only'), (2, 'hide'), )
- # Like set() but orders
- def organ(seq):
- seen = set()
- seen_add = seen.add
- return [x for x in seq if not (x in seen or seen_add(x))]
- class UserManager(BaseUserManager):
- # idk why this is even here
- def create_user(self, username, password):
- user = self.model(
- username=username,
- )
- user.set_password(password)
- user.save(using=self._db)
- return user
- def closed_create_user(self, username, password, email, addr, nick, nn, gravatar):
- user = self.model(
- username = username,
- nickname = util.filterchars(nick),
- addr = addr,
- email = email,
- )
- if settings.PROD:
- if util.iphub(addr):
- spamuser = True
- if settings.disallow_proxy:
- # This was for me, a server error will email admins of course.
- raise ValueError
- else:
- spamuser = False
- else:
- spamuser = False
- profile = Profile.objects.model()
- if nn:
- user.avatar = nn[0]
- profile.origin_id = nn[2]
- profile.origin_info = json.dumps(nn)
- user.has_mh = True
- else:
- user.avatar = util.get_gravatar(email) or ('s' if getrandbits(1) else '')
- user.has_mh = False
- user.set_password(password)
- user.save(using=self._db)
- if spamuser:
- profile.let_freedom = False
- else:
- profile.let_freedom = True
- profile.user = user
- profile.save()
- return user
- def create_superuser(self, username, password):
- user = self.model(
- username=username,
- )
- user.set_password(password)
- user.staff = True
- user.save()
- profile = Profile.objects.model()
- profile.user = user
- profile.save()
- return user
- def authenticate(self, username, password):
- if not username or username.isspace():
- return None
- user = self.filter(Q(username__iexact=username.replace(' ', '')) | Q(username__iexact=username) | Q(email=username))
- if not user.exists():
- return None
- user = user.first()
- # If the user is an admin, say that they don't exist
- # Or, if the user doesn't want username login, don't let them if they didn't enter their email
- if user.profile('email_login') == 2 and not user.email == username:
- return None
- elif user.profile('email_login') == 0 and user.email == username:
- return None
- try:
- passwd = user.check_password(password)
- # Check if the password is a valid bcrypt
- except ValueError:
- return (user, 2)
- else:
- if not passwd:
- if user.can_manage():
- return None
- return (user, False)
- return (user, True)
- class PostManager(models.Manager):
- def get_queryset(self):
- return super(PostManager, self).get_queryset().filter(is_rm=False)
- class CommunityFavoriteManager(models.Manager):
- def get_queryset(self):
- return super(CommunityFavoriteManager, self).get_queryset().filter(community__is_rm=False).exclude(community__type=3)
- # Taken from https://github.com/jaredly/django-colorfield/blob/master/colorfield/fields.py
- color_re = re.compile('^#([A-Fa-f0-9]{6})$')
- validate_color = RegexValidator(color_re, "Enter a valid color", 'invalid')
- class ColorField(models.CharField):
- default_validators = [validate_color]
- def __init__(self, *args, **kwargs):
- kwargs['max_length'] = 18
- super(ColorField, self).__init__(*args, **kwargs)
- class User(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- username = models.CharField(max_length=32, unique=True)
- # Todo: Don't allow nickname to be null (once this is fixed, un-str-ify line 357 of views; the one with ogdata description)
- # Also don't let lots of other strings to be null
- nickname = models.CharField(max_length=64, null=True)
- password = models.CharField(max_length=128)
- email = models.EmailField(null=True, blank=True, default='')
- has_mh = models.BooleanField(default=False)
- avatar = models.CharField(max_length=1200, blank=True, default='')
- # LEVEL: 0-1 is default, everything else is just levels
- level = models.SmallIntegerField(default=0)
- # ROLE: This doesn't have anything
- role = models.SmallIntegerField(default=0, choices=((0, 'normal'), (1, 'bot'), (2, 'administrator'), (3, 'moderator'), (4, 'openverse'), (5, 'donator'), (6, 'tester'), (7, 'urapp'), (8, 'developer'), (9, 'pipinstalldjango') (10, 'staff') (11, 'Kanna' )( 12, 'realshit' ) (13, 'artcon') (14, 'contest') (15, 'gamecon') (16, 'mp') )
- addr = models.CharField(max_length=64, null=True, blank=True)
- # Things that don't have to do with auth lol
- hide_online = models.BooleanField(default=False)
- color = ColorField(default='', null=True, blank=True)
- staff = models.BooleanField(default=False)
- #active = models.SmallIntegerField(default=1, choices=((0, 'Disabled'), (1, 'Good'), (2, 'Redirect')))
- active = models.BooleanField(default=True)
- is_anonymous = False
- is_authenticated = True
- last_login = models.DateTimeField(auto_now=True)
- created = models.DateTimeField(auto_now_add=True)
- USERNAME_FIELD = 'username'
- EMAIL_FIELD = 'email'
- REQUIRED_FIELDS = []
- objects = UserManager()
- def __str__(self):
- return self.username
- def get_full_name(self):
- return self.username
- def get_short_name(self):
- return self.nickname
- def get_username(self):
- return self.username
- def has_module_perms(self, a):
- return True
- def has_perm(self, a):
- return self.staff
- def is_staff(self):
- return self.staff
- def is_active(self):
- return self.active
- def set_password(self, raw_password):
- self.password = bcrypt_sha256.using(rounds=13).hash(raw_password)
- def check_password(self, raw_password):
- return bcrypt_sha256.using(rounds=13).verify(raw_password, hash=self.password)
- def profile(self, thing=None):
- # If thing is specified, that field is retrieved
- if thing:
- return self.profile_set.all().values_list(thing, flat=True).first()
- # Otherwise just get full profile
- return self.profile_set.filter(user=self).first()
- def gravatar(self):
- g = util.get_gravatar(self.email)
- if not g:
- return settings.STATIC_URL + '/img/anonymous-mii.png'
- return g
- def mh(self):
- origin_info = self.profile('origin_info')
- if not origin_info:
- return None
- try:
- infodecode = json.loads(origin_info)
- except:
- return None
- return infodecode[0]
- def has_plain_avatar(self):
- if not self.has_mh and 'http' in self.avatar and not 'gravatar.com' in self.avatar:
- return True
- def has_avatar(self):
- if not self.avatar or len(self.avatar) == 1:
- return False
- return True
- def let_yeahnotifs(self):
- return self.profile('let_yeahnotifs')
- def limit_remaining(self):
- limit = self.profile('limit_post')
- # If False is returned, no post limit is assumed.
- if limit == 0:
- return False
- today_min = timezone.datetime.combine(timezone.datetime.today(), time.min)
- today_max = timezone.datetime.combine(timezone.datetime.today(), time.max)
- # recent_posts =
- # Posts made by the user today + posts made by the IP today +
- # same thing except with comments
- recent_posts = Post.real.filter(Q(creator=self.id, created__range=(today_min, today_max)) | Q(creator__addr=self.addr, created__range=(today_min, today_max))).count() + Comment.real.filter(Q(creator=self.id, created__range=(today_min, today_max)) | Q(creator__addr=self.addr, created__range=(today_min, today_max))).count()
- # Posts remaining
- return int(limit) - recent_posts
- def get_class(self):
- first = {
- 1: 'tester',
- 2: 'administrator',
- 3: 'moderator',
- 4: 'openverse',
- 5: 'donator',
- 6: 'tester',
- 7: 'urapp',
- 8: 'developer',
- 9: 'pipinstalldjango'
- 10: 'staff'
- 11: 'Kanna'
- 12: 'Verified'
- 13: 'artcon'
- 14: 'contest'
- 15: 'gamecom'
- 16: 'mp'
- }.get(self.role, '')
- second = {
- 1: "Bot",
- 2: "Administrator",
- 3: "Moderator",
- 4: "O-PHP-enverse Man",
- 5: "Donator",
- 6: "Tester",
- 7: "cave story is okay",
- 8: "Friendship ended with PHP / Now PYTHON is my best friend",
- 9: "Pip install Django"
- 10: "Staff"
- 11: "Kanna Kamui from Miss Kobayashi's Dragon Maid aka Best Girl"
- 12: "Verified"
- 13: "Art Contest Winner"
- 14: "Contest Winner"
- 15: "Game Contest Winner"
- 16: "Mario's Princess"
- }.get(self.role, '')
- if first:
- first = 'official ' + first
- return [first, second]
- def is_me(self, request):
- if request.user.is_authenticated:
- return (self == request.user)
- else:
- return False
- def has_freedom(self):
- return self.profile('let_freedom')
- # This is the coolest one
- def online_status(self, force=False):
- # Okay so this returns True if the user's offline, 2 if they're AFK, False if they're offline and None if they hide it
- if self.hide_online:
- return None
- if (timezone.now() - timedelta(seconds=50)) > self.last_login:
- return False
- elif (timezone.now() - timedelta(seconds=48)) > self.last_login:
- return 2
- else:
- return True
- def do_avatar(self, feeling=0):
- if self.has_mh and self.avatar:
- feeling = {
- 0: 'normal',
- 1: 'happy',
- 2: 'like',
- 3: 'surprised',
- 4: 'frustrated',
- 5: 'puzzled',
- }.get(feeling, "normal")
- url = 'https://mii-secure.cdn.nintendo.net/{0}_{1}_face.png'.format(self.avatar, feeling)
- return url
- elif not self.avatar:
- return settings.STATIC_URL + '/img/anonymous-mii.png'
- elif self.avatar == 's':
- return settings.STATIC_URL + '/img/anonymous-mii-sad.png'
- else:
- return self.avatar
- def num_yeahs(self):
- return self.yeah_set.filter(by=self).count()
- def num_posts(self):
- return self.post_set.filter(creator=self).count()
- def num_comments(self):
- return self.comment_set.filter().count()
- def num_following(self):
- return self.follow_source.filter().count()
- def num_followers(self):
- return self.follow_target.filter().count()
- def num_friends(self):
- return self.friend_source.filter().count() + self.friend_target.filter().count()
- def can_follow(self, user):
- #if UserBlock.find_block(self, user):
- # return False
- return True
- def can_view(self, user):
- #if UserBlock.find_block(self, user, full=True):
- # return False
- return True
- def is_following(self, me):
- if not me.is_authenticated:
- return False
- if self == me:
- return True
- #if hasattr(self, 'has_follow'):
- # return self.has_follow
- return self.follow_target.filter(source=me).count() > 0
- def follow(self, source):
- if self.is_following(source) or source == self:
- return False
- if not self.can_follow(source):
- return False
- # Todo: put a follow limit here
- return self.follow_target.create(source=source, target=self)
- def unfollow(self, source):
- if not self.is_following(source) or source == self:
- return False
- return self.follow_target.filter(source=source, target=self).delete()
- def can_block(self, source):
- if self.can_manage():
- return False
- #if source.profile('moyenne'):
- # return False
- return True
- # BLOCK this user from SOURCE
- def make_block(self, source, full=False):
- if find_block(source, self):
- return False
- return UserBlock.objects.create(source=source, target=self, full=full)
- def get_posts(self, limit=50, offset=0, request=None):
- if request.user.is_authenticated:
- has_yeah = Yeah.objects.filter(post=OuterRef('id'), by=request.user.id)
- posts = self.post_set.select_related('community').select_related('creator').annotate(num_yeahs=Count('yeah', distinct=True), num_comments=Count('comment', distinct=True), yeah_given=Exists(has_yeah, distinct=True)).filter().order_by('-created')[offset:offset + limit]
- else:
- posts = self.post_set.select_related('community').select_related('creator').annotate(num_yeahs=Count('yeah', distinct=True), num_comments=Count('comment', distinct=True)).filter().order_by('-created')[offset:offset + limit]
- if request:
- for post in posts:
- post.setup(request)
- post.recent_comment = post.recent_comment()
- return posts
- def get_comments(self, limit=50, offset=0, request=None):
- if request.user.is_authenticated:
- has_yeah = Yeah.objects.filter(comment=OuterRef('id'), by=request.user.id)
- posts = self.comment_set.select_related('original_post').select_related('creator').select_related('original_post__creator').annotate(num_yeahs=Count('yeah', distinct=True), yeah_given=Exists(has_yeah, distinct=True)).filter().order_by('-created')[offset:offset + limit]
- else:
- posts = self.comment_set.select_related('original_post').select_related('original_post__creator').select_related('creator').annotate(num_yeahs=Count('yeah', distinct=True)).filter().order_by('-created')[offset:offset + limit]
- if request:
- for post in posts:
- post.setup(request)
- return posts
- def get_yeahed(self, type=0, limit=20, offset=0):
- # 0 - post, 1 - comment, 2 - any
- if type == 2:
- yeahs = self.yeah_set.select_related('post').select_related('comment').select_related('comment__original_post').select_related('comment__original_post__creator').annotate(num_yeahs_post=Count('post__yeah', distinct=True), num_yeahs_comment=Count('comment__yeah', distinct=True), num_comments=Count('post__comment', distinct=True)).filter().order_by('-created')[offset:offset + limit]
- else:
- yeahs = self.yeah_set.select_related('post').select_related('post__creator').annotate(num_yeahs_post=Count('post__yeah', distinct=True), num_comments=Count('post__comment', distinct=True)).filter(type=type, post__is_rm=False).order_by('-created')[offset:offset + limit]
- for thing in yeahs:
- if thing.post:
- thing.post.num_yeahs = thing.num_yeahs_post
- thing.post.num_comments = thing.num_comments
- elif thing.comment:
- thing.comment.num_yeahs = thing.num_yeahs_comment
- return yeahs
- def get_following(self, limit=50, offset=0, request=None):
- return self.follow_source.select_related('target').filter().order_by('-created')[offset:offset + limit]
- def get_followers(self, limit=50, offset=0, request=None):
- return self.follow_target.select_related('source').filter().order_by('-created')[offset:offset + limit]
- def notification_count(self):
- return self.notification_to.filter(read=False).count()
- def notification_read(self):
- return self.notification_to.filter(read=False).update(read=True)
- def get_notifications(self):
- return self.notification_to.select_related('context_post').select_related('context_comment').select_related('source').filter().order_by('-latest')[0:64]
- def notifications_clean(self):
- """ Broken - gives OperationError on MySQL
- notif_get = self.notification_to.all().values_list('id', flat=True)
- if notif_get.count() > 64:
- self.notification_to.filter().exclude(id__in=notif_get).delete()
- """
- # Admin can-manage
- def can_manage(self):
- if (self.level >= 2) or self.is_staff():
- return True
- return False
- # Does user have authority over self?
- def has_authority(self, user):
- if self.is_staff():
- return False
- if (self.level < user.level):
- return True
- return False
- def friend_state(self, other):
- # Todo: return -1 for cannot, 0 for nothing, 1 for my friend pending, 2 for their friend pending, 3 for friends
- query1 = other.fr_source.filter(target=self, finished=False).exists()
- if query1:
- return 1
- query2 = self.fr_source.filter(target=other, finished=False).exists()
- if query2:
- return 2
- query3 = Friendship.find_friendship(self, other)
- if query3:
- return 3
- return 0
- def get_fr(self, other):
- return FriendRequest.objects.filter(Q(source=self) & Q(target=other) | Q(target=self) & Q(source=other)).exclude(finished=True)
- def get_frs_target(self):
- return FriendRequest.objects.filter(target=self, finished=False).order_by('-created')
- def get_frs_notif(self):
- return FriendRequest.objects.filter(target=self, finished=False, read=False).count()
- def reject_fr(self, target):
- fr = self.get_fr(target)
- if fr:
- try:
- fr.first().finish()
- except:
- pass
- def send_fr(self, source, body=None):
- if not self.get_fr(source):
- return FriendRequest.objects.create(source=source, target=self, body=body)
- def accept_fr(self, target):
- fr = self.get_fr(target)
- if fr:
- try:
- fr.first().finish()
- except:
- pass
- return Friendship.objects.create(source=self, target=target)
- def cancel_fr(self, target):
- fr = target.get_fr(self)
- if fr:
- try:
- fr.first().finish()
- except:
- pass
- def read_fr(self):
- return self.get_frs_target().update(read=True)
- def delete_friend(self, target):
- fr = Friendship.find_friendship(self, target)
- if fr:
- fr.conversation().all_read()
- fr.delete()
- def get_activity(self, limit=20, offset=0, distinct=False, friends_only=False, request=None):
- #Todo: make distinct work; combine friends and following, but then get posts from them
- friends = Friendship.get_friendships(self, 0)
- friend_ids = []
- for friend in friends:
- friend_ids.append(friend.other(self))
- follows = self.follow_source.filter().values_list('target', flat=True)
- if not friends_only:
- friend_ids.append(self.id)
- for thing in follows:
- friend_ids.append(thing)
- if request.user.is_authenticated:
- has_yeah = Yeah.objects.filter(post=OuterRef('id'), by=request.user.id)
- if distinct:
- posts = Post.objects.select_related('creator').select_related('community').annotate(num_yeahs=Count('yeah', distinct=True), num_comments=Count('comment', distinct=True), yeah_given=Exists(has_yeah, distinct=True)).annotate(max_created=Max('creator__post__created')).filter(created=F('max_created')).filter(creator__in=friend_ids).order_by('-created')[offset:offset + limit]
- else:
- posts = Post.objects.select_related('creator').select_related('community').annotate(num_yeahs=Count('yeah', distinct=True), num_comments=Count('comment', distinct=True), yeah_given=Exists(has_yeah, distinct=True)).filter(creator__in=friend_ids).order_by('-created')[offset:offset + limit]
- if request:
- for post in posts:
- post.setup(request)
- post.recent_comment = post.recent_comment()
- return posts
- def community_favorites(self, all=False):
- if not all:
- favorites = self.communityfavorite_set.order_by('-created')[:8]
- else:
- favorites = self.communityfavorite_set.order_by('-created')
- communities = []
- for fav in favorites:
- communities.append(fav.community)
- del(favorites)
- return communities
- def wake(self, addr=None):
- if addr and not addr == self.addr:
- self.addr = addr
- return self.save(update_fields=['addr', 'last_login'])
- return self.save(update_fields=['last_login'])
- def has_postspam(self, body, screenshot=None, drawing=None):
- latest_post = self.post_set.filter().order_by('-created')[:1]
- if not latest_post:
- return False
- latest_post = latest_post.first()
- if drawing and latest_post.drawing:
- if drawing == latest_post.drawing:
- return True
- elif latest_post.screenshot and screenshot and not drawing:
- if latest_post.screenshot == screenshot and latest_post.body == body:
- return True
- elif latest_post.body and body and not latest_post.screenshot and not latest_post.drawing:
- if latest_post.body == body:
- return True
- return False
- def get_latest_msg(self, me):
- conversation = Conversation.objects.filter(Q(source=self) & Q(target=me) | Q(target=self) & Q(source=me)).order_by('-created')[:1].first()
- if not conversation:
- return False
- return conversation.latest_message(me)
- def conversations(self):
- return Conversation.objects.filter(Q(source=self) | Q(target=self)).order_by('-created')
- def msg_count(self):
- # Gets messages with conversations I am involved in, then looks for those unread and not by me, gets count and returns it
- messages = Message.objects.filter(Q(conversation__source=self.id) | Q(conversation__target=self.id)).filter(read=False).exclude(creator=self.id).count()
- return messages
- def password_reset_email(self, request):
- htmlmsg = render_to_string('closedverse_main/help/email.html', {
- 'menulogo': request.build_absolute_uri(settings.STATIC_URL + 'img/menu-logo.png'),
- 'contact': request.build_absolute_uri(reverse('main:help-contact')),
- 'link': request.build_absolute_uri(reverse('main:forgot-passwd')) + "?token=" + base64.urlsafe_b64encode(bytes(self.password, 'utf-8')).decode(),
- })
- subj = 'Closedverse password reset for "{0}"'.format(self.username)
- return send_mail(subject=subj, html_message=htmlmsg, from_email="Closedverse not Openverse <{0}>".format(settings.DEFAULT_FROM_EMAIL), recipient_list=[self.email], fail_silently=False)
- def find_related(self):
- return User.objects.filter(id__in=LoginAttempt.objects.filter(Q(addr=self.addr), Q(user=self.id)).values_list('user', flat=True)).exclude(id=self.id)
- @staticmethod
- def search(query='', limit=50, offset=0, request=None):
- return User.objects.filter(Q(username__icontains=query) | Q(nickname__icontains=query)).order_by('-created')[offset:offset + limit]
- @staticmethod
- def email_in_use(addr, request=None):
- if not addr:
- return False
- if request:
- return User.objects.filter(email__iexact=addr).exclude(id=request.user.id).exists()
- else:
- return User.objects.filter(email__iexact=addr).exists()
- @staticmethod
- def nnid_in_use(id, request=None):
- if not id:
- return False
- if request:
- nnid_real = id.lower().replace('-', '').replace('.', '')
- return Profile.objects.filter(origin_id__iexact=nnid_real).exclude(user__id=request.user.id).exists()
- else:
- return Profile.objects.filter(origin_id=id).exists()
- @staticmethod
- def get_from_passwd(passwd):
- try:
- user = User.objects.get(password=base64.urlsafe_b64decode(passwd))
- # Too lazy to make except cases
- except:
- return False
- return user
- class Community(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- name = models.CharField(max_length=255)
- description = models.TextField(blank=True, default='')
- ico = models.CharField(max_length=255, blank=True)
- banner = models.CharField(max_length=255, blank=True)
- # Type: 0 - general, 1 - game, 2 - special
- type = models.SmallIntegerField(default=0, choices=((0, 'general'), (1, 'game'), (2, 'special'), (3, 'hide')))
- # Platform - 0/none, 1/3DS, 2/Wii U, 3/both
- platform = models.SmallIntegerField(default=0, choices=((0, 'none'), (1, '3ds'), (2, 'wii u'), (3, 'both')))
- tags = models.CharField(blank=True, null=True, max_length=255, choices=(('announcements', 'main announcement community'), ('changelog', 'main changelog'), ('activity', 'Activity Feed posting community')))
- created = models.DateTimeField(auto_now_add=True)
- updated = models.DateTimeField(auto_now=True)
- is_rm = models.BooleanField(default=False)
- is_feature = models.BooleanField(default=False)
- allowed_users = models.TextField(null=True, blank=True)
- creator = models.ForeignKey(User, blank=True, null=True, on_delete=models.CASCADE)
- objects = PostManager()
- real = models.Manager()
- def __str__(self):
- return self.name
- def icon(self):
- if not self.ico:
- return settings.STATIC_URL + "/img/title-icon-default.png"
- return self.ico
- def type_txt(self):
- if self.type == 1:
- return {
- 0: "",
- 1: "3DS Games",
- 2: "Wii U Games",
- 3: "Wii U Games\u30FB3DS Games",
- }.get(self.platform)
- else:
- return {
- 0: "General community",
- 1: "Game community",
- 2: "Special community",
- }.get(self.type)
- def type_platform(self):
- thing = {
- 0: "",
- 1: "3ds",
- 2: "wiiu",
- 3: "wiiu-3ds",
- }.get(self.platform)
- if thing == "":
- return None
- return "img/platform-tag-" + thing + ".png"
- def is_activity(self):
- return self.tags == 'activity'
- def clickable(self):
- return not self.is_activity() and not self.type == 3
- def get_posts(self, limit=50, offset=0, request=None, favorite=False):
- if request.user.is_authenticated:
- has_yeah = Yeah.objects.filter(post=OuterRef('id'), by=request.user.id)
- posts = Post.objects.select_related('creator').annotate(num_yeahs=Count('yeah', distinct=True), num_comments=Count('comment', distinct=True), yeah_given=Exists(has_yeah, distinct=True)).filter(community_id=self.id).order_by('-created')[offset:offset + limit]
- else:
- posts = Post.objects.select_related('creator').annotate(num_yeahs=Count('yeah', distinct=True), num_comments=Count('comment', distinct=True)).filter(community_id=self.id).order_by('-created')[offset:offset + limit]
- if request:
- for post in posts:
- post.setup(request)
- post.recent_comment = post.recent_comment()
- return posts
- def post_perm(self, request):
- if self.allowed_users:
- allows = self.allowed_users.split(',')
- if not request.user.is_authenticated or str(request.user.id) not in allows:
- return False
- return True
- else:
- return True
- def has_favorite(self, request):
- if request.user.communityfavorite_set.filter(community=self).exists():
- return True
- return False
- def favorite_add(self, request):
- if not self.has_favorite(request):
- return request.user.communityfavorite_set.create(community=self)
- def favorite_rm(self, request):
- if self.has_favorite(request):
- return request.user.communityfavorite_set.filter(community=self).delete()
- def setup(self, request):
- if request.user.is_authenticated:
- self.post_perm = self.post_perm(request)
- self.has_favorite = self.has_favorite(request)
- def create_post(self, request):
- if not self.post_perm(request):
- return 4
- limit = request.user.limit_remaining()
- if not limit is False and not limit > 0:
- return 8
- del(limit)
- if Post.real.filter(Q(creator=request.user) | Q(creator__addr=request.user.addr), created__gt=timezone.now() - timedelta(seconds=10)).exists():
- return 3
- if request.POST.get('url'):
- try:
- URLValidator()(value=request.POST['url'])
- except ValidationError:
- return 5
- if not request.user.has_freedom() and (request.POST.get('url') or request.FILES.get('screen')):
- return 6
- if len(request.POST['body']) > 2200 or (len(request.POST['body']) < 1 and not request.POST.get('_post_type') == 'painting'):
- return 1
- upload = None
- drawing = None
- if request.FILES.get('screen'):
- upload = util.image_upload(request.FILES['screen'], True)
- if upload == 1:
- return 2
- if request.POST.get('_post_type') == 'painting':
- if not request.POST.get('painting'):
- return 2
- drawing = util.image_upload(request.POST['painting'], False, True)
- if drawing == 1:
- return 2
- # Check for spam using our OWN ALGO!!!!!!!!!
- if request.user.has_postspam(request.POST.get('body'), upload, drawing):
- return 7
- new_post = self.post_set.create(body=request.POST.get('body'), creator=request.user, community=self, feeling=int(request.POST.get('feeling_id', 0)), spoils=bool(request.POST.get('is_spoiler')), screenshot=upload, drawing=drawing, url=request.POST.get('url'))
- new_post.is_mine = True
- return new_post
- def search(query='', limit=50, offset=0, request=None):
- return Community.objects.filter(Q(name__icontains=query) | Q(description__contains=query)).exclude(type=3).order_by('-created')[offset:offset + limit]
- def get_all(type=0, offset=0, limit=12):
- return Community.objects.filter(type=type).order_by('-created')[offset:offset + limit]
- class Meta:
- verbose_name_plural = "communities"
- # Links between communities for "related" communities
- class CommunityClink(models.Model):
- # root/also order doesn't matter, time does though
- root = models.ForeignKey(Community, related_name='one', on_delete=models.CASCADE)
- also = models.ForeignKey(Community, related_name='two', on_delete=models.CASCADE)
- created = models.DateTimeField(auto_now_add=True)
- # type: related (f) / sub (t)
- kind = models.BooleanField(default=False)
- objects = CommunityFavoriteManager()
- # Do this, or not
- class CommunityFavorite(models.Model):
- id = models.AutoField(primary_key=True)
- by = models.ForeignKey(User, on_delete=models.CASCADE)
- community = models.ForeignKey(Community, on_delete=models.CASCADE)
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- return "Community favorite by " + str(self.by) + " for " + str(self.community)
- class Post(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- community = models.ForeignKey(Community, null=True, on_delete=models.CASCADE)
- feeling = models.SmallIntegerField(default=0, choices=feelings)
- body = models.TextField(null=True)
- drawing = models.CharField(max_length=200, null=True, blank=True)
- screenshot = models.CharField(max_length=1200, null=True, blank=True, default='')
- url = models.URLField(max_length=1200, null=True, blank=True, default='')
- spoils = models.BooleanField(default=False)
- created = models.DateTimeField(auto_now_add=True)
- edited = models.DateTimeField(auto_now=True)
- befores = models.TextField(null=True, blank=True)
- poll = models.ForeignKey('Poll', null=True, blank=True, on_delete=models.CASCADE)
- has_edit = models.BooleanField(default=False)
- is_rm = models.BooleanField(default=False)
- status = models.SmallIntegerField(default=0, choices=post_status)
- creator = models.ForeignKey(User, on_delete=models.CASCADE)
- objects = PostManager()
- real = models.Manager()
- def __str__(self):
- return self.body[:250]
- def is_reply(self):
- return False
- def trun(self):
- if self.is_rm:
- return 'deleted'
- if self.drawing:
- return 'drawing'
- else:
- return self.body
- def yt_vid(self):
- try:
- thing = re.search('(https?://)?(www\.)?(youtube|youtu|youtube-nocookie)\.(com|be)/(watch\?v=|embed/|v/|.+\?v=)?([^&=%\?]{11})', self.url).group(6)
- except:
- return False
- return thing
- def has_line_trun(self):
- if self.body and len(self.body.splitlines()) > 10:
- return True
- return False
- def is_mine(self, user):
- if user.is_authenticated:
- return (self.creator == user)
- else:
- return False
- #def yeah_notification(self, request):
- # ???? What is this
- #Notification.give_notification
- def number_yeahs(self):
- if hasattr(self, 'num_yeahs'):
- return self.num_yeahs
- return self.yeah_set.filter(post=self).count()
- def has_yeah(self, request):
- if request.user.is_authenticated:
- if hasattr(self, 'yeah_given'):
- return self.yeah_given
- else:
- return self.yeah_set.filter(post=self, by=request.user).exists()
- else:
- return False
- def can_yeah(self, request):
- if request.user.is_authenticated:
- #if UserBlock.find_block(self.creator, request.user):
- # return False
- return not self.is_mine(request.user)
- else:
- return False
- def can_rm(self, request):
- if self.creator.has_authority(request.user):
- return True
- return False
- def give_yeah(self, request):
- if not request.user.has_freedom() and Yeah.objects.filter(by=request.user, created__gt=timezone.now() - timedelta(seconds=5)).exists():
- return False
- if self.has_yeah(request):
- return True
- if not self.can_yeah(request):
- return False
- return self.yeah_set.create(by=request.user, post=self)
- def remove_yeah(self, request):
- if not self.has_yeah(request):
- return True
- return self.yeah_set.filter(post=self, by=request.user).delete()
- def number_comments(self):
- # Number of comments cannot be accurate due to comment deleting
- #if hasattr(self, 'num_comments'):
- # return self.num_comments
- return self.comment_set.filter(original_post=self).count()
- def get_yeahs(self, request):
- return Yeah.objects.filter(type=0, post=self).order_by('-created')[0:30]
- def can_comment(self, request):
- # TODO: Make this so that if a post's comments exceeds 100, make the user able to close the comments section
- if self.number_comments() > 500:
- return False
- #if UserBlock.find_block(self.creator, request.user):
- # return False
- return True
- def get_comments(self, request=None, limit=0, offset=0):
- if request.user.is_authenticated:
- has_yeah = Yeah.objects.filter(comment=OuterRef('id'), by=request.user.id)
- if limit:
- comments = self.comment_set.select_related('creator').annotate(num_yeahs=Count('yeah'), yeah_given=Exists(has_yeah)).filter(original_post=self).order_by('created')[offset:offset + limit]
- elif offset:
- comments = self.comment_set.select_related('creator').annotate(num_yeahs=Count('yeah'), yeah_given=Exists(has_yeah)).filter(original_post=self).order_by('created')[offset:]
- else:
- comments = self.comment_set.select_related('creator').annotate(num_yeahs=Count('yeah'), yeah_given=Exists(has_yeah)).filter(original_post=self).order_by('created')
- else:
- if limit:
- comments = self.comment_set.select_related('creator').annotate(num_yeahs=Count('yeah')).filter(original_post=self).order_by('created')[offset:offset + limit]
- elif offset:
- comments = self.comment_set.select_related('creator').annotate(num_yeahs=Count('yeah')).filter(original_post=self).order_by('created')[offset:]
- else:
- comments = self.comment_set.select_related('creator').annotate(num_yeahs=Count('yeah')).filter(original_post=self).order_by('created')
- if request:
- for post in comments:
- post.setup(request)
- return comments
- def create_comment(self, request):
- if not self.can_comment(request):
- return False
- limit = request.user.limit_remaining()
- if not limit is False and not limit > 0:
- return 8
- del(limit)
- if self.is_mine(request.user) and Comment.real.filter(creator=request.user, created__gt=timezone.now() - timedelta(seconds=2)).exists():
- return 3
- elif not self.is_mine(request.user) and Comment.real.filter(creator=request.user, created__gt=timezone.now() - timedelta(seconds=10)).exists():
- return 3
- if not request.user.has_freedom() and (request.POST.get('url') or request.FILES.get('screen')):
- return 6
- if len(request.POST['body']) > 2200 or (len(request.POST['body']) < 1 and not request.POST.get('_post_type') == 'painting'):
- return 1
- upload = None
- drawing = None
- if request.FILES.get('screen'):
- upload = util.image_upload(request.FILES['screen'], True)
- if upload == 1:
- return 2
- if request.POST.get('_post_type') == 'painting':
- if not request.POST.get('painting'):
- return 2
- drawing = util.image_upload(request.POST['painting'], False, True)
- if drawing == 1:
- return 2
- new_post = self.comment_set.create(body=request.POST.get('body'), creator=request.user, community=self.community, original_post=self, feeling=int(request.POST.get('feeling_id', 0)), spoils=bool(request.POST.get('is_spoiler')), drawing=drawing, screenshot=upload)
- new_post.is_mine = True
- return new_post
- def recent_comment(self):
- if self.number_comments() < 1:
- return False
- comments = self.comment_set.filter(spoils=False).exclude(creator=self.creator).order_by('-created')[:1]
- return comments.first()
- def change(self, request):
- if not self.is_mine(request.user) or self.has_edit:
- return 1
- if len(request.POST['body']) > 2200 or len(request.POST['body']) < 1:
- return 1
- if not self.befores:
- befores_json = []
- else:
- befores_json = json.loads(self.befores)
- befores_json.append(self.body)
- self.befores = json.dumps(befores_json)
- self.body = request.POST['body']
- self.spoils = request.POST.get('is_spoiler', False)
- self.feeling = request.POST.get('feeling_id', 0)
- if not timezone.now() < self.created + timezone.timedelta(minutes=2):
- self.has_edit = True
- return self.save()
- def is_favorite(self, user):
- profile = user.profile()
- if profile.favorite == self:
- return True
- else:
- return False
- def favorite(self, user):
- if not self.is_mine(user):
- return False
- profile = user.profile()
- if profile.favorite == self:
- return False
- profile.favorite = self
- return profile.save()
- def unfavorite(self, user):
- if not self.is_mine(user):
- return False
- profile = user.profile()
- if profile.favorite == self:
- profile.favorite = None
- return profile.save()
- def rm(self, request):
- if request and not self.is_mine(request.user) and not self.can_rm(request):
- return False
- if self.is_favorite(self.creator):
- self.unfavorite(self.creator)
- self.is_rm = True
- if self.is_mine(request.user):
- self.status = 1
- else:
- self.status = 2
- AuditLog.objects.create(type=0, post=self, user=self.creator, by=request.user)
- if self.screenshot:
- util.image_rm(self.screenshot)
- self.screenshot = None
- if self.drawing:
- util.image_rm(self.drawing)
- self.drawing = None
- self.save()
- def setup(self, request):
- self.has_yeah = self.has_yeah(request)
- self.can_yeah = self.can_yeah(request)
- self.is_mine = self.is_mine(request.user)
- def max_yeahs():
- try:
- max_yeahs_post = Post.objects.annotate(num_yeahs=Count('yeah')).aggregate(max_yeahs=Max('num_yeahs'))['max_yeahs']
- except:
- return None
- the_post = Post.objects.annotate(num_yeahs=Count('yeah')).filter(num_yeahs=max_yeahs_post).order_by('-created')
- return the_post.first()
- class Comment(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- original_post = models.ForeignKey(Post, on_delete=models.CASCADE)
- community = models.ForeignKey(Community, on_delete=models.CASCADE)
- feeling = models.SmallIntegerField(default=0, choices=feelings)
- body = models.TextField(null=True)
- screenshot = models.CharField(max_length=1200, null=True, blank=True, default='')
- drawing = models.CharField(max_length=200, null=True, blank=True)
- spoils = models.BooleanField(default=False)
- created = models.DateTimeField(auto_now_add=True)
- edited = models.DateTimeField(auto_now=True)
- befores = models.TextField(null=True, blank=True)
- has_edit = models.BooleanField(default=False)
- is_rm = models.BooleanField(default=False)
- status = models.SmallIntegerField(default=0, choices=post_status)
- creator = models.ForeignKey(User, blank=True, null=True, on_delete=models.CASCADE)
- objects = PostManager()
- real = models.Manager()
- def __str__(self):
- return self.body[:250]
- def is_reply(self):
- return True
- def trun(self):
- if self.is_rm:
- return 'deleted'
- if self.drawing:
- return '(drawing)'
- else:
- return self.body
- def is_mine(self, user):
- if user.is_authenticated:
- return (self.creator == user)
- else:
- return False
- def number_yeahs(self):
- if hasattr(self, 'num_yeahs'):
- return self.num_yeahs
- return self.yeah_set.filter(comment=self, type=1).count()
- def has_yeah(self, request):
- if request.user.is_authenticated:
- if hasattr(self, 'yeah_given'):
- return self.yeah_given
- else:
- return self.yeah_set.filter(comment=self, type=1, by=request.user).exists()
- else:
- return False
- def can_yeah(self, request):
- if request.user.is_authenticated:
- return not self.is_mine(request.user)
- #if UserBlock.find_block(self.creator, request.user):
- # return False
- else:
- return False
- def can_rm(self, request):
- if self.creator.has_authority(request.user):
- return True
- #if self.original_post.is_mine(request.user):
- # return True
- return False
- def give_yeah(self, request):
- if not request.user.has_freedom() and Yeah.objects.filter(by=request.user, created__gt=timezone.now() - timedelta(seconds=5)).exists():
- return False
- if self.has_yeah(request):
- return True
- if not self.can_yeah(request):
- return False
- return self.yeah_set.create(by=request.user, type=1, comment=self)
- def remove_yeah(self, request):
- if not self.has_yeah(request):
- return True
- return self.yeah_set.filter(comment=self, type=1, by=request.user).delete()
- def get_yeahs(self, request):
- return Yeah.objects.filter(type=1, comment=self).order_by('-created')[0:30]
- def owner_post(self):
- return (self.creator == self.original_post.creator)
- def change(self, request):
- if not self.is_mine(request.user) or self.has_edit:
- return 1
- if len(request.POST['body']) > 2200 or len(request.POST['body']) < 1:
- return 1
- if not self.befores:
- befores_json = []
- else:
- befores_json = json.loads(self.befores)
- befores_json.append(self.body)
- self.befores = json.dumps(befores_json)
- self.body = request.POST['body']
- self.spoils = request.POST.get('is_spoiler', False)
- self.feeling = request.POST.get('feeling_id', 0)
- if not timezone.now() < self.created + timezone.timedelta(minutes=2):
- self.has_edit = True
- return self.save()
- def rm(self, request):
- if request and not self.is_mine(request.user) and not self.can_rm(request):
- return False
- self.is_rm = True
- if self.is_mine(request.user):
- self.status = 1
- else:
- self.status = 2
- AuditLog.objects.create(type=1, comment=self, user=self.creator, by=request.user)
- if self.screenshot:
- util.image_rm(self.screenshot)
- self.screenshot = None
- if self.drawing:
- util.image_rm(self.drawing)
- self.drawing = None
- self.save()
- def setup(self, request):
- self.has_yeah = self.has_yeah(request)
- self.can_yeah = self.can_yeah(request)
- self.is_mine = self.is_mine(request.user)
- class Yeah(models.Model):
- # Todo: make this a plain int at some point
- id = models.UUIDField(primary_key=True, default=uuid.uuid4, editable=False, unique=True)
- by = models.ForeignKey(User, on_delete=models.CASCADE)
- type = models.SmallIntegerField(default=0, choices=((0, 'post'), (1, 'comment'), ))
- post = models.ForeignKey(Post, null=True, blank=True, on_delete=models.CASCADE)
- # kldsjfldsfsdfd
- #spam = models.BooleanField(default=False)
- comment = models.ForeignKey(Comment, null=True, blank=True, on_delete=models.CASCADE)
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- a = "from " + self.by.username + " to "
- if self.post:
- a += str(self.post.unique_id)
- elif self.comment:
- a += str(self.comment.unique_id)
- return a
- class Profile(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- user = models.ForeignKey(User, on_delete=models.CASCADE)
- origin_id = models.CharField(max_length=16, null=True, blank=True)
- origin_info = models.CharField(max_length=255, null=True, blank=True)
- comment = models.TextField(blank=True, default='')
- country = models.CharField(max_length=120, blank=True, default='')
- #birthday = models.DateField(null=True, blank=True)
- id_visibility = models.SmallIntegerField(default=0, choices=visibility)
- pronoun_is = models.IntegerField(default=0, choices=(
- (0, "I don't know"), (1, "He/him"), (2, "She/her"), (3, "He/she"), (4, "It")
- ))
- let_friendrequest = models.SmallIntegerField(default=0, choices=visibility)
- yeahs_visibility = models.SmallIntegerField(default=0, choices=visibility)
- comments_visibility = models.SmallIntegerField(default=2, choices=visibility)
- #relationship_visibility = models.SmallIntegerField(default=0, choices=visibility)
- weblink = models.CharField(max_length=1200, blank=True, default='')
- #gameskill = models.SmallIntegerField(default=0)
- external = models.CharField(max_length=255, blank=True, default='')
- favorite = models.ForeignKey(Post, blank=True, null=True, on_delete=models.CASCADE)
- let_yeahnotifs = models.BooleanField(default=True)
- let_freedom = models.BooleanField(default=True)
- # Todo: When you see this, implement it; make it a bool that determines whether the user should be able to edit their avatar; if this is true and
- #let_avatar = models.BooleanField(default=False)
- adopted = models.ForeignKey(User, null=True, blank=True, related_name='children', on_delete=models.CASCADE)
- # Post limit, 0 for none
- limit_post = models.SmallIntegerField(default=0)
- # If this is true, the user can't change their avatar or nickname
- cannot_edit = models.BooleanField(default=False)
- email_login = models.SmallIntegerField(default=1, choices=((0, 'Do not allow'), (1, 'Okay'), (2, 'Only allow')))
- def __str__(self):
- return "profile " + str(self.unique_id) + " for " + self.user.username
- def origin_id_public(self, user=None):
- if user == self.user:
- return self.origin_id
- if self.id_visibility == 2:
- return 1
- elif self.id_visibility == 1:
- if not user.is_authenticated or not Friendship.find_friendship(self.user, user):
- return 1
- return self.origin_id
- elif not self.origin_id:
- return None
- return self.origin_id
- def yeahs_visible(self, user=None):
- if user == self.user:
- return True
- if self.yeahs_visibility == 2:
- return False
- elif self.yeahs_visibility == 1:
- if not user.is_authenticated or not Friendship.find_friendship(self.user, user):
- return False
- return True
- return True
- def comments_visible(self, user=None):
- if user == self.user:
- return True
- if self.comments_visibility == 2:
- return False
- elif self.comments_visibility == 1:
- if not user.is_authenticated or not Friendship.find_friendship(self.user, user):
- return False
- return True
- return True
- def can_friend(self, user=None):
- if self.let_friendrequest == 2:
- return False
- #if user.is_authenticated and UserBlock.find_block(self.user, user):
- # return False
- elif self.let_friendrequest == 1:
- if not user.is_following(self.user):
- return False
- return True
- return True
- def got_fullurl(self):
- if self.weblink:
- try:
- URLValidator()(value=self.weblink)
- except ValidationError:
- return False
- return True
- return False
- def setup(self, request):
- self.origin_id_public = self.origin_id_public(request.user)
- self.yeahs_visible = self.yeahs_visible(request.user)
- self.comments_visible = self.comments_visible(request.user)
- class Follow(models.Model):
- # Todo: remove this
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- source = models.ForeignKey(User, related_name='follow_source', on_delete=models.CASCADE)
- target = models.ForeignKey(User, related_name='follow_target', on_delete=models.CASCADE)
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- return "follow: from " + self.source.username + " to " + self.target.username
- class Notification(models.Model):
- # Todo: make this a plain int at some point
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- to = models.ForeignKey(User, related_name='notification_to', on_delete=models.CASCADE)
- source = models.ForeignKey(User, related_name='notification_sender', on_delete=models.CASCADE)
- read = models.BooleanField(default=False)
- type = models.SmallIntegerField(choices=(
- (0, 'Yeah on post'),
- (1, 'Yeah on comment'),
- (2, 'Comment on my post'),
- (3, 'Comment on others\' post'),
- (4, 'Follow to me'),
- ))
- merges = models.TextField(blank=True, default='')
- context_post = models.ForeignKey(Post, null=True, blank=True, on_delete=models.CASCADE)
- context_comment = models.ForeignKey(Comment, null=True, blank=True, on_delete=models.CASCADE)
- created = models.DateTimeField(auto_now_add=True)
- latest = models.DateTimeField(auto_now=True)
- def __str__(self):
- return "Notification from " + str(self.source) + " to " + str(self.to) + " with type \"" + self.get_type_display() + "\""
- def url(self):
- what_type = {
- 0: 'main:post-view',
- 1: 'main:comment-view',
- 2: 'main:post-view',
- 3: 'main:post-view',
- 4: 'main:user-followers',
- }.get(self.type)
- if self.type == 0 or self.type == 2 or self.type == 3:
- what_id = self.context_post_id
- elif self.type == 1:
- what_id = self.context_comment_id
- elif self.type == 4:
- what_id = self.to.username
- return reverse(what_type, args=[what_id])
- def merge(self, user):
- self.latest = timezone.now()
- self.read = False
- if not self.merges:
- u = []
- else:
- u = json.loads(self.merges)
- u.append(user.id)
- self.merges = json.dumps(u)
- self.save()
- #return self.merged.create(source=user, to=self.to, type=self.type, context_post=self.context_post, context_comment=self.context_comment)
- def set_unread(self):
- self.read = False
- self.latest = timezone.now()
- return self.save()
- def all_users(self):
- if not self.merges:
- u = []
- else:
- u = organ(json.loads(self.merges))
- arr = []
- arr.append(self.source)
- for user in u:
- # Todo: Clean this up and make this block better
- if not u in arr:
- try:
- arr.append(User.objects.get(id=user))
- except:
- pass
- del(u)
- return arr
- """
- arr = []
- arr.append(self.source)
- merges = self.merged.filter().order_by('created')
- for merge in merges:
- arr.append(merge.source)
- return arr
- """
- def setup(self, user):
- # Only have is_following if the type is a follow; save SQL queries
- if self.type == 4:
- self.source.is_following = self.source.is_following(user)
- else:
- self.source.is_following = False
- # In the future, please put giving notifications for classes into their respective classes (right now they're in views)
- @staticmethod
- def give_notification(user, type, to, post=None, comment=None):
- # Just keeping this simple for now, might want to make it better later
- # If the user sent a notification to this user at least 5 seconds ago, return False
- # Or if user is to
- # Or if yeah notifications are off and this is a yeah notification
- user_is_self_unk = (not type == 3 and user == to)
- is_notification_too_fast = (user.notification_sender.filter(created__gt=timezone.now() - timedelta(seconds=5), type=type).exclude(type=4) and not type == 3)
- user_no_yeahnotif = (not to.let_yeahnotifs() and (type == 0 or type == 1))
- if user_is_self_unk or is_notification_too_fast or user_no_yeahnotif:
- return False
- # Search for my own notifiaction. If it exists, set it as unread.
- merge_own = user.notification_sender.filter(created__gt=timezone.now() - timedelta(hours=8), to=to, type=type, context_post=post, context_comment=comment)
- if merge_own:
- # If it's merged, don't unread that one, but unread what it's merging.
- return merge_own.first().set_unread()
- # Search for a notification already there so we can merge with it if it exists
- merge_s = Notification.objects.filter(created__gt=timezone.now() - timedelta(hours=8), to=to, type=type, context_post=post, context_comment=comment)
- # If it exists, merge with it. Else, create a new notification.
- if merge_s:
- return merge_s.first().merge(user)
- else:
- return user.notification_sender.create(source=user, type=type, to=to, context_post=post, context_comment=comment)
- class Complaint(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- creator = models.ForeignKey(User, on_delete=models.CASCADE)
- type = models.SmallIntegerField(choices=(
- (0, 'Bug report'),
- (1, 'Suggestion'),
- (2, 'Want'),
- ))
- body = models.TextField(blank=True, default='')
- sex = models.SmallIntegerField(null=True, choices=((0, 'girl'), (1, 'privileged one'), (2, '(none)'),
- ))
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- return "\"" + str(self.body) + "\" from " + str(self.creator) + " as a " + str(self.get_sex_display())
- def has_past_sent(user):
- return user.complaint_set.filter(created__gt=timezone.now() - timedelta(minutes=5)).exists()
- class FriendRequest(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- source = models.ForeignKey(User, related_name='fr_source', on_delete=models.CASCADE)
- target = models.ForeignKey(User, related_name='fr_target', on_delete=models.CASCADE)
- body = models.TextField(blank=True, null=True, default='')
- read = models.BooleanField(default=False)
- finished = models.BooleanField(default=False)
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- return "friend request ("+str(self.finished)+"): from " + str(self.source) + " to " + str(self.target)
- def finish(self):
- self.finished = True
- self.save()
- class Friendship(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- source = models.ForeignKey(User, related_name='friend_source', on_delete=models.CASCADE)
- target = models.ForeignKey(User, related_name='friend_target', on_delete=models.CASCADE)
- created = models.DateTimeField(auto_now_add=True)
- latest = models.DateTimeField(auto_now=True)
- def __str__(self):
- return "friendship with " + str(self.source) + " and " + str(self.target)
- def update(self):
- return self.save(update_fields=['latest'])
- def other(self, user):
- if self.source == user:
- return self.target
- return self.source
- def conversation(self):
- conv = Conversation.objects.filter(Q(source=self.source) & Q(target=self.target) | Q(target=self.source) & Q(source=self.target)).order_by('-created')
- if not conv:
- return Conversation.objects.create(source=self.source, target=self.target)
- return conv.first()
- @staticmethod
- def get_friendships(user, limit=50, offset=0, latest=False, online_only=False):
- if not limit:
- return Friendship.objects.filter(Q(source=user) | Q(target=user)).order_by('-created')
- if latest:
- if online_only:
- delta = timezone.now() - timedelta(seconds=48)
- awman = []
- for friend in Friendship.objects.filter(Q(source=user) | Q(target=user)).order_by('-latest')[offset:offset + limit]:
- if friend.other(user).last_login > delta:
- awman.append(friend)
- return awman
- # Fix all of this at some point
- # return Friendship.objects.filter(
- #source=When(source__ne=user, source__last_login__gt=delta),
- #target=When(target__ne=user, target__last_login__gt=delta)
- #).order_by('-latest')[offset:offset + limit]
- else:
- return Friendship.objects.filter(Q(source=user) | Q(target=user)).order_by('-latest')[offset:offset + limit]
- else:
- return Friendship.objects.filter(Q(source=user) | Q(target=user)).order_by('-created')[offset:offset + limit]
- @staticmethod
- def find_friendship(first, second):
- return Friendship.objects.filter(Q(source=first) & Q(target=second) | Q(target=first) & Q(source=second)).order_by('-created').first()
- @staticmethod
- def get_friendships_message(user, limit=20, offset=0, online_only=False):
- friends_list = Friendship.get_friendships(user, limit, offset, True, online_only)
- friends = []
- for friend in friends_list:
- friends.append(friend.other(user))
- del(friends_list)
- for friend in friends:
- friend.get_latest_msg = friend.get_latest_msg(user)
- return friends
- class Conversation(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- source = models.ForeignKey(User, related_name='conv_source', on_delete=models.CASCADE)
- target = models.ForeignKey(User, related_name='conv_target', on_delete=models.CASCADE)
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- return "conversation with " + str(self.source) + " and " + str(self.target)
- def latest_message(self, user):
- msgs = Message.objects.filter(conversation=self).order_by('-created')[:5]
- if not msgs:
- return False
- message = msgs.first()
- message.mine = message.mine(user)
- return message
- def unread(self, user):
- return self.message_set.filter(read=False).exclude(creator=user).order_by('-created')
- def set_read(self, user):
- return self.unread(user).update(read=True)
- def all_read(self):
- return self.message_set.filter().update(read=True)
- def messages(self, request, limit=50, offset=0):
- msgs = self.message_set.filter().order_by('-created')[offset:offset + limit]
- for msg in msgs:
- msg.mine = msg.mine(request.user)
- return msgs
- def make_message(self, request):
- if not request.user.has_freedom() and (request.POST.get('url') or request.FILES.get('screen')):
- return 6
- if Message.real.filter(creator=request.user, created__gt=timezone.now() - timedelta(seconds=2)).exists():
- return 3
- if len(request.POST['body']) > 50000 or (len(request.POST['body']) < 1 and not request.POST.get('_post_type') == 'painting'):
- return 1
- upload = None
- drawing = None
- if request.FILES.get('screen'):
- upload = util.image_upload(request.FILES['screen'], True)
- if upload == 1:
- return 2
- if request.POST.get('_post_type') == 'painting':
- if not request.POST.get('painting'):
- return 2
- drawing = util.image_upload(request.POST['painting'], False, True)
- if drawing == 1:
- return 2
- new_post = self.message_set.create(body=request.POST.get('body'), creator=request.user, feeling=int(request.POST.get('feeling_id', 0)), drawing=drawing, screenshot=upload)
- new_post.mine = True
- return new_post
- class Message(models.Model):
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- conversation = models.ForeignKey(Conversation, on_delete=models.CASCADE)
- feeling = models.SmallIntegerField(default=0, choices=feelings)
- body = models.TextField(null=True)
- drawing = models.CharField(max_length=200, null=True, blank=True)
- screenshot = models.CharField(max_length=1200, null=True, blank=True, default='')
- url = models.URLField(max_length=1200, null=True, blank=True, default='')
- created = models.DateTimeField(auto_now_add=True)
- read = models.BooleanField(default=False)
- is_rm = models.BooleanField(default=False)
- creator = models.ForeignKey(User, on_delete=models.CASCADE)
- objects = PostManager()
- real = models.Manager()
- def __str__(self):
- return self.body[:250]
- def trun(self):
- if self.is_rm:
- return 'deleted'
- if self.drawing:
- return '(drawing)'
- else:
- return self.body
- def mine(self, user):
- if self.creator == user:
- return True
- return False
- def rm(self, request):
- if self.conversation.source == request.user or self.conversation.target == request.user:
- self.is_rm = True
- if self.screenshot:
- util.image_rm(self.screenshot)
- self.screenshot = None
- if self.drawing:
- util.image_rm(self.drawing)
- self.drawing = None
- self.save()
- def makeopt(ls):
- if len(ls) < 1:
- raise ValueError
- return json.dumps(ls)
- class ConversationInvite(models.Model):
- id = models.AutoField(primary_key=True)
- conversation = models.ForeignKey(Conversation, on_delete=models.CASCADE)
- source = models.ForeignKey(User, related_name='convinvite_source', on_delete=models.CASCADE)
- target = models.ForeignKey(User, related_name='convinvite_target', on_delete=models.CASCADE)
- body = models.TextField(blank=True, null=True, default='')
- read = models.BooleanField(default=False)
- finished = models.BooleanField(default=False)
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- return "Invite to conversation " + str(self.conversation) + " from " + str(self.source) + " to " + str(self.target)
- class Poll(models.Model):
- # Todo: make this a plain int at some point
- unique_id = models.UUIDField(default=uuid.uuid4, editable=False, unique=True)
- id = models.AutoField(primary_key=True)
- able_vote = models.BooleanField(default=True)
- choices = models.TextField(default="[]")
- created = models.DateTimeField(auto_now_add=True)
- def __str__(self):
- return "A poll created at " + str(self.created)
- def num_votes(self):
- return self.pollvote_set.count()
- def vote(self, user, opt):
- ex_query = self.pollvote_set.filter(by=user)
- if ex_query.exists():
- ex_query.first().delete()
- self.pollvote_set.create(by=user, choice=opt)
- def unvote(self, user):
- vote = self.pollvote_set.filter(by=user).first()
- if vote:
- vote.delete()
- def has_vote(self, user):
- if not user.is_authenticated:
- return False
- vote = self.pollvote_set.filter(by=user)
- if vote:
- return (True, self.choices[vote.first().choice])
- return False
- def setup(self, user):
- self.choices = json.loads(self.choices)
- self.num_votes = self.num_votes()
- self.has_vote = self.has_vote(user)
- class PollVote(models.Model):
- id = models.AutoField(primary_key=True)
- done = models.DateTimeField(auto_now_add=True)
- choice = models.SmallIntegerField(default=0)
- poll = models.ForeignKey(Poll, on_delete=models.CASCADE)
- by = models.ForeignKey(User, on_delete=models.CASCADE)
- def __str__(self):
- return "A vote on option " + str(self.choice) + " for poll \"" + str(self.poll) + "\" by " + str(self.by)
- #def choice_votes(self):
- # return PollVote.objects.filter(poll=self.poll, choice=self.choice).count()
- class RedFlag(models.Model):
- id = models.AutoField(primary_key=True)
- created = models.DateTimeField(auto_now_add=True)
- post = models.ForeignKey(Post, blank=True, null=True, on_delete=models.CASCADE)
- comment = models.ForeignKey(Comment, blank=True, null=True, on_delete=models.CASCADE)
- user = models.ForeignKey(User, blank=True, null=True, on_delete=models.CASCADE)
- type = models.SmallIntegerField(choices=((0, 'Post'), (1, 'Comment'), (2, 'User'), ))
- reason = models.SmallIntegerField(choices=((0, "Actual harassment"), (1, "Spam"), (2, "I don't like this"), (3, "Personal info"), (4, "Obscene use of swearing"), (5, "NSFW where not allowed"), (6, "Overly advertising/spam"), (7, "Please delete this")))
- reasoning = models.TextField(default='', null=True, blank=True)
- def __str__(self):
- return "Report on a " + self.get_type_display() + " for " + self.get_reason_display() + ": " + str(self.reasoning)
- # Login attempts: for incorrect passwords
- class LoginAttempt(models.Model):
- id = models.AutoField(primary_key=True)
- created = models.DateTimeField(auto_now_add=True)
- user = models.ForeignKey(User, on_delete=models.CASCADE)
- success = models.BooleanField(default=False)
- addr = models.CharField(max_length=64, null=True, blank=True)
- def __str__(self):
- return 'A login attempt to ' + str(self.user) + ' from ' + self.addr + ', ' + str(self.success)
- # Fun
- class ThermostatTouch(models.Model):
- id = models.AutoField(primary_key=True)
- created = models.DateTimeField(auto_now_add=True)
- who = models.ForeignKey(User, blank=True, null=True, on_delete=models.CASCADE)
- lvl = models.IntegerField(default=1)
- def __str__(self):
- return str(created) + " touched the thermostat, setting it to " + str(lvl) + " degrees celsius"
- # Finally
- class UserBlock(models.Model):
- id = models.AutoField(primary_key=True)
- created = models.DateTimeField(auto_now_add=True)
- source = models.ForeignKey(User, related_name='block_source', on_delete=models.CASCADE)
- target = models.ForeignKey(User, related_name='block_target', on_delete=models.CASCADE)
- full = models.BooleanField(default=False)
- def __str__(self):
- return "Block created from " + str(self.source) + " to " + str(self.target)
- @staticmethod
- def find_block(first, second, full=False):
- if full:
- return UserBlock.objects.filter(Q(source=first, full=True) & Q(target=second, full=True) | Q(target=first, full=True) & Q(source=second, full=True)).exists()
- return UserBlock.objects.filter(Q(source=first) & Q(target=second) | Q(target=first) & Q(source=second)).exists()
- class AuditLog(models.Model):
- id = models.AutoField(primary_key=True)
- created = models.DateTimeField(auto_now_add=True)
- type = models.SmallIntegerField(choices=((0, "Post delete"), (1, "Comment delete"), (2, "User edit"), (3, "Generate passwd reset"), (4, "User delete"), (5, "Image delete"), (6, "Purge 1"), (7, "Purge 2"), (8, "Purge 3"), (9, "Purge 4"), (10, "Purge 5"), (11, "Un-purge 1"), ))
- post = models.ForeignKey(Post, related_name='audit_post', null=True, on_delete=models.CASCADE)
- comment = models.ForeignKey(Comment, related_name='audit_comment', null=True, on_delete=models.CASCADE)
- user = models.ForeignKey(User, related_name='audit_user', null=True, on_delete=models.CASCADE)
- reasoning = models.TextField(null=True, blank=True, default="")
- by = models.ForeignKey(User, related_name='audit_by', on_delete=models.CASCADE)
- reversed_by = models.ForeignKey(User, null=True, related_name='audit_reverse_by', on_delete=models.CASCADE)
- def __str__(self):
- return str(self.by) + " did " + self.get_type_display() + " at " + str(self.created)
- def reverse(self, user=None):
- # Try to reverse what this did
- if user:
- self.reversed_by = user
- # No switches in Python, so
- if self.type == 0:
- self.post.is_rm = False
- self.post.status = 0
- self.post.save()
- return True
- elif self.type == 1:
- self.post.is_rm = False
- self.post.status = 0
- self.post.save()
- return True
- else:
- return False
- # TODO: MAKE THIS ACTUALLY WORK
- class Restriction(models.Model):
- id = models.AutoField(primary_key=True)
- created = models.DateTimeField(auto_now_add=True)
- type = models.SmallIntegerField(choices=((0, "Prevent yeah"), (1, "Prevent follow"), (2, "Prevent comment"), ))
- post = models.ForeignKey(Post, related_name='restriction_post', null=True, on_delete=models.CASCADE)
- comment = models.ForeignKey(Comment, related_name='restriction_comment', null=True, on_delete=models.CASCADE)
- user = models.ForeignKey(User, related_name='restriction_user', null=True, on_delete=models.CASCADE)
- by = models.ForeignKey(User, related_name='restriction_by', on_delete=models.CASCADE)
- def __str__(self):
- return "Restrict " + str(self.user) + " on " + self.get_type_display() + " at " + str(self.created)
- class UserRequest(User):
- # USER AGENT
- ua = models.TextField(default='', null=True, blank=True)
- latest = models.DateTimeField(auto_now=True)
- status = models.SmallIntegerField(default=0, choices=((0, 'submitted'), (1, 'viewed'), (2, 'accepted'), (3, 'decline'), (4, 'ignore'), ))
- # blah blah blah
- # this method will be executed when...
- def rm_post_image(sender, instance, **kwargs):
- if instance.screenshot:
- util.image_rm(instance.screenshot)
- if instance.drawing:
- util.image_rm(instance.drawing)
- # when pre_delete happens on these
- pre_delete.connect(rm_post_image, sender=Post)
- pre_delete.connect(rm_post_image, sender=Comment)
- pre_delete.connect(rm_post_image, sender=Message)
Add Comment
Please, Sign In to add comment