From b321baf307b52c093f332c0f899ee9251250c090 Mon Sep 17 00:00:00 2001 From: Bob Mottram Date: Thu, 4 Jul 2019 17:24:23 +0100 Subject: [PATCH] Generic in/outbox functions --- daemon.py | 34 ++++++++------ person.py | 59 +++++++++++++++++++++--- posts.py | 134 +++++++++++++++++++++++++++++++----------------------- tests.py | 12 +++-- 4 files changed, 157 insertions(+), 82 deletions(-) diff --git a/daemon.py b/daemon.py index a4dd5b2f..912dd8f1 100644 --- a/daemon.py +++ b/daemon.py @@ -16,10 +16,10 @@ from webfinger import webfingerMeta from webfinger import webfingerLookup from webfinger import webfingerHandle from person import personLookup -from person import personOutboxJson +from person import personBoxJson from posts import getPersonPubKey from posts import outboxMessageCreateWrap -from posts import savePostToOutbox +from posts import savePostToBox from inbox import inboxPermittedMessage from inbox import inboxMessageHasParams from inbox import runInboxQueue @@ -142,7 +142,7 @@ class PubServer(BaseHTTPRequestHandler): postId=messageJson['id'] else: postId=None - savePostToOutbox(self.server.baseDir,postId,self.postToNickname,self.server.domain,messageJson) + savePostToBox(self.server.baseDir,postId,self.postToNickname,self.server.domain,messageJson,'outbox') return True def do_GET(self): @@ -175,13 +175,20 @@ class PubServer(BaseHTTPRequestHandler): if self.path.endswith('/inbox'): if '/users/' in self.path: if self.headers.get('Authorization'): - if authorize(self.server.baseDir,self.path,self.headers['Authorization'],self.server.debug): - # TODO - print('inbox access not supported yet') - self.send_response(405) - self.end_headers() - self.server.POSTbusy=False - return + if authorize(self.server.baseDir,self.path, \ + self.headers['Authorization'], \ + self.server.debug): + inboxFeed=personBoxJson(self.server.baseDir, \ + self.server.domain, \ + self.server.port, \ + self.path, \ + self.server.httpPrefix, \ + maxPostsInFeed, 'inbox') + if inboxFeed: + self._set_headers('application/json') + self.wfile.write(json.dumps(inboxFeed).encode('utf-8')) + self.server.GETbusy=False + return else: if self.server.debug: print('DEBUG: '+nickname+' was not authorized to access '+self.path) @@ -193,9 +200,10 @@ class PubServer(BaseHTTPRequestHandler): return # get outbox feed for a person - outboxFeed=personOutboxJson(self.server.baseDir,self.server.domain, \ - self.server.port,self.path, \ - self.server.httpPrefix,maxPostsInFeed) + outboxFeed=personBoxJson(self.server.baseDir,self.server.domain, \ + self.server.port,self.path, \ + self.server.httpPrefix, \ + maxPostsInFeed, 'outbox') if outboxFeed: self._set_headers('application/json') self.wfile.write(json.dumps(outboxFeed).encode('utf-8')) diff --git a/person.py b/person.py index 01168314..55fab8d5 100644 --- a/person.py +++ b/person.py @@ -148,11 +148,14 @@ def personLookup(domain: str,path: str,baseDir: str) -> {}: personJson=commentjson.load(fp) return personJson -def personOutboxJson(baseDir: str,domain: str,port: int,path: str, \ - httpPrefix: str,noOfItems: int) -> []: - """Obtain the outbox feed for the given person +def personBoxJson(baseDir: str,domain: str,port: int,path: str, \ + httpPrefix: str,noOfItems: int,boxname: str) -> []: + """Obtain the inbox/outbox feed for the given person """ - if not '/outbox' in path: + if boxname!='inbox' and boxname!='outbox': + return None + + if not '/'+boxname in path: return None # Only show the header by default @@ -172,20 +175,62 @@ def personOutboxJson(baseDir: str,domain: str,port: int,path: str, \ path=path.split('?page=')[0] headerOnly=False - if not path.endswith('/outbox'): + if not path.endswith('/'+boxname): return None nickname=None if path.startswith('/users/'): - nickname=path.replace('/users/','',1).replace('/outbox','') + nickname=path.replace('/users/','',1).replace('/'+boxname,'') if path.startswith('/@'): - nickname=path.replace('/@','',1).replace('/outbox','') + nickname=path.replace('/@','',1).replace('/'+boxname,'') if not nickname: return None if not validNickname(nickname): return None + if boxname=='inbox': + return createInbox(baseDir,nickname,domain,port,httpPrefix, \ + noOfItems,headerOnly,pageNumber) return createOutbox(baseDir,nickname,domain,port,httpPrefix, \ noOfItems,headerOnly,pageNumber) +def personInboxJson(baseDir: str,domain: str,port: int,path: str, \ + httpPrefix: str,noOfItems: int) -> []: + """Obtain the inbox feed for the given person + Authentication is expected to have already happened + """ + if not '/inbox' in path: + return None + + # Only show the header by default + headerOnly=True + + # handle page numbers + 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('/inbox'): + return None + nickname=None + if path.startswith('/users/'): + nickname=path.replace('/users/','',1).replace('/inbox','') + if path.startswith('/@'): + nickname=path.replace('/@','',1).replace('/inbox','') + if not nickname: + return None + if not validNickname(nickname): + return None + return createInbox(baseDir,nickname,domain,port,httpPrefix, \ + noOfItems,headerOnly,pageNumber) + def setPreferredNickname(baseDir: str,nickname: str, domain: str, \ preferredName: str) -> bool: if len(preferredName)>32: diff --git a/posts.py b/posts.py index 69aa163c..f330ba7d 100644 --- a/posts.py +++ b/posts.py @@ -28,7 +28,7 @@ from session import postJson from webfinger import webfingerHandle from httpsig import createSignedHeader from utils import getStatusNumber -from utils import createOutboxDir +from utils import createPersonDir from utils import urlPermitted try: from BeautifulSoup import BeautifulSoup @@ -234,23 +234,25 @@ def getPosts(session,outboxUrl: str,maxPosts: int,maxMentions: int, \ break return personPosts -def createOutboxArchive(nickname: str,domain: str,baseDir: str) -> str: - """Creates an archive directory for outbox posts +def createBoxArchive(nickname: str,domain: str,baseDir: str,boxname: str) -> str: + """Creates an archive directory for inbox/outbox posts """ handle=nickname.lower()+'@'+domain.lower() if not os.path.isdir(baseDir+'/accounts/'+handle): os.mkdir(baseDir+'/accounts/'+handle) - outboxArchiveDir=baseDir+'/accounts/'+handle+'/outboxarchive' - if not os.path.isdir(outboxArchiveDir): - os.mkdir(outboxArchiveDir) - return outboxArchiveDir + boxArchiveDir=baseDir+'/accounts/'+handle+'/'+boxname+'archive' + if not os.path.isdir(boxArchiveDir): + os.mkdir(boxArchiveDir) + return boxArchiveDir -def deleteAllPosts(baseDir: str,nickname: str, domain: str) -> None: - """Deletes all posts for a person +def deleteAllPosts(baseDir: str,nickname: str, domain: str,boxname: str) -> None: + """Deletes all posts for a person from inbox or outbox """ - outboxDir = createOutboxDir(nickname,domain,baseDir) - for deleteFilename in os.listdir(outboxDir): - filePath = os.path.join(outboxDir, deleteFilename) + if boxname!='inbox' and boxname!='outbox': + return + boxDir = createPersonDir(nickname,domain,baseDir,boxname) + for deleteFilename in os.listdir(boxDir): + filePath = os.path.join(boxDir, deleteFilename) try: if os.path.isfile(filePath): os.unlink(filePath) @@ -258,11 +260,14 @@ def deleteAllPosts(baseDir: str,nickname: str, domain: str) -> None: except Exception as e: print(e) -def savePostToOutbox(baseDir: str,httpPrefix: str,postId: str,nickname: str, domain: str,postJson: {}) -> None: - """Saves the give json to the outbox +def savePostToBox(baseDir: str,httpPrefix: str,postId: str,nickname: str, domain: str,postJson: {},boxname: str) -> None: + """Saves the give json to the give box """ + if boxname!='inbox' and boxname!='outbox': + return if ':' in domain: domain=domain.split(':')[0] + if not postId: statusNumber,published = getStatusNumber() postId=httpPrefix+'://'+domain+'/users/'+nickname+'/statuses/'+statusNumber @@ -271,8 +276,8 @@ def savePostToOutbox(baseDir: str,httpPrefix: str,postId: str,nickname: str, dom postJson['object']['id']=postId postJson['object']['atomUri']=postId - outboxDir = createOutboxDir(nickname,domain,baseDir) - filename=outboxDir+'/'+postId.replace('/','#')+'.json' + boxDir = createPersonDir(nickname,domain,baseDir,boxname) + filename=boxDir+'/'+postId.replace('/','#')+'.json' with open(filename, 'w') as fp: commentjson.dump(postJson, fp, indent=4, sort_keys=False) @@ -366,7 +371,7 @@ def createPostBase(baseDir: str,nickname: str, domain: str, port: int, \ newPost['cc']=ccUrl newPost['object']['cc']=ccUrl if saveToFile: - savePostToOutbox(baseDir,httpPrefix,newPostId,nickname,domain,newPost) + savePostToBox(baseDir,httpPrefix,newPostId,nickname,domain,newPost,'outbox') return newPost def outboxMessageCreateWrap(httpPrefix: str,nickname: str,domain: str,messageJson: {}) -> {}: @@ -499,11 +504,22 @@ def sendPost(session,baseDir: str,nickname: str, domain: str, port: int, \ thr.start() return 0 +def createInbox(baseDir: str,nickname: str,domain: str,port: int,httpPrefix: str, \ + itemsPerPage: int,headerOnly: bool,pageNumber=None) -> {}: + return createBoxBase(baseDir,'inbox',nickname,domain,port,httpPrefix, \ + itemsPerPage,headerOnly,pageNumber) def createOutbox(baseDir: str,nickname: str,domain: str,port: int,httpPrefix: str, \ itemsPerPage: int,headerOnly: bool,pageNumber=None) -> {}: - """Constructs the outbox feed + return createBoxBase(baseDir,'outbox',nickname,domain,port,httpPrefix, \ + itemsPerPage,headerOnly,pageNumber) + +def createBoxBase(baseDir: str,boxname: str,nickname: str,domain: str,port: int,httpPrefix: str, \ + itemsPerPage: int,headerOnly: bool,pageNumber=None) -> {}: + """Constructs the box feed """ - outboxDir = createOutboxDir(nickname,domain,baseDir) + if boxname!='inbox' and boxname!='outbox': + return None + boxDir = createPersonDir(nickname,domain,baseDir,boxname) if port!=80 and port!=443: domain = domain+':'+str(port) @@ -514,69 +530,69 @@ def createOutbox(baseDir: str,nickname: str,domain: str,port: int,httpPrefix: st pageStr='?page='+str(pageNumber) except: pass - outboxHeader = {'@context': 'https://www.w3.org/ns/activitystreams', - 'first': httpPrefix+'://'+domain+'/users/'+nickname+'/outbox?page=true', - 'id': httpPrefix+'://'+domain+'/users/'+nickname+'/outbox', - 'last': httpPrefix+'://'+domain+'/users/'+nickname+'/outbox?page=true', - 'totalItems': 0, - 'type': 'OrderedCollection'} - outboxItems = {'@context': 'https://www.w3.org/ns/activitystreams', - 'id': httpPrefix+'://'+domain+'/users/'+nickname+'/outbox'+pageStr, - 'orderedItems': [ - ], - 'partOf': httpPrefix+'://'+domain+'/users/'+nickname+'/outbox', - 'type': 'OrderedCollectionPage'} + boxHeader = {'@context': 'https://www.w3.org/ns/activitystreams', + 'first': httpPrefix+'://'+domain+'/users/'+nickname+'/'+boxname+'?page=true', + 'id': httpPrefix+'://'+domain+'/users/'+nickname+'/'+boxname, + 'last': httpPrefix+'://'+domain+'/users/'+nickname+'/'+boxname+'?page=true', + 'totalItems': 0, + 'type': 'OrderedCollection'} + boxItems = {'@context': 'https://www.w3.org/ns/activitystreams', + 'id': httpPrefix+'://'+domain+'/users/'+nickname+'/'+boxname+pageStr, + 'orderedItems': [ + ], + 'partOf': httpPrefix+'://'+domain+'/users/'+nickname+'/'+boxname, + 'type': 'OrderedCollectionPage'} # counter for posts loop postsOnPageCtr=0 # post filenames sorted in descending order - postsInOutbox=sorted(os.listdir(outboxDir), reverse=True) + postsInBox=sorted(os.listdir(boxDir), reverse=True) - # number of posts in outbox - outboxHeader['totalItems']=len(postsInOutbox) + # number of posts in box + boxHeader['totalItems']=len(postsInBox) prevPostFilename=None if not pageNumber: pageNumber=1 # Generate first and last entries within header - if len(postsInOutbox)>0: - lastPage=int(len(postsInOutbox)/itemsPerPage) + if len(postsInBox)>0: + lastPage=int(len(postsInBox)/itemsPerPage) if lastPage<1: lastPage=1 - outboxHeader['last']= \ - httpPrefix+'://'+domain+'/users/'+nickname+'/outbox?page='+str(lastPage) + boxHeader['last']= \ + httpPrefix+'://'+domain+'/users/'+nickname+'/'+boxname+'?page='+str(lastPage) # Insert posts currPage=1 postsCtr=0 - for postFilename in postsInOutbox: + for postFilename in postsInBox: # Are we at the starting page yet? if prevPostFilename and currPage==pageNumber and postsCtr==0: # update the prev entry for the last message id postId = prevPostFilename.split('#statuses#')[1].replace('#activity','') - outboxHeader['prev']= \ - httpPrefix+'://'+domain+'/users/'+nickname+'/outbox?min_id='+postId+'&page=true' + boxHeader['prev']= \ + httpPrefix+'://'+domain+'/users/'+nickname+'/'+boxname+'?min_id='+postId+'&page=true' # get the full path of the post file - filePath = os.path.join(outboxDir, postFilename) + filePath = os.path.join(boxDir, postFilename) try: if os.path.isfile(filePath): if currPage == pageNumber and postsOnPageCtr <= itemsPerPage: # get the post as json with open(filePath, 'r') as fp: p=commentjson.load(fp) - # insert it into the outbox feed + # insert it into the box feed if postsOnPageCtr < itemsPerPage: if not headerOnly: - outboxItems['orderedItems'].append(p) + boxItems['orderedItems'].append(p) elif postsOnPageCtr == itemsPerPage: # if this is the last post update the next message ID if '/statuses/' in p['id']: postId = p['id'].split('/statuses/')[1].replace('/activity','') - outboxHeader['next']= \ + boxHeader['next']= \ httpPrefix+'://'+domain+'/users/'+ \ - nickname+'/outbox?max_id='+ \ + nickname+'/'+boxname+'?max_id='+ \ postId+'&page=true' postsOnPageCtr += 1 # remember the last post filename for use with prev @@ -591,29 +607,31 @@ def createOutbox(baseDir: str,nickname: str,domain: str,port: int,httpPrefix: st except Exception as e: print(e) if headerOnly: - return outboxHeader - return outboxItems + return boxHeader + return boxItems def archivePosts(nickname: str,domain: str,baseDir: str, \ - maxPostsInOutbox=256) -> None: - """Retain a maximum number of posts within the outbox + boxname: str,maxPostsInBox=256) -> None: + """Retain a maximum number of posts within the given box Move any others to an archive directory """ - outboxDir = createOutboxDir(nickname,domain,baseDir) - archiveDir = createOutboxArchive(nickname,domain,baseDir) - postsInOutbox=sorted(os.listdir(outboxDir), reverse=False) - noOfPosts=len(postsInOutbox) - if noOfPosts<=maxPostsInOutbox: + if boxname!='inbox' and boxname!='outbox': + return + boxDir = createPersonDir(nickname,domain,baseDir,boxname) + archiveDir = createBoxArchive(nickname,domain,baseDir,boxname) + postsInBox=sorted(os.listdir(boxDir), reverse=False) + noOfPosts=len(postsInBox) + if noOfPosts<=maxPostsInBox: return - for postFilename in postsInOutbox: - filePath = os.path.join(outboxDir, postFilename) + for postFilename in postsInBox: + filePath = os.path.join(boxDir, postFilename) if os.path.isfile(filePath): archivePath = os.path.join(archiveDir, postFilename) os.rename(filePath,archivePath) # TODO: possibly archive any associated media files noOfPosts -= 1 - if noOfPosts <= maxPostsInOutbox: + if noOfPosts <= maxPostsInBox: break def getPublicPostsOfPerson(nickname,domain,raw,simple): diff --git a/tests.py b/tests.py index dd1f31a9..9514eb43 100644 --- a/tests.py +++ b/tests.py @@ -115,7 +115,8 @@ def createServerAlice(path: str,domain: str,port: int,federationList: []): useTor=False clientToServer=False privateKeyPem,publicKeyPem,person,wfEndpoint=createPerson(path,nickname,domain,port,httpPrefix,True) - deleteAllPosts(path,nickname,domain) + deleteAllPosts(path,nickname,domain,'inbox') + deleteAllPosts(path,nickname,domain,'outbox') followPerson(path,nickname,domain,'bob','127.0.0.100:61936',federationList) followerOfPerson(path,nickname,domain,'bob','127.0.0.100:61936',federationList) createPublicPost(path,nickname, domain, port,httpPrefix, "No wise fish would go anywhere without a porpoise", False, True, clientToServer) @@ -137,7 +138,8 @@ def createServerBob(path: str,domain: str,port: int,federationList: []): useTor=False clientToServer=False privateKeyPem,publicKeyPem,person,wfEndpoint=createPerson(path,nickname,domain,port,httpPrefix,True) - deleteAllPosts(path,nickname,domain) + deleteAllPosts(path,nickname,domain,'inbox') + deleteAllPosts(path,nickname,domain,'outbox') followPerson(path,nickname,domain,'alice','127.0.0.50:61935',federationList) followerOfPerson(path,nickname,domain,'alice','127.0.0.50:61935',federationList) createPublicPost(path,nickname, domain, port,httpPrefix, "It's your life, live it your way.", False, True, clientToServer) @@ -289,10 +291,12 @@ def testCreatePerson(): os.chdir(baseDir) privateKeyPem,publicKeyPem,person,wfEndpoint=createPerson(baseDir,nickname,domain,port,httpPrefix,True) - deleteAllPosts(baseDir,nickname,domain) + deleteAllPosts(baseDir,nickname,domain,'inbox') + deleteAllPosts(baseDir,nickname,domain,'outbox') setPreferredNickname(baseDir,nickname,domain,'badger') setBio(baseDir,nickname,domain,'Randomly roaming in your backyard') - archivePosts(nickname,domain,baseDir,4) + archivePosts(nickname,domain,baseDir,'inbox',4) + archivePosts(nickname,domain,baseDir,'outbox',4) createPublicPost(baseDir,nickname, domain, port,httpPrefix, "G'day world!", False, True, clientToServer, None, None, 'Not suitable for Vogons') os.chdir(currDir)