epicyon/webapp_profile.py

3780 lines
157 KiB
Python
Raw Normal View History

2020-11-09 22:44:03 +00:00
__filename__ = "webapp_profile.py"
__author__ = "Bob Mottram"
__license__ = "AGPL3+"
2024-12-22 23:37:30 +00:00
__version__ = "1.6.0"
2020-11-09 22:44:03 +00:00
__maintainer__ = "Bob Mottram"
2021-09-10 16:14:50 +00:00
__email__ = "bob@libreserver.org"
2020-11-09 22:44:03 +00:00
__status__ = "Production"
2021-06-15 15:08:12 +00:00
__module_group__ = "Web Interface"
2020-11-09 22:44:03 +00:00
import os
from pprint import pprint
2021-12-29 21:55:09 +00:00
from webfinger import webfinger_handle
from flags import is_dormant
from flags import is_artist
from flags import is_system_account
from flags import is_group_account
from flags import is_valid_date
from flags import is_premium_account
from utils import text_mode_removals
2024-08-09 12:29:22 +00:00
from utils import replace_strings
2024-05-12 12:35:26 +00:00
from utils import data_dir
from utils import time_days_ago
2024-02-04 13:52:31 +00:00
from utils import uninvert_text
2024-02-04 13:18:08 +00:00
from utils import get_attributed_to
2023-12-09 14:18:24 +00:00
from utils import get_url_from_post
2023-08-31 22:29:10 +00:00
from utils import get_memorials
from utils import text_in_file
2023-07-13 08:22:49 +00:00
from utils import dangerous_markup
2023-05-12 14:39:26 +00:00
from utils import ap_proxy_type
2022-11-27 21:11:16 +00:00
from utils import remove_id_ending
from utils import standardize_text
2021-12-27 21:59:07 +00:00
from utils import get_display_name
2021-12-26 10:57:03 +00:00
from utils import has_object_dict
2021-12-26 18:58:06 +00:00
from utils import get_occupation_name
2021-12-26 12:16:36 +00:00
from utils import get_locked_account
2021-12-26 12:45:03 +00:00
from utils import get_full_domain
2021-12-27 22:19:18 +00:00
from utils import get_nickname_from_actor
2021-12-27 19:05:25 +00:00
from utils import get_domain_from_actor
2021-12-27 15:43:22 +00:00
from utils import remove_html
2021-12-26 15:13:34 +00:00
from utils import load_json
2021-12-26 14:08:58 +00:00
from utils import get_config_param
2021-12-26 15:44:28 +00:00
from utils import get_image_formats
2021-12-26 12:02:29 +00:00
from utils import acct_dir
2021-12-26 17:26:55 +00:00
from utils import get_supported_languages
2021-12-26 10:19:59 +00:00
from utils import local_actor_url
2021-12-28 10:25:50 +00:00
from utils import get_reply_interval_hours
2022-02-25 19:57:31 +00:00
from utils import get_account_timezone
2022-06-21 11:58:50 +00:00
from utils import remove_eol
2024-01-09 16:59:23 +00:00
from utils import get_actor_from_post
2024-01-27 17:04:21 +00:00
from utils import resembles_url
2021-12-29 21:55:09 +00:00
from languages import get_actor_languages
from skills import get_skills
from theme import get_themes_list
2023-05-03 10:34:54 +00:00
from person import get_featured_hashtags_as_html
from person import get_featured_hashtags
2021-12-28 16:50:20 +00:00
from person import person_box_json
2021-12-29 21:55:09 +00:00
from person import get_actor_json
from person import get_person_avatar_url
from person import get_person_notes
from posts import get_post_expiry_keep_dms
from posts import get_post_expiry_days
2021-12-29 21:55:09 +00:00
from posts import get_person_box
2021-12-28 19:33:29 +00:00
from posts import is_moderator
2021-12-29 21:55:09 +00:00
from posts import parse_user_feed
from posts import is_create_inside_announce
from posts import get_max_profile_posts
from website import get_website
from website import get_gemini_link
2021-12-28 18:13:52 +00:00
from donate import get_donation_url
from pronouns import get_pronouns
from pixelfed import get_pixelfed
from discord import get_discord
2024-08-15 19:58:26 +00:00
from art import get_art_site_url
from music import get_music_site_url
from youtube import get_youtube
from peertube import get_peertube
2021-12-28 17:33:54 +00:00
from xmpp import get_xmpp_address
2021-12-28 18:13:52 +00:00
from matrix import get_matrix_address
2021-12-28 17:33:54 +00:00
from ssb import get_ssb_address
from pgp import get_email_address
from pgp import get_pgp_fingerprint
from pgp import get_pgp_pub_key
2021-12-28 17:20:43 +00:00
from enigma import get_enigma_pub_key
2021-12-28 17:33:54 +00:00
from tox import get_tox_address
from briar import get_briar_address
from cwtch import get_cwtch_address
2021-12-29 21:55:09 +00:00
from filters import is_filtered
from follow import is_follower_of_person
from follow import get_follower_domains
from follow import is_following_actor
2021-12-29 21:55:09 +00:00
from webapp_frontscreen import html_front_screen
from webapp_utils import text_mode_browser
from webapp_utils import html_following_dropdown
from webapp_utils import edit_number_field
2021-12-29 21:55:09 +00:00
from webapp_utils import html_keyboard_navigation
from webapp_utils import html_hide_from_screen_reader
from webapp_utils import scheduled_posts_exist
from webapp_utils import html_header_with_external_style
from webapp_utils import html_header_with_person_markup
from webapp_utils import html_footer
from webapp_utils import add_emoji_to_display_name
from webapp_utils import get_profile_background_file
2021-12-29 21:55:09 +00:00
from webapp_utils import html_post_separator
from webapp_utils import edit_check_box
from webapp_utils import edit_text_field
from webapp_utils import edit_text_area
from webapp_utils import begin_edit_section
from webapp_utils import end_edit_section
2021-12-28 22:22:09 +00:00
from blog import get_blog_address
2023-05-19 20:44:43 +00:00
from blog import account_has_blog
2021-12-29 21:55:09 +00:00
from webapp_post import individual_post_as_html
from webapp_timeline import html_individual_share
2022-07-08 16:33:38 +00:00
from webapp_timeline import page_number_buttons
2023-03-20 14:50:19 +00:00
from cwlists import get_cw_list_variable
2022-11-23 18:40:45 +00:00
from blocking import get_account_blocks
2022-05-14 17:07:20 +00:00
from blocking import is_blocked
2024-12-24 18:17:36 +00:00
from blocking import sending_is_blocked2
from content import remove_link_trackers_from_content
2022-03-24 13:14:41 +00:00
from content import bold_reading_string
2022-09-02 10:29:42 +00:00
from roles import is_devops
2023-08-13 09:58:02 +00:00
from session import get_json_valid
2022-11-09 12:10:29 +00:00
from session import site_is_verified
from session import get_json
2023-08-23 19:20:55 +00:00
from shares import actor_attached_shares_as_html
from git import get_repo_url
from reading import html_profile_book_list
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
THEME_FORMATS = '.zip, .gz'
2023-02-10 12:02:10 +00:00
BLOCKFILE_FORMATS = '.csv'
2021-12-30 23:46:06 +00:00
2020-11-09 22:44:03 +00:00
def _import_export_blocks(translate: {}) -> str:
""" import and export blocks on edit profile screen
"""
edit_profile_form = \
' <label class="labels">' + \
translate['Import Blocks'] + '</label>\n'
edit_profile_form += ' <input type="file" id="importBlocks" '
edit_profile_form += 'name="importBlocks" '
edit_profile_form += 'accept="' + BLOCKFILE_FORMATS + '">\n'
edit_profile_form += \
' <label class="labels">' + \
translate['Export Blocks'] + '</label><br>\n'
edit_profile_form += \
' <button type="submit" class="button" ' + \
'name="submitExportBlocks">➤</button><br>\n'
return edit_profile_form
def _get_profile_short_description(profile_description: str) -> str:
"""Returns a short version of the profile description
"""
profile_description_short = profile_description
if '\n' in profile_description:
if len(profile_description.split('\n')) > 6:
profile_description_short = ''
else:
if '<br>' in profile_description:
if len(profile_description.split('<br>')) > 6:
profile_description_short = ''
# keep the profile description short
if len(profile_description_short) > 2048:
profile_description_short = profile_description_short[:2048]
return profile_description_short
2021-12-29 21:55:09 +00:00
def _valid_profile_preview_post(post_json_object: {},
2021-12-30 23:46:06 +00:00
person_url: str) -> (bool, {}):
2021-11-19 13:47:02 +00:00
"""Returns true if the given post should appear on a person/group profile
after searching for a handle
"""
2022-12-30 19:40:48 +00:00
if not isinstance(post_json_object, dict):
return False, None
2021-12-30 23:46:06 +00:00
is_announced_feed_item = False
2021-12-29 21:55:09 +00:00
if is_create_inside_announce(post_json_object):
2021-12-30 23:46:06 +00:00
is_announced_feed_item = True
2021-12-25 22:09:19 +00:00
post_json_object = post_json_object['object']
2023-07-07 08:33:26 +00:00
if not post_json_object.get('type'):
return False, None
2021-12-25 22:09:19 +00:00
if post_json_object['type'] == 'Create':
2021-12-26 10:57:03 +00:00
if not has_object_dict(post_json_object):
2021-11-19 13:47:02 +00:00
return False, None
2023-12-09 19:18:14 +00:00
if post_json_object['type'] not in ('Create', 'Announce'):
2024-01-09 10:41:19 +00:00
if post_json_object['type'] not in ('Note', 'Event', 'Video', 'Page'):
2021-11-19 13:47:02 +00:00
return False, None
2021-12-25 22:09:19 +00:00
if not post_json_object.get('to'):
2021-11-19 13:47:02 +00:00
return False, None
2021-12-25 22:09:19 +00:00
if not post_json_object.get('id'):
2021-11-19 13:47:02 +00:00
return False, None
# wrap in create
2024-12-23 18:23:47 +00:00
cc_list: list[str] = []
2021-12-25 22:09:19 +00:00
if post_json_object.get('cc'):
2021-12-30 23:46:06 +00:00
cc_list = post_json_object['cc']
new_post_json_object = {
2021-12-25 22:09:19 +00:00
'object': post_json_object,
'to': post_json_object['to'],
2021-12-30 23:46:06 +00:00
'cc': cc_list,
2021-12-25 22:09:19 +00:00
'id': post_json_object['id'],
2021-12-30 23:46:06 +00:00
'actor': person_url,
2021-11-19 13:47:02 +00:00
'type': 'Create'
}
2021-12-30 23:46:06 +00:00
post_json_object = new_post_json_object
2021-12-25 22:09:19 +00:00
if not post_json_object.get('actor'):
2021-11-19 13:47:02 +00:00
return False, None
2023-02-12 22:21:40 +00:00
# convert actor back to id
2023-02-12 21:42:53 +00:00
if isinstance(post_json_object['actor'], dict):
2024-01-09 16:59:23 +00:00
actor_url = get_actor_from_post(post_json_object)
if actor_url:
post_json_object['actor'] = actor_url
2023-02-12 22:21:40 +00:00
if has_object_dict(post_json_object):
2023-02-12 22:24:01 +00:00
# convert attributedTo actor back to id
2023-02-12 22:21:40 +00:00
if post_json_object['object'].get('attributedTo'):
2024-02-04 13:18:08 +00:00
attrib_field = post_json_object['object']['attributedTo']
if attrib_field:
if isinstance(attrib_field, dict):
attrib = get_attributed_to(attrib_field)
if attrib:
post_json_object['object']['attributedTo'] = attrib
2023-02-12 22:21:40 +00:00
if not is_announced_feed_item:
2024-01-09 16:59:23 +00:00
actor_url = get_actor_from_post(post_json_object)
if actor_url != person_url and \
2022-03-12 20:09:43 +00:00
post_json_object['object']['type'] != 'Page':
return False, None
2021-12-25 22:09:19 +00:00
return True, post_json_object
2021-11-19 13:47:02 +00:00
def html_profile_after_search(authorized: bool,
recent_posts_cache: {}, max_recent_posts: int,
2021-12-29 21:55:09 +00:00
translate: {},
base_dir: str, path: str, http_prefix: str,
nickname: str, domain: str, port: int,
2021-12-30 23:46:06 +00:00
profile_handle: str,
2021-12-29 21:55:09 +00:00
session, cached_webfingers: {}, person_cache: {},
debug: bool, project_version: str,
yt_replace_domain: str,
twitter_replacement_domain: str,
show_published_date_only: bool,
2021-12-30 23:46:06 +00:00
default_timeline: str,
2021-12-29 21:55:09 +00:00
peertube_instances: [],
allow_local_network_access: bool,
theme_name: str,
2021-12-30 23:46:06 +00:00
access_keys: {},
2021-12-29 21:55:09 +00:00
system_language: str,
max_like_count: int,
signing_priv_key_pem: str,
2022-02-25 19:12:40 +00:00
cw_lists: {}, lists_enabled: str,
timezone: str,
2022-03-24 13:14:41 +00:00
onion_domain: str, i2p_domain: str,
bold_reading: bool, dogwhistles: {},
2023-01-13 15:04:48 +00:00
min_images_for_accounts: [],
buy_sites: {},
max_shares_on_profile: int,
no_of_books: int,
2024-12-17 13:50:48 +00:00
auto_cw_cache: {},
mitm_servers: [],
ua_str: str) -> str:
2020-11-09 22:44:03 +00:00
"""Show a profile page after a search for a fediverse address
"""
2021-06-03 19:46:35 +00:00
http = False
gnunet = False
2022-04-29 13:54:13 +00:00
ipfs = False
ipns = False
2021-12-25 17:09:22 +00:00
if http_prefix == 'http':
2021-06-03 19:46:35 +00:00
http = True
2021-12-25 17:09:22 +00:00
elif http_prefix == 'gnunet':
2021-06-03 19:46:35 +00:00
gnunet = True
2022-04-29 13:54:13 +00:00
elif http_prefix == 'ipfs':
ipfs = True
elif http_prefix == 'ipns':
ipns = True
from_domain = domain
if onion_domain:
if '.onion/' in profile_handle or profile_handle.endswith('.onion'):
from_domain = onion_domain
http = True
if i2p_domain:
if '.i2p/' in profile_handle or profile_handle.endswith('.i2p'):
from_domain = i2p_domain
http = True
2021-12-30 23:46:06 +00:00
profile_json, as_header = \
2022-04-29 13:54:13 +00:00
get_actor_json(from_domain, profile_handle, http,
gnunet, ipfs, ipns, debug, False,
2024-12-17 13:50:48 +00:00
signing_priv_key_pem, session,
mitm_servers)
2021-12-26 10:08:06 +00:00
if not profile_json:
2020-11-09 22:44:03 +00:00
return None
2022-09-15 11:26:22 +00:00
if not profile_json.get('id'):
return None
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
person_url = profile_json['id']
search_domain, search_port = get_domain_from_actor(person_url)
if not search_domain:
return None
2021-12-30 23:46:06 +00:00
search_nickname = get_nickname_from_actor(person_url)
if not search_nickname:
return None
2021-12-30 23:46:06 +00:00
search_domain_full = get_full_domain(search_domain, search_port)
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
profile_str = ''
css_filename = base_dir + '/epicyon-profile.css'
2021-12-25 16:17:53 +00:00
if os.path.isfile(base_dir + '/epicyon.css'):
2021-12-30 23:46:06 +00:00
css_filename = base_dir + '/epicyon.css'
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
is_group = False
2021-12-26 10:08:06 +00:00
if profile_json.get('type'):
if profile_json['type'] == 'Group':
2021-12-30 23:46:06 +00:00
is_group = True
2021-09-21 18:57:12 +00:00
# shared items attached to the actor
# https://codeberg.org/fediverse/fep/src/branch/main/fep/0837/fep-0837.md
attached_shared_items = \
actor_attached_shares_as_html(profile_json, max_shares_on_profile)
2021-12-30 23:46:06 +00:00
avatar_url = ''
2021-12-26 10:08:06 +00:00
if profile_json.get('icon'):
2024-03-13 20:09:03 +00:00
icon_dict = profile_json['icon']
if isinstance(icon_dict, list):
if len(icon_dict) > 0:
2024-03-13 20:11:12 +00:00
icon_dict = icon_dict[-1]
2024-03-13 20:09:03 +00:00
if isinstance(icon_dict, dict):
if icon_dict.get('url'):
url_str = get_url_from_post(icon_dict['url'])
2024-03-13 20:00:55 +00:00
avatar_url = remove_html(url_str)
2021-12-30 23:46:06 +00:00
if not avatar_url:
2022-06-12 12:30:14 +00:00
avatar_url = get_person_avatar_url(base_dir, person_url, person_cache)
2021-12-30 23:46:06 +00:00
display_name = search_nickname
2021-12-26 10:08:06 +00:00
if profile_json.get('name'):
2021-12-30 23:46:06 +00:00
display_name = profile_json['name']
2023-02-07 17:31:17 +00:00
display_name = remove_html(display_name)
2024-02-04 13:52:31 +00:00
display_name = uninvert_text(display_name)
display_name = \
add_emoji_to_display_name(session, base_dir, http_prefix,
nickname, domain,
2023-02-07 17:41:30 +00:00
display_name, False, translate)
2021-12-30 23:46:06 +00:00
locked_account = get_locked_account(profile_json)
if locked_account:
display_name += '🔒'
2024-08-12 11:16:32 +00:00
pronouns = get_pronouns(profile_json)
discord = get_discord(profile_json)
2024-08-15 19:58:26 +00:00
art_site_url = get_art_site_url(profile_json)
music_site_url = get_music_site_url(profile_json)
youtube = get_youtube(profile_json)
peertube = get_peertube(profile_json)
pixelfed = get_pixelfed(profile_json)
2024-08-12 11:16:32 +00:00
2021-12-30 23:46:06 +00:00
moved_to = ''
2021-12-26 10:08:06 +00:00
if profile_json.get('movedTo'):
2021-12-30 23:46:06 +00:00
moved_to = profile_json['movedTo']
if '"' in moved_to:
moved_to = moved_to.split('"')[1]
2023-07-13 08:22:49 +00:00
moved_to = remove_html(moved_to)
2021-12-30 23:46:06 +00:00
display_name += ''
2023-09-19 12:49:09 +00:00
you_follow = \
is_following_actor(base_dir,
nickname, domain, person_url)
2021-12-30 23:46:06 +00:00
follows_you = \
2021-12-29 21:55:09 +00:00
is_follower_of_person(base_dir,
nickname, domain,
2021-12-30 23:46:06 +00:00
search_nickname,
search_domain_full)
2021-12-30 23:46:06 +00:00
profile_description = ''
2021-12-26 10:08:06 +00:00
if profile_json.get('summary'):
2023-07-13 08:22:49 +00:00
if not dangerous_markup(profile_json['summary'],
False, []):
profile_description = profile_json['summary']
2023-07-28 13:33:42 +00:00
else:
profile_description = remove_html(profile_json['summary'])
profile_description = \
remove_link_trackers_from_content(profile_description)
profile_description = \
add_emoji_to_display_name(session, base_dir, http_prefix,
nickname, domain,
2023-02-07 17:41:30 +00:00
profile_description, False, translate)
2023-05-03 11:07:34 +00:00
featured_hashtags = \
get_featured_hashtags_as_html(profile_json, profile_description)
2021-12-30 23:46:06 +00:00
outbox_url = None
2021-12-26 10:08:06 +00:00
if not profile_json.get('outbox'):
2020-11-12 17:05:38 +00:00
if debug:
2021-12-26 10:08:06 +00:00
pprint(profile_json)
2020-11-12 17:05:38 +00:00
print('DEBUG: No outbox found')
return None
2021-12-30 23:46:06 +00:00
outbox_url = profile_json['outbox']
2020-11-12 17:05:38 +00:00
# profileBackgroundImage = ''
2021-12-26 10:08:06 +00:00
# if profile_json.get('image'):
# if profile_json['image'].get('url'):
2023-12-09 14:18:24 +00:00
# url_str = get_url_from_post(profile_json['image']['url'])
# profileBackgroundImage = remove_html(url_str)
2020-11-12 17:05:38 +00:00
# url to return to
2021-12-30 23:46:06 +00:00
back_url = path
if not back_url.endswith('/inbox'):
back_url += '/inbox'
profile_description_short = \
_get_profile_short_description(profile_description)
2020-11-12 17:05:38 +00:00
# remove formatting from profile description used on title
2021-12-30 23:46:06 +00:00
avatar_description = ''
2021-12-26 10:08:06 +00:00
if profile_json.get('summary'):
if isinstance(profile_json['summary'], str):
2021-12-30 23:46:06 +00:00
avatar_description = \
2021-12-26 10:08:06 +00:00
profile_json['summary'].replace('<br>', '\n')
2021-12-30 23:46:06 +00:00
avatar_description = avatar_description.replace('<p>', '')
avatar_description = avatar_description.replace('</p>', '')
if '<' in avatar_description:
avatar_description = remove_html(avatar_description)
2020-11-12 22:41:41 +00:00
2021-12-30 23:46:06 +00:00
image_url = ''
2021-12-26 10:08:06 +00:00
if profile_json.get('image'):
if profile_json['image'].get('url'):
2023-12-09 14:18:24 +00:00
url_str = get_url_from_post(profile_json['image']['url'])
image_url = remove_html(url_str)
2020-11-12 23:44:16 +00:00
2021-12-30 23:46:06 +00:00
also_known_as = None
2021-12-26 10:08:06 +00:00
if profile_json.get('alsoKnownAs'):
2023-07-13 08:22:49 +00:00
also_known_as = remove_html(profile_json['alsoKnownAs'])
elif profile_json.get('sameAs'):
2023-07-13 08:22:49 +00:00
also_known_as = remove_html(profile_json['sameAs'])
2021-01-22 20:48:52 +00:00
2021-12-30 23:46:06 +00:00
joined_date = None
2021-12-26 10:08:06 +00:00
if profile_json.get('published'):
if 'T' in profile_json['published']:
2023-07-13 08:22:49 +00:00
joined_date = remove_html(profile_json['published'])
2023-05-12 14:39:26 +00:00
actor_proxied = ap_proxy_type(profile_json)
2023-05-12 13:47:28 +00:00
website_url = get_website(profile_json, translate)
repo_url = get_repo_url(profile_json)
# is sending posts to this account blocked?
send_blocks_str = ''
2024-12-24 18:17:36 +00:00
if sending_is_blocked2(base_dir, nickname, domain,
search_domain_full, person_url):
send_block_filename = \
acct_dir(base_dir, nickname, domain) + '/send_blocks.txt'
if text_in_file(person_url,
send_block_filename, False):
send_blocks_str = translate['FollowAccountWarning']
elif text_in_file('://' + search_domain_full + '\n',
send_block_filename, False):
send_blocks_str = translate['FollowWarning']
2024-01-07 22:19:29 +00:00
birth_date = ''
if profile_json.get('vcard:bday'):
birth_date = profile_json['vcard:bday']
2021-12-30 23:46:06 +00:00
profile_str = \
_get_profile_header_after_search(base_dir, nickname, domain,
default_timeline,
2021-12-30 23:46:06 +00:00
search_nickname,
search_domain_full,
2021-12-29 21:55:09 +00:00
translate,
2024-08-12 11:16:32 +00:00
display_name, pronouns,
2023-09-19 12:49:09 +00:00
you_follow, follows_you,
2021-12-30 23:46:06 +00:00
profile_description_short,
2023-05-03 11:00:46 +00:00
featured_hashtags,
2021-12-30 23:46:06 +00:00
avatar_url, image_url,
moved_to, profile_json['id'],
also_known_as, access_keys,
joined_date, actor_proxied,
attached_shared_items,
website_url, repo_url,
send_blocks_str,
authorized,
2024-01-07 22:19:29 +00:00
person_url, no_of_books,
birth_date,
youtube, peertube, pixelfed,
2024-08-15 19:58:26 +00:00
discord, music_site_url,
art_site_url)
2020-11-12 22:41:41 +00:00
2021-12-26 12:45:03 +00:00
domain_full = get_full_domain(domain, port)
2020-11-20 12:14:22 +00:00
2021-12-30 23:46:06 +00:00
follow_is_permitted = True
2021-12-26 10:08:06 +00:00
if not profile_json.get('followers'):
# no followers collection specified within actor
2021-12-30 23:46:06 +00:00
follow_is_permitted = False
elif search_nickname == 'news' and search_domain_full == domain_full:
2020-11-20 12:20:34 +00:00
# currently the news actor is not something you can follow
2021-12-30 23:46:06 +00:00
follow_is_permitted = False
elif search_nickname == nickname and search_domain_full == domain_full:
2020-11-20 12:20:34 +00:00
# don't follow yourself!
2021-12-30 23:46:06 +00:00
follow_is_permitted = False
2020-11-20 12:20:34 +00:00
2022-05-14 17:07:20 +00:00
blocked = \
is_blocked(base_dir, nickname, domain, search_nickname, search_domain,
None, None)
2022-05-14 17:07:20 +00:00
2021-12-30 23:46:06 +00:00
if follow_is_permitted:
follow_str = 'Follow'
if is_group:
follow_str = 'Join'
2021-09-21 18:57:12 +00:00
2021-12-30 23:46:06 +00:00
profile_str += \
2021-07-22 10:22:01 +00:00
'<div class="container">\n' + \
' <form method="POST" action="' + \
2021-12-30 23:46:06 +00:00
back_url + '/followconfirm">\n' + \
2022-05-14 17:07:20 +00:00
' <center>\n'
profile_str += \
2020-11-20 12:14:22 +00:00
' <input type="hidden" name="actor" value="' + \
person_url + '">\n'
2023-09-19 12:49:09 +00:00
if not you_follow:
if is_moderator(base_dir, nickname):
profile_str += \
' <button type="submit" class="button" ' + \
'name="submitInfo" ' + \
'accesskey="' + access_keys['infoButton'] + '">' + \
translate['Info'] + '</button>\n'
profile_str += \
' <button type="submit" class="button" ' + \
'name="submitYes" ' + \
'accesskey="' + access_keys['followButton'] + '">' + \
translate[follow_str] + '</button>\n'
2022-05-14 17:07:20 +00:00
profile_str += \
2021-07-28 14:02:07 +00:00
' <button type="submit" class="button" name="submitView" ' + \
2021-12-30 23:46:06 +00:00
'accesskey="' + access_keys['viewButton'] + '">' + \
2022-05-14 17:07:20 +00:00
translate['View'] + '</button>\n'
if blocked:
profile_str += \
' <button type="submit" ' + \
'class="button" name="submitUnblock" ' + \
'accesskey="' + access_keys['unblockButton'] + '">' + \
translate['Unblock'] + '</button>\n'
profile_str += \
2021-07-28 14:02:07 +00:00
' </center>\n' + \
' </form>\n' + \
'</div>\n'
else:
2021-12-30 23:46:06 +00:00
profile_str += \
2021-07-28 14:02:07 +00:00
'<div class="container">\n' + \
' <form method="POST" action="' + \
2021-12-30 23:46:06 +00:00
back_url + '/followconfirm">\n' + \
2021-07-28 14:02:07 +00:00
' <center>\n' + \
' <input type="hidden" name="actor" value="' + \
2021-12-30 23:46:06 +00:00
person_url + '">\n' + \
2021-07-28 14:02:07 +00:00
' <button type="submit" class="button" name="submitView" ' + \
2021-12-30 23:46:06 +00:00
'accesskey="' + access_keys['viewButton'] + '">' + \
2021-07-28 14:02:07 +00:00
translate['View'] + '</button>\n' + \
' </center>\n' + \
' </form>\n' + \
'</div>\n'
2020-11-12 17:05:38 +00:00
2023-01-01 17:53:22 +00:00
text_mode_separator = '<div class="transparent"><hr></div>'
2021-12-30 23:46:06 +00:00
user_feed = \
2021-12-29 21:55:09 +00:00
parse_user_feed(signing_priv_key_pem,
2021-12-30 23:46:06 +00:00
session, outbox_url, as_header, project_version,
2024-12-17 13:50:48 +00:00
http_prefix, from_domain, debug, 0,
mitm_servers)
2023-10-12 14:04:51 +00:00
if not user_feed:
if debug:
print('DEBUG: no user feed in profile preview')
else:
minimize_all_images = False
if nickname in min_images_for_accounts:
minimize_all_images = True
2021-08-02 20:17:55 +00:00
i = 0
2021-12-30 23:46:06 +00:00
for item in user_feed:
2024-07-26 09:51:57 +00:00
if not isinstance(item, dict):
continue
if item.get('type') and item.get('object'):
if str(item['type']) == 'Announce' and \
isinstance(item['object'], str):
# resolve the announce
2023-06-18 14:47:04 +00:00
profile_str2 = 'https://www.w3.org/ns/activitystreams'
as_header2_str = 'application/ld+json; profile="' + \
2023-06-18 14:47:04 +00:00
profile_str2 + '"'
as_header2 = {
'Accept': as_header2_str
}
item = \
get_json(signing_priv_key_pem, session, item['object'],
2024-12-17 13:50:48 +00:00
as_header2, None, debug, mitm_servers,
__version__, http_prefix, from_domain)
2023-06-18 14:50:42 +00:00
if debug:
print('DEBUG: resolved public feed announce ' +
str(item))
2023-08-13 09:58:02 +00:00
if not get_json_valid(item):
2023-10-12 14:04:51 +00:00
if debug:
print('DEBUG: ' +
'announce json is not valid in timeline ' +
str(item))
continue
2021-12-30 23:46:06 +00:00
show_item, post_json_object = \
_valid_profile_preview_post(item, person_url)
if not show_item:
2023-02-12 22:35:13 +00:00
if debug:
print('DEBUG: item not valid in profile posts: ' +
str(item))
continue
2020-12-18 18:12:33 +00:00
2023-10-12 15:53:19 +00:00
profile_post_html = \
2021-12-29 21:55:09 +00:00
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,
2021-12-30 23:46:06 +00:00
post_json_object, avatar_url,
2021-12-29 21:55:09 +00:00
False, False,
http_prefix, project_version, 'inbox',
yt_replace_domain,
twitter_replacement_domain,
show_published_date_only,
peertube_instances,
allow_local_network_access,
theme_name, system_language,
max_like_count,
False, False, False,
False, False, False,
2022-02-25 19:12:40 +00:00
cw_lists, lists_enabled,
2022-03-24 13:14:41 +00:00
timezone, False,
bold_reading, dogwhistles,
2023-01-13 15:04:48 +00:00
minimize_all_images, None,
2024-12-17 13:50:48 +00:00
buy_sites, auto_cw_cache,
mitm_servers)
2023-10-12 15:53:19 +00:00
if not profile_post_html:
if debug:
print('DEBUG: no html produced for profile post: ' +
str(item))
continue
profile_str += text_mode_separator + profile_post_html
2021-08-02 20:17:55 +00:00
i += 1
if i >= 8:
2021-08-02 20:17:55 +00:00
break
2020-11-09 22:44:03 +00:00
if text_mode_browser(ua_str):
profile_str = text_mode_removals(profile_str, translate)
2021-12-30 23:46:06 +00:00
instance_title = get_config_param(base_dir, 'instanceTitle')
2024-12-23 18:23:47 +00:00
preload_images: list[str] = []
2021-12-30 23:46:06 +00:00
return html_header_with_external_style(css_filename,
2024-10-22 12:12:16 +00:00
instance_title, None,
preload_images) + \
profile_str + text_mode_separator + html_footer()
2021-12-29 21:55:09 +00:00
2023-08-25 19:47:33 +00:00
def _profile_shared_items_list(attached_shared_items: str,
translate: {}) -> str:
"""Complete html for shared items shown on profile
"""
return \
'<p><b>' + translate['Shares'] + ':</b><br>\n' + \
attached_shared_items + '</p>\n'
2022-08-20 13:18:26 +00:00
def _get_profile_header(base_dir: str, http_prefix: str, nickname: str,
domain: str, domain_full: str, translate: {},
2021-12-30 23:46:06 +00:00
default_timeline: str,
2024-08-12 10:57:15 +00:00
display_name: str, pronouns: str,
2021-12-30 23:46:06 +00:00
profile_description_short: str,
2023-05-03 11:00:46 +00:00
featured_hashtags: str,
2021-12-30 23:46:06 +00:00
login_button: str, avatar_url: str,
theme: str, moved_to: str,
also_known_as: [],
pinned_content: str,
2023-08-23 19:20:55 +00:00
attached_shared_items: str,
2021-12-30 23:46:06 +00:00
access_keys: {},
joined_date: str,
2023-05-12 13:47:28 +00:00
occupation_name: str,
actor_proxied: str,
person_url: str,
no_of_books: int,
2024-01-07 22:19:29 +00:00
authorized: bool,
2024-03-20 20:42:12 +00:00
birth_date: str,
premium: bool) -> str:
2020-11-12 22:33:00 +00:00
"""The header of the profile screen, containing background
image and avatar
"""
2022-08-20 13:18:26 +00:00
banner_file, _ = \
get_profile_background_file(base_dir, nickname, domain, theme)
2021-12-30 23:46:06 +00:00
html_str = \
2021-07-22 10:22:01 +00:00
'\n\n <figure class="profileHeader">\n' + \
' <a href="/users/' + \
2021-12-30 23:46:06 +00:00
nickname + '/' + default_timeline + '" title="' + \
2022-05-25 13:54:53 +00:00
translate['Switch to timeline view'] + '" tabindex="1" ' + \
'accesskey="' + access_keys['menuTimeline'] + '">\n' + \
2021-07-22 10:22:01 +00:00
' <img class="profileBackground" ' + \
'alt="" ' + \
2022-08-20 13:18:26 +00:00
'src="/users/' + nickname + '/' + banner_file + '" /></a>\n' + \
2021-07-22 10:22:01 +00:00
' <figcaption>\n' + \
2020-11-12 23:17:37 +00:00
' <a href="/users/' + \
2021-12-30 23:46:06 +00:00
nickname + '/' + default_timeline + '" title="' + \
2020-11-12 23:17:37 +00:00
translate['Switch to timeline view'] + '">\n' + \
2022-03-28 08:47:53 +00:00
' <img loading="lazy" decoding="async" ' + \
'src="' + avatar_url + '" alt="" class="title"></a>\n'
2021-05-12 18:31:41 +00:00
2021-12-30 23:46:06 +00:00
occupation_str = ''
if occupation_name:
occupation_str += \
' <b>' + occupation_name + '</b><br>\n'
2021-05-12 18:31:41 +00:00
2024-03-20 20:42:12 +00:00
html_str += ' <h1>' + display_name + '\n</h1>\n'
if premium:
html_str += \
' <b>' + translate['Premium account'] + '</b><br>\n'
2024-08-12 10:57:15 +00:00
if pronouns:
2024-08-12 11:19:27 +00:00
html_str += ' ' + pronouns + '<br><br>\n'
2024-08-12 10:57:15 +00:00
2024-03-20 20:42:12 +00:00
html_str += occupation_str
2021-05-12 18:31:41 +00:00
2023-05-19 20:58:23 +00:00
# show if the actor is proxied
2023-05-12 13:47:28 +00:00
if not actor_proxied:
actor_proxied = ''
else:
actor_proxied = remove_html(actor_proxied)
2024-01-27 17:04:21 +00:00
if resembles_url(actor_proxied):
2023-05-12 14:11:42 +00:00
proxy_str = 'Proxy'
if translate.get(proxy_str):
proxy_str = translate[proxy_str]
actor_proxied = '<a href="' + actor_proxied + \
'" target="_blank" rel="nofollow noopener noreferrer">' + \
proxy_str + '</a>'
elif '/' in actor_proxied:
2023-05-12 13:47:28 +00:00
actor_proxied = actor_proxied.split('/')[-1]
actor_proxied = ' [' + actor_proxied + ']'
2023-05-19 20:58:23 +00:00
# show blog icon if this account has a blog
2023-05-19 20:44:43 +00:00
acct_blog_str = ''
2023-05-19 20:50:47 +00:00
has_blog = account_has_blog(base_dir, nickname, domain)
2023-05-19 20:44:43 +00:00
if has_blog:
acct_blog_str = \
2023-05-19 20:50:47 +00:00
' <a href="/blog/' + nickname + '" title="' + \
2023-05-19 20:44:43 +00:00
translate['Blog'] + '">📖</a>'
2023-05-19 20:50:47 +00:00
html_str += \
' <p><b>@' + nickname + '@' + domain_full + \
actor_proxied + acct_blog_str + '</b><br>\n'
2021-12-30 23:46:06 +00:00
if joined_date:
joined_str = translate['Joined']
if time_days_ago(joined_date) < 7:
joined_str = '<b>' + translate['New account'] + '</b>'
2021-12-30 23:46:06 +00:00
html_str += \
' <p>' + joined_str + ' ' + \
2021-12-30 23:46:06 +00:00
joined_date.split('T')[0] + '<br>\n'
if moved_to:
new_nickname = get_nickname_from_actor(moved_to)
new_domain, new_port = get_domain_from_actor(moved_to)
if new_nickname and new_domain:
2023-01-15 14:33:18 +00:00
new_domain_full = get_full_domain(new_domain, new_port)
2021-12-30 23:46:06 +00:00
html_str += \
' <p>' + translate['New account'] + ': ' + \
2021-12-30 23:46:06 +00:00
'<a href="' + moved_to + '">@' + \
new_nickname + '@' + new_domain_full + '</a><br>\n'
elif also_known_as:
other_accounts_html = \
2021-01-22 20:59:43 +00:00
' <p>' + translate['Other accounts'] + ': '
2021-12-26 10:19:59 +00:00
actor = local_actor_url(http_prefix, nickname, domain_full)
ctr = 0
2021-12-30 23:46:06 +00:00
if isinstance(also_known_as, list):
for alt_actor in also_known_as:
if alt_actor == actor:
continue
2021-01-22 20:59:43 +00:00
if ctr > 0:
2021-12-30 23:46:06 +00:00
other_accounts_html += ' '
2021-01-22 20:59:43 +00:00
ctr += 1
2021-12-30 23:46:06 +00:00
alt_domain, _ = get_domain_from_actor(alt_actor)
2023-01-15 14:33:18 +00:00
if alt_domain:
other_accounts_html += \
'<a href="' + alt_actor + \
'" tabindex="1">' + alt_domain + '</a>'
2021-12-30 23:46:06 +00:00
elif isinstance(also_known_as, str):
if also_known_as != actor:
ctr += 1
2021-12-30 23:46:06 +00:00
alt_domain, _ = get_domain_from_actor(also_known_as)
2023-01-15 14:33:18 +00:00
if alt_domain:
other_accounts_html += \
'<a href="' + also_known_as + '">' + \
alt_domain + '</a>'
2021-12-30 23:46:06 +00:00
other_accounts_html += '</p>\n'
if ctr > 0:
2021-12-30 23:46:06 +00:00
html_str += other_accounts_html
2024-01-07 22:19:29 +00:00
if is_valid_date(birth_date):
birth_date = remove_html(birth_date)
html_str += \
' <p>' + translate['Birthday'] + ': ' + \
birth_date + '</p>\n'
2023-05-03 11:00:46 +00:00
if featured_hashtags:
featured_hashtags += '\n'
2021-12-30 23:46:06 +00:00
html_str += \
2020-11-12 23:00:36 +00:00
' <a href="/users/' + nickname + \
2020-11-12 22:33:00 +00:00
'/qrcode.png" alt="' + translate['QR Code'] + '" title="' + \
2022-05-25 14:09:01 +00:00
translate['QR Code'] + '" tabindex="1">' + \
2021-02-01 18:42:39 +00:00
'<img class="qrcode" alt="' + translate['QR Code'] + \
2021-07-22 10:22:01 +00:00
'" src="/icons/qrcode.png" /></a></p>\n' + \
2023-05-03 11:00:46 +00:00
' <p>' + profile_description_short + '</p>\n' + \
featured_hashtags + login_button
2021-12-30 23:46:06 +00:00
if pinned_content:
html_str += pinned_content.replace('<p>', '<p>📎', 1)
2023-08-23 19:20:55 +00:00
if attached_shared_items:
html_str += \
2023-08-25 19:47:33 +00:00
_profile_shared_items_list(attached_shared_items,
translate)
# show vcard download link
# NOTE: don't use download="preferredfilename" which is
# unsupported by some browsers
html_str += \
' <a href="/users/' + nickname + '.vcf" ' + \
'type="text/vcard" tabindex="1" class="imageAnchor">' + \
2024-08-14 11:06:48 +00:00
'<img class="vcard" src="/icons/vcard.png" ' + \
'title="vCard" alt="vCard" /></a>\n'
2021-12-30 23:46:06 +00:00
html_str += \
2021-07-22 10:22:01 +00:00
' </figcaption>\n' + \
' </figure>\n\n'
# book events for this actor
html_str += html_profile_book_list(base_dir, person_url, no_of_books,
translate, nickname, domain,
authorized)
2021-12-30 23:46:06 +00:00
return html_str
2020-11-12 22:33:00 +00:00
def _get_profile_header_after_search(base_dir: str,
nickname: str, domain: str,
default_timeline: str,
2021-12-30 23:46:06 +00:00
search_nickname: str,
search_domain_full: str,
2021-12-29 21:55:09 +00:00
translate: {},
2021-12-30 23:46:06 +00:00
display_name: str,
2024-08-12 11:16:32 +00:00
pronouns: str,
2023-09-19 12:49:09 +00:00
you_follow: bool,
2021-12-30 23:46:06 +00:00
follows_you: bool,
profile_description_short: str,
2023-05-03 11:00:46 +00:00
featured_hashtags: str,
2021-12-30 23:46:06 +00:00
avatar_url: str, image_url: str,
moved_to: str, actor: str,
also_known_as: [],
access_keys: {},
2023-05-12 13:47:28 +00:00
joined_date: str,
actor_proxied: str,
attached_shared_items: str,
website_url: str,
repo_url: str,
send_blocks_str: str,
authorized: bool,
person_url: str,
2024-01-07 22:19:29 +00:00
no_of_books: str,
birth_date: str,
youtube: str,
peertube: str,
pixelfed: str,
discord: str,
2024-08-15 19:58:26 +00:00
music_site_url: str,
art_site_url: str) -> str:
2020-11-12 23:38:58 +00:00
"""The header of a searched for handle, containing background
image and avatar
"""
2021-12-30 23:46:06 +00:00
if not image_url:
image_url = '/defaultprofilebackground'
html_str = \
2021-07-22 10:22:01 +00:00
'\n\n <figure class="profileHeader">\n' + \
' <a href="/users/' + \
2021-12-30 23:46:06 +00:00
nickname + '/' + default_timeline + '" title="' + \
2021-04-23 12:04:42 +00:00
translate['Switch to timeline view'] + '" ' + \
2022-05-25 13:50:43 +00:00
'accesskey="' + access_keys['menuTimeline'] + '" tabindex="1">\n' + \
2021-07-22 10:22:01 +00:00
' <img class="profileBackground" ' + \
'alt="" ' + \
2021-12-30 23:46:06 +00:00
'src="' + image_url + '" /></a>\n' + \
2021-07-22 10:22:01 +00:00
' <figcaption>\n'
2021-12-30 23:46:06 +00:00
if avatar_url:
html_str += \
2021-07-22 10:22:01 +00:00
' <a href="/users/' + \
2021-12-30 23:46:06 +00:00
nickname + '/' + default_timeline + '" title="' + \
2021-07-22 10:22:01 +00:00
translate['Switch to timeline view'] + '">\n' + \
2022-03-28 08:47:53 +00:00
' <img loading="lazy" decoding="async" src="' + \
avatar_url + '" ' + 'alt="" class="title"></a>\n'
2021-12-30 23:46:06 +00:00
if not display_name:
display_name = search_nickname
2023-05-12 14:11:42 +00:00
2023-05-12 13:47:28 +00:00
if not actor_proxied:
actor_proxied = ''
else:
actor_proxied = remove_html(actor_proxied)
2024-01-27 17:04:21 +00:00
if resembles_url(actor_proxied):
2023-05-12 14:11:42 +00:00
proxy_str = 'Proxy'
if translate.get(proxy_str):
proxy_str = translate[proxy_str]
actor_proxied = '<a href="' + actor_proxied + \
'" target="_blank" rel="nofollow noopener noreferrer">' + \
proxy_str + '</a>'
elif '/' in actor_proxied:
2023-05-12 13:47:28 +00:00
actor_proxied = actor_proxied.split('/')[-1]
actor_proxied = ' [' + actor_proxied + ']'
2023-05-12 14:11:42 +00:00
2021-12-30 23:46:06 +00:00
html_str += \
2022-11-02 19:16:07 +00:00
' <h1>\n' + \
2023-02-07 17:31:17 +00:00
' ' + display_name + '\n' + \
2022-11-02 19:16:07 +00:00
' </h1>\n' + \
2024-08-12 11:16:32 +00:00
' <p><b>'
if pronouns:
2024-08-12 11:20:41 +00:00
html_str += pronouns + '<br><br>\n'
2024-08-12 11:16:32 +00:00
html_str += \
'@' + search_nickname + '@' + search_domain_full + \
2023-05-12 13:47:28 +00:00
actor_proxied + '</b><br>\n'
2021-12-30 23:46:06 +00:00
if joined_date:
joined_str = translate['Joined']
if time_days_ago(joined_date) < 7:
joined_str = '<b>' + translate['New account'] + '</b>'
html_str += ' <p>' + joined_str + ' ' + \
2021-12-30 23:46:06 +00:00
joined_date.split('T')[0] + '</p>\n'
if follows_you:
2023-09-19 12:49:09 +00:00
if not you_follow:
html_str += ' <p><b>' + \
translate['Follows you'] + '</b></p>\n'
else:
html_str += ' <p><b>' + \
translate['Mutuals'] + '</b></p>\n'
if send_blocks_str:
html_str += ' <p><b>' + send_blocks_str + '</b></p>\n'
2021-12-30 23:46:06 +00:00
if moved_to:
new_nickname = get_nickname_from_actor(moved_to)
new_domain, new_port = get_domain_from_actor(moved_to)
if new_nickname and new_domain:
2023-01-15 14:33:18 +00:00
new_domain_full = get_full_domain(new_domain, new_port)
2021-12-30 23:46:06 +00:00
new_handle = new_nickname + '@' + new_domain_full
html_str += ' <p>' + translate['New account'] + \
': <a href="' + moved_to + '">@' + new_handle + '</a></p>\n'
elif also_known_as:
other_accounts_html = \
2021-01-22 20:48:52 +00:00
' <p>' + translate['Other accounts'] + ': '
ctr = 0
2021-12-30 23:46:06 +00:00
if isinstance(also_known_as, list):
for alt_actor in also_known_as:
if alt_actor == actor:
continue
2021-01-22 20:48:52 +00:00
if ctr > 0:
2021-12-30 23:46:06 +00:00
other_accounts_html += ' '
2021-01-22 20:48:52 +00:00
ctr += 1
2021-12-30 23:46:06 +00:00
alt_domain, _ = get_domain_from_actor(alt_actor)
2023-01-15 14:33:18 +00:00
if alt_domain:
other_accounts_html += \
'<a href="' + alt_actor + \
'" tabindex="1">' + alt_domain + '</a>'
2021-12-30 23:46:06 +00:00
elif isinstance(also_known_as, str):
if also_known_as != actor:
ctr += 1
2021-12-30 23:46:06 +00:00
alt_domain, _ = get_domain_from_actor(also_known_as)
2023-01-15 14:33:18 +00:00
if alt_domain:
other_accounts_html += \
'<a href="' + also_known_as + '">' + \
alt_domain + '</a>'
2021-12-30 23:46:06 +00:00
other_accounts_html += '</p>\n'
if ctr > 0:
2021-12-30 23:46:06 +00:00
html_str += other_accounts_html
2024-01-07 22:19:29 +00:00
if is_valid_date(birth_date):
birth_date = remove_html(birth_date)
html_str += \
' <p>' + translate['Birthday'] + ': ' + \
birth_date + '</p>\n'
2023-05-03 11:00:46 +00:00
if featured_hashtags:
featured_hashtags += '\n'
if website_url:
html_str += ' <p>🌐 <a href="' + website_url + '">' + \
website_url + '</a></p>\n'
if youtube:
html_str += ' <p>YouTube: <a href="' + youtube + '">' + \
youtube + '</a></p>\n'
if peertube:
html_str += ' <p>PeerTube: <a href="' + peertube + '">' + \
peertube + '</a></p>\n'
if pixelfed:
html_str += ' <p>Pixelfed: <a href="' + pixelfed + '">' + \
pixelfed + '</a></p>\n'
if discord:
html_str += ' <p>Discord: <a href="' + discord + '">' + \
discord + '</a></p>\n'
2024-08-15 19:58:26 +00:00
if art_site_url:
html_str += ' <p>' + translate['Art'] + ': <a href="' + \
art_site_url + '">' + art_site_url + '</a></p>\n'
if music_site_url:
html_str += ' <p>' + translate['Music'] + ': <a href="' + \
music_site_url + '">' + music_site_url + '</a></p>\n'
if repo_url:
html_str += ' <p>💻 <a href="' + repo_url + '">' + \
repo_url + '</a></p>\n'
2021-12-30 23:46:06 +00:00
html_str += \
' <p>' + profile_description_short + '</p>\n' + \
2023-12-20 18:55:50 +00:00
featured_hashtags
# show any notes about this account
if authorized:
handle = search_nickname + '@' + search_domain_full
person_notes = \
get_person_notes(base_dir, nickname, domain, handle)
if person_notes:
person_notes_html = person_notes.replace('\n', '<br>')
html_str += ' <p><b>' + \
translate['Notes'].upper() + ': ' + \
person_notes_html + '</b></p>\n'
2023-12-20 18:55:50 +00:00
html_str += \
' </figcaption>\n' + \
' </figure>\n\n'
if attached_shared_items:
html_str += \
_profile_shared_items_list(attached_shared_items,
translate)
# book events for this actor
html_str += html_profile_book_list(base_dir, person_url, no_of_books,
translate,
nickname, domain, authorized)
2021-12-30 23:46:06 +00:00
return html_str
2020-11-12 23:38:58 +00:00
2021-12-29 21:55:09 +00:00
def html_profile(signing_priv_key_pem: str,
rss_icon_at_top: bool,
2022-07-12 19:03:30 +00:00
icons_as_buttons: bool,
2021-12-30 23:46:06 +00:00
default_timeline: str,
2021-12-29 21:55:09 +00:00
recent_posts_cache: {}, max_recent_posts: int,
translate: {}, project_version: str,
base_dir: str, http_prefix: str, authorized: bool,
profile_json: {}, selected: str,
session, cached_webfingers: {}, person_cache: {},
yt_replace_domain: str,
twitter_replacement_domain: str,
show_published_date_only: bool,
newswire: {}, theme: str, dormant_months: int,
peertube_instances: [],
allow_local_network_access: bool,
text_mode_banner: str,
2021-12-30 23:46:06 +00:00
debug: bool, access_keys: {}, city: str,
2021-12-29 21:55:09 +00:00
system_language: str, max_like_count: int,
shared_items_federated_domains: [],
2022-05-30 18:33:51 +00:00
extra_json: {}, page_number: int,
2021-12-30 23:46:06 +00:00
max_items_per_page: int,
2021-12-29 21:55:09 +00:00
cw_lists: {}, lists_enabled: str,
2022-02-25 19:12:40 +00:00
content_license_url: str,
2023-01-13 15:04:48 +00:00
timezone: str, bold_reading: bool,
2023-05-12 13:47:28 +00:00
buy_sites: {},
actor_proxied: str,
2023-09-16 13:55:02 +00:00
max_shares_on_profile: int,
sites_unavailable: [],
no_of_books: int,
auto_cw_cache: {},
2024-12-17 13:50:48 +00:00
known_epicyon_instances: [],
mitm_servers: []) -> str:
2020-11-09 22:44:03 +00:00
"""Show the profile page as html
"""
2022-11-28 12:05:13 +00:00
show_moved_accounts = False
if authorized:
2024-05-12 12:35:26 +00:00
moved_accounts_filename = data_dir(base_dir) + '/actors_moved.txt'
2022-11-28 12:11:34 +00:00
if os.path.isfile(moved_accounts_filename):
show_moved_accounts = True
2022-11-28 12:05:13 +00:00
2021-12-26 10:08:06 +00:00
nickname = profile_json['preferredUsername']
2020-11-09 22:44:03 +00:00
if not nickname:
return ""
2021-12-27 15:41:04 +00:00
if is_system_account(nickname):
2024-12-23 18:23:47 +00:00
min_images_for_accounts: list[str] = []
2021-12-29 21:55:09 +00:00
return html_front_screen(signing_priv_key_pem,
rss_icon_at_top,
2022-07-12 19:03:30 +00:00
icons_as_buttons,
2021-12-30 23:46:06 +00:00
default_timeline,
2021-12-29 21:55:09 +00:00
recent_posts_cache, max_recent_posts,
translate, project_version,
base_dir, http_prefix, authorized,
2024-02-26 11:24:00 +00:00
profile_json,
2021-12-29 21:55:09 +00:00
session, cached_webfingers, person_cache,
yt_replace_domain,
twitter_replacement_domain,
show_published_date_only,
2022-05-30 18:33:51 +00:00
newswire, theme, extra_json,
2021-12-30 23:46:06 +00:00
allow_local_network_access, access_keys,
2021-12-29 21:55:09 +00:00
system_language, max_like_count,
2024-02-26 11:24:00 +00:00
shared_items_federated_domains, cw_lists,
lists_enabled, {},
min_images_for_accounts, buy_sites,
auto_cw_cache,
2024-12-17 13:50:48 +00:00
known_epicyon_instances,
mitm_servers)
2021-12-27 19:05:25 +00:00
domain, port = get_domain_from_actor(profile_json['id'])
2020-11-09 22:44:03 +00:00
if not domain:
return ""
2023-02-07 17:35:50 +00:00
display_name = remove_html(profile_json['name'])
2024-02-02 13:13:24 +00:00
display_name = standardize_text(display_name)
2021-12-30 23:46:06 +00:00
display_name = \
2021-12-29 21:55:09 +00:00
add_emoji_to_display_name(session, base_dir, http_prefix,
nickname, domain,
2023-02-08 11:42:42 +00:00
display_name, False, translate)
2021-12-26 12:45:03 +00:00
domain_full = get_full_domain(domain, port)
2023-07-28 13:33:42 +00:00
if not dangerous_markup(profile_json['summary'], False, []):
profile_description = profile_json['summary']
else:
profile_description = remove_html(profile_json['summary'])
profile_description = \
remove_link_trackers_from_content(profile_description)
2021-12-30 23:46:06 +00:00
profile_description = \
2021-12-29 21:55:09 +00:00
add_emoji_to_display_name(session, base_dir, http_prefix,
nickname, domain,
2023-02-07 17:35:50 +00:00
profile_description, False, translate)
if profile_description:
profile_description = standardize_text(profile_description)
2023-05-03 11:07:34 +00:00
featured_hashtags = \
get_featured_hashtags_as_html(profile_json, profile_description)
2021-12-30 23:46:06 +00:00
posts_button = 'button'
following_button = 'button'
2022-11-27 22:59:24 +00:00
moved_button = 'button'
moved_button = 'button'
2022-12-01 19:44:12 +00:00
inactive_button = 'button'
2021-12-30 23:46:06 +00:00
followers_button = 'button'
roles_button = 'button'
skills_button = 'button'
# shares_button = 'button'
# wanted_button = 'button'
2020-11-09 22:44:03 +00:00
if selected == 'posts':
2021-12-30 23:46:06 +00:00
posts_button = 'buttonselected'
2020-11-09 22:44:03 +00:00
elif selected == 'following':
2021-12-30 23:46:06 +00:00
following_button = 'buttonselected'
2022-11-27 22:59:24 +00:00
elif selected == 'moved':
moved_button = 'buttonselected'
2022-12-01 19:44:12 +00:00
elif selected == 'inactive':
inactive_button = 'buttonselected'
2020-11-09 22:44:03 +00:00
elif selected == 'followers':
2021-12-30 23:46:06 +00:00
followers_button = 'buttonselected'
2020-11-09 22:44:03 +00:00
elif selected == 'roles':
2021-12-30 23:46:06 +00:00
roles_button = 'buttonselected'
2020-11-09 22:44:03 +00:00
elif selected == 'skills':
2021-12-30 23:46:06 +00:00
skills_button = 'buttonselected'
# elif selected == 'shares':
# shares_button = 'buttonselected'
# elif selected == 'wanted':
# wanted_button = 'buttonselected'
login_button = ''
follow_approvals_section = ''
follow_approvals = False
edit_profile_str = ''
logout_str = ''
2021-12-26 10:08:06 +00:00
actor = profile_json['id']
2021-12-30 23:46:06 +00:00
users_path = '/users/' + actor.split('/users/')[1]
donate_section = ''
donate_url = get_donation_url(profile_json)
website_url = get_website(profile_json, translate)
repo_url = get_repo_url(profile_json)
2024-02-28 19:44:47 +00:00
gemini_link = get_gemini_link(profile_json)
2021-12-30 23:46:06 +00:00
blog_address = get_blog_address(profile_json)
enigma_pub_key = get_enigma_pub_key(profile_json)
pgp_pub_key = get_pgp_pub_key(profile_json)
pgp_fingerprint = get_pgp_fingerprint(profile_json)
email_address = get_email_address(profile_json)
pronouns = get_pronouns(profile_json)
pixelfed = get_pixelfed(profile_json)
discord = get_discord(profile_json)
2024-08-15 19:58:26 +00:00
art_site_url = get_art_site_url(profile_json)
music_site_url = get_music_site_url(profile_json)
youtube = get_youtube(profile_json)
peertube = get_peertube(profile_json)
2021-12-30 23:46:06 +00:00
xmpp_address = get_xmpp_address(profile_json)
matrix_address = get_matrix_address(profile_json)
ssb_address = get_ssb_address(profile_json)
tox_address = get_tox_address(profile_json)
briar_address = get_briar_address(profile_json)
2021-12-30 20:48:38 +00:00
cwtch_address = get_cwtch_address(profile_json)
2022-11-09 13:08:32 +00:00
verified_site_checkmark = ''
premium = is_premium_account(base_dir, nickname, domain)
if donate_url or website_url or repo_url or pronouns or discord or \
2024-08-15 19:58:26 +00:00
art_site_url or music_site_url or youtube or peertube or pixelfed or \
xmpp_address or matrix_address or \
ssb_address or tox_address or briar_address or cwtch_address or \
pgp_pub_key or enigma_pub_key or pgp_fingerprint or email_address:
2021-12-30 23:46:06 +00:00
donate_section = '<div class="container">\n'
donate_section += ' <center>\n'
if donate_url and not is_system_account(nickname):
2024-03-20 20:42:12 +00:00
donate_str = translate['Donate']
if premium:
donate_str = translate['Subscribe']
2021-12-30 23:46:06 +00:00
donate_section += \
2023-07-09 09:24:46 +00:00
' <p><a href="' + donate_url + \
'" tabindex="1" rel="donation">' + \
2024-03-20 20:42:12 +00:00
'<button class="donateButton">' + donate_str + \
2020-11-09 22:44:03 +00:00
'</button></a></p>\n'
2021-12-30 23:46:06 +00:00
if website_url:
2022-11-09 12:10:29 +00:00
if site_is_verified(session, base_dir, http_prefix,
nickname, domain,
2024-12-17 13:50:48 +00:00
website_url, False, debug,
mitm_servers):
2022-11-09 12:34:48 +00:00
donate_section += \
2022-11-09 13:11:22 +00:00
'<p><div class="verified_site">' + \
translate['Website'] + ': ' + \
2022-11-09 13:18:56 +00:00
verified_site_checkmark + '<a href="' + \
2022-11-09 13:08:32 +00:00
website_url + '" rel="me" tabindex="1">' + \
website_url + '</a></div></p>\n'
2022-11-09 12:34:48 +00:00
else:
2022-11-09 13:08:32 +00:00
donate_section += \
2022-11-09 13:11:22 +00:00
'<p>' + translate['Website'] + ': ' + \
2022-11-09 13:08:32 +00:00
'<a href="' + \
website_url + '" rel="me" tabindex="1">' + \
website_url + '</a></p>\n'
if repo_url:
donate_section += \
'<p>💻 <a href="' + repo_url + '" tabindex="1">' + \
repo_url + '</a></p>\n'
2022-11-04 21:20:01 +00:00
if gemini_link:
donate_section += \
'<p>' + 'Gemini' + ': <a href="' + \
2022-11-07 16:00:16 +00:00
gemini_link + '" tabindex="1">' + \
gemini_link + '</a></p>\n'
2021-12-30 23:46:06 +00:00
if email_address:
donate_section += \
2020-11-09 22:44:03 +00:00
'<p>' + translate['Email'] + ': <a href="mailto:' + \
2022-05-25 14:09:01 +00:00
email_address + '" tabindex="1">' + \
email_address + '</a></p>\n'
2021-12-30 23:46:06 +00:00
if blog_address:
2022-11-09 12:10:29 +00:00
if site_is_verified(session, base_dir, http_prefix,
nickname, domain,
2024-12-17 13:50:48 +00:00
blog_address, False, debug,
mitm_servers):
2022-11-09 12:34:48 +00:00
donate_section += \
2022-11-09 13:18:56 +00:00
'<p><div class="verified_site">' + \
'Blog: ' + verified_site_checkmark + \
'<a href="' + \
blog_address + '" rel="me" tabindex="1">' + \
blog_address + '</a></div></p>\n'
2022-11-09 12:34:48 +00:00
else:
2022-11-09 13:18:56 +00:00
donate_section += \
'<p>Blog: <a href="' + \
blog_address + '" rel="me" tabindex="1">' + \
blog_address + '</a></p>\n'
if pronouns:
donate_section += \
'<p>' + translate['Pronouns'] + ': ' + pronouns + '</p>\n'
2021-12-30 23:46:06 +00:00
if xmpp_address:
donate_section += \
2020-11-09 22:44:03 +00:00
'<p>' + translate['XMPP'] + ': <a href="xmpp:' + \
2022-05-25 14:09:01 +00:00
xmpp_address + '" tabindex="1">' + xmpp_address + '</a></p>\n'
if pixelfed:
donate_section += \
'<p>Pixelfed: <a href="' + \
pixelfed + '" tabindex="1">' + pixelfed + '</a></p>\n'
if discord:
donate_section += \
'<p>Discord: <a href="' + \
discord + '" tabindex="1">' + discord + '</a></p>\n'
2024-08-15 19:58:26 +00:00
if art_site_url:
donate_section += \
'<p>' + translate['Art'] + ': <a href="' + \
art_site_url + '" tabindex="1">' + \
art_site_url + '</a></p>\n'
if music_site_url:
donate_section += \
'<p>' + translate['Music'] + ': <a href="' + \
music_site_url + '" tabindex="1">' + \
music_site_url + '</a></p>\n'
if youtube:
donate_section += \
'<p>YouTube: <a href="' + \
youtube + '" tabindex="1">' + youtube + '</a></p>\n'
if peertube:
donate_section += \
'<p>PeerTube: <a href="' + \
peertube + '" tabindex="1">' + peertube + '</a></p>\n'
2021-12-30 23:46:06 +00:00
if matrix_address:
donate_section += \
'<p>' + translate['Matrix'] + ': ' + matrix_address + '</p>\n'
if ssb_address:
donate_section += \
2020-11-09 22:44:03 +00:00
'<p>SSB: <label class="ssbaddr">' + \
2021-12-30 23:46:06 +00:00
ssb_address + '</label></p>\n'
if tox_address:
donate_section += \
2020-11-09 22:44:03 +00:00
'<p>Tox: <label class="toxaddr">' + \
2021-12-30 23:46:06 +00:00
tox_address + '</label></p>\n'
if briar_address:
if briar_address.startswith('briar://'):
donate_section += \
2020-12-24 17:11:18 +00:00
'<p><label class="toxaddr">' + \
2021-12-30 23:46:06 +00:00
briar_address + '</label></p>\n'
2020-12-24 17:11:18 +00:00
else:
2021-12-30 23:46:06 +00:00
donate_section += \
2020-12-24 17:11:18 +00:00
'<p>briar://<label class="toxaddr">' + \
2021-12-30 23:46:06 +00:00
briar_address + '</label></p>\n'
2021-12-30 20:48:38 +00:00
if cwtch_address:
2021-12-30 23:46:06 +00:00
donate_section += \
2021-06-27 11:48:03 +00:00
'<p>Cwtch: <label class="toxaddr">' + \
2021-12-30 20:48:38 +00:00
cwtch_address + '</label></p>\n'
2021-12-30 23:46:06 +00:00
if enigma_pub_key:
donate_section += \
2021-12-11 10:53:38 +00:00
'<p>Enigma: <label class="toxaddr">' + \
2021-12-30 23:46:06 +00:00
enigma_pub_key + '</label></p>\n'
if pgp_fingerprint:
donate_section += \
2024-06-10 12:38:09 +00:00
'<p class="pgp">' + translate['PGP Fingerprint'] + ': ' + \
2021-12-30 23:46:06 +00:00
pgp_fingerprint.replace('\n', '<br>') + '</p>\n'
if pgp_pub_key:
donate_section += \
'<details><summary class="cw" tabindex="10">' + \
2024-06-10 12:38:09 +00:00
translate['PGP Public Key'] + \
'</summary><div class="pgp">' + \
pgp_pub_key.replace('\n', '<br>') + \
'</div></details>\n'
2021-12-30 23:46:06 +00:00
donate_section += ' </center>\n'
donate_section += '</div>\n'
2020-11-09 22:44:03 +00:00
if authorized:
2021-12-30 23:46:06 +00:00
edit_profile_str = \
2022-05-25 14:09:01 +00:00
'<a class="imageAnchor" href="' + users_path + \
'/editprofile" tabindex="1">' + \
2022-03-28 08:47:53 +00:00
'<img loading="lazy" decoding="async" src="/icons' + \
2020-11-09 22:44:03 +00:00
'/edit.png" title="' + translate['Edit'] + \
'" alt="| ' + translate['Edit'] + '" class="timelineicon"/></a>\n'
2021-12-30 23:46:06 +00:00
logout_str = \
2022-05-25 14:09:01 +00:00
'<a class="imageAnchor" href="/logout" tabindex="1">' + \
2022-03-28 08:47:53 +00:00
'<img loading="lazy" decoding="async" src="/icons' + \
2020-11-09 22:44:03 +00:00
'/logout.png" title="' + translate['Logout'] + \
'" alt="| ' + translate['Logout'] + \
'" class="timelineicon"/></a>\n'
# are there any follow requests?
2021-12-30 23:46:06 +00:00
follow_requests_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/followrequests.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(follow_requests_filename):
try:
with open(follow_requests_filename, 'r',
2024-07-14 13:01:46 +00:00
encoding='utf-8') as fp_foll:
for line in fp_foll:
if len(line) > 0:
follow_approvals = True
followers_button = 'buttonhighlighted'
if selected == 'followers':
followers_button = 'buttonselectedhighlighted'
break
2024-07-14 13:01:46 +00:00
except OSError as exc:
print('EX: html_profile unable to read ' +
2024-07-14 13:01:46 +00:00
follow_requests_filename + ' ' + str(exc))
2020-11-09 22:44:03 +00:00
if selected == 'followers':
2021-12-30 23:46:06 +00:00
if follow_approvals:
curr_follower_domains = \
2021-12-29 21:55:09 +00:00
get_follower_domains(base_dir, nickname, domain)
2022-06-09 14:46:30 +00:00
with open(follow_requests_filename, 'r',
2024-07-14 13:01:46 +00:00
encoding='utf-8') as fp_req:
for follower_handle in fp_req:
2021-12-30 23:46:06 +00:00
if len(follower_handle) > 0:
2022-06-21 11:58:50 +00:00
follower_handle = \
remove_eol(follower_handle)
2021-12-30 23:46:06 +00:00
if '://' in follower_handle:
follower_actor = follower_handle
2020-11-09 22:44:03 +00:00
else:
2021-12-30 23:46:06 +00:00
nick = follower_handle.split('@')[0]
dom = follower_handle.split('@')[1]
follower_actor = \
2021-12-26 10:19:59 +00:00
local_actor_url(http_prefix, nick, dom)
# is this a new domain?
# if so then append a new instance indicator
2021-12-30 23:46:06 +00:00
follower_domain, _ = \
get_domain_from_actor(follower_actor)
new_follower_domain = ''
if follower_domain not in curr_follower_domains:
new_follower_domain = ''
# Show the handle of the potential follower
# being approved, linking to search on that handle
2021-12-30 23:46:06 +00:00
base_path = '/users/' + nickname
follow_approvals_section += \
'<div class="container">\n' + \
' <form method="POST" action="' + \
base_path + '/searchhandle?page=1">\n' + \
' <input type="hidden" ' + \
2023-01-04 10:48:37 +00:00
'name="actor" value="' + actor + '">\n' + \
2023-01-04 10:21:40 +00:00
' <input type="hidden" ' + \
'name="searchtext" value="' + \
2023-01-04 10:48:37 +00:00
follower_actor + \
2023-01-04 10:21:40 +00:00
'">\n <button type="submit" ' + \
'class="followApproveHandle" ' + \
'name="submitSearch" tabindex="2">' + \
follower_handle + new_follower_domain + \
'</button>\n </form>\n'
# show Approve and Deny buttons
2021-12-30 23:46:06 +00:00
follow_approvals_section += \
'<a href="' + base_path + \
2022-05-25 14:09:01 +00:00
'/followapprove=' + follower_handle + \
'" tabindex="2">'
2021-12-30 23:46:06 +00:00
follow_approvals_section += \
2020-11-09 22:44:03 +00:00
'<button class="followApprove">' + \
translate['Approve'] + '</button></a><br><br>'
2021-12-30 23:46:06 +00:00
follow_approvals_section += \
'<a href="' + base_path + \
2022-05-25 14:09:01 +00:00
'/followdeny=' + follower_handle + \
'" tabindex="3">'
2021-12-30 23:46:06 +00:00
follow_approvals_section += \
2020-11-09 22:44:03 +00:00
'<button class="followDeny">' + \
translate['Deny'] + '</button></a>'
2021-12-30 23:46:06 +00:00
follow_approvals_section += '</div>'
2020-11-09 22:44:03 +00:00
profile_description_short = \
_get_profile_short_description(profile_description)
2020-11-09 22:44:03 +00:00
# remove formatting from profile description used on title
2021-12-30 23:46:06 +00:00
avatar_description = ''
2021-12-26 10:08:06 +00:00
if profile_json.get('summary'):
2021-12-30 23:46:06 +00:00
avatar_description = profile_json['summary'].replace('<br>', '\n')
2024-08-09 12:29:22 +00:00
replacements = {
'<p>': '',
'</p>': ''
}
avatar_description = replace_strings(avatar_description, replacements)
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
moved_to = ''
2021-12-26 10:08:06 +00:00
if profile_json.get('movedTo'):
2021-12-30 23:46:06 +00:00
moved_to = profile_json['movedTo']
if '"' in moved_to:
moved_to = moved_to.split('"')[1]
2021-12-30 23:46:06 +00:00
also_known_as = None
2021-12-26 10:08:06 +00:00
if profile_json.get('alsoKnownAs'):
2021-12-30 23:46:06 +00:00
also_known_as = profile_json['alsoKnownAs']
2021-01-22 20:59:43 +00:00
2021-12-30 23:46:06 +00:00
joined_date = None
2021-12-26 10:08:06 +00:00
if profile_json.get('published'):
if 'T' in profile_json['published']:
2021-12-30 23:46:06 +00:00
joined_date = profile_json['published']
occupation_name = None
2021-12-26 10:08:06 +00:00
if profile_json.get('hasOccupation'):
2021-12-30 23:46:06 +00:00
occupation_name = get_occupation_name(profile_json)
2023-12-09 14:18:24 +00:00
url_str = get_url_from_post(profile_json['icon']['url'])
avatar_url = remove_html(url_str)
2021-06-13 08:48:59 +00:00
# use alternate path for local avatars to avoid any caching issues
2021-12-30 23:46:06 +00:00
if '://' + domain_full + '/system/accounts/avatars/' in avatar_url:
avatar_url = \
avatar_url.replace('://' + domain_full +
'/system/accounts/avatars/',
'://' + domain_full + '/users/')
2021-01-24 18:09:21 +00:00
2021-12-30 23:46:06 +00:00
account_dir = acct_dir(base_dir, nickname, domain)
2023-08-23 19:20:55 +00:00
# get pinned post content
2021-12-30 23:46:06 +00:00
pinned_filename = account_dir + '/pinToProfile.txt'
pinned_content = None
if os.path.isfile(pinned_filename):
try:
2024-07-14 13:01:46 +00:00
with open(pinned_filename, 'r', encoding='utf-8') as fp_pin:
pinned_content = fp_pin.read()
except OSError:
print('EX: html_profile unable to read ' + pinned_filename)
2021-12-30 23:46:06 +00:00
2023-08-23 19:20:55 +00:00
# shared items attached to the actor
# https://codeberg.org/fediverse/fep/src/branch/main/fep/0837/fep-0837.md
attached_shared_items = \
actor_attached_shares_as_html(profile_json, max_shares_on_profile)
2023-08-23 19:20:55 +00:00
2024-01-07 22:19:29 +00:00
birth_date = ''
if profile_json.get('vcard:bday'):
birth_date = profile_json['vcard:bday']
2021-12-30 23:46:06 +00:00
profile_header_str = \
2022-08-20 13:18:26 +00:00
_get_profile_header(base_dir, http_prefix,
2022-06-01 17:45:59 +00:00
nickname,
2022-08-20 13:18:26 +00:00
domain, domain_full, translate,
2021-12-30 23:46:06 +00:00
default_timeline, display_name,
2024-08-12 10:57:15 +00:00
pronouns, profile_description_short,
2023-05-03 11:00:46 +00:00
featured_hashtags,
2021-12-30 23:46:06 +00:00
login_button, avatar_url, theme,
moved_to, also_known_as,
2023-08-23 19:20:55 +00:00
pinned_content,
attached_shared_items,
access_keys, joined_date,
occupation_name,
actor_proxied, actor,
2024-01-07 22:19:29 +00:00
no_of_books, authorized,
2024-03-20 20:42:12 +00:00
birth_date, premium)
2020-11-09 22:44:03 +00:00
2021-02-05 15:30:51 +00:00
# keyboard navigation
2021-12-30 23:46:06 +00:00
user_path_str = '/users/' + nickname
deft = default_timeline
is_group = False
followers_str = translate['Followers']
if premium:
followers_str = translate['Fans']
2021-12-26 17:33:24 +00:00
if is_group_account(base_dir, nickname, domain):
2021-12-30 23:46:06 +00:00
is_group = True
followers_str = translate['Members']
menu_timeline = \
2021-12-29 21:55:09 +00:00
html_hide_from_screen_reader('🏠') + ' ' + \
2021-02-06 10:46:03 +00:00
translate['Switch to timeline view']
2021-12-30 23:46:06 +00:00
menu_edit = \
2021-12-29 21:55:09 +00:00
html_hide_from_screen_reader('') + ' ' + translate['Edit']
2021-12-30 23:46:06 +00:00
menu_followers = \
html_hide_from_screen_reader('👪') + ' ' + followers_str
2022-11-28 12:05:13 +00:00
if show_moved_accounts:
menu_moved = \
html_hide_from_screen_reader('') + ' ' + translate['Moved']
2022-12-01 19:44:12 +00:00
menu_inactive = \
html_hide_from_screen_reader('💤') + ' ' + translate['Inactive']
2021-12-30 23:46:06 +00:00
menu_logout = \
2021-12-29 21:55:09 +00:00
html_hide_from_screen_reader('') + ' ' + translate['Logout']
2022-11-28 12:05:13 +00:00
if not show_moved_accounts:
nav_links = {
menu_timeline: user_path_str + '/' + deft,
menu_edit: user_path_str + '/editprofile',
menu_followers: user_path_str + '/followers#timeline',
menu_logout: '/logout'
}
else:
nav_links = {
menu_timeline: user_path_str + '/' + deft,
menu_edit: user_path_str + '/editprofile',
menu_followers: user_path_str + '/followers#timeline',
menu_moved: user_path_str + '/moved#timeline',
2022-12-01 19:44:12 +00:00
menu_inactive: user_path_str + '/inactive#timeline',
2022-11-28 12:05:13 +00:00
menu_logout: '/logout'
}
2022-05-16 00:47:20 +00:00
if not is_group:
menu_following = \
html_hide_from_screen_reader('👥') + ' ' + translate['Following']
nav_links[menu_following] = user_path_str + '/following#timeline'
menu_roles = \
html_hide_from_screen_reader('🤚') + ' ' + translate['Roles']
nav_links[menu_roles] = user_path_str + '/roles#timeline'
menu_skills = \
html_hide_from_screen_reader('🛠') + ' ' + translate['Skills']
nav_links[menu_skills] = user_path_str + '/skills#timeline'
2021-12-26 14:17:13 +00:00
if is_artist(base_dir, nickname):
2021-12-30 23:46:06 +00:00
menu_theme_designer = \
2021-12-29 21:55:09 +00:00
html_hide_from_screen_reader('🎨') + ' ' + \
translate['Theme Designer']
2021-12-30 23:46:06 +00:00
nav_links[menu_theme_designer] = user_path_str + '/themedesigner'
nav_access_keys = {}
for variable_name, key in access_keys.items():
if not locals().get(variable_name):
2021-04-22 14:12:59 +00:00
continue
2021-12-30 23:46:06 +00:00
nav_access_keys[locals()[variable_name]] = key
2021-04-22 14:12:59 +00:00
2021-12-30 23:46:06 +00:00
profile_str = html_keyboard_navigation(text_mode_banner,
2024-02-19 18:31:04 +00:00
nav_links, nav_access_keys,
None, None, None, False)
2021-12-30 23:46:06 +00:00
profile_str += profile_header_str + donate_section
profile_str += '<div class="container" id="buttonheader">\n'
profile_str += ' <center>'
profile_str += \
2022-05-25 14:09:01 +00:00
' <a href="' + users_path + '#buttonheader" tabindex="2">' + \
'<button class="' + \
2021-12-30 23:46:06 +00:00
posts_button + '"><span>' + translate['Posts'] + \
' </span></button></a>'
2021-12-30 23:46:06 +00:00
if not is_group:
profile_str += \
2022-05-25 14:09:01 +00:00
' <a href="' + users_path + \
'/following#buttonheader" tabindex="2">' + \
2021-12-30 23:46:06 +00:00
'<button class="' + following_button + '"><span>' + \
translate['Following'] + ' </span></button></a>'
2021-12-30 23:46:06 +00:00
profile_str += \
2022-05-25 14:09:01 +00:00
' <a href="' + users_path + \
'/followers#buttonheader" tabindex="2">' + \
2021-12-30 23:46:06 +00:00
'<button class="' + followers_button + \
'"><span>' + followers_str + ' </span></button></a>'
2022-12-01 19:44:12 +00:00
profile_str += \
' <a href="' + users_path + \
'/inactive#buttonheader" tabindex="2">' + \
'<button class="' + inactive_button + \
'"><span>' + translate['Inactive'] + ' </span></button></a>'
2021-12-30 23:46:06 +00:00
if not is_group:
2022-11-28 12:05:13 +00:00
if show_moved_accounts:
2022-11-27 22:59:24 +00:00
profile_str += \
' <a href="' + users_path + \
'/moved#buttonheader" tabindex="2">' + \
'<button class="' + moved_button + '"><span>' + \
translate['Moved'] + ' </span></button></a>'
2021-12-30 23:46:06 +00:00
profile_str += \
2022-05-25 14:09:01 +00:00
' <a href="' + users_path + \
'/roles#buttonheader" tabindex="2">' + \
2021-12-30 23:46:06 +00:00
'<button class="' + roles_button + '"><span>' + \
translate['Roles'] + \
' </span></button></a>'
2021-12-30 23:46:06 +00:00
profile_str += \
2022-05-25 14:09:01 +00:00
' <a href="' + users_path + \
'/skills#buttonheader" tabindex="2">' + \
2021-12-30 23:46:06 +00:00
'<button class="' + skills_button + '"><span>' + \
translate['Skills'] + ' </span></button></a>'
2021-12-30 23:46:06 +00:00
# profile_str += \
2022-05-25 14:09:01 +00:00
# ' <a href="' + users_path + \
# '/shares#buttonheader" tabindex="2">' + \
2021-12-30 23:46:06 +00:00
# '<button class="' + shares_button + '"><span>' + \
# translate['Shares'] + ' </span></button></a>'
2021-12-30 23:46:06 +00:00
# profile_str += \
2022-05-25 14:09:01 +00:00
# ' <a href="' + users_path + \
# '/wanted#buttonheader" tabindex="2">' + \
2021-12-30 23:46:06 +00:00
# '<button class="' + wanted_button + '"><span>' + \
# translate['Wanted'] + ' </span></button></a>'
2021-12-30 23:46:06 +00:00
profile_str += logout_str + edit_profile_str
profile_str += ' </center>'
profile_str += '</div>'
2020-11-09 22:44:03 +00:00
# search for following or followers
if authorized:
if selected in ('following', 'followers'):
follow_search_str = '<div class="container">\n'
follow_search_str += \
'<form method="POST" action="' + users_path + \
'/searchhandle?page=1">\n'
follow_search_str += \
' <input type="hidden" ' + \
'name="actor" value="' + actor + '">\n'
follow_search_str += \
html_following_dropdown(base_dir, nickname, domain,
domain_full, selected, False)
follow_search_str += \
' <button type="submit" class="button" ' + \
2023-01-01 10:32:08 +00:00
'name="submitSearch">' + translate['View'] + '</button>\n'
follow_search_str += '</form>\n</div>\n'
profile_str += follow_search_str
2021-02-05 17:55:49 +00:00
# start of #timeline
2021-12-30 23:46:06 +00:00
profile_str += '<div id="timeline">\n'
2021-02-05 17:55:49 +00:00
2021-12-30 23:46:06 +00:00
profile_str += follow_approvals_section
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
css_filename = base_dir + '/epicyon-profile.css'
2021-12-25 16:17:53 +00:00
if os.path.isfile(base_dir + '/epicyon.css'):
2021-12-30 23:46:06 +00:00
css_filename = base_dir + '/epicyon.css'
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
license_str = \
2022-05-25 14:09:01 +00:00
'<a href="https://gitlab.com/bashrc2/epicyon" tabindex="2">' + \
2022-03-28 08:47:53 +00:00
'<img loading="lazy" decoding="async" class="license" alt="' + \
translate['Get the source code'] + '" title="' + \
translate['Get the source code'] + '" src="/icons/agpl.png" /></a>'
2024-03-20 20:42:12 +00:00
if selected == 'posts' and not premium:
max_profile_posts = \
get_max_profile_posts(base_dir, nickname, domain, 20)
2024-12-23 18:23:47 +00:00
min_images_for_accounts: list[str] = []
2021-12-30 23:46:06 +00:00
profile_str += \
_html_profile_posts(recent_posts_cache, max_profile_posts,
2021-12-29 21:55:09 +00:00
translate,
base_dir, http_prefix, authorized,
nickname, domain, port,
session, cached_webfingers, person_cache,
project_version,
yt_replace_domain,
twitter_replacement_domain,
show_published_date_only,
peertube_instances,
allow_local_network_access,
theme, system_language,
max_like_count,
signing_priv_key_pem,
2022-02-25 19:12:40 +00:00
cw_lists, lists_enabled,
timezone, bold_reading, {},
min_images_for_accounts,
2023-01-13 15:04:48 +00:00
max_profile_posts,
buy_sites,
2024-12-17 13:50:48 +00:00
auto_cw_cache,
mitm_servers) + license_str
2021-12-30 23:46:06 +00:00
if not is_group:
if selected == 'following':
2021-12-30 23:46:06 +00:00
profile_str += \
2021-12-29 21:55:09 +00:00
_html_profile_following(translate, base_dir, http_prefix,
authorized, nickname,
2022-06-01 17:45:59 +00:00
domain, session,
2021-12-29 21:55:09 +00:00
cached_webfingers,
2022-05-30 18:33:51 +00:00
person_cache, extra_json,
2021-12-29 21:55:09 +00:00
project_version, ["unfollow"],
selected,
2021-12-30 23:46:06 +00:00
users_path, page_number,
max_items_per_page,
2021-12-29 21:55:09 +00:00
dormant_months, debug,
2023-09-16 13:55:02 +00:00
signing_priv_key_pem,
2023-10-25 19:55:40 +00:00
sites_unavailable,
2024-12-17 13:50:48 +00:00
system_language,
mitm_servers)
2022-11-28 12:05:13 +00:00
if show_moved_accounts and selected == 'moved':
2022-11-27 22:59:24 +00:00
profile_str += \
_html_profile_following(translate, base_dir, http_prefix,
authorized, nickname,
domain, session,
cached_webfingers,
person_cache, extra_json,
2022-11-29 22:46:40 +00:00
project_version, ["moveAccount"],
2022-11-27 22:59:24 +00:00
selected,
users_path, page_number,
max_items_per_page,
dormant_months, debug,
2023-09-16 13:55:02 +00:00
signing_priv_key_pem,
2023-10-25 19:55:40 +00:00
sites_unavailable,
2024-12-17 13:50:48 +00:00
system_language,
mitm_servers)
if selected == 'followers':
2022-12-01 19:44:12 +00:00
profile_str += \
_html_profile_following(translate, base_dir, http_prefix,
authorized, nickname,
domain, session,
cached_webfingers,
person_cache, extra_json,
project_version, ["block"],
selected, users_path, page_number,
max_items_per_page, dormant_months, debug,
2023-10-25 19:55:40 +00:00
signing_priv_key_pem, sites_unavailable,
2024-12-17 13:50:48 +00:00
system_language, mitm_servers)
2022-12-01 19:44:12 +00:00
if authorized and selected == 'inactive':
2021-12-30 23:46:06 +00:00
profile_str += \
2021-12-29 21:55:09 +00:00
_html_profile_following(translate, base_dir, http_prefix,
authorized, nickname,
2022-06-01 17:45:59 +00:00
domain, session,
2021-12-29 21:55:09 +00:00
cached_webfingers,
2022-05-30 18:33:51 +00:00
person_cache, extra_json,
2021-12-29 21:55:09 +00:00
project_version, ["block"],
2021-12-30 23:46:06 +00:00
selected, users_path, page_number,
max_items_per_page, dormant_months, debug,
2023-10-25 19:55:40 +00:00
signing_priv_key_pem, sites_unavailable,
2024-12-17 13:50:48 +00:00
system_language, mitm_servers)
2021-12-30 23:46:06 +00:00
if not is_group:
if selected == 'roles':
2021-12-30 23:46:06 +00:00
profile_str += \
2021-12-29 21:55:09 +00:00
_html_profile_roles(translate, nickname, domain_full,
2022-05-30 18:33:51 +00:00
extra_json)
2021-09-21 17:47:25 +00:00
elif selected == 'skills':
2021-12-30 23:46:06 +00:00
profile_str += \
2022-06-01 17:45:59 +00:00
_html_profile_skills(extra_json)
# elif selected == 'shares':
2021-12-30 23:46:06 +00:00
# profile_str += \
2021-12-29 21:55:09 +00:00
# _html_profile_shares(actor, translate,
2022-06-01 17:45:59 +00:00
# domain_full,
# extra_json, 'shares') + license_str
# elif selected == 'wanted':
2021-12-30 23:46:06 +00:00
# profile_str += \
2021-12-29 21:55:09 +00:00
# _html_profile_shares(actor, translate,
2022-06-01 17:45:59 +00:00
# domain_full,
# extra_json, 'wanted') + license_str
2021-02-05 17:55:49 +00:00
# end of #timeline
2022-12-23 13:29:08 +00:00
profile_str += '</div>\n<br>\n'
2020-11-12 17:05:38 +00:00
2021-12-30 23:46:06 +00:00
instance_title = \
2021-12-26 14:08:58 +00:00
get_config_param(base_dir, 'instanceTitle')
2021-12-30 23:46:06 +00:00
profile_str = \
html_header_with_person_markup(css_filename, instance_title,
2021-12-29 21:55:09 +00:00
profile_json, city,
content_license_url) + \
2021-12-30 23:46:06 +00:00
profile_str + html_footer()
return profile_str
2020-11-09 22:44:03 +00:00
2021-12-29 21:55:09 +00:00
def _html_profile_posts(recent_posts_cache: {}, max_recent_posts: int,
translate: {},
base_dir: str, http_prefix: str,
authorized: bool,
nickname: str, domain: str, port: int,
session, cached_webfingers: {}, person_cache: {},
project_version: str,
yt_replace_domain: str,
twitter_replacement_domain: str,
show_published_date_only: bool,
peertube_instances: [],
allow_local_network_access: bool,
theme_name: str, system_language: str,
max_like_count: int,
signing_priv_key_pem: str,
2022-02-25 19:12:40 +00:00
cw_lists: {}, lists_enabled: str,
2022-07-05 14:40:26 +00:00
timezone: str, bold_reading: bool,
dogwhistles: {},
min_images_for_accounts: [],
2023-01-13 15:04:48 +00:00
max_profile_posts: int,
buy_sites: {},
2024-12-17 13:50:48 +00:00
auto_cw_cache: {},
mitm_servers: []) -> str:
2020-11-09 22:44:03 +00:00
"""Shows posts on the profile screen
These should only be public posts
"""
2021-12-30 23:46:06 +00:00
separator_str = html_post_separator(base_dir, None)
profile_str = ''
max_items = max_profile_posts
2020-11-09 22:44:03 +00:00
ctr = 0
2021-12-30 23:46:06 +00:00
curr_page = 1
box_name = 'outbox'
minimize_all_images = False
if nickname in min_images_for_accounts:
minimize_all_images = True
2021-12-30 23:46:06 +00:00
while ctr < max_items and curr_page < 4:
outbox_feed_path_str = \
'/users/' + nickname + '/' + box_name + '?page=' + \
str(curr_page)
outbox_feed = \
2022-09-08 17:59:19 +00:00
person_box_json({}, base_dir, domain,
2021-12-28 16:50:20 +00:00
port,
2021-12-30 23:46:06 +00:00
outbox_feed_path_str,
2021-12-28 16:50:20 +00:00
http_prefix,
2021-12-30 23:46:06 +00:00
10, box_name,
2021-12-28 16:50:20 +00:00
authorized, 0, False, 0)
2021-12-30 23:46:06 +00:00
if not outbox_feed:
2020-11-27 16:14:54 +00:00
break
2021-12-30 23:46:06 +00:00
if len(outbox_feed['orderedItems']) == 0:
2020-11-09 22:44:03 +00:00
break
2024-12-23 18:23:47 +00:00
shown_items: list[str] = []
2021-12-30 23:46:06 +00:00
for item in outbox_feed['orderedItems']:
2020-11-09 22:44:03 +00:00
if item['type'] == 'Create':
2022-11-27 21:11:16 +00:00
if not item['object'].get('id'):
continue
item_id = remove_id_ending(item['object']['id'])
2021-12-30 23:46:06 +00:00
post_str = \
2021-12-29 21:55:09 +00:00
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, item,
None, True, False,
http_prefix, project_version,
'inbox',
yt_replace_domain,
twitter_replacement_domain,
show_published_date_only,
peertube_instances,
allow_local_network_access,
theme_name, system_language,
max_like_count,
False, False, False,
True, False, False,
2022-02-25 19:12:40 +00:00
cw_lists, lists_enabled,
2022-03-24 13:14:41 +00:00
timezone, False,
bold_reading, dogwhistles,
2023-01-13 15:04:48 +00:00
minimize_all_images, None,
2024-12-17 13:50:48 +00:00
buy_sites, auto_cw_cache,
mitm_servers)
2022-11-27 21:11:16 +00:00
if post_str and item_id not in shown_items:
2021-12-30 23:46:06 +00:00
profile_str += post_str + separator_str
2022-11-27 21:11:16 +00:00
shown_items.append(item_id)
2020-11-09 22:44:03 +00:00
ctr += 1
2021-12-30 23:46:06 +00:00
if ctr >= max_items:
2020-11-09 22:44:03 +00:00
break
2021-12-30 23:46:06 +00:00
curr_page += 1
return profile_str
2020-11-09 22:44:03 +00:00
2021-12-29 21:55:09 +00:00
def _html_profile_following(translate: {}, base_dir: str, http_prefix: str,
2022-06-01 17:45:59 +00:00
authorized: bool, nickname: str, domain: str,
2021-12-29 21:55:09 +00:00
session, cached_webfingers: {}, person_cache: {},
2021-12-30 23:46:06 +00:00
following_json: {}, project_version: str,
2021-12-29 21:55:09 +00:00
buttons: [],
2022-05-30 20:47:23 +00:00
feed_name: str, actor: str,
2021-12-30 23:46:06 +00:00
page_number: int,
max_items_per_page: int,
2021-12-29 21:55:09 +00:00
dormant_months: int, debug: bool,
2023-09-16 13:55:02 +00:00
signing_priv_key_pem: str,
2023-10-25 19:55:40 +00:00
sites_unavailable: [],
2024-12-17 13:50:48 +00:00
system_language: str,
mitm_servers: []) -> str:
2020-11-09 22:44:03 +00:00
"""Shows following on the profile screen
"""
2021-12-30 23:46:06 +00:00
profile_str = ''
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
if authorized and page_number:
if authorized and page_number > 1:
2020-11-09 22:44:03 +00:00
# page up arrow
2021-12-30 23:46:06 +00:00
profile_str += \
2020-11-09 22:44:03 +00:00
' <center>\n' + \
2022-05-30 20:47:23 +00:00
' <a href="' + actor + '/' + feed_name + \
2021-12-30 23:46:06 +00:00
'?page=' + str(page_number - 1) + '#buttonheader' + \
2022-03-28 08:47:53 +00:00
'"><img loading="lazy" decoding="async" ' + \
'class="pageicon" src="/' + \
2020-12-09 13:08:26 +00:00
'icons/pageup.png" title="' + \
2020-11-09 22:44:03 +00:00
translate['Page up'] + '" alt="' + \
translate['Page up'] + '"></a>\n' + \
' </center>\n'
if not following_json:
following_json = {
'orderedItems': []
}
2021-12-30 23:46:06 +00:00
for following_actor in following_json['orderedItems']:
2020-12-13 12:48:04 +00:00
# is this a dormant followed account?
dormant = False
2023-09-16 13:55:02 +00:00
offline = False
following_domain, _ = get_domain_from_actor(following_actor)
if authorized:
if following_domain in sites_unavailable:
dormant = True
offline = True
else:
if feed_name == 'following':
dormant = \
is_dormant(base_dir, nickname, domain,
following_actor, dormant_months)
2020-12-13 12:48:04 +00:00
2021-12-30 23:46:06 +00:00
profile_str += \
2021-12-29 21:55:09 +00:00
_individual_follow_as_html(signing_priv_key_pem,
translate, base_dir, session,
cached_webfingers, person_cache,
2021-12-30 23:46:06 +00:00
domain, following_actor,
2021-12-29 21:55:09 +00:00
authorized, nickname,
2023-09-16 13:55:02 +00:00
http_prefix, project_version,
dormant, offline,
2024-12-17 13:50:48 +00:00
debug, system_language,
mitm_servers,
buttons)
2020-12-13 12:48:04 +00:00
2021-12-30 23:46:06 +00:00
if authorized and max_items_per_page and page_number:
if len(following_json['orderedItems']) >= max_items_per_page:
2020-11-09 22:44:03 +00:00
# page down arrow
2021-12-30 23:46:06 +00:00
profile_str += \
2020-11-09 22:44:03 +00:00
' <center>\n' + \
2022-05-30 20:47:23 +00:00
' <a href="' + actor + '/' + feed_name + \
2021-12-30 23:46:06 +00:00
'?page=' + str(page_number + 1) + '#buttonheader' + \
2022-03-28 08:47:53 +00:00
'"><img loading="lazy" decoding="async" ' + \
'class="pageicon" src="/' + \
2020-12-09 13:08:26 +00:00
'icons/pagedown.png" title="' + \
2020-11-09 22:44:03 +00:00
translate['Page down'] + '" alt="' + \
translate['Page down'] + '"></a>\n' + \
' </center>\n'
2022-07-08 16:33:38 +00:00
# list of page numbers
profile_str += \
page_number_buttons(actor, feed_name, page_number,
'buttonheader')
2022-07-08 16:36:02 +00:00
# some vertical padding to allow "finger space" on mobile
2022-07-08 16:38:13 +00:00
profile_str += '<br>'
2021-12-30 23:46:06 +00:00
return profile_str
2020-11-09 22:44:03 +00:00
2021-12-29 21:55:09 +00:00
def _html_profile_roles(translate: {}, nickname: str, domain: str,
2022-05-30 18:33:51 +00:00
roles_list: []) -> str:
2020-11-09 22:44:03 +00:00
"""Shows roles on the profile screen
"""
2021-12-30 23:46:06 +00:00
profile_str = ''
profile_str += \
'<div class="roles">\n<div class="roles-inner">\n'
2022-05-30 18:33:51 +00:00
for role in roles_list:
if translate.get(role):
2021-12-30 23:46:06 +00:00
profile_str += '<h3>' + translate[role] + '</h3>\n'
else:
2021-12-30 23:46:06 +00:00
profile_str += '<h3>' + role + '</h3>\n'
profile_str += '</div></div>\n'
if len(profile_str) == 0:
profile_str += \
2020-11-09 22:44:03 +00:00
'<p>@' + nickname + '@' + domain + ' has no roles assigned</p>\n'
else:
2021-12-30 23:46:06 +00:00
profile_str = '<div>' + profile_str + '</div>\n'
return profile_str
2020-11-09 22:44:03 +00:00
2022-06-01 17:45:59 +00:00
def _html_profile_skills(skills_json: {}) -> str:
2020-11-09 22:44:03 +00:00
"""Shows skills on the profile screen
"""
2021-12-30 23:46:06 +00:00
profile_str = ''
2022-06-01 17:45:59 +00:00
for skill, level in skills_json.items():
2021-12-30 23:46:06 +00:00
profile_str += \
2020-11-09 22:44:03 +00:00
'<div>' + skill + \
'<br><div id="myProgress"><div id="myBar" style="width:' + \
str(level) + '%"></div></div></div>\n<br>\n'
2021-12-30 23:46:06 +00:00
if len(profile_str) > 0:
profile_str = '<center><div class="skill-title">' + \
profile_str + '</div></center>\n'
return profile_str
2020-11-09 22:44:03 +00:00
2021-12-29 21:55:09 +00:00
def _html_profile_shares(actor: str, translate: {},
2022-06-01 17:45:59 +00:00
domain: str, shares_json: {},
2021-12-30 23:46:06 +00:00
shares_file_type: str) -> str:
2020-11-09 22:44:03 +00:00
"""Shows shares on the profile screen
"""
2021-12-30 23:46:06 +00:00
profile_str = ''
for item in shares_json['orderedItems']:
profile_str += html_individual_share(domain, item['shareId'],
actor, item, translate,
False, False,
shares_file_type)
if len(profile_str) > 0:
profile_str = '<div class="share-title">' + profile_str + '</div>\n'
return profile_str
2020-11-09 22:44:03 +00:00
2021-12-29 21:55:09 +00:00
def _grayscale_enabled(base_dir: str) -> bool:
2021-06-27 17:59:46 +00:00
"""Is grayscale UI enabled?
"""
2024-05-12 12:35:26 +00:00
dir_str = data_dir(base_dir)
return os.path.isfile(dir_str + '/.grayscale')
2021-06-27 17:59:46 +00:00
2021-12-29 21:55:09 +00:00
def _html_themes_dropdown(base_dir: str, translate: {}) -> str:
2021-06-27 17:59:46 +00:00
"""Returns the html for theme selection dropdown
"""
# Themes section
2021-12-29 21:55:09 +00:00
themes = get_themes_list(base_dir)
2021-12-30 23:46:06 +00:00
themes_dropdown = ' <label class="labels">' + \
2021-06-27 17:59:46 +00:00
translate['Theme'] + '</label><br>\n'
2021-12-29 21:55:09 +00:00
grayscale = _grayscale_enabled(base_dir)
2021-12-30 23:46:06 +00:00
themes_dropdown += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Grayscale'], 'grayscale', grayscale)
dyslexic_font = get_config_param(base_dir, 'dyslexicFont')
themes_dropdown += \
edit_check_box(translate['Dyslexic font'], 'dyslexicFont',
dyslexic_font)
2021-12-30 23:46:06 +00:00
themes_dropdown += ' <select id="themeDropdown" ' + \
2021-06-27 17:59:46 +00:00
'name="themeDropdown" class="theme">'
2021-12-25 23:35:50 +00:00
for theme_name in themes:
2021-12-30 23:46:06 +00:00
translated_theme_name = theme_name
2021-12-25 23:35:50 +00:00
if translate.get(theme_name):
2021-12-30 23:46:06 +00:00
translated_theme_name = translate[theme_name]
themes_dropdown += ' <option value="' + \
2021-12-25 23:35:50 +00:00
theme_name.lower() + '">' + \
2021-12-30 23:46:06 +00:00
translated_theme_name + '</option>'
themes_dropdown += ' </select><br>'
2021-12-25 16:17:53 +00:00
if os.path.isfile(base_dir + '/fonts/custom.woff') or \
os.path.isfile(base_dir + '/fonts/custom.woff2') or \
os.path.isfile(base_dir + '/fonts/custom.otf') or \
os.path.isfile(base_dir + '/fonts/custom.ttf'):
2021-12-30 23:46:06 +00:00
themes_dropdown += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Remove the custom font'],
'removeCustomFont', False)
2021-12-26 14:08:58 +00:00
theme_name = get_config_param(base_dir, 'theme')
2021-12-30 23:46:06 +00:00
themes_dropdown = \
themes_dropdown.replace('<option value="' + theme_name + '">',
'<option value="' + theme_name +
'" selected>')
return themes_dropdown
2021-06-27 17:59:46 +00:00
2021-12-29 21:55:09 +00:00
def _html_edit_profile_graphic_design(base_dir: str, translate: {}) -> str:
"""Graphic design section on Edit Profile screen
"""
2021-12-30 23:46:06 +00:00
graphics_str = begin_edit_section(translate['Graphic Design'])
2021-12-30 13:56:38 +00:00
low_bandwidth = get_config_param(base_dir, 'lowBandwidth')
2021-12-25 18:20:56 +00:00
if not low_bandwidth:
low_bandwidth = False
2021-12-30 23:46:06 +00:00
graphics_str += _html_themes_dropdown(base_dir, translate)
graphics_str += \
' <label class="labels">' + \
translate['Import Theme'] + '</label>\n'
2023-02-10 19:41:31 +00:00
graphics_str += ' <input type="file" id="importTheme" '
graphics_str += 'name="importTheme" '
2021-12-30 23:46:06 +00:00
graphics_str += 'accept="' + THEME_FORMATS + '">\n'
graphics_str += \
' <label class="labels">' + \
translate['Export Theme'] + '</label><br>\n'
2021-12-30 23:46:06 +00:00
graphics_str += \
' <button type="submit" class="button" ' + \
2021-08-13 17:35:31 +00:00
'name="submitExportTheme">➤</button><br>\n'
2021-12-30 23:46:06 +00:00
graphics_str += \
2021-12-30 13:56:38 +00:00
edit_check_box(translate['Low Bandwidth'], 'lowBandwidth',
2021-12-29 21:55:09 +00:00
bool(low_bandwidth))
2021-12-30 23:46:06 +00:00
graphics_str += end_edit_section()
return graphics_str
2021-12-29 21:55:09 +00:00
def _html_edit_profile_twitter(base_dir: str, translate: {},
2021-12-30 23:46:06 +00:00
remove_twitter: str) -> str:
2021-09-18 17:08:14 +00:00
"""Edit twitter settings within profile
"""
# Twitter section
2021-12-30 23:46:06 +00:00
twitter_str = begin_edit_section(translate['Twitter'])
twitter_str += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Remove Twitter posts'],
2021-12-30 23:46:06 +00:00
'removeTwitter', remove_twitter)
2021-12-26 14:08:58 +00:00
twitter_replacement_domain = get_config_param(base_dir, "twitterdomain")
2021-12-25 20:55:47 +00:00
if not twitter_replacement_domain:
twitter_replacement_domain = ''
2021-12-30 23:46:06 +00:00
twitter_str += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Twitter Replacement Domain'],
'twitterdomain', twitter_replacement_domain)
2021-12-30 23:46:06 +00:00
twitter_str += end_edit_section()
return twitter_str
2021-09-18 17:08:14 +00:00
2021-12-29 21:55:09 +00:00
def _html_edit_profile_instance(base_dir: str, translate: {},
peertube_instances: [],
2021-12-30 23:46:06 +00:00
media_instance_str: str,
blogs_instance_str: str,
news_instance_str: str) -> (str, str,
2023-08-31 22:29:10 +00:00
str, str, str):
"""Edit profile instance settings
"""
2021-12-30 23:46:06 +00:00
image_formats = get_image_formats()
# Instance details section
2021-12-30 23:46:06 +00:00
instance_description = \
2021-12-26 14:08:58 +00:00
get_config_param(base_dir, 'instanceDescription')
2021-12-30 23:46:06 +00:00
custom_submit_text = \
2021-12-26 14:08:58 +00:00
get_config_param(base_dir, 'customSubmitText')
2021-12-30 23:46:06 +00:00
instance_description_short = \
2021-12-26 14:08:58 +00:00
get_config_param(base_dir, 'instanceDescriptionShort')
2021-12-30 23:46:06 +00:00
instance_title = \
2021-12-26 14:08:58 +00:00
get_config_param(base_dir, 'instanceTitle')
2021-12-25 17:13:38 +00:00
content_license_url = \
2021-12-30 13:56:38 +00:00
get_config_param(base_dir, 'contentLicenseUrl')
2021-12-25 17:13:38 +00:00
if not content_license_url:
content_license_url = 'https://creativecommons.org/licenses/by-nc/4.0'
2021-12-30 23:46:06 +00:00
instance_str = begin_edit_section(translate['Instance Settings'])
2021-07-22 12:36:31 +00:00
2021-12-30 23:46:06 +00:00
instance_str += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Instance Title'],
2021-12-30 23:46:06 +00:00
'instanceTitle', instance_title)
instance_str += '<br>\n'
instance_str += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Instance Short Description'],
2021-12-30 23:46:06 +00:00
'instanceDescriptionShort', instance_description_short)
instance_str += '<br>\n'
instance_str += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['Instance Description'], None,
2021-12-30 23:46:06 +00:00
'instanceDescription', instance_description, 200,
2021-12-29 21:55:09 +00:00
'', True)
2021-12-30 23:46:06 +00:00
instance_str += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Content License'],
2021-12-30 13:56:38 +00:00
'contentLicenseUrl', content_license_url)
2021-12-30 23:46:06 +00:00
instance_str += '<br>\n'
instance_str += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Custom post submit button text'],
2021-12-30 23:46:06 +00:00
'customSubmitText', custom_submit_text)
instance_str += '<br>\n'
instance_str += \
' <label class="labels">' + \
2021-07-22 10:22:01 +00:00
translate['Instance Logo'] + '</label>' + \
' <input type="file" id="instanceLogo" name="instanceLogo"' + \
' accept="' + image_formats + '"><br>\n <br>\n'
public_replies_unlisted = False
if get_config_param(base_dir, "publicRepliesUnlisted") is True:
public_replies_unlisted = True
instance_str += \
edit_check_box(translate['Public replies default to unlisted scope'],
'publicRepliesUnlisted', public_replies_unlisted)
registrations_open = False
if get_config_param(base_dir, "registration") == 'open':
registrations_open = True
instance_str += \
2022-12-05 14:41:02 +00:00
edit_check_box(translate['Registrations open'],
'regOpen', registrations_open)
2022-12-05 20:19:19 +00:00
if registrations_open:
remaining_config_exists = \
get_config_param(base_dir, 'registrationsRemaining')
registrations_remaining = 0
if remaining_config_exists is not None:
registrations_remaining = int(remaining_config_exists)
instance_str += \
edit_number_field(translate['Registrations remaining'],
'regRemaining',
registrations_remaining, 0, 10, 10)
2022-12-05 20:24:41 +00:00
instance_str += ' <br>\n'
instance_str += \
' <label class="labels">\n' + \
translate['Security'] + '</label><br>\n'
2021-12-30 23:46:06 +00:00
node_info_str = \
translate['Show numbers of accounts within instance metadata']
2021-12-30 13:56:38 +00:00
if get_config_param(base_dir, "showNodeInfoAccounts"):
2021-12-30 23:46:06 +00:00
instance_str += \
edit_check_box(node_info_str, 'showNodeInfoAccounts', True)
else:
2021-12-30 23:46:06 +00:00
instance_str += \
edit_check_box(node_info_str, 'showNodeInfoAccounts', False)
2021-12-30 23:46:06 +00:00
node_info_str = \
translate['Show version number within instance metadata']
2021-12-30 13:56:38 +00:00
if get_config_param(base_dir, "showNodeInfoVersion"):
2021-12-30 23:46:06 +00:00
instance_str += \
edit_check_box(node_info_str, 'showNodeInfoVersion', True)
else:
2021-12-30 23:46:06 +00:00
instance_str += \
edit_check_box(node_info_str, 'showNodeInfoVersion', False)
2021-12-30 13:56:38 +00:00
if get_config_param(base_dir, "verifyAllSignatures"):
2021-12-30 23:46:06 +00:00
instance_str += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Verify all signatures'],
'verifyallsignatures', True)
else:
2021-12-30 23:46:06 +00:00
instance_str += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Verify all signatures'],
'verifyallsignatures', False)
2021-12-30 23:46:06 +00:00
instance_str += translate['Enabling broch mode'] + '<br>\n'
2021-12-30 13:56:38 +00:00
if get_config_param(base_dir, "brochMode"):
2021-12-30 23:46:06 +00:00
instance_str += \
2021-12-30 13:56:38 +00:00
edit_check_box(translate['Broch mode'], 'brochMode', True)
else:
2021-12-30 23:46:06 +00:00
instance_str += \
2021-12-30 13:56:38 +00:00
edit_check_box(translate['Broch mode'], 'brochMode', False)
# Instance type
2021-12-30 23:46:06 +00:00
instance_str += \
' <br><label class="labels">' + \
2021-07-22 12:36:31 +00:00
translate['Type of instance'] + '</label><br>\n'
2021-12-30 23:46:06 +00:00
instance_str += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['This is a media instance'],
2021-12-30 23:46:06 +00:00
'mediaInstance', media_instance_str)
instance_str += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['This is a blogging instance'],
2021-12-30 23:46:06 +00:00
'blogsInstance', blogs_instance_str)
instance_str += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['This is a news instance'],
2021-12-30 23:46:06 +00:00
'newsInstance', news_instance_str)
2021-07-22 12:36:31 +00:00
2021-12-30 23:46:06 +00:00
instance_str += end_edit_section()
# Role assignments section
2022-09-02 15:57:06 +00:00
role_assign_str = \
begin_edit_section(translate['Role Assignment']) + \
' <b><label class="labels">'
# site moderators
moderators = ''
2024-05-12 12:35:26 +00:00
moderators_file = data_dir(base_dir) + '/moderators.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(moderators_file):
try:
2024-07-14 13:01:46 +00:00
with open(moderators_file, 'r', encoding='utf-8') as fp_mod:
moderators = fp_mod.read()
except OSError:
print('EX: _html_edit_profile_instance unable to read ' +
moderators_file)
2022-09-02 15:57:06 +00:00
subtitle = translate['A list of moderator nicknames. One per line.']
role_assign_str += \
edit_text_area('<b>' + translate['Moderators'] + '</b>', subtitle,
'moderators', moderators, 200, '', False)
# site editors
editors = ''
2024-05-12 12:35:26 +00:00
editors_file = data_dir(base_dir) + '/editors.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(editors_file):
try:
2024-07-14 13:01:46 +00:00
with open(editors_file, 'r', encoding='utf-8') as fp_edit:
editors = fp_edit.read()
except OSError:
print('EX: _html_edit_profile_instance unable to read ' +
editors_file)
2022-09-02 16:16:11 +00:00
subtitle = translate['A list of editor nicknames. One per line.']
2021-12-30 23:46:06 +00:00
role_assign_str += \
2022-09-02 15:57:06 +00:00
edit_text_area('<b>' + translate['Site Editors'] + '</b>',
2022-09-02 16:16:11 +00:00
subtitle, 'editors', editors, 200, '', False)
# counselors
counselors = ''
2024-05-12 12:35:26 +00:00
counselors_file = data_dir(base_dir) + '/counselors.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(counselors_file):
try:
2024-07-14 13:01:46 +00:00
with open(counselors_file, 'r', encoding='utf-8') as fp_co:
counselors = fp_co.read()
except OSError:
print('EX: _html_edit_profile_instance unable to read ' +
counselors_file)
2021-12-30 23:46:06 +00:00
role_assign_str += \
2022-09-02 15:57:06 +00:00
edit_text_area('<b>' + translate['Counselors'] + '</b>', None,
2022-09-02 10:55:28 +00:00
'counselors', counselors, 200, '', False)
# artists
artists = ''
2024-05-12 12:35:26 +00:00
artists_file = data_dir(base_dir) + '/artists.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(artists_file):
try:
2024-07-14 13:01:46 +00:00
with open(artists_file, 'r', encoding='utf-8') as fp_art:
artists = fp_art.read()
except OSError:
print('EX: _html_edit_profile_instance unable to read ' +
artists_file)
2021-12-30 23:46:06 +00:00
role_assign_str += \
2022-09-02 15:57:06 +00:00
edit_text_area('<b>' + translate['Artists'] + '</b>', None,
2022-09-02 10:55:28 +00:00
'artists', artists, 200, '', False)
2022-09-02 10:29:42 +00:00
# site devops
devops = ''
2024-05-12 12:35:26 +00:00
devops_file = data_dir(base_dir) + '/devops.txt'
2022-09-02 10:29:42 +00:00
if os.path.isfile(devops_file):
try:
2024-07-14 13:01:46 +00:00
with open(devops_file, 'r', encoding='utf-8') as fp_edit:
devops = fp_edit.read()
except OSError:
print('EX: _html_edit_profile_instance unable to read ' +
devops_file)
2022-09-02 16:16:11 +00:00
subtitle = translate['A list of devops nicknames. One per line.']
2022-09-02 10:29:42 +00:00
role_assign_str += \
2022-09-02 15:57:06 +00:00
edit_text_area('<b>' + translate['Site DevOps'] + '</b>',
2022-09-02 16:16:11 +00:00
subtitle, 'devopslist', devops, 200, '', False)
2022-09-02 10:29:42 +00:00
2021-12-30 23:46:06 +00:00
role_assign_str += end_edit_section()
# Video section
2021-12-30 23:46:06 +00:00
peertube_str = begin_edit_section(translate['Video Settings'])
peertube_instances_str = ''
2021-12-25 23:38:53 +00:00
for url in peertube_instances:
2021-12-30 23:46:06 +00:00
peertube_instances_str += url + '\n'
peertube_str += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['Peertube Instances'], None,
'ptInstances', peertube_instances_str, 200, '', False)
2021-12-30 23:46:06 +00:00
peertube_str += \
2021-07-22 12:36:31 +00:00
' <br>\n'
2021-12-26 14:08:58 +00:00
yt_replace_domain = get_config_param(base_dir, "youtubedomain")
2021-12-25 17:15:52 +00:00
if not yt_replace_domain:
yt_replace_domain = ''
2021-12-30 23:46:06 +00:00
peertube_str += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['YouTube Replacement Domain'],
'ytdomain', yt_replace_domain)
2021-12-30 23:46:06 +00:00
peertube_str += end_edit_section()
2021-12-30 23:46:06 +00:00
libretranslate_url = get_config_param(base_dir, 'libretranslateUrl')
libretranslate_api_key = get_config_param(base_dir, 'libretranslateApiKey')
libretranslate_str = \
2022-06-01 17:45:59 +00:00
_html_edit_profile_libre_translate(libretranslate_url,
2021-12-30 23:46:06 +00:00
libretranslate_api_key)
2023-08-31 22:29:10 +00:00
memorial_str = \
_html_edit_profile_memorial(base_dir, translate)
2023-08-31 22:29:10 +00:00
return instance_str, role_assign_str, peertube_str, \
libretranslate_str, memorial_str
2021-12-29 21:55:09 +00:00
def _html_edit_profile_danger_zone(translate: {}) -> str:
"""danger zone section of Edit Profile screen
"""
2021-12-30 23:46:06 +00:00
edit_profile_form = begin_edit_section(translate['Danger Zone'])
2021-07-22 12:36:31 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-07-22 10:22:01 +00:00
' <b><label class="labels">' + \
2021-07-22 12:36:31 +00:00
translate['Danger Zone'] + '</label></b><br>\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Deactivate this account'],
'deactivateThisAccount', False)
2021-07-22 12:36:31 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
2021-12-29 21:55:09 +00:00
def _html_system_monitor(nickname: str, translate: {}) -> str:
"""Links to performance graphs
"""
2021-12-30 23:46:06 +00:00
system_monitor_str = begin_edit_section(translate['System Monitor'])
system_monitor_str += '<p><a href="/users/' + nickname + \
2021-10-20 10:41:58 +00:00
'/performance?graph=get">📊 GET</a></p>'
2022-04-23 18:26:54 +00:00
system_monitor_str += '<p><a href="/users/' + nickname + \
'/performance?graph=inbox">📊 INBOX</a></p>'
2021-12-30 23:46:06 +00:00
system_monitor_str += '<p><a href="/users/' + nickname + \
2021-10-20 10:41:58 +00:00
'/performance?graph=post">📊 POST</a></p>'
2021-12-30 23:46:06 +00:00
system_monitor_str += end_edit_section()
return system_monitor_str
2021-12-29 21:55:09 +00:00
def _html_edit_profile_skills(base_dir: str, nickname: str, domain: str,
translate: {}) -> str:
"""skills section of Edit Profile screen
"""
2022-09-25 17:26:11 +00:00
system_language = 'en'
2021-12-29 21:55:09 +00:00
skills = get_skills(base_dir, nickname, domain)
2021-12-30 23:46:06 +00:00
skills_str = ''
skill_ctr = 1
if skills:
2021-12-30 23:46:06 +00:00
for skill_desc, skill_value in skills.items():
2022-09-25 17:26:11 +00:00
if is_filtered(base_dir, nickname, domain, skill_desc,
system_language):
continue
2021-12-30 23:46:06 +00:00
skills_str += \
'<p><input type="text" placeholder="' + translate['Skill'] + \
2021-12-30 23:46:06 +00:00
' ' + str(skill_ctr) + '" name="skillName' + str(skill_ctr) + \
'" value="' + skill_desc + '" style="width:40%">' + \
'<input type="range" min="1" max="100" ' + \
'class="slider" name="skillValue' + \
2021-12-30 23:46:06 +00:00
str(skill_ctr) + '" value="' + str(skill_value) + '"></p>'
skill_ctr += 1
2021-12-30 23:46:06 +00:00
skills_str += \
'<p><input type="text" placeholder="Skill ' + str(skill_ctr) + \
'" name="skillName' + str(skill_ctr) + \
2021-07-22 10:22:01 +00:00
'" value="" style="width:40%">' + \
'<input type="range" min="1" max="100" ' + \
'class="slider" name="skillValue' + \
2021-12-30 23:46:06 +00:00
str(skill_ctr) + '" value="50"></p>' + end_edit_section()
idx = 'If you want to participate within organizations then you ' + \
'can indicate some skills that you have and approximate ' + \
'proficiency levels. This helps organizers to construct ' + \
'teams with an appropriate combination of skills.'
2021-12-30 23:46:06 +00:00
edit_profile_form = \
2021-12-29 21:55:09 +00:00
begin_edit_section(translate['Skills']) + \
2021-07-22 10:22:01 +00:00
' <b><label class="labels">' + \
translate['Skills'] + '</label></b><br>\n' + \
' <label class="labels">' + \
2021-12-30 23:46:06 +00:00
translate[idx] + '</label>\n' + skills_str
return edit_profile_form
2021-12-29 21:55:09 +00:00
def _html_edit_profile_git_projects(base_dir: str, nickname: str, domain: str,
translate: {}) -> str:
"""git projects section of edit profile screen
"""
2021-12-30 23:46:06 +00:00
git_projects_str = ''
git_projects_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/gitprojects.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(git_projects_filename):
try:
with open(git_projects_filename, 'r',
2024-07-14 13:01:46 +00:00
encoding='utf-8') as fp_git:
git_projects_str = fp_git.read()
except OSError:
print('EX: _html_edit_profile_git_projects unable to read ' +
git_projects_filename)
2021-12-30 23:46:06 +00:00
edit_profile_form = begin_edit_section(translate['Git Projects'])
2021-07-22 19:00:37 +00:00
idx = 'List of project names that you wish to receive git patches for'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate[idx], None, 'gitProjects', git_projects_str,
2021-12-29 21:55:09 +00:00
100, '', False)
2021-12-30 23:46:06 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
2022-06-15 12:39:08 +00:00
def _html_edit_profile_shared_items(base_dir: str, translate: {}) -> str:
"""shared items section of edit profile screen
"""
2021-12-30 23:46:06 +00:00
shared_items_str = ''
shared_items_federated_domains_str = \
2021-12-30 13:56:38 +00:00
get_config_param(base_dir, 'sharedItemsFederatedDomains')
2021-12-30 23:46:06 +00:00
if shared_items_federated_domains_str:
shared_items_federated_domains_list = \
shared_items_federated_domains_str.split(',')
for shared_federated_domain in shared_items_federated_domains_list:
shared_items_str += shared_federated_domain.strip() + '\n'
2021-12-30 23:46:06 +00:00
edit_profile_form = begin_edit_section(translate['Shares'])
idx = 'List of domains which can access the shared items catalog'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate[idx], None, 'shareDomainList',
2021-12-30 23:46:06 +00:00
shared_items_str, 200, '', False)
edit_profile_form += end_edit_section()
return edit_profile_form
2021-12-29 21:55:09 +00:00
def _html_edit_profile_filtering(base_dir: str, nickname: str, domain: str,
user_agents_blocked: str,
2022-03-06 12:56:26 +00:00
crawlers_allowed: str,
2021-12-30 23:46:06 +00:00
translate: {}, reply_interval_hours: int,
2023-01-13 21:15:02 +00:00
cw_lists: {}, lists_enabled: str,
2024-02-09 18:39:44 +00:00
buy_sites: {}, block_military: {},
2024-12-15 13:01:26 +00:00
block_government: {},
2024-12-14 18:02:43 +00:00
block_bluesky: {},
2024-02-09 18:39:44 +00:00
block_federated_endpoints: []) -> str:
"""Filtering and blocking section of edit profile screen
"""
2021-12-30 23:46:06 +00:00
filter_str = ''
filter_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/filters.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(filter_filename):
try:
2024-07-16 12:20:58 +00:00
with open(filter_filename, 'r', encoding='utf-8') as fp_filter:
filter_str = fp_filter.read()
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
filter_filename)
2021-12-30 23:46:06 +00:00
filter_bio_str = ''
filter_bio_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/filters_bio.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(filter_bio_filename):
try:
with open(filter_bio_filename, 'r',
encoding='utf-8') as fp_filter:
filter_bio_str = fp_filter.read()
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
filter_bio_filename)
2021-12-14 14:27:56 +00:00
2021-12-30 23:46:06 +00:00
switch_str = ''
switch_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/replacewords.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(switch_filename):
try:
2024-07-16 12:20:58 +00:00
with open(switch_filename, 'r', encoding='utf-8') as fp_switch:
switch_str = fp_switch.read()
except OSError:
print('EX: _html_edit_profile_filtering unable to save ' +
switch_filename)
2021-12-30 23:46:06 +00:00
auto_tags = ''
auto_tags_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/autotags.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(auto_tags_filename):
try:
2024-07-14 13:01:46 +00:00
with open(auto_tags_filename, 'r', encoding='utf-8') as fp_auto:
auto_tags = fp_auto.read()
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
auto_tags_filename)
2021-12-30 23:46:06 +00:00
auto_cw = ''
auto_cw_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/autocw.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(auto_cw_filename):
try:
2024-07-14 13:01:46 +00:00
with open(auto_cw_filename, 'r', encoding='utf-8') as fp_cw:
auto_cw = fp_cw.read()
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
auto_cw_filename)
2022-11-23 18:40:45 +00:00
blocked_str = get_account_blocks(base_dir, nickname, domain)
2021-12-30 23:46:06 +00:00
dm_allowed_instances_str = ''
dm_allowed_instances_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/dmAllowedInstances.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(dm_allowed_instances_filename):
try:
with open(dm_allowed_instances_filename, 'r',
2024-07-14 13:01:46 +00:00
encoding='utf-8') as fp_dm:
dm_allowed_instances_str = fp_dm.read()
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
dm_allowed_instances_filename)
2021-12-30 23:46:06 +00:00
allowed_instances_str = ''
allowed_instances_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, domain) + '/allowedinstances.txt'
2021-12-30 23:46:06 +00:00
if os.path.isfile(allowed_instances_filename):
try:
with open(allowed_instances_filename, 'r',
2024-07-14 13:01:46 +00:00
encoding='utf-8') as fp_allow:
allowed_instances_str = fp_allow.read()
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
allowed_instances_filename)
2021-12-30 23:46:06 +00:00
edit_profile_form = begin_edit_section(translate['Filtering and Blocking'])
idx = 'Hours after posting during which replies are allowed'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
' <label class="labels">' + \
translate[idx] + \
':</label> <input type="number" name="replyhours" ' + \
'min="0" max="999999999999" step="1" ' + \
2021-12-30 23:46:06 +00:00
'value="' + str(reply_interval_hours) + '"><br>\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
'<label class="labels">' + \
translate['City for spoofed GPS image metadata'] + \
'</label><br>\n'
city = ''
2021-12-30 23:46:06 +00:00
city_filename = acct_dir(base_dir, nickname, domain) + '/city.txt'
if os.path.isfile(city_filename):
try:
2024-07-14 13:01:46 +00:00
with open(city_filename, 'r', encoding='utf-8') as fp_city:
city1 = fp_city.read()
city = remove_eol(city1)
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
city_filename)
2021-12-30 23:46:06 +00:00
locations_filename = base_dir + '/custom_locations.txt'
if not os.path.isfile(locations_filename):
locations_filename = base_dir + '/locations.txt'
2024-12-23 18:23:47 +00:00
cities: list[str] = []
try:
2024-07-14 13:01:46 +00:00
with open(locations_filename, 'r', encoding='utf-8') as fp_loc:
cities = fp_loc.readlines()
cities.sort()
except OSError:
print('EX: _html_edit_profile_filtering unable to read ' +
locations_filename)
2021-12-30 23:46:06 +00:00
edit_profile_form += ' <select id="cityDropdown" ' + \
'name="cityDropdown" class="theme">\n'
city = city.lower()
2021-12-30 23:46:06 +00:00
for city_name in cities:
if ':' not in city_name:
continue
2021-12-30 23:46:06 +00:00
city_selected = ''
city_name = city_name.split(':')[0]
city_name = city_name.lower()
if city:
2021-12-30 23:46:06 +00:00
if city in city_name:
city_selected = ' selected'
edit_profile_form += \
' <option value="' + city_name + \
'"' + city_selected.title() + '>' + \
city_name + '</option>\n'
edit_profile_form += ' </select><br>\n'
edit_profile_form += \
' <b><label class="labels">' + \
2021-07-22 10:22:01 +00:00
translate['Filtered words'] + '</label></b>\n' + \
' <br><label class="labels">' + \
translate['One per line'] + '</label>\n' + \
' <textarea id="message" ' + \
2024-05-06 19:44:14 +00:00
'name="filteredWords" style="height:500px" spellcheck="false">' + \
2021-12-30 23:46:06 +00:00
filter_str + '</textarea>\n' + \
' <br><b><label class="labels">' + \
translate['Filtered words within bio'] + '</label></b>\n' + \
' <br><label class="labels">' + \
translate['One per line'] + '</label>\n' + \
' <textarea id="message" ' + \
'name="filteredWordsBio" style="height:200px" spellcheck="false">' + \
2021-12-30 23:46:06 +00:00
filter_bio_str + '</textarea>\n' + \
' <br><b><label class="labels">' + \
2021-07-22 10:22:01 +00:00
translate['Word Replacements'] + '</label></b>\n' + \
' <br><label class="labels">A -> B</label>\n' + \
2021-12-30 12:23:55 +00:00
' <textarea id="message" name="switchwords" ' + \
2024-05-06 19:44:14 +00:00
'style="height:500px" spellcheck="false">' + \
2021-12-30 23:46:06 +00:00
switch_str + '</textarea>\n' + \
' <br><b><label class="labels">' + \
2021-07-22 10:22:01 +00:00
translate['Autogenerated Hashtags'] + '</label></b>\n' + \
' <br><label class="labels">A -> #B</label>\n' + \
' <textarea id="message" name="autoTags" ' + \
'style="height:200px" spellcheck="false">' + \
2021-12-30 23:46:06 +00:00
auto_tags + '</textarea>\n' + \
' <br><b><label class="labels">' + \
2021-07-22 10:22:01 +00:00
translate['Autogenerated Content Warnings'] + '</label></b>\n' + \
' <br><label class="labels">A -> B</label>\n' + \
' <textarea id="message" name="autoCW" ' + \
2024-05-06 19:44:14 +00:00
'style="height:500px" spellcheck="true">' + auto_cw + '</textarea>\n'
idx = 'Blocked accounts, one per line, in the form ' + \
'nickname@domain or *@blockeddomain'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['Blocked accounts'], None, 'blocked',
2024-05-06 19:44:14 +00:00
blocked_str, 500, '', False)
2023-02-10 12:02:10 +00:00
# import and export blocks
edit_profile_form += _import_export_blocks(translate)
2023-02-10 12:02:10 +00:00
idx = 'Direct messages are always allowed from these instances.'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['Direct Message permitted instances'], None,
2021-12-30 23:46:06 +00:00
'dmAllowedInstances', dm_allowed_instances_str,
2021-12-29 21:55:09 +00:00
200, '', False)
idx = 'Federate only with a defined set of instances. ' + \
'One domain name per line.'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-07-22 10:22:01 +00:00
' <br><b><label class="labels">' + \
translate['Federation list'] + '</label></b>\n' + \
' <br><label class="labels">' + \
2021-07-22 10:22:01 +00:00
translate[idx] + '</label>\n' + \
' <textarea id="message" name="allowedInstances" ' + \
'style="height:200px" spellcheck="false">' + \
2021-12-30 23:46:06 +00:00
allowed_instances_str + '</textarea>\n'
2021-12-28 19:33:29 +00:00
if is_moderator(base_dir, nickname):
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-10-24 09:57:10 +00:00
'<a href="/users/' + nickname + '/crawlers">' + \
translate['Known Web Crawlers'] + '</a><br>\n'
2021-12-30 13:56:38 +00:00
user_agents_blocked_str = ''
2021-12-30 23:46:06 +00:00
for uagent in user_agents_blocked:
2021-12-30 13:56:38 +00:00
if user_agents_blocked_str:
user_agents_blocked_str += '\n'
2021-12-30 23:46:06 +00:00
user_agents_blocked_str += uagent
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['Blocked User Agents'], None,
2021-12-30 13:56:38 +00:00
'userAgentsBlockedStr', user_agents_blocked_str,
2021-12-29 21:55:09 +00:00
200, '', False)
2021-10-21 19:45:39 +00:00
2022-03-06 15:28:02 +00:00
edit_profile_form += \
'<a href="/users/' + nickname + '/bots.txt">' + \
translate['Known Search Bots'] + '</a><br>\n'
2022-03-06 12:56:26 +00:00
crawlers_allowed_str = ''
for uagent in crawlers_allowed:
if crawlers_allowed_str:
crawlers_allowed_str += '\n'
crawlers_allowed_str += uagent
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['Web Bots Allowed'], None,
2022-03-06 12:56:26 +00:00
'crawlersAllowedStr', crawlers_allowed_str,
200, '', False)
2023-01-13 21:15:02 +00:00
buy_domains_list_str = ''
for buy_icon_text, buy_url in buy_sites.items():
if buy_icon_text != buy_url:
buy_domains_list_str += \
buy_icon_text + ' ' + buy_url.strip() + '\n'
else:
buy_domains_list_str += buy_url.strip() + '\n'
buy_domains_str = \
"Buy links are allowed from the following domains"
edit_profile_form += \
edit_text_area(translate[buy_domains_str], None,
'buySitesStr', buy_domains_list_str,
200, '', False)
2024-02-09 18:39:44 +00:00
block_federated_endpoints_list_str = ''
for block_api_url in block_federated_endpoints:
block_federated_endpoints_list_str += block_api_url.strip() + '\n'
block_federated_str = "Blocking API endpoints"
edit_profile_form += \
edit_text_area(translate[block_federated_str], None,
'blockFederated',
block_federated_endpoints_list_str,
200, '', False)
robots_txt_filename = data_dir(base_dir) + '/robots.txt'
robots_txt = ''
if os.path.isfile(robots_txt_filename):
new_robots_txt = ''
try:
with open(robots_txt_filename, 'r',
encoding='utf-8') as fp_robots:
new_robots_txt = fp_robots.read()
except OSError:
print('EX: error reading 2 ' + robots_txt_filename)
if new_robots_txt:
robots_txt = new_robots_txt
edit_profile_form += \
edit_text_area("robots.txt", None,
'robotsTxt', robots_txt,
200, '', False)
2024-02-09 21:41:32 +00:00
idx = 'Block military instances'
if translate.get(idx):
name = translate[idx]
block_mil = False
if block_military.get(nickname):
block_mil = block_military[nickname]
edit_profile_form += \
edit_check_box(idx, 'blockMilitary', block_mil)
2024-12-15 13:01:26 +00:00
idx = 'Block government instances'
if translate.get(idx):
name = translate[idx]
block_gov = False
if block_government.get(nickname):
block_gov = block_government[nickname]
edit_profile_form += \
edit_check_box(idx, 'blockGovernment', block_gov)
2024-12-14 18:02:43 +00:00
idx = 'Block BlueSky bridges'
if translate.get(idx):
name = translate[idx]
block_bsky = False
if block_bluesky.get(nickname):
block_bsky = block_bluesky[nickname]
edit_profile_form += \
edit_check_box(idx, 'blockBlueSky', block_bsky)
2021-12-30 23:46:06 +00:00
cw_lists_str = ''
for name, _ in cw_lists.items():
variablename = get_cw_list_variable(name)
list_is_enabled = False
2021-12-25 18:12:13 +00:00
if lists_enabled:
if name in lists_enabled:
2021-12-30 23:46:06 +00:00
list_is_enabled = True
2021-10-21 19:45:39 +00:00
if translate.get(name):
name = translate[name]
2021-12-30 23:46:06 +00:00
cw_lists_str += \
edit_check_box(name, variablename, list_is_enabled)
if cw_lists_str:
2021-10-21 19:45:39 +00:00
idx = 'Add content warnings for the following sites'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-10-21 19:45:39 +00:00
'<label class="labels">' + translate[idx] + ':</label>\n' + \
2021-12-30 23:46:06 +00:00
'<br>' + cw_lists_str
2021-10-21 16:58:01 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
2021-12-29 21:55:09 +00:00
def _html_edit_profile_change_password(translate: {}) -> str:
"""Change password section of edit profile screen
"""
2021-12-30 23:46:06 +00:00
edit_profile_form = \
2021-12-29 21:55:09 +00:00
begin_edit_section(translate['Change Password']) + \
'<label class="labels">' + translate['Change Password'] + \
2021-07-06 10:18:56 +00:00
'</label><br>\n' + \
2021-07-20 08:44:32 +00:00
' <input type="password" name="password" ' + \
2021-07-06 10:18:56 +00:00
'value=""><br>\n' + \
'<label class="labels">' + translate['Confirm Password'] + \
2021-07-06 10:18:56 +00:00
'</label><br>\n' + \
2021-07-20 08:44:32 +00:00
' <input type="password" name="passwordconfirm" value="">\n' + \
2021-12-29 21:55:09 +00:00
end_edit_section()
2021-12-30 23:46:06 +00:00
return edit_profile_form
2022-06-01 17:45:59 +00:00
def _html_edit_profile_libre_translate(libretranslate_url: str,
2021-12-30 23:46:06 +00:00
libretranslate_api_key: str) -> str:
2021-07-19 19:40:04 +00:00
"""Change automatic translation settings
"""
2021-12-30 23:46:06 +00:00
edit_profile_form = begin_edit_section('LibreTranslate')
2021-07-19 19:40:04 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += \
edit_text_field('URL', 'libretranslateUrl', libretranslate_url,
2021-12-29 21:55:09 +00:00
'http://0.0.0.0:5000')
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_text_field('API Key', 'libretranslateApiKey',
2021-12-30 23:46:06 +00:00
libretranslate_api_key)
2021-07-22 13:26:43 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
2021-07-19 19:40:04 +00:00
2023-08-31 22:29:10 +00:00
def _html_edit_profile_memorial(base_dir: str, translate: {}) -> str:
"""Change memorial accounts
"""
memorial_accounts_str = get_memorials(base_dir)
edit_profile_form = begin_edit_section(translate['Memorials'])
title_str = \
translate['Nicknames of memorial accounts (one per line)']
2023-08-31 22:44:39 +00:00
edit_profile_form += \
edit_text_area(title_str, None, 'memorialAccounts',
memorial_accounts_str, 200, '', True)
2023-08-31 22:29:10 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
2021-12-29 21:55:09 +00:00
def _html_edit_profile_background(news_instance: bool, translate: {}) -> str:
"""Background images section of edit profile screen
"""
idx = 'The files attached below should be no larger than ' + \
'10MB in total uploaded at once.'
2021-12-30 23:46:06 +00:00
edit_profile_form = \
2021-12-29 21:55:09 +00:00
begin_edit_section(translate['Background Images']) + \
' <label class="labels">' + translate[idx] + '</label><br><br>\n'
2021-12-25 20:20:08 +00:00
if not news_instance:
2021-12-30 23:46:06 +00:00
image_formats = get_image_formats()
edit_profile_form += \
' <label class="labels">' + \
2021-07-06 10:18:56 +00:00
translate['Background image'] + '</label>\n' + \
' <input type="file" id="image" name="image"' + \
2021-12-30 23:46:06 +00:00
' accept="' + image_formats + '">\n' + \
2021-07-06 10:18:56 +00:00
' <br><label class="labels">' + \
translate['Timeline banner image'] + '</label>\n' + \
' <input type="file" id="banner" name="banner"' + \
2021-12-30 23:46:06 +00:00
' accept="' + image_formats + '">\n' + \
2021-07-06 10:18:56 +00:00
' <br><label class="labels">' + \
translate['Search banner image'] + '</label>\n' + \
' <input type="file" id="search_banner" ' + \
'name="search_banner"' + \
2021-12-30 23:46:06 +00:00
' accept="' + image_formats + '">\n' + \
2021-07-06 10:18:56 +00:00
' <br><label class="labels">' + \
translate['Left column image'] + '</label>\n' + \
' <input type="file" id="left_col_image" ' + \
'name="left_col_image"' + \
2021-12-30 23:46:06 +00:00
' accept="' + image_formats + '">\n' + \
2021-07-06 10:18:56 +00:00
' <br><label class="labels">' + \
translate['Right column image'] + '</label>\n' + \
' <input type="file" id="right_col_image" ' + \
'name="right_col_image"' + \
' accept="' + image_formats + '">\n' + \
' <br><label class="labels">' + \
translate['Watermark image'] + '</label>\n' + \
' <input type="file" id="watermark_image" ' + \
'name="watermark_image"' + \
2021-12-30 23:46:06 +00:00
' accept="' + image_formats + '">\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
2022-09-08 18:11:48 +00:00
def _html_edit_profile_contact_info(email_address: str,
2021-12-30 23:46:06 +00:00
xmpp_address: str,
matrix_address: str,
ssb_address: str,
tox_address: str,
briar_address: str,
2021-12-30 20:48:38 +00:00
cwtch_address: str,
translate: {},
youtube: str,
peertube: str,
pixelfed: str,
discord: str,
2024-08-15 19:58:26 +00:00
music_site_url: str,
art_site_url: str) -> str:
"""Contact Information section of edit profile screen
"""
2021-12-30 23:46:06 +00:00
edit_profile_form = begin_edit_section(translate['Contact Details'])
edit_profile_form += edit_text_field(translate['Email'],
'email', email_address)
edit_profile_form += edit_text_field(translate['XMPP'],
'xmppAddress', xmpp_address)
edit_profile_form += edit_text_field(translate['Matrix'],
'matrixAddress', matrix_address)
edit_profile_form += edit_text_field('SSB', 'ssbAddress', ssb_address)
edit_profile_form += edit_text_field('Tox', 'toxAddress', tox_address)
edit_profile_form += edit_text_field('Briar', 'briarAddress',
briar_address)
edit_profile_form += edit_text_field('Cwtch', 'cwtchAddress',
cwtch_address)
edit_profile_form += edit_text_field('YouTube', 'youtubeChannel',
youtube)
edit_profile_form += edit_text_field('PeerTube', 'peertubeChannel',
peertube)
edit_profile_form += edit_text_field('Pixelfed', 'pixelfedChannel',
pixelfed)
edit_profile_form += edit_text_field('Discord', 'discordChannel',
discord)
2024-08-15 19:58:26 +00:00
edit_profile_form += edit_text_field(translate['Art'],
'artSiteUrl', art_site_url)
edit_profile_form += edit_text_field(translate['Music'],
'musicSiteUrl', music_site_url)
edit_profile_form += end_edit_section()
return edit_profile_form
def _html_edit_notifications(base_dir: str, nickname: str, domain: str,
translate: {}) -> str:
"""Notifications settings
"""
ntfy_url = "ntfy.sh"
ntfy_topic = ''
ntfy_url_file = \
2022-12-18 13:58:48 +00:00
acct_dir(base_dir, nickname, domain) + '/.ntfy_url'
ntfy_topic_file = \
2022-12-18 13:58:48 +00:00
acct_dir(base_dir, nickname, domain) + '/.ntfy_topic'
if os.path.isfile(ntfy_url_file):
try:
with open(ntfy_url_file, 'r', encoding='utf-8') as fp_ntfy:
ntfy_url = fp_ntfy.read()
except OSError:
print('EX: unable to read ' + ntfy_url_file)
if os.path.isfile(ntfy_topic_file):
try:
with open(ntfy_topic_file, 'r', encoding='utf-8') as fp_ntfy:
ntfy_topic = fp_ntfy.read()
except OSError:
print('EX: unable to read ' + ntfy_topic_file)
edit_profile_form = begin_edit_section(translate['Notifications'])
edit_profile_form += edit_text_field(translate['ntfy URL'],
2022-08-19 10:25:29 +00:00
'ntfyUrl', ntfy_url)
edit_profile_form += edit_text_field(translate['ntfy topic'],
'ntfyTopic', ntfy_topic)
edit_profile_form += end_edit_section()
return edit_profile_form
def _html_edit_profile_import_export(nickname: str, domain: str,
translate: {}, premium: bool) -> str:
"""Contact Information section of edit profile screen
"""
edit_profile_form = begin_edit_section(translate['Import and Export'])
2022-07-21 09:58:28 +00:00
edit_profile_form += \
'<p><label class="labels">' + \
translate['Import Follows'] + '</label>\n'
2023-02-10 19:30:25 +00:00
edit_profile_form += '<input type="file" id="importFollows" '
edit_profile_form += 'name="importFollows" '
2022-07-21 09:58:28 +00:00
edit_profile_form += 'accept=".csv"></p>\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
'<p><a href="/users/' + nickname + \
'/followingaccounts"><label class="labels">' + \
2022-07-20 13:04:09 +00:00
translate['Following'] + '</label></a>'
# NOTE: don't use download="preferredfilename" which is
# unsupported by some browsers
2022-07-19 13:31:13 +00:00
edit_profile_form += \
2022-07-20 13:04:09 +00:00
' <a href="/users/' + nickname + '/followingaccounts.csv" ' + \
'download>' + \
2024-07-12 11:33:27 +00:00
'<label class="labels">⇩ CSV</label></a></p><br>\n'
followers_str = translate['Followers']
if premium:
followers_str = translate['Fans']
2022-07-20 13:04:09 +00:00
edit_profile_form += \
'<p><a href="/users/' + nickname + \
2022-07-19 13:31:13 +00:00
'/followersaccounts"><label class="labels">' + \
followers_str + '</label></a><br></p>\n'
edit_profile_form += _import_export_blocks(translate)
2021-12-30 23:46:06 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
2021-12-30 23:46:06 +00:00
def _html_edit_profile_encryption_keys(pgp_fingerprint: str,
pgp_pub_key: str,
enigma_pub_key: str,
2021-12-29 21:55:09 +00:00
translate: {}) -> str:
"""Contact Information section of edit profile screen
"""
2021-12-30 23:46:06 +00:00
edit_profile_form = begin_edit_section(translate['Encryption Keys'])
enigma_url = 'https://github.com/enigma-reloaded/enigma-reloaded'
edit_profile_form += \
edit_text_field('<a href="' + enigma_url + '">Enigma</a>',
'enigmapubkey', enigma_pub_key)
edit_profile_form += edit_text_field(translate['PGP Fingerprint'],
'openpgp', pgp_fingerprint)
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['PGP'], None, 'pgp', pgp_pub_key, 600,
2021-12-29 21:55:09 +00:00
'-----BEGIN PGP PUBLIC KEY BLOCK-----', False)
2021-07-22 10:22:01 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += end_edit_section()
return edit_profile_form
def _html_edit_profile_reply_controls(translate: {},
follow_dms: bool,
premium: bool,
show_replies_followers: bool,
show_replies_mutuals: bool,
no_reply_boosts: bool) -> str:
"""option checkboxes for reply controls
"""
edit_profile_form = begin_edit_section(translate['Reply Controls'])
edit_profile_form += \
edit_check_box(translate['Only people I follow can send me DMs'],
'followDMs', follow_dms)
show_replies_followers_str = translate['Only allow replies from followers']
if premium:
show_replies_followers_str = translate['Only allow replies from fans']
edit_profile_form += \
edit_check_box(show_replies_followers_str, 'repliesFromFollowersOnly',
show_replies_followers)
show_replies_mutuals_str = translate['Only allow replies from mutuals']
edit_profile_form += \
edit_check_box(show_replies_mutuals_str, 'repliesFromMutualsOnly',
show_replies_mutuals)
no_reply_boosts_str = translate["Don't show boosted replies"]
edit_profile_form += \
edit_check_box(no_reply_boosts_str, 'noReplyBoosts', no_reply_boosts)
edit_profile_form += end_edit_section()
return edit_profile_form
def _html_edit_profile_options(is_admin: bool,
manually_approves_followers: str,
reject_spam_actors: str,
is_bot: str, is_group: str,
remove_twitter: str,
notify_likes: str, notify_reactions: str,
hide_like_button: str,
hide_reaction_button: str,
translate: {}, bold_reading: bool,
nickname: str,
min_images_for_accounts: [],
reverse_sequence: [],
show_quote_toots: bool,
show_vote_posts: bool,
hide_follows: bool,
premium: bool,
no_seen_posts: bool,
watermark_enabled: bool) -> str:
"""option checkboxes section of edit profile screen
"""
edit_profile_form = ' <div class="container">\n'
edit_profile_form += \
edit_check_box(translate['Premium account'], 'premiumAccount', premium)
approve_followers_str = translate['Approve follower requests']
if premium:
approve_followers_str = translate['Approve fans']
2021-12-30 23:46:06 +00:00
edit_profile_form += \
edit_check_box(approve_followers_str,
2021-12-30 23:46:06 +00:00
'approveFollowers', manually_approves_followers)
edit_profile_form += \
edit_check_box(translate['Reject spam accounts'],
'rejectSpamActors', reject_spam_actors)
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['This is a bot account'],
2021-12-30 23:46:06 +00:00
'isBot', is_bot)
if is_admin:
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['This is a group account'],
2021-12-30 23:46:06 +00:00
'isGroup', is_group)
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Remove Twitter posts'],
2021-12-30 23:46:06 +00:00
'removeTwitter', remove_twitter)
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Notify when posts are liked'],
2021-12-30 23:46:06 +00:00
'notifyLikes', notify_likes)
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Notify on emoji reactions'],
2021-12-30 23:46:06 +00:00
'notifyReactions', notify_reactions)
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate["Don't show the Like button"],
2021-12-30 23:46:06 +00:00
'hideLikeButton', hide_like_button)
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate["Don't show the Reaction button"],
2021-12-30 23:46:06 +00:00
'hideReactionButton', hide_reaction_button)
2022-03-24 13:14:41 +00:00
bold_str = bold_reading_string(translate['Bold reading'])
edit_profile_form += \
edit_check_box(bold_str, 'boldReading', bold_reading)
minimize_all_images = False
if nickname in min_images_for_accounts:
minimize_all_images = True
2023-04-13 16:19:19 +00:00
minimize_all_images_str = translate['Minimize all images']
edit_profile_form += \
edit_check_box(minimize_all_images_str, 'minimizeAllImages',
minimize_all_images)
reverse = False
if nickname in reverse_sequence:
reverse = True
2023-04-13 16:19:19 +00:00
reverse_str = translate['Reverse timelines']
edit_profile_form += \
edit_check_box(reverse_str, 'reverseTimelines', reverse)
show_quote_toots_str = translate['Show quote posts']
edit_profile_form += \
edit_check_box(show_quote_toots_str, 'showQuotes', show_quote_toots)
2023-04-13 16:19:19 +00:00
show_vote_posts_str = translate['Show vote posts']
2023-04-05 17:13:22 +00:00
edit_profile_form += \
edit_check_box(show_vote_posts_str, 'showVotes', show_vote_posts)
hide_follows_str = translate['Do not show follows on your profile']
if premium:
hide_follows_str = translate['Do not show fans on your profile']
edit_profile_form += \
edit_check_box(hide_follows_str, 'hideFollows', hide_follows)
no_seen_posts_str = translate["Don't show already seen posts"]
edit_profile_form += \
edit_check_box(no_seen_posts_str, 'noSeenPosts', no_seen_posts)
2024-07-29 16:23:59 +00:00
watermark_str = translate["Apply a watermark to uploaded images"]
edit_profile_form += \
edit_check_box(watermark_str, 'watermarkEnabled', watermark_enabled)
2021-12-30 23:46:06 +00:00
edit_profile_form += ' </div>\n'
return edit_profile_form
2021-12-26 17:26:55 +00:00
def _get_supported_languagesSorted(base_dir: str) -> str:
"""Returns a list of supported languages
"""
2021-12-26 17:26:55 +00:00
lang_list = get_supported_languages(base_dir)
2021-12-26 10:35:37 +00:00
if not lang_list:
2021-07-18 18:57:08 +00:00
return ''
2021-12-26 10:35:37 +00:00
lang_list.sort()
2021-12-30 23:46:06 +00:00
languages_str = ''
2021-12-26 10:35:37 +00:00
for lang in lang_list:
2021-12-30 23:46:06 +00:00
if languages_str:
languages_str += ' / ' + lang
2021-07-18 18:57:08 +00:00
else:
2021-12-30 23:46:06 +00:00
languages_str = lang
return languages_str
def _html_edit_profile_main(base_dir: str, display_nickname: str,
pronouns: str, bio_str: str,
2021-12-30 23:46:06 +00:00
moved_to: str, donate_url: str, website_url: str,
2022-11-04 21:07:55 +00:00
gemini_link: str, blog_address: str,
actor_json: {}, translate: {},
nickname: str, domain: str,
max_recent_posts: int,
featured_hashtags: str) -> str:
"""main info on edit profile screen
"""
2021-12-30 23:46:06 +00:00
image_formats = get_image_formats()
2021-12-30 23:46:06 +00:00
edit_profile_form = ' <div class="container">\n'
2021-07-06 10:18:56 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Nickname'], 'displayNickname',
2021-12-30 23:46:06 +00:00
display_nickname)
2021-07-06 10:18:56 +00:00
edit_profile_form += \
edit_text_field(translate['Pronouns'], 'setPronouns',
pronouns)
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2022-09-02 15:57:06 +00:00
edit_text_area(translate['Your bio'], None, 'bio', bio_str,
200, '', True)
2021-07-06 10:18:56 +00:00
2024-01-07 15:59:05 +00:00
birth_date = ''
birth_date_field = 'vcard:bday'
if actor_json.get(birth_date_field):
if '-' in actor_json[birth_date_field]:
if len(actor_json[birth_date_field].split('-')) == 3:
if 'T' in actor_json[birth_date_field]:
actor_json[birth_date_field] = \
actor_json[birth_date_field].split('T')[0]
birth_date = actor_json[birth_date_field]
2024-01-07 16:03:01 +00:00
edit_profile_form += '<br>\n<label class="labels">' + \
2024-01-07 15:59:05 +00:00
translate['Birthday'] + ': </label>\n'
edit_profile_form += \
'<input type="date" name="birthDate" value="' + birth_date + '">\n'
2024-01-07 16:03:01 +00:00
edit_profile_form += '<br>\n' + \
edit_text_field(translate['Featured hashtags'], 'featuredHashtags',
featured_hashtags, '#tag1 #tag2')
2021-12-30 23:46:06 +00:00
edit_profile_form += \
' <label class="labels">' + translate['Avatar image'] + \
2021-07-06 10:18:56 +00:00
'</label>\n' + \
' <input type="file" id="avatar" name="avatar"' + \
2021-12-30 23:46:06 +00:00
' accept="' + image_formats + '">\n'
2021-12-30 23:46:06 +00:00
occupation_name = ''
2021-12-26 10:29:52 +00:00
if actor_json.get('hasOccupation'):
2021-12-30 23:46:06 +00:00
occupation_name = get_occupation_name(actor_json)
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Occupation'], 'occupationName',
2021-12-30 23:46:06 +00:00
occupation_name)
2021-12-30 23:46:06 +00:00
also_known_as_str = ''
2021-12-26 10:29:52 +00:00
if actor_json.get('alsoKnownAs'):
2021-12-30 23:46:06 +00:00
also_known_as = actor_json['alsoKnownAs']
ctr = 0
2021-12-30 23:46:06 +00:00
for alt_actor in also_known_as:
if ctr > 0:
2021-12-30 23:46:06 +00:00
also_known_as_str += ', '
ctr += 1
2021-12-30 23:46:06 +00:00
also_known_as_str += alt_actor
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Other accounts'], 'alsoKnownAs',
2021-12-30 23:46:06 +00:00
also_known_as_str, 'https://...')
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Moved to new account address'], 'movedTo',
2021-12-30 23:46:06 +00:00
moved_to, 'https://...')
2021-07-06 10:18:56 +00:00
2024-03-20 20:42:12 +00:00
donate_str = translate['Donations link']
if is_premium_account(base_dir, nickname, domain):
donate_str = translate['Premium account subscription link']
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2024-03-20 20:42:12 +00:00
edit_text_field(donate_str, 'donateUrl', donate_url, 'https://...')
2021-07-06 10:18:56 +00:00
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Website'], 'websiteUrl',
2021-12-30 23:46:06 +00:00
website_url, 'https://...')
2021-08-12 20:40:23 +00:00
2022-11-04 21:07:55 +00:00
edit_profile_form += \
edit_text_field('Gemini', 'geminiLink',
gemini_link, 'gemini://...')
2021-12-30 23:46:06 +00:00
edit_profile_form += \
edit_text_field('Blog', 'blogAddress', blog_address, 'https://...')
2021-12-30 23:46:06 +00:00
languages_list_str = _get_supported_languagesSorted(base_dir)
show_languages = get_actor_languages(actor_json)
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_text_field(translate['Languages'], 'showLanguages',
2021-12-30 23:46:06 +00:00
show_languages, languages_list_str)
2021-07-22 13:26:43 +00:00
2022-02-25 19:57:31 +00:00
timezone = get_account_timezone(base_dir, nickname, domain)
edit_profile_form += \
edit_text_field(translate['Time Zone'], 'timeZone',
timezone, 'Europe/London')
post_expiry_period_days = \
get_post_expiry_days(base_dir, nickname, domain)
edit_profile_form += \
edit_number_field(translate['Post expiry period in days'],
'postExpiryPeriod', post_expiry_period_days,
0, 9999999999999999999999, 0)
keep_dms = get_post_expiry_keep_dms(base_dir, nickname, domain)
2022-08-10 20:42:32 +00:00
edit_profile_form += '<br>\n' + \
edit_check_box(translate['Keep DMs during post expiry'],
'expiryKeepDMs', keep_dms)
max_profile_posts = \
get_max_profile_posts(base_dir, nickname, domain, max_recent_posts)
edit_profile_form += \
edit_number_field(translate['Preview posts on profile screen'],
'maxRecentProfilePosts', max_profile_posts,
1, 20, max_recent_posts)
2021-12-30 23:46:06 +00:00
edit_profile_form += ' </div>\n'
return edit_profile_form
2021-12-29 21:55:09 +00:00
def _html_edit_profile_top_banner(base_dir: str,
nickname: str, domain: str, domain_full: str,
2021-12-30 23:46:06 +00:00
default_timeline: str, banner_file: str,
path: str, access_keys: {},
2021-12-29 21:55:09 +00:00
translate: {}) -> str:
"""top banner on edit profile screen
"""
2021-12-30 23:46:06 +00:00
edit_profile_form = \
'<a href="/users/' + nickname + '/' + default_timeline + '">' + \
2022-03-28 08:47:53 +00:00
'<img loading="lazy" decoding="async" ' + \
'class="timeline-banner" src="' + \
2021-12-30 23:46:06 +00:00
'/users/' + nickname + '/' + banner_file + '" alt="" /></a>\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += \
'<form enctype="multipart/form-data" method="POST" ' + \
'accept-charset="UTF-8" action="' + path + '/profiledata">\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += ' <div class="vertical-center">\n'
edit_profile_form += \
' <h1>' + translate['Profile for'] + \
2021-12-26 10:00:46 +00:00
' ' + nickname + '@' + domain_full + '</h1>'
2021-12-30 23:46:06 +00:00
edit_profile_form += ' <div class="container">\n'
edit_profile_form += \
' <center>\n' + \
' <input type="submit" name="submitProfile" ' + \
2021-12-30 23:46:06 +00:00
'accesskey="' + access_keys['submitButton'] + '" ' + \
2022-05-23 18:48:45 +00:00
'value="' + translate['Publish'] + '">\n' + \
' </center>\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += ' </div>\n'
2021-12-29 21:55:09 +00:00
if scheduled_posts_exist(base_dir, nickname, domain):
2021-12-30 23:46:06 +00:00
edit_profile_form += ' <div class="container">\n'
edit_profile_form += \
2021-12-29 21:55:09 +00:00
edit_check_box(translate['Remove scheduled posts'],
2021-12-30 13:56:38 +00:00
'removeScheduledPosts', False)
2021-12-30 23:46:06 +00:00
edit_profile_form += ' </div>\n'
return edit_profile_form
2022-06-15 12:39:08 +00:00
def html_edit_profile(server, translate: {},
2022-03-24 13:14:41 +00:00
base_dir: str, path: str,
2022-06-15 12:39:08 +00:00
domain: str, port: int,
2021-12-30 23:46:06 +00:00
default_timeline: str, theme: str,
2021-12-29 21:55:09 +00:00
peertube_instances: [],
2022-06-15 12:39:08 +00:00
text_mode_banner: str,
2022-03-06 12:56:26 +00:00
user_agents_blocked: [],
crawlers_allowed: [],
2021-12-30 23:46:06 +00:00
access_keys: {},
2021-12-29 21:55:09 +00:00
default_reply_interval_hrs: int,
2022-09-25 17:26:11 +00:00
cw_lists: {}, lists_enabled: str,
system_language: str,
min_images_for_accounts: [],
max_recent_posts: int,
2023-01-13 21:15:02 +00:00
reverse_sequence: [],
2023-07-18 14:05:54 +00:00
buy_sites: {},
2024-02-09 18:39:44 +00:00
block_military: {},
2024-12-15 13:01:26 +00:00
block_government: {},
2024-12-14 18:02:43 +00:00
block_bluesky: {},
2024-02-09 18:39:44 +00:00
block_federated_endpoints: []) -> str:
2020-11-09 22:44:03 +00:00
"""Shows the edit profile screen
"""
2024-08-09 12:29:22 +00:00
replacements = {
'/inbox': '',
'/outbox': '',
'/shares': '',
'/wanted': ''
}
path = replace_strings(path, replacements)
2021-12-27 22:19:18 +00:00
nickname = get_nickname_from_actor(path)
2020-11-09 22:44:03 +00:00
if not nickname:
return ''
2021-12-26 12:45:03 +00:00
domain_full = get_full_domain(domain, port)
2020-11-09 22:44:03 +00:00
2022-03-24 13:14:41 +00:00
bold_reading = False
2022-03-24 13:21:40 +00:00
if server.bold_reading.get(nickname):
2022-03-24 13:14:41 +00:00
bold_reading = True
2021-12-30 23:46:06 +00:00
actor_filename = acct_dir(base_dir, nickname, domain) + '.json'
if not os.path.isfile(actor_filename):
2020-11-09 22:44:03 +00:00
return ''
# filename of the banner shown at the top
2021-12-30 23:46:06 +00:00
banner_file, _ = \
get_profile_background_file(base_dir, nickname, domain, theme)
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
display_nickname = nickname
is_bot = is_group = follow_dms = remove_twitter = ''
notify_likes = notify_reactions = ''
hide_like_button = hide_reaction_button = media_instance_str = ''
blogs_instance_str = news_instance_str = moved_to = twitter_str = ''
bio_str = donate_url = website_url = gemini_link = ''
email_address = featured_hashtags = pgp_pub_key = enigma_pub_key = ''
pgp_fingerprint = pronouns = peertube = youtube = pixelfed = ''
ssb_address = blog_address = matrix_address = tox_address = ''
cwtch_address = briar_address = xmpp_address = ''
2024-08-15 19:58:26 +00:00
discord = music_site_url = art_site_url = ''
manually_approves_followers = reject_spam_actors = ''
2021-12-30 23:46:06 +00:00
actor_json = load_json(actor_filename)
2021-12-26 10:29:52 +00:00
if actor_json:
if actor_json.get('movedTo'):
2021-12-30 23:46:06 +00:00
moved_to = actor_json['movedTo']
featured_hashtags = get_featured_hashtags(actor_json)
2021-12-30 23:46:06 +00:00
donate_url = get_donation_url(actor_json)
website_url = get_website(actor_json, translate)
2024-02-28 19:44:47 +00:00
gemini_link = get_gemini_link(actor_json)
pronouns = get_pronouns(actor_json)
pixelfed = get_pixelfed(actor_json)
discord = get_discord(actor_json)
2024-08-15 19:58:26 +00:00
art_site_url = get_art_site_url(actor_json)
music_site_url = get_music_site_url(actor_json)
youtube = get_youtube(actor_json)
peertube = get_peertube(actor_json)
2021-12-30 23:46:06 +00:00
xmpp_address = get_xmpp_address(actor_json)
matrix_address = get_matrix_address(actor_json)
ssb_address = get_ssb_address(actor_json)
blog_address = get_blog_address(actor_json)
tox_address = get_tox_address(actor_json)
briar_address = get_briar_address(actor_json)
2021-12-30 20:48:38 +00:00
cwtch_address = get_cwtch_address(actor_json)
2021-12-30 23:46:06 +00:00
email_address = get_email_address(actor_json)
enigma_pub_key = get_enigma_pub_key(actor_json)
pgp_pub_key = get_pgp_pub_key(actor_json)
pgp_fingerprint = get_pgp_fingerprint(actor_json)
2021-12-26 10:29:52 +00:00
if actor_json.get('name'):
2022-09-25 17:26:11 +00:00
if not is_filtered(base_dir, nickname, domain, actor_json['name'],
system_language):
2021-12-30 23:46:06 +00:00
display_nickname = actor_json['name']
2021-12-26 10:29:52 +00:00
if actor_json.get('summary'):
2021-12-30 23:46:06 +00:00
bio_str = \
2021-12-26 10:29:52 +00:00
actor_json['summary'].replace('<p>', '').replace('</p>', '')
2022-09-25 17:26:11 +00:00
if is_filtered(base_dir, nickname, domain, bio_str,
system_language):
2021-12-30 23:46:06 +00:00
bio_str = ''
bio_str = remove_html(bio_str)
2021-12-26 10:29:52 +00:00
if actor_json.get('manuallyApprovesFollowers'):
if actor_json['manuallyApprovesFollowers']:
2021-12-30 23:46:06 +00:00
manually_approves_followers = 'checked'
2020-11-09 22:44:03 +00:00
else:
2021-12-30 23:46:06 +00:00
manually_approves_followers = ''
reject_spam_actors = ''
actor_spam_filter_filename = \
acct_dir(base_dir, nickname, domain) + '/.reject_spam_actors'
if os.path.isfile(actor_spam_filter_filename):
reject_spam_actors = 'checked'
2021-12-26 10:29:52 +00:00
if actor_json.get('type'):
if actor_json['type'] == 'Service':
2021-12-30 23:46:06 +00:00
is_bot = 'checked'
is_group = ''
2021-12-26 10:29:52 +00:00
elif actor_json['type'] == 'Group':
2021-12-30 23:46:06 +00:00
is_group = 'checked'
is_bot = ''
account_dir = acct_dir(base_dir, nickname, domain)
if os.path.isfile(account_dir + '/.followDMs'):
follow_dms = 'checked'
if os.path.isfile(account_dir + '/.removeTwitter'):
remove_twitter = 'checked'
if os.path.isfile(account_dir + '/.notifyLikes'):
notify_likes = 'checked'
if os.path.isfile(account_dir + '/.notifyReactions'):
notify_reactions = 'checked'
if os.path.isfile(account_dir + '/.hideLikeButton'):
hide_like_button = 'checked'
if os.path.isfile(account_dir + '/.hideReactionButton'):
hide_reaction_button = 'checked'
2020-11-09 22:44:03 +00:00
2021-12-30 13:56:38 +00:00
media_instance = get_config_param(base_dir, "mediaInstance")
2021-12-25 20:25:07 +00:00
if media_instance:
if media_instance is True:
2021-12-30 23:46:06 +00:00
media_instance_str = 'checked'
blogs_instance_str = news_instance_str = ''
2020-11-09 22:44:03 +00:00
2021-12-30 13:56:38 +00:00
news_instance = get_config_param(base_dir, "newsInstance")
2021-12-25 20:20:08 +00:00
if news_instance:
if news_instance is True:
2021-12-30 23:46:06 +00:00
news_instance_str = 'checked'
blogs_instance_str = media_instance_str = ''
2020-11-09 22:44:03 +00:00
2021-12-30 13:56:38 +00:00
blogs_instance = get_config_param(base_dir, "blogsInstance")
2021-12-25 20:22:25 +00:00
if blogs_instance:
if blogs_instance is True:
2021-12-30 23:46:06 +00:00
blogs_instance_str = 'checked'
media_instance_str = news_instance_str = ''
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
css_filename = base_dir + '/epicyon-profile.css'
2021-12-25 16:17:53 +00:00
if os.path.isfile(base_dir + '/epicyon.css'):
2021-12-30 23:46:06 +00:00
css_filename = base_dir + '/epicyon.css'
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
instance_str = ''
role_assign_str = ''
peertube_str = ''
libretranslate_str = ''
2023-08-31 22:29:10 +00:00
memorial_str = ''
2021-12-30 23:46:06 +00:00
system_monitor_str = ''
graphics_str = ''
shares_federation_str = ''
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
admin_nickname = get_config_param(base_dir, 'admin')
2021-12-26 14:17:13 +00:00
if is_artist(base_dir, nickname) or \
2021-12-30 23:46:06 +00:00
path.startswith('/users/' + str(admin_nickname) + '/'):
graphics_str = _html_edit_profile_graphic_design(base_dir, translate)
is_admin = False
if admin_nickname:
if path.startswith('/users/' + admin_nickname + '/'):
is_admin = True
twitter_str = \
_html_edit_profile_twitter(base_dir, translate, remove_twitter)
# shared items section
2021-12-30 23:46:06 +00:00
shares_federation_str = \
2022-06-15 12:39:08 +00:00
_html_edit_profile_shared_items(base_dir, translate)
2023-08-31 22:29:10 +00:00
(instance_str, role_assign_str, peertube_str,
libretranslate_str, memorial_str) = \
2021-12-29 21:55:09 +00:00
_html_edit_profile_instance(base_dir, translate,
peertube_instances,
2021-12-30 23:46:06 +00:00
media_instance_str,
blogs_instance_str,
news_instance_str)
2022-09-02 10:52:11 +00:00
if is_admin or is_devops(base_dir, nickname):
2022-09-02 10:29:42 +00:00
system_monitor_str = _html_system_monitor(nickname, translate)
2020-12-24 11:56:17 +00:00
2021-12-30 23:46:06 +00:00
instance_title = get_config_param(base_dir, 'instanceTitle')
2024-12-23 18:23:47 +00:00
preload_images: list[str] = []
2021-12-30 23:46:06 +00:00
edit_profile_form = \
2024-10-22 12:12:16 +00:00
html_header_with_external_style(css_filename, instance_title, None,
preload_images)
2020-11-09 22:44:03 +00:00
# keyboard navigation
2021-12-30 23:46:06 +00:00
user_path_str = '/users/' + nickname
user_timeline_str = '/users/' + nickname + '/' + default_timeline
menu_timeline = \
2021-12-29 21:55:09 +00:00
html_hide_from_screen_reader('🏠') + ' ' + \
2021-02-06 10:46:03 +00:00
translate['Switch to timeline view']
2021-12-30 23:46:06 +00:00
menu_profile = \
2021-12-29 21:55:09 +00:00
html_hide_from_screen_reader('👤') + ' ' + \
2021-02-06 10:46:03 +00:00
translate['Switch to profile view']
2021-12-30 23:46:06 +00:00
nav_links = {
menu_profile: user_path_str,
menu_timeline: user_timeline_str
2021-02-05 17:05:53 +00:00
}
2021-12-30 23:46:06 +00:00
nav_access_keys = {
menu_profile: 'p',
menu_timeline: 't'
2021-04-22 11:51:19 +00:00
}
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2024-02-19 18:31:04 +00:00
html_keyboard_navigation(text_mode_banner, nav_links, nav_access_keys,
None, None, None, False)
# top banner
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
_html_edit_profile_top_banner(base_dir, nickname, domain, domain_full,
2021-12-30 23:46:06 +00:00
default_timeline, banner_file,
path, access_keys, translate)
2020-11-09 22:44:03 +00:00
# main info
2021-12-30 23:46:06 +00:00
edit_profile_form += \
_html_edit_profile_main(base_dir, display_nickname,
pronouns, bio_str,
2021-12-30 23:46:06 +00:00
moved_to, donate_url, website_url,
2022-11-04 21:07:55 +00:00
gemini_link,
2022-02-25 19:57:31 +00:00
blog_address, actor_json, translate,
nickname, domain, max_recent_posts,
featured_hashtags)
# whether to show quote toots
show_quote_toots = False
if os.path.isfile(account_dir + '/.allowQuotes'):
show_quote_toots = True
2023-04-05 17:13:22 +00:00
# whether to show votes
show_vote_posts = True
2023-04-05 19:03:28 +00:00
if os.path.isfile(account_dir + '/.noVotes'):
2023-04-05 17:13:22 +00:00
show_vote_posts = False
# only show replies from followers
show_replies_followers = False
if os.path.isfile(account_dir + '/.repliesFromFollowersOnly'):
2023-06-20 10:48:44 +00:00
show_replies_followers = True
# only show replies from mutuals
show_replies_mutuals = False
if os.path.isfile(account_dir + '/.repliesFromMutualsOnly'):
show_replies_mutuals = True
# don't show follows on profile
hide_follows = False
if os.path.isfile(account_dir + '/.hideFollows'):
hide_follows = True
# is this a premium account?
premium = is_premium_account(base_dir, nickname, domain)
# are boosts of replies permitted in the inbox?
no_reply_boosts_filename = account_dir + '/.noReplyBoosts'
no_reply_boosts = False
if os.path.isfile(no_reply_boosts_filename):
no_reply_boosts = True
# are seen posts permitted in timelines?
no_seen_posts_filename = account_dir + '/.noSeenPosts'
no_seen_posts = False
if os.path.isfile(no_seen_posts_filename):
no_seen_posts = True
2024-07-29 16:23:59 +00:00
# is watermarking enabled?
watermark_enabled_filename = account_dir + '/.watermarkEnabled'
watermark_enabled = False
if os.path.isfile(watermark_enabled_filename):
watermark_enabled = True
2021-03-06 11:57:40 +00:00
# Option checkboxes
2021-12-30 23:46:06 +00:00
edit_profile_form += \
_html_edit_profile_options(is_admin, manually_approves_followers,
reject_spam_actors, is_bot, is_group,
remove_twitter, notify_likes,
notify_reactions, hide_like_button,
hide_reaction_button, translate,
bold_reading, nickname,
min_images_for_accounts,
reverse_sequence, show_quote_toots,
show_vote_posts, hide_follows,
premium, no_seen_posts, watermark_enabled)
# reply controls
edit_profile_form += \
_html_edit_profile_reply_controls(translate, follow_dms,
premium,
show_replies_followers,
show_replies_mutuals,
no_reply_boosts)
2021-03-06 11:57:40 +00:00
# Contact information
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2022-09-08 18:11:48 +00:00
_html_edit_profile_contact_info(email_address,
2021-12-30 23:46:06 +00:00
xmpp_address, matrix_address,
ssb_address, tox_address,
briar_address,
cwtch_address, translate,
youtube, peertube, pixelfed,
2024-08-15 19:58:26 +00:00
discord, music_site_url,
art_site_url)
# notification settings
2022-08-19 10:17:31 +00:00
edit_profile_form += \
_html_edit_notifications(base_dir, nickname, domain, translate)
# Import and export
edit_profile_form += \
_html_edit_profile_import_export(nickname, domain, translate, premium)
# Encryption Keys
2021-12-30 23:46:06 +00:00
edit_profile_form += \
_html_edit_profile_encryption_keys(pgp_fingerprint,
pgp_pub_key, enigma_pub_key,
translate)
2021-03-06 11:55:09 +00:00
# Customize images and banners
2021-12-30 23:46:06 +00:00
edit_profile_form += \
_html_edit_profile_background(news_instance, translate)
2021-03-06 11:55:09 +00:00
# Change password
2021-12-30 23:46:06 +00:00
edit_profile_form += _html_edit_profile_change_password(translate)
2020-11-09 22:44:03 +00:00
2021-07-19 19:40:04 +00:00
# automatic translations
2021-12-30 23:46:06 +00:00
edit_profile_form += libretranslate_str
2021-07-19 19:40:04 +00:00
2023-08-31 22:29:10 +00:00
# memorial accounts
edit_profile_form += memorial_str
# system monitor
2021-12-30 23:46:06 +00:00
edit_profile_form += system_monitor_str
# Filtering and blocking section
2021-12-30 23:46:06 +00:00
reply_interval_hours = \
get_reply_interval_hours(base_dir, nickname, domain,
default_reply_interval_hrs)
edit_profile_form += \
2021-12-29 21:55:09 +00:00
_html_edit_profile_filtering(base_dir, nickname, domain,
2022-03-06 12:56:26 +00:00
user_agents_blocked, crawlers_allowed,
translate, reply_interval_hours,
2023-07-18 14:05:54 +00:00
cw_lists, lists_enabled, buy_sites,
2024-12-15 13:01:26 +00:00
block_military, block_government,
block_bluesky,
2024-12-14 18:02:43 +00:00
block_federated_endpoints)
# git projects section
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
_html_edit_profile_git_projects(base_dir, nickname, domain, translate)
# Skills section
2021-12-30 23:46:06 +00:00
edit_profile_form += \
2021-12-29 21:55:09 +00:00
_html_edit_profile_skills(base_dir, nickname, domain, translate)
2021-12-30 23:46:06 +00:00
edit_profile_form += role_assign_str + peertube_str + graphics_str
edit_profile_form += shares_federation_str + twitter_str + instance_str
2021-03-06 15:18:56 +00:00
# danger zone section
2021-12-30 23:46:06 +00:00
edit_profile_form += _html_edit_profile_danger_zone(translate)
2021-12-30 23:46:06 +00:00
edit_profile_form += ' <div class="container">\n'
edit_profile_form += \
' <center>\n' + \
' <input type="submit" name="submitProfile" value="' + \
2022-05-23 18:48:45 +00:00
translate['Publish'] + '">\n' + \
' </center>\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += ' </div>\n'
2021-12-30 23:46:06 +00:00
edit_profile_form += ' </div>\n'
edit_profile_form += '</form>\n'
edit_profile_form += html_footer()
return edit_profile_form
2020-11-09 22:44:03 +00:00
2021-12-29 21:55:09 +00:00
def _individual_follow_as_html(signing_priv_key_pem: str,
translate: {},
base_dir: str, session,
cached_webfingers: {},
person_cache: {}, domain: str,
2022-05-30 18:33:51 +00:00
follow_url: str,
2021-12-29 21:55:09 +00:00
authorized: bool,
2021-12-30 23:46:06 +00:00
actor_nickname: str,
2021-12-29 21:55:09 +00:00
http_prefix: str,
project_version: str,
dormant: bool,
2023-09-16 13:55:02 +00:00
offline: bool,
2021-12-29 21:55:09 +00:00
debug: bool,
2023-10-25 19:55:40 +00:00
system_language: str,
2024-12-17 13:50:48 +00:00
mitm_servers: [],
2024-12-23 15:39:55 +00:00
buttons: list[str] = []) -> str:
2020-11-09 22:44:03 +00:00
"""An individual follow entry on the profile screen
"""
2022-05-30 18:33:51 +00:00
follow_url_nickname = get_nickname_from_actor(follow_url)
if not follow_url_nickname:
return ''
2022-05-30 18:33:51 +00:00
follow_url_domain, follow_url_port = get_domain_from_actor(follow_url)
2023-01-15 14:33:18 +00:00
if not follow_url_domain:
return ''
2021-12-30 23:46:06 +00:00
follow_url_domain_full = \
get_full_domain(follow_url_domain, follow_url_port)
title_str = '@' + follow_url_nickname + '@' + follow_url_domain_full
2022-05-30 18:33:51 +00:00
avatar_url = \
2022-06-12 12:30:14 +00:00
get_person_avatar_url(base_dir, follow_url, person_cache)
2021-12-30 23:46:06 +00:00
if not avatar_url:
2022-05-30 18:33:51 +00:00
avatar_url = follow_url + '/avatar.png'
2021-12-30 23:46:06 +00:00
2022-05-30 18:33:51 +00:00
display_name = get_display_name(base_dir, follow_url, person_cache)
2021-12-30 23:46:06 +00:00
is_group = False
if not display_name:
2022-05-30 18:33:51 +00:00
# lookup the correct webfinger for the follow_url
2021-12-30 23:46:06 +00:00
follow_url_handle = follow_url_nickname + '@' + follow_url_domain_full
follow_url_wf = \
webfinger_handle(session, follow_url_handle, http_prefix,
2021-12-29 21:55:09 +00:00
cached_webfingers,
domain, __version__, debug, False,
2024-12-17 13:50:48 +00:00
signing_priv_key_pem, mitm_servers)
2021-12-30 23:46:06 +00:00
origin_domain = domain
(_, _, _, _, _, avatar_url2,
display_name, is_group) = get_person_box(signing_priv_key_pem,
origin_domain,
base_dir, session,
follow_url_wf,
person_cache,
project_version,
http_prefix,
follow_url_nickname,
2023-10-25 19:55:40 +00:00
domain, 'outbox', 43036,
2024-12-17 13:50:48 +00:00
system_language,
mitm_servers)
2021-12-30 23:46:06 +00:00
if avatar_url2:
avatar_url = avatar_url2
if display_name:
display_name = \
2021-12-29 21:55:09 +00:00
add_emoji_to_display_name(None, base_dir, http_prefix,
2021-12-30 23:46:06 +00:00
actor_nickname, domain,
2022-07-18 16:18:04 +00:00
display_name, False, translate)
2021-12-30 23:46:06 +00:00
title_str = display_name
2021-01-12 10:23:25 +00:00
2023-09-16 13:55:02 +00:00
if offline:
title_str += ' <b>[' + translate['offline'].upper() + ']</b>'
elif dormant:
title_str += ' 💤'
2020-11-09 22:44:03 +00:00
2021-12-30 23:46:06 +00:00
buttons_str = ''
2020-11-09 22:44:03 +00:00
if authorized:
2021-12-30 23:46:06 +00:00
for btn in buttons:
if btn == 'block':
buttons_str += \
'<a href="/users/' + actor_nickname + \
2022-05-30 18:33:51 +00:00
'?options=' + follow_url + \
2021-12-30 23:46:06 +00:00
';1;' + avatar_url + \
'"><button class="buttonunfollow">' + \
2020-11-09 22:44:03 +00:00
translate['Block'] + '</button></a>\n'
2021-12-30 23:46:06 +00:00
elif btn == 'unfollow':
unfollow_str = 'Unfollow'
if is_group or \
2021-12-26 17:33:24 +00:00
is_group_account(base_dir,
2021-12-30 23:46:06 +00:00
follow_url_nickname, follow_url_domain):
unfollow_str = 'Leave'
buttons_str += \
'<a href="/users/' + actor_nickname + \
2022-05-30 18:33:51 +00:00
'?options=' + follow_url + \
2021-12-30 23:46:06 +00:00
';1;' + avatar_url + \
'"><button class="buttonunfollow">' + \
translate[unfollow_str] + '</button></a>\n'
2022-11-29 22:46:40 +00:00
elif btn == 'moveAccount':
buttons_str += \
'<a href="/users/' + actor_nickname + \
'?options=' + follow_url + \
';1;' + avatar_url + \
'"><button class="buttonunfollow">' + \
translate['Move'] + '</button></a>\n'
2021-12-30 23:46:06 +00:00
result_str = '<div class="container">\n'
result_str += \
'<a href="/users/' + actor_nickname + '?options=' + \
2022-05-30 18:33:51 +00:00
follow_url + ';1;' + avatar_url + '">\n'
2022-03-28 08:47:53 +00:00
result_str += '<p><img loading="lazy" decoding="async" ' + \
'src="' + avatar_url + '" alt=" ">'
2021-12-30 23:46:06 +00:00
result_str += title_str + '</a>' + buttons_str + '</p>\n'
result_str += '</div>\n'
return result_str