mirror of https://gitlab.com/bashrc2/epicyon
				
				
				
			
		
			
				
	
	
		
			462 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Python
		
	
	
			
		
		
	
	
			462 lines
		
	
	
		
			18 KiB
		
	
	
	
		
			Python
		
	
	
| """ ActivityPub Accept or Reject json """
 | |
| 
 | |
| __filename__ = "acceptreject.py"
 | |
| __author__ = "Bob Mottram"
 | |
| __license__ = "AGPL3+"
 | |
| __version__ = "1.6.0"
 | |
| __maintainer__ = "Bob Mottram"
 | |
| __email__ = "bob@libreserver.org"
 | |
| __status__ = "Production"
 | |
| __module_group__ = "ActivityPub"
 | |
| 
 | |
| import os
 | |
| import time
 | |
| from posts import send_signed_json
 | |
| from flags import has_group_type
 | |
| from flags import url_permitted
 | |
| from utils import get_status_number
 | |
| from utils import get_attributed_to
 | |
| from utils import get_user_paths
 | |
| from utils import text_in_file
 | |
| from utils import has_object_string_object
 | |
| from utils import has_users_path
 | |
| from utils import get_full_domain
 | |
| from utils import get_domain_from_actor
 | |
| from utils import get_nickname_from_actor
 | |
| from utils import domain_permitted
 | |
| from utils import follow_person
 | |
| from utils import acct_dir
 | |
| from utils import local_actor_url
 | |
| from utils import has_actor
 | |
| from utils import has_object_string_type
 | |
| from utils import get_actor_from_post
 | |
| 
 | |
| 
 | |
| def _create_quote_accept_reject(receiving_actor: str,
 | |
|                                 sending_actor: str,
 | |
|                                 federation_list: [],
 | |
|                                 debug: bool,
 | |
|                                 quote_request_id: str,
 | |
|                                 quote_request_object: str,
 | |
|                                 quote_request_instrument: str,
 | |
|                                 accept_type: str) -> {}:
 | |
|     """Creates an Accept or Reject response to QuoteRequest
 | |
|     https://codeberg.org/fediverse/fep/src/branch/main/fep/044f/fep-044f.md
 | |
|     """
 | |
|     if not receiving_actor or \
 | |
|        not sending_actor or \
 | |
|        not quote_request_id or \
 | |
|        not quote_request_object or \
 | |
|        not quote_request_instrument:
 | |
|         return None
 | |
|     if not url_permitted(sending_actor, federation_list):
 | |
|         return None
 | |
| 
 | |
|     status_number, _ = get_status_number()
 | |
| 
 | |
|     new_accept = {
 | |
|         "@context": [
 | |
|             "https://www.w3.org/ns/activitystreams",
 | |
|             {
 | |
|                 "toot": "http://joinmastodon.org/ns#",
 | |
|                 "Quote": "toot:QuoteRequest"
 | |
|             }
 | |
|         ],
 | |
|         "type": accept_type,
 | |
|         "to": [sending_actor],
 | |
|         "id": receiving_actor + "/statuses/" + status_number,
 | |
|         "actor": receiving_actor,
 | |
|         "object": {
 | |
|             "type": "QuoteRequest",
 | |
|             "id": quote_request_id,
 | |
|             "actor": sending_actor,
 | |
|             "object": "https://example.com/users/alice/statuses/1",
 | |
|             "instrument": "https://example.org/users/bob/statuses/1"
 | |
|         }
 | |
|     }
 | |
|     if debug:
 | |
|         print('REJECT: QuoteRequest ' + str(new_accept))
 | |
|     return new_accept
 | |
| 
 | |
| 
 | |
| def _create_accept_reject(federation_list: [],
 | |
|                           nickname: str, domain: str, port: int,
 | |
|                           to_url: str, cc_url: str, http_prefix: str,
 | |
|                           object_json: {}, accept_type: str) -> {}:
 | |
|     """Accepts or rejects something (eg. a follow request or offer)
 | |
|     Typically to_url will be https://www.w3.org/ns/activitystreams#Public
 | |
|     and cc_url might be a specific person favorited or repeated and
 | |
|     the followers url objectUrl is typically the url of the message,
 | |
|     corresponding to url or atomUri in createPostBase
 | |
|     """
 | |
|     if not object_json.get('actor'):
 | |
|         return None
 | |
| 
 | |
