__filename__ = "webapp_create_post.py" __author__ = "Bob Mottram" __license__ = "AGPL3+" __version__ = "1.5.0" __maintainer__ = "Bob Mottram" __email__ = "bob@libreserver.org" __status__ = "Production" __module_group__ = "Web Interface" import os from flags import is_public_post_from_url from flags import is_premium_account from utils import data_dir from utils import dangerous_markup from utils import remove_html from utils import get_content_from_post from utils import has_object_dict from utils import load_json from utils import locate_post from utils import get_new_post_endpoints from utils import get_nickname_from_actor from utils import get_domain_from_actor from utils import get_media_formats from utils import get_config_param from utils import acct_dir from utils import get_currencies from utils import get_category_types from utils import get_account_timezone from utils import get_supported_languages from utils import text_in_file from utils import get_attributed_to from utils import get_full_domain from webapp_utils import open_content_warning from webapp_utils import edit_check_box from webapp_utils import get_buy_links from webapp_utils import html_following_data_list from webapp_utils import html_common_emoji from webapp_utils import begin_edit_section from webapp_utils import end_edit_section from webapp_utils import get_banner_file from webapp_utils import html_header_with_external_style from webapp_utils import html_footer from webapp_utils import edit_text_field from webapp_utils import edit_number_field from webapp_utils import edit_currency_field from webapp_post import individual_post_as_html from maps import get_map_preferences_url from maps import get_map_preferences_coords from maps import get_location_from_post from cache import get_person_from_cache from person import get_person_notes def _html_new_post_drop_down(scope_icon: str, scope_description: str, reply_str: str, translate: {}, show_public_on_dropdown: bool, default_timeline: str, path_base: str, dropdown_new_post_suffix: str, dropdown_new_blog_suffix: str, dropdown_unlisted_suffix: str, dropdown_followers_suffix: str, dropdown_dm_suffix: str, dropdown_reminder_suffix: str, dropdown_report_suffix: str, no_drop_down: bool, access_keys: {}, account_dir: str, premium: bool) -> str: """Returns the html for a drop down list of new post types """ drop_down_content = '\n' return drop_down_content drop_down_content += ' \n' drop_down_content += '\n' return drop_down_content def _get_date_from_tags(tags: []) -> (str, str): """Returns the date from the tags list """ for tag_item in tags: if not tag_item.get('type'): continue if tag_item['type'] != 'Event': continue if not tag_item.get('startTime'): continue if not isinstance(tag_item['startTime'], str): continue if 'T' not in tag_item['startTime']: continue start_time = tag_item['startTime'] if not tag_item.get('endTime'): return start_time, '' if not isinstance(tag_item['endTime'], str): return start_time, '' if 'T' not in tag_item['endTime']: return start_time, '' end_time = tag_item['endTime'] return start_time, end_time return '', '' def _remove_initial_mentions_from_content(content: str) -> str: """ Removes initial @mentions from content This happens when a the html content is converted back to plain text """ if not content.startswith('@'): return content words = content.split(' ') new_content = '' for wrd in words: if wrd.startswith('@'): continue if new_content: new_content += ' ' + wrd else: new_content += wrd return new_content def html_new_post(edit_post_params: {}, media_instance: bool, translate: {}, base_dir: str, http_prefix: str, path: str, in_reply_to: str, mentions: [], share_description: str, report_url: str, page_number: int, category: str, nickname: str, domain: str, domain_full: str, default_timeline: str, newswire: {}, theme: str, no_drop_down: bool, access_keys: {}, custom_submit_text: str, conversation_id: str, convthread_id: str, recent_posts_cache: {}, max_recent_posts: int, session, cached_webfingers: {}, person_cache: {}, port: int, post_json_object: {}, project_version: str, yt_replace_domain: str, twitter_replacement_domain: str, show_published_date_only: bool, peertube_instances: [], allow_local_network_access: bool, system_language: str, languages_understood: [], max_like_count: int, signing_priv_key_pem: str, cw_lists: {}, lists_enabled: str, box_name: str, reply_is_chat: bool, bold_reading: bool, dogwhistles: {}, min_images_for_accounts: [], default_month: int, default_year: int, default_post_language: str, buy_sites: {}, default_buy_site: str, auto_cw_cache: {}, searchable_by_default: str) -> str: """New post screen """ # get the json if this is an edited post edited_post_json = None edited_published = '' if edit_post_params: if edit_post_params.get('post_url'): edited_post_filename = \ locate_post(base_dir, nickname, domain, edit_post_params['post_url']) if edited_post_filename: edited_post_json = load_json(edited_post_filename) if not has_object_dict(edited_post_json): return '' if not edited_post_json: return '' buy_links = \ get_buy_links(edited_post_json, translate, buy_sites) if buy_links: for _, buy_url in buy_links.items(): default_buy_site = buy_url break # Due to lack of AP specification maintenance, a conversation can also # be referred to as a thread or (confusingly) "context" if edited_post_json['object'].get('conversation'): conversation_id = edited_post_json['object']['conversation'] elif edited_post_json['object'].get('context'): conversation_id = edited_post_json['object']['context'] if edited_post_json['object'].get('thread'): convthread_id = edited_post_json['object']['thread'] if edit_post_params.get('replyTo'): in_reply_to = edit_post_params['replyTo'] if edit_post_params['scope'] == 'dm': mentions = edited_post_json['object']['to'] edited_published = \ edited_post_json['object']['published'] # default subject line or content warning default_subject = '' if share_description: default_subject = share_description default_location = '' default_start_time = '' default_end_time = '' if default_month and default_year: default_month_str = str(default_month) if default_month < 10: default_month_str = '0' + default_month_str default_start_time = \ str(default_year) + '-' + default_month_str + '-01T09:00:00' if edited_post_json: # if this is an edited post then get the subject line or # content warning summary_str = get_content_from_post(edited_post_json, system_language, languages_understood, "summary") if summary_str: default_subject = remove_html(summary_str) if edited_post_json['object'].get('tag'): # if this is an edited post then get the location location_str = get_location_from_post(edited_post_json) if location_str: default_location = location_str # if this is an edited post then get the start and end time default_start_time, default_end_time = \ _get_date_from_tags(edited_post_json['object']['tag']) reply_str = '' is_new_reminder = False if path.endswith('/newreminder'): is_new_reminder = True # the date and time date_and_time_str = '

