epicyon/follow.py

411 lines
17 KiB
Python
Raw Normal View History

2019-06-29 18:23:13 +00:00
__filename__ = "follow.py"
__author__ = "Bob Mottram"
__license__ = "AGPL3+"
__version__ = "0.0.1"
__maintainer__ = "Bob Mottram"
__email__ = "bob@freedombone.net"
__status__ = "Production"
import json
2019-07-08 17:15:55 +00:00
import commentjson
2019-06-29 18:23:13 +00:00
from pprint import pprint
import os
import sys
2019-07-03 09:40:27 +00:00
from person import validNickname
2019-07-02 10:39:55 +00:00
from utils import domainPermitted
2019-07-06 15:17:21 +00:00
from utils import getDomainFromActor
from utils import getNicknameFromActor
2019-07-06 19:24:52 +00:00
from utils import getStatusNumber
from utils import followPerson
2019-07-05 18:57:19 +00:00
from posts import sendSignedJson
2019-07-06 13:49:25 +00:00
from acceptreject import createAccept
2019-06-29 18:23:13 +00:00
2019-07-06 17:00:22 +00:00
def getFollowersOfPerson(baseDir: str, \
nickname: str,domain: str, \
followFile='following.txt') -> []:
"""Returns a list containing the followers of the given person
Used by the shared inbox to know who to send incoming mail to
"""
followers=[]
2019-07-11 12:29:31 +00:00
if ':' in domain:
domain=domain.split(':')[0]
handle=nickname.lower()+'@'+domain.lower()
if not os.path.isdir(baseDir+'/accounts/'+handle):
return followers
for subdir, dirs, files in os.walk(baseDir+'/accounts'):
for account in dirs:
filename = os.path.join(subdir, account)+'/'+followFile
2019-07-08 13:30:04 +00:00
if account == handle or account.startswith('inbox@'):
continue
if not os.path.isfile(filename):
continue
with open(filename, 'r') as followingfile:
for followingHandle in followingfile:
if followingHandle.replace('\n','')==handle:
if account not in followers:
followers.append(account)
break
return followers
2019-07-03 09:40:27 +00:00
def followerOfPerson(baseDir: str,nickname: str, domain: str, \
followerNickname: str, followerDomain: str, \
2019-07-06 19:24:52 +00:00
federationList: [],debug :bool) -> bool:
"""Adds a follower of the given person
"""
2019-07-06 19:24:52 +00:00
return followPerson(baseDir,nickname,domain, \
followerNickname,followerDomain, \
federationList,debug,'followers.txt')
2019-06-29 18:23:13 +00:00
2019-07-03 09:40:27 +00:00
def unfollowPerson(baseDir: str,nickname: str, domain: str, \
followNickname: str, followDomain: str, \
2019-07-02 19:05:59 +00:00
followFile='following.txt') -> None:
2019-06-29 18:23:13 +00:00
"""Removes a person to the follow list
"""
2019-07-03 09:40:27 +00:00
handle=nickname.lower()+'@'+domain.lower()
handleToUnfollow=followNickname.lower()+'@'+followDomain.lower()
2019-07-02 17:20:15 +00:00
if not os.path.isdir(baseDir+'/accounts'):
os.mkdir(baseDir+'/accounts')
2019-06-29 18:23:13 +00:00
if not os.path.isdir(baseDir+'/accounts/'+handle):
os.mkdir(baseDir+'/accounts/'+handle)
filename=baseDir+'/accounts/'+handle+'/'+followFile
if os.path.isfile(filename):
if handleToUnfollow not in open(filename).read():
return
with open(filename, "r") as f:
lines = f.readlines()
with open(filename, "w") as f:
for line in lines:
if line.strip("\n") != handleToUnfollow:
f.write(line)
2019-07-03 09:40:27 +00:00
def unfollowerOfPerson(baseDir: str,nickname: str,domain: str, \
followerNickname: str,followerDomain: str) -> None:
"""Remove a follower of a person
"""
2019-07-06 17:00:22 +00:00
unfollowPerson(baseDir,nickname,domain, \
followerNickname,followerDomain,'followers.txt')
2019-06-29 18:23:13 +00:00
2019-07-06 17:00:22 +00:00
def clearFollows(baseDir: str,nickname: str,domain: str, \
followFile='following.txt') -> None:
2019-06-29 18:23:13 +00:00
"""Removes all follows
"""
2019-07-03 09:40:27 +00:00
handle=nickname.lower()+'@'+domain.lower()
2019-07-02 17:20:15 +00:00
if not os.path.isdir(baseDir+'/accounts'):
os.mkdir(baseDir+'/accounts')
2019-06-29 18:23:13 +00:00
if not os.path.isdir(baseDir+'/accounts/'+handle):
os.mkdir(baseDir+'/accounts/'+handle)
filename=baseDir+'/accounts/'+handle+'/'+followFile
if os.path.isfile(filename):
os.remove(filename)
2019-07-03 09:40:27 +00:00
def clearFollowers(baseDir: str,nickname: str,domain: str) -> None:
"""Removes all followers
"""
2019-07-03 09:40:27 +00:00
clearFollows(baseDir,nickname, domain,'followers.txt')
2019-06-29 20:21:37 +00:00
2019-07-06 17:00:22 +00:00
def getNoOfFollows(baseDir: str,nickname: str,domain: str, \
followFile='following.txt') -> int:
"""Returns the number of follows or followers
"""
2019-07-03 09:40:27 +00:00
handle=nickname.lower()+'@'+domain.lower()
2019-06-29 20:21:37 +00:00
filename=baseDir+'/accounts/'+handle+'/'+followFile
if not os.path.isfile(filename):
return 0
ctr = 0
with open(filename, "r") as f:
lines = f.readlines()
for line in lines:
if '#' not in line:
if '@' in line and '.' in line and not line.startswith('http'):
ctr += 1
elif line.startswith('http') and '/users/' in line:
ctr += 1
return ctr
2019-07-03 09:40:27 +00:00
def getNoOfFollowers(baseDir: str,nickname: str,domain: str) -> int:
"""Returns the number of followers of the given person
"""
2019-07-03 09:40:27 +00:00
return getNoOfFollows(baseDir,nickname,domain,'followers.txt')
2019-06-29 20:21:37 +00:00
2019-07-06 17:00:22 +00:00
def getFollowingFeed(baseDir: str,domain: str,port: int,path: str, \
httpPrefix: str, followsPerPage=12, \
followFile='following') -> {}:
"""Returns the following and followers feeds from GET requests
"""
2019-06-29 20:21:37 +00:00
if '/'+followFile not in path:
return None
# handle page numbers
headerOnly=True
pageNumber=None
if '?page=' in path:
pageNumber=path.split('?page=')[1]
if pageNumber=='true':
pageNumber=1
else:
try:
pageNumber=int(pageNumber)
except:
pass
path=path.split('?page=')[0]
headerOnly=False
if not path.endswith('/'+followFile):
return None
2019-07-03 09:40:27 +00:00
nickname=None
2019-06-29 20:21:37 +00:00
if path.startswith('/users/'):
2019-07-03 09:40:27 +00:00
nickname=path.replace('/users/','',1).replace('/'+followFile,'')
2019-06-29 20:21:37 +00:00
if path.startswith('/@'):
2019-07-03 09:40:27 +00:00
nickname=path.replace('/@','',1).replace('/'+followFile,'')
if not nickname:
2019-06-29 20:21:37 +00:00
return None
2019-07-03 09:40:27 +00:00
if not validNickname(nickname):
2019-06-29 20:21:37 +00:00
return None
2019-06-30 19:01:43 +00:00
if port!=80 and port!=443:
domain=domain+':'+str(port)
2019-06-29 20:21:37 +00:00
if headerOnly:
following = {
'@context': 'https://www.w3.org/ns/activitystreams',
2019-07-03 19:00:03 +00:00
'first': httpPrefix+'://'+domain+'/users/'+nickname+'/'+followFile+'?page=1',
'id': httpPrefix+'://'+domain+'/users/'+nickname+'/'+followFile,
2019-07-06 19:54:09 +00:00
'totalItems': getNoOfFollows(baseDir,nickname,domain),
2019-06-29 20:21:37 +00:00
'type': 'OrderedCollection'}
return following
if not pageNumber:
pageNumber=1
nextPageNumber=int(pageNumber+1)
following = {
'@context': 'https://www.w3.org/ns/activitystreams',
2019-07-03 19:00:03 +00:00
'id': httpPrefix+'://'+domain+'/users/'+nickname+'/'+followFile+'?page='+str(pageNumber),
2019-06-29 20:21:37 +00:00
'orderedItems': [],
2019-07-03 19:00:03 +00:00
'partOf': httpPrefix+'://'+domain+'/users/'+nickname+'/'+followFile,
2019-06-29 20:21:37 +00:00
'totalItems': 0,
'type': 'OrderedCollectionPage'}
2019-07-03 09:40:27 +00:00
handle=nickname.lower()+'@'+domain.lower()
2019-06-29 20:21:37 +00:00
filename=baseDir+'/accounts/'+handle+'/'+followFile+'.txt'
if not os.path.isfile(filename):
return following
currPage=1
pageCtr=0
totalCtr=0
with open(filename, "r") as f:
lines = f.readlines()
for line in lines:
if '#' not in line:
if '@' in line and '.' in line and not line.startswith('http'):
pageCtr += 1
totalCtr += 1
if currPage==pageNumber:
2019-07-03 19:00:03 +00:00
url = httpPrefix + '://' + line.lower().replace('\n','').split('@')[1] + \
2019-06-29 20:21:37 +00:00
'/users/' + line.lower().replace('\n','').split('@')[0]
following['orderedItems'].append(url)
2019-07-03 19:00:03 +00:00
elif (line.startswith('http') or line.startswith('dat')) and '/users/' in line:
2019-06-29 20:21:37 +00:00
pageCtr += 1
totalCtr += 1
if currPage==pageNumber:
following['orderedItems'].append(line.lower().replace('\n',''))
if pageCtr>=followsPerPage:
pageCtr=0
currPage += 1
following['totalItems']=totalCtr
lastPage=int(totalCtr/followsPerPage)
if lastPage<1:
lastPage=1
if nextPageNumber>lastPage:
2019-07-03 19:00:03 +00:00
following['next']=httpPrefix+'://'+domain+'/users/'+nickname+'/'+followFile+'?page='+str(lastPage)
2019-06-29 20:21:37 +00:00
return following
2019-07-02 18:17:04 +00:00
2019-07-06 17:00:22 +00:00
def receiveFollowRequest(session,baseDir: str,httpPrefix: str, \
port: int,sendThreads: [],postLog: [], \
cachedWebfingers: {},personCache: {}, \
messageJson: {},federationList: [], \
2019-07-09 17:54:08 +00:00
debug : bool, \
acceptedCaps=["inbox:write","objects:read"]) -> bool:
2019-07-02 18:38:51 +00:00
"""Receives a follow request within the POST section of HTTPServer
"""
2019-07-02 18:17:04 +00:00
if not messageJson['type'].startswith('Follow'):
return False
2019-07-06 13:49:25 +00:00
if not messageJson.get('actor'):
if debug:
print('DEBUG: follow request has no actor')
return False
2019-07-02 18:17:04 +00:00
if '/users/' not in messageJson['actor']:
2019-07-06 13:49:25 +00:00
if debug:
print('DEBUG: "users" missing from actor')
2019-07-02 18:17:04 +00:00
return False
2019-07-06 15:17:21 +00:00
domain,tempPort=getDomainFromActor(messageJson['actor'])
fromPort=port
if tempPort:
fromPort=tempPort
2019-07-02 18:17:04 +00:00
if not domainPermitted(domain,federationList):
2019-07-06 13:49:25 +00:00
if debug:
print('DEBUG: follower from domain not permitted - '+domain)
2019-07-02 18:17:04 +00:00
return False
2019-07-06 15:17:21 +00:00
nickname=getNicknameFromActor(messageJson['actor'])
if not nickname:
if debug:
print('DEBUG: follow request does not contain a nickname')
return False
2019-07-03 09:40:27 +00:00
handle=nickname.lower()+'@'+domain.lower()
2019-07-02 18:17:04 +00:00
if '/users/' not in messageJson['object']:
2019-07-06 13:49:25 +00:00
if debug:
print('DEBUG: "users" not found within object')
2019-07-02 18:17:04 +00:00
return False
2019-07-06 15:17:21 +00:00
domainToFollow,tempPort=getDomainFromActor(messageJson['object'])
2019-07-02 18:17:04 +00:00
if not domainPermitted(domainToFollow,federationList):
2019-07-06 13:49:25 +00:00
if debug:
print('DEBUG: follow domain not permitted '+domainToFollow)
2019-07-02 18:17:04 +00:00
return False
2019-07-06 15:17:21 +00:00
nicknameToFollow=getNicknameFromActor(messageJson['object'])
if not nicknameToFollow:
if debug:
print('DEBUG: follow request does not contain a nickname for the account followed')
return False
2019-07-03 09:40:27 +00:00
handleToFollow=nicknameToFollow.lower()+'@'+domainToFollow.lower()
2019-07-02 18:17:04 +00:00
if domainToFollow==domain:
if not os.path.isdir(baseDir+'/accounts/'+handleToFollow):
2019-07-06 13:49:25 +00:00
if debug:
2019-07-06 17:00:22 +00:00
print('DEBUG: followed account not found - '+ \
baseDir+'/accounts/'+handleToFollow)
2019-07-02 18:17:04 +00:00
return False
2019-07-06 17:15:03 +00:00
if not followerOfPerson(baseDir,nicknameToFollow,domainToFollow, \
2019-07-06 19:24:52 +00:00
nickname,domain,federationList,debug):
2019-07-06 13:49:25 +00:00
if debug:
2019-07-06 17:00:22 +00:00
print('DEBUG: '+nickname+'@'+domain+ \
' is already a follower of '+ \
nicknameToFollow+'@'+domainToFollow)
2019-07-05 18:57:19 +00:00
return False
# send accept back
2019-07-06 13:49:25 +00:00
if debug:
2019-07-06 17:00:22 +00:00
print('DEBUG: sending Accept for follow request which arrived at '+ \
nicknameToFollow+'@'+domainToFollow+' back to '+nickname+'@'+domain)
2019-07-05 18:57:19 +00:00
personUrl=messageJson['actor']
2019-07-09 14:20:23 +00:00
acceptJson=createAccept(baseDir,federationList, \
2019-07-07 13:53:12 +00:00
nicknameToFollow,domainToFollow,port, \
2019-07-09 17:54:08 +00:00
personUrl,'',httpPrefix,messageJson,acceptedCaps)
2019-07-06 15:17:21 +00:00
if debug:
pprint(acceptJson)
2019-07-06 17:00:22 +00:00
print('DEBUG: sending follow Accept from '+ \
nicknameToFollow+'@'+domainToFollow+ \
' port '+str(port)+' to '+ \
nickname+'@'+domain+' port '+ str(fromPort))
2019-07-06 13:49:25 +00:00
clientToServer=False
2019-07-06 17:00:22 +00:00
return sendSignedJson(acceptJson,session,baseDir, \
nicknameToFollow,domainToFollow,port, \
2019-07-06 15:17:21 +00:00
nickname,domain,fromPort, '', \
httpPrefix,True,clientToServer, \
2019-07-09 14:20:23 +00:00
federationList, \
2019-07-06 17:00:22 +00:00
sendThreads,postLog,cachedWebfingers, \
personCache,debug)
2019-07-02 18:38:51 +00:00
2019-07-06 17:00:22 +00:00
def sendFollowRequest(session,baseDir: str, \
nickname: str,domain: str,port: int,httpPrefix: str, \
followNickname: str,followDomain: str, \
followPort: bool,followHttpPrefix: str, \
2019-07-09 14:20:23 +00:00
clientToServer: bool,federationList: [], \
2019-07-06 17:00:22 +00:00
sendThreads: [],postLog: [],cachedWebfingers: {}, \
personCache: {},debug : bool) -> {}:
2019-07-02 20:54:22 +00:00
"""Gets the json object for sending a follow request
2019-07-06 13:49:25 +00:00
"""
2019-07-02 18:38:51 +00:00
if not domainPermitted(followDomain,federationList):
return None
2019-07-07 13:53:12 +00:00
fullDomain=domain
2019-07-06 13:49:25 +00:00
followActor=httpPrefix+'://'+domain+'/users/'+nickname
2019-07-02 18:38:51 +00:00
if port!=80 and port!=443:
2019-07-07 13:53:12 +00:00
fullDomain=domain+':'+str(port)
2019-07-06 13:49:25 +00:00
followActor=httpPrefix+'://'+domain+':'+str(port)+'/users/'+nickname
2019-07-02 18:38:51 +00:00
2019-07-06 13:49:25 +00:00
requestDomain=followDomain
2019-07-02 18:38:51 +00:00
if followPort!=80 and followPort!=443:
2019-07-06 13:49:25 +00:00
requestDomain=followDomain+':'+str(followPort)
2019-07-06 10:33:57 +00:00
2019-07-06 19:24:52 +00:00
statusNumber,published = getStatusNumber()
followedId=followHttpPrefix+'://'+requestDomain+'/users/'+followNickname
2019-07-05 20:32:21 +00:00
newFollowJson = {
2019-07-07 13:53:12 +00:00
'id': httpPrefix+'://'+fullDomain+'/users/'+nickname+'/statuses/'+statusNumber,
2019-07-02 18:38:51 +00:00
'type': 'Follow',
2019-07-06 10:33:57 +00:00
'actor': followActor,
2019-07-06 19:24:52 +00:00
'object': followedId,
'to': [followedId],
'cc': ['https://www.w3.org/ns/activitystreams#Public'],
'published': published
2019-07-02 18:38:51 +00:00
}
2019-07-02 19:05:59 +00:00
2019-07-05 20:32:21 +00:00
sendSignedJson(newFollowJson,session,baseDir,nickname,domain,port, \
2019-07-06 19:24:52 +00:00
followNickname,followDomain,followPort, \
'https://www.w3.org/ns/activitystreams#Public', \
2019-07-06 13:49:25 +00:00
httpPrefix,True,clientToServer, \
2019-07-09 14:20:23 +00:00
federationList, \
2019-07-06 13:49:25 +00:00
sendThreads,postLog,cachedWebfingers,personCache, debug)
2019-07-05 20:32:21 +00:00
return newFollowJson
2019-07-11 12:29:31 +00:00
def getFollowersOfActor(baseDir :str,actor :str,debug: bool) -> {}:
"""In a shared inbox if we receive a post we know who it's from
2019-07-08 17:15:55 +00:00
and if it's addressed to followers then we need to get a list of those.
This returns a list of account handles which follow the given actor
and also the corresponding capability id if it exists
"""
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: getting followers of '+actor)
recipientsDict={}
2019-07-08 17:15:55 +00:00
if ':' not in actor:
2019-07-08 22:12:24 +00:00
return recipientsDict
2019-07-08 17:15:55 +00:00
httpPrefix=actor.split(':')[0]
nickname=getNicknameFromActor(actor)
if not nickname:
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: no nickname found in '+actor)
2019-07-08 22:12:24 +00:00
return recipientsDict
domain,port=getDomainFromActor(actor)
if not domain:
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: no domain found in '+actor)
2019-07-08 22:12:24 +00:00
return recipientsDict
actorHandle=nickname+'@'+domain
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: searching for handle '+actorHandle)
# for each of the accounts
for subdir, dirs, files in os.walk(baseDir+'/accounts'):
for account in dirs:
if '@' in account and not account.startswith('inbox@'):
followingFilename = os.path.join(subdir, account)+'/following.txt'
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: examining follows of '+account)
print(followingFilename)
if os.path.isfile(followingFilename):
# does this account follow the given actor?
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: checking if '+actorHandle+' in '+followingFilename)
2019-07-08 17:15:55 +00:00
if actorHandle in open(followingFilename).read():
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: '+account+' follows '+actorHandle)
2019-07-08 22:12:24 +00:00
ocapFilename=baseDir+'/accounts/'+account+'/ocap/accept/'+httpPrefix+':##'+domain+':'+str(port)+'#users#'+nickname+'.json'
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: checking capabilities of'+account)
2019-07-08 17:15:55 +00:00
if os.path.isfile(ocapFilename):
with open(ocapFilename, 'r') as fp:
ocapJson=commentjson.load(fp)
2019-07-11 12:29:31 +00:00
if ocapJson.get('id'):
if debug:
print('DEBUG: capabilities id found for '+account)
recipientsDict[account]=ocapJson['id']
else:
if debug:
print('DEBUG: capabilities has no id attribute')
recipientsDict[account]=None
2019-07-08 17:15:55 +00:00
else:
2019-07-11 12:29:31 +00:00
if debug:
print('DEBUG: No capabilities file found for '+account+' granted by '+actorHandle)
print(ocapFilename)
2019-07-08 22:12:24 +00:00
recipientsDict[account]=None
return recipientsDict