|     actor_url = get_actor_from_post(object_json)
 | |
|     if not url_permitted(actor_url, federation_list):
 | |
|         return None
 | |
| 
 | |
|     domain = get_full_domain(domain, port)
 | |
| 
 | |
|     new_accept = {
 | |
|         "@context": [
 | |
|             'https://www.w3.org/ns/activitystreams',
 | |
|             'https://w3id.org/security/v1'
 | |
|         ],
 | |
|         'type': accept_type,
 | |
|         'actor': local_actor_url(http_prefix, nickname, domain),
 | |
|         'to': [to_url],
 | |
|         'cc': [],
 | |
|         'object': object_json
 | |
|     }
 | |
|     if cc_url:
 | |
|         new_accept['cc'] = [cc_url]
 | |
|     return new_accept
 | |
| 
 | |
| 
 | |
| def create_accept(federation_list: [],
 | |
|                   nickname: str, domain: str, port: int,
 | |
|                   to_url: str, cc_url: str, http_prefix: str,
 | |
|                   object_json: {}) -> {}:
 | |
|     """ Create json for ActivityPub Accept """
 | |
|     return _create_accept_reject(federation_list,
 | |
|                                  nickname, domain, port,
 | |
|                                  to_url, cc_url, http_prefix,
 | |
|                                  object_json, 'Accept')
 | |
| 
 | |
| 
 | |
| def create_reject(federation_list: [],
 | |
|                   nickname: str, domain: str, port: int,
 | |
|                   to_url: str, cc_url: str, http_prefix: str,
 | |
|                   object_json: {}) -> {}:
 | |
|     """ Create json for ActivityPub Reject """
 | |
|     return _create_accept_reject(federation_list,
 | |
|                                  nickname, domain, port,
 | |
|                                  to_url, cc_url,
 | |
|                                  http_prefix, object_json, 'Reject')
 | |
| 
 | |
| 
 | |
| def _reject_quote_request(message_json: {}, domain_full: str,
 | |
|                           federation_list: [],
 | |
|                           debug: bool,
 | |
|                           session, session_onion, session_i2p,
 | |
|                           base_dir: str,
 | |
|                           http_prefix: str,
 | |
|                           send_threads: [], post_log: [],
 | |
|                           cached_webfingers: {},
 | |
|                           person_cache: {}, project_version: str,
 | |
|                           signing_priv_key_pem: str,
 | |
|                           onion_domain: str, i2p_domain: str,
 | |
|                           extra_headers: {},
 | |
|                           sites_unavailable: {},
 | |
|                           system_language: str,
 | |
|                           mitm_servers: []) -> bool:
 | |
|     """ Rejects a QuoteRequest
 | |
|     https://codeberg.org/fediverse/fep/src/branch/main/fep/044f/fep-044f.md
 | |
|     """
 | |
|     sending_actor = None
 | |
|     receiving_actor = None
 | |
|     quote_request_id = None
 | |
|     quote_request_object = None
 | |
|     quote_request_instrument = None
 | |
|     if message_json.get('actor'):
 | |
|         sending_actor = message_json['actor']
 | |
|     elif message_json.get('instrument'):
 | |
|         if isinstance(message_json['instrument'], dict):
 | |
|             instrument_dict = message_json['instrument']
 | |
|             if instrument_dict.get('attributedTo'):
 | |
|                 sending_actor = \
 | |
|                     get_attributed_to(instrument_dict['attributedTo'])
 | |
|             if instrument_dict.get('to'):
 | |
|                 if isinstance(instrument_dict['to'], str):
 | |
|                     receiving_actor = instrument_dict['to']
 | |
|                 elif isinstance(instrument_dict['to'], list):
 | |
|                     for receiver in instrument_dict['to']:
 | |
|                         if '#Public' not in receiver and \
 | |
|                            '://' + domain_full + '/' in receiver:
 | |
|                             receiving_actor = receiver
 | |
|                             break
 | |
|             if instrument_dict.get('id'):
 | |
|                 quote_request_instrument = instrument_dict['id']
 | |
|             if instrument_dict.get('object'):
 | |
|                 quote_request_object = instrument_dict['object']
 | |
|     if message_json.get('id'):
 | |
|         quote_request_id = message_json['id']
 | |
|     if not sending_actor:
 | |
|         return False
 | |
|     if receiving_actor:
 | |
|         quote_request_object = receiving_actor
 | |