\n' if not is_new_reminder: date_and_time_str += \ '\n' # select a date and time for this post date_and_time_str += '\n' date_default = '' time_default = '' if default_start_time: date_default = ' value="' + default_start_time.split('T')[0] + '"' time_default = ' value="' + default_start_time.split('T')[1] + '"' end_time_default = '' if default_end_time: end_time_default = ' value="' + default_end_time.split('T')[1] + '"' date_and_time_str += \ '\n' date_and_time_str += '\n
\n' date_and_time_str += '\n

\n' show_public_on_dropdown = True message_box_height = 400 image_description_height = 150 transcript_height = 1000 # filename of the banner shown at the top banner_file, _ = \ get_banner_file(base_dir, nickname, domain, theme) banner_path = '/users/' + nickname + '/' + banner_file if not path.endswith('/newshare') and not path.endswith('/newwanted'): if not path.endswith('/newreport'): if not in_reply_to or is_new_reminder: new_post_text = '

' + \ translate['Write your post text below.'] + '

\n' else: new_post_text = '' if category != 'accommodation': new_post_text = \ '

' + \ translate['Write your reply to'] + \ ' ' + \ translate['this post'] + '

\n' # is sending posts to this account blocked? send_block_filename = \ acct_dir(base_dir, nickname, domain) + \ '/send_blocks.txt' if os.path.isfile(send_block_filename): reply_actor = in_reply_to reply_nickname = get_nickname_from_actor(in_reply_to) if reply_nickname: reply_actor = \ in_reply_to.split('/' + reply_nickname)[0] + \ '/' + reply_nickname if text_in_file(reply_actor, send_block_filename, False): new_post_text += \ '

' + \ translate['FollowAccountWarning'] + \ '

\n' else: reply_domain, _ = \ get_domain_from_actor(reply_actor) if text_in_file('://' + reply_domain + '\n', send_block_filename, False): new_post_text += \ '

