epicyon/acceptreject.py

237 lines
9.2 KiB
Python
Raw Normal View History

2020-04-01 18:16:29 +00:00
__filename__ = "acceptreject.py"
__author__ = "Bob Mottram"
__license__ = "AGPL3+"
2022-02-03 13:58:20 +00:00
__version__ = "1.3.0"
2020-04-01 18:16:29 +00:00
__maintainer__ = "Bob Mottram"
2021-09-10 16:14:50 +00:00
__email__ = "bob@libreserver.org"
2020-04-01 18:16:29 +00:00
__status__ = "Production"
2021-06-15 15:08:12 +00:00
__module_group__ = "ActivityPub"
2019-07-02 11:31:26 +00:00
2020-03-01 00:04:27 +00:00
import os
2021-12-26 15:54:46 +00:00
from utils import has_object_string_object
2021-12-26 12:19:00 +00:00
from utils import has_users_path
2021-12-26 12:45:03 +00:00
from utils import get_full_domain
2021-12-27 20:47:05 +00:00
from utils import url_permitted
2021-12-27 19:05:25 +00:00
from utils import get_domain_from_actor
2021-12-27 22:19:18 +00:00
from utils import get_nickname_from_actor
2021-12-27 18:28:26 +00:00
from utils import domain_permitted
2021-12-27 17:08:19 +00:00
from utils import follow_person
2021-12-26 12:02:29 +00:00
from utils import acct_dir
2021-12-26 17:53:07 +00:00
from utils import has_group_type
2021-12-26 10:19:59 +00:00
from utils import local_actor_url
2021-12-26 17:15:04 +00:00
from utils import has_actor
2021-12-26 17:12:07 +00:00
from utils import has_object_stringType
2019-07-02 11:31:26 +00:00
2020-04-01 18:16:29 +00:00
2021-12-25 23:45:30 +00:00
def _create_accept_reject(base_dir: str, federation_list: [],
nickname: str, domain: str, port: int,
toUrl: str, ccUrl: str, http_prefix: str,
objectJson: {}, acceptType: str) -> {}:
2019-07-06 17:00:22 +00:00
"""Accepts or rejects something (eg. a follow request or offer)
2019-07-02 11:31:26 +00:00
Typically toUrl will be https://www.w3.org/ns/activitystreams#Public
2019-07-06 17:00:22 +00:00
and ccUrl 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
2019-07-02 11:31:26 +00:00
"""
2019-07-06 19:24:52 +00:00
if not objectJson.get('actor'):
return None
2021-12-27 20:47:05 +00:00
if not url_permitted(objectJson['actor'], federation_list):
2019-07-02 11:31:26 +00:00
return None
2021-12-26 12:45:03 +00:00
domain = get_full_domain(domain, port)
2019-07-02 11:31:26 +00:00
2021-12-25 23:51:19 +00:00
new_accept = {
2019-08-18 11:07:06 +00:00
"@context": "https://www.w3.org/ns/activitystreams",
2019-07-02 11:31:26 +00:00
'type': acceptType,
2021-12-26 10:19:59 +00:00
'actor': local_actor_url(http_prefix, nickname, domain),
2019-07-02 11:31:26 +00:00
'to': [toUrl],
'cc': [],
2019-07-06 19:24:52 +00:00
'object': objectJson
2019-07-02 11:31:26 +00:00
}
if ccUrl:
2020-04-01 18:16:29 +00:00
if len(ccUrl) > 0:
2021-12-25 23:51:19 +00:00
new_accept['cc'] = [ccUrl]
return new_accept
2019-07-02 11:31:26 +00:00
2020-04-01 18:16:29 +00:00
2021-12-29 21:55:09 +00:00
def create_accept(base_dir: str, federation_list: [],
nickname: str, domain: str, port: int,
toUrl: str, ccUrl: str, http_prefix: str,
objectJson: {}) -> {}:
2021-12-25 23:45:30 +00:00
return _create_accept_reject(base_dir, federation_list,
nickname, domain, port,
toUrl, ccUrl, http_prefix,
objectJson, 'Accept')
2020-04-01 18:16:29 +00:00
2021-12-29 21:55:09 +00:00
def create_reject(base_dir: str, federation_list: [],
nickname: str, domain: str, port: int,
toUrl: str, ccUrl: str, http_prefix: str,
objectJson: {}) -> {}:
2021-12-25 23:45:30 +00:00
return _create_accept_reject(base_dir, federation_list,
nickname, domain, port,
toUrl, ccUrl,
http_prefix, objectJson, 'Reject')
2019-07-06 19:24:52 +00:00
2020-04-01 18:16:29 +00:00
2021-12-26 00:07:44 +00:00
def _accept_follow(base_dir: str, domain: str, message_json: {},
federation_list: [], debug: bool,
curr_domain: str,
onion_domain: str, i2p_domain: str) -> None:
2019-07-20 08:33:18 +00:00
"""Receiving a follow Accept activity
"""
2021-12-26 17:12:07 +00:00
if not has_object_stringType(message_json, debug):
2020-03-22 21:16:02 +00:00
return
2021-12-25 23:51:19 +00:00
if not message_json['object']['type'] == 'Follow':
if not message_json['object']['type'] == 'Join':
return
2019-11-12 14:18:23 +00:00
if debug:
print('DEBUG: receiving Follow activity')
2021-12-25 23:51:19 +00:00
if not message_json['object'].get('actor'):
2019-11-12 14:18:23 +00:00
print('DEBUG: no actor in Follow activity')
2019-07-06 19:24:52 +00:00
return
# no, this isn't a mistake
2021-12-26 15:54:46 +00:00
if not has_object_string_object(message_json, debug):
2020-03-22 21:16:02 +00:00
return
2021-12-25 23:51:19 +00:00
if not message_json.get('to'):
2019-07-07 13:53:12 +00:00
if debug:
print('DEBUG: No "to" parameter in follow Accept')
return
2019-07-06 19:24:52 +00:00
if debug:
2022-02-28 20:05:47 +00:00
print('DEBUG: follow Accept received ' + str(message_json))
2021-12-26 00:07:44 +00:00
this_actor = message_json['object']['actor']
2021-12-27 22:19:18 +00:00
nickname = get_nickname_from_actor(this_actor)
2019-09-02 09:43:43 +00:00
if not nickname:
2021-12-26 00:07:44 +00:00
print('WARN: no nickname found in ' + this_actor)
2019-09-02 09:43:43 +00:00
return
2021-12-29 22:45:26 +00:00
accepted_domain, accepted_port = get_domain_from_actor(this_actor)
if not accepted_domain:
2019-07-06 19:24:52 +00:00
if debug:
2021-12-26 00:07:44 +00:00
print('DEBUG: domain not found in ' + this_actor)
2019-07-06 19:24:52 +00:00
return
if not nickname:
if debug:
2021-12-26 00:07:44 +00:00
print('DEBUG: nickname not found in ' + this_actor)
2019-07-06 19:24:52 +00:00
return
2021-12-29 22:45:26 +00:00
if accepted_port:
if '/' + accepted_domain + ':' + str(accepted_port) + \
2021-12-26 00:07:44 +00:00
'/users/' + nickname not in this_actor:
2019-07-06 19:24:52 +00:00
if debug:
2021-12-29 22:45:26 +00:00
print('Port: ' + str(accepted_port))
print('Expected: /' + accepted_domain + ':' +
str(accepted_port) + '/users/' + nickname)
2021-12-26 00:07:44 +00:00
print('Actual: ' + this_actor)
print('DEBUG: unrecognized actor ' + this_actor)
2019-07-06 19:24:52 +00:00
return
else:
2021-12-29 22:45:26 +00:00
if not '/' + accepted_domain + '/users/' + nickname in this_actor:
2019-07-06 19:24:52 +00:00
if debug:
2021-12-29 22:45:26 +00:00
print('Expected: /' + accepted_domain + '/users/' + nickname)
2021-12-26 00:07:44 +00:00
print('Actual: ' + this_actor)
print('DEBUG: unrecognized actor ' + this_actor)
2020-03-22 21:16:02 +00:00
return
2021-12-26 00:07:44 +00:00
followed_actor = message_json['object']['object']
2021-12-27 19:05:25 +00:00
followed_domain, port = get_domain_from_actor(followed_actor)
2021-12-26 00:07:44 +00:00
if not followed_domain:
2020-04-01 18:16:29 +00:00
print('DEBUG: no domain found within Follow activity object ' +
2021-12-26 00:07:44 +00:00
followed_actor)
2019-07-06 19:24:52 +00:00
return
2021-12-26 00:07:44 +00:00
followed_domain_full = followed_domain
2019-07-06 19:24:52 +00:00
if port:
2021-12-26 00:07:44 +00:00
followed_domain_full = followed_domain + ':' + str(port)
2021-12-27 22:19:18 +00:00
followed_nickname = get_nickname_from_actor(followed_actor)
2021-12-26 00:07:44 +00:00
if not followed_nickname:
2020-04-01 18:16:29 +00:00
print('DEBUG: no nickname found within Follow activity object ' +
2021-12-26 00:07:44 +00:00
followed_actor)
2019-07-06 19:24:52 +00:00
return
2019-07-07 11:53:32 +00:00
# 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
2021-12-29 22:45:26 +00:00
accepted_domain_full = accepted_domain
if accepted_port:
accepted_domain_full = accepted_domain + ':' + str(accepted_port)
2019-07-16 22:57:45 +00:00
# has this person already been unfollowed?
2021-12-26 00:07:44 +00:00
unfollowed_filename = \
2021-12-26 12:02:29 +00:00
acct_dir(base_dir, nickname, accepted_domain_full) + '/unfollowed.txt'
2021-12-26 00:07:44 +00:00
if os.path.isfile(unfollowed_filename):
if followed_nickname + '@' + followed_domain_full in \
open(unfollowed_filename).read():
if debug:
2020-04-01 18:16:29 +00:00
print('DEBUG: follow accept arrived for ' +
2021-12-26 00:07:44 +00:00
nickname + '@' + accepted_domain_full +
' from ' +
followed_nickname + '@' + followed_domain_full +
' but they have been unfollowed')
2020-03-22 21:16:02 +00:00
return
2021-07-30 16:06:34 +00:00
# does the url path indicate that this is a group actor
2021-12-26 17:53:07 +00:00
group_account = has_group_type(base_dir, followed_actor, None, debug)
2021-07-31 11:56:28 +00:00
if debug:
2021-12-26 00:07:44 +00:00
print('Accepted follow is a group: ' + str(group_account) +
' ' + followed_actor + ' ' + base_dir)
2021-07-30 16:06:34 +00:00
2021-12-27 17:08:19 +00:00
if follow_person(base_dir,
nickname, accepted_domain_full,
followed_nickname, followed_domain_full,
federation_list, debug, group_account):
2019-07-06 19:24:52 +00:00
if debug:
2021-12-26 00:07:44 +00:00
print('DEBUG: ' + nickname + '@' + accepted_domain_full +
' followed ' +
followed_nickname + '@' + followed_domain_full)
2019-07-06 19:24:52 +00:00
else:
if debug:
2020-04-01 18:16:29 +00:00
print('DEBUG: Unable to create follow - ' +
2021-12-29 22:45:26 +00:00
nickname + '@' + accepted_domain + ' -> ' +
2021-12-26 00:07:44 +00:00
followed_nickname + '@' + followed_domain)
2020-04-01 18:16:29 +00:00
2021-12-29 21:55:09 +00:00
def receive_accept_reject(session, base_dir: str,
http_prefix: str, domain: str, port: int,
2021-12-29 22:45:26 +00:00
send_threads: [], post_log: [],
2021-12-29 21:55:09 +00:00
cached_webfingers: {},
person_cache: {}, message_json: {},
federation_list: [],
debug: bool,
curr_domain: str,
onion_domain: str, i2p_domain: str) -> bool:
2019-07-06 15:17:21 +00:00
"""Receives an Accept or Reject within the POST section of HTTPServer
"""
2021-12-25 23:51:19 +00:00
if message_json['type'] != 'Accept' and message_json['type'] != 'Reject':
2019-07-06 15:17:21 +00:00
return False
2021-12-26 17:15:04 +00:00
if not has_actor(message_json, debug):
2019-07-06 15:17:21 +00:00
return False
2021-12-26 12:19:00 +00:00
if not has_users_path(message_json['actor']):
2019-07-06 15:17:21 +00:00
if debug:
2020-04-01 18:16:29 +00:00
print('DEBUG: "users" or "profile" missing from actor in ' +
2021-12-25 23:51:19 +00:00
message_json['type'] + '. Assuming single user instance.')
2021-12-27 19:05:25 +00:00
domain, _ = get_domain_from_actor(message_json['actor'])
2021-12-27 18:28:26 +00:00
if not domain_permitted(domain, federation_list):
2019-07-06 15:17:21 +00:00
if debug:
2021-12-25 23:51:19 +00:00
print('DEBUG: ' + message_json['type'] +
2020-04-01 18:16:29 +00:00
' from domain not permitted - ' + domain)
2019-07-06 15:17:21 +00:00
return False
2021-12-27 22:19:18 +00:00
nickname = get_nickname_from_actor(message_json['actor'])
2019-07-06 15:17:21 +00:00
if not nickname:
# single user instance
2020-04-01 18:16:29 +00:00
nickname = 'dev'
2019-07-06 15:17:21 +00:00
if debug:
2021-12-25 23:51:19 +00:00
print('DEBUG: ' + message_json['type'] +
2020-04-01 18:16:29 +00:00
' does not contain a nickname. ' +
2020-03-30 19:09:45 +00:00
'Assuming single user instance.')
2019-07-20 08:33:18 +00:00
# receive follow accept
_accept_follow(base_dir, domain, message_json, federation_list, debug,
curr_domain, onion_domain, i2p_domain)
2019-07-06 15:17:21 +00:00
if debug:
2021-12-25 23:51:19 +00:00
print('DEBUG: Uh, ' + message_json['type'] + ', I guess')
2019-07-06 15:17:21 +00:00
return True