__filename__ = "manualapprove.py" __author__ = "Bob Mottram" __license__ = "AGPL3+" __version__ = "1.2.0" __maintainer__ = "Bob Mottram" __email__ = "bob@libreserver.org" __status__ = "Production" __module_group__ = "ActivityPub" import os from follow import followedAccountAccepts from follow import followedAccountRejects from follow import removeFromFollowRequests from utils import loadJson from utils import removeDomainPort from utils import getPortFromDomain from utils import get_user_paths from utils import acct_dir from threads import threadWithTrace def manualDenyFollowRequest(session, base_dir: str, http_prefix: str, nickname: str, domain: str, port: int, denyHandle: str, federation_list: [], send_threads: [], postLog: [], cached_webfingers: {}, person_cache: {}, debug: bool, project_version: str, signing_priv_key_pem: str) -> None: """Manually deny a follow request """ accountsDir = acct_dir(base_dir, nickname, domain) # has this handle already been rejected? rejectedFollowsFilename = accountsDir + '/followrejects.txt' if os.path.isfile(rejectedFollowsFilename): if denyHandle in open(rejectedFollowsFilename).read(): removeFromFollowRequests(base_dir, nickname, domain, denyHandle, debug) print(denyHandle + ' has already been rejected as a follower of ' + nickname) return removeFromFollowRequests(base_dir, nickname, domain, denyHandle, debug) # Store rejected follows try: with open(rejectedFollowsFilename, 'a+') as rejectsFile: rejectsFile.write(denyHandle + '\n') except OSError: print('EX: unable to append ' + rejectedFollowsFilename) denyNickname = denyHandle.split('@')[0] denyDomain = \ denyHandle.split('@')[1].replace('\n', '').replace('\r', '') denyPort = port if ':' in denyDomain: denyPort = getPortFromDomain(denyDomain) denyDomain = removeDomainPort(denyDomain) followedAccountRejects(session, base_dir, http_prefix, nickname, domain, port, denyNickname, denyDomain, denyPort, federation_list, send_threads, postLog, cached_webfingers, person_cache, debug, project_version, signing_priv_key_pem) print('Follow request from ' + denyHandle + ' was denied.') def manualDenyFollowRequestThread(session, base_dir: str, http_prefix: str, nickname: str, domain: str, port: int, denyHandle: str, federation_list: [], send_threads: [], postLog: [], cached_webfingers: {}, person_cache: {}, debug: bool, project_version: str, signing_priv_key_pem: str) -> None: """Manually deny a follow request, within a thread so that the user interface doesn't lag """ thr = \ threadWithTrace(target=manualDenyFollowRequest, args=(session, base_dir, http_prefix, nickname, domain, port, denyHandle, federation_list, send_threads, postLog, cached_webfingers, person_cache, debug, project_version, signing_priv_key_pem), daemon=True) thr.start() send_threads.append(thr) def _approveFollowerHandle(accountDir: str, approveHandle: str) -> None: """ Record manually approved handles so that if they unfollow and then re-follow later then they don't need to be manually approved again """ approvedFilename = accountDir + '/approved.txt' if os.path.isfile(approvedFilename): if approveHandle not in open(approvedFilename).read(): try: with open(approvedFilename, 'a+') as approvedFile: approvedFile.write(approveHandle + '\n') except OSError: print('EX: unable to append ' + approvedFilename) else: try: with open(approvedFilename, 'w+') as approvedFile: approvedFile.write(approveHandle + '\n') except OSError: print('EX: unable to write ' + approvedFilename) def manualApproveFollowRequest(session, base_dir: str, http_prefix: str, nickname: str, domain: str, port: int, approveHandle: str, federation_list: [], send_threads: [], postLog: [], cached_webfingers: {}, person_cache: {}, debug: bool, project_version: str, signing_priv_key_pem: str) -> None: """Manually approve a follow request """ handle = nickname + '@' + domain print('Manual follow accept: ' + handle + ' approving follow request from ' + approveHandle) accountDir = base_dir + '/accounts/' + handle approveFollowsFilename = accountDir + '/followrequests.txt' if not os.path.isfile(approveFollowsFilename): print('Manual follow accept: follow requests file ' + approveFollowsFilename + ' not found') return # is the handle in the requests file? approveFollowsStr = '' with open(approveFollowsFilename, 'r') as fpFollowers: approveFollowsStr = fpFollowers.read() exists = False approveHandleFull = approveHandle if approveHandle in approveFollowsStr: exists = True elif '@' in approveHandle: group_account = False if approveHandle.startswith('!'): group_account = True reqNick = approveHandle.split('@')[0].replace('!', '') reqDomain = approveHandle.split('@')[1].strip() reqPrefix = http_prefix + '://' + reqDomain paths = get_user_paths() for userPath in paths: if reqPrefix + userPath + reqNick in approveFollowsStr: exists = True approveHandleFull = reqPrefix + userPath + reqNick if group_account: approveHandleFull = '!' + approveHandleFull break if not exists: print('Manual follow accept: ' + approveHandleFull + ' not in requests file "' + approveFollowsStr.replace('\n', ' ') + '" ' + approveFollowsFilename) return with open(approveFollowsFilename + '.new', 'w+') as approvefilenew: updateApprovedFollowers = False followActivityfilename = None with open(approveFollowsFilename, 'r') as approvefile: for handleOfFollowRequester in approvefile: # is this the approved follow? if handleOfFollowRequester.startswith(approveHandleFull): handleOfFollowRequester = \ handleOfFollowRequester.replace('\n', '') handleOfFollowRequester = \ handleOfFollowRequester.replace('\r', '') port2 = port if ':' in handleOfFollowRequester: port2 = getPortFromDomain(handleOfFollowRequester) requestsDir = accountDir + '/requests' followActivityfilename = \ requestsDir + '/' + handleOfFollowRequester + '.follow' if os.path.isfile(followActivityfilename): followJson = loadJson(followActivityfilename) if followJson: approveNickname = approveHandle.split('@')[0] approveDomain = approveHandle.split('@')[1] approveDomain = \ approveDomain.replace('\n', '') approveDomain = \ approveDomain.replace('\r', '') approvePort = port2 if ':' in approveDomain: approvePort = getPortFromDomain(approveDomain) approveDomain = removeDomainPort(approveDomain) print('Manual follow accept: Sending Accept for ' + handle + ' follow request from ' + approveNickname + '@' + approveDomain) followedAccountAccepts(session, base_dir, http_prefix, nickname, domain, port, approveNickname, approveDomain, approvePort, followJson['actor'], federation_list, followJson, send_threads, postLog, cached_webfingers, person_cache, debug, project_version, False, signing_priv_key_pem) updateApprovedFollowers = True else: # this isn't the approved follow so it will remain # in the requests file approvefilenew.write(handleOfFollowRequester) followersFilename = accountDir + '/followers.txt' if updateApprovedFollowers: # update the followers print('Manual follow accept: updating ' + followersFilename) if os.path.isfile(followersFilename): if approveHandleFull not in open(followersFilename).read(): try: with open(followersFilename, 'r+') as followersFile: content = followersFile.read() if approveHandleFull + '\n' not in content: followersFile.seek(0, 0) followersFile.write(approveHandleFull + '\n' + content) except Exception as ex: print('WARN: Manual follow accept. ' + 'Failed to write entry to followers file ' + str(ex)) else: print('WARN: Manual follow accept: ' + approveHandleFull + ' already exists in ' + followersFilename) else: print('Manual follow accept: first follower accepted for ' + handle + ' is ' + approveHandleFull) try: with open(followersFilename, 'w+') as followersFile: followersFile.write(approveHandleFull + '\n') except OSError: print('EX: unable to write ' + followersFilename) # only update the follow requests file if the follow is confirmed to be # in followers.txt if approveHandleFull in open(followersFilename).read(): # mark this handle as approved for following _approveFollowerHandle(accountDir, approveHandle) # update the follow requests with the handles not yet approved os.rename(approveFollowsFilename + '.new', approveFollowsFilename) # remove the .follow file if followActivityfilename: if os.path.isfile(followActivityfilename): try: os.remove(followActivityfilename) except OSError: print('EX: manualApproveFollowRequest unable to delete ' + followActivityfilename) else: try: os.remove(approveFollowsFilename + '.new') except OSError: print('EX: manualApproveFollowRequest unable to delete ' + approveFollowsFilename + '.new') def manualApproveFollowRequestThread(session, base_dir: str, http_prefix: str, nickname: str, domain: str, port: int, approveHandle: str, federation_list: [], send_threads: [], postLog: [], cached_webfingers: {}, person_cache: {}, debug: bool, project_version: str, signing_priv_key_pem: str) -> None: """Manually approve a follow request, in a thread so as not to cause the UI to lag """ thr = \ threadWithTrace(target=manualApproveFollowRequest, args=(session, base_dir, http_prefix, nickname, domain, port, approveHandle, federation_list, send_threads, postLog, cached_webfingers, person_cache, debug, project_version, signing_priv_key_pem), daemon=True) thr.start() send_threads.append(thr)