mirror of https://gitlab.com/bashrc2/epicyon
875 lines
37 KiB
Python
875 lines
37 KiB
Python
__filename__ = "outbox.py"
|
|
__author__ = "Bob Mottram"
|
|
__license__ = "AGPL3+"
|
|
__version__ = "1.6.0"
|
|
__maintainer__ = "Bob Mottram"
|
|
__email__ = "bob@libreserver.org"
|
|
__status__ = "Production"
|
|
__module_group__ = "Timeline"
|
|
|
|
import os
|
|
from shutil import copyfile
|
|
from auth import create_password
|
|
from posts import is_image_media
|
|
from posts import outbox_message_create_wrap
|
|
from posts import save_post_to_box
|
|
from posts import send_to_followers_thread
|
|
from posts import send_to_named_addresses_thread
|
|
from flags import is_featured_writer
|
|
from flags import is_quote_toot
|
|
from utils import data_dir
|
|
from utils import quote_toots_allowed
|
|
from utils import get_post_attachments
|
|
from utils import get_attributed_to
|
|
from utils import contains_invalid_actor_url_chars
|
|
from utils import get_attachment_property_value
|
|
from utils import get_account_timezone
|
|
from utils import has_object_string_type
|
|
from utils import get_base_content_from_post
|
|
from utils import has_object_dict
|
|
from utils import get_local_network_addresses
|
|
from utils import get_full_domain
|
|
from utils import remove_id_ending
|
|
from utils import get_domain_from_actor
|
|
from utils import dangerous_markup
|
|
from utils import load_json
|
|
from utils import save_json
|
|
from utils import acct_dir
|
|
from utils import local_actor_url
|
|
from utils import has_actor
|
|
from utils import get_actor_from_post
|
|
from blocking import is_blocked_domain
|
|
from blocking import outbox_block
|
|
from blocking import outbox_undo_block
|
|
from blocking import outbox_mute
|
|
from blocking import outbox_undo_mute
|
|
from media import replace_you_tube
|
|
from media import replace_twitter
|
|
from media import get_media_path
|
|
from media import create_media_dirs
|
|
from announce import outbox_announce
|
|
from announce import outbox_undo_announce
|
|
from follow import outbox_undo_follow
|
|
from follow import follower_approval_active
|
|
from skills import outbox_skills
|
|
from availability import outbox_availability
|
|
from like import outbox_like
|
|
from like import outbox_undo_like
|
|
from reaction import outbox_reaction
|
|
from reaction import outbox_undo_reaction
|
|
from bookmarks import outbox_bookmark
|
|
from bookmarks import outbox_undo_bookmark
|
|
from delete import outbox_delete
|
|
from shares import outbox_share_upload
|
|
from shares import outbox_undo_share_upload
|
|
from webapp_post import individual_post_as_html
|
|
from webapp_hashtagswarm import store_hash_tags
|
|
from speaker import update_speaker
|
|
from reading import store_book_events
|
|
from reading import has_edition_tag
|
|
from inbox_receive import inbox_update_index
|
|
|
|
|
|
def _localonly_not_local(message_json: {}, domain_full: str) -> bool:
|
|
"""If this is a "local only" post return true if it is not local
|
|
"""
|
|
# if this is a local only post, is it really local?
|
|
if 'localOnly' in message_json['object'] and \
|
|
message_json['object'].get('to') and \
|
|
message_json['object'].get('attributedTo'):
|
|
if message_json['object']['localOnly'] is True:
|
|
# check that the to addresses are local
|
|
if isinstance(message_json['object']['to'], list):
|
|
for to_actor in message_json['object']['to']:
|
|
to_domain, to_port = get_domain_from_actor(to_actor)
|
|
if not to_domain:
|
|
continue
|
|
to_domain_full = get_full_domain(to_domain, to_port)
|
|
if domain_full != to_domain_full:
|
|
print("REJECT: local only post isn't local " +
|
|
str(message_json))
|
|
return True
|
|
# check that the sender is local
|
|
local_actor = \
|
|
get_attributed_to(message_json['object']['attributedTo'])
|
|
local_domain, local_port = get_domain_from_actor(local_actor)
|
|
if local_domain:
|
|
local_domain_full = \
|
|
get_full_domain(local_domain, local_port)
|
|
if domain_full != local_domain_full:
|
|
print("REJECT: local only post isn't local " +
|
|
str(message_json))
|
|
return True
|
|
return False
|
|
|
|
|
|
def _valid_person_update_outbox(message_json: {}, debug: bool) -> bool:
|
|
"""is an actor update valid?
|
|
"""
|
|
if not message_json.get('type'):
|
|
return False
|
|
if not isinstance(message_json['type'], str):
|
|
if debug:
|
|
print('DEBUG: c2s actor update type is not a string')
|
|
return False
|
|
if message_json['type'] != 'Update':
|
|
return False
|
|
if not has_object_string_type(message_json, debug):
|
|
return False
|
|
if not isinstance(message_json['object']['type'], str):
|
|
if debug:
|
|
print('DEBUG: c2s actor update object type is not a string')
|
|
return False
|
|
if message_json['object']['type'] != 'Person':
|
|
if debug:
|
|
print('DEBUG: not a c2s actor update')
|
|
return False
|
|
if not message_json.get('to'):
|
|
if debug:
|
|
print('DEBUG: c2s actor update has no "to" field')
|
|
return False
|
|
if not has_actor(message_json, debug):
|
|
return False
|
|
if not message_json.get('id'):
|
|
if debug:
|
|
print('DEBUG: c2s actor update has no id field')
|
|
return False
|
|
if not isinstance(message_json['id'], str):
|
|
if debug:
|
|
print('DEBUG: c2s actor update id is not a string')
|
|
return False
|
|
if not isinstance(message_json['to'], list):
|
|
if debug:
|
|
print('DEBUG: c2s actor update - to field is not a list ' +
|
|
str(message_json['to']))
|
|
return False
|
|
if len(message_json['to']) != 1:
|
|
if debug:
|
|
print('DEBUG: c2s actor update - to does not contain one actor ' +
|
|
str(message_json['to']))
|
|
return False
|
|
return True
|
|
|
|
|
|
def _person_receive_update_outbox(base_dir: str, http_prefix: str,
|
|
nickname: str, domain: str, port: int,
|
|
message_json: {}, debug: bool) -> None:
|
|
""" Receive an actor update from c2s
|
|
For example, setting the PGP key from the desktop client
|
|
"""
|
|
if not _valid_person_update_outbox(message_json, debug):
|
|
return
|
|
|
|
domain_full = get_full_domain(domain, port)
|
|
actor = local_actor_url(http_prefix, nickname, domain_full)
|
|
if message_json['to'][0] != actor:
|
|
if debug:
|
|
print('DEBUG: c2s actor update - to does not contain actor ' +
|
|
str(message_json['to']) + ' ' + actor)
|
|
return
|
|
if not message_json['id'].startswith(actor + '#updates/'):
|
|
if debug:
|
|
print('DEBUG: c2s actor update - unexpected id ' +
|
|
message_json['id'])
|
|
return
|
|
updated_actor_json = message_json['object']
|
|
# load actor from file
|
|
actor_filename = acct_dir(base_dir, nickname, domain) + '.json'
|
|
if not os.path.isfile(actor_filename):
|
|
print('actor_filename not found: ' + actor_filename)
|
|
return
|
|
actor_json = load_json(actor_filename)
|
|
if not actor_json:
|
|
return
|
|
actor_changed = False
|
|
# update fields within actor
|
|
if 'attachment' in updated_actor_json:
|
|
# these attachments are updatable via c2s
|
|
updatable_attachments = ('PGP', 'OpenPGP', 'Email')
|
|
|
|
for new_property_value in updated_actor_json['attachment']:
|
|
name_value = None
|
|
if new_property_value.get('name'):
|
|
name_value = new_property_value['name']
|
|
elif new_property_value.get('schema:name'):
|
|
name_value = new_property_value['schema:name']
|
|
if not name_value:
|
|
continue
|
|
if name_value not in updatable_attachments:
|
|
continue
|
|
if not new_property_value.get('type'):
|
|
continue
|
|
prop_value_name, _ = \
|
|
get_attachment_property_value(new_property_value)
|
|
if not prop_value_name:
|
|
continue
|
|
if not new_property_value['type'].endswith('PropertyValue'):
|
|
continue
|
|
if 'attachment' not in actor_json:
|
|
continue
|
|
found = False
|
|
for attach_idx, _ in enumerate(actor_json['attachment']):
|
|
attach_type = actor_json['attachment'][attach_idx]['type']
|
|
if not attach_type.endswith('PropertyValue'):
|
|
continue
|
|
attach_name = ''
|
|
if actor_json['attachment'][attach_idx].get('name'):
|
|
attach_name = \
|
|
actor_json['attachment'][attach_idx]['name']
|
|
elif actor_json['attachment'][attach_idx].get('schema:name'):
|
|
attach_name = \
|
|
actor_json['attachment'][attach_idx]['schema:name']
|
|
if attach_name != name_value:
|
|
continue
|
|
if actor_json['attachment'][attach_idx][prop_value_name] != \
|
|
new_property_value[prop_value_name]:
|
|
actor_json['attachment'][attach_idx][prop_value_name] = \
|
|
new_property_value[prop_value_name]
|
|
actor_changed = True
|
|
found = True
|
|
break
|
|
if not found:
|
|
actor_json['attachment'].append({
|
|
"name": name_value,
|
|
"type": "PropertyValue",
|
|
"value": new_property_value[prop_value_name]
|
|
})
|
|
actor_changed = True
|
|
# save actor to file
|
|
if actor_changed:
|
|
save_json(actor_json, actor_filename)
|
|
if debug:
|
|
print('actor saved: ' + actor_filename)
|
|
if debug:
|
|
print('New attachment: ' + str(actor_json['attachment']))
|
|
message_json['object'] = actor_json
|
|
if debug:
|
|
print('DEBUG: actor update via c2s - ' + nickname + '@' + domain)
|
|
|
|
|
|
def _capitalize_hashtag(content: str, message_json: {},
|
|
system_language: str, translate: {},
|
|
original_tag: str,
|
|
capitalized_tag: str) -> None:
|
|
"""If a nowplaying hashtag exists then ensure it is capitalized
|
|
"""
|
|
if translate.get(original_tag) and \
|
|
translate.get(capitalized_tag):
|
|
original_tag = translate[original_tag].replace(' ', '_')
|
|
capitalized_tag = translate[capitalized_tag].replace(' ', '_')
|
|
|
|
if '#' + original_tag not in content:
|
|
return
|
|
content = content.replace('#' + original_tag, '#' + capitalized_tag)
|
|
if 'contentMap' in message_json['object']:
|
|
if message_json['object']['contentMap'].get(system_language):
|
|
message_json['object']['contentMap'][system_language] = content
|
|
message_json['object']['contentMap'][system_language] = content
|
|
|
|
|
|
def post_message_to_outbox(session, translate: {},
|
|
message_json: {}, post_to_nickname: str,
|
|
server, base_dir: str, http_prefix: str,
|
|
domain: str, domain_full: str,
|
|
onion_domain: str, i2p_domain: str, port: int,
|
|
recent_posts_cache: {}, followers_threads: [],
|
|
federation_list: [], send_threads: [],
|
|
post_log: [], cached_webfingers: {},
|
|
person_cache: {}, allow_deletion: bool,
|
|
proxy_type: str, version: str, debug: bool,
|
|
yt_replace_domain: str,
|
|
twitter_replacement_domain: str,
|
|
show_published_date_only: bool,
|
|
allow_local_network_access: bool,
|
|
city: str, system_language: str,
|
|
shared_items_federated_domains: [],
|
|
shared_item_federation_tokens: {},
|
|
low_bandwidth: bool,
|
|
signing_priv_key_pem: str,
|
|
peertube_instances: str, theme: str,
|
|
max_like_count: int,
|
|
max_recent_posts: int, cw_lists: {},
|
|
lists_enabled: str,
|
|
content_license_url: str,
|
|
dogwhistles: {},
|
|
min_images_for_accounts: [],
|
|
buy_sites: {},
|
|
sites_unavailable: [],
|
|
max_recent_books: int,
|
|
books_cache: {},
|
|
max_cached_readers: int,
|
|
auto_cw_cache: {},
|
|
block_federated: [],
|
|
mitm_servers: []) -> bool:
|
|
"""post is received by the outbox
|
|
Client to server message post
|
|
https://www.w3.org/TR/activitypub/#client-to-server-outbox-delivery
|
|
"""
|
|
if not message_json.get('type'):
|
|
if debug:
|
|
print('DEBUG: POST to outbox has no "type" parameter')
|
|
return False
|
|
if not message_json.get('object') and message_json.get('content'):
|
|
if message_json['type'] != 'Create':
|
|
# https://www.w3.org/TR/activitypub/#object-without-create
|
|
if debug:
|
|
print('DEBUG: POST to outbox - adding Create wrapper')
|
|
message_json = \
|
|
outbox_message_create_wrap(http_prefix,
|
|
post_to_nickname,
|
|
domain, port,
|
|
message_json)
|
|
|
|
# is Bold Reading enabled for this account?
|
|
bold_reading = False
|
|
if server.bold_reading.get(post_to_nickname):
|
|
bold_reading = True
|
|
|
|
if has_object_dict(message_json):
|
|
# if this is "local only" and it is not local then reject the post
|
|
if _localonly_not_local(message_json, domain_full):
|
|
return False
|
|
|
|
# if quote toots are not allowed then reject the post
|
|
if is_quote_toot(message_json, ''):
|
|
allow_quotes = \
|
|
quote_toots_allowed(base_dir, post_to_nickname, domain,
|
|
None, None)
|
|
if not allow_quotes:
|
|
print('REJECT: POST quote toot ' + str(message_json))
|
|
return False
|
|
|
|
# get the content of the post
|
|
content_str = get_base_content_from_post(message_json, system_language)
|
|
if content_str:
|
|
# convert #nowplaying to #NowPlaying
|
|
_capitalize_hashtag(content_str, message_json,
|
|
system_language, translate,
|
|
'nowplaying', 'NowPlaying')
|
|
|
|
# check that the outgoing post doesn't contain any markup
|
|
# which can be used to implement exploits
|
|
if dangerous_markup(content_str, allow_local_network_access, []):
|
|
print('POST to outbox contains dangerous markup: ' +
|
|
str(message_json))
|
|
return False
|
|
|
|
if message_json['type'] == 'Create':
|
|
# check that a Create post has the expected fields
|
|
if not (message_json.get('id') and
|
|
message_json.get('type') and
|
|
message_json.get('actor') and
|
|
message_json.get('object') and
|
|
message_json.get('to')):
|
|
if not message_json.get('id'):
|
|
if debug:
|
|
print('DEBUG: POST to outbox - ' +
|
|
'Create does not have the id parameter ' +
|
|
str(message_json))
|
|
elif not message_json.get('id'):
|
|
if debug:
|
|
print('DEBUG: POST to outbox - ' +
|
|
'Create does not have the type parameter ' +
|
|
str(message_json))
|
|
elif not message_json.get('id'):
|
|
if debug:
|
|
print('DEBUG: POST to outbox - ' +
|
|
'Create does not have the actor parameter ' +
|
|
str(message_json))
|
|
elif not message_json.get('id'):
|
|
if debug:
|
|
print('DEBUG: POST to outbox - ' +
|
|
'Create does not have the object parameter ' +
|
|
str(message_json))
|
|
else:
|
|
if debug:
|
|
print('DEBUG: POST to outbox - ' +
|
|
'Create does not have the "to" parameter ' +
|
|
str(message_json))
|
|
return False
|
|
|
|
# actor should be a string
|
|
actor_url = get_actor_from_post(message_json)
|
|
if not actor_url:
|
|
return False
|
|
|
|
# actor should look like a url
|
|
if '://' not in actor_url or \
|
|
'.' not in actor_url:
|
|
return False
|
|
|
|
if contains_invalid_actor_url_chars(actor_url):
|
|
return False
|
|
|
|
# sent by an actor on a local network address?
|
|
if not allow_local_network_access:
|
|
local_network_pattern_list = get_local_network_addresses()
|
|
for local_network_pattern in local_network_pattern_list:
|
|
if local_network_pattern in actor_url:
|
|
return False
|
|
|
|
# is the post actor blocked?
|
|
test_domain, test_port = get_domain_from_actor(actor_url)
|
|
if test_domain:
|
|
test_domain = get_full_domain(test_domain, test_port)
|
|
if is_blocked_domain(base_dir, test_domain, None, None):
|
|
if debug:
|
|
print('DEBUG: domain is blocked: ' + actor_url)
|
|
return False
|
|
|
|
# replace youtube, so that google gets less tracking data
|
|
replace_you_tube(message_json, yt_replace_domain, system_language)
|
|
|
|
# replace twitter, so that twitter posts can be shown without
|
|
# having a twitter account
|
|
replace_twitter(message_json, twitter_replacement_domain,
|
|
system_language)
|
|
|
|
# https://www.w3.org/TR/activitypub/#create-activity-outbox
|
|
message_json['object']['attributedTo'] = actor_url
|
|
message_attachments = get_post_attachments(message_json['object'])
|
|
if message_attachments:
|
|
attachment_index = 0
|
|
attach = message_attachments[attachment_index]
|
|
if attach.get('mediaType'):
|
|
file_extension = 'png'
|
|
media_type_str = \
|
|
attach['mediaType']
|
|
|
|
extensions = {
|
|
"jpeg": "jpg",
|
|
"jxl": "jxl",
|
|
"gif": "gif",
|
|
"svg": "svg",
|
|
"webp": "webp",
|
|
"avif": "avif",
|
|
"heic": "heic",
|
|
"audio/mpeg": "mp3",
|
|
"ogg": "ogg",
|
|
"audio/wav": "wav",
|
|
"audio/x-wav": "wav",
|
|
"audio/x-pn-wave": "wav",
|
|
"audio/vnd.wave": "wav",
|
|
"flac": "flac",
|
|
"opus": "opus",
|
|
"audio/speex": "spx",
|
|
"audio/x-speex": "spx",
|
|
"mp4": "mp4",
|
|
"webm": "webm",
|
|
"ogv": "ogv"
|
|
}
|
|
for match_ext, ext in extensions.items():
|
|
if media_type_str.endswith(match_ext):
|
|
file_extension = ext
|
|
break
|
|
|
|
media_dir = \
|
|
data_dir(base_dir) + '/' + \
|
|
post_to_nickname + '@' + domain
|
|
upload_media_filename = media_dir + '/upload.' + file_extension
|
|
if not os.path.isfile(upload_media_filename):
|
|
del message_json['object']['attachment']
|
|
else:
|
|
# generate a path for the uploaded image
|
|
mpath = get_media_path()
|
|
media_path = mpath + '/' + \
|
|
create_password(16).lower() + '.' + file_extension
|
|
create_media_dirs(base_dir, mpath)
|
|
media_filename = base_dir + '/' + media_path
|
|
# move the uploaded image to its new path
|
|
os.rename(upload_media_filename, media_filename)
|
|
# convert dictionary to list if needed
|
|
if isinstance(message_json['object']['attachment'], dict):
|
|
message_json['object']['attachment'] = \
|
|
[message_json['object']['attachment']]
|
|
attach_idx = attachment_index
|
|
attach = \
|
|
message_json['object']['attachment'][attach_idx]
|
|
# change the url of the attachment
|
|
attach['url'] = \
|
|
http_prefix + '://' + domain_full + '/' + media_path
|
|
attach['url'] = \
|
|
attach['url'].replace('/media/',
|
|
'/system/' +
|
|
'media_attachments/files/')
|
|
|
|
permitted_outbox_types = (
|
|
'Create', 'Announce', 'Like', 'EmojiReact', 'Follow', 'Undo',
|
|
'Update', 'Add', 'Remove', 'Block', 'Delete', 'Skill', 'Ignore',
|
|
'Move', 'Edition'
|
|
)
|
|
if message_json['type'] not in permitted_outbox_types:
|
|
if debug:
|
|
print('DEBUG: POST to outbox - ' + message_json['type'] +
|
|
' is not a permitted activity type')
|
|
return False
|
|
if message_json.get('id'):
|
|
post_id = remove_id_ending(message_json['id'])
|
|
if debug:
|
|
print('DEBUG: id attribute exists within POST to outbox')
|
|
else:
|
|
if debug:
|
|
print('DEBUG: No id attribute within POST to outbox')
|
|
post_id = None
|
|
if debug:
|
|
print('DEBUG: save_post_to_box')
|
|
|
|
is_edited_post = False
|
|
if message_json['type'] == 'Update' and \
|
|
message_json['object']['type'] in ('Note', 'Event'):
|
|
is_edited_post = True
|
|
message_json['type'] = 'Create'
|
|
|
|
outbox_name = 'outbox'
|
|
|
|
store_hash_tags(base_dir, post_to_nickname, domain,
|
|
http_prefix, domain_full,
|
|
message_json, translate, session)
|
|
|
|
# if this is a blog post or an event then save to its own box
|
|
if message_json['type'] == 'Create':
|
|
if has_object_dict(message_json):
|
|
if message_json['object'].get('type'):
|
|
if message_json['object']['type'] == 'Article':
|
|
outbox_name = 'tlblogs'
|
|
|
|
saved_filename = \
|
|
save_post_to_box(base_dir,
|
|
http_prefix,
|
|
post_id,
|
|
post_to_nickname, domain_full,
|
|
message_json, outbox_name)
|
|
if not saved_filename:
|
|
print('WARN: post not saved to outbox ' + outbox_name)
|
|
return False
|
|
|
|
# update the speaker endpoint for speech synthesis
|
|
actor_url = get_actor_from_post(message_json)
|
|
update_speaker(base_dir, http_prefix,
|
|
post_to_nickname, domain, domain_full,
|
|
message_json, person_cache,
|
|
translate, actor_url,
|
|
theme, system_language,
|
|
outbox_name)
|
|
|
|
if has_edition_tag(message_json):
|
|
store_book_events(base_dir,
|
|
message_json,
|
|
system_language, [],
|
|
translate, debug,
|
|
max_recent_books,
|
|
books_cache,
|
|
max_cached_readers)
|
|
|
|
# save all instance blogs to the news actor
|
|
if post_to_nickname != 'news' and outbox_name == 'tlblogs':
|
|
if '/' in saved_filename:
|
|
if is_featured_writer(base_dir, post_to_nickname, domain):
|
|
saved_post_id = saved_filename.split('/')[-1]
|
|
blogs_dir = \
|
|
data_dir(base_dir) + '/news@' + domain + '/tlblogs'
|
|
if not os.path.isdir(blogs_dir):
|
|
os.mkdir(blogs_dir)
|
|
copyfile(saved_filename, blogs_dir + '/' + saved_post_id)
|
|
inbox_update_index('tlblogs', base_dir,
|
|
'news@' + domain,
|
|
saved_filename, debug)
|
|
|
|
# clear the citations file if it exists
|
|
citations_filename = \
|
|
data_dir(base_dir) + '/' + \
|
|
post_to_nickname + '@' + domain + '/.citations.txt'
|
|
if os.path.isfile(citations_filename):
|
|
try:
|
|
os.remove(citations_filename)
|
|
except OSError:
|
|
print('EX: post_message_to_outbox unable to delete ' +
|
|
citations_filename)
|
|
|
|
# The following activity types get added to the index files
|
|
indexed_activities = (
|
|
'Create', 'Question', 'Note', 'Event', 'EncryptedMessage', 'Article',
|
|
'Patch', 'Announce', 'ChatMessage'
|
|
)
|
|
if message_json['type'] in indexed_activities:
|
|
indexes = [outbox_name, "inbox"]
|
|
self_actor = \
|
|
local_actor_url(http_prefix, post_to_nickname, domain_full)
|
|
for box_name_index in indexes:
|
|
if not box_name_index:
|
|
continue
|
|
|
|
# should this also go to the media timeline?
|
|
if box_name_index == 'inbox':
|
|
show_vote_posts = True
|
|
show_vote_file = \
|
|
acct_dir(base_dir, post_to_nickname, domain) + '/.noVotes'
|
|
if os.path.isfile(show_vote_file):
|
|
show_vote_posts = False
|
|
languages_understood: list[str] = []
|
|
if is_image_media(session, base_dir, http_prefix,
|
|
post_to_nickname, domain,
|
|
message_json,
|
|
yt_replace_domain,
|
|
twitter_replacement_domain,
|
|
allow_local_network_access,
|
|
recent_posts_cache, debug,
|
|
system_language,
|
|
domain_full, person_cache,
|
|
signing_priv_key_pem,
|
|
bold_reading,
|
|
show_vote_posts,
|
|
languages_understood,
|
|
mitm_servers):
|
|
inbox_update_index('tlmedia', base_dir,
|
|
post_to_nickname + '@' + domain,
|
|
saved_filename, debug)
|
|
|
|
if box_name_index == 'inbox' and outbox_name == 'tlblogs':
|
|
continue
|
|
|
|
# avoid duplicates of the message if already going
|
|
# back to the inbox of the same account
|
|
if self_actor not in message_json['to']:
|
|
# show sent post within the inbox,
|
|
# as is the typical convention
|
|
inbox_update_index(box_name_index, base_dir,
|
|
post_to_nickname + '@' + domain,
|
|
saved_filename, debug)
|
|
|
|
# regenerate the html
|
|
use_cache_only = False
|
|
page_number = 1
|
|
show_individual_post_icons = True
|
|
manually_approve_followers = \
|
|
follower_approval_active(base_dir,
|
|
post_to_nickname, domain)
|
|
timezone = \
|
|
get_account_timezone(base_dir,
|
|
post_to_nickname, domain)
|
|
mitm = False
|
|
if os.path.isfile(saved_filename.replace('.json', '') +
|
|
'.mitm'):
|
|
mitm = True
|
|
minimize_all_images = False
|
|
if post_to_nickname in min_images_for_accounts:
|
|
minimize_all_images = True
|
|
individual_post_as_html(signing_priv_key_pem,
|
|
False, recent_posts_cache,
|
|
max_recent_posts,
|
|
translate, page_number,
|
|
base_dir, session,
|
|
cached_webfingers,
|
|
person_cache,
|
|
post_to_nickname, domain, port,
|
|
message_json, None, True,
|
|
allow_deletion,
|
|
http_prefix, __version__,
|
|
box_name_index,
|
|
yt_replace_domain,
|
|
twitter_replacement_domain,
|
|
show_published_date_only,
|
|
peertube_instances,
|
|
allow_local_network_access,
|
|
theme, system_language,
|
|
max_like_count,
|
|
box_name_index != 'dm',
|
|
show_individual_post_icons,
|
|
manually_approve_followers,
|
|
False, True, use_cache_only,
|
|
cw_lists, lists_enabled,
|
|
timezone, mitm,
|
|
bold_reading, dogwhistles,
|
|
minimize_all_images, None,
|
|
buy_sites, auto_cw_cache,
|
|
mitm_servers)
|
|
|
|
if is_edited_post:
|
|
message_json['type'] = 'Update'
|
|
|
|
if outbox_announce(recent_posts_cache,
|
|
base_dir, message_json, debug):
|
|
if debug:
|
|
print('DEBUG: Updated announcements (shares) collection ' +
|
|
'for the post associated with the Announce activity')
|
|
if debug:
|
|
print('DEBUG: sending c2s post to followers')
|
|
# remove inactive threads
|
|
inactive_follower_threads = []
|
|
for thr in followers_threads:
|
|
if not thr.is_alive():
|
|
inactive_follower_threads.append(thr)
|
|
for thr in inactive_follower_threads:
|
|
followers_threads.remove(thr)
|
|
if debug:
|
|
print('DEBUG: ' + str(len(followers_threads)) +
|
|
' followers threads active')
|
|
# retain up to 200 threads
|
|
if len(followers_threads) > 200:
|
|
# kill the thread if it is still alive
|
|
if followers_threads[0].is_alive():
|
|
followers_threads[0].kill()
|
|
# remove it from the list
|
|
followers_threads.pop(0)
|
|
# create a thread to send the post to followers
|
|
followers_thread = \
|
|
send_to_followers_thread(server, server.session,
|
|
server.session_onion,
|
|
server.session_i2p,
|
|
base_dir,
|
|
post_to_nickname,
|
|
domain, onion_domain, i2p_domain,
|
|
port, http_prefix,
|
|
federation_list,
|
|
send_threads,
|
|
post_log,
|
|
cached_webfingers,
|
|
person_cache,
|
|
message_json, debug,
|
|
version,
|
|
shared_items_federated_domains,
|
|
shared_item_federation_tokens,
|
|
signing_priv_key_pem,
|
|
sites_unavailable,
|
|
system_language,
|
|
mitm_servers)
|
|
followers_threads.append(followers_thread)
|
|
|
|
if debug:
|
|
print('DEBUG: handle any unfollow requests')
|
|
outbox_undo_follow(base_dir, message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle skills changes requests')
|
|
outbox_skills(base_dir, post_to_nickname, message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle availability changes requests')
|
|
outbox_availability(base_dir, post_to_nickname, message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle any like requests')
|
|
outbox_like(recent_posts_cache,
|
|
base_dir, post_to_nickname, domain,
|
|
message_json, debug)
|
|
if debug:
|
|
print('DEBUG: handle any undo like requests')
|
|
outbox_undo_like(recent_posts_cache,
|
|
base_dir, post_to_nickname, domain,
|
|
message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle any emoji reaction requests')
|
|
outbox_reaction(recent_posts_cache,
|
|
base_dir, post_to_nickname, domain,
|
|
message_json, debug)
|
|
if debug:
|
|
print('DEBUG: handle any undo emoji reaction requests')
|
|
outbox_undo_reaction(recent_posts_cache,
|
|
base_dir, post_to_nickname, domain,
|
|
message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle any undo announce requests')
|
|
outbox_undo_announce(recent_posts_cache,
|
|
base_dir, post_to_nickname, domain,
|
|
message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle any bookmark requests')
|
|
outbox_bookmark(recent_posts_cache,
|
|
base_dir, http_prefix,
|
|
post_to_nickname, domain, port,
|
|
message_json, debug)
|
|
if debug:
|
|
print('DEBUG: handle any undo bookmark requests')
|
|
outbox_undo_bookmark(recent_posts_cache,
|
|
base_dir, http_prefix,
|
|
post_to_nickname, domain, port,
|
|
message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle delete requests')
|
|
outbox_delete(base_dir, http_prefix,
|
|
post_to_nickname, domain,
|
|
message_json, debug,
|
|
allow_deletion,
|
|
recent_posts_cache)
|
|
|
|
if debug:
|
|
print('DEBUG: handle block requests')
|
|
outbox_block(base_dir, post_to_nickname, domain,
|
|
message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle undo block requests')
|
|
outbox_undo_block(base_dir, post_to_nickname, domain, message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle mute requests')
|
|
outbox_mute(base_dir, http_prefix,
|
|
post_to_nickname, domain,
|
|
port,
|
|
message_json, debug,
|
|
recent_posts_cache)
|
|
|
|
if debug:
|
|
print('DEBUG: handle undo mute requests')
|
|
outbox_undo_mute(base_dir, http_prefix,
|
|
post_to_nickname, domain,
|
|
port,
|
|
message_json, debug,
|
|
recent_posts_cache)
|
|
|
|
if debug:
|
|
print('DEBUG: handle share uploads')
|
|
outbox_share_upload(base_dir, http_prefix, post_to_nickname, domain,
|
|
port, message_json, debug, city,
|
|
system_language, translate, low_bandwidth,
|
|
content_license_url, block_federated)
|
|
|
|
if debug:
|
|
print('DEBUG: handle undo share uploads')
|
|
outbox_undo_share_upload(base_dir, post_to_nickname, domain,
|
|
message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: handle actor updates from c2s')
|
|
_person_receive_update_outbox(base_dir, http_prefix,
|
|
post_to_nickname, domain, port,
|
|
message_json, debug)
|
|
|
|
if debug:
|
|
print('DEBUG: sending c2s post to named addresses')
|
|
if message_json.get('to'):
|
|
print('c2s sender: ' +
|
|
post_to_nickname + '@' + domain + ':' + str(port) +
|
|
' recipient: ' + str(message_json['to']))
|
|
else:
|
|
print('c2s sender: ' +
|
|
post_to_nickname + '@' + domain + ':' + str(port))
|
|
named_addresses_thread = \
|
|
send_to_named_addresses_thread(server, server.session,
|
|
server.session_onion,
|
|
server.session_i2p,
|
|
base_dir, post_to_nickname,
|
|
domain, onion_domain, i2p_domain, port,
|
|
http_prefix,
|
|
federation_list,
|
|
send_threads,
|
|
post_log,
|
|
cached_webfingers,
|
|
person_cache,
|
|
message_json, debug,
|
|
version,
|
|
shared_items_federated_domains,
|
|
shared_item_federation_tokens,
|
|
signing_priv_key_pem,
|
|
proxy_type,
|
|
server.followers_sync_cache,
|
|
server.sites_unavailable,
|
|
server.system_language,
|
|
server.mitm_servers)
|
|
followers_threads.append(named_addresses_thread)
|
|
return True
|