|     reject_json = \
 | |
|         _create_quote_accept_reject(receiving_actor,
 | |
|                                     sending_actor,
 | |
|                                     federation_list,
 | |
|                                     debug,
 | |
|                                     quote_request_id,
 | |
|                                     quote_request_object,
 | |
|                                     quote_request_instrument,
 | |
|                                     'Reject')
 | |
|     if reject_json:
 | |
|         print('REJECT: QuoteRequest from ' + sending_actor)
 | |
|         nickname = get_nickname_from_actor(receiving_actor)
 | |
|         domain, from_port = get_domain_from_actor(receiving_actor)
 | |
|         nickname_to_follow = get_nickname_from_actor(sending_actor)
 | |
|         domain_to_follow, port = get_domain_from_actor(sending_actor)
 | |
|         group_account = \
 | |
|             has_group_type(base_dir, receiving_actor, person_cache)
 | |
|         if nickname and domain and \
 | |
|            nickname_to_follow and domain_to_follow:
 | |
|             if debug:
 | |
|                 print('REJECT: QuoteRequest sending reject ' +
 | |
|                       str(reject_json))
 | |
| 
 | |
|             curr_session = session
 | |
|             curr_domain = domain
 | |
|             curr_port = from_port
 | |
|             curr_http_prefix = http_prefix
 | |
|             if onion_domain and \
 | |
|                not curr_domain.endswith('.onion') and \
 | |
|                domain_to_follow.endswith('.onion'):
 | |
|                 curr_session = session_onion
 | |
|                 curr_http_prefix = 'http'
 | |
|                 curr_domain = onion_domain
 | |
|                 curr_port = 80
 | |
|                 port = 80
 | |
|                 if debug:
 | |
|                     print('Domain switched from ' + domain +
 | |
|                           ' to ' + curr_domain)
 | |
|             elif (i2p_domain and
 | |
|                   not curr_domain.endswith('.i2p') and
 | |
|                   domain_to_follow.endswith('.i2p')):
 | |
|                 curr_session = session_i2p
 | |
|                 curr_http_prefix = 'http'
 | |
|                 curr_domain = i2p_domain
 | |
|                 curr_port = 80
 | |
|                 port = 80
 | |
|                 if debug:
 | |
|                     print('Domain switched from ' + domain +
 | |
|                           ' to ' + curr_domain)
 | |
| 
 | |
|             client_to_server = False
 | |
|             send_signed_json(reject_json, curr_session, base_dir,
 | |
|                              nickname_to_follow, domain_to_follow, port,
 | |
|                              nickname, domain, curr_port,
 | |
|                              curr_http_prefix, client_to_server,
 | |
|                              federation_list,
 | |
|                              send_threads, post_log, cached_webfingers,
 | |
|                              person_cache, debug, project_version, None,
 | |
|                              group_account, signing_priv_key_pem,
 | |
|                              726235284, curr_domain, onion_domain, i2p_domain,
 | |
|                              extra_headers, sites_unavailable,
 | |
|                              system_language, mitm_servers)
 | |
|         return True
 | |
|     return False
 | |
| 
 | |
| 
 | |
| def _accept_follow(base_dir: str, message_json: {},
 | |
|                    federation_list: [], debug: bool,
 | |
|                    curr_domain: str,
 | |
|                    onion_domain: str, i2p_domain: str) -> None:
 | |
|     """ Receiving an ActivityPub follow Accept activity
 | |
|     Your follow was accepted
 | |
|     """
 | |
|     if not has_object_string_type(message_json, debug):
 | |
|         return
 | |
|     if message_json['object']['type'] not in ('Follow', 'Join'):
 | |
|         return
 | |
|     if debug:
 | |
|         print('DEBUG: receiving Follow activity')
 | |
|     if not message_json['object'].get('actor'):
 | |
|         print('DEBUG: no actor in Follow activity')
 | |
|         return
 | |
|     # no, this isn't a mistake
 | |
|     if not has_object_string_object(message_json, debug):
 | |
|         return
 | |
|     if not message_json.get('to'):
 | |
|         if debug:
 | |
|             print('DEBUG: No "to" parameter in follow Accept')
 | |
|         return
 | |
|     if debug:
 | |
|         print('DEBUG: follow Accept received ' + str(message_json))
 | |
|     this_actor = get_actor_from_post(message_json['object'])
 | |
