2011-12-21 09:43:17 -08:00
|
|
|
import datetime
|
2011-12-14 09:52:06 -08:00
|
|
|
import zlib
|
2011-12-22 13:36:03 -08:00
|
|
|
import hashlib
|
2012-01-09 13:55:26 -08:00
|
|
|
import redis
|
2011-12-14 09:52:06 -08:00
|
|
|
import mongoengine as mongo
|
2012-01-31 10:15:11 -08:00
|
|
|
# from mongoengine.queryset import OperationError
|
2011-12-21 09:43:17 -08:00
|
|
|
from django.conf import settings
|
2011-12-22 13:36:03 -08:00
|
|
|
from django.contrib.auth.models import User
|
2012-01-31 10:15:11 -08:00
|
|
|
from django.core.urlresolvers import reverse
|
2012-01-30 08:53:19 -08:00
|
|
|
from apps.reader.models import UserSubscription, MUserStory
|
2011-12-20 11:49:49 -08:00
|
|
|
from vendor import facebook
|
|
|
|
from vendor import tweepy
|
2011-12-24 00:30:37 -08:00
|
|
|
from utils import log as logging
|
2012-01-10 10:20:49 -08:00
|
|
|
from utils.feed_functions import relative_timesince
|
2011-12-14 09:52:06 -08:00
|
|
|
|
2011-12-20 11:49:49 -08:00
|
|
|
|
2012-01-04 09:42:35 -08:00
|
|
|
class MSocialProfile(mongo.Document):
|
|
|
|
user_id = mongo.IntField()
|
|
|
|
username = mongo.StringField(max_length=30)
|
|
|
|
email = mongo.StringField()
|
|
|
|
bio = mongo.StringField(max_length=80)
|
2012-01-31 10:15:11 -08:00
|
|
|
blog_title = mongo.StringField(max_length=256)
|
|
|
|
custom_css = mongo.StringField()
|
2012-01-04 09:42:35 -08:00
|
|
|
photo_url = mongo.StringField()
|
|
|
|
photo_service = mongo.StringField()
|
|
|
|
location = mongo.StringField(max_length=40)
|
|
|
|
website = mongo.StringField(max_length=200)
|
|
|
|
subscription_count = mongo.IntField(default=0)
|
|
|
|
shared_stories_count = mongo.IntField(default=0)
|
|
|
|
following_count = mongo.IntField(default=0)
|
|
|
|
follower_count = mongo.IntField(default=0)
|
|
|
|
following_user_ids = mongo.ListField(mongo.IntField())
|
|
|
|
follower_user_ids = mongo.ListField(mongo.IntField())
|
|
|
|
unfollowed_user_ids = mongo.ListField(mongo.IntField())
|
2012-01-20 09:38:41 -08:00
|
|
|
stories_last_month = mongo.IntField(default=0)
|
|
|
|
average_stories_per_month = mongo.IntField(default=0)
|
|
|
|
favicon_color = mongo.StringField(max_length=6)
|
2012-01-04 09:42:35 -08:00
|
|
|
|
|
|
|
meta = {
|
|
|
|
'collection': 'social_profile',
|
|
|
|
'indexes': ['user_id', 'following_user_ids', 'follower_user_ids', 'unfollowed_user_ids'],
|
|
|
|
'allow_inheritance': False,
|
|
|
|
}
|
|
|
|
|
|
|
|
def __unicode__(self):
|
|
|
|
return "%s [%s] %s/%s" % (self.username, self.user_id,
|
|
|
|
self.subscription_count, self.shared_stories_count)
|
|
|
|
|
|
|
|
def save(self, *args, **kwargs):
|
|
|
|
if not self.username:
|
|
|
|
self.update_user(skip_save=True)
|
|
|
|
if not self.subscription_count:
|
|
|
|
self.count(skip_save=True)
|
|
|
|
super(MSocialProfile, self).save(*args, **kwargs)
|
|
|
|
|
2012-01-20 09:38:41 -08:00
|
|
|
@classmethod
|
|
|
|
def user_statistics(cls, user):
|
|
|
|
try:
|
|
|
|
profile = cls.objects.get(user_id=user.pk)
|
|
|
|
except cls.DoesNotExist:
|
|
|
|
return None
|
|
|
|
|
|
|
|
values = {
|
|
|
|
'followers': profile.follower_count,
|
|
|
|
'following': profile.following_count,
|
|
|
|
'shared_stories': profile.shared_stories_count,
|
|
|
|
}
|
|
|
|
return values
|
|
|
|
|
2012-01-04 09:42:35 -08:00
|
|
|
@classmethod
|
|
|
|
def profiles(cls, user_ids):
|
|
|
|
profiles = cls.objects.filter(user_id__in=user_ids)
|
|
|
|
return profiles
|
2012-01-20 09:38:41 -08:00
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def profile_feeds(cls, user_ids):
|
|
|
|
profiles = cls.objects.filter(user_id__in=user_ids, shared_stories_count__gte=1)
|
2012-01-21 16:12:54 -08:00
|
|
|
profiles = dict((p.user_id, p.feed()) for p in profiles)
|
2012-01-20 09:38:41 -08:00
|
|
|
return profiles
|
2012-01-04 09:42:35 -08:00
|
|
|
|
2012-01-10 10:20:49 -08:00
|
|
|
@classmethod
|
2012-01-20 09:38:41 -08:00
|
|
|
def sync_all_redis(cls):
|
2012-01-10 10:20:49 -08:00
|
|
|
r = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
for profile in cls.objects.all():
|
2012-01-20 09:38:41 -08:00
|
|
|
profile.sync_redis(redis_conn=r)
|
|
|
|
|
|
|
|
def sync_redis(self, redis_conn=None):
|
|
|
|
if not redis_conn:
|
|
|
|
redis_conn = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
|
|
|
|
for user_id in self.following_user_ids:
|
|
|
|
following_key = "F:%s:F" % (self.user_id)
|
|
|
|
redis_conn.sadd(following_key, user_id)
|
|
|
|
follower_key = "F:%s:f" % (user_id)
|
|
|
|
redis_conn.sadd(follower_key, self.user_id)
|
2012-01-31 10:15:11 -08:00
|
|
|
|
|
|
|
@property
|
|
|
|
def title(self):
|
|
|
|
return self.blog_title if self.blog_title else self.username + "'s blurblog"
|
|
|
|
|
2012-01-21 16:12:54 -08:00
|
|
|
def feed(self):
|
|
|
|
params = self.to_json(compact=True)
|
|
|
|
params.update({
|
2012-01-31 10:15:11 -08:00
|
|
|
'feed_title': self.title,
|
|
|
|
'page_url': reverse('load-social-page', kwargs={'user_id': self.user_id, 'username': self.username})
|
|
|
|
})
|
|
|
|
return params
|
|
|
|
|
|
|
|
def page(self):
|
|
|
|
params = self.to_json(full=True)
|
|
|
|
params.update({
|
|
|
|
'feed_title': self.title,
|
|
|
|
'custom_css': self.custom_css,
|
2012-01-21 16:12:54 -08:00
|
|
|
})
|
|
|
|
return params
|
|
|
|
|
2012-01-13 09:20:37 -08:00
|
|
|
def to_json(self, compact=False, full=False):
|
|
|
|
if compact:
|
|
|
|
params = {
|
2012-01-26 08:43:46 -08:00
|
|
|
'id': 'social:%s' % self.user_id,
|
2012-01-13 09:20:37 -08:00
|
|
|
'user_id': self.user_id,
|
|
|
|
'username': self.username,
|
|
|
|
'photo_url': self.photo_url
|
|
|
|
}
|
|
|
|
else:
|
|
|
|
params = {
|
2012-01-26 08:43:46 -08:00
|
|
|
'id': 'social:%s' % self.user_id,
|
2012-01-13 09:20:37 -08:00
|
|
|
'user_id': self.user_id,
|
|
|
|
'username': self.username,
|
|
|
|
'photo_url': self.photo_url,
|
|
|
|
'bio': self.bio,
|
|
|
|
'location': self.location,
|
|
|
|
'website': self.website,
|
|
|
|
'subscription_count': self.subscription_count,
|
|
|
|
'shared_stories_count': self.shared_stories_count,
|
|
|
|
'following_count': self.following_count,
|
|
|
|
'follower_count': self.follower_count,
|
|
|
|
}
|
2012-01-04 09:42:35 -08:00
|
|
|
if full:
|
|
|
|
params['photo_service'] = self.photo_service
|
|
|
|
params['following_user_ids'] = self.following_user_ids
|
|
|
|
params['follower_user_ids'] = self.follower_user_ids
|
|
|
|
params['unfollowed_user_ids'] = self.unfollowed_user_ids
|
|
|
|
return params
|
|
|
|
|
|
|
|
def update_user(self, skip_save=False):
|
|
|
|
user = User.objects.get(pk=self.user_id)
|
|
|
|
self.username = user.username
|
|
|
|
self.email = user.email
|
|
|
|
if not skip_save:
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
def count(self, skip_save=False):
|
|
|
|
self.subscription_count = UserSubscription.objects.filter(user__pk=self.user_id).count()
|
|
|
|
self.shared_stories_count = MSharedStory.objects.filter(user_id=self.user_id).count()
|
|
|
|
self.following_count = len(self.following_user_ids)
|
|
|
|
self.follower_count = len(self.follower_user_ids)
|
|
|
|
if not skip_save:
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
def follow_user(self, user_id, check_unfollowed=False):
|
2012-01-09 13:55:26 -08:00
|
|
|
r = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
|
2012-01-04 09:42:35 -08:00
|
|
|
if check_unfollowed and user_id in self.unfollowed_user_ids:
|
|
|
|
return
|
|
|
|
|
|
|
|
if user_id not in self.following_user_ids:
|
|
|
|
self.following_user_ids.append(user_id)
|
|
|
|
if user_id in self.unfollowed_user_ids:
|
|
|
|
self.unfollowed_user_ids.remove(user_id)
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
followee, _ = MSocialProfile.objects.get_or_create(user_id=user_id)
|
|
|
|
if self.user_id not in followee.follower_user_ids:
|
|
|
|
followee.follower_user_ids.append(self.user_id)
|
2012-01-05 20:54:24 -08:00
|
|
|
followee.count()
|
2012-01-04 09:42:35 -08:00
|
|
|
followee.save()
|
|
|
|
self.count()
|
2012-01-09 13:55:26 -08:00
|
|
|
|
|
|
|
following_key = "F:%s:F" % (self.user_id)
|
|
|
|
r.sadd(following_key, user_id)
|
|
|
|
follower_key = "F:%s:f" % (user_id)
|
|
|
|
r.sadd(follower_key, self.user_id)
|
2012-01-20 09:38:41 -08:00
|
|
|
|
|
|
|
MSocialSubscription.objects.create(user_id=self.user_id, subscription_user_id=user_id)
|
2012-01-04 09:42:35 -08:00
|
|
|
|
|
|
|
def unfollow_user(self, user_id):
|
2012-01-09 13:55:26 -08:00
|
|
|
r = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
|
2012-01-04 09:42:35 -08:00
|
|
|
if user_id in self.following_user_ids:
|
|
|
|
self.following_user_ids.remove(user_id)
|
|
|
|
if user_id not in self.unfollowed_user_ids:
|
|
|
|
self.unfollowed_user_ids.append(user_id)
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
followee = MSocialProfile.objects.get(user_id=user_id)
|
|
|
|
if self.user_id in followee.follower_user_ids:
|
|
|
|
followee.follower_user_ids.remove(self.user_id)
|
2012-01-05 20:54:24 -08:00
|
|
|
followee.count()
|
2012-01-04 09:42:35 -08:00
|
|
|
followee.save()
|
|
|
|
self.count()
|
|
|
|
|
2012-01-09 13:55:26 -08:00
|
|
|
following_key = "F:%s:F" % (self.user_id)
|
|
|
|
r.srem(following_key, user_id)
|
|
|
|
follower_key = "F:%s:f" % (user_id)
|
|
|
|
r.srem(follower_key, self.user_id)
|
2012-01-20 09:38:41 -08:00
|
|
|
|
|
|
|
MSocialSubscription.objects.filter(user_id=self.user_id, subscription_user_id=user_id).delete()
|
|
|
|
|
|
|
|
|
|
|
|
class MSocialSubscription(mongo.Document):
|
|
|
|
UNREAD_CUTOFF = datetime.datetime.utcnow() - datetime.timedelta(days=settings.DAYS_OF_UNREAD)
|
2012-01-04 09:42:35 -08:00
|
|
|
|
2012-01-20 09:38:41 -08:00
|
|
|
user_id = mongo.IntField()
|
|
|
|
subscription_user_id = mongo.IntField(unique_with='user_id')
|
|
|
|
last_read_date = mongo.DateTimeField(default=UNREAD_CUTOFF)
|
|
|
|
mark_read_date = mongo.DateTimeField(default=UNREAD_CUTOFF)
|
|
|
|
unread_count_neutral = mongo.IntField(default=0)
|
|
|
|
unread_count_positive = mongo.IntField(default=0)
|
|
|
|
unread_count_negative = mongo.IntField(default=0)
|
|
|
|
unread_count_updated = mongo.DateTimeField()
|
|
|
|
oldest_unread_story_date = mongo.DateTimeField()
|
|
|
|
needs_unread_recalc = mongo.BooleanField(default=False)
|
|
|
|
feed_opens = mongo.IntField(default=0)
|
|
|
|
is_trained = mongo.BooleanField(default=False)
|
|
|
|
|
|
|
|
meta = {
|
|
|
|
'collection': 'social_subscription',
|
|
|
|
'indexes': [('user_id', 'subscription_user_id')],
|
|
|
|
'allow_inheritance': False,
|
|
|
|
}
|
|
|
|
|
|
|
|
def __unicode__(self):
|
|
|
|
return "%s:%s" % (self.user_id, self.subscription_user_id)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def feeds(cls, *args, **kwargs):
|
|
|
|
user_id = kwargs['user_id']
|
|
|
|
social_subs = cls.objects.filter(user_id=user_id)
|
2012-01-26 09:54:48 -08:00
|
|
|
social_feeds = []
|
|
|
|
if social_subs:
|
|
|
|
social_subs = dict((s.subscription_user_id, s.to_json()) for s in social_subs)
|
|
|
|
social_user_ids = social_subs.keys()
|
|
|
|
social_profiles = MSocialProfile.profile_feeds(social_user_ids)
|
|
|
|
social_feeds = {}
|
|
|
|
for user_id, social_sub in social_subs.items():
|
|
|
|
print social_profiles[user_id]['id'], social_sub, social_profiles[user_id]
|
|
|
|
social_feeds[social_profiles[user_id]['id']] = dict(social_sub.items() + social_profiles[user_id].items())
|
2012-01-21 16:12:54 -08:00
|
|
|
|
|
|
|
return social_feeds
|
|
|
|
|
|
|
|
def to_json(self):
|
|
|
|
return {
|
|
|
|
'user_id': self.user_id,
|
|
|
|
'subscription_user_id': self.subscription_user_id,
|
|
|
|
'nt': self.unread_count_neutral + 2,
|
|
|
|
'ps': self.unread_count_positive + 3,
|
2012-02-02 17:43:17 -08:00
|
|
|
'ng': self.unread_count_negative + 1,
|
2012-01-21 16:12:54 -08:00
|
|
|
'is_trained': self.is_trained,
|
|
|
|
}
|
2012-01-20 09:38:41 -08:00
|
|
|
|
2012-02-03 11:41:01 -08:00
|
|
|
def mark_story_ids_as_read(self, story_ids, feed_id, request=None):
|
2012-01-30 08:53:19 -08:00
|
|
|
data = dict(code=0, payload=story_ids)
|
|
|
|
|
|
|
|
if not request:
|
|
|
|
request = self.user
|
|
|
|
|
|
|
|
if not self.needs_unread_recalc:
|
|
|
|
self.needs_unread_recalc = True
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
if len(story_ids) > 1:
|
|
|
|
logging.user(request, "~FYRead %s stories in social subscription: %s" % (len(story_ids), self.subscription_user_id))
|
|
|
|
else:
|
|
|
|
logging.user(request, "~FYRead story in social subscription: %s" % (self.subscription_user_id))
|
|
|
|
|
|
|
|
for story_id in set(story_ids):
|
|
|
|
story = MSharedStory.objects.get(user_id=self.subscription_user_id, story_guid=story_id)
|
|
|
|
now = datetime.datetime.utcnow()
|
|
|
|
date = now if now > story.story_date else story.story_date # For handling future stories
|
2012-02-03 11:41:01 -08:00
|
|
|
m = MUserStory(user_id=self.user_id,
|
|
|
|
feed_id=feed_id, read_date=date,
|
2012-01-30 08:53:19 -08:00
|
|
|
story_id=story_id, story_date=story.story_date)
|
|
|
|
m.save()
|
|
|
|
|
|
|
|
return data
|
|
|
|
|
2012-01-20 09:38:41 -08:00
|
|
|
|
2012-02-02 17:43:17 -08:00
|
|
|
class MSharedStory(mongo.Document):
|
|
|
|
user_id = mongo.IntField()
|
|
|
|
shared_date = mongo.DateTimeField()
|
|
|
|
comments = mongo.StringField()
|
|
|
|
has_comments = mongo.BooleanField(default=False)
|
|
|
|
story_feed_id = mongo.IntField()
|
|
|
|
story_date = mongo.DateTimeField()
|
|
|
|
story_title = mongo.StringField(max_length=1024)
|
|
|
|
story_content = mongo.StringField()
|
|
|
|
story_content_z = mongo.BinaryField()
|
|
|
|
story_original_content = mongo.StringField()
|
|
|
|
story_original_content_z = mongo.BinaryField()
|
|
|
|
story_content_type = mongo.StringField(max_length=255)
|
|
|
|
story_author_name = mongo.StringField()
|
|
|
|
story_permalink = mongo.StringField()
|
|
|
|
story_guid = mongo.StringField(unique_with=('user_id',))
|
|
|
|
story_tags = mongo.ListField(mongo.StringField(max_length=250))
|
|
|
|
|
|
|
|
meta = {
|
|
|
|
'collection': 'shared_stories',
|
|
|
|
'indexes': [('user_id', '-shared_date'), ('user_id', 'story_feed_id'), 'story_feed_id'],
|
|
|
|
'index_drop_dups': True,
|
|
|
|
'ordering': ['shared_date'],
|
|
|
|
'allow_inheritance': False,
|
|
|
|
}
|
|
|
|
|
|
|
|
@property
|
|
|
|
def guid_hash(self):
|
|
|
|
return hashlib.sha1(self.story_guid).hexdigest()
|
|
|
|
|
|
|
|
def save(self, *args, **kwargs):
|
|
|
|
if self.story_content:
|
|
|
|
self.story_content_z = zlib.compress(self.story_content)
|
|
|
|
self.story_content = None
|
|
|
|
if self.story_original_content:
|
|
|
|
self.story_original_content_z = zlib.compress(self.story_original_content)
|
|
|
|
self.story_original_content = None
|
|
|
|
|
|
|
|
r = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
share_key = "S:%s:%s" % (self.story_feed_id, self.guid_hash)
|
|
|
|
r.sadd(share_key, self.user_id)
|
|
|
|
comment_key = "C:%s:%s" % (self.story_feed_id, self.guid_hash)
|
|
|
|
if self.has_comments:
|
|
|
|
r.sadd(comment_key, self.user_id)
|
|
|
|
else:
|
|
|
|
r.srem(comment_key, self.user_id)
|
|
|
|
|
|
|
|
self.shared_date = datetime.datetime.utcnow()
|
|
|
|
|
|
|
|
super(MSharedStory, self).save(*args, **kwargs)
|
|
|
|
|
|
|
|
author, _ = MSocialProfile.objects.get_or_create(user_id=self.user_id)
|
|
|
|
author.count()
|
|
|
|
|
|
|
|
def delete(self, *args, **kwargs):
|
|
|
|
r = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
share_key = "S:%s:%s" % (self.story_feed_id, self.guid_hash)
|
|
|
|
r.srem(share_key, self.user_id)
|
|
|
|
|
|
|
|
super(MSharedStory, self).delete(*args, **kwargs)
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def sync_all_redis(cls):
|
|
|
|
r = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
for story in cls.objects.all():
|
|
|
|
story.sync_redis(redis_conn=r)
|
|
|
|
|
|
|
|
def sync_redis(self, redis_conn=None):
|
|
|
|
if not redis_conn:
|
|
|
|
redis_conn = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
|
|
|
|
share_key = "S:%s:%s" % (self.story_feed_id, self.guid_hash)
|
|
|
|
comment_key = "C:%s:%s" % (self.story_feed_id, self.guid_hash)
|
|
|
|
redis_conn.sadd(share_key, self.user_id)
|
|
|
|
if self.has_comments:
|
|
|
|
redis_conn.sadd(comment_key, self.user_id)
|
|
|
|
else:
|
|
|
|
redis_conn.srem(comment_key, self.user_id)
|
|
|
|
|
|
|
|
def comments_with_author(self, compact=False, full=False):
|
|
|
|
comments = {
|
|
|
|
'user_id': self.user_id,
|
|
|
|
'comments': self.comments,
|
|
|
|
'shared_date': relative_timesince(self.shared_date),
|
|
|
|
}
|
|
|
|
if full or compact:
|
|
|
|
author = MSocialProfile.objects.get(user_id=self.user_id)
|
|
|
|
comments['author'] = author.to_json(compact=compact, full=full)
|
|
|
|
return comments
|
|
|
|
|
|
|
|
@classmethod
|
|
|
|
def stories_with_comments(cls, stories, user, check_all=False):
|
|
|
|
r = redis.Redis(connection_pool=settings.REDIS_POOL)
|
|
|
|
friend_key = "F:%s:F" % (user.pk)
|
|
|
|
for story in stories:
|
|
|
|
if check_all or story['comment_count']:
|
|
|
|
comment_key = "C:%s:%s" % (story['story_feed_id'], story['guid_hash'])
|
|
|
|
if check_all:
|
|
|
|
story['comment_count'] = r.scard(comment_key)
|
|
|
|
friends_with_comments = r.sinter(comment_key, friend_key)
|
|
|
|
shared_stories = []
|
|
|
|
if friends_with_comments:
|
|
|
|
params = {
|
|
|
|
'story_guid': story['id'],
|
|
|
|
'story_feed_id': story['story_feed_id'],
|
|
|
|
'user_id__in': friends_with_comments,
|
|
|
|
}
|
|
|
|
shared_stories = cls.objects.filter(**params)
|
|
|
|
story['comments'] = []
|
|
|
|
for shared_story in shared_stories:
|
|
|
|
story['comments'].append(shared_story.comments_with_author(compact=True))
|
|
|
|
story['comment_count_public'] = story['comment_count'] - len(shared_stories)
|
|
|
|
story['comment_count_friends'] = len(shared_stories)
|
|
|
|
|
|
|
|
if check_all or story['share_count']:
|
|
|
|
share_key = "S:%s:%s" % (story['story_feed_id'], story['guid_hash'])
|
|
|
|
if check_all:
|
|
|
|
story['share_count'] = r.scard(share_key)
|
|
|
|
friends_with_shares = [int(f) for f in r.sinter(share_key, friend_key)]
|
|
|
|
nonfriend_user_ids = list(set(story['share_user_ids']).difference(friends_with_shares))
|
|
|
|
profiles = MSocialProfile.objects.filter(user_id__in=nonfriend_user_ids)
|
|
|
|
friend_profiles = []
|
|
|
|
if friends_with_shares:
|
|
|
|
friend_profiles = MSocialProfile.objects.filter(user_id__in=friends_with_shares)
|
|
|
|
story['shared_by_public'] = []
|
|
|
|
story['shared_by_friends'] = []
|
|
|
|
for profile in profiles:
|
|
|
|
story['shared_by_public'].append(profile.to_json(compact=True))
|
|
|
|
for profile in friend_profiles:
|
|
|
|
story['shared_by_friends'].append(profile.to_json(compact=True))
|
|
|
|
story['share_count_public'] = story['share_count'] - len(friend_profiles)
|
|
|
|
story['share_count_friends'] = len(friend_profiles)
|
|
|
|
|
|
|
|
return stories
|
|
|
|
|
|
|
|
|
2011-12-20 11:49:49 -08:00
|
|
|
class MSocialServices(mongo.Document):
|
|
|
|
user_id = mongo.IntField()
|
2011-12-22 13:36:03 -08:00
|
|
|
autofollow = mongo.BooleanField(default=True)
|
2011-12-20 11:49:49 -08:00
|
|
|
twitter_uid = mongo.StringField()
|
|
|
|
twitter_access_key = mongo.StringField()
|
|
|
|
twitter_access_secret = mongo.StringField()
|
|
|
|
twitter_friend_ids = mongo.ListField(mongo.StringField())
|
|
|
|
twitter_picture_url = mongo.StringField()
|
|
|
|
twitter_username = mongo.StringField()
|
|
|
|
twitter_refresh_date = mongo.DateTimeField()
|
|
|
|
facebook_uid = mongo.StringField()
|
|
|
|
facebook_access_token = mongo.StringField()
|
|
|
|
facebook_friend_ids = mongo.ListField(mongo.StringField())
|
|
|
|
facebook_picture_url = mongo.StringField()
|
|
|
|
facebook_refresh_date = mongo.DateTimeField()
|
2011-12-24 00:30:37 -08:00
|
|
|
upload_picture_url = mongo.StringField()
|
2011-12-20 11:49:49 -08:00
|
|
|
|
|
|
|
meta = {
|
|
|
|
'collection': 'social_services',
|
|
|
|
'indexes': ['user_id', 'twitter_friend_ids', 'facebook_friend_ids', 'twitter_uid', 'facebook_uid'],
|
|
|
|
'allow_inheritance': False,
|
|
|
|
}
|
2011-12-25 20:50:59 -08:00
|
|
|
|
|
|
|
def __unicode__(self):
|
|
|
|
return "%s" % self.user_id
|
2011-12-22 13:36:03 -08:00
|
|
|
|
|
|
|
def to_json(self):
|
|
|
|
user = User.objects.get(pk=self.user_id)
|
|
|
|
return {
|
2011-12-25 12:27:05 -08:00
|
|
|
'twitter': {
|
|
|
|
'twitter_username': self.twitter_username,
|
|
|
|
'twitter_picture_url': self.twitter_picture_url,
|
|
|
|
'twitter_uid': self.twitter_uid,
|
|
|
|
},
|
|
|
|
'facebook': {
|
|
|
|
'facebook_uid': self.facebook_uid,
|
|
|
|
'facebook_picture_url': self.facebook_picture_url,
|
|
|
|
},
|
|
|
|
'gravatar': {
|
|
|
|
'gravatar_picture_url': "http://www.gravatar.com/avatar/" + \
|
|
|
|
hashlib.md5(user.email).hexdigest()
|
|
|
|
},
|
|
|
|
'upload': {
|
|
|
|
'upload_picture_url': self.upload_picture_url
|
2011-12-22 13:36:03 -08:00
|
|
|
}
|
|
|
|
}
|
2011-12-20 11:49:49 -08:00
|
|
|
|
2011-12-21 09:43:17 -08:00
|
|
|
def twitter_api(self):
|
|
|
|
twitter_consumer_key = settings.TWITTER_CONSUMER_KEY
|
|
|
|
twitter_consumer_secret = settings.TWITTER_CONSUMER_SECRET
|
|
|
|
auth = tweepy.OAuthHandler(twitter_consumer_key, twitter_consumer_secret)
|
|
|
|
auth.set_access_token(self.twitter_access_key, self.twitter_access_secret)
|
|
|
|
api = tweepy.API(auth)
|
|
|
|
return api
|
|
|
|
|
|
|
|
def facebook_api(self):
|
2011-12-20 11:49:49 -08:00
|
|
|
graph = facebook.GraphAPI(self.facebook_access_token)
|
2011-12-21 09:43:17 -08:00
|
|
|
return graph
|
|
|
|
|
|
|
|
def sync_twitter_friends(self):
|
|
|
|
api = self.twitter_api()
|
|
|
|
if not api:
|
|
|
|
return
|
|
|
|
|
|
|
|
friend_ids = list(unicode(friend.id) for friend in tweepy.Cursor(api.friends).items())
|
|
|
|
if not friend_ids:
|
|
|
|
return
|
|
|
|
|
|
|
|
twitter_user = api.me()
|
|
|
|
self.twitter_picture_url = twitter_user.profile_image_url
|
|
|
|
self.twitter_username = twitter_user.screen_name
|
|
|
|
self.twitter_friend_ids = friend_ids
|
|
|
|
self.twitter_refreshed_date = datetime.datetime.utcnow()
|
|
|
|
self.save()
|
|
|
|
|
2011-12-25 20:50:59 -08:00
|
|
|
self.follow_twitter_friends()
|
|
|
|
|
|
|
|
profile, _ = MSocialProfile.objects.get_or_create(user_id=self.user_id)
|
|
|
|
profile.location = profile.location or twitter_user.location
|
|
|
|
profile.bio = profile.bio or twitter_user.description
|
|
|
|
profile.website = profile.website or twitter_user.url
|
|
|
|
profile.save()
|
2011-12-27 21:38:32 -08:00
|
|
|
profile.count()
|
2011-12-25 20:50:59 -08:00
|
|
|
if not profile.photo_url or not profile.photo_service:
|
|
|
|
self.set_photo('twitter')
|
|
|
|
|
2011-12-21 09:43:17 -08:00
|
|
|
def sync_facebook_friends(self):
|
|
|
|
graph = self.facebook_api()
|
2011-12-20 11:49:49 -08:00
|
|
|
if not graph:
|
|
|
|
return
|
|
|
|
|
|
|
|
friends = graph.get_connections("me", "friends")
|
|
|
|
if not friends:
|
|
|
|
return
|
|
|
|
|
2011-12-21 09:43:17 -08:00
|
|
|
facebook_friend_ids = [unicode(friend["id"]) for friend in friends["data"]]
|
2011-12-20 11:49:49 -08:00
|
|
|
self.facebook_friend_ids = facebook_friend_ids
|
2011-12-21 09:43:17 -08:00
|
|
|
self.facebook_refresh_date = datetime.datetime.utcnow()
|
2011-12-22 13:36:03 -08:00
|
|
|
self.facebook_picture_url = "//graph.facebook.com/%s/picture" % self.facebook_uid
|
|
|
|
self.save()
|
2011-12-24 00:30:37 -08:00
|
|
|
|
|
|
|
self.follow_facebook_friends()
|
|
|
|
|
2011-12-25 20:50:59 -08:00
|
|
|
facebook_user = graph.request('me', args={'fields':'website,bio,location'})
|
|
|
|
profile, _ = MSocialProfile.objects.get_or_create(user_id=self.user_id)
|
|
|
|
profile.location = profile.location or (facebook_user.get('location') and facebook_user['location']['name'])
|
|
|
|
profile.bio = profile.bio or facebook_user.get('bio')
|
|
|
|
profile.website = profile.website or facebook_user.get('website')
|
|
|
|
profile.save()
|
2011-12-27 21:38:32 -08:00
|
|
|
profile.count()
|
2011-12-25 20:50:59 -08:00
|
|
|
if not profile.photo_url or not profile.photo_service:
|
|
|
|
self.set_photo('facebook')
|
|
|
|
|
2011-12-24 00:30:37 -08:00
|
|
|
def follow_twitter_friends(self):
|
|
|
|
social_profile, _ = MSocialProfile.objects.get_or_create(user_id=self.user_id)
|
|
|
|
following = []
|
|
|
|
followers = 0
|
|
|
|
|
|
|
|
if self.autofollow:
|
|
|
|
# Follow any friends already on NewsBlur
|
|
|
|
for twitter_uid in self.twitter_friend_ids:
|
|
|
|
user_social_services = MSocialServices.objects.filter(twitter_uid=twitter_uid)
|
|
|
|
if user_social_services:
|
|
|
|
followee_user_id = user_social_services[0].user_id
|
|
|
|
social_profile.follow_user(followee_user_id)
|
|
|
|
following.append(followee_user_id)
|
|
|
|
|
|
|
|
# Follow any friends already on NewsBlur
|
|
|
|
following_users = MSocialServices.objects.filter(twitter_friend_ids__contains=self.twitter_uid)
|
|
|
|
for following_user in following_users:
|
|
|
|
if following_user.autofollow:
|
|
|
|
following_user_profile = MSocialProfile.objects.get(user_id=following_user.user_id)
|
|
|
|
following_user_profile.follow_user(self.user_id, check_unfollowed=True)
|
|
|
|
followers += 1
|
|
|
|
|
|
|
|
user = User.objects.get(pk=self.user_id)
|
|
|
|
logging.user(user, "~BB~FRTwitter import: following ~SB%s~SN with ~SB%s~SN followers" % (following, followers))
|
|
|
|
|
|
|
|
return following
|
|
|
|
|
|
|
|
def follow_facebook_friends(self):
|
|
|
|
social_profile, _ = MSocialProfile.objects.get_or_create(user_id=self.user_id)
|
|
|
|
following = []
|
|
|
|
followers = 0
|
|
|
|
|
|
|
|
if self.autofollow:
|
|
|
|
# Follow any friends already on NewsBlur
|
|
|
|
for facebook_uid in self.facebook_friend_ids:
|
|
|
|
user_social_services = MSocialServices.objects.filter(facebook_uid=facebook_uid)
|
|
|
|
if user_social_services:
|
|
|
|
followee_user_id = user_social_services[0].user_id
|
|
|
|
social_profile.follow_user(followee_user_id)
|
|
|
|
following.append(followee_user_id)
|
|
|
|
|
|
|
|
# Follow any friends already on NewsBlur
|
|
|
|
following_users = MSocialServices.objects.filter(facebook_friend_ids__contains=self.facebook_uid)
|
|
|
|
for following_user in following_users:
|
|
|
|
if following_user.autofollow:
|
|
|
|
following_user_profile = MSocialProfile.objects.get(user_id=following_user.user_id)
|
|
|
|
following_user_profile.follow_user(self.user_id, check_unfollowed=True)
|
|
|
|
followers += 1
|
|
|
|
|
|
|
|
user = User.objects.get(pk=self.user_id)
|
|
|
|
logging.user(user, "~BB~FRFacebook import: following ~SB%s~SN with ~SB%s~SN followers" % (len(following), followers))
|
|
|
|
|
|
|
|
return following
|
|
|
|
|
2011-12-22 13:36:03 -08:00
|
|
|
def disconnect_twitter(self):
|
|
|
|
self.twitter_uid = None
|
|
|
|
self.save()
|
|
|
|
|
|
|
|
def disconnect_facebook(self):
|
|
|
|
self.facebook_uid = None
|
2011-12-21 09:43:17 -08:00
|
|
|
self.save()
|
2011-12-25 20:50:59 -08:00
|
|
|
|
|
|
|
def set_photo(self, service):
|
2012-01-20 09:38:41 -08:00
|
|
|
profile, _ = MSocialProfile.objects.get_or_create(user_id=self.user_id)
|
2011-12-25 20:50:59 -08:00
|
|
|
profile.photo_service = service
|
|
|
|
if service == 'twitter':
|
|
|
|
profile.photo_url = self.twitter_picture_url
|
|
|
|
elif service == 'facebook':
|
|
|
|
profile.photo_url = self.facebook_picture_url
|
|
|
|
elif service == 'upload':
|
|
|
|
profile.photo_url = self.upload_picture_url
|
|
|
|
elif service == 'gravatar':
|
|
|
|
user = User.objects.get(pk=self.user_id)
|
|
|
|
profile.photo_url = "http://www.gravatar.com/avatar/" + \
|
|
|
|
hashlib.md5(user.email).hexdigest()
|
|
|
|
profile.save()
|