' + \ translate['FollowWarning'] + '

\n' if post_json_object: timezone = \ get_account_timezone(base_dir, nickname, domain) minimize_all_images = False if nickname in min_images_for_accounts: minimize_all_images = True replied_to_post = \ individual_post_as_html(signing_priv_key_pem, True, recent_posts_cache, max_recent_posts, translate, None, base_dir, session, cached_webfingers, person_cache, nickname, domain, port, post_json_object, None, True, False, http_prefix, project_version, box_name, yt_replace_domain, twitter_replacement_domain, show_published_date_only, peertube_instances, allow_local_network_access, theme, system_language, max_like_count, False, False, False, False, False, False, cw_lists, lists_enabled, timezone, False, bold_reading, dogwhistles, minimize_all_images, None, buy_sites, auto_cw_cache) new_post_text += \ open_content_warning(replied_to_post, translate) # about the author if has_object_dict(post_json_object): if post_json_object['object'].get('attributedTo'): attrib_field = \ post_json_object['object']['attributedTo'] attrib_url = get_attributed_to(attrib_field) domain_full = get_full_domain(domain, port) this_account_url = \ '://' + domain_full + '/users/' + nickname if attrib_url and \ not attrib_url.endswith(this_account_url): reply_to_actor = \ get_person_from_cache(base_dir, attrib_url, person_cache) if reply_to_actor: summary = None if reply_to_actor.get('summary'): summary = reply_to_actor['summary'] attrib_nickname = \ get_nickname_from_actor(attrib_url) attrib_domain, attrib_port = \ get_domain_from_actor(attrib_url) if attrib_nickname and attrib_domain: attrib_domain_full = \ get_full_domain(attrib_domain, attrib_port) attrib_handle = \ attrib_nickname + '@' + \ attrib_domain_full person_notes = \ get_person_notes(base_dir, nickname, domain, attrib_handle) if person_notes: if summary: summary = \ '' + \ person_notes + \ '' + \ '

' + summary else: summary = \ '' + \ person_notes + \ '' if summary: if not dangerous_markup(summary, False, []): reply_to_description = \ summary else: reply_to_description = \ remove_html(summary) about_author_str = \ translate['About the author'] new_post_text += \ '
\n' + \ '
\n' + \ ' ' + about_author_str + \ '\n
\n' + \ ' ' + reply_to_description + \ '\n
\n' reply_str = '\n' # if replying to a non-public post then also make # this post non-public if not is_public_post_from_url(base_dir, nickname, domain, in_reply_to): new_post_path = path if '?' in new_post_path: new_post_path = new_post_path.split('?')[0] if new_post_path.endswith('/newpost'): path = path.replace('/newpost', '/newfollowers') show_public_on_dropdown = False else: new_post_text = \ '

' + translate['Write your report below.'] + '

\n' # custom report header with any additional instructions dir_str = data_dir(base_dir) if os.path.isfile(dir_str + '/report.txt'): try: with open(dir_str + '/report.txt', 'r', encoding='utf-8') as fp_report: custom_report_text = fp_report.read() if '

' not in custom_report_text: custom_report_text = \ '

' + \ custom_report_text + '

\n' rep_str = '

' custom_report_text = \ custom_report_text.replace('

', rep_str) new_post_text += custom_report_text except OSError as exc: print('EX: html_new_post unable to read ' + dir_str + '/report.txt ' + str(exc)) idx = 'This message only goes to moderators, even if it ' + \ 'mentions other fediverse addresses.' new_post_text += \ '

' + translate[idx] + '

\n' + \ '

' + translate['Also see'] + \ ' ' + \ translate['Terms of Service'] + '

\n' else: if path.endswith('/newshare'): new_post_text = \ '

' + \ translate['Enter the details for your shared item below.'] + \ '

\n' else: new_post_text = \ '

' + \ translate['Enter the details for your wanted item below.'] + \ '

\n' if path.endswith('/newquestion'): new_post_text = \ '

' + \ translate['Enter the choices for your question below.'] + \ '