|     nickname = get_nickname_from_actor(this_actor)
 | |
|     if not nickname:
 | |
|         print('WARN: no nickname found in ' + this_actor)
 | |
|         return
 | |
|     accepted_domain, accepted_port = get_domain_from_actor(this_actor)
 | |
|     if not accepted_domain:
 | |
|         if debug:
 | |
|             print('DEBUG: domain not found in ' + this_actor)
 | |
|         return
 | |
|     if not nickname:
 | |
|         if debug:
 | |
|             print('DEBUG: nickname not found in ' + this_actor)
 | |
|         return
 | |
|     if accepted_port:
 | |
|         if '/' + accepted_domain + ':' + str(accepted_port) + \
 | |
|            '/users/' + nickname not in this_actor:
 | |
|             if debug:
 | |
|                 print('Port: ' + str(accepted_port))
 | |
|                 print('Expected: /' + accepted_domain + ':' +
 | |
|                       str(accepted_port) + '/users/' + nickname)
 | |
|                 print('Actual:   ' + this_actor)
 | |
|                 print('DEBUG: unrecognized actor ' + this_actor)
 | |
|             return
 | |
|     else:
 | |
|         actor_found = False
 | |
|         users_list = get_user_paths()
 | |
|         for users_str in users_list:
 | |
|             if '/' + accepted_domain + users_str + nickname in this_actor:
 | |
|                 actor_found = True
 | |
|                 break
 | |
| 
 | |
|         if not actor_found:
 | |
|             if debug:
 | |
|                 print('Expected: /' + accepted_domain + '/users/' + nickname)
 | |
|                 print('Actual:   ' + this_actor)
 | |
|                 print('DEBUG: unrecognized actor ' + this_actor)
 | |
|             return
 | |
|     followed_actor = message_json['object']['object']
 | |
|     followed_domain, port = get_domain_from_actor(followed_actor)
 | |
|     if not followed_domain:
 | |
|         print('DEBUG: no domain found within Follow activity object ' +
 | |
|               followed_actor)
 | |
|         return
 | |
|     followed_domain_full = followed_domain
 | |
|     if port:
 | |
|         followed_domain_full = followed_domain + ':' + str(port)
 | |
|     followed_nickname = get_nickname_from_actor(followed_actor)
 | |
|     if not followed_nickname:
 | |
|         print('DEBUG: no nickname found within Follow activity object ' +
 | |
|               followed_actor)
 | |
|         return
 | |
| 
 | |
|     # convert from onion/i2p to clearnet accepted domain
 | |
|     if onion_domain:
 | |
|         if accepted_domain.endswith('.onion') and \
 | |
|            not curr_domain.endswith('.onion'):
 | |
|             accepted_domain = curr_domain
 | |
|     if i2p_domain:
 | |
|         if accepted_domain.endswith('.i2p') and \
 | |
|            not curr_domain.endswith('.i2p'):
 | |
|             accepted_domain = curr_domain
 | |
| 
 | |
|     accepted_domain_full = accepted_domain
 | |
|     if accepted_port:
 | |
|         accepted_domain_full = accepted_domain + ':' + str(accepted_port)
 | |
| 
 | |
|     # has this person already been unfollowed?
 | |
|     unfollowed_filename = \
 | |
|         acct_dir(base_dir, nickname, accepted_domain_full) + '/unfollowed.txt'
 | |
|     if os.path.isfile(unfollowed_filename):
 | |
|         if text_in_file(followed_nickname + '@' + followed_domain_full,
 | |
|                         unfollowed_filename):
 | |
|             if debug:
 | |
|                 print('DEBUG: follow accept arrived for ' +
 | |
|                       nickname + '@' + accepted_domain_full +
 | |
|                       ' from ' +
 | |
|                       followed_nickname + '@' + followed_domain_full +
 | |
|                       ' but they have been unfollowed')
 | |
|             return
 | |
| 
 | |
|     # does the url path indicate that this is a group actor
 | |
|     group_account = has_group_type(base_dir, followed_actor, None, debug)
 | |
|     if debug:
 | |
|         print('Accepted follow is a group: ' + str(group_account) +
 | |
|               ' ' + followed_actor + ' ' + base_dir)
 | |
| 
 | |
|     if follow_person(base_dir,
 | |
|                      nickname, accepted_domain_full,
 | |
|                      followed_nickname, followed_domain_full,
 | |
|                      federation_list, debug, group_account,
 | |
|                      'following.txt'):
 | |
