2010-04-27 17:35:51 -04:00
|
|
|
import datetime
|
2011-01-07 16:26:17 -05:00
|
|
|
import time
|
2010-06-14 01:01:21 -04:00
|
|
|
from django.shortcuts import render_to_response, get_object_or_404
|
2009-06-16 03:08:55 +00:00
|
|
|
from django.contrib.auth.decorators import login_required
|
|
|
|
from django.template import RequestContext
|
2011-05-07 17:58:53 -04:00
|
|
|
from django.template.loader import render_to_string
|
2009-09-08 04:27:27 +00:00
|
|
|
from django.db import IntegrityError
|
2010-01-24 22:53:46 -05:00
|
|
|
from django.views.decorators.cache import never_cache
|
|
|
|
from django.core.urlresolvers import reverse
|
2010-04-27 17:56:16 -04:00
|
|
|
from django.contrib.auth import login as login_user
|
2010-11-12 10:55:44 -05:00
|
|
|
from django.contrib.auth import logout as logout_user
|
2010-07-20 23:59:56 -04:00
|
|
|
from django.contrib.auth.models import User
|
2010-08-30 19:57:27 -04:00
|
|
|
from django.http import HttpResponse, HttpResponseRedirect, HttpResponseForbidden, Http404
|
2010-07-20 23:59:56 -04:00
|
|
|
from django.conf import settings
|
2010-10-19 19:09:08 -04:00
|
|
|
from django.core.mail import mail_admins
|
2011-05-07 17:58:53 -04:00
|
|
|
from django.core.validators import email_re
|
|
|
|
from django.core.mail import EmailMultiAlternatives
|
2011-11-15 18:19:09 -08:00
|
|
|
from pymongo.helpers import OperationFailure
|
2011-01-07 19:06:36 -05:00
|
|
|
from collections import defaultdict
|
2011-01-11 10:11:25 -05:00
|
|
|
from operator import itemgetter
|
2011-03-02 12:05:58 -05:00
|
|
|
from apps.recommendations.models import RecommendedFeed
|
2010-08-22 18:34:40 -04:00
|
|
|
from apps.analyzer.models import MClassifierTitle, MClassifierAuthor, MClassifierFeed, MClassifierTag
|
2010-01-21 13:12:29 -05:00
|
|
|
from apps.analyzer.models import apply_classifier_titles, apply_classifier_feeds, apply_classifier_authors, apply_classifier_tags
|
2010-03-23 20:03:40 -04:00
|
|
|
from apps.analyzer.models import get_classifiers_for_user
|
2011-09-19 09:46:36 -07:00
|
|
|
from apps.profile.models import Profile
|
2010-08-21 23:49:36 -04:00
|
|
|
from apps.reader.models import UserSubscription, UserSubscriptionFolders, MUserStory, Feature
|
2010-06-12 21:20:06 -04:00
|
|
|
from apps.reader.forms import SignupForm, LoginForm, FeatureForm
|
2011-04-21 23:10:43 -04:00
|
|
|
from apps.rss_feeds.models import MFeedIcon
|
2011-05-07 17:22:41 -04:00
|
|
|
from apps.statistics.models import MStatistics, MFeedback
|
2010-04-27 17:35:51 -04:00
|
|
|
try:
|
2010-11-30 10:30:18 -05:00
|
|
|
from apps.rss_feeds.models import Feed, MFeedPage, DuplicateFeed, MStory, MStarredStory, FeedLoadtime
|
2010-04-27 17:35:51 -04:00
|
|
|
except:
|
|
|
|
pass
|
2011-01-23 02:13:55 -05:00
|
|
|
from utils import json_functions as json
|
2010-07-24 15:54:25 -04:00
|
|
|
from utils.user_functions import get_user, ajax_login_required
|
2011-01-23 02:13:55 -05:00
|
|
|
from utils.feed_functions import relative_timesince
|
2010-10-31 16:03:50 -04:00
|
|
|
from utils.story_functions import format_story_link_date__short
|
|
|
|
from utils.story_functions import format_story_link_date__long
|
2011-01-07 16:26:17 -05:00
|
|
|
from utils.story_functions import bunch
|
2011-01-15 18:41:41 -05:00
|
|
|
from utils.story_functions import story_score
|
2010-08-16 12:52:39 -04:00
|
|
|
from utils import log as logging
|
2011-11-01 10:32:11 -07:00
|
|
|
from utils.view_functions import get_argument_or_404
|
2011-11-08 13:48:54 -08:00
|
|
|
from utils.ratelimit import ratelimit
|
2011-04-11 21:57:45 -04:00
|
|
|
from vendor.timezones.utilities import localtime_for_timezone
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2009-07-28 02:27:27 +00:00
|
|
|
SINGLE_DAY = 60*60*24
|
2009-08-08 16:52:11 +00:00
|
|
|
|
2010-01-24 22:53:46 -05:00
|
|
|
@never_cache
|
2009-06-16 03:08:55 +00:00
|
|
|
def index(request):
|
2010-04-03 00:59:03 -04:00
|
|
|
if request.method == "POST":
|
2010-04-22 21:17:00 -04:00
|
|
|
if request.POST['submit'] == 'login':
|
2011-05-11 20:22:53 -04:00
|
|
|
login_form = LoginForm(request.POST, prefix='login')
|
2010-04-22 21:17:00 -04:00
|
|
|
signup_form = SignupForm(prefix='signup')
|
|
|
|
else:
|
2011-05-11 20:22:53 -04:00
|
|
|
login_form = LoginForm(prefix='login')
|
2010-04-22 21:17:00 -04:00
|
|
|
signup_form = SignupForm(request.POST, prefix='signup')
|
2010-04-03 00:59:03 -04:00
|
|
|
else:
|
2011-05-11 20:22:53 -04:00
|
|
|
login_form = LoginForm(prefix='login')
|
2010-04-22 21:17:00 -04:00
|
|
|
signup_form = SignupForm(prefix='signup')
|
2010-10-17 17:25:10 -04:00
|
|
|
|
2011-05-11 20:22:53 -04:00
|
|
|
user = get_user(request)
|
|
|
|
authed = request.user.is_authenticated()
|
|
|
|
features = Feature.objects.all()[:3]
|
|
|
|
feature_form = FeatureForm() if request.user.is_staff else None
|
|
|
|
feed_count = UserSubscription.objects.filter(user=request.user).count() if authed else 0
|
|
|
|
active_count = UserSubscription.objects.filter(user=request.user, active=True).count() if authed else 0
|
|
|
|
train_count = UserSubscription.objects.filter(user=request.user, active=True, is_trained=False, feed__stories_last_month__gte=1).count() if authed else 0
|
|
|
|
recommended_feeds = RecommendedFeed.objects.filter(is_public=True, approved_date__lte=datetime.datetime.now()).select_related('feed')[:2]
|
2011-07-11 18:22:28 -07:00
|
|
|
unmoderated_feeds = RecommendedFeed.objects.filter(is_public=False, declined_date__isnull=True).select_related('feed')[:2]
|
2011-05-11 20:22:53 -04:00
|
|
|
statistics = MStatistics.all()
|
|
|
|
feedbacks = MFeedback.all()
|
2011-08-24 21:51:34 -07:00
|
|
|
start_import_from_google_reader = request.session.get('import_from_google_reader', False)
|
|
|
|
if start_import_from_google_reader:
|
|
|
|
del request.session['import_from_google_reader']
|
2010-08-17 23:40:03 -04:00
|
|
|
|
2010-04-22 21:17:00 -04:00
|
|
|
return render_to_response('reader/feeds.xhtml', {
|
2011-07-17 20:53:30 -07:00
|
|
|
'user_profile' : hasattr(user, 'profile') and user.profile,
|
2011-05-11 20:22:53 -04:00
|
|
|
'login_form' : login_form,
|
|
|
|
'signup_form' : signup_form,
|
|
|
|
'feature_form' : feature_form,
|
|
|
|
'features' : features,
|
|
|
|
'feed_count' : feed_count,
|
|
|
|
'active_count' : active_count,
|
|
|
|
'train_count' : active_count - train_count,
|
|
|
|
'account_images' : range(1, 4),
|
|
|
|
'recommended_feeds' : recommended_feeds,
|
2011-07-11 18:22:28 -07:00
|
|
|
'unmoderated_feeds' : unmoderated_feeds,
|
2011-05-11 20:22:53 -04:00
|
|
|
'statistics' : statistics,
|
|
|
|
'feedbacks' : feedbacks,
|
2011-08-24 21:51:34 -07:00
|
|
|
'start_import_from_google_reader': start_import_from_google_reader,
|
2010-04-22 21:17:00 -04:00
|
|
|
}, context_instance=RequestContext(request))
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2010-01-24 22:53:46 -05:00
|
|
|
@never_cache
|
2010-01-17 20:00:12 -05:00
|
|
|
def login(request):
|
2010-11-12 10:55:44 -05:00
|
|
|
code = -1
|
2011-08-13 23:00:51 -07:00
|
|
|
message = ""
|
2010-01-24 22:53:46 -05:00
|
|
|
if request.method == "POST":
|
2010-04-22 21:17:00 -04:00
|
|
|
form = LoginForm(request.POST, prefix='login')
|
2010-01-24 22:53:46 -05:00
|
|
|
if form.is_valid():
|
2010-04-27 17:56:16 -04:00
|
|
|
login_user(request, form.get_user())
|
2010-11-10 21:54:40 -05:00
|
|
|
if request.POST.get('api'):
|
2011-02-23 13:46:47 -05:00
|
|
|
logging.user(form.get_user(), "~FG~BB~SKiPhone Login~FW")
|
2010-11-12 10:55:44 -05:00
|
|
|
code = 1
|
2010-11-10 21:54:40 -05:00
|
|
|
else:
|
2011-02-23 13:46:47 -05:00
|
|
|
logging.user(form.get_user(), "~FG~BBLogin~FW")
|
2010-11-10 21:54:40 -05:00
|
|
|
return HttpResponseRedirect(reverse('index'))
|
2011-08-13 23:00:51 -07:00
|
|
|
else:
|
|
|
|
message = form.errors.items()[0][1][0]
|
2010-01-24 22:53:46 -05:00
|
|
|
|
2010-11-10 21:54:40 -05:00
|
|
|
if request.POST.get('api'):
|
2011-08-13 23:00:51 -07:00
|
|
|
return HttpResponse(json.encode(dict(code=code, message=message)), mimetype='application/json')
|
2010-11-10 21:54:40 -05:00
|
|
|
else:
|
|
|
|
return index(request)
|
2010-04-22 21:17:00 -04:00
|
|
|
|
|
|
|
@never_cache
|
|
|
|
def signup(request):
|
|
|
|
if request.method == "POST":
|
|
|
|
form = SignupForm(prefix='signup', data=request.POST)
|
|
|
|
if form.is_valid():
|
|
|
|
new_user = form.save()
|
2010-04-27 17:56:16 -04:00
|
|
|
login_user(request, new_user)
|
2011-02-23 13:46:47 -05:00
|
|
|
logging.user(new_user, "~FG~SB~BBNEW SIGNUP~FW")
|
2010-04-22 21:17:00 -04:00
|
|
|
return HttpResponseRedirect(reverse('index'))
|
|
|
|
|
|
|
|
return index(request)
|
2010-01-17 20:00:12 -05:00
|
|
|
|
2010-01-24 22:53:46 -05:00
|
|
|
@never_cache
|
|
|
|
def logout(request):
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FG~BBLogout~FW")
|
2010-11-12 10:55:44 -05:00
|
|
|
logout_user(request)
|
2010-01-24 22:53:46 -05:00
|
|
|
|
2010-11-11 23:48:27 -05:00
|
|
|
if request.GET.get('api'):
|
|
|
|
return HttpResponse(json.encode(dict(code=1)), mimetype='application/json')
|
|
|
|
else:
|
|
|
|
return HttpResponseRedirect(reverse('index'))
|
2011-09-19 09:46:36 -07:00
|
|
|
|
|
|
|
def autologin(request, username, secret):
|
2011-09-21 17:49:26 -07:00
|
|
|
next = request.GET.get('next', '')
|
|
|
|
|
2011-09-19 09:46:36 -07:00
|
|
|
if not username or not secret:
|
|
|
|
return HttpResponseForbidden()
|
|
|
|
|
|
|
|
profile = Profile.objects.filter(user__username=username, secret_token=secret)
|
2011-09-21 17:49:26 -07:00
|
|
|
if not profile:
|
2011-09-19 09:46:36 -07:00
|
|
|
return HttpResponseForbidden()
|
2011-09-21 17:49:26 -07:00
|
|
|
|
|
|
|
user = profile[0].user
|
|
|
|
user.backend = settings.AUTHENTICATION_BACKENDS[0]
|
|
|
|
login_user(request, user)
|
|
|
|
logging.user(user, "~FG~BB~SKAuto-Login. Next stop: %s~FW" % (next if next else 'Homepage',))
|
2011-09-19 09:46:36 -07:00
|
|
|
|
2011-09-21 17:49:26 -07:00
|
|
|
if next:
|
|
|
|
next = '?next=' + next
|
|
|
|
|
|
|
|
return HttpResponseRedirect(reverse('index') + next)
|
2010-01-24 22:53:46 -05:00
|
|
|
|
2011-11-15 18:19:09 -08:00
|
|
|
@ratelimit(minutes=1, requests=20)
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2009-06-16 03:08:55 +00:00
|
|
|
def load_feeds(request):
|
2011-04-04 12:01:29 -04:00
|
|
|
user = get_user(request)
|
|
|
|
feeds = {}
|
|
|
|
not_yet_fetched = False
|
2011-05-01 17:19:01 -04:00
|
|
|
include_favicons = request.REQUEST.get('include_favicons', False)
|
2011-04-20 09:35:59 -04:00
|
|
|
flat = request.REQUEST.get('flat', False)
|
2011-11-07 08:27:59 -08:00
|
|
|
update_counts = request.REQUEST.get('update_counts', False)
|
2011-04-20 09:35:59 -04:00
|
|
|
|
2011-12-01 11:10:25 -08:00
|
|
|
if include_favicons == 'false': include_favicons = False
|
|
|
|
if update_counts == 'false': update_counts = False
|
|
|
|
if flat == 'false': flat = False
|
|
|
|
|
2011-04-20 09:35:59 -04:00
|
|
|
if flat: return load_feeds_flat(request)
|
2010-02-11 01:28:47 -05:00
|
|
|
|
2010-03-02 10:56:25 -05:00
|
|
|
try:
|
|
|
|
folders = UserSubscriptionFolders.objects.get(user=user)
|
|
|
|
except UserSubscriptionFolders.DoesNotExist:
|
|
|
|
data = dict(feeds=[], folders=[])
|
2010-07-25 23:13:27 -04:00
|
|
|
return data
|
2010-11-04 19:32:19 -04:00
|
|
|
except UserSubscriptionFolders.MultipleObjectsReturned:
|
|
|
|
UserSubscriptionFolders.objects.filter(user=user)[1:].delete()
|
|
|
|
folders = UserSubscriptionFolders.objects.get(user=user)
|
2011-08-29 20:51:40 -07:00
|
|
|
|
2011-04-05 19:24:12 -04:00
|
|
|
user_subs = UserSubscription.objects.select_related('feed').filter(user=user)
|
2010-08-13 10:43:48 -04:00
|
|
|
|
2010-02-11 01:28:47 -05:00
|
|
|
for sub in user_subs:
|
2011-08-29 21:28:12 -07:00
|
|
|
pk = sub.feed.pk
|
2011-11-07 08:27:59 -08:00
|
|
|
if update_counts:
|
|
|
|
sub.calculate_feed_scores(silent=True)
|
2011-08-22 18:17:42 -07:00
|
|
|
feeds[pk] = sub.canonical(include_favicon=include_favicons)
|
|
|
|
if feeds[pk].get('not_yet_fetched'):
|
2010-08-11 22:02:47 -04:00
|
|
|
not_yet_fetched = True
|
2010-10-07 19:56:23 -04:00
|
|
|
if not sub.feed.active and not sub.feed.has_feed_exception and not sub.feed.has_page_exception:
|
|
|
|
sub.feed.count_subscribers()
|
|
|
|
sub.feed.schedule_feed_fetch_immediately()
|
2011-02-25 20:23:05 -05:00
|
|
|
if sub.active and sub.feed.active_subscribers <= 0:
|
2011-02-23 23:23:52 -05:00
|
|
|
sub.feed.count_subscribers()
|
|
|
|
sub.feed.schedule_feed_fetch_immediately()
|
2010-08-11 22:02:47 -04:00
|
|
|
|
|
|
|
if not_yet_fetched:
|
|
|
|
for f in feeds:
|
|
|
|
if 'not_yet_fetched' not in feeds[f]:
|
|
|
|
feeds[f]['not_yet_fetched'] = False
|
2010-12-01 14:11:42 -05:00
|
|
|
|
2010-12-02 20:18:33 -05:00
|
|
|
starred_count = MStarredStory.objects(user_id=user.pk).count()
|
2010-12-01 14:11:42 -05:00
|
|
|
|
|
|
|
data = {
|
|
|
|
'feeds': feeds,
|
|
|
|
'folders': json.decode(folders.folders),
|
|
|
|
'starred_count': starred_count,
|
|
|
|
}
|
2010-07-25 23:13:27 -04:00
|
|
|
return data
|
2010-04-14 22:58:00 -04:00
|
|
|
|
2011-04-05 19:24:12 -04:00
|
|
|
@json.json_view
|
|
|
|
def load_feed_favicons(request):
|
|
|
|
user = get_user(request)
|
|
|
|
feed_ids = request.REQUEST.getlist('feed_ids')
|
2011-04-21 10:44:50 -04:00
|
|
|
user_subs = UserSubscription.objects.select_related('feed').filter(user=user, active=True)
|
2011-05-01 17:48:10 -04:00
|
|
|
if feed_ids and len(feed_ids) > 0:
|
2011-04-05 19:24:12 -04:00
|
|
|
user_subs = user_subs.filter(feed__in=feed_ids)
|
|
|
|
|
2011-04-21 23:10:43 -04:00
|
|
|
feed_ids = [sub['feed__pk'] for sub in user_subs.values('feed__pk')]
|
2011-08-29 21:28:12 -07:00
|
|
|
feed_icons = dict([(i.feed_id, i.data) for i in MFeedIcon.objects(feed_id__in=feed_ids)])
|
2011-04-05 19:24:12 -04:00
|
|
|
|
2011-04-21 23:10:43 -04:00
|
|
|
return feed_icons
|
2011-07-20 21:08:57 -07:00
|
|
|
|
2011-04-20 09:35:59 -04:00
|
|
|
def load_feeds_flat(request):
|
2011-07-20 21:08:57 -07:00
|
|
|
user = request.user
|
2011-06-07 10:17:56 -04:00
|
|
|
include_favicons = request.REQUEST.get('include_favicons', False)
|
2010-06-20 11:04:23 -04:00
|
|
|
feeds = {}
|
|
|
|
|
2011-12-01 18:55:54 -08:00
|
|
|
if include_favicons == 'false': include_favicons = False
|
|
|
|
|
2011-07-20 21:08:57 -07:00
|
|
|
if not user.is_authenticated():
|
|
|
|
return HttpResponseForbidden()
|
|
|
|
|
2010-06-20 11:04:23 -04:00
|
|
|
try:
|
|
|
|
folders = UserSubscriptionFolders.objects.get(user=user)
|
|
|
|
except UserSubscriptionFolders.DoesNotExist:
|
|
|
|
data = dict(folders=[])
|
2010-07-25 23:13:27 -04:00
|
|
|
return data
|
2010-06-20 11:04:23 -04:00
|
|
|
|
2011-09-15 18:28:19 -07:00
|
|
|
user_subs = UserSubscription.objects.select_related('feed').filter(user=user, active=True)
|
2010-06-20 11:04:23 -04:00
|
|
|
|
|
|
|
for sub in user_subs:
|
|
|
|
if sub.needs_unread_recalc:
|
2010-12-01 09:30:56 -05:00
|
|
|
sub.calculate_feed_scores(silent=True)
|
2011-08-29 21:28:12 -07:00
|
|
|
feeds[sub.feed.pk] = sub.canonical(include_favicon=include_favicons)
|
2010-06-20 11:04:23 -04:00
|
|
|
|
|
|
|
folders = json.decode(folders.folders)
|
|
|
|
flat_folders = {}
|
|
|
|
|
|
|
|
def make_feeds_folder(items, parent_folder="", depth=0):
|
|
|
|
for item in items:
|
2011-08-30 08:35:22 -07:00
|
|
|
if isinstance(item, int) and item in feeds:
|
2010-06-20 11:04:23 -04:00
|
|
|
if not parent_folder:
|
|
|
|
parent_folder = ' '
|
|
|
|
if parent_folder in flat_folders:
|
2011-06-07 10:17:56 -04:00
|
|
|
flat_folders[parent_folder].append(item)
|
2010-06-20 11:04:23 -04:00
|
|
|
else:
|
2011-06-07 10:17:56 -04:00
|
|
|
flat_folders[parent_folder] = [item]
|
2010-06-20 11:04:23 -04:00
|
|
|
elif isinstance(item, dict):
|
|
|
|
for folder_name in item:
|
|
|
|
folder = item[folder_name]
|
|
|
|
flat_folder_name = "%s%s%s" % (
|
2011-10-10 09:57:54 -07:00
|
|
|
parent_folder if parent_folder and parent_folder != ' ' else "",
|
2011-06-07 10:17:56 -04:00
|
|
|
" - " if parent_folder and parent_folder != ' ' else "",
|
2010-06-20 11:04:23 -04:00
|
|
|
folder_name
|
|
|
|
)
|
2011-10-10 09:57:54 -07:00
|
|
|
flat_folders[flat_folder_name] = []
|
2010-06-20 11:04:23 -04:00
|
|
|
make_feeds_folder(folder, flat_folder_name, depth+1)
|
|
|
|
|
|
|
|
make_feeds_folder(folders)
|
2011-12-07 09:57:33 -08:00
|
|
|
data = dict(flat_folders=flat_folders, feeds=feeds, user=user.username, iphone_version="1.2")
|
2010-07-25 23:13:27 -04:00
|
|
|
return data
|
2010-06-20 11:04:23 -04:00
|
|
|
|
2011-11-08 19:12:56 -08:00
|
|
|
@ratelimit(minutes=1, requests=10)
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2010-04-14 22:58:00 -04:00
|
|
|
def refresh_feeds(request):
|
2011-03-08 10:06:55 -05:00
|
|
|
start = datetime.datetime.utcnow()
|
2010-04-14 22:58:00 -04:00
|
|
|
user = get_user(request)
|
2011-03-09 09:48:24 -05:00
|
|
|
feed_ids = request.REQUEST.getlist('feed_id')
|
2010-04-14 22:58:00 -04:00
|
|
|
feeds = {}
|
2011-01-30 23:00:22 -05:00
|
|
|
user_subs = UserSubscription.objects.select_related('feed').filter(user=user, active=True)
|
2011-09-04 14:47:47 -07:00
|
|
|
feed_ids = [f for f in feed_ids if f and not f.startswith('river')]
|
2011-03-09 09:48:24 -05:00
|
|
|
if feed_ids:
|
|
|
|
user_subs = user_subs.filter(feed__in=feed_ids)
|
2011-01-15 18:41:41 -05:00
|
|
|
UNREAD_CUTOFF = datetime.datetime.utcnow() - datetime.timedelta(days=settings.DAYS_OF_UNREAD)
|
2011-04-20 09:35:59 -04:00
|
|
|
favicons_fetching = [int(f) for f in request.REQUEST.getlist('favicons_fetching') if f]
|
2011-04-21 23:10:43 -04:00
|
|
|
feed_icons = dict([(i.feed_id, i) for i in MFeedIcon.objects(feed_id__in=favicons_fetching)])
|
2011-11-28 18:01:39 -08:00
|
|
|
|
|
|
|
for i, sub in enumerate(user_subs):
|
2011-08-30 09:23:16 -07:00
|
|
|
pk = sub.feed.pk
|
2011-01-15 18:41:41 -05:00
|
|
|
if (sub.needs_unread_recalc or
|
|
|
|
sub.unread_count_updated < UNREAD_CUTOFF or
|
|
|
|
sub.oldest_unread_story_date < UNREAD_CUTOFF):
|
2011-11-28 18:01:39 -08:00
|
|
|
sub = sub.calculate_feed_scores(silent=True)
|
2011-11-29 11:43:55 -08:00
|
|
|
if not sub: continue # TODO: Figure out the correct sub and give it a new feed_id
|
2011-08-22 18:17:42 -07:00
|
|
|
feeds[pk] = {
|
2010-04-14 22:58:00 -04:00
|
|
|
'ps': sub.unread_count_positive,
|
|
|
|
'nt': sub.unread_count_neutral,
|
|
|
|
'ng': sub.unread_count_negative,
|
|
|
|
}
|
2010-08-26 10:04:32 -04:00
|
|
|
if sub.feed.has_feed_exception or sub.feed.has_page_exception:
|
2011-08-22 18:17:42 -07:00
|
|
|
feeds[pk]['has_exception'] = True
|
|
|
|
feeds[pk]['exception_type'] = 'feed' if sub.feed.has_feed_exception else 'page'
|
|
|
|
feeds[pk]['feed_address'] = sub.feed.feed_address
|
|
|
|
feeds[pk]['exception_code'] = sub.feed.exception_code
|
2011-04-20 09:35:59 -04:00
|
|
|
if request.REQUEST.get('check_fetch_status', False):
|
2011-08-22 18:17:42 -07:00
|
|
|
feeds[pk]['not_yet_fetched'] = not sub.feed.fetched_once
|
2011-08-29 20:51:40 -07:00
|
|
|
|
2011-05-01 17:19:01 -04:00
|
|
|
if sub.feed.pk in favicons_fetching and sub.feed.pk in feed_icons:
|
2011-08-22 18:17:42 -07:00
|
|
|
feeds[pk]['favicon'] = feed_icons[sub.feed.pk].data
|
|
|
|
feeds[pk]['favicon_color'] = feed_icons[sub.feed.pk].color
|
2011-11-30 21:29:31 -08:00
|
|
|
feeds[pk]['favicon_fetching'] = sub.feed.favicon_fetching
|
2011-04-05 19:24:12 -04:00
|
|
|
|
2011-11-28 18:01:39 -08:00
|
|
|
user_subs = UserSubscription.objects.select_related('feed').filter(user=user, active=True)
|
|
|
|
|
2011-08-29 20:51:40 -07:00
|
|
|
if favicons_fetching:
|
|
|
|
sub_feed_ids = [s.feed.pk for s in user_subs]
|
|
|
|
moved_feed_ids = [f for f in favicons_fetching if f not in sub_feed_ids]
|
|
|
|
for moved_feed_id in moved_feed_ids:
|
2011-08-30 08:04:42 -07:00
|
|
|
duplicate_feeds = DuplicateFeed.objects.filter(duplicate_feed_id=moved_feed_id)
|
|
|
|
if duplicate_feeds and duplicate_feeds[0].feed.pk in feeds:
|
|
|
|
feeds[moved_feed_id] = feeds[duplicate_feeds[0].feed.pk]
|
|
|
|
feeds[moved_feed_id]['dupe_feed_id'] = duplicate_feeds[0].feed.pk
|
2011-08-29 20:51:40 -07:00
|
|
|
|
2011-12-05 10:42:25 -08:00
|
|
|
if settings.DEBUG or request.REQUEST.get('check_fetch_status'):
|
2011-04-05 19:24:12 -04:00
|
|
|
diff = datetime.datetime.utcnow()-start
|
|
|
|
timediff = float("%s.%.2s" % (diff.seconds, (diff.microseconds / 1000)))
|
2011-11-28 18:01:39 -08:00
|
|
|
logging.user(request, "~FBRefreshing %s feeds (%s seconds) (%s/%s)" % (user_subs.count(), timediff, request.REQUEST.get('check_fetch_status', False), len(favicons_fetching)))
|
2011-03-08 10:06:55 -05:00
|
|
|
|
2010-08-11 22:02:47 -04:00
|
|
|
return {'feeds': feeds}
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2011-08-21 13:46:43 -07:00
|
|
|
def refresh_feed(request, feed_id):
|
|
|
|
user = get_user(request)
|
|
|
|
feed = get_object_or_404(Feed, pk=feed_id)
|
|
|
|
|
|
|
|
feed = feed.update(force=True, compute_scores=False)
|
|
|
|
usersub = UserSubscription.objects.get(user=user, feed=feed)
|
|
|
|
usersub.calculate_feed_scores(silent=False)
|
|
|
|
|
|
|
|
return load_single_feed(request, feed_id)
|
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2011-04-21 10:44:50 -04:00
|
|
|
def load_single_feed(request, feed_id):
|
2011-06-09 14:59:22 -04:00
|
|
|
start = time.time()
|
|
|
|
user = get_user(request)
|
|
|
|
offset = int(request.REQUEST.get('offset', 0))
|
|
|
|
limit = int(request.REQUEST.get('limit', 12))
|
|
|
|
page = int(request.REQUEST.get('page', 1))
|
2011-01-14 00:59:51 -05:00
|
|
|
dupe_feed_id = None
|
2011-06-10 10:52:27 -04:00
|
|
|
userstories_db = None
|
2011-11-30 09:38:31 -08:00
|
|
|
|
2011-06-09 14:59:22 -04:00
|
|
|
if page: offset = limit * (page-1)
|
|
|
|
if not feed_id: raise Http404
|
2010-10-25 20:20:59 -04:00
|
|
|
|
|
|
|
try:
|
|
|
|
feed = Feed.objects.get(id=feed_id)
|
|
|
|
except Feed.DoesNotExist:
|
|
|
|
feed_address = request.REQUEST.get('feed_address')
|
|
|
|
dupe_feed = DuplicateFeed.objects.filter(duplicate_address=feed_address)
|
|
|
|
if dupe_feed:
|
|
|
|
feed = dupe_feed[0].feed
|
2011-01-14 00:59:51 -05:00
|
|
|
dupe_feed_id = feed_id
|
2010-10-25 20:20:59 -04:00
|
|
|
else:
|
|
|
|
raise Http404
|
|
|
|
|
2011-11-01 21:55:23 -07:00
|
|
|
stories = feed.get_stories(offset, limit)
|
2009-07-05 21:45:46 +00:00
|
|
|
|
2010-01-21 13:12:29 -05:00
|
|
|
# Get intelligence classifier for user
|
2011-06-09 15:36:24 -04:00
|
|
|
classifier_feeds = list(MClassifierFeed.objects(user_id=user.pk, feed_id=feed_id))
|
|
|
|
classifier_authors = list(MClassifierAuthor.objects(user_id=user.pk, feed_id=feed_id))
|
|
|
|
classifier_titles = list(MClassifierTitle.objects(user_id=user.pk, feed_id=feed_id))
|
|
|
|
classifier_tags = list(MClassifierTag.objects(user_id=user.pk, feed_id=feed_id))
|
2010-01-21 13:12:29 -05:00
|
|
|
|
2011-06-09 14:59:22 -04:00
|
|
|
checkpoint1 = time.time()
|
2010-01-21 13:12:29 -05:00
|
|
|
|
2011-02-09 18:52:36 -05:00
|
|
|
usersub = UserSubscription.objects.get(user=user, feed=feed)
|
2010-09-10 08:32:48 -07:00
|
|
|
userstories = []
|
2011-08-18 21:47:38 -07:00
|
|
|
if usersub and stories:
|
2011-09-14 18:03:09 -07:00
|
|
|
story_ids = [story['id'] for story in stories]
|
2011-03-15 23:42:27 -04:00
|
|
|
userstories_db = MUserStory.objects(user_id=user.pk,
|
|
|
|
feed_id=feed.pk,
|
2011-09-14 18:03:09 -07:00
|
|
|
story_id__in=story_ids).only('story_id')
|
|
|
|
starred_stories = MStarredStory.objects(user_id=user.pk,
|
|
|
|
story_feed_id=feed_id,
|
|
|
|
story_guid__in=story_ids).only('story_guid', 'starred_date')
|
2011-03-15 23:42:27 -04:00
|
|
|
starred_stories = dict([(story.story_guid, story.starred_date) for story in starred_stories])
|
2011-09-14 18:03:09 -07:00
|
|
|
userstories = set(us.story_id for us in userstories_db)
|
2010-09-10 08:32:48 -07:00
|
|
|
|
2011-06-09 14:59:22 -04:00
|
|
|
checkpoint2 = time.time()
|
|
|
|
|
2009-07-28 02:27:27 +00:00
|
|
|
for story in stories:
|
2010-10-31 19:32:41 -04:00
|
|
|
story_date = localtime_for_timezone(story['story_date'], user.profile.timezone)
|
2011-01-12 23:30:38 -05:00
|
|
|
now = localtime_for_timezone(datetime.datetime.now(), user.profile.timezone)
|
|
|
|
story['short_parsed_date'] = format_story_link_date__short(story_date, now)
|
|
|
|
story['long_parsed_date'] = format_story_link_date__long(story_date, now)
|
2011-03-15 23:42:27 -04:00
|
|
|
if usersub:
|
|
|
|
if story['id'] in userstories:
|
|
|
|
story['read_status'] = 1
|
|
|
|
elif not story.get('read_status') and story['story_date'] < usersub.mark_read_date:
|
|
|
|
story['read_status'] = 1
|
|
|
|
elif not story.get('read_status') and story['story_date'] > usersub.last_read_date:
|
|
|
|
story['read_status'] = 0
|
|
|
|
if story['id'] in starred_stories:
|
|
|
|
story['starred'] = True
|
|
|
|
starred_date = localtime_for_timezone(starred_stories[story['id']], user.profile.timezone)
|
|
|
|
story['starred_date'] = format_story_link_date__long(starred_date, now)
|
|
|
|
else:
|
2009-07-28 02:27:27 +00:00
|
|
|
story['read_status'] = 1
|
2010-01-21 13:12:29 -05:00
|
|
|
story['intelligence'] = {
|
|
|
|
'feed': apply_classifier_feeds(classifier_feeds, feed),
|
|
|
|
'author': apply_classifier_authors(classifier_authors, story),
|
|
|
|
'tags': apply_classifier_tags(classifier_tags, story),
|
|
|
|
'title': apply_classifier_titles(classifier_titles, story),
|
|
|
|
}
|
2011-06-09 19:00:31 -04:00
|
|
|
|
2011-06-09 14:59:22 -04:00
|
|
|
checkpoint3 = time.time()
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2010-01-21 13:12:29 -05:00
|
|
|
# Intelligence
|
2011-01-17 22:48:38 -05:00
|
|
|
feed_tags = json.decode(feed.data.popular_tags) if feed.data.popular_tags else []
|
|
|
|
feed_authors = json.decode(feed.data.popular_authors) if feed.data.popular_authors else []
|
2010-03-23 20:03:40 -04:00
|
|
|
classifiers = get_classifiers_for_user(user, feed_id, classifier_feeds,
|
|
|
|
classifier_authors, classifier_titles, classifier_tags)
|
2010-01-12 01:19:37 +00:00
|
|
|
|
2011-03-15 23:42:27 -04:00
|
|
|
if usersub:
|
|
|
|
usersub.feed_opens += 1
|
|
|
|
usersub.save()
|
2011-08-18 21:47:38 -07:00
|
|
|
diff1 = checkpoint1-start
|
|
|
|
diff2 = checkpoint2-start
|
|
|
|
diff3 = checkpoint3-start
|
2011-06-09 14:59:22 -04:00
|
|
|
timediff = time.time()-start
|
2010-10-31 16:03:50 -04:00
|
|
|
last_update = relative_timesince(feed.last_update)
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FYLoading feed: ~SB%s%s ~SN(%.4s seconds, ~SB%.4s/%.4s(%s)/%.4s~SN)" % (
|
2011-08-18 21:47:38 -07:00
|
|
|
feed.feed_title[:32], ('~SN/p%s' % page) if page > 1 else '', timediff,
|
|
|
|
diff1, diff2, userstories_db and userstories_db.count() or '~SN0~SB', diff3))
|
2010-09-23 10:29:18 -04:00
|
|
|
FeedLoadtime.objects.create(feed=feed, loadtime=timediff)
|
|
|
|
|
2010-07-26 21:38:56 -04:00
|
|
|
data = dict(stories=stories,
|
|
|
|
feed_tags=feed_tags,
|
|
|
|
feed_authors=feed_authors,
|
2010-09-22 19:59:07 -04:00
|
|
|
classifiers=classifiers,
|
2010-10-25 20:44:52 -04:00
|
|
|
last_update=last_update,
|
2011-11-25 11:58:40 -05:00
|
|
|
feed_id=feed.pk,
|
|
|
|
elapsed_time=round(float(timediff), 2))
|
2011-01-14 00:59:51 -05:00
|
|
|
|
|
|
|
if dupe_feed_id: data['dupe_feed_id'] = dupe_feed_id
|
2011-03-15 23:42:27 -04:00
|
|
|
if not usersub:
|
|
|
|
data.update(feed.canonical())
|
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
return data
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2011-04-21 22:36:26 -04:00
|
|
|
def load_feed_page(request, feed_id):
|
2011-02-03 18:47:53 -05:00
|
|
|
if not feed_id:
|
2010-08-30 19:57:27 -04:00
|
|
|
raise Http404
|
|
|
|
|
2011-01-29 19:16:40 -05:00
|
|
|
data = MFeedPage.get_data(feed_id=feed_id)
|
2011-01-14 00:59:51 -05:00
|
|
|
|
2010-08-16 12:52:39 -04:00
|
|
|
if not data:
|
2011-01-14 00:59:51 -05:00
|
|
|
data = "Fetching feed..."
|
2009-08-13 03:26:12 +00:00
|
|
|
|
|
|
|
return HttpResponse(data, mimetype='text/html')
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2010-12-02 11:09:09 -05:00
|
|
|
@json.json_view
|
|
|
|
def load_starred_stories(request):
|
|
|
|
user = get_user(request)
|
|
|
|
offset = int(request.REQUEST.get('offset', 0))
|
2010-12-03 09:49:38 -05:00
|
|
|
limit = int(request.REQUEST.get('limit', 10))
|
2011-09-07 09:56:54 -07:00
|
|
|
page = int(request.REQUEST.get('page', 0))
|
2011-08-29 22:25:25 -07:00
|
|
|
if page: offset = limit * (page - 1)
|
2010-12-02 11:09:09 -05:00
|
|
|
|
2010-12-02 20:18:33 -05:00
|
|
|
mstories = MStarredStory.objects(user_id=user.pk).order_by('-starred_date')[offset:offset+limit]
|
|
|
|
stories = Feed.format_stories(mstories)
|
|
|
|
|
|
|
|
for story in stories:
|
|
|
|
story_date = localtime_for_timezone(story['story_date'], user.profile.timezone)
|
2011-01-12 23:30:38 -05:00
|
|
|
now = localtime_for_timezone(datetime.datetime.now(), user.profile.timezone)
|
|
|
|
story['short_parsed_date'] = format_story_link_date__short(story_date, now)
|
|
|
|
story['long_parsed_date'] = format_story_link_date__long(story_date, now)
|
2010-12-04 13:51:46 -05:00
|
|
|
starred_date = localtime_for_timezone(story['starred_date'], user.profile.timezone)
|
2011-01-12 23:30:38 -05:00
|
|
|
story['starred_date'] = format_story_link_date__long(starred_date, now)
|
2010-12-02 20:18:33 -05:00
|
|
|
story['read_status'] = 1
|
|
|
|
story['starred'] = True
|
|
|
|
story['intelligence'] = {
|
|
|
|
'feed': 0,
|
|
|
|
'author': 0,
|
|
|
|
'tags': 0,
|
|
|
|
'title': 0,
|
|
|
|
}
|
|
|
|
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FCLoading starred stories: ~SB%s stories" % (len(stories)))
|
2010-12-04 23:34:57 -05:00
|
|
|
|
2010-12-02 20:18:33 -05:00
|
|
|
return dict(stories=stories)
|
2010-12-12 22:52:15 -05:00
|
|
|
|
|
|
|
@json.json_view
|
|
|
|
def load_river_stories(request):
|
2011-12-11 16:14:36 -08:00
|
|
|
limit = 18
|
|
|
|
offset = int(request.REQUEST.get('offset', 0))
|
|
|
|
start = time.time()
|
|
|
|
user = get_user(request)
|
|
|
|
feed_ids = [int(feed_id) for feed_id in request.REQUEST.getlist('feeds') if feed_id]
|
|
|
|
original_feed_ids = list(feed_ids)
|
|
|
|
page = int(request.REQUEST.get('page', 1))
|
|
|
|
read_stories_count = int(request.REQUEST.get('read_stories_count', 0))
|
|
|
|
days_to_keep_unreads = datetime.timedelta(days=settings.DAYS_OF_UNREAD)
|
2010-12-16 12:59:49 -05:00
|
|
|
|
2011-01-10 09:49:26 -05:00
|
|
|
if not feed_ids:
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FCLoading empty river stories: page %s" % (page))
|
2011-01-10 09:49:26 -05:00
|
|
|
return dict(stories=[])
|
|
|
|
|
2010-12-16 12:59:49 -05:00
|
|
|
# Fetch all stories at and before the page number.
|
|
|
|
# Not a single page, because reading stories can move them up in the unread order.
|
|
|
|
# `read_stories_count` is an optimization, works best when all 25 stories before have been read.
|
2011-02-09 15:45:41 -05:00
|
|
|
limit = limit * page - read_stories_count
|
2010-12-13 21:53:22 -05:00
|
|
|
|
2010-12-16 12:59:49 -05:00
|
|
|
# Read stories to exclude
|
2011-09-14 18:03:09 -07:00
|
|
|
read_stories = MUserStory.objects(user_id=user.pk, feed_id__in=feed_ids).only('story_id')
|
|
|
|
read_stories = [rs.story_id for rs in read_stories]
|
2010-12-16 12:59:49 -05:00
|
|
|
|
2011-01-07 19:20:34 -05:00
|
|
|
# Determine mark_as_read dates for all feeds to ignore all stories before this date.
|
2011-02-09 15:45:41 -05:00
|
|
|
feed_counts = {}
|
|
|
|
feed_last_reads = {}
|
2011-01-11 10:11:25 -05:00
|
|
|
for feed_id in feed_ids:
|
2011-02-03 18:38:10 -05:00
|
|
|
try:
|
|
|
|
usersub = UserSubscription.objects.get(feed__pk=feed_id, user=user)
|
|
|
|
except UserSubscription.DoesNotExist:
|
|
|
|
continue
|
2011-02-22 22:38:50 -05:00
|
|
|
if not usersub: continue
|
|
|
|
feed_counts[feed_id] = (usersub.unread_count_negative * 1 +
|
|
|
|
usersub.unread_count_neutral * 10 +
|
2011-01-30 10:46:53 -05:00
|
|
|
usersub.unread_count_positive * 20)
|
2011-08-29 21:47:58 -07:00
|
|
|
feed_last_reads[feed_id] = int(time.mktime(usersub.mark_read_date.timetuple()))
|
2011-12-11 16:14:36 -08:00
|
|
|
|
2011-11-15 18:19:09 -08:00
|
|
|
feed_counts = sorted(feed_counts.items(), key=itemgetter(1))[:40]
|
2011-01-11 10:11:25 -05:00
|
|
|
feed_ids = [f[0] for f in feed_counts]
|
2011-08-29 22:11:11 -07:00
|
|
|
feed_last_reads = dict([(str(feed_id), feed_last_reads[feed_id]) for feed_id in feed_ids
|
|
|
|
if feed_id in feed_last_reads])
|
2011-01-30 10:46:53 -05:00
|
|
|
feed_counts = dict(feed_counts)
|
2011-08-29 22:11:11 -07:00
|
|
|
|
2011-01-07 16:26:17 -05:00
|
|
|
# After excluding read stories, all that's left are stories
|
|
|
|
# past the mark_read_date. Everything returned is guaranteed to be unread.
|
2010-12-13 21:53:22 -05:00
|
|
|
mstories = MStory.objects(
|
2011-09-14 21:00:38 -07:00
|
|
|
story_guid__nin=read_stories,
|
2011-01-15 18:41:41 -05:00
|
|
|
story_feed_id__in=feed_ids,
|
2011-12-11 16:14:36 -08:00
|
|
|
# story_date__gte=start - days_to_keep_unreads
|
2011-01-07 16:26:17 -05:00
|
|
|
).map_reduce("""function() {
|
2011-02-05 15:34:43 -05:00
|
|
|
var d = feed_last_reads[this[~story_feed_id]];
|
|
|
|
if (this[~story_date].getTime()/1000 > d) {
|
|
|
|
emit(this[~id], this);
|
2011-01-07 18:44:36 -05:00
|
|
|
}
|
|
|
|
}""",
|
|
|
|
"""function(key, values) {
|
|
|
|
return values[0];
|
|
|
|
}""",
|
2011-05-18 14:23:43 -04:00
|
|
|
output='inline',
|
2011-01-15 18:41:41 -05:00
|
|
|
scope={
|
|
|
|
'feed_last_reads': feed_last_reads
|
|
|
|
}
|
|
|
|
)
|
2011-11-15 18:19:09 -08:00
|
|
|
try:
|
|
|
|
mstories = [story.value for story in mstories if story and story.value]
|
|
|
|
except OperationFailure, e:
|
|
|
|
raise e
|
2011-01-15 18:41:41 -05:00
|
|
|
|
2011-12-11 16:14:36 -08:00
|
|
|
mstories = sorted(mstories, cmp=lambda x, y: cmp(story_score(y, days_to_keep_unreads),
|
|
|
|
story_score(x, days_to_keep_unreads)))
|
2011-01-15 18:41:41 -05:00
|
|
|
|
2011-12-11 16:14:36 -08:00
|
|
|
# Prune the river to only include a set number of stories per feed
|
2011-01-15 18:41:41 -05:00
|
|
|
# story_feed_counts = defaultdict(int)
|
|
|
|
# mstories_pruned = []
|
|
|
|
# for story in mstories:
|
|
|
|
# print story['story_title'], story_feed_counts[story['story_feed_id']]
|
|
|
|
# if story_feed_counts[story['story_feed_id']] >= 3: continue
|
|
|
|
# mstories_pruned.append(story)
|
|
|
|
# story_feed_counts[story['story_feed_id']] += 1
|
2011-01-07 16:26:17 -05:00
|
|
|
stories = []
|
|
|
|
for i, story in enumerate(mstories):
|
2011-01-07 18:44:36 -05:00
|
|
|
if i < offset: continue
|
2011-01-07 16:26:17 -05:00
|
|
|
if i >= offset + limit: break
|
|
|
|
stories.append(bunch(story))
|
|
|
|
stories = Feed.format_stories(stories)
|
2011-01-15 18:41:41 -05:00
|
|
|
found_feed_ids = list(set([story['story_feed_id'] for story in stories]))
|
2010-06-14 01:01:21 -04:00
|
|
|
|
2011-01-07 19:06:36 -05:00
|
|
|
# Find starred stories
|
2011-11-29 17:57:20 -08:00
|
|
|
try:
|
|
|
|
starred_stories = MStarredStory.objects(
|
|
|
|
user_id=user.pk,
|
|
|
|
story_feed_id__in=found_feed_ids
|
|
|
|
).only('story_guid', 'starred_date')
|
|
|
|
starred_stories = dict([(story.story_guid, story.starred_date)
|
|
|
|
for story in starred_stories])
|
|
|
|
except OperationFailure:
|
|
|
|
logging.info(" ***> Starred stories failure")
|
|
|
|
starred_stories = {}
|
2010-12-12 22:52:15 -05:00
|
|
|
|
2011-01-07 19:06:36 -05:00
|
|
|
# Intelligence classifiers for all feeds involved
|
|
|
|
def sort_by_feed(classifiers):
|
|
|
|
feed_classifiers = defaultdict(list)
|
|
|
|
for classifier in classifiers:
|
|
|
|
feed_classifiers[classifier.feed_id].append(classifier)
|
|
|
|
return feed_classifiers
|
2011-10-24 08:55:28 -07:00
|
|
|
classifiers = {}
|
2011-11-29 17:57:20 -08:00
|
|
|
try:
|
|
|
|
classifier_feeds = sort_by_feed(MClassifierFeed.objects(user_id=user.pk, feed_id__in=found_feed_ids))
|
|
|
|
classifier_authors = sort_by_feed(MClassifierAuthor.objects(user_id=user.pk, feed_id__in=found_feed_ids))
|
|
|
|
classifier_titles = sort_by_feed(MClassifierTitle.objects(user_id=user.pk, feed_id__in=found_feed_ids))
|
|
|
|
classifier_tags = sort_by_feed(MClassifierTag.objects(user_id=user.pk, feed_id__in=found_feed_ids))
|
|
|
|
except OperationFailure:
|
|
|
|
logging.info(" ***> Classifiers failure")
|
|
|
|
else:
|
|
|
|
for feed_id in found_feed_ids:
|
|
|
|
classifiers[feed_id] = get_classifiers_for_user(user, feed_id, classifier_feeds[feed_id],
|
|
|
|
classifier_authors[feed_id],
|
|
|
|
classifier_titles[feed_id],
|
|
|
|
classifier_tags[feed_id])
|
2011-10-24 08:55:28 -07:00
|
|
|
|
2011-01-07 19:20:34 -05:00
|
|
|
# Just need to format stories
|
2010-12-12 22:52:15 -05:00
|
|
|
for story in stories:
|
|
|
|
story_date = localtime_for_timezone(story['story_date'], user.profile.timezone)
|
2011-01-12 23:30:38 -05:00
|
|
|
now = localtime_for_timezone(datetime.datetime.now(), user.profile.timezone)
|
|
|
|
story['short_parsed_date'] = format_story_link_date__short(story_date, now)
|
|
|
|
story['long_parsed_date'] = format_story_link_date__long(story_date, now)
|
2010-12-14 19:23:16 -05:00
|
|
|
story['read_status'] = 0
|
2010-12-12 22:52:15 -05:00
|
|
|
if story['id'] in starred_stories:
|
|
|
|
story['starred'] = True
|
|
|
|
starred_date = localtime_for_timezone(starred_stories[story['id']], user.profile.timezone)
|
2011-01-12 23:30:38 -05:00
|
|
|
story['starred_date'] = format_story_link_date__long(starred_date, now)
|
2010-12-12 22:52:15 -05:00
|
|
|
story['intelligence'] = {
|
2011-01-07 19:20:34 -05:00
|
|
|
'feed': apply_classifier_feeds(classifier_feeds[story['story_feed_id']], story['story_feed_id']),
|
2011-01-07 19:06:36 -05:00
|
|
|
'author': apply_classifier_authors(classifier_authors[story['story_feed_id']], story),
|
2011-01-07 19:20:34 -05:00
|
|
|
'tags': apply_classifier_tags(classifier_tags[story['story_feed_id']], story),
|
|
|
|
'title': apply_classifier_titles(classifier_titles[story['story_feed_id']], story),
|
2010-12-12 22:52:15 -05:00
|
|
|
}
|
2011-11-25 11:58:40 -05:00
|
|
|
|
|
|
|
diff = time.time() - start
|
|
|
|
timediff = round(float(diff), 2)
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FCLoading river stories: page %s - ~SB%s/%s "
|
2011-02-23 13:46:47 -05:00
|
|
|
"stories ~SN(%s/%s/%s feeds) ~FB(%s seconds)" %
|
|
|
|
(page, len(stories), len(mstories), len(found_feed_ids),
|
|
|
|
len(feed_ids), len(original_feed_ids), timediff))
|
2010-12-12 22:52:15 -05:00
|
|
|
|
2011-11-25 11:58:40 -05:00
|
|
|
return dict(stories=stories, classifiers=classifiers, elapsed_time=timediff)
|
2011-01-10 09:49:26 -05:00
|
|
|
|
|
|
|
|
2010-07-24 15:54:25 -04:00
|
|
|
@ajax_login_required
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2010-06-14 01:01:21 -04:00
|
|
|
def mark_all_as_read(request):
|
|
|
|
code = 1
|
2011-04-24 00:05:39 -04:00
|
|
|
days = int(request.POST.get('days', 0))
|
2010-06-14 01:01:21 -04:00
|
|
|
|
|
|
|
feeds = UserSubscription.objects.filter(user=request.user)
|
|
|
|
for sub in feeds:
|
|
|
|
if days == 0:
|
|
|
|
sub.mark_feed_read()
|
|
|
|
else:
|
2010-09-16 22:04:18 -04:00
|
|
|
read_date = datetime.datetime.utcnow() - datetime.timedelta(days=days)
|
2010-06-30 17:17:07 -04:00
|
|
|
if sub.mark_read_date < read_date:
|
|
|
|
sub.needs_unread_recalc = True
|
|
|
|
sub.mark_read_date = read_date
|
|
|
|
sub.save()
|
2010-06-14 01:01:21 -04:00
|
|
|
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FMMarking all as read: ~SB%s days" % (days,))
|
2010-07-25 23:13:27 -04:00
|
|
|
return dict(code=code)
|
2010-06-14 01:01:21 -04:00
|
|
|
|
2010-07-24 15:54:25 -04:00
|
|
|
@ajax_login_required
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2009-06-16 03:08:55 +00:00
|
|
|
def mark_story_as_read(request):
|
2010-08-23 08:22:10 -04:00
|
|
|
story_ids = request.REQUEST.getlist('story_id')
|
2011-11-01 10:32:11 -07:00
|
|
|
feed_id = int(get_argument_or_404(request, 'feed_id'))
|
2010-11-09 09:55:44 -05:00
|
|
|
|
|
|
|
try:
|
|
|
|
usersub = UserSubscription.objects.select_related('feed').get(user=request.user, feed=feed_id)
|
2011-02-09 18:52:36 -05:00
|
|
|
except (UserSubscription.DoesNotExist, Feed.DoesNotExist):
|
2010-11-09 09:55:44 -05:00
|
|
|
duplicate_feed = DuplicateFeed.objects.filter(duplicate_feed_id=feed_id)
|
|
|
|
if duplicate_feed:
|
|
|
|
try:
|
|
|
|
usersub = UserSubscription.objects.get(user=request.user,
|
|
|
|
feed=duplicate_feed[0].feed)
|
2011-02-09 18:52:36 -05:00
|
|
|
except (UserSubscription.DoesNotExist, Feed.DoesNotExist):
|
2010-11-09 09:55:44 -05:00
|
|
|
return dict(code=-1)
|
2011-02-22 11:27:05 -05:00
|
|
|
else:
|
|
|
|
return dict(code=-1)
|
2010-02-17 03:22:45 -05:00
|
|
|
|
2011-11-05 16:25:04 -07:00
|
|
|
data = usersub.mark_story_ids_as_read(story_ids, request=request)
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
return data
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2011-11-05 16:25:04 -07:00
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def mark_feed_stories_as_read(request):
|
2011-11-10 18:29:41 -08:00
|
|
|
feeds_stories = request.REQUEST.get('feeds_stories', "{}")
|
|
|
|
feeds_stories = json.decode(feeds_stories)
|
2011-11-05 16:25:04 -07:00
|
|
|
for feed_id, story_ids in feeds_stories.items():
|
2011-11-10 18:29:41 -08:00
|
|
|
feed_id = int(feed_id)
|
2011-11-05 16:25:04 -07:00
|
|
|
try:
|
|
|
|
usersub = UserSubscription.objects.select_related('feed').get(user=request.user, feed=feed_id)
|
|
|
|
except (UserSubscription.DoesNotExist, Feed.DoesNotExist):
|
|
|
|
duplicate_feed = DuplicateFeed.objects.filter(duplicate_feed_id=feed_id)
|
|
|
|
if duplicate_feed:
|
|
|
|
try:
|
|
|
|
usersub = UserSubscription.objects.get(user=request.user,
|
|
|
|
feed=duplicate_feed[0].feed)
|
|
|
|
except (UserSubscription.DoesNotExist, Feed.DoesNotExist):
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
continue
|
|
|
|
usersub.mark_story_ids_as_read(story_ids)
|
|
|
|
|
|
|
|
return dict(code=1)
|
|
|
|
|
2010-12-30 19:24:52 -05:00
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def mark_story_as_unread(request):
|
2010-12-31 10:34:31 -05:00
|
|
|
story_id = request.POST['story_id']
|
|
|
|
feed_id = int(request.POST['feed_id'])
|
|
|
|
|
2011-11-08 19:12:56 -08:00
|
|
|
usersub = UserSubscription.objects.select_related('feed').get(user=request.user, feed=feed_id)
|
2010-12-31 10:34:31 -05:00
|
|
|
|
|
|
|
if not usersub.needs_unread_recalc:
|
|
|
|
usersub.needs_unread_recalc = True
|
|
|
|
usersub.save()
|
|
|
|
|
|
|
|
data = dict(code=0, payload=dict(story_id=story_id))
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FY~SBUnread~SN story in feed: %s" % (usersub.feed))
|
2011-11-08 19:12:56 -08:00
|
|
|
|
2010-12-31 10:34:31 -05:00
|
|
|
story = MStory.objects(story_feed_id=feed_id, story_guid=story_id)[0]
|
2011-11-08 19:12:56 -08:00
|
|
|
|
|
|
|
if story.story_date < usersub.mark_read_date:
|
|
|
|
# Story is outside the mark as read range, so invert all stories before.
|
2011-11-09 09:51:42 -08:00
|
|
|
newer_stories = MStory.objects(story_feed_id=story.story_feed_id,
|
2011-11-08 19:12:56 -08:00
|
|
|
story_date__gte=story.story_date,
|
|
|
|
story_date__lte=usersub.mark_read_date
|
|
|
|
).only('story_guid')
|
|
|
|
newer_stories = [s.story_guid for s in newer_stories]
|
|
|
|
usersub.mark_read_date = story.story_date - datetime.timedelta(minutes=1)
|
|
|
|
usersub.needs_unread_recalc = True
|
|
|
|
usersub.save()
|
|
|
|
|
|
|
|
# Mark stories as read only after the mark_read_date has been moved, otherwise
|
|
|
|
# these would be ignored.
|
|
|
|
data = usersub.mark_story_ids_as_read(newer_stories, request=request)
|
|
|
|
|
|
|
|
m = MUserStory.objects(story_id=story_id, user_id=request.user.pk, feed_id=feed_id)
|
2010-12-31 10:34:31 -05:00
|
|
|
m.delete()
|
|
|
|
|
|
|
|
return data
|
2010-12-30 19:24:52 -05:00
|
|
|
|
2010-07-24 15:54:25 -04:00
|
|
|
@ajax_login_required
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2009-06-16 03:08:55 +00:00
|
|
|
def mark_feed_as_read(request):
|
2011-02-09 18:52:36 -05:00
|
|
|
feed_ids = [int(f) for f in request.REQUEST.getlist('feed_id') if f]
|
2011-10-31 18:40:07 -07:00
|
|
|
feed_count = len(feed_ids)
|
|
|
|
multiple = feed_count > 1
|
2010-10-06 20:58:29 -04:00
|
|
|
code = 0
|
2010-09-16 10:35:36 -04:00
|
|
|
for feed_id in feed_ids:
|
2010-11-05 20:34:17 -04:00
|
|
|
try:
|
|
|
|
feed = Feed.objects.get(id=feed_id)
|
|
|
|
except Feed.DoesNotExist:
|
|
|
|
continue
|
2010-09-16 10:35:36 -04:00
|
|
|
code = 0
|
2009-06-16 03:08:55 +00:00
|
|
|
|
2010-09-16 10:35:36 -04:00
|
|
|
us = UserSubscription.objects.get(feed=feed, user=request.user)
|
|
|
|
try:
|
2011-11-04 17:52:26 -07:00
|
|
|
if us:
|
|
|
|
us.mark_feed_read()
|
2010-09-16 10:35:36 -04:00
|
|
|
except IntegrityError:
|
|
|
|
code = -1
|
|
|
|
else:
|
|
|
|
code = 1
|
2009-09-08 04:37:38 +00:00
|
|
|
|
2011-10-31 18:40:07 -07:00
|
|
|
if not multiple:
|
|
|
|
logging.user(request, "~FMMarking feed as read: ~SB%s" % (feed,))
|
|
|
|
|
|
|
|
if multiple:
|
|
|
|
logging.user(request, "~FMMarking ~SB%s~SN feeds as read" % (feed_count,))
|
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
return dict(code=code)
|
2010-08-23 09:55:51 -04:00
|
|
|
|
2009-06-16 03:08:55 +00:00
|
|
|
def _parse_user_info(user):
|
|
|
|
return {
|
|
|
|
'user_info': {
|
2009-08-29 19:34:42 +00:00
|
|
|
'is_anonymous': json.encode(user.is_anonymous()),
|
|
|
|
'is_authenticated': json.encode(user.is_authenticated()),
|
|
|
|
'username': json.encode(user.username if user.is_authenticated() else 'Anonymous')
|
2009-06-16 03:08:55 +00:00
|
|
|
}
|
2009-06-22 15:28:20 +00:00
|
|
|
}
|
2010-03-28 17:06:19 -04:00
|
|
|
|
2010-07-24 15:54:25 -04:00
|
|
|
@ajax_login_required
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2010-04-06 20:41:00 -04:00
|
|
|
def add_url(request):
|
|
|
|
code = 0
|
|
|
|
url = request.POST['url']
|
2011-05-07 21:12:13 -04:00
|
|
|
if not url:
|
|
|
|
code = -1
|
|
|
|
message = 'Enter in the website address or the feed URL.'
|
|
|
|
else:
|
|
|
|
folder = request.POST.get('folder', '')
|
|
|
|
code, message, _ = UserSubscription.add_subscription(user=request.user, feed_address=url, folder=folder)
|
2010-04-06 20:41:00 -04:00
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
return dict(code=code, message=message)
|
2010-04-06 20:41:00 -04:00
|
|
|
|
2010-07-24 15:54:25 -04:00
|
|
|
@ajax_login_required
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2010-04-06 20:41:00 -04:00
|
|
|
def add_folder(request):
|
|
|
|
folder = request.POST['folder']
|
2011-04-24 00:05:39 -04:00
|
|
|
parent_folder = request.POST.get('parent_folder', '')
|
2011-10-10 09:57:54 -07:00
|
|
|
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FRAdding Folder: ~SB%s (in %s)" % (folder, parent_folder))
|
2010-08-17 17:45:51 -04:00
|
|
|
|
2010-04-06 20:41:00 -04:00
|
|
|
if folder:
|
|
|
|
code = 1
|
|
|
|
message = ""
|
|
|
|
user_sub_folders_object, _ = UserSubscriptionFolders.objects.get_or_create(user=request.user)
|
2011-03-21 10:15:18 -04:00
|
|
|
user_sub_folders_object.add_folder(parent_folder, folder)
|
2010-04-06 20:41:00 -04:00
|
|
|
else:
|
|
|
|
code = -1
|
|
|
|
message = "Gotta write in a folder name."
|
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
return dict(code=code, message=message)
|
2010-08-23 16:23:16 -04:00
|
|
|
|
2010-07-24 15:54:25 -04:00
|
|
|
@ajax_login_required
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2010-03-28 17:06:19 -04:00
|
|
|
def delete_feed(request):
|
|
|
|
feed_id = int(request.POST['feed_id'])
|
2010-09-14 23:47:21 -04:00
|
|
|
in_folder = request.POST.get('in_folder', '')
|
2011-10-20 10:10:09 -07:00
|
|
|
if in_folder == ' ':
|
|
|
|
in_folder = ""
|
2010-09-14 20:49:28 -04:00
|
|
|
|
2010-09-16 10:35:36 -04:00
|
|
|
user_sub_folders = get_object_or_404(UserSubscriptionFolders, user=request.user)
|
|
|
|
user_sub_folders.delete_feed(feed_id, in_folder)
|
|
|
|
|
2011-03-13 16:24:49 -04:00
|
|
|
feed = Feed.objects.filter(pk=feed_id)
|
|
|
|
if feed:
|
|
|
|
feed[0].count_subscribers()
|
|
|
|
|
2010-09-14 20:49:28 -04:00
|
|
|
return dict(code=1)
|
2010-03-28 17:06:19 -04:00
|
|
|
|
2010-09-14 20:49:28 -04:00
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def delete_folder(request):
|
2011-11-30 09:53:00 -08:00
|
|
|
folder_to_delete = request.POST.get('folder_name') or request.POST.get('folder_to_delete')
|
2010-09-16 10:35:36 -04:00
|
|
|
in_folder = request.POST.get('in_folder', '')
|
2011-02-09 18:52:36 -05:00
|
|
|
feed_ids_in_folder = [int(f) for f in request.REQUEST.getlist('feed_id') if f]
|
2010-09-22 10:12:38 -04:00
|
|
|
|
2010-09-16 10:35:36 -04:00
|
|
|
# Works piss poor with duplicate folder titles, if they are both in the same folder.
|
2010-09-22 10:12:38 -04:00
|
|
|
# Deletes all, but only in the same folder parent. But nobody should be doing that, right?
|
2010-09-16 10:35:36 -04:00
|
|
|
user_sub_folders = get_object_or_404(UserSubscriptionFolders, user=request.user)
|
2010-09-22 10:12:38 -04:00
|
|
|
user_sub_folders.delete_folder(folder_to_delete, in_folder, feed_ids_in_folder)
|
2010-05-04 10:39:25 -04:00
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
return dict(code=1)
|
2010-06-12 21:20:06 -04:00
|
|
|
|
2010-12-11 15:26:45 -05:00
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def rename_feed(request):
|
|
|
|
feed = get_object_or_404(Feed, pk=int(request.POST['feed_id']))
|
|
|
|
user_sub = UserSubscription.objects.get(user=request.user, feed=feed)
|
|
|
|
feed_title = request.POST['feed_title']
|
|
|
|
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FRRenaming feed '~SB%s~SN' to: ~SB%s" % (
|
2011-02-23 13:46:47 -05:00
|
|
|
feed.feed_title, feed_title))
|
2010-12-11 17:19:47 -05:00
|
|
|
|
2010-12-11 15:26:45 -05:00
|
|
|
user_sub.user_title = feed_title
|
|
|
|
user_sub.save()
|
|
|
|
|
|
|
|
return dict(code=1)
|
|
|
|
|
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def rename_folder(request):
|
2011-11-30 09:53:00 -08:00
|
|
|
folder_to_rename = request.POST.get('folder_name') or request.POST.get('folder_to_rename')
|
2010-12-11 17:16:12 -05:00
|
|
|
new_folder_name = request.POST['new_folder_name']
|
2010-12-11 15:26:45 -05:00
|
|
|
in_folder = request.POST.get('in_folder', '')
|
2011-11-30 09:53:00 -08:00
|
|
|
code = 0
|
2010-12-11 15:26:45 -05:00
|
|
|
|
|
|
|
# Works piss poor with duplicate folder titles, if they are both in the same folder.
|
|
|
|
# renames all, but only in the same folder parent. But nobody should be doing that, right?
|
2011-11-30 09:53:00 -08:00
|
|
|
if folder_to_rename and new_folder_name:
|
2010-12-11 17:16:12 -05:00
|
|
|
user_sub_folders = get_object_or_404(UserSubscriptionFolders, user=request.user)
|
|
|
|
user_sub_folders.rename_folder(folder_to_rename, new_folder_name, in_folder)
|
2011-11-30 09:53:00 -08:00
|
|
|
code = 1
|
|
|
|
else:
|
|
|
|
code = -1
|
|
|
|
|
|
|
|
return dict(code=code)
|
2010-12-11 15:26:45 -05:00
|
|
|
|
2011-11-07 20:50:46 -08:00
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def move_feed_to_folder(request):
|
|
|
|
feed_id = int(request.POST['feed_id'])
|
|
|
|
in_folder = request.POST.get('in_folder', '')
|
|
|
|
to_folder = request.POST.get('to_folder', '')
|
2011-12-03 18:22:14 -08:00
|
|
|
|
2011-11-07 20:50:46 -08:00
|
|
|
user_sub_folders = get_object_or_404(UserSubscriptionFolders, user=request.user)
|
|
|
|
user_sub_folders = user_sub_folders.move_feed_to_folder(feed_id, in_folder=in_folder, to_folder=to_folder)
|
|
|
|
|
|
|
|
return dict(code=1, folders=json.decode(user_sub_folders.folders))
|
|
|
|
|
2011-11-08 09:20:10 -08:00
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def move_folder_to_folder(request):
|
|
|
|
folder_name = request.POST['folder_name']
|
|
|
|
in_folder = request.POST.get('in_folder', '')
|
|
|
|
to_folder = request.POST.get('to_folder', '')
|
|
|
|
|
|
|
|
user_sub_folders = get_object_or_404(UserSubscriptionFolders, user=request.user)
|
|
|
|
user_sub_folders = user_sub_folders.move_folder_to_folder(folder_name, in_folder=in_folder, to_folder=to_folder)
|
|
|
|
|
|
|
|
return dict(code=1, folders=json.decode(user_sub_folders.folders))
|
|
|
|
|
2010-06-12 21:20:06 -04:00
|
|
|
@login_required
|
|
|
|
def add_feature(request):
|
|
|
|
if not request.user.is_staff:
|
|
|
|
return HttpResponseForbidden()
|
|
|
|
|
|
|
|
code = -1
|
|
|
|
form = FeatureForm(request.POST)
|
|
|
|
|
|
|
|
if form.is_valid():
|
|
|
|
form.save()
|
|
|
|
code = 1
|
|
|
|
return HttpResponseRedirect(reverse('index'))
|
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
return dict(code=code)
|
2010-06-30 12:17:22 -04:00
|
|
|
|
2010-07-25 23:13:27 -04:00
|
|
|
@json.json_view
|
2010-06-30 12:17:22 -04:00
|
|
|
def load_features(request):
|
2011-03-06 12:42:32 -05:00
|
|
|
user = get_user(request)
|
2011-04-20 09:35:59 -04:00
|
|
|
page = int(request.REQUEST.get('page', 0))
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FBBrowse features: ~SBPage #%s" % (page+1))
|
2010-06-30 16:18:55 -04:00
|
|
|
features = Feature.objects.all()[page*3:(page+1)*3+1].values()
|
2010-08-23 16:23:16 -04:00
|
|
|
features = [{
|
|
|
|
'description': f['description'],
|
2011-03-06 12:42:32 -05:00
|
|
|
'date': localtime_for_timezone(f['date'], user.profile.timezone).strftime("%b %d, %Y")
|
2010-08-23 16:23:16 -04:00
|
|
|
} for f in features]
|
2010-07-25 23:13:27 -04:00
|
|
|
return features
|
2010-07-11 11:10:45 -04:00
|
|
|
|
2010-11-05 10:35:52 -04:00
|
|
|
@ajax_login_required
|
2010-07-11 11:10:45 -04:00
|
|
|
@json.json_view
|
|
|
|
def save_feed_order(request):
|
|
|
|
folders = request.POST.get('folders')
|
|
|
|
if folders:
|
|
|
|
# Test that folders can be JSON decoded
|
|
|
|
folders_list = json.decode(folders)
|
2010-07-20 20:23:49 -04:00
|
|
|
assert folders_list is not None
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FBFeed re-ordering: ~SB%s folders/feeds" % (len(folders_list)))
|
2010-07-11 11:10:45 -04:00
|
|
|
user_sub_folders = UserSubscriptionFolders.objects.get(user=request.user)
|
|
|
|
user_sub_folders.folders = folders
|
|
|
|
user_sub_folders.save()
|
|
|
|
|
2010-07-20 23:59:56 -04:00
|
|
|
return {}
|
|
|
|
|
2010-08-01 19:12:42 -04:00
|
|
|
@json.json_view
|
2011-04-24 21:27:31 -04:00
|
|
|
def feeds_trainer(request):
|
2010-08-01 19:12:42 -04:00
|
|
|
classifiers = []
|
2011-04-24 21:27:31 -04:00
|
|
|
feed_id = request.REQUEST.get('feed_id')
|
2010-12-04 22:01:26 -05:00
|
|
|
user = get_user(request)
|
|
|
|
usersubs = UserSubscription.objects.filter(user=user, active=True)
|
2010-09-17 12:40:42 -04:00
|
|
|
if feed_id:
|
|
|
|
feed = get_object_or_404(Feed, pk=feed_id)
|
|
|
|
usersubs = usersubs.filter(feed=feed)
|
|
|
|
usersubs = usersubs.select_related('feed').order_by('-feed__stories_last_month')
|
2010-08-01 19:12:42 -04:00
|
|
|
|
|
|
|
for us in usersubs:
|
2010-09-17 12:40:42 -04:00
|
|
|
if (not us.is_trained and us.feed.stories_last_month > 0) or feed_id:
|
2010-08-01 19:12:42 -04:00
|
|
|
classifier = dict()
|
2010-12-04 22:01:26 -05:00
|
|
|
classifier['classifiers'] = get_classifiers_for_user(user, us.feed.pk)
|
2010-08-01 19:12:42 -04:00
|
|
|
classifier['feed_id'] = us.feed.pk
|
|
|
|
classifier['stories_last_month'] = us.feed.stories_last_month
|
2011-07-27 22:17:34 -07:00
|
|
|
classifier['num_subscribers'] = us.feed.num_subscribers
|
2011-01-17 22:48:38 -05:00
|
|
|
classifier['feed_tags'] = json.decode(us.feed.data.popular_tags) if us.feed.data.popular_tags else []
|
|
|
|
classifier['feed_authors'] = json.decode(us.feed.data.popular_authors) if us.feed.data.popular_authors else []
|
2010-08-01 19:12:42 -04:00
|
|
|
classifiers.append(classifier)
|
|
|
|
|
2011-02-23 13:46:47 -05:00
|
|
|
logging.user(user, "~FGLoading Trainer: ~SB%s feeds" % (len(classifiers)))
|
2010-08-02 23:09:47 -04:00
|
|
|
|
2010-08-01 19:12:42 -04:00
|
|
|
return classifiers
|
2010-08-13 19:21:29 -04:00
|
|
|
|
2010-09-28 18:53:57 -04:00
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def save_feed_chooser(request):
|
2011-02-03 18:38:10 -05:00
|
|
|
approved_feeds = [int(feed_id) for feed_id in request.POST.getlist('approved_feeds') if feed_id][:64]
|
2010-09-28 18:53:57 -04:00
|
|
|
activated = 0
|
|
|
|
usersubs = UserSubscription.objects.filter(user=request.user)
|
2010-11-12 10:55:44 -05:00
|
|
|
|
2010-09-28 18:53:57 -04:00
|
|
|
for sub in usersubs:
|
2010-11-12 10:55:44 -05:00
|
|
|
try:
|
|
|
|
if sub.feed.pk in approved_feeds:
|
|
|
|
activated += 1
|
2011-11-28 18:01:39 -08:00
|
|
|
if not sub.active:
|
|
|
|
sub.active = True
|
|
|
|
sub.save()
|
|
|
|
sub.feed.count_subscribers()
|
2010-11-12 10:55:44 -05:00
|
|
|
elif sub.active:
|
|
|
|
sub.active = False
|
|
|
|
sub.save()
|
|
|
|
except Feed.DoesNotExist:
|
|
|
|
pass
|
2010-10-26 14:04:26 -04:00
|
|
|
|
|
|
|
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~BB~FW~SBActivated standard account: ~FC%s~SN/~SB%s" % (
|
2011-02-23 13:46:47 -05:00
|
|
|
activated,
|
|
|
|
usersubs.count()
|
|
|
|
))
|
2011-01-30 23:56:51 -05:00
|
|
|
request.user.profile.queue_new_feeds()
|
|
|
|
request.user.profile.refresh_stale_feeds(exclude_new=True)
|
2011-01-30 15:24:40 -05:00
|
|
|
|
2010-09-28 18:53:57 -04:00
|
|
|
return {'activated': activated}
|
2010-10-05 19:05:01 -04:00
|
|
|
|
2010-10-29 11:34:33 -04:00
|
|
|
@ajax_login_required
|
|
|
|
def retrain_all_sites(request):
|
|
|
|
for sub in UserSubscription.objects.filter(user=request.user):
|
|
|
|
sub.is_trained = False
|
|
|
|
sub.save()
|
|
|
|
|
2011-04-24 21:27:31 -04:00
|
|
|
return feeds_trainer(request)
|
2010-10-29 11:34:33 -04:00
|
|
|
|
2010-10-05 19:05:01 -04:00
|
|
|
@login_required
|
|
|
|
def activate_premium_account(request):
|
2010-10-19 19:09:08 -04:00
|
|
|
try:
|
|
|
|
usersubs = UserSubscription.objects.select_related('feed').filter(user=request.user)
|
|
|
|
for sub in usersubs:
|
|
|
|
sub.active = True
|
|
|
|
sub.save()
|
|
|
|
if sub.feed.premium_subscribers <= 0:
|
|
|
|
sub.feed.count_subscribers()
|
|
|
|
sub.feed.schedule_feed_fetch_immediately()
|
|
|
|
except Exception, e:
|
|
|
|
subject = "Premium activation failed"
|
|
|
|
message = "%s -- %s\n\n%s" % (request.user, usersubs, e)
|
|
|
|
mail_admins(subject, message, fail_silently=True)
|
|
|
|
|
2010-10-05 19:05:01 -04:00
|
|
|
request.user.profile.is_premium = True
|
|
|
|
request.user.profile.save()
|
|
|
|
|
|
|
|
return HttpResponseRedirect(reverse('index'))
|
|
|
|
|
2010-10-29 11:34:33 -04:00
|
|
|
@login_required
|
|
|
|
def login_as(request):
|
|
|
|
if not request.user.is_staff:
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~SKNON-STAFF LOGGING IN AS ANOTHER USER!")
|
2010-10-29 11:34:33 -04:00
|
|
|
assert False
|
|
|
|
return HttpResponseForbidden()
|
|
|
|
username = request.GET['user']
|
2011-02-22 09:37:09 -05:00
|
|
|
user = get_object_or_404(User, username__iexact=username)
|
2010-10-29 11:34:33 -04:00
|
|
|
user.backend = settings.AUTHENTICATION_BACKENDS[0]
|
|
|
|
login_user(request, user)
|
|
|
|
return HttpResponseRedirect(reverse('index'))
|
|
|
|
|
2010-08-13 19:21:29 -04:00
|
|
|
def iframe_buster(request):
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FB~SBiFrame bust!")
|
2010-11-30 10:30:18 -05:00
|
|
|
return HttpResponse(status=204)
|
|
|
|
|
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def mark_story_as_starred(request):
|
|
|
|
code = 1
|
|
|
|
feed_id = int(request.POST['feed_id'])
|
|
|
|
story_id = request.POST['story_id']
|
|
|
|
|
|
|
|
story = MStory.objects(story_feed_id=feed_id, story_guid=story_id).limit(1)
|
|
|
|
if story:
|
|
|
|
story_db = dict([(k, v) for k, v in story[0]._data.items()
|
|
|
|
if k is not None and v is not None])
|
2010-12-02 20:18:33 -05:00
|
|
|
now = datetime.datetime.now()
|
|
|
|
story_values = dict(user_id=request.user.pk, starred_date=now, **story_db)
|
2010-11-30 10:30:18 -05:00
|
|
|
MStarredStory.objects.create(**story_values)
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FCStarring: ~SB%s" % (story[0].story_title[:50]))
|
2010-11-30 10:30:18 -05:00
|
|
|
else:
|
|
|
|
code = -1
|
|
|
|
|
|
|
|
return {'code': code}
|
|
|
|
|
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def mark_story_as_unstarred(request):
|
|
|
|
code = 1
|
|
|
|
story_id = request.POST['story_id']
|
|
|
|
|
2010-12-02 20:18:33 -05:00
|
|
|
starred_story = MStarredStory.objects(user_id=request.user.pk, story_guid=story_id)
|
2010-11-30 10:30:18 -05:00
|
|
|
if starred_story:
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, "~FCUnstarring: ~SB%s" % (starred_story[0].story_title[:50]))
|
2010-12-04 16:42:51 -05:00
|
|
|
starred_story.delete()
|
2010-11-30 10:30:18 -05:00
|
|
|
else:
|
|
|
|
code = -1
|
|
|
|
|
2010-12-04 13:32:13 -05:00
|
|
|
return {'code': code}
|
2011-05-06 10:06:13 -04:00
|
|
|
|
|
|
|
@ajax_login_required
|
|
|
|
@json.json_view
|
|
|
|
def send_story_email(request):
|
|
|
|
code = 1
|
2011-05-07 17:58:53 -04:00
|
|
|
message = 'OK'
|
2011-05-06 10:06:13 -04:00
|
|
|
story_id = request.POST['story_id']
|
|
|
|
feed_id = request.POST['feed_id']
|
|
|
|
to_address = request.POST['to']
|
2011-05-07 21:12:13 -04:00
|
|
|
from_name = request.POST['from_name']
|
|
|
|
from_email = request.POST['from_email']
|
2011-05-06 10:06:13 -04:00
|
|
|
comments = request.POST['comments']
|
2011-05-09 10:17:14 -04:00
|
|
|
comments = comments[:2048] # Separated due to PyLint
|
2011-05-08 20:32:02 -04:00
|
|
|
from_address = 'share@newsblur.com'
|
2011-05-07 17:58:53 -04:00
|
|
|
|
|
|
|
if not email_re.match(to_address):
|
|
|
|
code = -1
|
|
|
|
message = 'You need to send the email to a valid email address.'
|
2011-05-09 10:17:14 -04:00
|
|
|
elif not email_re.match(from_email):
|
2011-05-08 20:34:52 -04:00
|
|
|
code = -1
|
|
|
|
message = 'You need to provide your email address.'
|
2011-05-09 10:17:14 -04:00
|
|
|
elif not from_name:
|
2011-05-08 20:34:52 -04:00
|
|
|
code = -1
|
|
|
|
message = 'You need to provide your name.'
|
2011-05-07 17:58:53 -04:00
|
|
|
else:
|
|
|
|
story = MStory.objects(story_feed_id=feed_id, story_guid=story_id)[0]
|
2011-05-08 19:41:50 -04:00
|
|
|
story = Feed.format_story(story, feed_id, text=True)
|
|
|
|
feed = Feed.objects.get(pk=story['story_feed_id'])
|
2011-05-07 17:58:53 -04:00
|
|
|
text = render_to_string('mail/email_story_text.xhtml', locals())
|
|
|
|
html = render_to_string('mail/email_story_html.xhtml', locals())
|
2011-05-08 19:41:50 -04:00
|
|
|
subject = "%s is sharing a story with you: \"%s\"" % (from_name, story['story_title'])
|
2011-11-11 11:26:34 -08:00
|
|
|
subject = subject.replace('\n', ' ')
|
2011-05-08 20:32:02 -04:00
|
|
|
msg = EmailMultiAlternatives(subject, text,
|
2011-05-08 20:34:52 -04:00
|
|
|
from_email='NewsBlur <%s>' % from_address,
|
2011-05-08 20:32:02 -04:00
|
|
|
to=[to_address],
|
2011-05-08 20:21:09 -04:00
|
|
|
cc=['%s <%s>' % (from_name, from_email)],
|
|
|
|
headers={'Reply-To': '%s <%s>' % (from_name, from_email)})
|
2011-05-07 17:58:53 -04:00
|
|
|
msg.attach_alternative(html, "text/html")
|
|
|
|
msg.send()
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, '~BMSharing story by email: ~FY~SB%s~SN~BM~FY/~SB%s' %
|
2011-05-09 10:17:14 -04:00
|
|
|
(story['story_title'][:50], feed.feed_title[:50]))
|
2011-05-07 17:58:53 -04:00
|
|
|
|
2011-05-07 21:12:24 -04:00
|
|
|
return {'code': code, 'message': message}
|
2011-05-18 22:33:42 -04:00
|
|
|
|
|
|
|
@json.json_view
|
|
|
|
def load_tutorial(request):
|
2011-05-20 00:18:23 -04:00
|
|
|
if request.REQUEST.get('finished'):
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, '~BY~FW~SBFinishing Tutorial')
|
2011-05-20 00:18:23 -04:00
|
|
|
return {}
|
|
|
|
else:
|
|
|
|
newsblur_feed = Feed.objects.filter(feed_address__icontains='blog.newsblur.com').order_by('-pk')[0]
|
2011-09-16 09:26:22 -07:00
|
|
|
logging.user(request, '~BY~FW~SBLoading Tutorial')
|
2011-05-20 00:18:23 -04:00
|
|
|
return {
|
|
|
|
'newsblur_feed': newsblur_feed.canonical()
|
|
|
|
}
|