\n' dir_str = data_dir(base_dir) if os.path.isfile(dir_str + '/newpost.txt'): try: with open(dir_str + '/newpost.txt', 'r', encoding='utf-8') as fp_new: new_post_text = '

' + fp_new.read() + '

\n' except OSError: print('EX: html_new_post unable to read ' + dir_str + '/newpost.txt') css_filename = base_dir + '/epicyon-profile.css' if os.path.isfile(base_dir + '/epicyon.css'): css_filename = base_dir + '/epicyon.css' if '?' in path: path = path.split('?')[0] new_post_endpoints = get_new_post_endpoints() path_base = path for curr_post_type in new_post_endpoints: path_base = path_base.replace('/' + curr_post_type, '') attach_str = 'Attach an image, video or audio file' new_post_image_section = begin_edit_section('πŸ“· ' + translate[attach_str]) new_post_image_section += \ ' \n' new_post_image_section += \ ' \n' new_post_image_section += \ ' \n' media_creator_str = translate['Media creator'] new_post_image_section += \ edit_text_field(media_creator_str, 'mediaCreator', '', '') media_license_str = translate['Media license'] new_post_image_section += \ edit_text_field(media_license_str, 'mediaLicense', '', 'CC-BY-NC') new_post_image_section += \ ' \n' new_post_image_section += \ ' \n' new_post_image_section += end_edit_section() new_post_emoji_section = '' if not path.endswith('/newreadingstatus'): common_emoji_str = html_common_emoji(base_dir, 16) if common_emoji_str: new_post_emoji_section = \ begin_edit_section('πŸ˜€ ' + translate['Common emoji']) new_post_emoji_section += \ '
\n' new_post_emoji_section += common_emoji_str new_post_emoji_section += end_edit_section() scope_icon = 'scope_public.png' scope_description = translate['Public'] if share_description: if category == 'accommodation': placeholder_subject = translate['Request to stay'] else: placeholder_subject = translate['Ask about a shared item.'] + '..' else: placeholder_subject = \ translate['Subject or Content Warning (optional)'] + '...' placeholder_mentions = '' if in_reply_to: placeholder_mentions = \ translate['Replying to'] + '...' placeholder_message = '' if category != 'accommodation': if default_timeline == 'tlfeatures': placeholder_message = translate['Write your news report'] + '...' else: placeholder_message = translate['Write something'] + '...' else: idx = 'Introduce yourself and specify the date ' + \ 'and time when you wish to stay' placeholder_message = translate[idx] extra_fields = '' premium = is_premium_account(base_dir, nickname, domain) endpoint = 'newpost' if path.endswith('/newblog'): placeholder_subject = translate['Title'] scope_icon = 'scope_blog.png' if default_timeline != 'tlfeatures': scope_description = translate['Blog'] else: scope_description = translate['Article'] endpoint = 'newblog' elif path.endswith('/newunlisted'): scope_icon = 'scope_unlisted.png' scope_description = translate['Unlisted'] endpoint = 'newunlisted' elif path.endswith('/newfollowers'): scope_icon = 'scope_followers.png' scope_description = translate['Followers'] if premium: scope_description = translate['Fans'] endpoint = 'newfollowers' elif path.endswith('/newdm'): scope_icon = 'scope_dm.png' scope_description = translate['DM'] endpoint = 'newdm' placeholder_message = '⚠️ ' + translate['DM warning'] elif is_new_reminder: scope_icon = 'scope_reminder.png' scope_description = translate['Reminder'] endpoint = 'newreminder' elif path.endswith('/newreport'): scope_icon = 'scope_report.png' scope_description = translate['Report'] endpoint = 'newreport' elif path.endswith('/newquestion'): scope_icon = 'scope_question.png' scope_description = translate['Question'] placeholder_message = translate['Enter your question'] + '...' endpoint = 'newquestion' extra_fields = '
\n' extra_fields += '
\n' for question_ctr in range(8): extra_fields += \ '
\n' extra_fields += \ '
\n' extra_fields += '
' elif path.endswith('/newshare'): scope_icon = 'scope_share.png' scope_description = translate['Shared Item'] placeholder_subject = translate['Name of the shared item'] + '...' placeholder_message = \ translate['Description of the item being shared'] + '...' endpoint = 'newshare' extra_fields = '
\n' extra_fields += \ edit_number_field(translate['Quantity'], 'itemQty', 1, 1, 999999, 1) extra_fields += '
' + \ edit_text_field(translate['Type of shared item. eg. hat'] + ':', 'itemType', '', '', True) category_types = get_category_types(base_dir) cat_str = translate['Category of shared item. eg. clothing'] extra_fields += '
\n' extra_fields += '
\n' extra_fields += \ edit_number_field(translate['Duration of listing in days'], 'duration', 14, 1, 365, 1) extra_fields += '
\n' extra_fields += \ edit_check_box(translate['Display on your public profile'], 'shareOnProfile', False) extra_fields += '
\n' extra_fields += '
\n' city_or_loc_str = translate['City or location of the shared item'] extra_fields += edit_text_field(city_or_loc_str + ':', 'location', default_location, 'https://www.openstreetmap.org/#map=') extra_fields += '
\n' extra_fields += '
\n' extra_fields += \ edit_currency_field(translate['Price'] + ':', 'itemPrice', '0.00', '0.00', True) extra_fields += '
' extra_fields += \ '
\n' currencies = get_currencies() extra_fields += ' \n' extra_fields += '
\n' elif path.endswith('/newwanted'): scope_icon = 'scope_wanted.png' scope_description = translate['Wanted'] placeholder_subject = translate['Name of the wanted item'] + '...' placeholder_message = \ translate['Description of the item wanted'] + '...' endpoint = 'newwanted' extra_fields = '
\n' extra_fields += \ edit_number_field(translate['Quantity'], 'itemQty', 1, 1, 999999, 1) extra_fields += '
' + \ edit_text_field(translate['Type of wanted item. eg. hat'] + ':', 'itemType', '', '', True) category_types = get_category_types(base_dir) cat_str = translate['Category of wanted item. eg. clothes'] extra_fields += '
\n' extra_fields += '
\n' extra_fields += \ edit_number_field(translate['Duration of listing in days'], 'duration', 14, 1, 365, 1) extra_fields += '
\n' extra_fields += '
\n' city_or_loc_str = translate['City or location of the wanted item'] extra_fields += edit_text_field(city_or_loc_str + ':', 'location', default_location, 'https://www.openstreetmap.org/#map=') extra_fields += '
\n' extra_fields += '
\n' extra_fields += \ edit_currency_field(translate['Maximum Price'] + ':', 'itemPrice', '0.00', '0.00', True) extra_fields += '
' extra_fields += \ '
\n' currencies = get_currencies() extra_fields += ' \n' extra_fields += '
\n' elif path.endswith('/newreadingstatus'): scope_icon = 'scope_readingstatus.png' scope_description = translate['Reading Status'] endpoint = 'newreadingstatus' extra_fields = '
\n' cat_str = translate['Update type'] extra_fields += '
\n' extra_fields += '
\n' extra_fields += '
' + \ edit_text_field(translate['Title'] + ':', 'booktitle', '', '', True) books_url = 'https://en.wikipedia.org/wiki/Lists_of_books' extra_fields += '
' + \ edit_text_field('URL:', 'bookurl', '', 'https://...', True) extra_fields += '
' + \ edit_number_field(translate['Rating'], 'bookrating', '', 1, 5, None) extra_fields += '
\n' citations_str = '' if endpoint == 'newblog': citations_filename = \ acct_dir(base_dir, nickname, domain) + '/.citations.txt' if os.path.isfile(citations_filename): citations_str = '
\n' citations_str += '

\n' citations_str += ' \n' citations_str += '
\n' replies_section = '' date_and_location = '' if endpoint not in ('newshare', 'newwanted', 'newreport', 'newquestion', 'newreadingstatus'): if not is_new_reminder: replies_section = \ '
\n' if category != 'accommodation': replies_section += \ '

\n' if endpoint == 'newpost': replies_section += \ '

\n' else: replies_section += \ '\n' # Language used dropdown supported_languages = get_supported_languages(base_dir) languages_dropdown = '
' languages_dropdown = \ languages_dropdown.replace('