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
|
|
|
|
from pprint import pprint
|
|
|
|
import os
|
|
|
|
import sys
|
2019-06-29 20:21:37 +00:00
|
|
|
from person import validUsername
|
2019-07-02 10:39:55 +00:00
|
|
|
from utils import domainPermitted
|
2019-06-29 18:23:13 +00:00
|
|
|
|
2019-07-02 19:05:59 +00:00
|
|
|
def followPerson(baseDir: str,username: str, domain: str, \
|
|
|
|
followUsername: str, followDomain: str, \
|
|
|
|
federationList: [], followFile='following.txt') -> bool:
|
2019-06-29 18:23:13 +00:00
|
|
|
"""Adds a person to the follow list
|
|
|
|
"""
|
2019-07-02 10:39:55 +00:00
|
|
|
if not domainPermitted(followDomain.lower().replace('\n',''), federationList):
|
2019-06-29 21:13:44 +00:00
|
|
|
return False
|
2019-06-29 18:23:13 +00:00
|
|
|
handle=username.lower()+'@'+domain.lower()
|
|
|
|
handleToFollow=followUsername.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 handleToFollow in open(filename).read():
|
2019-06-29 21:13:44 +00:00
|
|
|
return True
|
2019-06-29 18:23:13 +00:00
|
|
|
with open(filename, "a") as followfile:
|
|
|
|
followfile.write(handleToFollow+'\n')
|
2019-06-29 21:13:44 +00:00
|
|
|
return True
|
2019-06-29 18:23:13 +00:00
|
|
|
with open(filename, "w") as followfile:
|
|
|
|
followfile.write(handleToFollow+'\n')
|
2019-06-29 21:13:44 +00:00
|
|
|
return True
|
2019-06-29 18:23:13 +00:00
|
|
|
|
2019-07-02 19:05:59 +00:00
|
|
|
def followerOfPerson(baseDir: str,username: str, domain: str, \
|
|
|
|
followerUsername: str, followerDomain: str, \
|
|
|
|
federationList: []) -> bool:
|
2019-06-29 21:13:44 +00:00
|
|
|
"""Adds a follower of the given person
|
|
|
|
"""
|
2019-07-02 19:05:59 +00:00
|
|
|
return followPerson(baseDir,username, domain, \
|
|
|
|
followerUsername, followerDomain, \
|
|
|
|
federationList, 'followers.txt')
|
2019-06-29 18:23:13 +00:00
|
|
|
|
2019-07-02 19:05:59 +00:00
|
|
|
def unfollowPerson(baseDir: str,username: str, domain: str, \
|
|
|
|
followUsername: str, followDomain: str, \
|
|
|
|
followFile='following.txt') -> None:
|
2019-06-29 18:23:13 +00:00
|
|
|
"""Removes a person to the follow list
|
|
|
|
"""
|
|
|
|
handle=username.lower()+'@'+domain.lower()
|
|
|
|
handleToUnfollow=followUsername.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-02 19:05:59 +00:00
|
|
|
def unfollowerOfPerson(baseDir: str,username: str,domain: str, \
|
|
|
|
followerUsername: str,followerDomain: str) -> None:
|
2019-06-29 21:13:44 +00:00
|
|
|
"""Remove a follower of a person
|
|
|
|
"""
|
2019-07-02 19:05:59 +00:00
|
|
|
unfollowPerson(baseDir,username,domain,followerUsername,followerDomain,'followers.txt')
|
2019-06-29 18:23:13 +00:00
|
|
|
|
2019-07-02 19:05:59 +00:00
|
|
|
def clearFollows(baseDir: str,username: str,domain: str,followFile='following.txt') -> None:
|
2019-06-29 18:23:13 +00:00
|
|
|
"""Removes all follows
|
|
|
|
"""
|
|
|
|
handle=username.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-02 19:05:59 +00:00
|
|
|
def clearFollowers(baseDir: str,username: str,domain: str) -> None:
|
2019-06-29 21:13:44 +00:00
|
|
|
"""Removes all followers
|
|
|
|
"""
|
2019-07-01 09:59:57 +00:00
|
|
|
clearFollows(baseDir,username, domain,'followers.txt')
|
2019-06-29 20:21:37 +00:00
|
|
|
|
2019-07-02 19:05:59 +00:00
|
|
|
def getNoOfFollows(baseDir: str,username: str,domain: str,followFile='following.txt') -> int:
|
2019-06-29 21:13:44 +00:00
|
|
|
"""Returns the number of follows or followers
|
|
|
|
"""
|
2019-06-29 20:21:37 +00:00
|
|
|
handle=username.lower()+'@'+domain.lower()
|
|
|
|
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-02 19:05:59 +00:00
|
|
|
def getNoOfFollowers(baseDir: str,username: str,domain: str) -> int:
|
2019-06-29 21:13:44 +00:00
|
|
|
"""Returns the number of followers of the given person
|
|
|
|
"""
|
2019-07-02 19:05:59 +00:00
|
|
|
return getNoOfFollows(baseDir,username,domain,'followers.txt')
|
2019-06-29 20:21:37 +00:00
|
|
|
|
2019-07-02 19:05:59 +00:00
|
|
|
def getFollowingFeed(baseDir: str,domain: str,port: int,path: str,https: bool, \
|
|
|
|
followsPerPage=12,followFile='following') -> {}:
|
2019-06-29 21:13:44 +00:00
|
|
|
"""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
|
|
|
|
username=None
|
|
|
|
if path.startswith('/users/'):
|
|
|
|
username=path.replace('/users/','',1).replace('/'+followFile,'')
|
|
|
|
if path.startswith('/@'):
|
|
|
|
username=path.replace('/@','',1).replace('/'+followFile,'')
|
|
|
|
if not username:
|
|
|
|
return None
|
|
|
|
if not validUsername(username):
|
|
|
|
return None
|
|
|
|
|
|
|
|
prefix='https'
|
|
|
|
if not https:
|
|
|
|
prefix='http'
|
|
|
|
|
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',
|
|
|
|
'first': prefix+'://'+domain+'/users/'+username+'/'+followFile+'?page=1',
|
|
|
|
'id': prefix+'://'+domain+'/users/'+username+'/'+followFile,
|
|
|
|
'totalItems': getNoOfFollows(username,domain),
|
|
|
|
'type': 'OrderedCollection'}
|
|
|
|
return following
|
|
|
|
|
|
|
|
if not pageNumber:
|
|
|
|
pageNumber=1
|
|
|
|
|
|
|
|
nextPageNumber=int(pageNumber+1)
|
|
|
|
following = {
|
|
|
|
'@context': 'https://www.w3.org/ns/activitystreams',
|
|
|
|
'id': prefix+'://'+domain+'/users/'+username+'/'+followFile+'?page='+str(pageNumber),
|
|
|
|
'orderedItems': [],
|
|
|
|
'partOf': prefix+'://'+domain+'/users/'+username+'/'+followFile,
|
|
|
|
'totalItems': 0,
|
|
|
|
'type': 'OrderedCollectionPage'}
|
|
|
|
|
|
|
|
handle=username.lower()+'@'+domain.lower()
|
|
|
|
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:
|
|
|
|
url = prefix + '://' + line.lower().replace('\n','').split('@')[1] + \
|
|
|
|
'/users/' + line.lower().replace('\n','').split('@')[0]
|
|
|
|
following['orderedItems'].append(url)
|
|
|
|
elif line.startswith('http') and '/users/' in line:
|
|
|
|
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:
|
|
|
|
following['next']=prefix+'://'+domain+'/users/'+username+'/'+followFile+'?page='+str(lastPage)
|
|
|
|
return following
|
2019-07-02 18:17:04 +00:00
|
|
|
|
|
|
|
def receiveFollowRequest(baseDir: str,messageJson: {},federationList: []) -> 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
|
|
|
|
if '/users/' not in messageJson['actor']:
|
|
|
|
return False
|
|
|
|
domain=messageJson['actor'].split('/users/')[0].replace('https://','').replace('http://','')
|
|
|
|
if not domainPermitted(domain,federationList):
|
|
|
|
return False
|
|
|
|
username=messageJson['actor'].split('/users/')[1].replace('@','')
|
|
|
|
handle=username.lower()+'@'+domain.lower()
|
|
|
|
if not os.path.isdir(baseDir+'/accounts/'+handle):
|
|
|
|
return False
|
|
|
|
if '/users/' not in messageJson['object']:
|
|
|
|
return False
|
|
|
|
domainToFollow=messageJson['object'].split('/users/')[0].replace('https://','').replace('http://','')
|
|
|
|
if not domainPermitted(domainToFollow,federationList):
|
|
|
|
return False
|
|
|
|
usernameToFollow=messageJson['object'].split('/users/')[1].replace('@','')
|
|
|
|
handleToFollow=usernameToFollow.lower()+'@'+domainToFollow.lower()
|
|
|
|
if domainToFollow==domain:
|
|
|
|
if not os.path.isdir(baseDir+'/accounts/'+handleToFollow):
|
|
|
|
return False
|
|
|
|
return followerOfPerson(baseDir,username,domain,usernameToFollow,domainToFollow,federationList)
|
2019-07-02 18:38:51 +00:00
|
|
|
|
2019-07-02 19:05:59 +00:00
|
|
|
def sendFollowRequest(baseDir: str,username: str,domain: str,port: int,https: bool, \
|
|
|
|
followUsername: str,followDomain: str,followPort: bool,followHttps: bool, \
|
2019-07-02 20:54:22 +00:00
|
|
|
federationList: []) -> {}:
|
|
|
|
"""Gets the json object for sending a follow request
|
2019-07-02 19:05:59 +00:00
|
|
|
"""
|
2019-07-02 18:38:51 +00:00
|
|
|
if not domainPermitted(followDomain,federationList):
|
|
|
|
return None
|
|
|
|
|
|
|
|
prefix='https'
|
|
|
|
if not https:
|
|
|
|
prefix='http'
|
|
|
|
|
|
|
|
followPrefix='https'
|
|
|
|
if not followHttps:
|
|
|
|
followPrefix='http'
|
|
|
|
|
|
|
|
if port!=80 and port!=443:
|
|
|
|
domain=domain+':'+str(port)
|
|
|
|
|
|
|
|
if followPort!=80 and followPort!=443:
|
|
|
|
followDomain=followDomain+':'+str(followPort)
|
|
|
|
|
|
|
|
newFollow = {
|
|
|
|
'type': 'Follow',
|
|
|
|
'actor': prefix+'://'+domain+'/users/'+username,
|
|
|
|
'object': followPrefix+'://'+followDomain+'/users/'+followUsername,
|
|
|
|
'to': [toUrl],
|
|
|
|
'cc': []
|
|
|
|
}
|
2019-07-02 19:05:59 +00:00
|
|
|
|
2019-07-02 18:38:51 +00:00
|
|
|
if ccUrl:
|
|
|
|
if len(ccUrl)>0:
|
|
|
|
newFollow['cc']=ccUrl
|
2019-07-02 20:54:22 +00:00
|
|
|
return newFollow
|