480 lines
17 KiB
Python
480 lines
17 KiB
Python
from taggit.utils import edit_string_for_tags, parse_tags
|
|
from tastypie.authentication import ApiKeyAuthentication
|
|
from tastypie.authorization import Authorization
|
|
from django.template.defaultfilters import date, urlize, linebreaksbr
|
|
from tastypie.resources import ModelResource
|
|
from tastypie.exceptions import Unauthorized
|
|
from django.contrib.auth.models import User
|
|
from tastypie.validation import Validation
|
|
from tastypie.models import create_api_key
|
|
from snipts.models import Favorite, Snipt
|
|
from haystack.query import SearchQuerySet
|
|
from accounts.models import UserProfile
|
|
from tastypie.cache import SimpleCache
|
|
from tastypie.fields import ListField
|
|
from taggit.models import Tag
|
|
from django.db import models
|
|
from tastypie import fields
|
|
|
|
import datetime, hashlib, time, re
|
|
|
|
import parsedatetime.parsedatetime as pdt
|
|
|
|
models.signals.post_save.connect(create_api_key, sender=User)
|
|
|
|
|
|
class PrivateFavoriteAuthorization(Authorization):
|
|
def read_list(self, object_list, bundle):
|
|
return object_list.filter(user=bundle.request.user)
|
|
|
|
def read_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
def create_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def create_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
def update_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
class PrivateSniptAuthorization(Authorization):
|
|
def read_list(self, object_list, bundle):
|
|
return object_list.filter(user=bundle.request.user)
|
|
|
|
def read_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
def create_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def create_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
def update_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
def delete_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
class PrivateTagAuthorization(Authorization):
|
|
def read_list(self, object_list, bundle):
|
|
object_list = object_list.filter(snipt__user=bundle.request.user)
|
|
object_list = object_list.annotate(count=models.Count('taggit_taggeditem_items__id'))
|
|
object_list = object_list.order_by('-count', 'name')
|
|
return object_list
|
|
|
|
def read_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def create_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def create_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
class PrivateUserProfileAuthorization(Authorization):
|
|
def read_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def read_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
def create_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def create_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_detail(self, object_list, bundle):
|
|
return bundle.obj.user == bundle.request.user
|
|
|
|
def delete_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
class PrivateUserAuthorization(Authorization):
|
|
def read_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def read_detail(self, object_list, bundle):
|
|
return bundle.obj == bundle.request.user
|
|
|
|
def create_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def create_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def update_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_list(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
def delete_detail(self, object_list, bundle):
|
|
raise Unauthorized()
|
|
|
|
|
|
class FavoriteValidation(Validation):
|
|
def is_valid(self, bundle, request=None):
|
|
errors = {}
|
|
snipt = bundle.data['snipt']
|
|
|
|
if Favorite.objects.filter(user=bundle.request.user, snipt=snipt).count():
|
|
errors['duplicate'] = 'User has already favorited this snipt.'
|
|
|
|
return errors
|
|
|
|
class UserProfileValidation(Validation):
|
|
def is_valid(self, bundle, request=None):
|
|
errors = {}
|
|
|
|
if not bundle.request.user.profile.is_pro:
|
|
return 'You must be a Pro to change these settings.'
|
|
|
|
for field in bundle.data:
|
|
if bundle.data[field]:
|
|
if not re.match('^[ A-Za-z0-9\/\@\._-]*$', bundle.data[field]):
|
|
errors[field] = 'Only spaces, letters, numbers, underscores, dashes, periods, forward slashes, and "at sign" are valid.'
|
|
|
|
return errors
|
|
|
|
|
|
class PublicUserResource(ModelResource):
|
|
class Meta:
|
|
queryset = User.objects.all()
|
|
resource_name = 'user'
|
|
fields = ['id', 'username',]
|
|
include_absolute_url = True
|
|
allowed_methods = ['get']
|
|
filtering = { 'username': 'exact', }
|
|
max_limit = 200
|
|
cache = SimpleCache()
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['snipts'] = '/api/public/snipt/?user=%d' % bundle.obj.id
|
|
return bundle
|
|
|
|
class PublicTagResource(ModelResource):
|
|
class Meta:
|
|
queryset = Tag.objects.filter(snipt__public=True)
|
|
queryset = queryset.annotate(count=models.Count('taggit_taggeditem_items__id'))
|
|
queryset = queryset.order_by('-count', 'name')
|
|
resource_name = 'tag'
|
|
fields = ['id', 'name',]
|
|
allowed_methods = ['get']
|
|
max_limit = 200
|
|
cache = SimpleCache()
|
|
|
|
def build_filters(self, filters=None):
|
|
if filters is None:
|
|
filters = {}
|
|
|
|
orm_filters = super(PublicTagResource, self).build_filters(filters)
|
|
|
|
if 'q' in filters:
|
|
orm_filters['slug'] = filters['q']
|
|
|
|
return orm_filters
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['absolute_url'] = '/public/tag/%s/' % bundle.obj.slug
|
|
bundle.data['snipts'] = '/api/public/snipt/?tag=%d' % bundle.obj.id
|
|
bundle.data['count'] = bundle.obj.taggit_taggeditem_items.filter(
|
|
snipt__public=True).count()
|
|
return bundle
|
|
|
|
class PublicSniptResource(ModelResource):
|
|
user = fields.ForeignKey(PublicUserResource, 'user', full=True)
|
|
tags = fields.ToManyField(PublicTagResource, 'tags', related_name='tag', full=True)
|
|
|
|
class Meta:
|
|
queryset = Snipt.objects.filter(public=True).order_by('-created')
|
|
resource_name = 'snipt'
|
|
fields = ['id', 'title', 'slug', 'lexer', 'code', 'description', 'line_count',
|
|
'stylized', 'created', 'modified',]
|
|
include_absolute_url = True
|
|
allowed_methods = ['get']
|
|
filtering = { 'user': 'exact', }
|
|
ordering = ['created', 'modified',]
|
|
max_limit = 200
|
|
cache = SimpleCache()
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['embed_url'] = bundle.obj.get_embed_url()
|
|
bundle.data['raw_url'] = bundle.obj.get_raw_url()
|
|
bundle.data['full_absolute_url'] = bundle.obj.get_full_absolute_url()
|
|
bundle.data['description_rendered'] = linebreaksbr(urlize(bundle.obj.description))
|
|
return bundle
|
|
|
|
def build_filters(self, filters=None):
|
|
if filters is None:
|
|
filters = {}
|
|
|
|
orm_filters = super(PublicSniptResource, self).build_filters(filters)
|
|
|
|
if 'tag' in filters:
|
|
tag = Tag.objects.get(pk=filters['tag'])
|
|
tagged_items = tag.taggit_taggeditem_items.all()
|
|
orm_filters['pk__in'] = [i.object_id for i in tagged_items]
|
|
|
|
if 'q' in filters:
|
|
sqs = SearchQuerySet().auto_query(filters['q'])
|
|
orm_filters['pk__in'] = [i.pk for i in sqs]
|
|
|
|
return orm_filters
|
|
|
|
|
|
class PrivateUserProfileResource(ModelResource):
|
|
class Meta:
|
|
queryset = UserProfile.objects.all()
|
|
resource_name = 'profile'
|
|
excludes = ['is_pro', 'stripe_id']
|
|
validation = UserProfileValidation()
|
|
include_absolute_url = False
|
|
allowed_methods = ['get', 'put']
|
|
list_allowed_methods = []
|
|
authentication = ApiKeyAuthentication()
|
|
authorization = PrivateUserProfileAuthorization()
|
|
always_return_data = True
|
|
max_limit = 200
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['email'] = bundle.obj.user.email
|
|
bundle.data['username'] = bundle.obj.user.username
|
|
bundle.data['user_id'] = bundle.obj.user.id
|
|
bundle.data['api_key'] = bundle.obj.user.api_key.key
|
|
bundle.data['is_pro'] = bundle.obj.user.profile.is_pro
|
|
return bundle
|
|
|
|
class PrivateUserResource(ModelResource):
|
|
profile = fields.ForeignKey(PrivateUserProfileResource, 'profile', full=False)
|
|
|
|
class Meta:
|
|
queryset = User.objects.all()
|
|
resource_name = 'user'
|
|
fields = ['id', 'username', 'email']
|
|
include_absolute_url = True
|
|
allowed_methods = ['get']
|
|
list_allowed_methods = []
|
|
authentication = ApiKeyAuthentication()
|
|
authorization = PrivateUserAuthorization()
|
|
always_return_data = True
|
|
max_limit = 200
|
|
cache = SimpleCache()
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['email_md5'] = hashlib.md5(bundle.obj.email.lower()).hexdigest()
|
|
bundle.data['is_pro'] = bundle.obj.profile.is_pro
|
|
bundle.data['lexers'] = [snipt['lexer'] for snipt in \
|
|
Snipt.objects.filter(user=bundle.obj).values('lexer').distinct()]
|
|
return bundle
|
|
|
|
class PrivateTagResource(ModelResource):
|
|
class Meta:
|
|
queryset = Tag.objects.all()
|
|
resource_name = 'tag'
|
|
fields = ['id', 'name',]
|
|
allowed_methods = ['get']
|
|
authentication = ApiKeyAuthentication()
|
|
authorization = PrivateTagAuthorization()
|
|
always_return_data = True
|
|
max_limit = 200
|
|
cache = SimpleCache()
|
|
|
|
def build_filters(self, filters=None):
|
|
if filters is None:
|
|
filters = {}
|
|
|
|
orm_filters = super(PrivateTagResource, self).build_filters(filters)
|
|
|
|
if 'q' in filters:
|
|
orm_filters['slug'] = filters['q']
|
|
|
|
return orm_filters
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['absolute_url'] = '/%s/tag/%s/' % (bundle.request.user.username,
|
|
bundle.obj.slug)
|
|
bundle.data['snipts'] = '/api/private/snipt/?tag=%d' % bundle.obj.id
|
|
|
|
bundle.data['count'] = bundle.obj.taggit_taggeditem_items.filter(
|
|
snipt__user=bundle.request.user
|
|
).count()
|
|
|
|
return bundle
|
|
|
|
class PrivateSniptResource(ModelResource):
|
|
user = fields.ForeignKey(PrivateUserResource, 'user', full=True)
|
|
tags = fields.ToManyField(PrivateTagResource, 'tags', related_name='tag', full=True)
|
|
tags_list = ListField()
|
|
|
|
class Meta:
|
|
queryset = Snipt.objects.all().order_by('-created')
|
|
resource_name = 'snipt'
|
|
fields = ['id', 'title', 'slug', 'lexer', 'code', 'description', 'line_count', 'stylized',
|
|
'key', 'public', 'blog_post', 'created', 'modified', 'publish_date',]
|
|
include_absolute_url = True
|
|
detail_allowed_methods = ['get', 'patch', 'put', 'delete']
|
|
list_allowed_methods = ['get', 'post']
|
|
authentication = ApiKeyAuthentication()
|
|
authorization = PrivateSniptAuthorization()
|
|
ordering = ['created', 'modified',]
|
|
always_return_data = True
|
|
max_limit = 200
|
|
cache = SimpleCache()
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['embed_url'] = bundle.obj.get_embed_url()
|
|
bundle.data['raw_url'] = bundle.obj.get_raw_url()
|
|
bundle.data['tags_list'] = edit_string_for_tags(bundle.obj.tags.all())
|
|
bundle.data['full_absolute_url'] = bundle.obj.get_full_absolute_url()
|
|
bundle.data['description_rendered'] = linebreaksbr(urlize(bundle.obj.description))
|
|
|
|
if bundle.data['user'].data['is_pro']:
|
|
bundle.data['views'] = bundle.obj.views
|
|
bundle.data['favs'] = bundle.obj.favs()
|
|
|
|
if bundle.data['publish_date']:
|
|
bundle.data['publish_date'] = date(bundle.data['publish_date'], 'M d, Y \\a\\t h:i A')
|
|
|
|
return bundle
|
|
|
|
def obj_create(self, bundle, **kwargs):
|
|
bundle.data['tags_list'] = bundle.data.get('tags')
|
|
bundle.data['tags'] = ''
|
|
|
|
if 'blog_post' in bundle.data:
|
|
bundle = self._clean_publish_date(bundle)
|
|
|
|
return super(PrivateSniptResource, self).obj_create(bundle,
|
|
user=bundle.request.user, **kwargs)
|
|
|
|
def obj_update(self, bundle, **kwargs):
|
|
bundle.data['user'] = bundle.request.user
|
|
bundle.data['created'] = None
|
|
bundle.data['modified'] = None
|
|
|
|
if type(bundle.data['tags']) in (str, unicode):
|
|
bundle.data['tags_list'] = bundle.data['tags']
|
|
else:
|
|
bundle.data['tags_list'] = ''
|
|
bundle.data['tags'] = ''
|
|
|
|
if 'blog_post' in bundle.data:
|
|
bundle = self._clean_publish_date(bundle)
|
|
|
|
return super(PrivateSniptResource, self).obj_update(bundle,
|
|
user=bundle.request.user, **kwargs)
|
|
|
|
def _clean_publish_date(self, bundle):
|
|
if bundle.data['blog_post'] and 'publish_date' not in bundle.data:
|
|
bundle.data['publish_date'] = datetime.datetime.now()
|
|
elif bundle.data['publish_date'] == '':
|
|
bundle.data['publish_date'] = datetime.datetime.now()
|
|
elif bundle.data['blog_post']:
|
|
c = pdt.Constants()
|
|
p = pdt.Calendar(c)
|
|
publish_date, result = p.parse(bundle.data['publish_date'])
|
|
|
|
if result != 0:
|
|
publish_date = time.strftime("%Y-%m-%d %H:%M:%S", publish_date)
|
|
else:
|
|
publish_date = datetime.datetime.now()
|
|
|
|
bundle.data['publish_date'] = publish_date
|
|
elif not bundle.data['blog_post']:
|
|
bundle.data['publish_date'] = None
|
|
|
|
return bundle
|
|
|
|
def build_filters(self, filters=None):
|
|
if filters is None:
|
|
filters = {}
|
|
|
|
orm_filters = super(PrivateSniptResource, self).build_filters(filters)
|
|
|
|
if 'tag' in filters:
|
|
tag = Tag.objects.get(pk=filters['tag'])
|
|
tagged_items = tag.taggit_taggeditem_items.all()
|
|
orm_filters['pk__in'] = [i.object_id for i in tagged_items]
|
|
|
|
if 'q' in filters:
|
|
sqs = SearchQuerySet().auto_query(filters['q'])
|
|
orm_filters['pk__in'] = [i.pk for i in sqs]
|
|
|
|
return orm_filters
|
|
|
|
def save_m2m(self, bundle):
|
|
tags = bundle.data.get('tags_list', [])
|
|
if tags != '':
|
|
bundle.obj.tags.set(*parse_tags(tags))
|
|
else:
|
|
bundle.obj.tags.set()
|
|
|
|
class PrivateFavoriteResource(ModelResource):
|
|
user = fields.ForeignKey(PrivateUserResource, 'user', full=True)
|
|
snipt = fields.ForeignKey(PrivateSniptResource, 'snipt', full=False)
|
|
|
|
class Meta:
|
|
queryset = Favorite.objects.all().order_by('-created')
|
|
resource_name = 'favorite'
|
|
fields = ['id',]
|
|
validation = FavoriteValidation()
|
|
detail_allowed_methods = ['get', 'post', 'delete']
|
|
list_allowed_methods = ['get', 'post',]
|
|
authentication = ApiKeyAuthentication()
|
|
authorization = PrivateFavoriteAuthorization()
|
|
ordering = ['created',]
|
|
always_return_data = True
|
|
max_limit = 200
|
|
cache = SimpleCache()
|
|
|
|
def dehydrate(self, bundle):
|
|
bundle.data['snipt'] = '/api/public/snipt/{}/'.format(
|
|
bundle.obj.snipt.pk)
|
|
return bundle
|
|
|
|
def obj_create(self, bundle, **kwargs):
|
|
bundle.data['user'] = bundle.request.user
|
|
bundle.data['snipt'] = Snipt.objects.get(pk=bundle.data['snipt'])
|
|
return super(PrivateFavoriteResource, self).obj_create(bundle,
|
|
user=bundle.request.user, **kwargs)
|