|         if debug:
 | |
|             print('DEBUG: ' + nickname + '@' + accepted_domain_full +
 | |
|                   ' followed ' +
 | |
|                   followed_nickname + '@' + followed_domain_full)
 | |
|     else:
 | |
|         if debug:
 | |
|             print('DEBUG: Unable to create follow - ' +
 | |
|                   nickname + '@' + accepted_domain + ' -> ' +
 | |
|                   followed_nickname + '@' + followed_domain)
 | |
| 
 | |
| 
 | |
| def receive_accept_reject(base_dir: str, domain: str, message_json: {},
 | |
|                           federation_list: [], debug: bool, curr_domain: str,
 | |
|                           onion_domain: str, i2p_domain: str) -> bool:
 | |
|     """Receives an Accept or Reject within the POST section of HTTPServer
 | |
|     """
 | |
|     if message_json['type'] not in ('Accept', 'Reject'):
 | |
|         return False
 | |
|     if not has_actor(message_json, debug):
 | |
|         return False
 | |
|     actor_url = get_actor_from_post(message_json)
 | |
|     if not has_users_path(actor_url):
 | |
|         if debug:
 | |
|             print('DEBUG: "users" or "profile" missing from actor in ' +
 | |
|                   message_json['type'] + '. Assuming single user instance.')
 | |
|     domain, _ = get_domain_from_actor(actor_url)
 | |
|     if not domain_permitted(domain, federation_list):
 | |
|         if debug:
 | |
|             print('DEBUG: ' + message_json['type'] +
 | |
|                   ' from domain not permitted - ' + domain)
 | |
|         return False
 | |
|     nickname = get_nickname_from_actor(actor_url)
 | |
|     if not nickname:
 | |
|         # single user instance
 | |
|         nickname = 'dev'
 | |
|         if debug:
 | |
|             print('DEBUG: ' + message_json['type'] +
 | |
|                   ' does not contain a nickname. ' +
 | |
|                   'Assuming single user instance.')
 | |
|     # receive follow accept
 | |
|     _accept_follow(base_dir, message_json, federation_list, debug,
 | |
|                    curr_domain, onion_domain, i2p_domain)
 | |
|     if debug:
 | |
|         print('DEBUG: Uh, ' + message_json['type'] + ', I guess')
 | |
|     return True
 | |
| 
 | |
| 
 | |
| def receive_quote_request(message_json: {}, federation_list: [],
 | |
|                           debug: bool,
 | |
|                           domain_full: str,
 | |
|                           session, session_onion, session_i2p,
 | |
|                           base_dir: str,
 | |
|                           http_prefix: str,
 | |
|                           send_threads: [], post_log: [],
 | |
|                           cached_webfingers: {},
 | |
|                           person_cache: {}, project_version: str,
 | |
|                           signing_priv_key_pem: str,
 | |
|                           onion_domain: str,
 | |
|                           i2p_domain: str,
 | |
|                           extra_headers: {},
 | |
|                           sites_unavailable: {},
 | |
|                           system_language: str,
 | |
|                           mitm_servers: [],
 | |
|                           last_quote_request) -> bool:
 | |
|     """Receives a QuoteRequest within the POST section of HTTPServer
 | |
|     https://codeberg.org/fediverse/fep/src/branch/main/fep/044f/fep-044f.md
 | |
|     """
 | |
|     if message_json['type'] != 'QuoteRequest':
 | |
|         return False
 | |
|     curr_time = int(time.time())
 | |
|     seconds_since_last_quote_request = curr_time - last_quote_request
 | |
|     if seconds_since_last_quote_request < 30:
 | |
|         # don't handle quote requests too often
 | |
|         return True
 | |
|     _reject_quote_request(message_json, domain_full,
 | |
|                           federation_list, debug,
 | |
|                           session, session_onion, session_i2p, base_dir,
 | |
|                           http_prefix,
 | |
|                           send_threads, post_log,
 | |
|                           cached_webfingers,
 | |
|                           person_cache, project_version,
 | |
|                           signing_priv_key_pem,
 | |
|                           onion_domain,
 | |
|                           i2p_domain,
 | |
|                           extra_headers,
 | |
|                           sites_unavailable,
 | |
|                           system_language,
 | |
|                           mitm_servers)
 | |
| 
 | |
|     return True
 |