2024-01-03 20:14:39 +13:00
|
|
|
from datetime import datetime, timedelta
|
|
|
|
from math import log
|
|
|
|
|
2023-12-17 00:12:49 +13:00
|
|
|
from sqlalchemy.sql.operators import or_
|
2023-08-10 21:13:37 +12:00
|
|
|
|
2023-12-03 22:41:15 +13:00
|
|
|
from app import db, cache
|
2023-12-27 15:47:17 +13:00
|
|
|
from app.activitypub.util import default_context, make_image_sizes_async, refresh_user_profile
|
2023-12-17 00:12:49 +13:00
|
|
|
from app.constants import SUBSCRIPTION_PENDING, SUBSCRIPTION_MEMBER, POST_TYPE_IMAGE, POST_TYPE_LINK, SUBSCRIPTION_OWNER
|
2023-07-28 16:22:12 +12:00
|
|
|
from app.main import bp
|
2023-12-21 22:14:43 +13:00
|
|
|
from flask import g, session, flash, request, current_app, url_for, redirect, make_response, jsonify
|
2023-07-28 16:22:12 +12:00
|
|
|
from flask_moment import moment
|
2023-08-26 15:41:11 +12:00
|
|
|
from flask_login import current_user
|
2023-07-28 16:22:12 +12:00
|
|
|
from flask_babel import _, get_locale
|
2023-12-17 00:12:49 +13:00
|
|
|
from sqlalchemy import select, desc
|
2023-09-05 20:25:02 +12:00
|
|
|
from sqlalchemy_searchable import search
|
2023-12-21 22:14:43 +13:00
|
|
|
from app.utils import render_template, get_setting, gibberish, request_etag_matches, return_304, blocked_domains, \
|
2024-01-01 16:26:57 +13:00
|
|
|
ap_datetime, ip_address
|
2024-01-03 20:14:39 +13:00
|
|
|
from app.models import Community, CommunityMember, Post, Site, User, utcnow
|
2023-08-22 21:24:11 +12:00
|
|
|
|
2023-07-28 16:22:12 +12:00
|
|
|
|
2023-12-21 22:14:43 +13:00
|
|
|
@bp.route('/', methods=['HEAD', 'GET', 'POST'])
|
2023-07-28 16:22:12 +12:00
|
|
|
@bp.route('/index', methods=['GET', 'POST'])
|
|
|
|
def index():
|
2023-12-21 22:14:43 +13:00
|
|
|
if 'application/ld+json' in request.headers.get('Accept', '') or 'application/activity+json' in request.headers.get(
|
|
|
|
'Accept', ''):
|
|
|
|
return activitypub_application()
|
|
|
|
|
2023-10-23 13:03:35 +13:00
|
|
|
verification_warning()
|
2023-12-17 00:12:49 +13:00
|
|
|
|
|
|
|
# If nothing has changed since their last visit, return HTTP 304
|
|
|
|
current_etag = f"home_{hash(str(g.site.last_active))}"
|
|
|
|
if current_user.is_anonymous and request_etag_matches(current_etag):
|
|
|
|
return return_304(current_etag)
|
|
|
|
|
|
|
|
page = request.args.get('page', 1, type=int)
|
|
|
|
|
|
|
|
if current_user.is_anonymous:
|
|
|
|
posts = Post.query.filter(Post.from_bot == False, Post.nsfw == False, Post.nsfl == False)
|
|
|
|
else:
|
|
|
|
posts = Post.query.join(CommunityMember, Post.community_id == CommunityMember.community_id).filter(CommunityMember.is_banned == False)
|
|
|
|
posts = posts.join(User, CommunityMember.user_id == User.id).filter(User.id == current_user.id)
|
|
|
|
domains_ids = blocked_domains(current_user.id)
|
|
|
|
if domains_ids:
|
|
|
|
posts = posts.filter(or_(Post.domain_id.not_in(domains_ids), Post.domain_id == None))
|
|
|
|
|
2024-01-03 20:14:39 +13:00
|
|
|
posts = posts.order_by(desc(Post.ranking)).paginate(page=page, per_page=100, error_out=False)
|
2023-12-17 00:12:49 +13:00
|
|
|
|
|
|
|
next_url = url_for('main.index', page=posts.next_num) if posts.has_next else None
|
|
|
|
prev_url = url_for('main.index', page=posts.prev_num) if posts.has_prev and page != 1 else None
|
|
|
|
|
|
|
|
active_communities = Community.query.filter_by(banned=False).order_by(desc(Community.last_active)).limit(5).all()
|
|
|
|
|
|
|
|
return render_template('index.html', posts=posts, active_communities=active_communities,
|
|
|
|
POST_TYPE_IMAGE=POST_TYPE_IMAGE, POST_TYPE_LINK=POST_TYPE_LINK,
|
|
|
|
SUBSCRIPTION_PENDING=SUBSCRIPTION_PENDING, SUBSCRIPTION_MEMBER=SUBSCRIPTION_MEMBER,
|
|
|
|
etag=f"home_{hash(str(g.site.last_active))}", next_url=next_url, prev_url=prev_url,
|
|
|
|
rss_feed=f"https://{current_app.config['SERVER_NAME']}/feed", rss_feed_name=f"Posts on " + g.site.name)
|
2023-08-22 21:24:11 +12:00
|
|
|
|
|
|
|
|
2024-01-03 20:14:39 +13:00
|
|
|
@bp.route('/new', methods=['HEAD', 'GET', 'POST'])
|
|
|
|
def new_posts():
|
|
|
|
verification_warning()
|
|
|
|
|
|
|
|
# If nothing has changed since their last visit, return HTTP 304
|
|
|
|
current_etag = f"new_{hash(str(g.site.last_active))}"
|
|
|
|
if current_user.is_anonymous and request_etag_matches(current_etag):
|
|
|
|
return return_304(current_etag)
|
|
|
|
|
|
|
|
page = request.args.get('page', 1, type=int)
|
|
|
|
|
|
|
|
if current_user.is_anonymous:
|
|
|
|
posts = Post.query.filter(Post.from_bot == False, Post.nsfw == False, Post.nsfl == False)
|
|
|
|
else:
|
|
|
|
posts = Post.query.join(CommunityMember, Post.community_id == CommunityMember.community_id).filter(
|
|
|
|
CommunityMember.is_banned == False)
|
|
|
|
posts = posts.join(User, CommunityMember.user_id == User.id).filter(User.id == current_user.id)
|
|
|
|
domains_ids = blocked_domains(current_user.id)
|
|
|
|
if domains_ids:
|
|
|
|
posts = posts.filter(or_(Post.domain_id.not_in(domains_ids), Post.domain_id == None))
|
|
|
|
|
|
|
|
posts = posts.order_by(desc(Post.posted_at)).paginate(page=page, per_page=100, error_out=False)
|
|
|
|
|
|
|
|
next_url = url_for('main.new_posts', page=posts.next_num) if posts.has_next else None
|
|
|
|
prev_url = url_for('main.new_posts', page=posts.prev_num) if posts.has_prev and page != 1 else None
|
|
|
|
|
|
|
|
active_communities = Community.query.filter_by(banned=False).order_by(desc(Community.last_active)).limit(5).all()
|
|
|
|
|
|
|
|
return render_template('new_posts.html', posts=posts, active_communities=active_communities,
|
|
|
|
POST_TYPE_IMAGE=POST_TYPE_IMAGE, POST_TYPE_LINK=POST_TYPE_LINK,
|
|
|
|
SUBSCRIPTION_PENDING=SUBSCRIPTION_PENDING, SUBSCRIPTION_MEMBER=SUBSCRIPTION_MEMBER,
|
|
|
|
etag=f"home_{hash(str(g.site.last_active))}", next_url=next_url, prev_url=prev_url,
|
|
|
|
rss_feed=f"https://{current_app.config['SERVER_NAME']}/feed",
|
|
|
|
rss_feed_name=f"Posts on " + g.site.name)
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/top', methods=['HEAD', 'GET', 'POST'])
|
|
|
|
def top_posts():
|
|
|
|
verification_warning()
|
|
|
|
|
|
|
|
# If nothing has changed since their last visit, return HTTP 304
|
|
|
|
current_etag = f"best_{hash(str(g.site.last_active))}"
|
|
|
|
if current_user.is_anonymous and request_etag_matches(current_etag):
|
|
|
|
return return_304(current_etag)
|
|
|
|
|
|
|
|
page = request.args.get('page', 1, type=int)
|
|
|
|
|
|
|
|
if current_user.is_anonymous:
|
|
|
|
posts = Post.query.filter(Post.from_bot == False, Post.nsfw == False, Post.nsfl == False)
|
|
|
|
else:
|
|
|
|
posts = Post.query.join(CommunityMember, Post.community_id == CommunityMember.community_id).filter(
|
|
|
|
CommunityMember.is_banned == False)
|
|
|
|
posts = posts.join(User, CommunityMember.user_id == User.id).filter(User.id == current_user.id)
|
|
|
|
domains_ids = blocked_domains(current_user.id)
|
|
|
|
if domains_ids:
|
|
|
|
posts = posts.filter(or_(Post.domain_id.not_in(domains_ids), Post.domain_id == None))
|
|
|
|
|
|
|
|
posts = posts.filter(Post.posted_at > utcnow() - timedelta(days=1)).order_by(desc(Post.score)).paginate(page=page, per_page=100, error_out=False)
|
|
|
|
|
|
|
|
next_url = url_for('main.top_posts', page=posts.next_num) if posts.has_next else None
|
|
|
|
prev_url = url_for('main.top_posts', page=posts.prev_num) if posts.has_prev and page != 1 else None
|
|
|
|
|
|
|
|
active_communities = Community.query.filter_by(banned=False).order_by(desc(Community.last_active)).limit(5).all()
|
|
|
|
|
|
|
|
return render_template('top_posts.html', posts=posts, active_communities=active_communities,
|
|
|
|
POST_TYPE_IMAGE=POST_TYPE_IMAGE, POST_TYPE_LINK=POST_TYPE_LINK,
|
|
|
|
SUBSCRIPTION_PENDING=SUBSCRIPTION_PENDING, SUBSCRIPTION_MEMBER=SUBSCRIPTION_MEMBER,
|
|
|
|
etag=f"home_{hash(str(g.site.last_active))}", next_url=next_url, prev_url=prev_url,
|
|
|
|
rss_feed=f"https://{current_app.config['SERVER_NAME']}/feed",
|
|
|
|
rss_feed_name=f"Posts on " + g.site.name)
|
|
|
|
|
|
|
|
|
2023-08-22 21:24:11 +12:00
|
|
|
@bp.route('/communities', methods=['GET'])
|
|
|
|
def list_communities():
|
2023-10-23 13:03:35 +13:00
|
|
|
verification_warning()
|
2023-09-05 20:25:02 +12:00
|
|
|
search_param = request.args.get('search', '')
|
|
|
|
if search_param == '':
|
2023-10-07 21:32:19 +13:00
|
|
|
communities = Community.query.filter_by(banned=False).all()
|
2023-09-05 20:25:02 +12:00
|
|
|
else:
|
2024-01-01 11:38:24 +13:00
|
|
|
query = search(select(Community), search_param, sort=True) # todo: exclude banned communities from search
|
2023-09-05 20:25:02 +12:00
|
|
|
communities = db.session.scalars(query).all()
|
|
|
|
|
2023-12-03 22:41:15 +13:00
|
|
|
return render_template('list_communities.html', communities=communities, search=search_param, title=_('Communities'),
|
2023-12-17 00:12:49 +13:00
|
|
|
SUBSCRIPTION_PENDING=SUBSCRIPTION_PENDING, SUBSCRIPTION_MEMBER=SUBSCRIPTION_MEMBER,
|
|
|
|
SUBSCRIPTION_OWNER=SUBSCRIPTION_OWNER)
|
2023-09-05 20:25:02 +12:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/communities/local', methods=['GET'])
|
|
|
|
def list_local_communities():
|
2023-10-23 13:03:35 +13:00
|
|
|
verification_warning()
|
2023-10-07 21:32:19 +13:00
|
|
|
communities = Community.query.filter_by(ap_id=None, banned=False).all()
|
2023-12-03 22:41:15 +13:00
|
|
|
return render_template('list_communities.html', communities=communities, title=_('Local communities'),
|
|
|
|
SUBSCRIPTION_PENDING=SUBSCRIPTION_PENDING, SUBSCRIPTION_MEMBER=SUBSCRIPTION_MEMBER)
|
2023-09-05 20:25:02 +12:00
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/communities/subscribed', methods=['GET'])
|
|
|
|
def list_subscribed_communities():
|
2023-10-23 13:03:35 +13:00
|
|
|
verification_warning()
|
2023-10-07 21:32:19 +13:00
|
|
|
communities = Community.query.filter_by(banned=False).join(CommunityMember).filter(CommunityMember.user_id == current_user.id).all()
|
2023-12-03 22:41:15 +13:00
|
|
|
return render_template('list_communities.html', communities=communities, title=_('Subscribed communities'),
|
|
|
|
SUBSCRIPTION_PENDING=SUBSCRIPTION_PENDING, SUBSCRIPTION_MEMBER=SUBSCRIPTION_MEMBER)
|
|
|
|
|
|
|
|
|
2023-12-17 00:12:49 +13:00
|
|
|
@bp.route('/login')
|
|
|
|
def login():
|
|
|
|
return redirect(url_for('auth.login'))
|
|
|
|
|
|
|
|
|
|
|
|
@bp.route('/robots.txt')
|
|
|
|
def robots():
|
|
|
|
resp = make_response(render_template('robots.txt'))
|
|
|
|
resp.mimetype = 'text/plain'
|
|
|
|
return resp
|
|
|
|
|
|
|
|
|
2023-12-03 22:41:15 +13:00
|
|
|
@bp.route('/test')
|
|
|
|
def test():
|
2024-01-03 20:14:39 +13:00
|
|
|
return 'done'
|
|
|
|
|
|
|
|
#ip = request.headers.get('X-Forwarded-For') or request.remote_addr
|
|
|
|
#if ',' in ip: # Remove all but first ip addresses
|
|
|
|
# ip = ip[:ip.index(',')].strip()
|
|
|
|
#return ip
|
2023-10-23 13:03:35 +13:00
|
|
|
|
|
|
|
|
|
|
|
def verification_warning():
|
|
|
|
if hasattr(current_user, 'verified') and current_user.verified is False:
|
|
|
|
flash(_('Please click the link in your email inbox to verify your account.'), 'warning')
|
2023-12-21 22:14:43 +13:00
|
|
|
|
|
|
|
|
2023-12-24 13:28:41 +13:00
|
|
|
@cache.cached(timeout=6)
|
2023-12-21 22:14:43 +13:00
|
|
|
def activitypub_application():
|
|
|
|
application_data = {
|
|
|
|
'@context': default_context(),
|
|
|
|
'type': 'Application',
|
|
|
|
'id': f"https://{current_app.config['SERVER_NAME']}/",
|
|
|
|
'name': g.site.name,
|
|
|
|
'summary': g.site.description,
|
|
|
|
'published': ap_datetime(g.site.created_at),
|
|
|
|
'updated': ap_datetime(g.site.updated),
|
|
|
|
'inbox': f"https://{current_app.config['SERVER_NAME']}/site_inbox",
|
|
|
|
'outbox': f"https://{current_app.config['SERVER_NAME']}/site_outbox",
|
|
|
|
}
|
|
|
|
resp = jsonify(application_data)
|
|
|
|
resp.content_type = 'application/activity+json'
|
|
|
|
return resp
|