2020-04-03 16:27:34 +00:00
|
|
|
__filename__ = "inbox.py"
|
|
|
|
__author__ = "Bob Mottram"
|
|
|
|
__license__ = "AGPL3+"
|
2021-01-26 10:07:42 +00:00
|
|
|
__version__ = "1.2.0"
|
2020-04-03 16:27:34 +00:00
|
|
|
__maintainer__ = "Bob Mottram"
|
2021-09-10 16:14:50 +00:00
|
|
|
__email__ = "bob@libreserver.org"
|
2020-04-03 16:27:34 +00:00
|
|
|
__status__ = "Production"
|
2021-06-15 15:08:12 +00:00
|
|
|
__module_group__ = "Timeline"
|
2019-06-28 21:59:54 +00:00
|
|
|
|
|
|
|
import json
|
|
|
|
import os
|
2019-06-29 10:08:59 +00:00
|
|
|
import datetime
|
2019-07-04 12:23:53 +00:00
|
|
|
import time
|
2021-06-05 13:36:03 +00:00
|
|
|
import random
|
2021-01-04 19:02:24 +00:00
|
|
|
from linked_data_sig import verifyJsonSignature
|
2021-07-19 08:46:21 +00:00
|
|
|
from languages import understoodPostLanguage
|
2021-09-03 12:00:23 +00:00
|
|
|
from like import updateLikesCollection
|
2021-11-10 12:16:03 +00:00
|
|
|
from reaction import updateReactionCollection
|
2021-11-10 13:10:02 +00:00
|
|
|
from reaction import validEmojiContent
|
2021-11-22 12:05:09 +00:00
|
|
|
from utils import invalidCiphertext
|
2021-11-10 12:16:03 +00:00
|
|
|
from utils import removeHtml
|
2021-10-20 13:33:34 +00:00
|
|
|
from utils import fileLastModified
|
2021-10-13 11:15:06 +00:00
|
|
|
from utils import hasObjectString
|
2021-10-13 10:37:52 +00:00
|
|
|
from utils import hasObjectStringObject
|
2021-09-08 18:37:04 +00:00
|
|
|
from utils import getReplyIntervalHours
|
|
|
|
from utils import canReplyTo
|
2021-07-29 17:14:33 +00:00
|
|
|
from utils import getUserPaths
|
2021-07-20 13:33:27 +00:00
|
|
|
from utils import getBaseContentFromPost
|
2021-07-13 21:59:53 +00:00
|
|
|
from utils import acctDir
|
2021-06-26 14:21:24 +00:00
|
|
|
from utils import removeDomainPort
|
|
|
|
from utils import getPortFromDomain
|
2021-06-22 15:45:59 +00:00
|
|
|
from utils import hasObjectDict
|
2021-04-22 09:27:20 +00:00
|
|
|
from utils import dmAllowedFromDomain
|
2021-03-03 20:16:53 +00:00
|
|
|
from utils import isRecentPost
|
2021-01-05 10:48:22 +00:00
|
|
|
from utils import getConfigParam
|
2020-12-23 10:57:44 +00:00
|
|
|
from utils import hasUsersPath
|
2020-12-21 12:11:45 +00:00
|
|
|
from utils import validPostDate
|
2020-12-16 10:48:40 +00:00
|
|
|
from utils import getFullDomain
|
2020-08-23 11:13:35 +00:00
|
|
|
from utils import removeIdEnding
|
2020-06-11 12:26:15 +00:00
|
|
|
from utils import getProtocolPrefixes
|
2020-02-25 15:24:29 +00:00
|
|
|
from utils import isBlogPost
|
2020-02-04 19:34:52 +00:00
|
|
|
from utils import removeAvatarFromCache
|
2019-12-12 17:34:31 +00:00
|
|
|
from utils import isPublicPost
|
2019-11-25 14:05:59 +00:00
|
|
|
from utils import getCachedPostFilename
|
|
|
|
from utils import removePostFromCache
|
2019-07-02 10:39:55 +00:00
|
|
|
from utils import urlPermitted
|
2019-07-04 10:02:56 +00:00
|
|
|
from utils import createInboxQueueDir
|
2019-07-06 13:49:25 +00:00
|
|
|
from utils import getStatusNumber
|
2019-07-09 14:20:23 +00:00
|
|
|
from utils import getDomainFromActor
|
|
|
|
from utils import getNicknameFromActor
|
2019-07-11 12:29:31 +00:00
|
|
|
from utils import locatePost
|
2019-07-14 16:37:01 +00:00
|
|
|
from utils import deletePost
|
2019-08-12 18:02:29 +00:00
|
|
|
from utils import removeModerationPostFromIndex
|
2019-10-22 11:55:06 +00:00
|
|
|
from utils import loadJson
|
|
|
|
from utils import saveJson
|
2020-09-05 16:13:25 +00:00
|
|
|
from utils import undoLikesCollectionEntry
|
2021-11-10 12:16:03 +00:00
|
|
|
from utils import undoReactionCollectionEntry
|
2021-07-31 11:56:28 +00:00
|
|
|
from utils import hasGroupType
|
2021-08-14 11:13:39 +00:00
|
|
|
from utils import localActorUrl
|
2021-10-13 10:11:02 +00:00
|
|
|
from utils import hasObjectStringType
|
2020-12-22 10:30:52 +00:00
|
|
|
from categories import getHashtagCategories
|
|
|
|
from categories import setHashtagCategory
|
2021-11-23 11:41:40 +00:00
|
|
|
from httpsig import getDigestAlgorithmFromHeaders
|
2019-07-04 12:23:53 +00:00
|
|
|
from httpsig import verifyPostHeaders
|
|
|
|
from session import createSession
|
2021-09-03 22:04:50 +00:00
|
|
|
from follow import followerApprovalActive
|
2021-02-24 09:54:37 +00:00
|
|
|
from follow import isFollowingActor
|
2019-07-04 12:23:53 +00:00
|
|
|
from follow import receiveFollowRequest
|
2019-07-08 18:55:39 +00:00
|
|
|
from follow import getFollowersOfActor
|
2020-12-22 13:57:24 +00:00
|
|
|
from follow import unfollowerOfAccount
|
2019-07-04 14:36:29 +00:00
|
|
|
from pprint import pprint
|
2019-07-04 20:25:19 +00:00
|
|
|
from cache import storePersonInCache
|
2021-07-31 11:56:28 +00:00
|
|
|
from cache import getPersonPubKey
|
2019-07-06 15:17:21 +00:00
|
|
|
from acceptreject import receiveAcceptReject
|
2019-11-17 14:02:59 +00:00
|
|
|
from bookmarks import updateBookmarksCollection
|
2019-11-17 14:01:49 +00:00
|
|
|
from bookmarks import undoBookmarksCollectionEntry
|
2019-07-14 20:12:05 +00:00
|
|
|
from blocking import isBlocked
|
2019-10-17 13:18:21 +00:00
|
|
|
from blocking import isBlockedDomain
|
2021-02-15 22:26:25 +00:00
|
|
|
from blocking import brochModeLapses
|
2019-07-14 20:50:27 +00:00
|
|
|
from filters import isFiltered
|
2020-12-22 13:57:24 +00:00
|
|
|
from utils import updateAnnounceCollection
|
|
|
|
from utils import undoAnnounceCollectionEntry
|
2021-01-31 11:05:17 +00:00
|
|
|
from utils import dangerousMarkup
|
2021-03-09 13:52:02 +00:00
|
|
|
from utils import isDM
|
|
|
|
from utils import isReply
|
2021-10-13 09:33:15 +00:00
|
|
|
from utils import hasActor
|
2019-08-16 17:19:23 +00:00
|
|
|
from httpsig import messageContentDigest
|
2021-10-14 15:12:35 +00:00
|
|
|
from posts import editedPostFilename
|
2021-09-18 09:57:03 +00:00
|
|
|
from posts import savePostToBox
|
2021-09-11 14:30:37 +00:00
|
|
|
from posts import isCreateInsideAnnounce
|
2021-02-24 11:01:44 +00:00
|
|
|
from posts import createDirectMessagePost
|
2020-08-25 19:45:15 +00:00
|
|
|
from posts import validContentWarning
|
2019-09-29 09:15:10 +00:00
|
|
|
from posts import downloadAnnounce
|
2020-08-27 17:40:09 +00:00
|
|
|
from posts import isMuted
|
2019-10-22 20:30:43 +00:00
|
|
|
from posts import isImageMedia
|
2019-10-04 12:39:46 +00:00
|
|
|
from posts import sendSignedJson
|
2019-11-29 22:02:16 +00:00
|
|
|
from posts import sendToFollowersThread
|
2020-11-28 10:54:48 +00:00
|
|
|
from webapp_post import individualPostAsHtml
|
2019-11-29 18:46:21 +00:00
|
|
|
from question import questionUpdateVotes
|
2020-01-15 10:56:39 +00:00
|
|
|
from media import replaceYouTube
|
2021-09-18 17:08:14 +00:00
|
|
|
from media import replaceTwitter
|
2020-05-02 11:08:38 +00:00
|
|
|
from git import isGitPatch
|
|
|
|
from git import receiveGitPatch
|
2020-07-03 19:20:31 +00:00
|
|
|
from followingCalendar import receivingCalendarEvents
|
2020-08-20 16:51:48 +00:00
|
|
|
from happening import saveEventPost
|
2020-11-09 19:41:01 +00:00
|
|
|
from delete import removeOldHashtags
|
2020-12-22 10:30:52 +00:00
|
|
|
from categories import guessHashtagCategory
|
2021-01-05 20:11:16 +00:00
|
|
|
from context import hasValidContext
|
2021-03-03 19:06:18 +00:00
|
|
|
from speaker import updateSpeaker
|
2021-06-03 13:21:57 +00:00
|
|
|
from announce import isSelfAnnounce
|
2021-08-02 14:19:09 +00:00
|
|
|
from announce import createAnnounce
|
2021-07-06 20:38:08 +00:00
|
|
|
from notifyOnPost import notifyWhenPersonPosts
|
2021-08-12 10:22:04 +00:00
|
|
|
from conversation import updateConversation
|
2021-09-15 19:48:01 +00:00
|
|
|
from content import validHashTag
|
2021-10-20 13:33:34 +00:00
|
|
|
from webapp_hashtagswarm import htmlHashTagSwarm
|
2020-12-05 11:11:32 +00:00
|
|
|
|
|
|
|
|
2021-10-18 15:20:22 +00:00
|
|
|
def _storeLastPostId(baseDir: str, nickname: str, domain: str,
|
|
|
|
postJsonObject: {}) -> None:
|
|
|
|
"""Stores the id of the last post made by an actor
|
2021-10-18 19:42:31 +00:00
|
|
|
When a new post arrives this allows it to be compared against the last
|
|
|
|
to see if it is an edited post.
|
|
|
|
It would be great if edited posts contained a back reference id to the
|
|
|
|
source but we don't live in that ideal world.
|
2021-10-18 15:20:22 +00:00
|
|
|
"""
|
|
|
|
actor = postId = None
|
|
|
|
if hasObjectDict(postJsonObject):
|
|
|
|
if postJsonObject['object'].get('attributedTo'):
|
|
|
|
if isinstance(postJsonObject['object']['attributedTo'], str):
|
|
|
|
actor = postJsonObject['object']['attributedTo']
|
|
|
|
postId = removeIdEnding(postJsonObject['object']['id'])
|
|
|
|
if not actor:
|
|
|
|
actor = postJsonObject['actor']
|
2021-10-18 19:43:42 +00:00
|
|
|
postId = removeIdEnding(postJsonObject['id'])
|
2021-10-18 15:20:22 +00:00
|
|
|
if not actor:
|
|
|
|
return
|
|
|
|
lastpostDir = acctDir(baseDir, nickname, domain) + '/lastpost'
|
|
|
|
if not os.path.isdir(lastpostDir):
|
|
|
|
os.mkdir(lastpostDir)
|
|
|
|
actorFilename = lastpostDir + '/' + actor.replace('/', '#')
|
|
|
|
try:
|
|
|
|
with open(actorFilename, 'w+') as fp:
|
|
|
|
fp.write(postId)
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: Unable to write last post id to ' + actorFilename)
|
2021-10-18 15:20:22 +00:00
|
|
|
|
|
|
|
|
2021-10-20 13:33:34 +00:00
|
|
|
def _updateCachedHashtagSwarm(baseDir: str, nickname: str, domain: str,
|
|
|
|
httpPrefix: str, domainFull: str,
|
|
|
|
translate: {}) -> bool:
|
|
|
|
"""Updates the hashtag swarm stored as a file
|
|
|
|
"""
|
|
|
|
cachedHashtagSwarmFilename = \
|
|
|
|
acctDir(baseDir, nickname, domain) + '/.hashtagSwarm'
|
|
|
|
saveSwarm = True
|
|
|
|
if os.path.isfile(cachedHashtagSwarmFilename):
|
|
|
|
lastModified = fileLastModified(cachedHashtagSwarmFilename)
|
|
|
|
modifiedDate = None
|
|
|
|
try:
|
|
|
|
modifiedDate = \
|
2021-10-20 14:35:56 +00:00
|
|
|
datetime.datetime.strptime(lastModified, "%Y-%m-%dT%H:%M:%SZ")
|
2021-10-20 13:33:34 +00:00
|
|
|
except BaseException:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: unable to parse last modified cache date ' +
|
2021-10-20 13:33:34 +00:00
|
|
|
str(lastModified))
|
|
|
|
pass
|
|
|
|
if modifiedDate:
|
|
|
|
currDate = datetime.datetime.utcnow()
|
|
|
|
timeDiff = currDate - modifiedDate
|
2021-10-20 14:20:27 +00:00
|
|
|
diffMins = int(timeDiff.total_seconds() / 60)
|
2021-10-20 13:33:34 +00:00
|
|
|
if diffMins < 10:
|
|
|
|
# was saved recently, so don't save again
|
|
|
|
# This avoids too much disk I/O
|
|
|
|
saveSwarm = False
|
|
|
|
else:
|
|
|
|
print('Updating cached hashtag swarm, last changed ' +
|
|
|
|
str(diffMins) + ' minutes ago')
|
2021-10-20 14:53:00 +00:00
|
|
|
else:
|
|
|
|
print('WARN: no modified date for ' + str(lastModified))
|
2021-10-20 13:33:34 +00:00
|
|
|
if saveSwarm:
|
2021-10-23 13:43:21 +00:00
|
|
|
actor = localActorUrl(httpPrefix, nickname, domainFull)
|
2021-10-20 13:33:34 +00:00
|
|
|
newSwarmStr = htmlHashTagSwarm(baseDir, actor, translate)
|
|
|
|
if newSwarmStr:
|
|
|
|
try:
|
|
|
|
with open(cachedHashtagSwarmFilename, 'w+') as fp:
|
|
|
|
fp.write(newSwarmStr)
|
|
|
|
return True
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: unable to write cached hashtag swarm ' +
|
2021-10-20 13:33:34 +00:00
|
|
|
cachedHashtagSwarmFilename)
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
|
|
|
def storeHashTags(baseDir: str, nickname: str, domain: str,
|
|
|
|
httpPrefix: str, domainFull: str,
|
|
|
|
postJsonObject: {}, translate: {}) -> None:
|
2019-12-12 17:34:31 +00:00
|
|
|
"""Extracts hashtags from an incoming post and updates the
|
|
|
|
relevant tags files.
|
|
|
|
"""
|
|
|
|
if not isPublicPost(postJsonObject):
|
|
|
|
return
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(postJsonObject):
|
2019-12-12 17:34:31 +00:00
|
|
|
return
|
|
|
|
if not postJsonObject['object'].get('tag'):
|
|
|
|
return
|
|
|
|
if not postJsonObject.get('id'):
|
|
|
|
return
|
|
|
|
if not isinstance(postJsonObject['object']['tag'], list):
|
|
|
|
return
|
2020-09-03 09:09:58 +00:00
|
|
|
tagsDir = baseDir + '/tags'
|
|
|
|
|
|
|
|
# add tags directory if it doesn't exist
|
|
|
|
if not os.path.isdir(tagsDir):
|
|
|
|
print('Creating tags directory')
|
|
|
|
os.mkdir(tagsDir)
|
|
|
|
|
2020-12-05 11:11:32 +00:00
|
|
|
hashtagCategories = getHashtagCategories(baseDir)
|
|
|
|
|
2021-10-20 13:33:34 +00:00
|
|
|
hashtagsCtr = 0
|
2019-12-12 17:34:31 +00:00
|
|
|
for tag in postJsonObject['object']['tag']:
|
|
|
|
if not tag.get('type'):
|
|
|
|
continue
|
2021-02-08 15:06:26 +00:00
|
|
|
if not isinstance(tag['type'], str):
|
|
|
|
continue
|
2020-04-03 16:27:34 +00:00
|
|
|
if tag['type'] != 'Hashtag':
|
2019-12-12 17:34:31 +00:00
|
|
|
continue
|
|
|
|
if not tag.get('name'):
|
|
|
|
continue
|
2020-04-03 16:27:34 +00:00
|
|
|
tagName = tag['name'].replace('#', '').strip()
|
2021-09-15 19:48:01 +00:00
|
|
|
if not validHashTag(tagName):
|
|
|
|
continue
|
2020-04-03 16:27:34 +00:00
|
|
|
tagsFilename = tagsDir + '/' + tagName + '.txt'
|
2020-08-23 11:13:35 +00:00
|
|
|
postUrl = removeIdEnding(postJsonObject['id'])
|
2020-04-03 16:27:34 +00:00
|
|
|
postUrl = postUrl.replace('/', '#')
|
|
|
|
daysDiff = datetime.datetime.utcnow() - datetime.datetime(1970, 1, 1)
|
|
|
|
daysSinceEpoch = daysDiff.days
|
|
|
|
tagline = str(daysSinceEpoch) + ' ' + nickname + ' ' + postUrl + '\n'
|
2021-10-20 13:33:34 +00:00
|
|
|
hashtagsCtr += 1
|
2019-12-12 17:34:31 +00:00
|
|
|
if not os.path.isfile(tagsFilename):
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(tagsFilename, 'w+') as tagsFile:
|
|
|
|
tagsFile.write(tagline)
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + tagsFilename)
|
2019-12-12 17:34:31 +00:00
|
|
|
else:
|
|
|
|
if postUrl not in open(tagsFilename).read():
|
2019-12-12 17:47:16 +00:00
|
|
|
try:
|
|
|
|
with open(tagsFilename, 'r+') as tagsFile:
|
2020-04-03 16:27:34 +00:00
|
|
|
content = tagsFile.read()
|
2020-12-29 20:22:28 +00:00
|
|
|
if tagline not in content:
|
|
|
|
tagsFile.seek(0, 0)
|
|
|
|
tagsFile.write(tagline + content)
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError as e:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: Failed to write entry to tags file ' +
|
2020-04-03 16:27:34 +00:00
|
|
|
tagsFilename + ' ' + str(e))
|
2020-05-31 16:31:33 +00:00
|
|
|
removeOldHashtags(baseDir, 3)
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-05 11:11:32 +00:00
|
|
|
# automatically assign a category to the tag if possible
|
|
|
|
categoryFilename = tagsDir + '/' + tagName + '.category'
|
|
|
|
if not os.path.isfile(categoryFilename):
|
|
|
|
categoryStr = \
|
|
|
|
guessHashtagCategory(tagName, hashtagCategories)
|
|
|
|
if categoryStr:
|
2021-08-15 11:39:20 +00:00
|
|
|
setHashtagCategory(baseDir, tagName, categoryStr, False)
|
2020-12-05 11:11:32 +00:00
|
|
|
|
2021-10-20 13:33:34 +00:00
|
|
|
# if some hashtags were found then recalculate the swarm
|
|
|
|
# ready for later display
|
|
|
|
if hashtagsCtr > 0:
|
|
|
|
_updateCachedHashtagSwarm(baseDir, nickname, domain,
|
|
|
|
httpPrefix, domainFull, translate)
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _inboxStorePostToHtmlCache(recentPostsCache: {}, maxRecentPosts: int,
|
|
|
|
translate: {},
|
|
|
|
baseDir: str, httpPrefix: str,
|
|
|
|
session, cachedWebfingers: {}, personCache: {},
|
|
|
|
nickname: str, domain: str, port: int,
|
|
|
|
postJsonObject: {},
|
|
|
|
allowDeletion: bool, boxname: str,
|
2020-12-23 23:59:49 +00:00
|
|
|
showPublishedDateOnly: bool,
|
2021-01-30 11:47:09 +00:00
|
|
|
peertubeInstances: [],
|
2021-03-09 19:52:10 +00:00
|
|
|
allowLocalNetworkAccess: bool,
|
2021-08-03 10:04:45 +00:00
|
|
|
themeName: str, systemLanguage: str,
|
2021-08-31 14:17:11 +00:00
|
|
|
maxLikeCount: int,
|
2021-10-21 13:08:21 +00:00
|
|
|
signingPrivateKeyPem: str,
|
2021-10-21 19:00:25 +00:00
|
|
|
CWlists: {},
|
|
|
|
listsEnabled: str) -> None:
|
2019-10-19 12:37:35 +00:00
|
|
|
"""Converts the json post into html and stores it in a cache
|
|
|
|
This enables the post to be quickly displayed later
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
pageNumber = -999
|
|
|
|
avatarUrl = None
|
2021-07-01 21:30:36 +00:00
|
|
|
if boxname != 'outbox':
|
2020-10-08 12:28:02 +00:00
|
|
|
boxname = 'inbox'
|
2020-12-18 18:12:33 +00:00
|
|
|
|
2021-09-03 22:04:50 +00:00
|
|
|
notDM = not isDM(postJsonObject)
|
2021-09-18 19:17:10 +00:00
|
|
|
YTReplacementDomain = getConfigParam(baseDir, 'youtubedomain')
|
|
|
|
twitterReplacementDomain = getConfigParam(baseDir, 'twitterdomain')
|
2021-08-31 14:17:11 +00:00
|
|
|
individualPostAsHtml(signingPrivateKeyPem,
|
|
|
|
True, recentPostsCache, maxRecentPosts,
|
2020-12-09 13:31:54 +00:00
|
|
|
translate, pageNumber,
|
2020-12-31 12:23:15 +00:00
|
|
|
baseDir, session, cachedWebfingers,
|
|
|
|
personCache,
|
2020-04-03 16:27:34 +00:00
|
|
|
nickname, domain, port, postJsonObject,
|
|
|
|
avatarUrl, True, allowDeletion,
|
2021-09-18 19:17:10 +00:00
|
|
|
httpPrefix, __version__, boxname,
|
|
|
|
YTReplacementDomain, twitterReplacementDomain,
|
2020-10-11 18:50:13 +00:00
|
|
|
showPublishedDateOnly,
|
2021-01-30 11:47:09 +00:00
|
|
|
peertubeInstances, allowLocalNetworkAccess,
|
2021-08-03 10:04:45 +00:00
|
|
|
themeName, systemLanguage, maxLikeCount,
|
2021-10-21 13:08:21 +00:00
|
|
|
notDM, True, True, False, True, False,
|
2021-10-21 19:00:25 +00:00
|
|
|
CWlists, listsEnabled)
|
2020-04-03 16:27:34 +00:00
|
|
|
|
|
|
|
|
|
|
|
def validInbox(baseDir: str, nickname: str, domain: str) -> bool:
|
2019-07-18 11:35:48 +00:00
|
|
|
"""Checks whether files were correctly saved to the inbox
|
|
|
|
"""
|
2021-06-23 21:31:50 +00:00
|
|
|
domain = removeDomainPort(domain)
|
2021-07-13 21:59:53 +00:00
|
|
|
inboxDir = acctDir(baseDir, nickname, domain) + '/inbox'
|
2019-07-18 09:26:47 +00:00
|
|
|
if not os.path.isdir(inboxDir):
|
|
|
|
return True
|
|
|
|
for subdir, dirs, files in os.walk(inboxDir):
|
|
|
|
for f in files:
|
2020-04-03 16:27:34 +00:00
|
|
|
filename = os.path.join(subdir, f)
|
2019-07-18 09:26:47 +00:00
|
|
|
if not os.path.isfile(filename):
|
2020-04-03 16:27:34 +00:00
|
|
|
print('filename: ' + filename)
|
2019-07-18 09:26:47 +00:00
|
|
|
return False
|
|
|
|
if 'postNickname' in open(filename).read():
|
2020-04-03 16:27:34 +00:00
|
|
|
print('queue file incorrectly saved to ' + filename)
|
2019-07-18 11:35:48 +00:00
|
|
|
return False
|
2020-12-13 22:13:45 +00:00
|
|
|
break
|
2020-03-22 21:16:02 +00:00
|
|
|
return True
|
2019-07-18 11:35:48 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
|
|
|
def validInboxFilenames(baseDir: str, nickname: str, domain: str,
|
|
|
|
expectedDomain: str, expectedPort: int) -> bool:
|
2019-07-18 11:35:48 +00:00
|
|
|
"""Used by unit tests to check that the port number gets appended to
|
|
|
|
domain names within saved post filenames
|
|
|
|
"""
|
2021-06-23 21:31:50 +00:00
|
|
|
domain = removeDomainPort(domain)
|
2021-07-13 21:59:53 +00:00
|
|
|
inboxDir = acctDir(baseDir, nickname, domain) + '/inbox'
|
2019-07-18 11:35:48 +00:00
|
|
|
if not os.path.isdir(inboxDir):
|
2021-08-01 13:25:11 +00:00
|
|
|
print('Not an inbox directory: ' + inboxDir)
|
2019-07-18 11:35:48 +00:00
|
|
|
return True
|
2020-04-03 16:27:34 +00:00
|
|
|
expectedStr = expectedDomain + ':' + str(expectedPort)
|
2021-08-01 13:25:11 +00:00
|
|
|
expectedFound = False
|
2021-08-20 11:22:20 +00:00
|
|
|
ctr = 0
|
2019-07-18 11:35:48 +00:00
|
|
|
for subdir, dirs, files in os.walk(inboxDir):
|
|
|
|
for f in files:
|
2020-04-03 16:27:34 +00:00
|
|
|
filename = os.path.join(subdir, f)
|
2021-08-20 11:22:20 +00:00
|
|
|
ctr += 1
|
2019-07-18 11:35:48 +00:00
|
|
|
if not os.path.isfile(filename):
|
2020-04-03 16:27:34 +00:00
|
|
|
print('filename: ' + filename)
|
2019-07-18 11:35:48 +00:00
|
|
|
return False
|
2021-08-01 13:25:11 +00:00
|
|
|
if expectedStr in filename:
|
|
|
|
expectedFound = True
|
2020-12-13 22:13:45 +00:00
|
|
|
break
|
2021-08-20 11:22:20 +00:00
|
|
|
if ctr == 0:
|
|
|
|
return True
|
2021-08-01 13:25:11 +00:00
|
|
|
if not expectedFound:
|
|
|
|
print('Expected file was not found: ' + expectedStr)
|
2021-08-20 11:22:20 +00:00
|
|
|
for subdir, dirs, files in os.walk(inboxDir):
|
|
|
|
for f in files:
|
|
|
|
filename = os.path.join(subdir, f)
|
|
|
|
print(filename)
|
|
|
|
break
|
2021-08-01 13:25:11 +00:00
|
|
|
return False
|
2020-03-22 21:16:02 +00:00
|
|
|
return True
|
2019-07-18 09:26:47 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2019-07-02 15:07:27 +00:00
|
|
|
def inboxMessageHasParams(messageJson: {}) -> bool:
|
|
|
|
"""Checks whether an incoming message contains expected parameters
|
|
|
|
"""
|
2020-08-23 14:45:58 +00:00
|
|
|
expectedParams = ['actor', 'type', 'object']
|
2019-07-02 15:07:27 +00:00
|
|
|
for param in expectedParams:
|
|
|
|
if not messageJson.get(param):
|
2020-08-23 14:45:58 +00:00
|
|
|
# print('inboxMessageHasParams: ' +
|
|
|
|
# param + ' ' + str(messageJson))
|
2019-07-02 15:07:27 +00:00
|
|
|
return False
|
2021-02-08 15:06:26 +00:00
|
|
|
|
|
|
|
# actor should be a string
|
|
|
|
if not isinstance(messageJson['actor'], str):
|
|
|
|
print('WARN: actor should be a string, but is actually: ' +
|
|
|
|
str(messageJson['actor']))
|
2021-08-19 22:31:25 +00:00
|
|
|
pprint(messageJson)
|
2021-02-08 15:06:26 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
# type should be a string
|
|
|
|
if not isinstance(messageJson['type'], str):
|
|
|
|
print('WARN: type from ' + str(messageJson['actor']) +
|
|
|
|
' should be a string, but is actually: ' +
|
|
|
|
str(messageJson['type']))
|
|
|
|
return False
|
|
|
|
|
|
|
|
# object should be a dict or a string
|
2021-07-06 10:00:19 +00:00
|
|
|
if not hasObjectDict(messageJson):
|
2021-02-08 15:06:26 +00:00
|
|
|
if not isinstance(messageJson['object'], str):
|
|
|
|
print('WARN: object from ' + str(messageJson['actor']) +
|
|
|
|
' should be a dict or string, but is actually: ' +
|
|
|
|
str(messageJson['object']))
|
|
|
|
return False
|
|
|
|
|
2019-07-06 13:49:25 +00:00
|
|
|
if not messageJson.get('to'):
|
2021-11-10 12:16:03 +00:00
|
|
|
allowedWithoutToParam = ['Like', 'EmojiReact',
|
|
|
|
'Follow', 'Join', 'Request',
|
2020-04-03 16:27:34 +00:00
|
|
|
'Accept', 'Capability', 'Undo']
|
2019-07-06 13:49:25 +00:00
|
|
|
if messageJson['type'] not in allowedWithoutToParam:
|
|
|
|
return False
|
2019-07-02 15:07:27 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
|
|
|
def inboxPermittedMessage(domain: str, messageJson: {},
|
|
|
|
federationList: []) -> bool:
|
2019-06-28 21:59:54 +00:00
|
|
|
""" check that we are receiving from a permitted domain
|
|
|
|
"""
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, False):
|
2019-06-28 21:59:54 +00:00
|
|
|
return False
|
2020-08-23 14:45:58 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
actor = messageJson['actor']
|
2019-06-28 21:59:54 +00:00
|
|
|
# always allow the local domain
|
2019-07-01 11:48:54 +00:00
|
|
|
if domain in actor:
|
2019-06-28 21:59:54 +00:00
|
|
|
return True
|
|
|
|
|
2020-09-27 19:27:24 +00:00
|
|
|
if not urlPermitted(actor, federationList):
|
2019-06-28 21:59:54 +00:00
|
|
|
return False
|
|
|
|
|
2021-11-10 12:16:03 +00:00
|
|
|
alwaysAllowedTypes = (
|
|
|
|
'Follow', 'Join', 'Like', 'EmojiReact', 'Delete', 'Announce'
|
|
|
|
)
|
2019-11-16 12:30:59 +00:00
|
|
|
if messageJson['type'] not in alwaysAllowedTypes:
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(messageJson):
|
2019-11-16 12:32:28 +00:00
|
|
|
return True
|
|
|
|
if messageJson['object'].get('inReplyTo'):
|
2020-04-03 16:27:34 +00:00
|
|
|
inReplyTo = messageJson['object']['inReplyTo']
|
2020-08-28 14:45:07 +00:00
|
|
|
if not isinstance(inReplyTo, str):
|
|
|
|
return False
|
2020-09-27 19:27:24 +00:00
|
|
|
if not urlPermitted(inReplyTo, federationList):
|
2019-07-15 09:20:16 +00:00
|
|
|
return False
|
2019-06-28 21:59:54 +00:00
|
|
|
|
|
|
|
return True
|
2019-06-29 10:08:59 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
|
|
|
def savePostToInboxQueue(baseDir: str, httpPrefix: str,
|
|
|
|
nickname: str, domain: str,
|
|
|
|
postJsonObject: {},
|
2021-01-03 18:20:25 +00:00
|
|
|
originalPostJsonObject: {},
|
2020-04-03 16:27:34 +00:00
|
|
|
messageBytes: str,
|
|
|
|
httpHeaders: {},
|
2021-06-21 14:32:17 +00:00
|
|
|
postPath: str, debug: bool,
|
2021-07-18 14:15:16 +00:00
|
|
|
blockedCache: [], systemLanguage: str) -> str:
|
2021-07-20 13:33:27 +00:00
|
|
|
"""Saves the given json to the inbox queue for the person
|
2019-07-04 10:02:56 +00:00
|
|
|
keyId specifies the actor sending the post
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
if len(messageBytes) > 10240:
|
|
|
|
print('WARN: inbox message too long ' +
|
|
|
|
str(len(messageBytes)) + ' bytes')
|
2019-11-15 23:43:07 +00:00
|
|
|
return None
|
2020-04-03 16:27:34 +00:00
|
|
|
originalDomain = domain
|
2021-06-23 21:31:50 +00:00
|
|
|
domain = removeDomainPort(domain)
|
2019-07-14 20:12:05 +00:00
|
|
|
|
|
|
|
# block at the ealiest stage possible, which means the data
|
|
|
|
# isn't written to file
|
2020-04-03 16:27:34 +00:00
|
|
|
postNickname = None
|
|
|
|
postDomain = None
|
|
|
|
actor = None
|
2019-07-14 20:12:05 +00:00
|
|
|
if postJsonObject.get('actor'):
|
2020-09-05 16:46:03 +00:00
|
|
|
if not isinstance(postJsonObject['actor'], str):
|
|
|
|
return None
|
2020-04-03 16:27:34 +00:00
|
|
|
actor = postJsonObject['actor']
|
|
|
|
postNickname = getNicknameFromActor(postJsonObject['actor'])
|
2019-09-01 19:20:28 +00:00
|
|
|
if not postNickname:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('No post Nickname in actor ' + postJsonObject['actor'])
|
2019-09-01 19:20:28 +00:00
|
|
|
return None
|
2020-04-03 16:27:34 +00:00
|
|
|
postDomain, postPort = getDomainFromActor(postJsonObject['actor'])
|
2019-09-01 19:20:28 +00:00
|
|
|
if not postDomain:
|
2019-10-29 20:23:49 +00:00
|
|
|
if debug:
|
|
|
|
pprint(postJsonObject)
|
2019-09-01 19:20:28 +00:00
|
|
|
print('No post Domain in actor')
|
|
|
|
return None
|
2021-06-21 14:32:17 +00:00
|
|
|
if isBlocked(baseDir, nickname, domain,
|
|
|
|
postNickname, postDomain, blockedCache):
|
2019-08-18 09:39:12 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: post from ' + postNickname + ' blocked')
|
2019-07-14 20:12:05 +00:00
|
|
|
return None
|
2020-12-16 10:48:40 +00:00
|
|
|
postDomain = getFullDomain(postDomain, postPort)
|
2019-07-14 20:50:27 +00:00
|
|
|
|
2021-06-22 15:45:59 +00:00
|
|
|
if hasObjectDict(postJsonObject):
|
|
|
|
if postJsonObject['object'].get('inReplyTo'):
|
|
|
|
if isinstance(postJsonObject['object']['inReplyTo'], str):
|
|
|
|
inReplyTo = \
|
|
|
|
postJsonObject['object']['inReplyTo']
|
|
|
|
replyDomain, replyPort = \
|
|
|
|
getDomainFromActor(inReplyTo)
|
|
|
|
if isBlockedDomain(baseDir, replyDomain, blockedCache):
|
|
|
|
if debug:
|
|
|
|
print('WARN: post contains reply from ' +
|
|
|
|
str(actor) +
|
|
|
|
' to a blocked domain: ' + replyDomain)
|
|
|
|
return None
|
|
|
|
else:
|
|
|
|
replyNickname = \
|
|
|
|
getNicknameFromActor(inReplyTo)
|
|
|
|
if replyNickname and replyDomain:
|
|
|
|
if isBlocked(baseDir, nickname, domain,
|
|
|
|
replyNickname, replyDomain,
|
|
|
|
blockedCache):
|
|
|
|
if debug:
|
|
|
|
print('WARN: post contains reply from ' +
|
|
|
|
str(actor) +
|
|
|
|
' to a blocked account: ' +
|
|
|
|
replyNickname + '@' + replyDomain)
|
|
|
|
return None
|
|
|
|
if postJsonObject['object'].get('content'):
|
2021-07-20 13:33:27 +00:00
|
|
|
contentStr = getBaseContentFromPost(postJsonObject, systemLanguage)
|
2021-07-18 14:15:16 +00:00
|
|
|
if contentStr:
|
|
|
|
if isFiltered(baseDir, nickname, domain, contentStr):
|
2021-06-22 15:45:59 +00:00
|
|
|
if debug:
|
|
|
|
print('WARN: post was filtered out due to content')
|
|
|
|
return None
|
2020-04-03 16:27:34 +00:00
|
|
|
originalPostId = None
|
2019-07-14 16:57:06 +00:00
|
|
|
if postJsonObject.get('id'):
|
2020-09-05 16:46:03 +00:00
|
|
|
if not isinstance(postJsonObject['id'], str):
|
|
|
|
return None
|
2020-08-23 11:13:35 +00:00
|
|
|
originalPostId = removeIdEnding(postJsonObject['id'])
|
2019-08-16 15:04:40 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
currTime = datetime.datetime.utcnow()
|
2019-08-16 15:04:40 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
postId = None
|
2019-08-16 15:04:40 +00:00
|
|
|
if postJsonObject.get('id'):
|
2020-08-23 11:13:35 +00:00
|
|
|
postId = removeIdEnding(postJsonObject['id'])
|
2020-04-03 16:27:34 +00:00
|
|
|
published = currTime.strftime("%Y-%m-%dT%H:%M:%SZ")
|
2019-08-16 15:04:40 +00:00
|
|
|
if not postId:
|
2020-04-03 16:27:34 +00:00
|
|
|
statusNumber, published = getStatusNumber()
|
2019-08-16 15:04:40 +00:00
|
|
|
if actor:
|
2020-04-03 16:27:34 +00:00
|
|
|
postId = actor + '/statuses/' + statusNumber
|
2019-08-16 15:04:40 +00:00
|
|
|
else:
|
2021-08-14 11:13:39 +00:00
|
|
|
postId = localActorUrl(httpPrefix, nickname, originalDomain) + \
|
|
|
|
'/statuses/' + statusNumber
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2019-08-16 13:47:01 +00:00
|
|
|
# NOTE: don't change postJsonObject['id'] before signature check
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
inboxQueueDir = createInboxQueueDir(nickname, domain, baseDir)
|
|
|
|
|
|
|
|
handle = nickname + '@' + domain
|
|
|
|
destination = baseDir + '/accounts/' + \
|
|
|
|
handle + '/inbox/' + postId.replace('/', '#') + '.json'
|
|
|
|
filename = inboxQueueDir + '/' + postId.replace('/', '#') + '.json'
|
|
|
|
|
|
|
|
sharedInboxItem = False
|
|
|
|
if nickname == 'inbox':
|
|
|
|
nickname = originalDomain
|
|
|
|
sharedInboxItem = True
|
|
|
|
|
|
|
|
digestStartTime = time.time()
|
2021-11-23 11:41:40 +00:00
|
|
|
digestAlgorithm = getDigestAlgorithmFromHeaders(httpHeaders)
|
|
|
|
digest = messageContentDigest(messageBytes, digestAlgorithm)
|
2020-04-03 16:27:34 +00:00
|
|
|
timeDiffStr = str(int((time.time() - digestStartTime) * 1000))
|
2019-11-16 10:12:40 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
while len(timeDiffStr) < 6:
|
|
|
|
timeDiffStr = '0' + timeDiffStr
|
|
|
|
print('DIGEST|' + timeDiffStr + '|' + filename)
|
2019-11-16 10:07:32 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
newQueueItem = {
|
2019-08-16 09:35:06 +00:00
|
|
|
'originalId': originalPostId,
|
2019-07-15 09:20:16 +00:00
|
|
|
'id': postId,
|
2019-08-16 09:35:06 +00:00
|
|
|
'actor': actor,
|
2019-07-07 15:51:04 +00:00
|
|
|
'nickname': nickname,
|
|
|
|
'domain': domain,
|
2019-07-15 10:22:19 +00:00
|
|
|
'postNickname': postNickname,
|
|
|
|
'postDomain': postDomain,
|
2019-07-05 11:39:03 +00:00
|
|
|
'sharedInbox': sharedInboxItem,
|
2019-07-04 10:09:27 +00:00
|
|
|
'published': published,
|
2019-08-15 21:34:25 +00:00
|
|
|
'httpHeaders': httpHeaders,
|
2019-07-05 22:13:20 +00:00
|
|
|
'path': postPath,
|
2019-07-14 16:57:06 +00:00
|
|
|
'post': postJsonObject,
|
2021-01-03 18:20:25 +00:00
|
|
|
'original': originalPostJsonObject,
|
2019-11-16 10:07:32 +00:00
|
|
|
'digest': digest,
|
2019-07-04 10:19:15 +00:00
|
|
|
'filename': filename,
|
2019-08-05 09:50:45 +00:00
|
|
|
'destination': destination
|
2019-07-04 10:02:56 +00:00
|
|
|
}
|
2019-07-06 13:49:25 +00:00
|
|
|
|
|
|
|
if debug:
|
|
|
|
print('Inbox queue item created')
|
2020-04-03 16:27:34 +00:00
|
|
|
saveJson(newQueueItem, filename)
|
2019-07-04 10:02:56 +00:00
|
|
|
return filename
|
2019-07-04 12:23:53 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _inboxPostRecipientsAdd(baseDir: str, httpPrefix: str, toList: [],
|
|
|
|
recipientsDict: {},
|
|
|
|
domainMatch: str, domain: str,
|
|
|
|
actor: str, debug: bool) -> bool:
|
2019-07-08 22:12:24 +00:00
|
|
|
"""Given a list of post recipients (toList) from 'to' or 'cc' parameters
|
2020-09-27 18:35:35 +00:00
|
|
|
populate a recipientsDict with the handle for each
|
2019-07-08 22:12:24 +00:00
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
followerRecipients = False
|
2019-07-08 22:12:24 +00:00
|
|
|
for recipient in toList:
|
2019-09-03 19:53:22 +00:00
|
|
|
if not recipient:
|
|
|
|
continue
|
2019-07-08 22:12:24 +00:00
|
|
|
# is this a to a local account?
|
|
|
|
if domainMatch in recipient:
|
|
|
|
# get the handle for the local account
|
2020-04-03 16:27:34 +00:00
|
|
|
nickname = recipient.split(domainMatch)[1]
|
2021-06-22 12:42:52 +00:00
|
|
|
handle = nickname + '@' + domain
|
2020-04-03 16:27:34 +00:00
|
|
|
if os.path.isdir(baseDir + '/accounts/' + handle):
|
2020-09-27 18:35:35 +00:00
|
|
|
recipientsDict[handle] = None
|
2019-07-11 12:29:31 +00:00
|
|
|
else:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' + baseDir + '/accounts/' +
|
|
|
|
handle + ' does not exist')
|
2019-07-11 12:29:31 +00:00
|
|
|
else:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' + recipient + ' is not local to ' +
|
|
|
|
domainMatch)
|
2019-07-11 12:29:31 +00:00
|
|
|
print(str(toList))
|
2019-07-08 22:12:24 +00:00
|
|
|
if recipient.endswith('followers'):
|
2019-07-11 12:29:31 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: followers detected as post recipients')
|
2020-04-03 16:27:34 +00:00
|
|
|
followerRecipients = True
|
|
|
|
return followerRecipients, recipientsDict
|
2019-07-08 22:12:24 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _inboxPostRecipients(baseDir: str, postJsonObject: {},
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
debug: bool) -> ([], []):
|
2019-07-10 15:33:19 +00:00
|
|
|
"""Returns dictionaries containing the recipients of the given post
|
|
|
|
The shared dictionary contains followers
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
recipientsDict = {}
|
|
|
|
recipientsDictFollowers = {}
|
2019-07-08 22:12:24 +00:00
|
|
|
|
|
|
|
if not postJsonObject.get('actor'):
|
2019-07-11 12:29:31 +00:00
|
|
|
if debug:
|
|
|
|
pprint(postJsonObject)
|
|
|
|
print('WARNING: inbox post has no actor')
|
2020-04-03 16:27:34 +00:00
|
|
|
return recipientsDict, recipientsDictFollowers
|
2019-07-08 22:12:24 +00:00
|
|
|
|
2021-06-23 21:31:50 +00:00
|
|
|
domain = removeDomainPort(domain)
|
2020-04-03 16:27:34 +00:00
|
|
|
domainBase = domain
|
2020-12-16 10:48:40 +00:00
|
|
|
domain = getFullDomain(domain, port)
|
2020-04-03 16:27:34 +00:00
|
|
|
domainMatch = '/' + domain + '/users/'
|
2019-07-08 22:12:24 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
actor = postJsonObject['actor']
|
2019-07-08 22:12:24 +00:00
|
|
|
# first get any specific people which the post is addressed to
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
followerRecipients = False
|
2021-06-22 15:45:59 +00:00
|
|
|
if hasObjectDict(postJsonObject):
|
|
|
|
if postJsonObject['object'].get('to'):
|
|
|
|
if isinstance(postJsonObject['object']['to'], list):
|
|
|
|
recipientsList = postJsonObject['object']['to']
|
2019-07-11 12:29:31 +00:00
|
|
|
else:
|
2021-06-22 15:45:59 +00:00
|
|
|
recipientsList = [postJsonObject['object']['to']]
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: resolving "to"')
|
|
|
|
includesFollowers, recipientsDict = \
|
|
|
|
_inboxPostRecipientsAdd(baseDir, httpPrefix,
|
|
|
|
recipientsList,
|
|
|
|
recipientsDict,
|
|
|
|
domainMatch, domainBase,
|
|
|
|
actor, debug)
|
|
|
|
if includesFollowers:
|
|
|
|
followerRecipients = True
|
|
|
|
else:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox post has no "to"')
|
2019-07-08 22:12:24 +00:00
|
|
|
|
2021-06-22 15:45:59 +00:00
|
|
|
if postJsonObject['object'].get('cc'):
|
|
|
|
if isinstance(postJsonObject['object']['cc'], list):
|
|
|
|
recipientsList = postJsonObject['object']['cc']
|
2019-07-11 12:29:31 +00:00
|
|
|
else:
|
2021-06-22 15:45:59 +00:00
|
|
|
recipientsList = [postJsonObject['object']['cc']]
|
|
|
|
includesFollowers, recipientsDict = \
|
|
|
|
_inboxPostRecipientsAdd(baseDir, httpPrefix,
|
|
|
|
recipientsList,
|
|
|
|
recipientsDict,
|
|
|
|
domainMatch, domainBase,
|
|
|
|
actor, debug)
|
|
|
|
if includesFollowers:
|
|
|
|
followerRecipients = True
|
2019-07-11 12:29:31 +00:00
|
|
|
else:
|
|
|
|
if debug:
|
2021-06-22 15:45:59 +00:00
|
|
|
print('DEBUG: inbox post has no cc')
|
|
|
|
else:
|
|
|
|
if debug and postJsonObject.get('object'):
|
|
|
|
if isinstance(postJsonObject['object'], str):
|
|
|
|
if '/statuses/' in postJsonObject['object']:
|
|
|
|
print('DEBUG: inbox item is a link to a post')
|
|
|
|
else:
|
|
|
|
if '/users/' in postJsonObject['object']:
|
|
|
|
print('DEBUG: inbox item is a link to an actor')
|
2019-07-08 22:12:24 +00:00
|
|
|
|
|
|
|
if postJsonObject.get('to'):
|
2019-08-16 17:51:00 +00:00
|
|
|
if isinstance(postJsonObject['to'], list):
|
2020-04-03 16:27:34 +00:00
|
|
|
recipientsList = postJsonObject['to']
|
2019-08-16 17:51:00 +00:00
|
|
|
else:
|
2020-04-03 16:27:34 +00:00
|
|
|
recipientsList = [postJsonObject['to']]
|
|
|
|
includesFollowers, recipientsDict = \
|
2020-12-22 18:06:23 +00:00
|
|
|
_inboxPostRecipientsAdd(baseDir, httpPrefix,
|
|
|
|
recipientsList,
|
|
|
|
recipientsDict,
|
|
|
|
domainMatch, domainBase,
|
|
|
|
actor, debug)
|
2019-07-08 22:12:24 +00:00
|
|
|
if includesFollowers:
|
2020-04-03 16:27:34 +00:00
|
|
|
followerRecipients = True
|
2019-07-08 22:12:24 +00:00
|
|
|
|
|
|
|
if postJsonObject.get('cc'):
|
2019-08-16 17:51:00 +00:00
|
|
|
if isinstance(postJsonObject['cc'], list):
|
2020-04-03 16:27:34 +00:00
|
|
|
recipientsList = postJsonObject['cc']
|
2019-08-16 17:51:00 +00:00
|
|
|
else:
|
2020-04-03 16:27:34 +00:00
|
|
|
recipientsList = [postJsonObject['cc']]
|
|
|
|
includesFollowers, recipientsDict = \
|
2020-12-22 18:06:23 +00:00
|
|
|
_inboxPostRecipientsAdd(baseDir, httpPrefix,
|
|
|
|
recipientsList,
|
|
|
|
recipientsDict,
|
|
|
|
domainMatch, domainBase,
|
|
|
|
actor, debug)
|
2019-07-08 22:12:24 +00:00
|
|
|
if includesFollowers:
|
2020-04-03 16:27:34 +00:00
|
|
|
followerRecipients = True
|
2019-07-08 22:12:24 +00:00
|
|
|
|
|
|
|
if not followerRecipients:
|
2019-07-11 12:29:31 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: no followers were resolved')
|
2020-04-03 16:27:34 +00:00
|
|
|
return recipientsDict, recipientsDictFollowers
|
2019-07-08 22:12:24 +00:00
|
|
|
|
|
|
|
# now resolve the followers
|
2020-04-03 16:27:34 +00:00
|
|
|
recipientsDictFollowers = \
|
|
|
|
getFollowersOfActor(baseDir, actor, debug)
|
|
|
|
|
|
|
|
return recipientsDict, recipientsDictFollowers
|
2019-07-08 22:12:24 +00:00
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveUndoFollow(session, baseDir: str, httpPrefix: str,
|
|
|
|
port: int, messageJson: {},
|
|
|
|
federationList: [],
|
|
|
|
debug: bool) -> bool:
|
2019-07-17 10:34:00 +00:00
|
|
|
if not messageJson['object'].get('actor'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: follow request has no actor within object')
|
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['object']['actor']):
|
2019-07-17 10:34:00 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "users" or "profile" missing ' +
|
|
|
|
'from actor within object')
|
2019-07-17 10:34:00 +00:00
|
|
|
return False
|
|
|
|
if messageJson['object']['actor'] != messageJson['actor']:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: actors do not match')
|
|
|
|
return False
|
2019-08-15 17:05:22 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
nicknameFollower = \
|
|
|
|
getNicknameFromActor(messageJson['object']['actor'])
|
2019-09-02 09:43:43 +00:00
|
|
|
if not nicknameFollower:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('WARN: unable to find nickname in ' +
|
|
|
|
messageJson['object']['actor'])
|
2019-09-02 09:43:43 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
domainFollower, portFollower = \
|
|
|
|
getDomainFromActor(messageJson['object']['actor'])
|
2020-12-16 10:48:40 +00:00
|
|
|
domainFollowerFull = getFullDomain(domainFollower, portFollower)
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
nicknameFollowing = \
|
|
|
|
getNicknameFromActor(messageJson['object']['object'])
|
2019-09-02 09:43:43 +00:00
|
|
|
if not nicknameFollowing:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('WARN: unable to find nickname in ' +
|
|
|
|
messageJson['object']['object'])
|
2019-09-02 09:43:43 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
domainFollowing, portFollowing = \
|
|
|
|
getDomainFromActor(messageJson['object']['object'])
|
2020-12-16 10:48:40 +00:00
|
|
|
domainFollowingFull = getFullDomain(domainFollowing, portFollowing)
|
2019-07-17 10:34:00 +00:00
|
|
|
|
2021-07-31 11:56:28 +00:00
|
|
|
groupAccount = hasGroupType(baseDir, messageJson['object']['actor'], None)
|
2020-12-22 13:57:24 +00:00
|
|
|
if unfollowerOfAccount(baseDir,
|
|
|
|
nicknameFollowing, domainFollowingFull,
|
|
|
|
nicknameFollower, domainFollowerFull,
|
2021-07-30 16:06:34 +00:00
|
|
|
debug, groupAccount):
|
2020-08-20 12:11:07 +00:00
|
|
|
print(nicknameFollowing + '@' + domainFollowingFull + ': '
|
|
|
|
'Follower ' + nicknameFollower + '@' + domainFollowerFull +
|
|
|
|
' was removed')
|
2019-07-17 11:54:13 +00:00
|
|
|
return True
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2019-07-17 11:54:13 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Follower ' +
|
|
|
|
nicknameFollower + '@' + domainFollowerFull +
|
|
|
|
' was not removed')
|
2019-07-17 11:54:13 +00:00
|
|
|
return False
|
2019-07-17 10:34:00 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveUndo(session, baseDir: str, httpPrefix: str,
|
|
|
|
port: int, sendThreads: [], postLog: [],
|
|
|
|
cachedWebfingers: {}, personCache: {},
|
|
|
|
messageJson: {}, federationList: [],
|
|
|
|
debug: bool) -> bool:
|
2019-07-17 10:34:00 +00:00
|
|
|
"""Receives an undo request within the POST section of HTTPServer
|
|
|
|
"""
|
|
|
|
if not messageJson['type'].startswith('Undo'):
|
|
|
|
return False
|
2019-07-17 11:24:11 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: Undo activity received')
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2019-07-17 10:34:00 +00:00
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
2019-07-17 10:34:00 +00:00
|
|
|
if debug:
|
2020-03-22 21:16:02 +00:00
|
|
|
print('DEBUG: "users" or "profile" missing from actor')
|
2019-07-17 10:34:00 +00:00
|
|
|
return False
|
2021-10-13 10:11:02 +00:00
|
|
|
if not hasObjectStringType(messageJson, debug):
|
2021-07-19 09:03:56 +00:00
|
|
|
return False
|
2021-10-13 10:37:52 +00:00
|
|
|
if not hasObjectStringObject(messageJson, debug):
|
2019-07-17 10:34:00 +00:00
|
|
|
return False
|
2021-02-08 14:48:37 +00:00
|
|
|
if messageJson['object']['type'] == 'Follow' or \
|
|
|
|
messageJson['object']['type'] == 'Join':
|
2020-12-22 18:06:23 +00:00
|
|
|
return _receiveUndoFollow(session, baseDir, httpPrefix,
|
|
|
|
port, messageJson,
|
|
|
|
federationList, debug)
|
2019-07-17 10:34:00 +00:00
|
|
|
return False
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _personReceiveUpdate(baseDir: str,
|
|
|
|
domain: str, port: int,
|
|
|
|
updateNickname: str, updateDomain: str,
|
|
|
|
updatePort: int,
|
|
|
|
personJson: {}, personCache: {},
|
|
|
|
debug: bool) -> bool:
|
2019-08-22 18:36:07 +00:00
|
|
|
"""Changes an actor. eg: avatar or display name change
|
2019-08-20 19:41:58 +00:00
|
|
|
"""
|
2021-03-14 19:42:25 +00:00
|
|
|
if debug:
|
|
|
|
print('Receiving actor update for ' + personJson['url'] +
|
|
|
|
' ' + str(personJson))
|
2020-12-16 10:48:40 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
|
|
|
updateDomainFull = getFullDomain(updateDomain, updatePort)
|
2021-07-29 17:14:33 +00:00
|
|
|
usersPaths = getUserPaths()
|
2020-12-23 22:18:19 +00:00
|
|
|
usersStrFound = False
|
|
|
|
for usersStr in usersPaths:
|
2021-07-29 17:14:33 +00:00
|
|
|
actor = updateDomainFull + usersStr + updateNickname
|
2020-12-23 22:18:19 +00:00
|
|
|
if actor in personJson['id']:
|
|
|
|
usersStrFound = True
|
|
|
|
break
|
|
|
|
if not usersStrFound:
|
|
|
|
if debug:
|
|
|
|
print('actor: ' + actor)
|
|
|
|
print('id: ' + personJson['id'])
|
|
|
|
print('DEBUG: Actor does not match id')
|
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
if updateDomainFull == domainFull:
|
2019-08-22 17:49:57 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: You can only receive actor updates ' +
|
|
|
|
'for domains other than your own')
|
2019-08-20 19:41:58 +00:00
|
|
|
return False
|
|
|
|
if not personJson.get('publicKey'):
|
|
|
|
if debug:
|
2020-03-22 21:16:02 +00:00
|
|
|
print('DEBUG: actor update does not contain a public key')
|
2019-08-20 19:41:58 +00:00
|
|
|
return False
|
|
|
|
if not personJson['publicKey'].get('publicKeyPem'):
|
|
|
|
if debug:
|
2020-03-22 21:16:02 +00:00
|
|
|
print('DEBUG: actor update does not contain a public key Pem')
|
2019-08-20 19:41:58 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
actorFilename = baseDir + '/cache/actors/' + \
|
|
|
|
personJson['id'].replace('/', '#') + '.json'
|
2019-08-20 19:41:58 +00:00
|
|
|
# check that the public keys match.
|
|
|
|
# If they don't then this may be a nefarious attempt to hack an account
|
2020-04-03 16:27:34 +00:00
|
|
|
idx = personJson['id']
|
|
|
|
if personCache.get(idx):
|
|
|
|
if personCache[idx]['actor']['publicKey']['publicKeyPem'] != \
|
|
|
|
personJson['publicKey']['publicKeyPem']:
|
2019-08-20 19:41:58 +00:00
|
|
|
if debug:
|
|
|
|
print('WARN: Public key does not match when updating actor')
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
if os.path.isfile(actorFilename):
|
2020-04-03 16:27:34 +00:00
|
|
|
existingPersonJson = loadJson(actorFilename)
|
2019-10-22 11:55:06 +00:00
|
|
|
if existingPersonJson:
|
2020-04-03 16:27:34 +00:00
|
|
|
if existingPersonJson['publicKey']['publicKeyPem'] != \
|
|
|
|
personJson['publicKey']['publicKeyPem']:
|
2019-08-20 19:41:58 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('WARN: Public key does not match ' +
|
|
|
|
'cached actor when updating')
|
2019-08-20 19:41:58 +00:00
|
|
|
return False
|
|
|
|
# save to cache in memory
|
2020-08-29 10:21:29 +00:00
|
|
|
storePersonInCache(baseDir, personJson['id'], personJson,
|
|
|
|
personCache, True)
|
2021-03-14 19:46:46 +00:00
|
|
|
# save to cache on file
|
2020-04-03 16:27:34 +00:00
|
|
|
if saveJson(personJson, actorFilename):
|
2021-03-14 19:42:25 +00:00
|
|
|
if debug:
|
|
|
|
print('actor updated for ' + personJson['id'])
|
2019-09-14 18:58:55 +00:00
|
|
|
|
|
|
|
# remove avatar if it exists so that it will be refreshed later
|
|
|
|
# when a timeline is constructed
|
2020-04-03 16:27:34 +00:00
|
|
|
actorStr = personJson['id'].replace('/', '-')
|
|
|
|
removeAvatarFromCache(baseDir, actorStr)
|
2019-08-20 19:41:58 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveUpdateToQuestion(recentPostsCache: {}, messageJson: {},
|
|
|
|
baseDir: str,
|
|
|
|
nickname: str, domain: str) -> None:
|
2019-11-26 10:43:37 +00:00
|
|
|
"""Updating a question as new votes arrive
|
|
|
|
"""
|
|
|
|
# message url of the question
|
|
|
|
if not messageJson.get('id'):
|
|
|
|
return
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, False):
|
2019-11-26 10:43:37 +00:00
|
|
|
return
|
2020-08-23 11:13:35 +00:00
|
|
|
messageId = removeIdEnding(messageJson['id'])
|
2019-11-26 10:43:37 +00:00
|
|
|
if '#' in messageId:
|
2020-04-03 16:27:34 +00:00
|
|
|
messageId = messageId.split('#', 1)[0]
|
2019-11-26 10:43:37 +00:00
|
|
|
# find the question post
|
2020-04-03 16:27:34 +00:00
|
|
|
postFilename = locatePost(baseDir, nickname, domain, messageId)
|
2019-11-26 10:43:37 +00:00
|
|
|
if not postFilename:
|
|
|
|
return
|
|
|
|
# load the json for the question
|
2020-04-03 16:27:34 +00:00
|
|
|
postJsonObject = loadJson(postFilename, 1)
|
2019-11-26 10:43:37 +00:00
|
|
|
if not postJsonObject:
|
|
|
|
return
|
|
|
|
if not postJsonObject.get('actor'):
|
|
|
|
return
|
|
|
|
# does the actor match?
|
2020-04-03 16:27:34 +00:00
|
|
|
if postJsonObject['actor'] != messageJson['actor']:
|
2019-11-26 10:43:37 +00:00
|
|
|
return
|
2020-04-03 16:27:34 +00:00
|
|
|
saveJson(messageJson, postFilename)
|
2019-11-26 10:43:37 +00:00
|
|
|
# ensure that the cached post is removed if it exists, so
|
|
|
|
# that it then will be recreated
|
2020-04-03 16:27:34 +00:00
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, nickname, domain, messageJson)
|
2019-11-26 10:43:37 +00:00
|
|
|
if cachedPostFilename:
|
|
|
|
if os.path.isfile(cachedPostFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(cachedPostFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: _receiveUpdateToQuestion unable to delete ' +
|
|
|
|
cachedPostFilename)
|
2019-11-26 10:43:37 +00:00
|
|
|
# remove from memory cache
|
2020-04-03 16:27:34 +00:00
|
|
|
removePostFromCache(messageJson, recentPostsCache)
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveUpdate(recentPostsCache: {}, session, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
|
|
|
nickname: str, debug: bool) -> bool:
|
2019-07-09 14:20:23 +00:00
|
|
|
"""Receives an Update activity within the POST section of HTTPServer
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['type'] != 'Update':
|
2019-07-09 14:20:23 +00:00
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2019-07-09 14:20:23 +00:00
|
|
|
return False
|
2021-10-13 10:11:02 +00:00
|
|
|
if not hasObjectStringType(messageJson, debug):
|
2021-07-19 09:07:20 +00:00
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
2019-07-09 14:20:23 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'])
|
2019-07-09 14:20:23 +00:00
|
|
|
return False
|
2019-08-22 17:25:12 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['object']['type'] == 'Question':
|
2020-12-22 18:06:23 +00:00
|
|
|
_receiveUpdateToQuestion(recentPostsCache, messageJson,
|
|
|
|
baseDir, nickname, domain)
|
2019-11-25 14:05:59 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: Question update was received')
|
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['object']['type'] == 'Person' or \
|
|
|
|
messageJson['object']['type'] == 'Application' or \
|
|
|
|
messageJson['object']['type'] == 'Group' or \
|
|
|
|
messageJson['object']['type'] == 'Service':
|
|
|
|
if messageJson['object'].get('url') and \
|
|
|
|
messageJson['object'].get('id'):
|
2021-03-14 19:42:25 +00:00
|
|
|
if debug:
|
|
|
|
print('Request to update actor: ' + str(messageJson))
|
2020-04-03 16:27:34 +00:00
|
|
|
updateNickname = getNicknameFromActor(messageJson['actor'])
|
2019-09-02 09:43:43 +00:00
|
|
|
if updateNickname:
|
2020-04-03 16:27:34 +00:00
|
|
|
updateDomain, updatePort = \
|
|
|
|
getDomainFromActor(messageJson['actor'])
|
2020-12-22 18:06:23 +00:00
|
|
|
if _personReceiveUpdate(baseDir,
|
|
|
|
domain, port,
|
|
|
|
updateNickname, updateDomain,
|
|
|
|
updatePort,
|
|
|
|
messageJson['object'],
|
|
|
|
personCache, debug):
|
2021-09-30 12:54:44 +00:00
|
|
|
print('Person Update: ' + str(messageJson))
|
2019-09-02 09:43:43 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Profile update was received for ' +
|
|
|
|
messageJson['object']['url'])
|
2019-09-02 09:43:43 +00:00
|
|
|
return True
|
2019-07-09 14:20:23 +00:00
|
|
|
return False
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveLike(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
onionDomain: str,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
2021-09-03 22:04:50 +00:00
|
|
|
debug: bool,
|
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int, translate: {},
|
|
|
|
allowDeletion: bool,
|
|
|
|
YTReplacementDomain: str,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain: str,
|
2021-09-03 22:04:50 +00:00
|
|
|
peertubeInstances: [],
|
|
|
|
allowLocalNetworkAccess: bool,
|
|
|
|
themeName: str, systemLanguage: str,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount: int, CWlists: {},
|
|
|
|
listsEnabled: str) -> bool:
|
2019-07-10 12:40:31 +00:00
|
|
|
"""Receives a Like activity within the POST section of HTTPServer
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['type'] != 'Like':
|
2019-07-10 12:40:31 +00:00
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2019-07-10 12:40:31 +00:00
|
|
|
return False
|
2021-10-13 11:15:06 +00:00
|
|
|
if not hasObjectString(messageJson, debug):
|
2019-07-10 12:40:31 +00:00
|
|
|
return False
|
|
|
|
if not messageJson.get('to'):
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' + messageJson['type'] + ' has no "to" list')
|
2019-07-10 12:40:31 +00:00
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
2019-07-10 12:40:31 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'])
|
2019-07-10 12:40:31 +00:00
|
|
|
return False
|
|
|
|
if '/statuses/' not in messageJson['object']:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "statuses" missing from object in ' +
|
|
|
|
messageJson['type'])
|
2019-07-10 12:40:31 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
if not os.path.isdir(baseDir + '/accounts/' + handle):
|
|
|
|
print('DEBUG: unknown recipient of like - ' + handle)
|
2019-07-10 18:00:14 +00:00
|
|
|
# if this post in the outbox of the person?
|
2020-12-22 21:24:46 +00:00
|
|
|
handleName = handle.split('@')[0]
|
|
|
|
handleDom = handle.split('@')[1]
|
2021-05-19 19:21:27 +00:00
|
|
|
postLikedId = messageJson['object']
|
|
|
|
postFilename = locatePost(baseDir, handleName, handleDom, postLikedId)
|
2019-07-10 12:40:31 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: post not found in inbox or outbox')
|
2021-05-19 19:21:27 +00:00
|
|
|
print(postLikedId)
|
2019-07-10 12:40:31 +00:00
|
|
|
return True
|
|
|
|
if debug:
|
2019-07-11 12:59:00 +00:00
|
|
|
print('DEBUG: liked post found in inbox')
|
2019-10-19 17:50:05 +00:00
|
|
|
|
2021-10-14 22:29:51 +00:00
|
|
|
likeActor = messageJson['actor']
|
2020-12-22 21:24:46 +00:00
|
|
|
handleName = handle.split('@')[0]
|
|
|
|
handleDom = handle.split('@')[1]
|
2020-12-22 18:06:23 +00:00
|
|
|
if not _alreadyLiked(baseDir,
|
2020-12-22 21:24:46 +00:00
|
|
|
handleName, handleDom,
|
2021-05-19 19:21:27 +00:00
|
|
|
postLikedId,
|
2021-10-14 22:29:51 +00:00
|
|
|
likeActor):
|
2020-12-22 18:06:23 +00:00
|
|
|
_likeNotify(baseDir, domain, onionDomain, handle,
|
2021-10-14 22:29:51 +00:00
|
|
|
likeActor, postLikedId)
|
2021-03-12 15:34:56 +00:00
|
|
|
updateLikesCollection(recentPostsCache, baseDir, postFilename,
|
2021-10-14 22:29:51 +00:00
|
|
|
postLikedId, likeActor,
|
2021-10-17 16:11:38 +00:00
|
|
|
handleName, domain, debug, None)
|
2021-09-03 22:04:50 +00:00
|
|
|
# regenerate the html
|
|
|
|
likedPostJson = loadJson(postFilename, 0, 1)
|
|
|
|
if likedPostJson:
|
2021-10-14 22:29:51 +00:00
|
|
|
if likedPostJson.get('type'):
|
|
|
|
if likedPostJson['type'] == 'Announce' and \
|
|
|
|
likedPostJson.get('object'):
|
|
|
|
if isinstance(likedPostJson['object'], str):
|
|
|
|
announceLikeUrl = likedPostJson['object']
|
|
|
|
announceLikedFilename = \
|
|
|
|
locatePost(baseDir, handleName,
|
|
|
|
domain, announceLikeUrl)
|
|
|
|
if announceLikedFilename:
|
|
|
|
postLikedId = announceLikeUrl
|
|
|
|
postFilename = announceLikedFilename
|
|
|
|
updateLikesCollection(recentPostsCache,
|
|
|
|
baseDir,
|
|
|
|
postFilename,
|
|
|
|
postLikedId,
|
|
|
|
likeActor,
|
|
|
|
handleName,
|
2021-10-17 16:11:38 +00:00
|
|
|
domain, debug, None)
|
2021-10-14 22:29:51 +00:00
|
|
|
if likedPostJson:
|
|
|
|
if debug:
|
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, handleName, domain,
|
|
|
|
likedPostJson)
|
|
|
|
print('Liked post json: ' + str(likedPostJson))
|
|
|
|
print('Liked post nickname: ' + handleName + ' ' + domain)
|
|
|
|
print('Liked post cache: ' + str(cachedPostFilename))
|
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, handleName, domain)
|
|
|
|
notDM = not isDM(likedPostJson)
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, False,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
handleName, domain, port, likedPostJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
2021-10-21 19:00:25 +00:00
|
|
|
False, True, False, CWlists,
|
|
|
|
listsEnabled)
|
2019-07-10 12:40:31 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveUndoLike(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
2021-09-03 22:10:54 +00:00
|
|
|
debug: bool,
|
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int, translate: {},
|
|
|
|
allowDeletion: bool,
|
|
|
|
YTReplacementDomain: str,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain: str,
|
2021-09-03 22:10:54 +00:00
|
|
|
peertubeInstances: [],
|
|
|
|
allowLocalNetworkAccess: bool,
|
|
|
|
themeName: str, systemLanguage: str,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount: int, CWlists: {},
|
|
|
|
listsEnabled: str) -> bool:
|
2019-07-12 09:10:09 +00:00
|
|
|
"""Receives an undo like activity within the POST section of HTTPServer
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['type'] != 'Undo':
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
2021-10-13 10:11:02 +00:00
|
|
|
if not hasObjectStringType(messageJson, debug):
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['object']['type'] != 'Like':
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
2021-10-13 10:37:52 +00:00
|
|
|
if not hasObjectStringObject(messageJson, debug):
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
2019-07-12 09:10:09 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'] + ' like')
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
|
|
|
if '/statuses/' not in messageJson['object']['object']:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "statuses" missing from like object in ' +
|
|
|
|
messageJson['type'])
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
if not os.path.isdir(baseDir + '/accounts/' + handle):
|
|
|
|
print('DEBUG: unknown recipient of undo like - ' + handle)
|
2019-07-12 09:10:09 +00:00
|
|
|
# if this post in the outbox of the person?
|
2020-12-22 21:24:46 +00:00
|
|
|
handleName = handle.split('@')[0]
|
|
|
|
handleDom = handle.split('@')[1]
|
2020-04-03 16:27:34 +00:00
|
|
|
postFilename = \
|
2020-12-22 21:24:46 +00:00
|
|
|
locatePost(baseDir, handleName, handleDom,
|
2020-04-03 16:27:34 +00:00
|
|
|
messageJson['object']['object'])
|
2019-07-12 09:10:09 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
2019-07-12 09:41:57 +00:00
|
|
|
print('DEBUG: unliked post not found in inbox or outbox')
|
2019-07-12 09:10:09 +00:00
|
|
|
print(messageJson['object']['object'])
|
|
|
|
return True
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: liked post found in inbox. Now undoing.')
|
2021-10-14 22:29:51 +00:00
|
|
|
likeActor = messageJson['actor']
|
|
|
|
postLikedId = messageJson['object']
|
2020-04-03 16:27:34 +00:00
|
|
|
undoLikesCollectionEntry(recentPostsCache, baseDir, postFilename,
|
2021-10-17 16:11:38 +00:00
|
|
|
postLikedId, likeActor, domain, debug, None)
|
2021-09-03 22:10:54 +00:00
|
|
|
# regenerate the html
|
|
|
|
likedPostJson = loadJson(postFilename, 0, 1)
|
|
|
|
if likedPostJson:
|
2021-10-14 22:29:51 +00:00
|
|
|
if likedPostJson.get('type'):
|
|
|
|
if likedPostJson['type'] == 'Announce' and \
|
|
|
|
likedPostJson.get('object'):
|
|
|
|
if isinstance(likedPostJson['object'], str):
|
|
|
|
announceLikeUrl = likedPostJson['object']
|
|
|
|
announceLikedFilename = \
|
|
|
|
locatePost(baseDir, handleName,
|
|
|
|
domain, announceLikeUrl)
|
|
|
|
if announceLikedFilename:
|
|
|
|
postLikedId = announceLikeUrl
|
|
|
|
postFilename = announceLikedFilename
|
|
|
|
undoLikesCollectionEntry(recentPostsCache, baseDir,
|
|
|
|
postFilename, postLikedId,
|
2021-10-17 16:11:38 +00:00
|
|
|
likeActor, domain, debug,
|
|
|
|
None)
|
2021-10-14 22:29:51 +00:00
|
|
|
if likedPostJson:
|
|
|
|
if debug:
|
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, handleName, domain,
|
|
|
|
likedPostJson)
|
|
|
|
print('Unliked post json: ' + str(likedPostJson))
|
|
|
|
print('Unliked post nickname: ' + handleName + ' ' + domain)
|
|
|
|
print('Unliked post cache: ' + str(cachedPostFilename))
|
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, handleName, domain)
|
|
|
|
notDM = not isDM(likedPostJson)
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, False,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
handleName, domain, port, likedPostJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
2021-10-21 19:00:25 +00:00
|
|
|
False, True, False, CWlists,
|
|
|
|
listsEnabled)
|
2019-07-12 09:10:09 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2021-11-10 12:16:03 +00:00
|
|
|
def _receiveReaction(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
onionDomain: str,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
|
|
|
debug: bool,
|
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int, translate: {},
|
|
|
|
allowDeletion: bool,
|
|
|
|
YTReplacementDomain: str,
|
|
|
|
twitterReplacementDomain: str,
|
|
|
|
peertubeInstances: [],
|
|
|
|
allowLocalNetworkAccess: bool,
|
|
|
|
themeName: str, systemLanguage: str,
|
|
|
|
maxLikeCount: int, CWlists: {},
|
|
|
|
listsEnabled: str) -> bool:
|
|
|
|
"""Receives an emoji reaction within the POST section of HTTPServer
|
|
|
|
"""
|
|
|
|
if messageJson['type'] != 'EmojiReact':
|
|
|
|
return False
|
|
|
|
if not hasActor(messageJson, debug):
|
|
|
|
return False
|
|
|
|
if not hasObjectString(messageJson, debug):
|
|
|
|
return False
|
|
|
|
if not messageJson.get('to'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: ' + messageJson['type'] + ' has no "to" list')
|
|
|
|
return False
|
|
|
|
if not messageJson.get('content'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: ' + messageJson['type'] + ' has no "content"')
|
|
|
|
return False
|
|
|
|
if not isinstance(messageJson['content'], str):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: ' + messageJson['type'] + ' content is not string')
|
|
|
|
return False
|
2021-11-10 13:10:02 +00:00
|
|
|
if not validEmojiContent(messageJson['content']):
|
|
|
|
print('_receiveReaction: Invalid emoji reaction: "' +
|
|
|
|
messageJson['content'] + '" from ' + messageJson['actor'])
|
|
|
|
return False
|
2021-11-10 12:16:03 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: "users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'])
|
|
|
|
return False
|
|
|
|
if '/statuses/' not in messageJson['object']:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: "statuses" missing from object in ' +
|
|
|
|
messageJson['type'])
|
|
|
|
return False
|
|
|
|
if not os.path.isdir(baseDir + '/accounts/' + handle):
|
|
|
|
print('DEBUG: unknown recipient of emoji reaction - ' + handle)
|
2021-11-17 14:25:24 +00:00
|
|
|
if os.path.isfile(baseDir + '/accounts/' + handle +
|
|
|
|
'/.hideReactionButton'):
|
|
|
|
print('Emoji reaction rejected by ' + handle +
|
|
|
|
' due to their settings')
|
|
|
|
return True
|
2021-11-10 12:16:03 +00:00
|
|
|
# if this post in the outbox of the person?
|
|
|
|
handleName = handle.split('@')[0]
|
|
|
|
handleDom = handle.split('@')[1]
|
2021-11-17 14:25:24 +00:00
|
|
|
|
2021-11-10 12:16:03 +00:00
|
|
|
postReactionId = messageJson['object']
|
|
|
|
emojiContent = removeHtml(messageJson['content'])
|
|
|
|
if not emojiContent:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: emoji reaction has no content')
|
|
|
|
return True
|
|
|
|
postFilename = locatePost(baseDir, handleName, handleDom, postReactionId)
|
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: emoji reaction post not found in inbox or outbox')
|
|
|
|
print(postReactionId)
|
|
|
|
return True
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: emoji reaction post found in inbox')
|
|
|
|
|
|
|
|
reactionActor = messageJson['actor']
|
|
|
|
handleName = handle.split('@')[0]
|
|
|
|
handleDom = handle.split('@')[1]
|
|
|
|
if not _alreadyReacted(baseDir,
|
|
|
|
handleName, handleDom,
|
|
|
|
postReactionId,
|
|
|
|
reactionActor,
|
|
|
|
emojiContent):
|
|
|
|
_reactionNotify(baseDir, domain, onionDomain, handle,
|
|
|
|
reactionActor, postReactionId, emojiContent)
|
|
|
|
updateReactionCollection(recentPostsCache, baseDir, postFilename,
|
|
|
|
postReactionId, reactionActor,
|
|
|
|
handleName, domain, debug, None, emojiContent)
|
|
|
|
# regenerate the html
|
|
|
|
reactionPostJson = loadJson(postFilename, 0, 1)
|
|
|
|
if reactionPostJson:
|
|
|
|
if reactionPostJson.get('type'):
|
|
|
|
if reactionPostJson['type'] == 'Announce' and \
|
|
|
|
reactionPostJson.get('object'):
|
|
|
|
if isinstance(reactionPostJson['object'], str):
|
|
|
|
announceReactionUrl = reactionPostJson['object']
|
|
|
|
announceReactionFilename = \
|
|
|
|
locatePost(baseDir, handleName,
|
|
|
|
domain, announceReactionUrl)
|
|
|
|
if announceReactionFilename:
|
|
|
|
postReactionId = announceReactionUrl
|
|
|
|
postFilename = announceReactionFilename
|
|
|
|
updateReactionCollection(recentPostsCache,
|
|
|
|
baseDir,
|
|
|
|
postFilename,
|
|
|
|
postReactionId,
|
|
|
|
reactionActor,
|
|
|
|
handleName,
|
|
|
|
domain, debug, None,
|
|
|
|
emojiContent)
|
|
|
|
if reactionPostJson:
|
|
|
|
if debug:
|
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, handleName, domain,
|
|
|
|
reactionPostJson)
|
|
|
|
print('Reaction post json: ' + str(reactionPostJson))
|
|
|
|
print('Reaction post nickname: ' + handleName + ' ' + domain)
|
|
|
|
print('Reaction post cache: ' + str(cachedPostFilename))
|
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, handleName, domain)
|
|
|
|
notDM = not isDM(reactionPostJson)
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, False,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
handleName, domain, port, reactionPostJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
|
|
|
False, True, False, CWlists,
|
|
|
|
listsEnabled)
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
def _receiveUndoReaction(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
|
|
|
debug: bool,
|
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int, translate: {},
|
|
|
|
allowDeletion: bool,
|
|
|
|
YTReplacementDomain: str,
|
|
|
|
twitterReplacementDomain: str,
|
|
|
|
peertubeInstances: [],
|
|
|
|
allowLocalNetworkAccess: bool,
|
|
|
|
themeName: str, systemLanguage: str,
|
|
|
|
maxLikeCount: int, CWlists: {},
|
|
|
|
listsEnabled: str) -> bool:
|
|
|
|
"""Receives an undo emoji reaction within the POST section of HTTPServer
|
|
|
|
"""
|
|
|
|
if messageJson['type'] != 'Undo':
|
|
|
|
return False
|
|
|
|
if not hasActor(messageJson, debug):
|
|
|
|
return False
|
|
|
|
if not hasObjectStringType(messageJson, debug):
|
|
|
|
return False
|
|
|
|
if messageJson['object']['type'] != 'EmojiReact':
|
|
|
|
return False
|
|
|
|
if not hasObjectStringObject(messageJson, debug):
|
|
|
|
return False
|
|
|
|
if not messageJson['object'].get('content'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: ' + messageJson['type'] + ' has no "content"')
|
|
|
|
return False
|
|
|
|
if not isinstance(messageJson['object']['content'], str):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: ' + messageJson['type'] + ' content is not string')
|
|
|
|
return False
|
|
|
|
if not hasUsersPath(messageJson['actor']):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: "users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'] + ' reaction')
|
|
|
|
return False
|
|
|
|
if '/statuses/' not in messageJson['object']['object']:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: "statuses" missing from reaction object in ' +
|
|
|
|
messageJson['type'])
|
|
|
|
return False
|
|
|
|
if not os.path.isdir(baseDir + '/accounts/' + handle):
|
|
|
|
print('DEBUG: unknown recipient of undo reaction - ' + handle)
|
|
|
|
# if this post in the outbox of the person?
|
|
|
|
handleName = handle.split('@')[0]
|
|
|
|
handleDom = handle.split('@')[1]
|
|
|
|
postFilename = \
|
|
|
|
locatePost(baseDir, handleName, handleDom,
|
|
|
|
messageJson['object']['object'])
|
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: unreaction post not found in inbox or outbox')
|
|
|
|
print(messageJson['object']['object'])
|
|
|
|
return True
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: reaction post found in inbox. Now undoing.')
|
|
|
|
reactionActor = messageJson['actor']
|
|
|
|
postReactionId = messageJson['object']
|
|
|
|
emojiContent = removeHtml(messageJson['object']['content'])
|
|
|
|
if not emojiContent:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: unreaction has no content')
|
|
|
|
return True
|
|
|
|
undoReactionCollectionEntry(recentPostsCache, baseDir, postFilename,
|
|
|
|
postReactionId, reactionActor, domain,
|
|
|
|
debug, None, emojiContent)
|
|
|
|
# regenerate the html
|
|
|
|
reactionPostJson = loadJson(postFilename, 0, 1)
|
|
|
|
if reactionPostJson:
|
|
|
|
if reactionPostJson.get('type'):
|
|
|
|
if reactionPostJson['type'] == 'Announce' and \
|
|
|
|
reactionPostJson.get('object'):
|
|
|
|
if isinstance(reactionPostJson['object'], str):
|
|
|
|
announceReactionUrl = reactionPostJson['object']
|
|
|
|
announceReactionFilename = \
|
|
|
|
locatePost(baseDir, handleName,
|
|
|
|
domain, announceReactionUrl)
|
|
|
|
if announceReactionFilename:
|
|
|
|
postReactionId = announceReactionUrl
|
|
|
|
postFilename = announceReactionFilename
|
|
|
|
undoReactionCollectionEntry(recentPostsCache, baseDir,
|
|
|
|
postFilename,
|
|
|
|
postReactionId,
|
|
|
|
reactionActor, domain,
|
|
|
|
debug, None,
|
|
|
|
emojiContent)
|
|
|
|
if reactionPostJson:
|
|
|
|
if debug:
|
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, handleName, domain,
|
|
|
|
reactionPostJson)
|
|
|
|
print('Unreaction post json: ' + str(reactionPostJson))
|
|
|
|
print('Unreaction post nickname: ' + handleName + ' ' + domain)
|
|
|
|
print('Unreaction post cache: ' + str(cachedPostFilename))
|
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, handleName, domain)
|
|
|
|
notDM = not isDM(reactionPostJson)
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, False,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
handleName, domain, port, reactionPostJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
|
|
|
False, True, False, CWlists,
|
|
|
|
listsEnabled)
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveBookmark(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
2021-09-03 22:28:50 +00:00
|
|
|
debug: bool, signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int, translate: {},
|
|
|
|
allowDeletion: bool,
|
|
|
|
YTReplacementDomain: str,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain: str,
|
2021-09-03 22:28:50 +00:00
|
|
|
peertubeInstances: [],
|
|
|
|
allowLocalNetworkAccess: bool,
|
|
|
|
themeName: str, systemLanguage: str,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount: int, CWlists: {},
|
|
|
|
listsEnabled: {}) -> bool:
|
2019-11-17 14:01:49 +00:00
|
|
|
"""Receives a bookmark activity within the POST section of HTTPServer
|
|
|
|
"""
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson.get('type'):
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
|
|
|
if messageJson['type'] != 'Add':
|
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson.get('target'):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2021-03-20 09:49:43 +00:00
|
|
|
print('DEBUG: no target in inbox bookmark Add')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-10-13 10:11:02 +00:00
|
|
|
if not hasObjectStringType(messageJson, debug):
|
2021-03-20 14:25:24 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not isinstance(messageJson['target'], str):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2021-03-20 09:49:43 +00:00
|
|
|
print('DEBUG: inbox bookmark Add target is not string')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2020-12-16 10:48:40 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
2020-04-03 16:27:34 +00:00
|
|
|
nickname = handle.split('@')[0]
|
|
|
|
if not messageJson['actor'].endswith(domainFull + '/users/' + nickname):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2021-03-20 09:49:43 +00:00
|
|
|
print('DEBUG: inbox bookmark Add unexpected actor')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson['target'].endswith(messageJson['actor'] +
|
|
|
|
'/tlbookmarks'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox bookmark Add target invalid ' +
|
|
|
|
messageJson['target'])
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if messageJson['object']['type'] != 'Document':
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox bookmark Add type is not Document')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson['object'].get('url'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox bookmark Add missing url')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if '/statuses/' not in messageJson['object']['url']:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox bookmark Add missing statuses un url')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: c2s inbox bookmark Add request arrived in outbox')
|
|
|
|
|
|
|
|
messageUrl = removeIdEnding(messageJson['object']['url'])
|
2021-06-23 21:31:50 +00:00
|
|
|
domain = removeDomainPort(domain)
|
2021-03-20 09:49:43 +00:00
|
|
|
postFilename = locatePost(baseDir, nickname, domain, messageUrl)
|
2019-11-17 14:01:49 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
2021-03-20 09:49:43 +00:00
|
|
|
print('DEBUG: c2s inbox like post not found in inbox or outbox')
|
|
|
|
print(messageUrl)
|
2019-11-17 14:01:49 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
updateBookmarksCollection(recentPostsCache, baseDir, postFilename,
|
2021-03-20 09:49:43 +00:00
|
|
|
messageJson['object']['url'],
|
2020-04-03 16:27:34 +00:00
|
|
|
messageJson['actor'], domain, debug)
|
2021-09-03 22:28:50 +00:00
|
|
|
# regenerate the html
|
|
|
|
bookmarkedPostJson = loadJson(postFilename, 0, 1)
|
|
|
|
if bookmarkedPostJson:
|
|
|
|
if debug:
|
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, nickname, domain,
|
|
|
|
bookmarkedPostJson)
|
|
|
|
print('Bookmarked post json: ' + str(bookmarkedPostJson))
|
|
|
|
print('Bookmarked post nickname: ' + nickname + ' ' + domain)
|
|
|
|
print('Bookmarked post cache: ' + str(cachedPostFilename))
|
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, nickname, domain)
|
|
|
|
notDM = not isDM(bookmarkedPostJson)
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, False,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
nickname, domain, port, bookmarkedPostJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
2021-09-18 17:08:14 +00:00
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
2021-09-03 22:28:50 +00:00
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
2021-10-21 19:00:25 +00:00
|
|
|
False, True, False, CWlists,
|
|
|
|
listsEnabled)
|
2019-11-17 14:01:49 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveUndoBookmark(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
2021-09-03 22:28:50 +00:00
|
|
|
debug: bool, signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int, translate: {},
|
|
|
|
allowDeletion: bool,
|
|
|
|
YTReplacementDomain: str,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain: str,
|
2021-09-03 22:28:50 +00:00
|
|
|
peertubeInstances: [],
|
|
|
|
allowLocalNetworkAccess: bool,
|
|
|
|
themeName: str, systemLanguage: str,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount: int, CWlists: {},
|
|
|
|
listsEnabled: str) -> bool:
|
2019-11-17 14:01:49 +00:00
|
|
|
"""Receives an undo bookmark activity within the POST section of HTTPServer
|
|
|
|
"""
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson.get('type'):
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
|
|
|
if messageJson['type'] != 'Remove':
|
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson.get('target'):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2021-03-20 09:49:43 +00:00
|
|
|
print('DEBUG: no target in inbox undo bookmark Remove')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-10-13 10:11:02 +00:00
|
|
|
if not hasObjectStringType(messageJson, debug):
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not isinstance(messageJson['target'], str):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2021-03-20 14:25:24 +00:00
|
|
|
print('DEBUG: inbox Remove bookmark target is not string')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2020-12-16 10:48:40 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
2020-04-03 16:27:34 +00:00
|
|
|
nickname = handle.split('@')[0]
|
|
|
|
if not messageJson['actor'].endswith(domainFull + '/users/' + nickname):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2021-03-20 09:49:43 +00:00
|
|
|
print('DEBUG: inbox undo bookmark Remove unexpected actor')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson['target'].endswith(messageJson['actor'] +
|
|
|
|
'/tlbookmarks'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox undo bookmark Remove target invalid ' +
|
|
|
|
messageJson['target'])
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if messageJson['object']['type'] != 'Document':
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox undo bookmark Remove type is not Document')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if not messageJson['object'].get('url'):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox undo bookmark Remove missing url')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if '/statuses/' not in messageJson['object']['url']:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: inbox undo bookmark Remove missing statuses un url')
|
2021-03-20 14:09:07 +00:00
|
|
|
return False
|
2021-03-20 09:49:43 +00:00
|
|
|
if debug:
|
2021-03-20 14:09:07 +00:00
|
|
|
print('DEBUG: c2s inbox Remove bookmark ' +
|
2021-03-20 09:49:43 +00:00
|
|
|
'request arrived in outbox')
|
|
|
|
|
|
|
|
messageUrl = removeIdEnding(messageJson['object']['url'])
|
2021-06-23 21:31:50 +00:00
|
|
|
domain = removeDomainPort(domain)
|
2021-03-20 09:49:43 +00:00
|
|
|
postFilename = locatePost(baseDir, nickname, domain, messageUrl)
|
2019-11-17 14:01:49 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
2021-03-20 09:49:43 +00:00
|
|
|
print('DEBUG: c2s inbox like post not found in inbox or outbox')
|
|
|
|
print(messageUrl)
|
2019-11-17 14:01:49 +00:00
|
|
|
return True
|
2021-03-20 09:49:43 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
undoBookmarksCollectionEntry(recentPostsCache, baseDir, postFilename,
|
2021-03-20 09:49:43 +00:00
|
|
|
messageJson['object']['url'],
|
2020-04-03 16:27:34 +00:00
|
|
|
messageJson['actor'], domain, debug)
|
2021-09-03 22:28:50 +00:00
|
|
|
# regenerate the html
|
|
|
|
bookmarkedPostJson = loadJson(postFilename, 0, 1)
|
|
|
|
if bookmarkedPostJson:
|
|
|
|
if debug:
|
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, nickname, domain,
|
|
|
|
bookmarkedPostJson)
|
|
|
|
print('Unbookmarked post json: ' + str(bookmarkedPostJson))
|
|
|
|
print('Unbookmarked post nickname: ' + nickname + ' ' + domain)
|
|
|
|
print('Unbookmarked post cache: ' + str(cachedPostFilename))
|
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, nickname, domain)
|
|
|
|
notDM = not isDM(bookmarkedPostJson)
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, False,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
nickname, domain, port, bookmarkedPostJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
2021-09-18 17:08:14 +00:00
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
2021-09-03 22:28:50 +00:00
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
2021-10-21 19:00:25 +00:00
|
|
|
False, True, False, CWlists, listsEnabled)
|
2019-11-17 14:01:49 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveDelete(session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
|
|
|
debug: bool, allowDeletion: bool,
|
|
|
|
recentPostsCache: {}) -> bool:
|
2019-07-11 21:38:28 +00:00
|
|
|
"""Receives a Delete activity within the POST section of HTTPServer
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['type'] != 'Delete':
|
2019-07-11 21:38:28 +00:00
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2019-07-11 21:38:28 +00:00
|
|
|
return False
|
2019-07-17 17:16:48 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: Delete activity arrived')
|
2021-10-13 11:15:06 +00:00
|
|
|
if not hasObjectString(messageJson, debug):
|
2019-07-11 21:38:28 +00:00
|
|
|
return False
|
2020-12-16 10:48:40 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
2020-04-03 16:27:34 +00:00
|
|
|
deletePrefix = httpPrefix + '://' + domainFull + '/'
|
|
|
|
if (not allowDeletion and
|
|
|
|
(not messageJson['object'].startswith(deletePrefix) or
|
|
|
|
not messageJson['actor'].startswith(deletePrefix))):
|
2019-08-12 18:02:29 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: delete not permitted from other instances')
|
2020-03-22 21:16:02 +00:00
|
|
|
return False
|
2019-07-11 21:38:28 +00:00
|
|
|
if not messageJson.get('to'):
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' + messageJson['type'] + ' has no "to" list')
|
2019-07-11 21:38:28 +00:00
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
2019-07-11 21:38:28 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' +
|
|
|
|
'"users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'])
|
2019-07-11 21:38:28 +00:00
|
|
|
return False
|
|
|
|
if '/statuses/' not in messageJson['object']:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "statuses" missing from object in ' +
|
|
|
|
messageJson['type'])
|
2019-07-11 21:38:28 +00:00
|
|
|
return False
|
2019-07-11 21:42:15 +00:00
|
|
|
if messageJson['actor'] not in messageJson['object']:
|
|
|
|
if debug:
|
2020-03-22 21:16:02 +00:00
|
|
|
print('DEBUG: actor is not the owner of the post to be deleted')
|
2020-04-03 16:27:34 +00:00
|
|
|
if not os.path.isdir(baseDir + '/accounts/' + handle):
|
|
|
|
print('DEBUG: unknown recipient of like - ' + handle)
|
2019-07-11 21:38:28 +00:00
|
|
|
# if this post in the outbox of the person?
|
2020-08-23 11:13:35 +00:00
|
|
|
messageId = removeIdEnding(messageJson['object'])
|
2020-04-03 16:27:34 +00:00
|
|
|
removeModerationPostFromIndex(baseDir, messageId, debug)
|
2020-11-27 10:38:51 +00:00
|
|
|
handleNickname = handle.split('@')[0]
|
|
|
|
handleDomain = handle.split('@')[1]
|
|
|
|
postFilename = locatePost(baseDir, handleNickname,
|
|
|
|
handleDomain, messageId)
|
2019-07-11 21:38:28 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: delete post not found in inbox or outbox')
|
2019-07-17 17:16:48 +00:00
|
|
|
print(messageId)
|
2019-07-14 14:42:00 +00:00
|
|
|
return True
|
2020-11-27 10:38:51 +00:00
|
|
|
deletePost(baseDir, httpPrefix, handleNickname,
|
|
|
|
handleDomain, postFilename, debug,
|
2020-06-24 13:30:50 +00:00
|
|
|
recentPostsCache)
|
2019-07-11 21:38:28 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: post deleted - ' + postFilename)
|
2020-11-27 10:38:51 +00:00
|
|
|
|
|
|
|
# also delete any local blogs saved to the news actor
|
|
|
|
if handleNickname != 'news' and handleDomain == domainFull:
|
|
|
|
postFilename = locatePost(baseDir, 'news',
|
|
|
|
handleDomain, messageId)
|
|
|
|
if postFilename:
|
|
|
|
deletePost(baseDir, httpPrefix, 'news',
|
|
|
|
handleDomain, postFilename, debug,
|
|
|
|
recentPostsCache)
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: blog post deleted - ' + postFilename)
|
2019-07-11 21:38:28 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveAnnounce(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str,
|
|
|
|
domain: str, onionDomain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
|
|
|
debug: bool, translate: {},
|
2021-01-30 11:47:09 +00:00
|
|
|
YTReplacementDomain: str,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain: str,
|
2021-03-09 19:52:10 +00:00
|
|
|
allowLocalNetworkAccess: bool,
|
2021-08-31 14:17:11 +00:00
|
|
|
themeName: str, systemLanguage: str,
|
2021-09-04 10:11:46 +00:00
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int,
|
|
|
|
allowDeletion: bool,
|
|
|
|
peertubeInstances: [],
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount: int, CWlists: {},
|
|
|
|
listsEnabled: str) -> bool:
|
2019-07-12 09:41:57 +00:00
|
|
|
"""Receives an announce activity within the POST section of HTTPServer
|
2019-07-11 19:31:02 +00:00
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['type'] != 'Announce':
|
2019-07-11 19:31:02 +00:00
|
|
|
return False
|
2019-09-29 09:15:10 +00:00
|
|
|
if '@' not in handle:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: bad handle ' + handle)
|
2020-03-22 21:16:02 +00:00
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2019-07-11 19:31:02 +00:00
|
|
|
return False
|
2019-07-16 22:57:45 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: receiving announce on ' + handle)
|
2021-10-13 11:15:06 +00:00
|
|
|
if not hasObjectString(messageJson, debug):
|
2019-07-11 19:31:02 +00:00
|
|
|
return False
|
|
|
|
if not messageJson.get('to'):
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' + messageJson['type'] + ' has no "to" list')
|
2019-07-11 19:31:02 +00:00
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
2019-07-11 19:31:02 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' +
|
|
|
|
'"users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'])
|
2019-09-09 09:41:31 +00:00
|
|
|
return False
|
2021-06-03 13:21:57 +00:00
|
|
|
if isSelfAnnounce(messageJson):
|
2021-06-03 08:33:50 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: self-boost rejected')
|
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['object']):
|
2019-09-09 09:41:31 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: ' +
|
|
|
|
'"users", "channel" or "profile" missing in ' +
|
|
|
|
messageJson['type'])
|
2019-07-11 19:31:02 +00:00
|
|
|
return False
|
2020-05-25 09:21:34 +00:00
|
|
|
|
2021-09-13 11:34:56 +00:00
|
|
|
blockedCache = {}
|
2020-06-11 12:26:15 +00:00
|
|
|
prefixes = getProtocolPrefixes()
|
2020-05-25 09:21:34 +00:00
|
|
|
# is the domain of the announce actor blocked?
|
2020-06-11 12:04:42 +00:00
|
|
|
objectDomain = messageJson['object']
|
|
|
|
for prefix in prefixes:
|
|
|
|
objectDomain = objectDomain.replace(prefix, '')
|
2019-09-09 16:02:14 +00:00
|
|
|
if '/' in objectDomain:
|
2020-04-03 16:27:34 +00:00
|
|
|
objectDomain = objectDomain.split('/')[0]
|
|
|
|
if isBlockedDomain(baseDir, objectDomain):
|
2019-09-09 16:02:14 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: announced domain is blocked')
|
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
if not os.path.isdir(baseDir + '/accounts/' + handle):
|
|
|
|
print('DEBUG: unknown recipient of announce - ' + handle)
|
2020-05-25 09:21:34 +00:00
|
|
|
|
|
|
|
# is the announce actor blocked?
|
2020-04-03 16:27:34 +00:00
|
|
|
nickname = handle.split('@')[0]
|
2020-05-25 09:21:34 +00:00
|
|
|
actorNickname = getNicknameFromActor(messageJson['actor'])
|
|
|
|
actorDomain, actorPort = getDomainFromActor(messageJson['actor'])
|
|
|
|
if isBlocked(baseDir, nickname, domain, actorNickname, actorDomain):
|
|
|
|
print('Receive announce blocked for actor: ' +
|
|
|
|
actorNickname + '@' + actorDomain)
|
|
|
|
return False
|
|
|
|
|
2021-08-23 12:30:16 +00:00
|
|
|
# also check the actor for the url being announced
|
|
|
|
announcedActorNickname = getNicknameFromActor(messageJson['object'])
|
|
|
|
announcedActorDomain, announcedActorPort = \
|
|
|
|
getDomainFromActor(messageJson['object'])
|
|
|
|
if isBlocked(baseDir, nickname, domain,
|
|
|
|
announcedActorNickname, announcedActorDomain):
|
2021-08-23 12:31:37 +00:00
|
|
|
print('Receive announce object blocked for actor: ' +
|
2021-08-23 12:30:16 +00:00
|
|
|
announcedActorNickname + '@' + announcedActorDomain)
|
|
|
|
return False
|
|
|
|
|
2020-05-25 09:21:34 +00:00
|
|
|
# is this post in the outbox of the person?
|
|
|
|
postFilename = locatePost(baseDir, nickname, domain,
|
2020-04-03 16:27:34 +00:00
|
|
|
messageJson['object'])
|
2019-07-11 19:31:02 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: announce post not found in inbox or outbox')
|
|
|
|
print(messageJson['object'])
|
|
|
|
return True
|
2020-04-03 16:27:34 +00:00
|
|
|
updateAnnounceCollection(recentPostsCache, baseDir, postFilename,
|
2021-05-19 19:42:45 +00:00
|
|
|
messageJson['actor'], nickname, domain, debug)
|
2019-09-29 10:13:00 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Downloading announce post ' + messageJson['actor'] +
|
|
|
|
' -> ' + messageJson['object'])
|
2021-07-19 08:46:21 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
2021-09-04 16:11:48 +00:00
|
|
|
|
2021-09-04 17:10:29 +00:00
|
|
|
# Generate html. This also downloads the announced post.
|
2021-09-04 16:11:48 +00:00
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, nickname, domain)
|
|
|
|
notDM = True
|
2021-09-04 18:05:52 +00:00
|
|
|
if debug:
|
|
|
|
print('Generating html for announce ' + messageJson['id'])
|
2021-09-04 16:11:48 +00:00
|
|
|
announceHtml = \
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, True,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
nickname, domain, port, messageJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
2021-09-18 17:08:14 +00:00
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
2021-09-04 16:11:48 +00:00
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
2021-10-21 19:00:25 +00:00
|
|
|
False, True, False, CWlists,
|
|
|
|
listsEnabled)
|
2021-09-04 16:11:48 +00:00
|
|
|
if not announceHtml:
|
|
|
|
print('WARN: Unable to generate html for announce ' +
|
|
|
|
str(messageJson))
|
|
|
|
else:
|
2021-09-04 18:05:52 +00:00
|
|
|
if debug:
|
|
|
|
print('Generated announce html ' + announceHtml.replace('\n', ''))
|
2021-09-04 16:11:48 +00:00
|
|
|
|
2021-03-03 17:09:31 +00:00
|
|
|
postJsonObject = downloadAnnounce(session, baseDir,
|
|
|
|
httpPrefix,
|
|
|
|
nickname, domain,
|
|
|
|
messageJson,
|
|
|
|
__version__, translate,
|
|
|
|
YTReplacementDomain,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain,
|
2021-03-05 19:23:33 +00:00
|
|
|
allowLocalNetworkAccess,
|
2021-07-18 14:15:16 +00:00
|
|
|
recentPostsCache, debug,
|
2021-07-19 08:46:21 +00:00
|
|
|
systemLanguage,
|
2021-08-31 14:17:11 +00:00
|
|
|
domainFull, personCache,
|
2021-09-13 11:34:56 +00:00
|
|
|
signingPrivateKeyPem,
|
|
|
|
blockedCache)
|
2020-12-21 21:40:29 +00:00
|
|
|
if not postJsonObject:
|
2021-09-04 14:32:02 +00:00
|
|
|
print('WARN: unable to download announce: ' + str(messageJson))
|
2021-02-28 18:21:12 +00:00
|
|
|
notInOnion = True
|
|
|
|
if onionDomain:
|
|
|
|
if onionDomain in messageJson['object']:
|
|
|
|
notInOnion = False
|
|
|
|
if domain not in messageJson['object'] and notInOnion:
|
2020-12-21 21:40:29 +00:00
|
|
|
if os.path.isfile(postFilename):
|
|
|
|
# if the announce can't be downloaded then remove it
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(postFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: _receiveAnnounce unable to delete ' +
|
|
|
|
str(postFilename))
|
2020-12-21 21:40:29 +00:00
|
|
|
else:
|
2019-10-01 13:23:22 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Announce post downloaded for ' +
|
|
|
|
messageJson['actor'] + ' -> ' + messageJson['object'])
|
2021-10-20 13:33:34 +00:00
|
|
|
storeHashTags(baseDir, nickname, domain,
|
|
|
|
httpPrefix, domainFull,
|
|
|
|
postJsonObject, translate)
|
2019-09-30 19:13:14 +00:00
|
|
|
# Try to obtain the actor for this person
|
|
|
|
# so that their avatar can be shown
|
2020-04-03 16:27:34 +00:00
|
|
|
lookupActor = None
|
2019-10-01 14:11:15 +00:00
|
|
|
if postJsonObject.get('attributedTo'):
|
2020-08-06 16:21:46 +00:00
|
|
|
if isinstance(postJsonObject['attributedTo'], str):
|
|
|
|
lookupActor = postJsonObject['attributedTo']
|
2019-10-01 14:11:15 +00:00
|
|
|
else:
|
2021-06-22 15:45:59 +00:00
|
|
|
if hasObjectDict(postJsonObject):
|
|
|
|
if postJsonObject['object'].get('attributedTo'):
|
|
|
|
attrib = postJsonObject['object']['attributedTo']
|
|
|
|
if isinstance(attrib, str):
|
|
|
|
lookupActor = attrib
|
2019-09-30 19:13:14 +00:00
|
|
|
if lookupActor:
|
2020-12-23 10:57:44 +00:00
|
|
|
if hasUsersPath(lookupActor):
|
2019-10-01 13:23:22 +00:00
|
|
|
if '/statuses/' in lookupActor:
|
2020-04-03 16:27:34 +00:00
|
|
|
lookupActor = lookupActor.split('/statuses/')[0]
|
2019-10-01 12:35:39 +00:00
|
|
|
|
2021-10-29 22:40:09 +00:00
|
|
|
if isRecentPost(postJsonObject, 3):
|
2021-03-03 20:16:53 +00:00
|
|
|
if not os.path.isfile(postFilename + '.tts'):
|
2021-03-09 13:52:02 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
|
|
|
updateSpeaker(baseDir, httpPrefix,
|
|
|
|
nickname, domain, domainFull,
|
2021-03-03 20:16:53 +00:00
|
|
|
postJsonObject, personCache,
|
2021-03-09 19:52:10 +00:00
|
|
|
translate, lookupActor,
|
|
|
|
themeName)
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(postFilename + '.tts', 'w+') as ttsFile:
|
|
|
|
ttsFile.write('\n')
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write recent post ' +
|
2021-11-25 21:18:53 +00:00
|
|
|
postFilename)
|
2021-03-01 22:27:36 +00:00
|
|
|
|
2019-10-01 12:50:06 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Obtaining actor for announce post ' +
|
|
|
|
lookupActor)
|
2019-10-01 13:23:22 +00:00
|
|
|
for tries in range(6):
|
2020-04-03 16:27:34 +00:00
|
|
|
pubKey = \
|
|
|
|
getPersonPubKey(baseDir, session, lookupActor,
|
|
|
|
personCache, debug,
|
|
|
|
__version__, httpPrefix,
|
2021-08-31 14:17:11 +00:00
|
|
|
domain, onionDomain,
|
|
|
|
signingPrivateKeyPem)
|
2019-10-01 13:23:22 +00:00
|
|
|
if pubKey:
|
2021-03-14 20:15:44 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: public key obtained for announce: ' +
|
|
|
|
lookupActor)
|
2019-10-01 13:23:22 +00:00
|
|
|
break
|
|
|
|
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Retry ' + str(tries + 1) +
|
|
|
|
' obtaining actor for ' + lookupActor)
|
2020-03-22 21:16:02 +00:00
|
|
|
time.sleep(5)
|
2020-12-21 21:40:29 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: announced/repeated post arrived in inbox')
|
2019-07-11 19:31:02 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _receiveUndoAnnounce(recentPostsCache: {},
|
|
|
|
session, handle: str, isGroup: bool, baseDir: str,
|
|
|
|
httpPrefix: str, domain: str, port: int,
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
|
|
|
personCache: {}, messageJson: {}, federationList: [],
|
|
|
|
debug: bool) -> bool:
|
2019-07-12 09:41:57 +00:00
|
|
|
"""Receives an undo announce activity within the POST section of HTTPServer
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['type'] != 'Undo':
|
2019-07-12 09:41:57 +00:00
|
|
|
return False
|
2021-10-13 09:33:15 +00:00
|
|
|
if not hasActor(messageJson, debug):
|
2019-07-12 09:41:57 +00:00
|
|
|
return False
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(messageJson):
|
2019-07-12 09:41:57 +00:00
|
|
|
return False
|
2021-10-13 10:37:52 +00:00
|
|
|
if not hasObjectStringObject(messageJson, debug):
|
2019-07-12 09:41:57 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
if messageJson['object']['type'] != 'Announce':
|
2020-03-22 21:16:02 +00:00
|
|
|
return False
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(messageJson['actor']):
|
2019-07-12 09:41:57 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: "users" or "profile" missing from actor in ' +
|
|
|
|
messageJson['type'] + ' announce')
|
2019-07-12 09:41:57 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
if not os.path.isdir(baseDir + '/accounts/' + handle):
|
|
|
|
print('DEBUG: unknown recipient of undo announce - ' + handle)
|
2019-07-12 09:41:57 +00:00
|
|
|
# if this post in the outbox of the person?
|
2020-12-22 21:24:46 +00:00
|
|
|
handleName = handle.split('@')[0]
|
|
|
|
handleDom = handle.split('@')[1]
|
|
|
|
postFilename = locatePost(baseDir, handleName, handleDom,
|
2020-04-03 16:27:34 +00:00
|
|
|
messageJson['object']['object'])
|
2019-07-12 09:41:57 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: undo announce post not found in inbox or outbox')
|
|
|
|
print(messageJson['object']['object'])
|
|
|
|
return True
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: announced/repeated post to be undone found in inbox')
|
2019-09-17 12:14:36 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
postJsonObject = loadJson(postFilename)
|
2019-10-22 11:55:06 +00:00
|
|
|
if postJsonObject:
|
2019-07-14 16:57:06 +00:00
|
|
|
if not postJsonObject.get('type'):
|
2020-04-03 16:27:34 +00:00
|
|
|
if postJsonObject['type'] != 'Announce':
|
2019-07-12 10:09:15 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print("DEBUG: Attempt to undo something " +
|
|
|
|
"which isn't an announcement")
|
2020-03-22 21:16:02 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
undoAnnounceCollectionEntry(recentPostsCache, baseDir, postFilename,
|
|
|
|
messageJson['actor'], domain, debug)
|
2019-10-13 12:22:27 +00:00
|
|
|
if os.path.isfile(postFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(postFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: _receiveUndoAnnounce unable to delete ' +
|
|
|
|
str(postFilename))
|
2019-07-12 09:41:57 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-08-21 18:32:16 +00:00
|
|
|
def jsonPostAllowsComments(postJsonObject: {}) -> bool:
|
|
|
|
"""Returns true if the given post allows comments/replies
|
|
|
|
"""
|
|
|
|
if 'commentsEnabled' in postJsonObject:
|
|
|
|
return postJsonObject['commentsEnabled']
|
2021-06-25 09:51:54 +00:00
|
|
|
if 'rejectReplies' in postJsonObject:
|
|
|
|
return not postJsonObject['rejectReplies']
|
2020-08-21 18:32:16 +00:00
|
|
|
if postJsonObject.get('object'):
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(postJsonObject):
|
2020-08-21 18:32:16 +00:00
|
|
|
return False
|
2021-06-22 15:45:59 +00:00
|
|
|
elif 'commentsEnabled' in postJsonObject['object']:
|
2020-08-21 18:32:16 +00:00
|
|
|
return postJsonObject['object']['commentsEnabled']
|
2021-06-25 09:51:54 +00:00
|
|
|
elif 'rejectReplies' in postJsonObject['object']:
|
|
|
|
return not postJsonObject['object']['rejectReplies']
|
2020-08-21 18:32:16 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _postAllowsComments(postFilename: str) -> bool:
|
2020-08-21 18:32:16 +00:00
|
|
|
"""Returns true if the given post allows comments/replies
|
|
|
|
"""
|
|
|
|
postJsonObject = loadJson(postFilename)
|
|
|
|
if not postJsonObject:
|
|
|
|
return False
|
|
|
|
return jsonPostAllowsComments(postJsonObject)
|
|
|
|
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
def populateReplies(baseDir: str, httpPrefix: str, domain: str,
|
|
|
|
messageJson: {}, maxReplies: int, debug: bool) -> bool:
|
2020-03-22 21:16:02 +00:00
|
|
|
"""Updates the list of replies for a post on this domain if
|
2019-07-12 12:35:38 +00:00
|
|
|
a reply to it arrives
|
|
|
|
"""
|
|
|
|
if not messageJson.get('id'):
|
|
|
|
return False
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(messageJson):
|
2019-07-13 20:16:07 +00:00
|
|
|
return False
|
|
|
|
if not messageJson['object'].get('inReplyTo'):
|
|
|
|
return False
|
|
|
|
if not messageJson['object'].get('to'):
|
2019-07-12 12:35:38 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
replyTo = messageJson['object']['inReplyTo']
|
2020-08-28 14:45:07 +00:00
|
|
|
if not isinstance(replyTo, str):
|
|
|
|
return False
|
2019-07-12 12:35:38 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: post contains a reply')
|
|
|
|
# is this a reply to a post on this domain?
|
2020-04-03 16:27:34 +00:00
|
|
|
if not replyTo.startswith(httpPrefix + '://' + domain + '/'):
|
2019-07-12 12:35:38 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: post is a reply to another not on this domain')
|
2019-08-02 18:04:31 +00:00
|
|
|
print(replyTo)
|
2020-04-03 16:27:34 +00:00
|
|
|
print('Expected: ' + httpPrefix + '://' + domain + '/')
|
2019-07-12 12:35:38 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
replyToNickname = getNicknameFromActor(replyTo)
|
2019-07-12 12:35:38 +00:00
|
|
|
if not replyToNickname:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: no nickname found for ' + replyTo)
|
2019-07-12 12:35:38 +00:00
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
replyToDomain, replyToPort = getDomainFromActor(replyTo)
|
2019-07-12 12:35:38 +00:00
|
|
|
if not replyToDomain:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: no domain found for ' + replyTo)
|
2019-07-12 12:35:38 +00:00
|
|
|
return False
|
2021-03-23 23:33:33 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
postFilename = locatePost(baseDir, replyToNickname,
|
|
|
|
replyToDomain, replyTo)
|
2019-07-12 12:35:38 +00:00
|
|
|
if not postFilename:
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: post may have expired - ' + replyTo)
|
2020-03-22 21:16:02 +00:00
|
|
|
return False
|
2021-03-23 22:52:00 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if not _postAllowsComments(postFilename):
|
2020-08-21 18:32:16 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: post does not allow comments - ' + replyTo)
|
|
|
|
return False
|
2019-07-13 19:28:14 +00:00
|
|
|
# populate a text file containing the ids of replies
|
2020-04-03 16:27:34 +00:00
|
|
|
postRepliesFilename = postFilename.replace('.json', '.replies')
|
2020-08-23 11:13:35 +00:00
|
|
|
messageId = removeIdEnding(messageJson['id'])
|
2019-07-13 19:28:14 +00:00
|
|
|
if os.path.isfile(postRepliesFilename):
|
2020-04-03 16:27:34 +00:00
|
|
|
numLines = sum(1 for line in open(postRepliesFilename))
|
|
|
|
if numLines > maxReplies:
|
2019-07-13 21:00:12 +00:00
|
|
|
return False
|
2019-07-13 19:28:14 +00:00
|
|
|
if messageId not in open(postRepliesFilename).read():
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(postRepliesFilename, 'a+') as repliesFile:
|
|
|
|
repliesFile.write(messageId + '\n')
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to append ' + postRepliesFilename)
|
2019-07-13 19:28:14 +00:00
|
|
|
else:
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(postRepliesFilename, 'w+') as repliesFile:
|
|
|
|
repliesFile.write(messageId + '\n')
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + postRepliesFilename)
|
2019-07-13 19:28:14 +00:00
|
|
|
return True
|
2019-09-30 09:43:46 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _estimateNumberOfMentions(content: str) -> int:
|
2019-09-30 10:15:20 +00:00
|
|
|
"""Returns a rough estimate of the number of mentions
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
return int(content.count('@') / 2)
|
|
|
|
|
2019-11-16 14:49:21 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _estimateNumberOfEmoji(content: str) -> int:
|
2019-11-16 14:49:21 +00:00
|
|
|
"""Returns a rough estimate of the number of emoji
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
return int(content.count(':') / 2)
|
2019-11-16 14:49:21 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _validPostContent(baseDir: str, nickname: str, domain: str,
|
|
|
|
messageJson: {}, maxMentions: int, maxEmoji: int,
|
2021-07-18 14:15:16 +00:00
|
|
|
allowLocalNetworkAccess: bool, debug: bool,
|
2021-07-19 08:46:21 +00:00
|
|
|
systemLanguage: str,
|
|
|
|
httpPrefix: str, domainFull: str,
|
|
|
|
personCache: {}) -> bool:
|
2019-09-30 09:43:46 +00:00
|
|
|
"""Is the content of a received post valid?
|
2019-09-30 10:15:20 +00:00
|
|
|
Check for bad html
|
|
|
|
Check for hellthreads
|
2019-09-30 10:35:49 +00:00
|
|
|
Check number of tags is reasonable
|
2019-09-30 09:43:46 +00:00
|
|
|
"""
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(messageJson):
|
2019-09-30 09:43:46 +00:00
|
|
|
return True
|
|
|
|
if not messageJson['object'].get('content'):
|
|
|
|
return True
|
2019-11-29 22:45:56 +00:00
|
|
|
|
|
|
|
if not messageJson['object'].get('published'):
|
|
|
|
return False
|
|
|
|
if 'T' not in messageJson['object']['published']:
|
|
|
|
return False
|
|
|
|
if 'Z' not in messageJson['object']['published']:
|
|
|
|
return False
|
2021-03-14 19:53:22 +00:00
|
|
|
if not validPostDate(messageJson['object']['published'], 90, debug):
|
2020-12-21 10:45:31 +00:00
|
|
|
return False
|
2020-07-10 14:15:01 +00:00
|
|
|
|
2021-08-19 22:16:32 +00:00
|
|
|
summary = None
|
2020-08-25 19:45:15 +00:00
|
|
|
if messageJson['object'].get('summary'):
|
|
|
|
summary = messageJson['object']['summary']
|
|
|
|
if not isinstance(summary, str):
|
|
|
|
print('WARN: content warning is not a string')
|
|
|
|
return False
|
|
|
|
if summary != validContentWarning(summary):
|
|
|
|
print('WARN: invalid content warning ' + summary)
|
|
|
|
return False
|
|
|
|
|
2021-09-19 09:03:19 +00:00
|
|
|
# check for patches before dangeousMarkup, which excludes code
|
2020-05-02 11:08:38 +00:00
|
|
|
if isGitPatch(baseDir, nickname, domain,
|
2020-05-03 10:56:29 +00:00
|
|
|
messageJson['object']['type'],
|
2021-08-19 22:16:32 +00:00
|
|
|
summary,
|
2020-05-02 11:08:38 +00:00
|
|
|
messageJson['object']['content']):
|
|
|
|
return True
|
2020-07-10 14:15:01 +00:00
|
|
|
|
2021-07-20 13:33:27 +00:00
|
|
|
contentStr = getBaseContentFromPost(messageJson, systemLanguage)
|
2021-07-18 14:15:16 +00:00
|
|
|
if dangerousMarkup(contentStr, allowLocalNetworkAccess):
|
2020-07-10 14:15:01 +00:00
|
|
|
if messageJson['object'].get('id'):
|
|
|
|
print('REJECT ARBITRARY HTML: ' + messageJson['object']['id'])
|
|
|
|
print('REJECT ARBITRARY HTML: bad string in post - ' +
|
2021-07-18 14:15:16 +00:00
|
|
|
contentStr)
|
2020-07-10 14:15:01 +00:00
|
|
|
return False
|
|
|
|
|
2019-09-30 10:35:49 +00:00
|
|
|
# check (rough) number of mentions
|
2021-07-18 14:15:16 +00:00
|
|
|
mentionsEst = _estimateNumberOfMentions(contentStr)
|
2020-04-03 16:27:34 +00:00
|
|
|
if mentionsEst > maxMentions:
|
2019-09-30 10:37:34 +00:00
|
|
|
if messageJson['object'].get('id'):
|
2020-04-03 16:27:34 +00:00
|
|
|
print('REJECT HELLTHREAD: ' + messageJson['object']['id'])
|
|
|
|
print('REJECT HELLTHREAD: Too many mentions in post - ' +
|
2021-07-18 14:15:16 +00:00
|
|
|
contentStr)
|
2019-11-16 14:49:21 +00:00
|
|
|
return False
|
2021-07-18 14:15:16 +00:00
|
|
|
if _estimateNumberOfEmoji(contentStr) > maxEmoji:
|
2019-11-16 14:49:21 +00:00
|
|
|
if messageJson['object'].get('id'):
|
2020-04-03 16:27:34 +00:00
|
|
|
print('REJECT EMOJI OVERLOAD: ' + messageJson['object']['id'])
|
|
|
|
print('REJECT EMOJI OVERLOAD: Too many emoji in post - ' +
|
2021-07-18 14:15:16 +00:00
|
|
|
contentStr)
|
2019-09-30 10:15:20 +00:00
|
|
|
return False
|
2019-09-30 10:35:49 +00:00
|
|
|
# check number of tags
|
|
|
|
if messageJson['object'].get('tag'):
|
2019-09-30 11:05:35 +00:00
|
|
|
if not isinstance(messageJson['object']['tag'], list):
|
2020-04-03 16:27:34 +00:00
|
|
|
messageJson['object']['tag'] = []
|
2019-09-30 11:05:35 +00:00
|
|
|
else:
|
2020-04-03 16:27:34 +00:00
|
|
|
if len(messageJson['object']['tag']) > int(maxMentions * 2):
|
2019-09-30 10:37:34 +00:00
|
|
|
if messageJson['object'].get('id'):
|
2020-04-03 16:27:34 +00:00
|
|
|
print('REJECT: ' + messageJson['object']['id'])
|
|
|
|
print('REJECT: Too many tags in post - ' +
|
|
|
|
messageJson['object']['tag'])
|
2019-09-30 10:35:49 +00:00
|
|
|
return False
|
2021-07-18 18:33:53 +00:00
|
|
|
# check that the post is in a language suitable for this account
|
2021-07-18 19:35:34 +00:00
|
|
|
if not understoodPostLanguage(baseDir, nickname, domain,
|
2021-07-19 08:46:21 +00:00
|
|
|
messageJson, systemLanguage,
|
|
|
|
httpPrefix, domainFull,
|
|
|
|
personCache):
|
2021-07-18 18:33:53 +00:00
|
|
|
return False
|
2020-02-05 17:29:38 +00:00
|
|
|
# check for filtered content
|
2021-07-18 14:15:16 +00:00
|
|
|
if isFiltered(baseDir, nickname, domain, contentStr):
|
2020-02-05 17:29:38 +00:00
|
|
|
print('REJECT: content filtered')
|
|
|
|
return False
|
2020-08-21 18:32:16 +00:00
|
|
|
if messageJson['object'].get('inReplyTo'):
|
|
|
|
if isinstance(messageJson['object']['inReplyTo'], str):
|
|
|
|
originalPostId = messageJson['object']['inReplyTo']
|
|
|
|
postPostFilename = locatePost(baseDir, nickname, domain,
|
|
|
|
originalPostId)
|
|
|
|
if postPostFilename:
|
2020-12-22 18:06:23 +00:00
|
|
|
if not _postAllowsComments(postPostFilename):
|
2020-08-21 18:32:16 +00:00
|
|
|
print('REJECT: reply to post which does not ' +
|
|
|
|
'allow comments: ' + originalPostId)
|
|
|
|
return False
|
2021-11-22 12:05:09 +00:00
|
|
|
if invalidCiphertext(messageJson['object']['content']):
|
2021-11-22 11:59:41 +00:00
|
|
|
print('REJECT: malformed ciphertext in content')
|
|
|
|
return False
|
2021-03-14 20:15:44 +00:00
|
|
|
if debug:
|
|
|
|
print('ACCEPT: post content is valid')
|
2019-09-30 09:43:46 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _obtainAvatarForReplyPost(session, baseDir: str, httpPrefix: str,
|
|
|
|
domain: str, onionDomain: str, personCache: {},
|
2021-08-31 14:17:11 +00:00
|
|
|
postJsonObject: {}, debug: bool,
|
|
|
|
signingPrivateKeyPem: str) -> None:
|
2019-09-30 19:23:53 +00:00
|
|
|
"""Tries to obtain the actor for the person being replied to
|
|
|
|
so that their avatar can later be shown
|
|
|
|
"""
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(postJsonObject):
|
2019-09-30 19:39:48 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if not postJsonObject['object'].get('inReplyTo'):
|
|
|
|
return
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
lookupActor = postJsonObject['object']['inReplyTo']
|
2019-10-21 12:49:16 +00:00
|
|
|
if not lookupActor:
|
|
|
|
return
|
|
|
|
|
2020-08-28 14:45:07 +00:00
|
|
|
if not isinstance(lookupActor, str):
|
|
|
|
return
|
|
|
|
|
2020-12-23 10:57:44 +00:00
|
|
|
if not hasUsersPath(lookupActor):
|
2019-10-21 12:49:16 +00:00
|
|
|
return
|
|
|
|
|
|
|
|
if '/statuses/' in lookupActor:
|
2020-04-03 16:27:34 +00:00
|
|
|
lookupActor = lookupActor.split('/statuses/')[0]
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2019-10-21 12:49:16 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Obtaining actor for reply post ' + lookupActor)
|
2019-10-01 13:23:22 +00:00
|
|
|
|
2019-10-21 12:49:16 +00:00
|
|
|
for tries in range(6):
|
2020-04-03 16:27:34 +00:00
|
|
|
pubKey = \
|
|
|
|
getPersonPubKey(baseDir, session, lookupActor,
|
|
|
|
personCache, debug,
|
|
|
|
__version__, httpPrefix,
|
2021-08-31 14:17:11 +00:00
|
|
|
domain, onionDomain, signingPrivateKeyPem)
|
2019-10-21 12:49:16 +00:00
|
|
|
if pubKey:
|
2021-03-14 20:15:44 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: public key obtained for reply: ' + lookupActor)
|
2019-10-21 12:49:16 +00:00
|
|
|
break
|
|
|
|
|
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Retry ' + str(tries + 1) +
|
|
|
|
' obtaining actor for ' + lookupActor)
|
2020-03-22 21:16:02 +00:00
|
|
|
time.sleep(5)
|
2019-09-30 19:23:53 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _dmNotify(baseDir: str, handle: str, url: str) -> None:
|
2019-10-03 16:22:34 +00:00
|
|
|
"""Creates a notification that a new DM has arrived
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
accountDir = baseDir + '/accounts/' + handle
|
2019-10-03 16:22:34 +00:00
|
|
|
if not os.path.isdir(accountDir):
|
|
|
|
return
|
2020-04-03 16:27:34 +00:00
|
|
|
dmFile = accountDir + '/.newDM'
|
2019-10-03 16:22:34 +00:00
|
|
|
if not os.path.isfile(dmFile):
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(dmFile, 'w+') as fp:
|
|
|
|
fp.write(url)
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + dmFile)
|
2019-10-03 16:22:34 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _alreadyLiked(baseDir: str, nickname: str, domain: str,
|
|
|
|
postUrl: str, likerActor: str) -> bool:
|
2020-07-13 13:36:45 +00:00
|
|
|
"""Is the given post already liked by the given handle?
|
|
|
|
"""
|
|
|
|
postFilename = \
|
|
|
|
locatePost(baseDir, nickname, domain, postUrl)
|
|
|
|
if not postFilename:
|
|
|
|
return False
|
|
|
|
postJsonObject = loadJson(postFilename, 1)
|
|
|
|
if not postJsonObject:
|
|
|
|
return False
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(postJsonObject):
|
2020-07-13 13:36:45 +00:00
|
|
|
return False
|
|
|
|
if not postJsonObject['object'].get('likes'):
|
|
|
|
return False
|
|
|
|
if not postJsonObject['object']['likes'].get('items'):
|
|
|
|
return False
|
|
|
|
for like in postJsonObject['object']['likes']['items']:
|
|
|
|
if not like.get('type'):
|
|
|
|
continue
|
|
|
|
if not like.get('actor'):
|
|
|
|
continue
|
|
|
|
if like['type'] != 'Like':
|
|
|
|
continue
|
|
|
|
if like['actor'] == likerActor:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2021-11-10 12:16:03 +00:00
|
|
|
def _alreadyReacted(baseDir: str, nickname: str, domain: str,
|
|
|
|
postUrl: str, reactionActor: str,
|
|
|
|
emojiContent: str) -> bool:
|
|
|
|
"""Is the given post already emoji reacted by the given handle?
|
|
|
|
"""
|
|
|
|
postFilename = \
|
|
|
|
locatePost(baseDir, nickname, domain, postUrl)
|
|
|
|
if not postFilename:
|
|
|
|
return False
|
|
|
|
postJsonObject = loadJson(postFilename, 1)
|
|
|
|
if not postJsonObject:
|
|
|
|
return False
|
|
|
|
if not hasObjectDict(postJsonObject):
|
|
|
|
return False
|
|
|
|
if not postJsonObject['object'].get('reactions'):
|
|
|
|
return False
|
|
|
|
if not postJsonObject['object']['reactions'].get('items'):
|
|
|
|
return False
|
|
|
|
for react in postJsonObject['object']['reactions']['items']:
|
|
|
|
if not react.get('type'):
|
|
|
|
continue
|
|
|
|
if not react.get('content'):
|
|
|
|
continue
|
|
|
|
if not react.get('actor'):
|
|
|
|
continue
|
|
|
|
if react['type'] != 'EmojiReact':
|
|
|
|
continue
|
|
|
|
if react['content'] != emojiContent:
|
|
|
|
continue
|
|
|
|
if react['actor'] == reactionActor:
|
|
|
|
return True
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _likeNotify(baseDir: str, domain: str, onionDomain: str,
|
|
|
|
handle: str, actor: str, url: str) -> None:
|
2020-07-08 19:49:15 +00:00
|
|
|
"""Creates a notification that a like has arrived
|
|
|
|
"""
|
2020-07-08 22:04:17 +00:00
|
|
|
# This is not you liking your own post
|
|
|
|
if actor in url:
|
|
|
|
return
|
|
|
|
|
|
|
|
# check that the liked post was by this handle
|
|
|
|
nickname = handle.split('@')[0]
|
|
|
|
if '/' + domain + '/users/' + nickname not in url:
|
|
|
|
if not onionDomain:
|
|
|
|
return
|
|
|
|
if '/' + onionDomain + '/users/' + nickname not in url:
|
|
|
|
return
|
|
|
|
|
2020-07-08 19:49:15 +00:00
|
|
|
accountDir = baseDir + '/accounts/' + handle
|
2020-08-27 09:35:26 +00:00
|
|
|
|
|
|
|
# are like notifications enabled?
|
2020-08-27 09:19:32 +00:00
|
|
|
notifyLikesEnabledFilename = accountDir + '/.notifyLikes'
|
|
|
|
if not os.path.isfile(notifyLikesEnabledFilename):
|
2020-07-08 19:49:15 +00:00
|
|
|
return
|
2020-08-27 09:19:32 +00:00
|
|
|
|
2020-07-08 19:49:15 +00:00
|
|
|
likeFile = accountDir + '/.newLike'
|
2020-07-08 21:41:48 +00:00
|
|
|
if os.path.isfile(likeFile):
|
2020-07-08 21:52:18 +00:00
|
|
|
if '##sent##' not in open(likeFile).read():
|
|
|
|
return
|
2020-07-08 19:49:15 +00:00
|
|
|
|
|
|
|
likerNickname = getNicknameFromActor(actor)
|
|
|
|
likerDomain, likerPort = getDomainFromActor(actor)
|
2020-07-08 21:04:19 +00:00
|
|
|
if likerNickname and likerDomain:
|
|
|
|
likerHandle = likerNickname + '@' + likerDomain
|
|
|
|
else:
|
2020-12-22 18:06:23 +00:00
|
|
|
print('_likeNotify likerHandle: ' +
|
2020-07-08 21:18:50 +00:00
|
|
|
str(likerNickname) + '@' + str(likerDomain))
|
2020-07-08 21:04:19 +00:00
|
|
|
likerHandle = actor
|
2020-07-08 19:49:15 +00:00
|
|
|
if likerHandle != handle:
|
2020-07-13 19:42:30 +00:00
|
|
|
likeStr = likerHandle + ' ' + url + '?likedBy=' + actor
|
2020-07-08 22:17:21 +00:00
|
|
|
prevLikeFile = accountDir + '/.prevLike'
|
|
|
|
# was there a previous like notification?
|
|
|
|
if os.path.isfile(prevLikeFile):
|
|
|
|
# is it the same as the current notification ?
|
2021-06-21 22:52:04 +00:00
|
|
|
with open(prevLikeFile, 'r') as fp:
|
|
|
|
prevLikeStr = fp.read()
|
|
|
|
if prevLikeStr == likeStr:
|
|
|
|
return
|
2021-06-21 22:53:04 +00:00
|
|
|
try:
|
|
|
|
with open(prevLikeFile, 'w+') as fp:
|
|
|
|
fp.write(likeStr)
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: ERROR: unable to save previous like notification ' +
|
2021-06-21 22:53:04 +00:00
|
|
|
prevLikeFile)
|
2021-11-25 21:18:53 +00:00
|
|
|
|
2021-06-21 22:53:04 +00:00
|
|
|
try:
|
|
|
|
with open(likeFile, 'w+') as fp:
|
|
|
|
fp.write(likeStr)
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: ERROR: unable to write like notification file ' +
|
2021-06-21 22:53:04 +00:00
|
|
|
likeFile)
|
2020-07-08 19:49:15 +00:00
|
|
|
|
|
|
|
|
2021-11-10 12:16:03 +00:00
|
|
|
def _reactionNotify(baseDir: str, domain: str, onionDomain: str,
|
|
|
|
handle: str, actor: str,
|
|
|
|
url: str, emojiContent: str) -> None:
|
|
|
|
"""Creates a notification that an emoji reaction has arrived
|
|
|
|
"""
|
|
|
|
# This is not you reacting to your own post
|
|
|
|
if actor in url:
|
|
|
|
return
|
|
|
|
|
|
|
|
# check that the reaction post was by this handle
|
|
|
|
nickname = handle.split('@')[0]
|
|
|
|
if '/' + domain + '/users/' + nickname not in url:
|
|
|
|
if not onionDomain:
|
|
|
|
return
|
|
|
|
if '/' + onionDomain + '/users/' + nickname not in url:
|
|
|
|
return
|
|
|
|
|
|
|
|
accountDir = baseDir + '/accounts/' + handle
|
|
|
|
|
|
|
|
# are reaction notifications enabled?
|
2021-11-10 19:33:28 +00:00
|
|
|
notifyReactionEnabledFilename = accountDir + '/.notifyReactions'
|
2021-11-10 12:16:03 +00:00
|
|
|
if not os.path.isfile(notifyReactionEnabledFilename):
|
|
|
|
return
|
|
|
|
|
|
|
|
reactionFile = accountDir + '/.newReaction'
|
|
|
|
if os.path.isfile(reactionFile):
|
|
|
|
if '##sent##' not in open(reactionFile).read():
|
|
|
|
return
|
|
|
|
|
|
|
|
reactionNickname = getNicknameFromActor(actor)
|
|
|
|
reactionDomain, reactionPort = getDomainFromActor(actor)
|
|
|
|
if reactionNickname and reactionDomain:
|
|
|
|
reactionHandle = reactionNickname + '@' + reactionDomain
|
|
|
|
else:
|
|
|
|
print('_reactionNotify reactionHandle: ' +
|
|
|
|
str(reactionNickname) + '@' + str(reactionDomain))
|
|
|
|
reactionHandle = actor
|
|
|
|
if reactionHandle != handle:
|
|
|
|
reactionStr = \
|
2021-11-10 19:33:28 +00:00
|
|
|
reactionHandle + ' ' + url + '?reactBy=' + actor + \
|
|
|
|
';emoj=' + emojiContent
|
2021-11-10 12:16:03 +00:00
|
|
|
prevReactionFile = accountDir + '/.prevReaction'
|
|
|
|
# was there a previous reaction notification?
|
|
|
|
if os.path.isfile(prevReactionFile):
|
|
|
|
# is it the same as the current notification ?
|
|
|
|
with open(prevReactionFile, 'r') as fp:
|
|
|
|
prevReactionStr = fp.read()
|
|
|
|
if prevReactionStr == reactionStr:
|
|
|
|
return
|
|
|
|
try:
|
|
|
|
with open(prevReactionFile, 'w+') as fp:
|
|
|
|
fp.write(reactionStr)
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError:
|
2021-11-10 12:16:03 +00:00
|
|
|
print('EX: ERROR: unable to save previous reaction notification ' +
|
|
|
|
prevReactionFile)
|
2021-11-25 21:18:53 +00:00
|
|
|
|
2021-11-10 12:16:03 +00:00
|
|
|
try:
|
|
|
|
with open(reactionFile, 'w+') as fp:
|
|
|
|
fp.write(reactionStr)
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError:
|
2021-11-10 12:16:03 +00:00
|
|
|
print('EX: ERROR: unable to write reaction notification file ' +
|
|
|
|
reactionFile)
|
|
|
|
|
|
|
|
|
2021-07-06 20:38:08 +00:00
|
|
|
def _notifyPostArrival(baseDir: str, handle: str, url: str) -> None:
|
2021-07-07 09:32:48 +00:00
|
|
|
"""Creates a notification that a new post has arrived.
|
|
|
|
This is for followed accounts with the notify checkbox enabled
|
|
|
|
on the person options screen
|
2021-07-06 20:38:08 +00:00
|
|
|
"""
|
|
|
|
accountDir = baseDir + '/accounts/' + handle
|
|
|
|
if not os.path.isdir(accountDir):
|
|
|
|
return
|
|
|
|
notifyFile = accountDir + '/.newNotifiedPost'
|
2021-07-07 09:30:15 +00:00
|
|
|
if os.path.isfile(notifyFile):
|
|
|
|
# check that the same notification is not repeatedly sent
|
|
|
|
with open(notifyFile, 'r') as fp:
|
|
|
|
existingNotificationMessage = fp.read()
|
|
|
|
if url in existingNotificationMessage:
|
|
|
|
return
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(notifyFile, 'w+') as fp:
|
|
|
|
fp.write(url)
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + notifyFile)
|
2021-07-06 20:38:08 +00:00
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _replyNotify(baseDir: str, handle: str, url: str) -> None:
|
2019-10-03 16:37:25 +00:00
|
|
|
"""Creates a notification that a new reply has arrived
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
accountDir = baseDir + '/accounts/' + handle
|
2019-10-03 16:37:25 +00:00
|
|
|
if not os.path.isdir(accountDir):
|
|
|
|
return
|
2020-04-03 16:27:34 +00:00
|
|
|
replyFile = accountDir + '/.newReply'
|
2019-10-03 16:37:25 +00:00
|
|
|
if not os.path.isfile(replyFile):
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(replyFile, 'w+') as fp:
|
|
|
|
fp.write(url)
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + replyFile)
|
2019-10-03 16:37:25 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _gitPatchNotify(baseDir: str, handle: str,
|
|
|
|
subject: str, content: str,
|
|
|
|
fromNickname: str, fromDomain: str) -> None:
|
2020-05-02 11:20:57 +00:00
|
|
|
"""Creates a notification that a new git patch has arrived
|
|
|
|
"""
|
|
|
|
accountDir = baseDir + '/accounts/' + handle
|
|
|
|
if not os.path.isdir(accountDir):
|
|
|
|
return
|
|
|
|
patchFile = accountDir + '/.newPatch'
|
2020-05-03 09:58:51 +00:00
|
|
|
subject = subject.replace('[PATCH]', '').strip()
|
|
|
|
handle = '@' + fromNickname + '@' + fromDomain
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(patchFile, 'w+') as fp:
|
|
|
|
fp.write('git ' + handle + ' ' + subject)
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + patchFile)
|
2020-05-02 17:16:24 +00:00
|
|
|
|
2020-05-02 11:20:57 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _groupHandle(baseDir: str, handle: str) -> bool:
|
2019-10-04 12:22:56 +00:00
|
|
|
"""Is the given account handle a group?
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
actorFile = baseDir + '/accounts/' + handle + '.json'
|
2019-10-04 12:22:56 +00:00
|
|
|
if not os.path.isfile(actorFile):
|
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
actorJson = loadJson(actorFile)
|
2019-10-04 12:22:56 +00:00
|
|
|
if not actorJson:
|
|
|
|
return False
|
2020-04-03 16:27:34 +00:00
|
|
|
return actorJson['type'] == 'Group'
|
|
|
|
|
2019-10-04 12:22:56 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _sendToGroupMembers(session, baseDir: str, handle: str, port: int,
|
|
|
|
postJsonObject: {},
|
|
|
|
httpPrefix: str, federationList: [],
|
|
|
|
sendThreads: [], postLog: [], cachedWebfingers: {},
|
2021-07-18 14:15:16 +00:00
|
|
|
personCache: {}, debug: bool,
|
2021-08-02 18:55:53 +00:00
|
|
|
systemLanguage: str,
|
2021-08-31 14:17:11 +00:00
|
|
|
onionDomain: str, i2pDomain: str,
|
|
|
|
signingPrivateKeyPem: str) -> None:
|
2019-10-04 12:22:56 +00:00
|
|
|
"""When a post arrives for a group send it out to the group members
|
|
|
|
"""
|
2021-08-01 13:25:11 +00:00
|
|
|
if debug:
|
|
|
|
print('\n\n=========================================================')
|
|
|
|
print(handle + ' sending to group members')
|
2021-08-02 19:01:31 +00:00
|
|
|
|
|
|
|
sharedItemFederationTokens = {}
|
|
|
|
sharedItemsFederatedDomains = []
|
|
|
|
sharedItemsFederatedDomainsStr = \
|
|
|
|
getConfigParam(baseDir, 'sharedItemsFederatedDomains')
|
|
|
|
if sharedItemsFederatedDomainsStr:
|
|
|
|
siFederatedDomainsList = \
|
|
|
|
sharedItemsFederatedDomainsStr.split(',')
|
|
|
|
for sharedFederatedDomain in siFederatedDomainsList:
|
|
|
|
domainStr = sharedFederatedDomain.strip()
|
|
|
|
sharedItemsFederatedDomains.append(domainStr)
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
followersFile = baseDir + '/accounts/' + handle + '/followers.txt'
|
2019-10-04 12:22:56 +00:00
|
|
|
if not os.path.isfile(followersFile):
|
|
|
|
return
|
2021-08-02 14:19:09 +00:00
|
|
|
if not postJsonObject.get('to'):
|
|
|
|
return
|
2019-10-04 13:31:30 +00:00
|
|
|
if not postJsonObject.get('object'):
|
|
|
|
return
|
2021-07-30 19:20:49 +00:00
|
|
|
if not hasObjectDict(postJsonObject):
|
|
|
|
return
|
2021-08-02 14:19:09 +00:00
|
|
|
nickname = handle.split('@')[0].replace('!', '')
|
2020-04-03 16:27:34 +00:00
|
|
|
domain = handle.split('@')[1]
|
2020-12-16 10:48:40 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
2021-08-14 11:13:39 +00:00
|
|
|
groupActor = localActorUrl(httpPrefix, nickname, domainFull)
|
2021-08-02 14:19:09 +00:00
|
|
|
if groupActor not in postJsonObject['to']:
|
|
|
|
return
|
2020-04-03 16:27:34 +00:00
|
|
|
cc = ''
|
2021-08-02 14:19:09 +00:00
|
|
|
nickname = handle.split('@')[0].replace('!', '')
|
|
|
|
|
2021-09-18 09:57:03 +00:00
|
|
|
# save to the group outbox so that replies will be to the group
|
|
|
|
# rather than the original sender
|
|
|
|
savePostToBox(baseDir, httpPrefix, None,
|
|
|
|
nickname, domain, postJsonObject, 'outbox')
|
|
|
|
|
2021-09-28 10:28:42 +00:00
|
|
|
postId = removeIdEnding(postJsonObject['object']['id'])
|
2021-08-02 14:19:09 +00:00
|
|
|
if debug:
|
2021-09-17 11:14:16 +00:00
|
|
|
print('Group announce: ' + postId)
|
2021-08-02 14:19:09 +00:00
|
|
|
announceJson = \
|
|
|
|
createAnnounce(session, baseDir, federationList,
|
|
|
|
nickname, domain, port,
|
|
|
|
groupActor + '/followers', cc,
|
2021-09-17 11:14:16 +00:00
|
|
|
httpPrefix, postId, False, False,
|
2021-08-02 14:19:09 +00:00
|
|
|
sendThreads, postLog,
|
|
|
|
personCache, cachedWebfingers,
|
2021-08-31 14:17:11 +00:00
|
|
|
debug, __version__, signingPrivateKeyPem)
|
2021-08-02 14:19:09 +00:00
|
|
|
|
2021-08-02 18:55:53 +00:00
|
|
|
sendToFollowersThread(session, baseDir, nickname, domain,
|
|
|
|
onionDomain, i2pDomain, port,
|
|
|
|
httpPrefix, federationList,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers, personCache,
|
|
|
|
announceJson, debug, __version__,
|
|
|
|
sharedItemsFederatedDomains,
|
2021-08-31 14:17:11 +00:00
|
|
|
sharedItemFederationTokens,
|
|
|
|
signingPrivateKeyPem)
|
2020-04-03 16:27:34 +00:00
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _inboxUpdateCalendar(baseDir: str, handle: str,
|
|
|
|
postJsonObject: {}) -> None:
|
2019-10-11 12:31:06 +00:00
|
|
|
"""Detects whether the tag list on a post contains calendar events
|
|
|
|
and if so saves the post id to a file in the calendar directory
|
|
|
|
for the account
|
|
|
|
"""
|
2020-07-03 18:49:00 +00:00
|
|
|
if not postJsonObject.get('actor'):
|
|
|
|
return
|
2021-06-22 15:45:59 +00:00
|
|
|
if not hasObjectDict(postJsonObject):
|
2019-10-11 12:31:06 +00:00
|
|
|
return
|
|
|
|
if not postJsonObject['object'].get('tag'):
|
|
|
|
return
|
|
|
|
if not isinstance(postJsonObject['object']['tag'], list):
|
|
|
|
return
|
|
|
|
|
2020-07-03 18:49:00 +00:00
|
|
|
actor = postJsonObject['actor']
|
|
|
|
actorNickname = getNicknameFromActor(actor)
|
|
|
|
actorDomain, actorPort = getDomainFromActor(actor)
|
2020-07-11 21:01:08 +00:00
|
|
|
handleNickname = handle.split('@')[0]
|
|
|
|
handleDomain = handle.split('@')[1]
|
2020-07-03 18:49:00 +00:00
|
|
|
if not receivingCalendarEvents(baseDir,
|
2020-07-11 21:01:08 +00:00
|
|
|
handleNickname, handleDomain,
|
|
|
|
actorNickname, actorDomain):
|
2020-07-03 18:49:00 +00:00
|
|
|
return
|
2020-08-13 09:37:11 +00:00
|
|
|
|
2020-08-23 11:13:35 +00:00
|
|
|
postId = removeIdEnding(postJsonObject['id']).replace('/', '#')
|
2020-08-13 11:58:05 +00:00
|
|
|
|
2020-08-13 09:37:11 +00:00
|
|
|
# look for events within the tags list
|
2019-10-11 12:31:06 +00:00
|
|
|
for tagDict in postJsonObject['object']['tag']:
|
2020-07-11 22:36:52 +00:00
|
|
|
if not tagDict.get('type'):
|
|
|
|
continue
|
2020-04-03 16:27:34 +00:00
|
|
|
if tagDict['type'] != 'Event':
|
2019-10-11 12:31:06 +00:00
|
|
|
continue
|
2019-10-11 16:16:56 +00:00
|
|
|
if not tagDict.get('startTime'):
|
2019-10-11 12:31:06 +00:00
|
|
|
continue
|
2020-08-20 16:51:48 +00:00
|
|
|
saveEventPost(baseDir, handle, postId, tagDict)
|
2019-10-11 16:20:16 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
|
|
|
def inboxUpdateIndex(boxname: str, baseDir: str, handle: str,
|
|
|
|
destinationFilename: str, debug: bool) -> bool:
|
2019-10-20 10:25:38 +00:00
|
|
|
"""Updates the index of received posts
|
|
|
|
The new entry is added to the top of the file
|
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
indexFilename = baseDir + '/accounts/' + handle + '/' + boxname + '.index'
|
2019-10-20 10:40:09 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Updating index ' + indexFilename)
|
2019-11-18 13:16:21 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
if '/' + boxname + '/' in destinationFilename:
|
|
|
|
destinationFilename = destinationFilename.split('/' + boxname + '/')[1]
|
2019-11-18 13:16:21 +00:00
|
|
|
|
|
|
|
# remove the path
|
|
|
|
if '/' in destinationFilename:
|
2020-04-03 16:27:34 +00:00
|
|
|
destinationFilename = destinationFilename.split('/')[-1]
|
2019-11-18 13:16:21 +00:00
|
|
|
|
2021-09-03 11:30:23 +00:00
|
|
|
written = False
|
2019-10-20 10:45:12 +00:00
|
|
|
if os.path.isfile(indexFilename):
|
2019-10-20 12:43:59 +00:00
|
|
|
try:
|
|
|
|
with open(indexFilename, 'r+') as indexFile:
|
2020-04-03 16:27:34 +00:00
|
|
|
content = indexFile.read()
|
2020-12-29 17:56:42 +00:00
|
|
|
if destinationFilename + '\n' not in content:
|
|
|
|
indexFile.seek(0, 0)
|
|
|
|
indexFile.write(destinationFilename + '\n' + content)
|
2021-09-03 11:30:23 +00:00
|
|
|
written = True
|
2019-10-20 12:43:59 +00:00
|
|
|
return True
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError as e:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: Failed to write entry to index ' + str(e))
|
2019-10-20 10:45:12 +00:00
|
|
|
else:
|
2021-06-21 22:53:04 +00:00
|
|
|
try:
|
2021-06-22 12:27:10 +00:00
|
|
|
with open(indexFilename, 'w+') as indexFile:
|
2021-06-21 22:53:04 +00:00
|
|
|
indexFile.write(destinationFilename + '\n')
|
2021-09-03 11:30:23 +00:00
|
|
|
written = True
|
2021-11-25 21:18:53 +00:00
|
|
|
except OSError as e:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: Failed to write initial entry to index ' + str(e))
|
2019-10-20 10:45:12 +00:00
|
|
|
|
2021-09-03 11:30:23 +00:00
|
|
|
return written
|
2019-10-20 10:25:38 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _updateLastSeen(baseDir: str, handle: str, actor: str) -> None:
|
2020-12-13 11:27:12 +00:00
|
|
|
"""Updates the time when the given handle last saw the given actor
|
2020-12-13 11:28:23 +00:00
|
|
|
This can later be used to indicate if accounts are dormant/abandoned/moved
|
2020-12-13 11:27:12 +00:00
|
|
|
"""
|
|
|
|
if '@' not in handle:
|
|
|
|
return
|
|
|
|
nickname = handle.split('@')[0]
|
|
|
|
domain = handle.split('@')[1]
|
2021-06-23 21:31:50 +00:00
|
|
|
domain = removeDomainPort(domain)
|
2021-07-13 21:59:53 +00:00
|
|
|
accountPath = acctDir(baseDir, nickname, domain)
|
2020-12-13 11:27:12 +00:00
|
|
|
if not os.path.isdir(accountPath):
|
|
|
|
return
|
|
|
|
if not isFollowingActor(baseDir, nickname, domain, actor):
|
|
|
|
return
|
|
|
|
lastSeenPath = accountPath + '/lastseen'
|
|
|
|
if not os.path.isdir(lastSeenPath):
|
|
|
|
os.mkdir(lastSeenPath)
|
|
|
|
lastSeenFilename = lastSeenPath + '/' + actor.replace('/', '#') + '.txt'
|
|
|
|
currTime = datetime.datetime.utcnow()
|
|
|
|
daysSinceEpoch = (currTime - datetime.datetime(1970, 1, 1)).days
|
2020-12-13 14:31:22 +00:00
|
|
|
# has the value changed?
|
|
|
|
if os.path.isfile(lastSeenFilename):
|
2021-06-21 22:52:04 +00:00
|
|
|
with open(lastSeenFilename, 'r') as lastSeenFile:
|
|
|
|
daysSinceEpochFile = lastSeenFile.read()
|
2020-12-13 14:31:22 +00:00
|
|
|
if int(daysSinceEpochFile) == daysSinceEpoch:
|
|
|
|
# value hasn't changed, so we can save writing anything to file
|
|
|
|
return
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(lastSeenFilename, 'w+') as lastSeenFile:
|
|
|
|
lastSeenFile.write(str(daysSinceEpoch))
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + lastSeenFilename)
|
2020-12-13 11:27:12 +00:00
|
|
|
|
|
|
|
|
2021-02-24 11:01:44 +00:00
|
|
|
def _bounceDM(senderPostId: str, session, httpPrefix: str,
|
|
|
|
baseDir: str, nickname: str, domain: str, port: int,
|
|
|
|
sendingHandle: str, federationList: [],
|
|
|
|
sendThreads: [], postLog: [],
|
|
|
|
cachedWebfingers: {}, personCache: {},
|
2021-02-24 11:43:48 +00:00
|
|
|
translate: {}, debug: bool,
|
2021-08-31 14:17:11 +00:00
|
|
|
lastBounceMessage: [], systemLanguage: str,
|
2021-11-08 18:09:24 +00:00
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
contentLicenseUrl: str) -> bool:
|
2021-02-24 11:01:44 +00:00
|
|
|
"""Sends a bounce message back to the sending handle
|
|
|
|
if a DM has been rejected
|
|
|
|
"""
|
|
|
|
print(nickname + '@' + domain +
|
|
|
|
' cannot receive DM from ' + sendingHandle +
|
|
|
|
' because they do not follow them')
|
2021-02-24 11:43:48 +00:00
|
|
|
|
|
|
|
# Don't send out bounce messages too frequently.
|
|
|
|
# Otherwise an adversary could try to DoS your instance
|
|
|
|
# by continuously sending DMs to you
|
|
|
|
currTime = int(time.time())
|
|
|
|
if currTime - lastBounceMessage[0] < 60:
|
|
|
|
return False
|
|
|
|
|
|
|
|
# record the last time that a bounce was generated
|
|
|
|
lastBounceMessage[0] = currTime
|
|
|
|
|
2021-02-24 11:01:44 +00:00
|
|
|
senderNickname = sendingHandle.split('@')[0]
|
2021-07-31 13:19:45 +00:00
|
|
|
groupAccount = False
|
|
|
|
if sendingHandle.startswith('!'):
|
|
|
|
sendingHandle = sendingHandle[1:]
|
|
|
|
groupAccount = True
|
2021-02-24 11:01:44 +00:00
|
|
|
senderDomain = sendingHandle.split('@')[1]
|
|
|
|
senderPort = port
|
|
|
|
if ':' in senderDomain:
|
2021-06-23 21:44:31 +00:00
|
|
|
senderPort = getPortFromDomain(senderDomain)
|
|
|
|
senderDomain = removeDomainPort(senderDomain)
|
2021-02-24 11:01:44 +00:00
|
|
|
cc = []
|
|
|
|
|
|
|
|
# create the bounce DM
|
|
|
|
subject = None
|
|
|
|
content = translate['DM bounce']
|
|
|
|
followersOnly = False
|
|
|
|
saveToFile = False
|
|
|
|
clientToServer = False
|
|
|
|
commentsEnabled = False
|
|
|
|
attachImageFilename = None
|
|
|
|
mediaType = None
|
|
|
|
imageDescription = ''
|
2021-05-09 19:29:53 +00:00
|
|
|
city = 'London, England'
|
2021-02-24 11:01:44 +00:00
|
|
|
inReplyTo = removeIdEnding(senderPostId)
|
|
|
|
inReplyToAtomUri = None
|
|
|
|
schedulePost = False
|
|
|
|
eventDate = None
|
|
|
|
eventTime = None
|
|
|
|
location = None
|
2021-08-08 16:52:32 +00:00
|
|
|
conversationId = None
|
2021-08-13 17:08:50 +00:00
|
|
|
lowBandwidth = False
|
2021-02-24 11:01:44 +00:00
|
|
|
postJsonObject = \
|
|
|
|
createDirectMessagePost(baseDir, nickname, domain, port,
|
|
|
|
httpPrefix, content, followersOnly,
|
|
|
|
saveToFile, clientToServer,
|
|
|
|
commentsEnabled,
|
|
|
|
attachImageFilename, mediaType,
|
2021-05-09 19:11:05 +00:00
|
|
|
imageDescription, city,
|
2021-02-24 11:01:44 +00:00
|
|
|
inReplyTo, inReplyToAtomUri,
|
|
|
|
subject, debug, schedulePost,
|
2021-07-18 09:55:49 +00:00
|
|
|
eventDate, eventTime, location,
|
2021-11-08 18:09:24 +00:00
|
|
|
systemLanguage, conversationId, lowBandwidth,
|
|
|
|
contentLicenseUrl)
|
2021-02-24 11:01:44 +00:00
|
|
|
if not postJsonObject:
|
|
|
|
print('WARN: unable to create bounce message to ' + sendingHandle)
|
2021-02-24 11:43:48 +00:00
|
|
|
return False
|
2021-02-24 11:01:44 +00:00
|
|
|
# bounce DM goes back to the sender
|
|
|
|
print('Sending bounce DM to ' + sendingHandle)
|
|
|
|
sendSignedJson(postJsonObject, session, baseDir,
|
|
|
|
nickname, domain, port,
|
|
|
|
senderNickname, senderDomain, senderPort, cc,
|
|
|
|
httpPrefix, False, False, federationList,
|
|
|
|
sendThreads, postLog, cachedWebfingers,
|
2021-08-31 14:17:11 +00:00
|
|
|
personCache, debug, __version__, None, groupAccount,
|
2021-09-20 16:05:11 +00:00
|
|
|
signingPrivateKeyPem, 7238634)
|
2021-02-24 11:43:48 +00:00
|
|
|
return True
|
2021-02-24 11:01:44 +00:00
|
|
|
|
|
|
|
|
2021-06-07 16:34:08 +00:00
|
|
|
def _isValidDM(baseDir: str, nickname: str, domain: str, port: int,
|
|
|
|
postJsonObject: {}, updateIndexList: [],
|
|
|
|
session, httpPrefix: str,
|
|
|
|
federationList: [],
|
|
|
|
sendThreads: [], postLog: [],
|
|
|
|
cachedWebfingers: {},
|
|
|
|
personCache: {},
|
|
|
|
translate: {}, debug: bool,
|
|
|
|
lastBounceMessage: [],
|
2021-08-31 14:17:11 +00:00
|
|
|
handle: str, systemLanguage: str,
|
2021-11-08 18:09:24 +00:00
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
contentLicenseUrl: str) -> bool:
|
2021-06-07 16:34:08 +00:00
|
|
|
"""Is the given message a valid DM?
|
|
|
|
"""
|
|
|
|
if nickname == 'inbox':
|
|
|
|
# going to the shared inbox
|
|
|
|
return True
|
|
|
|
|
|
|
|
# check for the flag file which indicates to
|
|
|
|
# only receive DMs from people you are following
|
2021-07-13 21:59:53 +00:00
|
|
|
followDMsFilename = acctDir(baseDir, nickname, domain) + '/.followDMs'
|
2021-06-07 16:34:08 +00:00
|
|
|
if not os.path.isfile(followDMsFilename):
|
|
|
|
# dm index will be updated
|
|
|
|
updateIndexList.append('dm')
|
2021-08-14 11:13:39 +00:00
|
|
|
actUrl = localActorUrl(httpPrefix, nickname, domain)
|
|
|
|
_dmNotify(baseDir, handle, actUrl + '/dm')
|
2021-06-07 16:34:08 +00:00
|
|
|
return True
|
2021-06-07 17:49:10 +00:00
|
|
|
|
2021-06-07 16:34:08 +00:00
|
|
|
# get the file containing following handles
|
2021-07-13 21:59:53 +00:00
|
|
|
followingFilename = acctDir(baseDir, nickname, domain) + '/following.txt'
|
2021-06-07 16:34:08 +00:00
|
|
|
# who is sending a DM?
|
|
|
|
if not postJsonObject.get('actor'):
|
|
|
|
return False
|
|
|
|
sendingActor = postJsonObject['actor']
|
|
|
|
sendingActorNickname = \
|
|
|
|
getNicknameFromActor(sendingActor)
|
|
|
|
if not sendingActorNickname:
|
|
|
|
return False
|
|
|
|
sendingActorDomain, sendingActorPort = \
|
|
|
|
getDomainFromActor(sendingActor)
|
|
|
|
if not sendingActorDomain:
|
|
|
|
return False
|
|
|
|
# Is this DM to yourself? eg. a reminder
|
|
|
|
sendingToSelf = False
|
|
|
|
if sendingActorNickname == nickname and \
|
|
|
|
sendingActorDomain == domain:
|
|
|
|
sendingToSelf = True
|
|
|
|
|
|
|
|
# check that the following file exists
|
|
|
|
if not sendingToSelf:
|
|
|
|
if not os.path.isfile(followingFilename):
|
|
|
|
print('No following.txt file exists for ' +
|
|
|
|
nickname + '@' + domain +
|
|
|
|
' so not accepting DM from ' +
|
|
|
|
sendingActorNickname + '@' +
|
|
|
|
sendingActorDomain)
|
|
|
|
return False
|
|
|
|
|
|
|
|
# Not sending to yourself
|
|
|
|
if not sendingToSelf:
|
|
|
|
# get the handle of the DM sender
|
|
|
|
sendH = sendingActorNickname + '@' + sendingActorDomain
|
|
|
|
# check the follow
|
|
|
|
if not isFollowingActor(baseDir, nickname, domain, sendH):
|
|
|
|
# DMs may always be allowed from some domains
|
|
|
|
if not dmAllowedFromDomain(baseDir,
|
|
|
|
nickname, domain,
|
|
|
|
sendingActorDomain):
|
|
|
|
# send back a bounce DM
|
|
|
|
if postJsonObject.get('id') and \
|
|
|
|
postJsonObject.get('object'):
|
|
|
|
# don't send bounces back to
|
|
|
|
# replies to bounce messages
|
|
|
|
obj = postJsonObject['object']
|
|
|
|
if isinstance(obj, dict):
|
|
|
|
if not obj.get('inReplyTo'):
|
2021-09-28 10:28:42 +00:00
|
|
|
bouncedId = removeIdEnding(postJsonObject['id'])
|
|
|
|
_bounceDM(bouncedId,
|
2021-06-07 16:34:08 +00:00
|
|
|
session, httpPrefix,
|
|
|
|
baseDir,
|
|
|
|
nickname, domain,
|
|
|
|
port, sendH,
|
|
|
|
federationList,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
translate, debug,
|
2021-07-18 09:55:49 +00:00
|
|
|
lastBounceMessage,
|
2021-08-31 14:17:11 +00:00
|
|
|
systemLanguage,
|
2021-11-08 18:09:24 +00:00
|
|
|
signingPrivateKeyPem,
|
|
|
|
contentLicenseUrl)
|
2021-06-07 16:34:08 +00:00
|
|
|
return False
|
|
|
|
|
|
|
|
# dm index will be updated
|
|
|
|
updateIndexList.append('dm')
|
2021-08-14 11:13:39 +00:00
|
|
|
actUrl = localActorUrl(httpPrefix, nickname, domain)
|
|
|
|
_dmNotify(baseDir, handle, actUrl + '/dm')
|
2021-06-07 16:34:08 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2021-11-04 11:48:23 +00:00
|
|
|
def _receiveQuestionVote(baseDir: str, nickname: str, domain: str,
|
|
|
|
httpPrefix: str, handle: str, debug: bool,
|
|
|
|
postJsonObject: {}, recentPostsCache: {},
|
|
|
|
session, onionDomain: str, i2pDomain: str, port: int,
|
|
|
|
federationList: [], sendThreads: [], postLog: [],
|
|
|
|
cachedWebfingers: {}, personCache: {},
|
2021-11-05 10:24:18 +00:00
|
|
|
signingPrivateKeyPem: str,
|
|
|
|
maxRecentPosts: int, translate: {},
|
|
|
|
allowDeletion: bool,
|
|
|
|
YTReplacementDomain: str,
|
|
|
|
twitterReplacementDomain: str,
|
|
|
|
peertubeInstances: [],
|
|
|
|
allowLocalNetworkAccess: bool,
|
|
|
|
themeName: str, systemLanguage: str,
|
|
|
|
maxLikeCount: int,
|
|
|
|
CWlists: {}, listsEnabled: bool) -> None:
|
2021-11-04 11:48:23 +00:00
|
|
|
"""Updates the votes on a Question/poll
|
|
|
|
"""
|
|
|
|
# if this is a reply to a question then update the votes
|
|
|
|
questionJson, questionPostFilename = \
|
|
|
|
questionUpdateVotes(baseDir, nickname, domain, postJsonObject)
|
|
|
|
if not questionJson:
|
|
|
|
return
|
|
|
|
if not questionPostFilename:
|
|
|
|
return
|
|
|
|
|
|
|
|
removePostFromCache(questionJson, recentPostsCache)
|
|
|
|
# ensure that the cached post is removed if it exists, so
|
|
|
|
# that it then will be recreated
|
|
|
|
cachedPostFilename = \
|
|
|
|
getCachedPostFilename(baseDir, nickname, domain, questionJson)
|
|
|
|
if cachedPostFilename:
|
|
|
|
if os.path.isfile(cachedPostFilename):
|
|
|
|
try:
|
|
|
|
os.remove(cachedPostFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-11-04 11:48:23 +00:00
|
|
|
print('EX: replytoQuestion unable to delete ' +
|
|
|
|
cachedPostFilename)
|
2021-11-05 10:24:18 +00:00
|
|
|
|
|
|
|
pageNumber = 1
|
|
|
|
showPublishedDateOnly = False
|
|
|
|
showIndividualPostIcons = True
|
|
|
|
manuallyApproveFollowers = \
|
|
|
|
followerApprovalActive(baseDir, nickname, domain)
|
|
|
|
notDM = not isDM(questionJson)
|
|
|
|
individualPostAsHtml(signingPrivateKeyPem, False,
|
|
|
|
recentPostsCache, maxRecentPosts,
|
|
|
|
translate, pageNumber, baseDir,
|
|
|
|
session, cachedWebfingers, personCache,
|
|
|
|
nickname, domain, port, questionJson,
|
|
|
|
None, True, allowDeletion,
|
|
|
|
httpPrefix, __version__,
|
|
|
|
'inbox',
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
showPublishedDateOnly,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, notDM,
|
|
|
|
showIndividualPostIcons,
|
|
|
|
manuallyApproveFollowers,
|
|
|
|
False, True, False, CWlists,
|
|
|
|
listsEnabled)
|
|
|
|
|
2021-11-05 10:26:20 +00:00
|
|
|
# add id to inbox index
|
|
|
|
inboxUpdateIndex('inbox', baseDir, handle,
|
|
|
|
questionPostFilename, debug)
|
|
|
|
|
2021-11-04 11:48:23 +00:00
|
|
|
# Is this a question created by this instance?
|
|
|
|
idPrefix = httpPrefix + '://' + domain
|
|
|
|
if not questionJson['object']['id'].startswith(idPrefix):
|
|
|
|
return
|
|
|
|
# if the votes on a question have changed then
|
|
|
|
# send out an update
|
|
|
|
questionJson['type'] = 'Update'
|
|
|
|
sharedItemsFederatedDomains = []
|
|
|
|
sharedItemFederationTokens = {}
|
|
|
|
sendToFollowersThread(session, baseDir, nickname, domain,
|
|
|
|
onionDomain, i2pDomain, port,
|
|
|
|
httpPrefix, federationList,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers, personCache,
|
|
|
|
postJsonObject, debug, __version__,
|
|
|
|
sharedItemsFederatedDomains,
|
|
|
|
sharedItemFederationTokens,
|
|
|
|
signingPrivateKeyPem)
|
|
|
|
|
|
|
|
|
2021-11-04 12:29:53 +00:00
|
|
|
def _createReplyNotificationFile(baseDir: str, nickname: str, domain: str,
|
|
|
|
handle: str, debug: bool, postIsDM: bool,
|
|
|
|
postJsonObject: {}, actor: str,
|
|
|
|
updateIndexList: [], httpPrefix: str,
|
|
|
|
defaultReplyIntervalHours: int) -> bool:
|
|
|
|
"""Generates a file indicating that a new reply has arrived
|
|
|
|
The file can then be used by other systems to create a notification
|
|
|
|
xmpp, matrix, email, etc
|
|
|
|
"""
|
|
|
|
isReplyToMutedPost = False
|
|
|
|
if postIsDM:
|
|
|
|
return isReplyToMutedPost
|
|
|
|
if not isReply(postJsonObject, actor):
|
|
|
|
return isReplyToMutedPost
|
|
|
|
if nickname == 'inbox':
|
|
|
|
return isReplyToMutedPost
|
|
|
|
# replies index will be updated
|
|
|
|
updateIndexList.append('tlreplies')
|
|
|
|
|
|
|
|
conversationId = None
|
|
|
|
if postJsonObject['object'].get('conversation'):
|
|
|
|
conversationId = postJsonObject['object']['conversation']
|
|
|
|
|
|
|
|
if not postJsonObject['object'].get('inReplyTo'):
|
|
|
|
return isReplyToMutedPost
|
|
|
|
inReplyTo = postJsonObject['object']['inReplyTo']
|
|
|
|
if not inReplyTo:
|
|
|
|
return isReplyToMutedPost
|
|
|
|
if not isinstance(inReplyTo, str):
|
|
|
|
return isReplyToMutedPost
|
|
|
|
if not isMuted(baseDir, nickname, domain, inReplyTo, conversationId):
|
|
|
|
# check if the reply is within the allowed time period
|
|
|
|
# after publication
|
|
|
|
replyIntervalHours = \
|
|
|
|
getReplyIntervalHours(baseDir, nickname, domain,
|
|
|
|
defaultReplyIntervalHours)
|
|
|
|
if canReplyTo(baseDir, nickname, domain, inReplyTo,
|
|
|
|
replyIntervalHours):
|
|
|
|
actUrl = localActorUrl(httpPrefix, nickname, domain)
|
|
|
|
_replyNotify(baseDir, handle, actUrl + '/tlreplies')
|
|
|
|
else:
|
|
|
|
if debug:
|
|
|
|
print('Reply to ' + inReplyTo + ' is outside of the ' +
|
|
|
|
'permitted interval of ' + str(replyIntervalHours) +
|
|
|
|
' hours')
|
|
|
|
return False
|
|
|
|
else:
|
|
|
|
isReplyToMutedPost = True
|
|
|
|
return isReplyToMutedPost
|
|
|
|
|
|
|
|
|
2021-11-04 12:43:46 +00:00
|
|
|
def _lowFrequencyPostNotification(baseDir: str, httpPrefix: str, nickname: str,
|
|
|
|
domain: str, port: int, handle: str,
|
|
|
|
postIsDM: bool, jsonObj: {}) -> None:
|
|
|
|
"""Should we notify that a post from this person has arrived?
|
|
|
|
This is for cases where the notify checkbox is enabled on the
|
|
|
|
person options screen
|
|
|
|
"""
|
|
|
|
if postIsDM:
|
|
|
|
return
|
|
|
|
if not jsonObj:
|
|
|
|
return
|
|
|
|
if not jsonObj.get('attributedTo'):
|
|
|
|
return
|
|
|
|
if not jsonObj.get('id'):
|
|
|
|
return
|
|
|
|
attributedTo = jsonObj['attributedTo']
|
|
|
|
if not isinstance(attributedTo, str):
|
|
|
|
return
|
|
|
|
fromNickname = getNicknameFromActor(attributedTo)
|
|
|
|
fromDomain, fromPort = getDomainFromActor(attributedTo)
|
|
|
|
fromDomainFull = getFullDomain(fromDomain, fromPort)
|
|
|
|
if notifyWhenPersonPosts(baseDir, nickname, domain,
|
|
|
|
fromNickname, fromDomainFull):
|
|
|
|
postId = removeIdEnding(jsonObj['id'])
|
|
|
|
domFull = getFullDomain(domain, port)
|
|
|
|
postLink = \
|
|
|
|
localActorUrl(httpPrefix, nickname, domFull) + \
|
|
|
|
'?notifypost=' + postId.replace('/', '-')
|
|
|
|
_notifyPostArrival(baseDir, handle, postLink)
|
|
|
|
|
|
|
|
|
2021-11-04 13:05:04 +00:00
|
|
|
def _checkForGitPatches(baseDir: str, nickname: str, domain: str,
|
|
|
|
handle: str, jsonObj: {}) -> int:
|
|
|
|
"""check for incoming git patches
|
|
|
|
"""
|
|
|
|
if not jsonObj:
|
|
|
|
return 0
|
|
|
|
if not jsonObj.get('content'):
|
|
|
|
return 0
|
|
|
|
if not jsonObj.get('summary'):
|
|
|
|
return 0
|
|
|
|
if not jsonObj.get('attributedTo'):
|
|
|
|
return 0
|
|
|
|
attributedTo = jsonObj['attributedTo']
|
|
|
|
if not isinstance(attributedTo, str):
|
|
|
|
return 0
|
|
|
|
fromNickname = getNicknameFromActor(attributedTo)
|
|
|
|
fromDomain, fromPort = getDomainFromActor(attributedTo)
|
|
|
|
fromDomainFull = getFullDomain(fromDomain, fromPort)
|
|
|
|
if receiveGitPatch(baseDir, nickname, domain,
|
|
|
|
jsonObj['type'], jsonObj['summary'],
|
|
|
|
jsonObj['content'],
|
|
|
|
fromNickname, fromDomainFull):
|
|
|
|
_gitPatchNotify(baseDir, handle,
|
|
|
|
jsonObj['summary'], jsonObj['content'],
|
|
|
|
fromNickname, fromDomainFull)
|
|
|
|
return 1
|
|
|
|
elif '[PATCH]' in jsonObj['content']:
|
|
|
|
print('WARN: git patch not accepted - ' + jsonObj['summary'])
|
|
|
|
return 2
|
|
|
|
return 0
|
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _inboxAfterInitial(recentPostsCache: {}, maxRecentPosts: int,
|
|
|
|
session, keyId: str, handle: str, messageJson: {},
|
|
|
|
baseDir: str, httpPrefix: str, sendThreads: [],
|
|
|
|
postLog: [], cachedWebfingers: {}, personCache: {},
|
|
|
|
queue: [], domain: str,
|
|
|
|
onionDomain: str, i2pDomain: str,
|
|
|
|
port: int, proxyType: str,
|
|
|
|
federationList: [], debug: bool,
|
|
|
|
queueFilename: str, destinationFilename: str,
|
|
|
|
maxReplies: int, allowDeletion: bool,
|
|
|
|
maxMentions: int, maxEmoji: int, translate: {},
|
2021-09-18 17:08:14 +00:00
|
|
|
unitTest: bool,
|
|
|
|
YTReplacementDomain: str,
|
|
|
|
twitterReplacementDomain: str,
|
2020-12-22 18:06:23 +00:00
|
|
|
showPublishedDateOnly: bool,
|
2020-12-23 23:59:49 +00:00
|
|
|
allowLocalNetworkAccess: bool,
|
2021-02-24 11:43:48 +00:00
|
|
|
peertubeInstances: [],
|
2021-03-09 19:52:10 +00:00
|
|
|
lastBounceMessage: [],
|
2021-08-03 10:04:45 +00:00
|
|
|
themeName: str, systemLanguage: str,
|
2021-09-03 22:04:50 +00:00
|
|
|
maxLikeCount: int,
|
2021-09-08 18:37:04 +00:00
|
|
|
signingPrivateKeyPem: str,
|
2021-10-21 13:08:21 +00:00
|
|
|
defaultReplyIntervalHours: int,
|
2021-11-08 18:09:24 +00:00
|
|
|
CWlists: {}, listsEnabled: str,
|
|
|
|
contentLicenseUrl: str) -> bool:
|
2020-09-27 18:35:35 +00:00
|
|
|
""" Anything which needs to be done after initial checks have passed
|
2019-07-10 09:59:22 +00:00
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
actor = keyId
|
2019-09-29 10:41:21 +00:00
|
|
|
if '#' in actor:
|
2020-04-03 16:27:34 +00:00
|
|
|
actor = keyId.split('#')[0]
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
_updateLastSeen(baseDir, handle, actor)
|
2020-12-13 11:27:12 +00:00
|
|
|
|
2021-07-06 20:38:08 +00:00
|
|
|
postIsDM = False
|
2020-12-22 18:06:23 +00:00
|
|
|
isGroup = _groupHandle(baseDir, handle)
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveLike(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
onionDomain,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
2021-09-03 22:04:50 +00:00
|
|
|
debug, signingPrivateKeyPem,
|
|
|
|
maxRecentPosts, translate,
|
|
|
|
allowDeletion,
|
|
|
|
YTReplacementDomain,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain,
|
2021-09-03 22:04:50 +00:00
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount, CWlists, listsEnabled):
|
2019-07-10 12:40:31 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Like accepted from ' + actor)
|
2019-07-10 12:40:31 +00:00
|
|
|
return False
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveUndoLike(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
2021-09-03 22:10:54 +00:00
|
|
|
debug, signingPrivateKeyPem,
|
|
|
|
maxRecentPosts, translate,
|
|
|
|
allowDeletion,
|
|
|
|
YTReplacementDomain,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain,
|
2021-09-03 22:10:54 +00:00
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount, CWlists, listsEnabled):
|
2019-07-12 09:10:09 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Undo like accepted from ' + actor)
|
2019-07-12 09:10:09 +00:00
|
|
|
return False
|
|
|
|
|
2021-11-10 12:16:03 +00:00
|
|
|
if _receiveReaction(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
onionDomain,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
|
|
|
debug, signingPrivateKeyPem,
|
|
|
|
maxRecentPosts, translate,
|
|
|
|
allowDeletion,
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, CWlists, listsEnabled):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: Reaction accepted from ' + actor)
|
|
|
|
return False
|
|
|
|
|
|
|
|
if _receiveUndoReaction(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
|
|
|
debug, signingPrivateKeyPem,
|
|
|
|
maxRecentPosts, translate,
|
|
|
|
allowDeletion,
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount, CWlists, listsEnabled):
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: Undo reaction accepted from ' + actor)
|
|
|
|
return False
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveBookmark(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
2021-09-03 22:28:50 +00:00
|
|
|
debug, signingPrivateKeyPem,
|
|
|
|
maxRecentPosts, translate,
|
|
|
|
allowDeletion,
|
|
|
|
YTReplacementDomain,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain,
|
2021-09-03 22:28:50 +00:00
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount, CWlists, listsEnabled):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Bookmark accepted from ' + actor)
|
2019-11-17 14:01:49 +00:00
|
|
|
return False
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveUndoBookmark(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
2021-09-03 22:28:50 +00:00
|
|
|
debug, signingPrivateKeyPem,
|
|
|
|
maxRecentPosts, translate,
|
|
|
|
allowDeletion,
|
|
|
|
YTReplacementDomain,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain,
|
2021-09-03 22:28:50 +00:00
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount, CWlists, listsEnabled):
|
2019-11-17 14:01:49 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Undo bookmark accepted from ' + actor)
|
2019-11-17 14:01:49 +00:00
|
|
|
return False
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2021-09-11 14:30:37 +00:00
|
|
|
if isCreateInsideAnnounce(messageJson):
|
|
|
|
messageJson = messageJson['object']
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveAnnounce(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, onionDomain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
|
|
|
debug, translate,
|
2021-01-30 11:47:09 +00:00
|
|
|
YTReplacementDomain,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain,
|
2021-03-09 19:52:10 +00:00
|
|
|
allowLocalNetworkAccess,
|
2021-08-31 14:17:11 +00:00
|
|
|
themeName, systemLanguage,
|
2021-09-04 10:11:46 +00:00
|
|
|
signingPrivateKeyPem,
|
|
|
|
maxRecentPosts,
|
|
|
|
allowDeletion,
|
|
|
|
peertubeInstances,
|
2021-10-21 19:00:25 +00:00
|
|
|
maxLikeCount, CWlists, listsEnabled):
|
2019-07-11 19:31:02 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Announce accepted from ' + actor)
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveUndoAnnounce(recentPostsCache,
|
|
|
|
session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
|
|
|
debug):
|
2019-07-12 09:41:57 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Undo announce accepted from ' + actor)
|
2019-07-12 11:35:03 +00:00
|
|
|
return False
|
2019-07-12 09:41:57 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveDelete(session, handle, isGroup,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
messageJson,
|
|
|
|
federationList,
|
|
|
|
debug, allowDeletion,
|
|
|
|
recentPostsCache):
|
2019-08-12 18:02:29 +00:00
|
|
|
if debug:
|
2020-04-03 16:27:34 +00:00
|
|
|
print('DEBUG: Delete accepted from ' + actor)
|
2019-08-12 18:02:29 +00:00
|
|
|
return False
|
|
|
|
|
2019-07-10 13:32:47 +00:00
|
|
|
if debug:
|
2020-09-27 18:35:35 +00:00
|
|
|
print('DEBUG: initial checks passed')
|
2020-04-03 16:27:34 +00:00
|
|
|
print('copy queue file from ' + queueFilename +
|
|
|
|
' to ' + destinationFilename)
|
2019-08-16 22:04:45 +00:00
|
|
|
|
2019-09-11 17:42:55 +00:00
|
|
|
if os.path.isfile(destinationFilename):
|
|
|
|
return True
|
2019-10-04 09:58:02 +00:00
|
|
|
|
2019-07-18 09:26:47 +00:00
|
|
|
if messageJson.get('postNickname'):
|
2020-04-03 16:27:34 +00:00
|
|
|
postJsonObject = messageJson['post']
|
2019-07-18 09:26:47 +00:00
|
|
|
else:
|
2020-04-03 16:27:34 +00:00
|
|
|
postJsonObject = messageJson
|
2019-10-04 12:22:56 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
nickname = handle.split('@')[0]
|
2021-07-07 09:52:05 +00:00
|
|
|
jsonObj = None
|
2021-07-19 08:46:21 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
2020-12-22 18:06:23 +00:00
|
|
|
if _validPostContent(baseDir, nickname, domain,
|
|
|
|
postJsonObject, maxMentions, maxEmoji,
|
2021-07-18 14:15:16 +00:00
|
|
|
allowLocalNetworkAccess, debug,
|
2021-07-19 08:46:21 +00:00
|
|
|
systemLanguage, httpPrefix,
|
|
|
|
domainFull, personCache):
|
2020-05-02 13:17:02 +00:00
|
|
|
|
2020-08-23 14:45:58 +00:00
|
|
|
if postJsonObject.get('object'):
|
|
|
|
jsonObj = postJsonObject['object']
|
|
|
|
if not isinstance(jsonObj, dict):
|
|
|
|
jsonObj = None
|
|
|
|
else:
|
|
|
|
jsonObj = postJsonObject
|
2021-11-04 13:05:04 +00:00
|
|
|
|
|
|
|
if _checkForGitPatches(baseDir, nickname, domain,
|
|
|
|
handle, jsonObj) == 2:
|
|
|
|
return False
|
2020-05-02 11:08:38 +00:00
|
|
|
|
2020-01-15 10:56:39 +00:00
|
|
|
# replace YouTube links, so they get less tracking data
|
2021-07-18 14:15:16 +00:00
|
|
|
replaceYouTube(postJsonObject, YTReplacementDomain, systemLanguage)
|
2021-09-18 17:08:14 +00:00
|
|
|
# replace twitter link domains, so that you can view twitter posts
|
|
|
|
# without having an account
|
|
|
|
replaceTwitter(postJsonObject, twitterReplacementDomain,
|
|
|
|
systemLanguage)
|
2020-01-15 10:56:39 +00:00
|
|
|
|
2019-10-22 20:07:12 +00:00
|
|
|
# list of indexes to be updated
|
2020-04-03 16:27:34 +00:00
|
|
|
updateIndexList = ['inbox']
|
|
|
|
populateReplies(baseDir, httpPrefix, domain, postJsonObject,
|
|
|
|
maxReplies, debug)
|
2019-11-29 19:22:11 +00:00
|
|
|
|
2021-11-04 11:48:23 +00:00
|
|
|
_receiveQuestionVote(baseDir, nickname, domain,
|
|
|
|
httpPrefix, handle, debug,
|
|
|
|
postJsonObject, recentPostsCache,
|
|
|
|
session, onionDomain, i2pDomain, port,
|
|
|
|
federationList, sendThreads, postLog,
|
|
|
|
cachedWebfingers, personCache,
|
2021-11-05 10:24:18 +00:00
|
|
|
signingPrivateKeyPem,
|
|
|
|
maxRecentPosts, translate,
|
|
|
|
allowDeletion,
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
|
|
|
peertubeInstances,
|
|
|
|
allowLocalNetworkAccess,
|
|
|
|
themeName, systemLanguage,
|
|
|
|
maxLikeCount,
|
|
|
|
CWlists, listsEnabled)
|
2019-11-29 19:22:11 +00:00
|
|
|
|
2020-08-27 17:40:09 +00:00
|
|
|
isReplyToMutedPost = False
|
|
|
|
|
2019-10-04 12:22:56 +00:00
|
|
|
if not isGroup:
|
|
|
|
# create a DM notification file if needed
|
2020-05-17 18:21:38 +00:00
|
|
|
postIsDM = isDM(postJsonObject)
|
|
|
|
if postIsDM:
|
2021-06-07 16:34:08 +00:00
|
|
|
if not _isValidDM(baseDir, nickname, domain, port,
|
|
|
|
postJsonObject, updateIndexList,
|
|
|
|
session, httpPrefix,
|
|
|
|
federationList,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
translate, debug,
|
|
|
|
lastBounceMessage,
|
2021-08-31 14:17:11 +00:00
|
|
|
handle, systemLanguage,
|
2021-11-08 18:09:24 +00:00
|
|
|
signingPrivateKeyPem,
|
|
|
|
contentLicenseUrl):
|
2021-06-07 16:34:08 +00:00
|
|
|
return False
|
2019-10-04 12:22:56 +00:00
|
|
|
|
|
|
|
# get the actor being replied to
|
2021-08-14 11:13:39 +00:00
|
|
|
actor = localActorUrl(httpPrefix, nickname, domainFull)
|
2019-10-04 12:22:56 +00:00
|
|
|
|
|
|
|
# create a reply notification file if needed
|
2021-11-04 12:29:53 +00:00
|
|
|
isReplyToMutedPost = \
|
|
|
|
_createReplyNotificationFile(baseDir, nickname, domain,
|
|
|
|
handle, debug, postIsDM,
|
|
|
|
postJsonObject, actor,
|
|
|
|
updateIndexList, httpPrefix,
|
|
|
|
defaultReplyIntervalHours)
|
2019-10-04 10:00:57 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
if isImageMedia(session, baseDir, httpPrefix,
|
2020-06-12 11:50:49 +00:00
|
|
|
nickname, domain, postJsonObject,
|
2021-09-18 17:08:14 +00:00
|
|
|
translate,
|
|
|
|
YTReplacementDomain,
|
|
|
|
twitterReplacementDomain,
|
2021-03-05 19:23:33 +00:00
|
|
|
allowLocalNetworkAccess,
|
2021-07-19 08:46:21 +00:00
|
|
|
recentPostsCache, debug, systemLanguage,
|
2021-08-31 14:17:11 +00:00
|
|
|
domainFull, personCache, signingPrivateKeyPem):
|
2019-10-22 20:30:43 +00:00
|
|
|
# media index will be updated
|
|
|
|
updateIndexList.append('tlmedia')
|
2020-02-24 14:39:25 +00:00
|
|
|
if isBlogPost(postJsonObject):
|
|
|
|
# blogs index will be updated
|
|
|
|
updateIndexList.append('tlblogs')
|
2019-10-22 20:30:43 +00:00
|
|
|
|
2019-10-04 10:00:57 +00:00
|
|
|
# get the avatar for a reply/announce
|
2020-12-22 18:06:23 +00:00
|
|
|
_obtainAvatarForReplyPost(session, baseDir,
|
|
|
|
httpPrefix, domain, onionDomain,
|
2021-08-31 14:17:11 +00:00
|
|
|
personCache, postJsonObject, debug,
|
|
|
|
signingPrivateKeyPem)
|
2019-10-04 10:00:57 +00:00
|
|
|
|
|
|
|
# save the post to file
|
2020-04-03 16:27:34 +00:00
|
|
|
if saveJson(postJsonObject, destinationFilename):
|
2021-11-04 12:43:46 +00:00
|
|
|
_lowFrequencyPostNotification(baseDir, httpPrefix,
|
|
|
|
nickname, domain, port,
|
|
|
|
handle, postIsDM, jsonObj)
|
2021-07-06 20:38:08 +00:00
|
|
|
|
2020-08-27 17:40:09 +00:00
|
|
|
# If this is a reply to a muted post then also mute it.
|
|
|
|
# This enables you to ignore a threat that's getting boring
|
|
|
|
if isReplyToMutedPost:
|
|
|
|
print('MUTE REPLY: ' + destinationFilename)
|
2021-11-25 21:18:53 +00:00
|
|
|
destinationFilenameMuted = destinationFilename + '.muted'
|
|
|
|
try:
|
|
|
|
with open(destinationFilenameMuted, 'w+') as muteFile:
|
|
|
|
muteFile.write('\n')
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to write ' + destinationFilenameMuted)
|
2020-08-27 17:40:09 +00:00
|
|
|
|
2019-10-22 20:07:12 +00:00
|
|
|
# update the indexes for different timelines
|
|
|
|
for boxname in updateIndexList:
|
2020-04-03 16:27:34 +00:00
|
|
|
if not inboxUpdateIndex(boxname, baseDir, handle,
|
|
|
|
destinationFilename, debug):
|
|
|
|
print('ERROR: unable to update ' + boxname + ' index')
|
2020-08-26 11:10:21 +00:00
|
|
|
else:
|
2021-03-01 15:24:12 +00:00
|
|
|
if boxname == 'inbox':
|
2021-10-29 22:40:09 +00:00
|
|
|
if isRecentPost(postJsonObject, 3):
|
2021-03-09 13:52:02 +00:00
|
|
|
domainFull = getFullDomain(domain, port)
|
|
|
|
updateSpeaker(baseDir, httpPrefix,
|
|
|
|
nickname, domain, domainFull,
|
2021-03-03 20:16:53 +00:00
|
|
|
postJsonObject, personCache,
|
2021-03-09 19:52:10 +00:00
|
|
|
translate, None, themeName)
|
2020-08-26 11:10:21 +00:00
|
|
|
if not unitTest:
|
2020-08-26 11:19:32 +00:00
|
|
|
if debug:
|
|
|
|
print('Saving inbox post as html to cache')
|
|
|
|
|
2020-08-26 11:10:21 +00:00
|
|
|
htmlCacheStartTime = time.time()
|
2020-12-22 21:24:46 +00:00
|
|
|
handleName = handle.split('@')[0]
|
2020-12-22 18:06:23 +00:00
|
|
|
_inboxStorePostToHtmlCache(recentPostsCache,
|
|
|
|
maxRecentPosts,
|
|
|
|
translate, baseDir,
|
|
|
|
httpPrefix,
|
|
|
|
session, cachedWebfingers,
|
|
|
|
personCache,
|
2020-12-22 21:24:46 +00:00
|
|
|
handleName,
|
2020-12-22 18:06:23 +00:00
|
|
|
domain, port,
|
|
|
|
postJsonObject,
|
|
|
|
allowDeletion,
|
|
|
|
boxname,
|
2020-12-23 23:59:49 +00:00
|
|
|
showPublishedDateOnly,
|
2021-01-30 11:47:09 +00:00
|
|
|
peertubeInstances,
|
2021-03-09 19:52:10 +00:00
|
|
|
allowLocalNetworkAccess,
|
2021-08-03 10:04:45 +00:00
|
|
|
themeName, systemLanguage,
|
2021-08-31 14:17:11 +00:00
|
|
|
maxLikeCount,
|
2021-10-21 13:08:21 +00:00
|
|
|
signingPrivateKeyPem,
|
2021-10-21 19:00:25 +00:00
|
|
|
CWlists, listsEnabled)
|
2020-08-26 11:19:32 +00:00
|
|
|
if debug:
|
|
|
|
timeDiff = \
|
|
|
|
str(int((time.time() - htmlCacheStartTime) *
|
|
|
|
1000))
|
|
|
|
print('Saved ' + boxname +
|
|
|
|
' post as html to cache in ' +
|
|
|
|
timeDiff + ' mS')
|
2019-10-20 10:25:38 +00:00
|
|
|
|
2021-08-12 10:22:04 +00:00
|
|
|
handleName = handle.split('@')[0]
|
2021-10-14 15:12:35 +00:00
|
|
|
|
|
|
|
# is this an edit of a previous post?
|
|
|
|
# in Mastodon "delete and redraft"
|
|
|
|
# NOTE: this must be done before updateConversation is called
|
|
|
|
editedFilename = \
|
|
|
|
editedPostFilename(baseDir, handleName, domain,
|
|
|
|
postJsonObject, debug, 300)
|
|
|
|
|
2021-08-12 10:22:04 +00:00
|
|
|
updateConversation(baseDir, handleName, domain, postJsonObject)
|
|
|
|
|
2021-10-14 15:12:35 +00:00
|
|
|
# If this was an edit then delete the previous version of the post
|
|
|
|
if editedFilename:
|
|
|
|
deletePost(baseDir, httpPrefix,
|
|
|
|
nickname, domain, editedFilename,
|
|
|
|
debug, recentPostsCache)
|
|
|
|
|
2021-10-18 15:20:22 +00:00
|
|
|
# store the id of the last post made by this actor
|
|
|
|
_storeLastPostId(baseDir, nickname, domain, postJsonObject)
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
_inboxUpdateCalendar(baseDir, handle, postJsonObject)
|
2019-10-19 18:08:47 +00:00
|
|
|
|
2021-10-20 13:33:34 +00:00
|
|
|
storeHashTags(baseDir, handleName, domain,
|
|
|
|
httpPrefix, domainFull,
|
|
|
|
postJsonObject, translate)
|
2019-12-12 17:34:31 +00:00
|
|
|
|
2019-10-19 13:00:46 +00:00
|
|
|
# send the post out to group members
|
|
|
|
if isGroup:
|
2020-12-22 18:06:23 +00:00
|
|
|
_sendToGroupMembers(session, baseDir, handle, port,
|
|
|
|
postJsonObject,
|
|
|
|
httpPrefix, federationList, sendThreads,
|
|
|
|
postLog, cachedWebfingers, personCache,
|
2021-08-02 18:55:53 +00:00
|
|
|
debug, systemLanguage,
|
2021-08-31 14:17:11 +00:00
|
|
|
onionDomain, i2pDomain,
|
|
|
|
signingPrivateKeyPem)
|
2019-10-04 12:22:56 +00:00
|
|
|
|
2019-10-04 10:00:57 +00:00
|
|
|
# if the post wasn't saved
|
2019-08-17 12:26:09 +00:00
|
|
|
if not os.path.isfile(destinationFilename):
|
|
|
|
return False
|
|
|
|
|
2019-07-10 09:59:22 +00:00
|
|
|
return True
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2020-05-22 11:48:13 +00:00
|
|
|
def clearQueueItems(baseDir: str, queue: []) -> None:
|
2020-05-22 12:57:15 +00:00
|
|
|
"""Clears the queue for each account
|
2020-05-22 11:48:13 +00:00
|
|
|
"""
|
|
|
|
ctr = 0
|
|
|
|
queue.clear()
|
|
|
|
for subdir, dirs, files in os.walk(baseDir + '/accounts'):
|
|
|
|
for account in dirs:
|
|
|
|
queueDir = baseDir + '/accounts/' + account + '/queue'
|
2020-06-02 09:05:55 +00:00
|
|
|
if not os.path.isdir(queueDir):
|
|
|
|
continue
|
|
|
|
for queuesubdir, queuedirs, queuefiles in os.walk(queueDir):
|
|
|
|
for qfile in queuefiles:
|
|
|
|
try:
|
|
|
|
os.remove(os.path.join(queueDir, qfile))
|
|
|
|
ctr += 1
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: clearQueueItems unable to delete ' + qfile)
|
2021-06-05 21:09:11 +00:00
|
|
|
break
|
2020-12-13 22:13:45 +00:00
|
|
|
break
|
2020-05-22 11:48:13 +00:00
|
|
|
if ctr > 0:
|
|
|
|
print('Removed ' + str(ctr) + ' inbox queue items')
|
|
|
|
|
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
def _restoreQueueItems(baseDir: str, queue: []) -> None:
|
2019-07-12 21:09:23 +00:00
|
|
|
"""Checks the queue for each account and appends filenames
|
|
|
|
"""
|
2019-08-15 16:45:07 +00:00
|
|
|
queue.clear()
|
2020-04-03 16:27:34 +00:00
|
|
|
for subdir, dirs, files in os.walk(baseDir + '/accounts'):
|
2019-07-12 21:09:23 +00:00
|
|
|
for account in dirs:
|
2020-04-03 16:27:34 +00:00
|
|
|
queueDir = baseDir + '/accounts/' + account + '/queue'
|
2020-06-02 09:05:55 +00:00
|
|
|
if not os.path.isdir(queueDir):
|
|
|
|
continue
|
|
|
|
for queuesubdir, queuedirs, queuefiles in os.walk(queueDir):
|
|
|
|
for qfile in queuefiles:
|
|
|
|
queue.append(os.path.join(queueDir, qfile))
|
2021-06-05 21:09:11 +00:00
|
|
|
break
|
2020-12-13 22:13:45 +00:00
|
|
|
break
|
2020-04-03 16:27:34 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
print('Restored ' + str(len(queue)) + ' inbox queue items')
|
2019-09-02 21:52:43 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
|
|
|
def runInboxQueueWatchdog(projectVersion: str, httpd) -> None:
|
2019-09-02 21:52:43 +00:00
|
|
|
"""This tries to keep the inbox thread running even if it dies
|
|
|
|
"""
|
|
|
|
print('Starting inbox queue watchdog')
|
2020-04-03 16:27:34 +00:00
|
|
|
inboxQueueOriginal = httpd.thrInboxQueue.clone(runInboxQueue)
|
2019-09-02 21:52:43 +00:00
|
|
|
httpd.thrInboxQueue.start()
|
|
|
|
while True:
|
2020-03-22 21:16:02 +00:00
|
|
|
time.sleep(20)
|
2020-12-18 15:29:12 +00:00
|
|
|
if not httpd.thrInboxQueue.is_alive() or httpd.restartInboxQueue:
|
2020-05-02 10:19:24 +00:00
|
|
|
httpd.restartInboxQueueInProgress = True
|
2019-09-02 21:52:43 +00:00
|
|
|
httpd.thrInboxQueue.kill()
|
2020-04-03 16:27:34 +00:00
|
|
|
httpd.thrInboxQueue = inboxQueueOriginal.clone(runInboxQueue)
|
2020-04-27 09:41:38 +00:00
|
|
|
httpd.inboxQueue.clear()
|
2019-09-02 21:52:43 +00:00
|
|
|
httpd.thrInboxQueue.start()
|
|
|
|
print('Restarting inbox queue...')
|
2020-05-02 10:19:24 +00:00
|
|
|
httpd.restartInboxQueueInProgress = False
|
|
|
|
httpd.restartInboxQueue = False
|
2019-09-02 21:52:43 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
|
2021-06-07 13:48:29 +00:00
|
|
|
def _inboxQuotaExceeded(queue: {}, queueFilename: str,
|
|
|
|
queueJson: {}, quotasDaily: {}, quotasPerMin: {},
|
|
|
|
domainMaxPostsPerDay: int,
|
|
|
|
accountMaxPostsPerDay: int,
|
|
|
|
debug: bool) -> bool:
|
|
|
|
"""limit the number of posts which can arrive per domain per day
|
|
|
|
"""
|
|
|
|
postDomain = queueJson['postDomain']
|
|
|
|
if not postDomain:
|
|
|
|
return False
|
|
|
|
|
|
|
|
if domainMaxPostsPerDay > 0:
|
|
|
|
if quotasDaily['domains'].get(postDomain):
|
|
|
|
if quotasDaily['domains'][postDomain] > \
|
|
|
|
domainMaxPostsPerDay:
|
|
|
|
print('Queue: Quota per day - Maximum posts for ' +
|
|
|
|
postDomain + ' reached (' +
|
|
|
|
str(domainMaxPostsPerDay) + ')')
|
|
|
|
if len(queue) > 0:
|
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: _inboxQuotaExceeded unable to delete ' +
|
|
|
|
str(queueFilename))
|
2021-06-07 13:48:29 +00:00
|
|
|
queue.pop(0)
|
|
|
|
return True
|
|
|
|
quotasDaily['domains'][postDomain] += 1
|
|
|
|
else:
|
|
|
|
quotasDaily['domains'][postDomain] = 1
|
|
|
|
|
|
|
|
if quotasPerMin['domains'].get(postDomain):
|
|
|
|
domainMaxPostsPerMin = \
|
|
|
|
int(domainMaxPostsPerDay / (24 * 60))
|
|
|
|
if domainMaxPostsPerMin < 5:
|
|
|
|
domainMaxPostsPerMin = 5
|
|
|
|
if quotasPerMin['domains'][postDomain] > \
|
|
|
|
domainMaxPostsPerMin:
|
|
|
|
print('Queue: Quota per min - Maximum posts for ' +
|
|
|
|
postDomain + ' reached (' +
|
|
|
|
str(domainMaxPostsPerMin) + ')')
|
|
|
|
if len(queue) > 0:
|
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: _inboxQuotaExceeded unable to delete ' +
|
|
|
|
str(queueFilename))
|
2021-06-07 13:48:29 +00:00
|
|
|
queue.pop(0)
|
|
|
|
return True
|
|
|
|
quotasPerMin['domains'][postDomain] += 1
|
|
|
|
else:
|
|
|
|
quotasPerMin['domains'][postDomain] = 1
|
|
|
|
|
|
|
|
if accountMaxPostsPerDay > 0:
|
|
|
|
postHandle = queueJson['postNickname'] + '@' + postDomain
|
|
|
|
if quotasDaily['accounts'].get(postHandle):
|
|
|
|
if quotasDaily['accounts'][postHandle] > \
|
|
|
|
accountMaxPostsPerDay:
|
|
|
|
print('Queue: Quota account posts per day -' +
|
|
|
|
' Maximum posts for ' +
|
|
|
|
postHandle + ' reached (' +
|
|
|
|
str(accountMaxPostsPerDay) + ')')
|
|
|
|
if len(queue) > 0:
|
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: _inboxQuotaExceeded unable to delete ' +
|
|
|
|
str(queueFilename))
|
2021-06-07 13:48:29 +00:00
|
|
|
queue.pop(0)
|
|
|
|
return True
|
|
|
|
quotasDaily['accounts'][postHandle] += 1
|
|
|
|
else:
|
|
|
|
quotasDaily['accounts'][postHandle] = 1
|
|
|
|
|
|
|
|
if quotasPerMin['accounts'].get(postHandle):
|
|
|
|
accountMaxPostsPerMin = \
|
|
|
|
int(accountMaxPostsPerDay / (24 * 60))
|
|
|
|
if accountMaxPostsPerMin < 5:
|
|
|
|
accountMaxPostsPerMin = 5
|
|
|
|
if quotasPerMin['accounts'][postHandle] > \
|
|
|
|
accountMaxPostsPerMin:
|
|
|
|
print('Queue: Quota account posts per min -' +
|
|
|
|
' Maximum posts for ' +
|
|
|
|
postHandle + ' reached (' +
|
|
|
|
str(accountMaxPostsPerMin) + ')')
|
|
|
|
if len(queue) > 0:
|
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: _inboxQuotaExceeded unable to delete ' +
|
|
|
|
str(queueFilename))
|
2021-06-07 13:48:29 +00:00
|
|
|
queue.pop(0)
|
|
|
|
return True
|
|
|
|
quotasPerMin['accounts'][postHandle] += 1
|
|
|
|
else:
|
|
|
|
quotasPerMin['accounts'][postHandle] = 1
|
|
|
|
|
|
|
|
if debug:
|
|
|
|
if accountMaxPostsPerDay > 0 or domainMaxPostsPerDay > 0:
|
|
|
|
pprint(quotasDaily)
|
|
|
|
return False
|
|
|
|
|
|
|
|
|
2021-06-07 14:07:15 +00:00
|
|
|
def _checkJsonSignature(baseDir: str, queueJson: {}) -> (bool, bool):
|
|
|
|
"""check if a json signature exists on this post
|
|
|
|
"""
|
|
|
|
hasJsonSignature = False
|
|
|
|
jwebsigType = None
|
|
|
|
originalJson = queueJson['original']
|
|
|
|
if not originalJson.get('@context') or \
|
|
|
|
not originalJson.get('signature'):
|
|
|
|
return hasJsonSignature, jwebsigType
|
|
|
|
if not isinstance(originalJson['signature'], dict):
|
|
|
|
return hasJsonSignature, jwebsigType
|
|
|
|
# see https://tools.ietf.org/html/rfc7515
|
|
|
|
jwebsig = originalJson['signature']
|
|
|
|
# signature exists and is of the expected type
|
|
|
|
if not jwebsig.get('type') or \
|
|
|
|
not jwebsig.get('signatureValue'):
|
|
|
|
return hasJsonSignature, jwebsigType
|
|
|
|
jwebsigType = jwebsig['type']
|
|
|
|
if jwebsigType == 'RsaSignature2017':
|
|
|
|
if hasValidContext(originalJson):
|
|
|
|
hasJsonSignature = True
|
|
|
|
else:
|
|
|
|
unknownContextsFile = \
|
|
|
|
baseDir + '/accounts/unknownContexts.txt'
|
|
|
|
unknownContext = str(originalJson['@context'])
|
|
|
|
|
2021-10-27 22:05:44 +00:00
|
|
|
print('unrecognized @context: ' + unknownContext)
|
2021-06-07 14:07:15 +00:00
|
|
|
|
|
|
|
alreadyUnknown = False
|
|
|
|
if os.path.isfile(unknownContextsFile):
|
|
|
|
if unknownContext in \
|
|
|
|
open(unknownContextsFile).read():
|
|
|
|
alreadyUnknown = True
|
|
|
|
|
|
|
|
if not alreadyUnknown:
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(unknownContextsFile, 'a+') as unknownFile:
|
|
|
|
unknownFile.write(unknownContext + '\n')
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to append ' + unknownContextsFile)
|
2021-06-07 14:07:15 +00:00
|
|
|
else:
|
2021-10-27 22:05:44 +00:00
|
|
|
print('Unrecognized jsonld signature type: ' + jwebsigType)
|
2021-06-07 14:07:15 +00:00
|
|
|
|
|
|
|
unknownSignaturesFile = \
|
|
|
|
baseDir + '/accounts/unknownJsonSignatures.txt'
|
|
|
|
|
|
|
|
alreadyUnknown = False
|
|
|
|
if os.path.isfile(unknownSignaturesFile):
|
|
|
|
if jwebsigType in \
|
|
|
|
open(unknownSignaturesFile).read():
|
|
|
|
alreadyUnknown = True
|
|
|
|
|
|
|
|
if not alreadyUnknown:
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
|
|
|
with open(unknownSignaturesFile, 'a+') as unknownFile:
|
|
|
|
unknownFile.write(jwebsigType + '\n')
|
|
|
|
except OSError:
|
2021-11-25 22:22:54 +00:00
|
|
|
print('EX: unable to append ' + unknownSignaturesFile)
|
2021-06-07 14:07:15 +00:00
|
|
|
return hasJsonSignature, jwebsigType
|
|
|
|
|
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
def runInboxQueue(recentPostsCache: {}, maxRecentPosts: int,
|
|
|
|
projectVersion: str,
|
|
|
|
baseDir: str, httpPrefix: str, sendThreads: [], postLog: [],
|
|
|
|
cachedWebfingers: {}, personCache: {}, queue: [],
|
2020-06-03 20:21:44 +00:00
|
|
|
domain: str,
|
2020-06-09 11:03:59 +00:00
|
|
|
onionDomain: str, i2pDomain: str, port: int, proxyType: str,
|
2020-09-27 19:27:24 +00:00
|
|
|
federationList: [], maxReplies: int,
|
2020-04-03 16:27:34 +00:00
|
|
|
domainMaxPostsPerDay: int, accountMaxPostsPerDay: int,
|
|
|
|
allowDeletion: bool, debug: bool, maxMentions: int,
|
|
|
|
maxEmoji: int, translate: {}, unitTest: bool,
|
2020-10-11 18:50:13 +00:00
|
|
|
YTReplacementDomain: str,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain: str,
|
2020-10-23 19:18:13 +00:00
|
|
|
showPublishedDateOnly: bool,
|
2020-12-23 23:59:49 +00:00
|
|
|
maxFollowers: int, allowLocalNetworkAccess: bool,
|
2021-01-05 10:29:37 +00:00
|
|
|
peertubeInstances: [],
|
2021-03-09 19:52:10 +00:00
|
|
|
verifyAllSignatures: bool,
|
2021-08-03 10:04:45 +00:00
|
|
|
themeName: str, systemLanguage: str,
|
2021-09-08 18:37:04 +00:00
|
|
|
maxLikeCount: int, signingPrivateKeyPem: str,
|
2021-10-21 13:08:21 +00:00
|
|
|
defaultReplyIntervalHours: int,
|
|
|
|
CWlists: {}) -> None:
|
2020-08-02 09:51:20 +00:00
|
|
|
"""Processes received items and moves them to the appropriate
|
|
|
|
directories
|
2019-07-04 12:23:53 +00:00
|
|
|
"""
|
2020-04-03 16:27:34 +00:00
|
|
|
currSessionTime = int(time.time())
|
|
|
|
sessionLastUpdate = currSessionTime
|
2020-06-24 09:04:58 +00:00
|
|
|
print('Starting new session when starting inbox queue')
|
2020-06-09 11:03:59 +00:00
|
|
|
session = createSession(proxyType)
|
2020-04-03 16:27:34 +00:00
|
|
|
inboxHandle = 'inbox@' + domain
|
2019-07-04 12:23:53 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: Inbox queue running')
|
|
|
|
|
2019-07-12 21:09:23 +00:00
|
|
|
# if queue processing was interrupted (eg server crash)
|
|
|
|
# then this loads any outstanding items back into the queue
|
2020-12-22 18:06:23 +00:00
|
|
|
_restoreQueueItems(baseDir, queue)
|
2019-07-15 10:22:19 +00:00
|
|
|
|
2020-03-25 10:21:25 +00:00
|
|
|
# keep track of numbers of incoming posts per day
|
2020-04-03 16:27:34 +00:00
|
|
|
quotasLastUpdateDaily = int(time.time())
|
|
|
|
quotasDaily = {
|
2019-07-15 10:22:19 +00:00
|
|
|
'domains': {},
|
|
|
|
'accounts': {}
|
|
|
|
}
|
2020-04-03 16:27:34 +00:00
|
|
|
quotasLastUpdatePerMin = int(time.time())
|
|
|
|
quotasPerMin = {
|
2020-03-25 10:36:37 +00:00
|
|
|
'domains': {},
|
|
|
|
'accounts': {}
|
|
|
|
}
|
2019-08-05 22:38:38 +00:00
|
|
|
|
2020-04-03 16:27:34 +00:00
|
|
|
heartBeatCtr = 0
|
|
|
|
queueRestoreCtr = 0
|
2019-09-03 08:46:26 +00:00
|
|
|
|
2021-02-24 11:43:48 +00:00
|
|
|
# time when the last DM bounce message was sent
|
|
|
|
# This is in a list so that it can be changed by reference
|
|
|
|
# within _bounceDM
|
|
|
|
lastBounceMessage = [int(time.time())]
|
|
|
|
|
2021-06-05 13:36:03 +00:00
|
|
|
# how long it takes for broch mode to lapse
|
|
|
|
brochLapseDays = random.randrange(7, 14)
|
|
|
|
|
2019-07-04 12:23:53 +00:00
|
|
|
while True:
|
2020-04-16 18:25:59 +00:00
|
|
|
time.sleep(1)
|
2019-09-03 08:46:26 +00:00
|
|
|
|
|
|
|
# heartbeat to monitor whether the inbox queue is running
|
2021-03-14 21:37:33 +00:00
|
|
|
heartBeatCtr += 1
|
2020-04-03 16:27:34 +00:00
|
|
|
if heartBeatCtr >= 10:
|
2021-02-15 22:26:25 +00:00
|
|
|
# turn off broch mode after it has timed out
|
2021-06-05 13:36:03 +00:00
|
|
|
if brochModeLapses(baseDir, brochLapseDays):
|
|
|
|
brochLapseDays = random.randrange(7, 14)
|
2020-04-16 10:14:05 +00:00
|
|
|
print('>>> Heartbeat Q:' + str(len(queue)) + ' ' +
|
2020-04-03 16:27:34 +00:00
|
|
|
'{:%F %T}'.format(datetime.datetime.now()))
|
|
|
|
heartBeatCtr = 0
|
|
|
|
|
|
|
|
if len(queue) == 0:
|
2019-09-03 09:11:33 +00:00
|
|
|
# restore any remaining queue items
|
2020-04-03 16:27:34 +00:00
|
|
|
queueRestoreCtr += 1
|
|
|
|
if queueRestoreCtr >= 30:
|
|
|
|
queueRestoreCtr = 0
|
2020-12-22 18:06:23 +00:00
|
|
|
_restoreQueueItems(baseDir, queue)
|
2020-04-16 09:49:57 +00:00
|
|
|
continue
|
2020-04-16 10:14:05 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
currTime = int(time.time())
|
|
|
|
|
|
|
|
# recreate the session periodically
|
2020-06-24 09:40:17 +00:00
|
|
|
if not session or currTime - sessionLastUpdate > 21600:
|
|
|
|
print('Regenerating inbox queue session at 6hr interval')
|
2020-06-09 11:03:59 +00:00
|
|
|
session = createSession(proxyType)
|
2020-06-08 20:18:02 +00:00
|
|
|
if not session:
|
|
|
|
continue
|
2020-04-16 09:49:57 +00:00
|
|
|
sessionLastUpdate = currTime
|
|
|
|
|
|
|
|
# oldest item first
|
|
|
|
queue.sort()
|
|
|
|
queueFilename = queue[0]
|
|
|
|
if not os.path.isfile(queueFilename):
|
|
|
|
print("Queue: queue item rejected because it has no file: " +
|
|
|
|
queueFilename)
|
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
2019-07-04 12:23:53 +00:00
|
|
|
|
2021-03-14 19:22:58 +00:00
|
|
|
if debug:
|
|
|
|
print('Loading queue item ' + queueFilename)
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
# Load the queue json
|
|
|
|
queueJson = loadJson(queueFilename, 1)
|
|
|
|
if not queueJson:
|
|
|
|
print('Queue: runInboxQueue failed to load inbox queue item ' +
|
|
|
|
queueFilename)
|
|
|
|
# Assume that the file is probably corrupt/unreadable
|
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
# delete the queue file
|
|
|
|
if os.path.isfile(queueFilename):
|
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 1 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
continue
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
# clear the daily quotas for maximum numbers of received posts
|
2021-01-05 10:48:22 +00:00
|
|
|
if currTime - quotasLastUpdateDaily > 60 * 60 * 24:
|
2020-04-16 09:49:57 +00:00
|
|
|
quotasDaily = {
|
|
|
|
'domains': {},
|
|
|
|
'accounts': {}
|
|
|
|
}
|
|
|
|
quotasLastUpdateDaily = currTime
|
|
|
|
|
2021-01-05 10:48:22 +00:00
|
|
|
if currTime - quotasLastUpdatePerMin > 60:
|
|
|
|
# clear the per minute quotas for maximum numbers of received posts
|
2020-04-16 09:49:57 +00:00
|
|
|
quotasPerMin = {
|
|
|
|
'domains': {},
|
|
|
|
'accounts': {}
|
|
|
|
}
|
2021-01-05 10:48:22 +00:00
|
|
|
# also check if the json signature enforcement has changed
|
|
|
|
verifyAllSigs = getConfigParam(baseDir, "verifyAllSignatures")
|
|
|
|
if verifyAllSigs is not None:
|
|
|
|
verifyAllSignatures = verifyAllSigs
|
|
|
|
# change the last time that this was done
|
2020-04-16 09:49:57 +00:00
|
|
|
quotasLastUpdatePerMin = currTime
|
|
|
|
|
2021-06-07 13:48:29 +00:00
|
|
|
if _inboxQuotaExceeded(queue, queueFilename,
|
|
|
|
queueJson, quotasDaily, quotasPerMin,
|
|
|
|
domainMaxPostsPerDay,
|
|
|
|
accountMaxPostsPerDay, debug):
|
|
|
|
continue
|
2019-07-15 10:22:19 +00:00
|
|
|
|
2021-03-14 19:22:58 +00:00
|
|
|
if debug and queueJson.get('actor'):
|
2020-08-23 14:45:58 +00:00
|
|
|
print('Obtaining public key for actor ' + queueJson['actor'])
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
# Try a few times to obtain the public key
|
|
|
|
pubKey = None
|
|
|
|
keyId = None
|
|
|
|
for tries in range(8):
|
2020-04-03 16:27:34 +00:00
|
|
|
keyId = None
|
2020-04-16 09:49:57 +00:00
|
|
|
signatureParams = \
|
|
|
|
queueJson['httpHeaders']['signature'].split(',')
|
|
|
|
for signatureItem in signatureParams:
|
|
|
|
if signatureItem.startswith('keyId='):
|
|
|
|
if '"' in signatureItem:
|
|
|
|
keyId = signatureItem.split('"')[1]
|
|
|
|
break
|
|
|
|
if not keyId:
|
|
|
|
print('Queue: No keyId in signature: ' +
|
|
|
|
queueJson['httpHeaders']['signature'])
|
|
|
|
pubKey = None
|
|
|
|
break
|
|
|
|
|
|
|
|
pubKey = \
|
|
|
|
getPersonPubKey(baseDir, session, keyId,
|
|
|
|
personCache, debug,
|
|
|
|
projectVersion, httpPrefix,
|
2021-08-31 14:17:11 +00:00
|
|
|
domain, onionDomain, signingPrivateKeyPem)
|
2020-04-16 09:49:57 +00:00
|
|
|
if pubKey:
|
2019-07-04 17:31:41 +00:00
|
|
|
if debug:
|
2020-04-16 09:49:57 +00:00
|
|
|
print('DEBUG: public key: ' + str(pubKey))
|
|
|
|
break
|
2019-07-04 17:31:41 +00:00
|
|
|
|
2019-08-15 08:36:49 +00:00
|
|
|
if debug:
|
2020-04-16 09:49:57 +00:00
|
|
|
print('DEBUG: Retry ' + str(tries+1) +
|
|
|
|
' obtaining public key for ' + keyId)
|
2020-04-16 18:25:59 +00:00
|
|
|
time.sleep(1)
|
2019-07-04 12:23:53 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
if not pubKey:
|
2021-03-14 19:39:00 +00:00
|
|
|
if debug:
|
|
|
|
print('Queue: public key could not be obtained from ' + keyId)
|
2020-04-16 09:49:57 +00:00
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 2 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
|
|
|
|
2021-01-03 09:44:33 +00:00
|
|
|
# check the http header signature
|
2020-04-16 09:49:57 +00:00
|
|
|
if debug:
|
2021-01-03 09:44:33 +00:00
|
|
|
print('DEBUG: checking http header signature')
|
2020-04-16 09:49:57 +00:00
|
|
|
pprint(queueJson['httpHeaders'])
|
2020-12-22 21:24:46 +00:00
|
|
|
postStr = json.dumps(queueJson['post'])
|
2021-02-14 15:22:03 +00:00
|
|
|
httpSignatureFailed = False
|
2020-04-16 09:49:57 +00:00
|
|
|
if not verifyPostHeaders(httpPrefix,
|
|
|
|
pubKey,
|
|
|
|
queueJson['httpHeaders'],
|
|
|
|
queueJson['path'], False,
|
|
|
|
queueJson['digest'],
|
2020-12-22 21:24:46 +00:00
|
|
|
postStr,
|
2020-04-16 09:49:57 +00:00
|
|
|
debug):
|
2021-02-14 15:22:03 +00:00
|
|
|
httpSignatureFailed = True
|
2020-04-16 09:49:57 +00:00
|
|
|
print('Queue: Header signature check failed')
|
2021-03-14 18:16:24 +00:00
|
|
|
pprint(queueJson['httpHeaders'])
|
2021-02-14 15:22:03 +00:00
|
|
|
else:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: http header signature check success')
|
2020-03-22 21:16:02 +00:00
|
|
|
|
2021-01-04 19:02:24 +00:00
|
|
|
# check if a json signature exists on this post
|
2021-06-07 14:07:15 +00:00
|
|
|
hasJsonSignature, jwebsigType = _checkJsonSignature(baseDir, queueJson)
|
2021-01-05 10:29:37 +00:00
|
|
|
|
2021-01-05 10:54:50 +00:00
|
|
|
# strict enforcement of json signatures
|
2021-02-14 15:22:03 +00:00
|
|
|
if not hasJsonSignature:
|
|
|
|
if httpSignatureFailed:
|
2021-02-14 15:45:42 +00:00
|
|
|
if jwebsigType:
|
|
|
|
print('Queue: Header signature check failed and does ' +
|
|
|
|
'not have a recognised jsonld signature type ' +
|
|
|
|
jwebsigType)
|
|
|
|
else:
|
|
|
|
print('Queue: Header signature check failed and ' +
|
2021-02-14 20:30:01 +00:00
|
|
|
'does not have jsonld signature')
|
2021-02-14 20:41:11 +00:00
|
|
|
if debug:
|
|
|
|
pprint(queueJson['httpHeaders'])
|
2021-01-05 10:29:37 +00:00
|
|
|
|
2021-02-14 15:22:03 +00:00
|
|
|
if verifyAllSignatures:
|
2021-06-07 14:07:15 +00:00
|
|
|
originalJson = queueJson['original']
|
2021-02-14 15:22:03 +00:00
|
|
|
print('Queue: inbox post does not have a jsonld signature ' +
|
|
|
|
keyId + ' ' + str(originalJson))
|
|
|
|
|
|
|
|
if httpSignatureFailed or verifyAllSignatures:
|
2021-01-05 20:55:11 +00:00
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 3 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2021-01-05 20:55:11 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
2021-02-14 15:22:03 +00:00
|
|
|
else:
|
|
|
|
if httpSignatureFailed or verifyAllSignatures:
|
|
|
|
# use the original json message received, not one which
|
|
|
|
# may have been modified along the way
|
2021-06-07 14:07:15 +00:00
|
|
|
originalJson = queueJson['original']
|
2021-02-14 15:22:03 +00:00
|
|
|
if not verifyJsonSignature(originalJson, pubKey):
|
|
|
|
if debug:
|
|
|
|
print('WARN: jsonld inbox signature check failed ' +
|
|
|
|
keyId + ' ' + pubKey + ' ' + str(originalJson))
|
|
|
|
else:
|
|
|
|
print('WARN: jsonld inbox signature check failed ' +
|
|
|
|
keyId)
|
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 4 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2021-02-14 15:22:03 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
|
|
|
else:
|
|
|
|
if httpSignatureFailed:
|
|
|
|
print('jsonld inbox signature check success ' +
|
|
|
|
'via relay ' + keyId)
|
|
|
|
else:
|
|
|
|
print('jsonld inbox signature check success ' + keyId)
|
2021-01-03 14:25:20 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
# set the id to the same as the post filename
|
|
|
|
# This makes the filename and the id consistent
|
|
|
|
# if queueJson['post'].get('id'):
|
2021-09-28 10:28:42 +00:00
|
|
|
# queueJson['post']['id'] = queueJson['id']
|
2020-04-16 09:49:57 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveUndo(session,
|
|
|
|
baseDir, httpPrefix, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
queueJson['post'],
|
|
|
|
federationList,
|
|
|
|
debug):
|
2020-04-16 09:49:57 +00:00
|
|
|
print('Queue: Undo accepted from ' + keyId)
|
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 5 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
2019-07-17 10:34:00 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
if debug:
|
|
|
|
print('DEBUG: checking for follow requests')
|
|
|
|
if receiveFollowRequest(session,
|
|
|
|
baseDir, httpPrefix, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
queueJson['post'],
|
|
|
|
federationList,
|
2020-10-23 19:18:13 +00:00
|
|
|
debug, projectVersion,
|
2021-08-31 14:17:11 +00:00
|
|
|
maxFollowers, onionDomain,
|
|
|
|
signingPrivateKeyPem):
|
2020-04-16 09:49:57 +00:00
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 6 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
print('Queue: Follow activity for ' + keyId +
|
2020-06-28 19:04:43 +00:00
|
|
|
' removed from queue')
|
2020-04-16 09:49:57 +00:00
|
|
|
continue
|
|
|
|
else:
|
2019-08-15 16:05:28 +00:00
|
|
|
if debug:
|
2020-04-16 09:49:57 +00:00
|
|
|
print('DEBUG: No follow requests')
|
|
|
|
|
|
|
|
if receiveAcceptReject(session,
|
|
|
|
baseDir, httpPrefix, domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers, personCache,
|
|
|
|
queueJson['post'],
|
|
|
|
federationList, debug):
|
|
|
|
print('Queue: Accept/Reject received from ' + keyId)
|
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 7 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
2019-07-06 15:17:21 +00:00
|
|
|
|
2020-12-22 18:06:23 +00:00
|
|
|
if _receiveUpdate(recentPostsCache, session,
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
domain, port,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache,
|
|
|
|
queueJson['post'],
|
|
|
|
federationList,
|
|
|
|
queueJson['postNickname'],
|
|
|
|
debug):
|
2021-03-14 19:46:46 +00:00
|
|
|
if debug:
|
|
|
|
print('Queue: Update accepted from ' + keyId)
|
2020-04-16 09:49:57 +00:00
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 8 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
|
|
|
|
|
|
|
# get recipients list
|
|
|
|
recipientsDict, recipientsDictFollowers = \
|
2020-12-22 18:06:23 +00:00
|
|
|
_inboxPostRecipients(baseDir, queueJson['post'],
|
|
|
|
httpPrefix, domain, port, debug)
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(recipientsDict.items()) == 0 and \
|
|
|
|
len(recipientsDictFollowers.items()) == 0:
|
2021-03-14 19:22:58 +00:00
|
|
|
if debug:
|
|
|
|
print('Queue: no recipients were resolved ' +
|
|
|
|
'for post arriving in inbox')
|
2020-04-16 09:49:57 +00:00
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 9 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|
|
|
|
continue
|
2019-07-09 14:20:23 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
# if there are only a small number of followers then
|
|
|
|
# process them as if they were specifically
|
|
|
|
# addresses to particular accounts
|
|
|
|
noOfFollowItems = len(recipientsDictFollowers.items())
|
|
|
|
if noOfFollowItems > 0:
|
|
|
|
# always deliver to individual inboxes
|
|
|
|
if noOfFollowItems < 999999:
|
|
|
|
if debug:
|
|
|
|
print('DEBUG: moving ' + str(noOfFollowItems) +
|
|
|
|
' inbox posts addressed to followers')
|
|
|
|
for handle, postItem in recipientsDictFollowers.items():
|
|
|
|
recipientsDict[handle] = postItem
|
|
|
|
recipientsDictFollowers = {}
|
|
|
|
# recipientsList = [recipientsDict, recipientsDictFollowers]
|
|
|
|
|
|
|
|
if debug:
|
|
|
|
print('*************************************')
|
|
|
|
print('Resolved recipients list:')
|
|
|
|
pprint(recipientsDict)
|
|
|
|
print('Resolved followers list:')
|
|
|
|
pprint(recipientsDictFollowers)
|
|
|
|
print('*************************************')
|
|
|
|
|
|
|
|
# Copy any posts addressed to followers into the shared inbox
|
|
|
|
# this avoid copying file multiple times to potentially many
|
|
|
|
# individual inboxes
|
|
|
|
if len(recipientsDictFollowers) > 0:
|
|
|
|
sharedInboxPostFilename = \
|
|
|
|
queueJson['destination'].replace(inboxHandle, inboxHandle)
|
|
|
|
if not os.path.isfile(sharedInboxPostFilename):
|
|
|
|
saveJson(queueJson['post'], sharedInboxPostFilename)
|
|
|
|
|
2021-10-21 19:00:25 +00:00
|
|
|
listsEnabled = getConfigParam(baseDir, "listsEnabled")
|
2021-11-08 18:09:24 +00:00
|
|
|
contentLicenseUrl = getConfigParam(baseDir, "contentLicenseUrl")
|
2021-10-21 19:00:25 +00:00
|
|
|
|
2020-04-16 09:49:57 +00:00
|
|
|
# for posts addressed to specific accounts
|
|
|
|
for handle, capsId in recipientsDict.items():
|
|
|
|
destination = \
|
|
|
|
queueJson['destination'].replace(inboxHandle, handle)
|
2020-12-22 18:06:23 +00:00
|
|
|
_inboxAfterInitial(recentPostsCache,
|
|
|
|
maxRecentPosts,
|
|
|
|
session, keyId, handle,
|
|
|
|
queueJson['post'],
|
|
|
|
baseDir, httpPrefix,
|
|
|
|
sendThreads, postLog,
|
|
|
|
cachedWebfingers,
|
|
|
|
personCache, queue,
|
|
|
|
domain,
|
|
|
|
onionDomain, i2pDomain,
|
|
|
|
port, proxyType,
|
|
|
|
federationList,
|
|
|
|
debug,
|
|
|
|
queueFilename, destination,
|
|
|
|
maxReplies, allowDeletion,
|
|
|
|
maxMentions, maxEmoji,
|
|
|
|
translate, unitTest,
|
|
|
|
YTReplacementDomain,
|
2021-09-18 17:08:14 +00:00
|
|
|
twitterReplacementDomain,
|
2020-12-22 18:06:23 +00:00
|
|
|
showPublishedDateOnly,
|
2020-12-23 23:59:49 +00:00
|
|
|
allowLocalNetworkAccess,
|
2021-02-24 11:43:48 +00:00
|
|
|
peertubeInstances,
|
2021-03-09 19:52:10 +00:00
|
|
|
lastBounceMessage,
|
2021-08-03 10:04:45 +00:00
|
|
|
themeName, systemLanguage,
|
2021-08-31 14:17:11 +00:00
|
|
|
maxLikeCount,
|
2021-09-08 18:37:04 +00:00
|
|
|
signingPrivateKeyPem,
|
2021-10-21 13:08:21 +00:00
|
|
|
defaultReplyIntervalHours,
|
2021-11-08 18:09:24 +00:00
|
|
|
CWlists, listsEnabled,
|
|
|
|
contentLicenseUrl)
|
2020-09-27 18:35:35 +00:00
|
|
|
if debug:
|
|
|
|
pprint(queueJson['post'])
|
2021-03-14 19:22:58 +00:00
|
|
|
print('Queue: Queue post accepted')
|
2020-04-16 09:49:57 +00:00
|
|
|
if os.path.isfile(queueFilename):
|
2021-09-05 10:17:43 +00:00
|
|
|
try:
|
|
|
|
os.remove(queueFilename)
|
2021-11-25 18:42:38 +00:00
|
|
|
except OSError:
|
2021-10-29 18:48:15 +00:00
|
|
|
print('EX: runInboxQueue 10 unable to delete ' +
|
|
|
|
str(queueFilename))
|
2020-04-16 09:49:57 +00:00
|
|
|
if len(queue) > 0:
|
|
|
|
queue.pop(0)
|