2020-05-02 10:07:50 +00:00
|
|
|
__filename__ = "git.py"
|
|
|
|
__author__ = "Bob Mottram"
|
|
|
|
__license__ = "AGPL3+"
|
2022-02-03 13:58:20 +00:00
|
|
|
__version__ = "1.3.0"
|
2020-05-02 10:07:50 +00:00
|
|
|
__maintainer__ = "Bob Mottram"
|
2021-09-10 16:14:50 +00:00
|
|
|
__email__ = "bob@libreserver.org"
|
2020-05-02 10:07:50 +00:00
|
|
|
__status__ = "Production"
|
2021-06-26 11:16:41 +00:00
|
|
|
__module_group__ = "Profile Metadata"
|
2020-05-02 10:07:50 +00:00
|
|
|
|
|
|
|
import os
|
2020-05-02 18:09:54 +00:00
|
|
|
import html
|
2021-12-26 12:02:29 +00:00
|
|
|
from utils import acct_dir
|
2021-12-26 17:12:07 +00:00
|
|
|
from utils import has_object_stringType
|
2020-05-02 10:07:50 +00:00
|
|
|
|
|
|
|
|
2021-12-29 21:55:09 +00:00
|
|
|
def _git_format_content(content: str) -> str:
|
2020-05-02 17:06:13 +00:00
|
|
|
""" replace html formatting, so that it's more
|
|
|
|
like the original patch file
|
|
|
|
"""
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_str = content.replace('<br>', '\n').replace('<br />', '\n')
|
|
|
|
patch_str = patch_str.replace('<p>', '').replace('</p>', '\n')
|
|
|
|
patch_str = html.unescape(patch_str)
|
|
|
|
if 'From ' in patch_str:
|
|
|
|
patch_str = 'From ' + patch_str.split('From ', 1)[1]
|
|
|
|
return patch_str
|
2020-05-02 17:06:13 +00:00
|
|
|
|
|
|
|
|
2021-12-29 21:55:09 +00:00
|
|
|
def _get_git_project_name(base_dir: str, nickname: str, domain: str,
|
|
|
|
subject: str) -> str:
|
2020-05-02 11:08:38 +00:00
|
|
|
"""Returns the project name for a git patch
|
|
|
|
The project name should be contained within the subject line
|
|
|
|
and should match against a list of projects which the account
|
|
|
|
holder wants to receive
|
|
|
|
"""
|
2022-01-02 14:59:05 +00:00
|
|
|
git_projects_filename = \
|
2021-12-26 12:02:29 +00:00
|
|
|
acct_dir(base_dir, nickname, domain) + '/gitprojects.txt'
|
2022-01-02 14:59:05 +00:00
|
|
|
if not os.path.isfile(git_projects_filename):
|
2020-05-02 11:08:38 +00:00
|
|
|
return None
|
2022-01-02 14:59:05 +00:00
|
|
|
subject_line_words = subject.lower().split(' ')
|
|
|
|
for word in subject_line_words:
|
|
|
|
if word in open(git_projects_filename).read():
|
2020-05-02 11:08:38 +00:00
|
|
|
return word
|
2020-05-02 16:24:17 +00:00
|
|
|
return None
|
2020-05-02 11:08:38 +00:00
|
|
|
|
|
|
|
|
2021-12-29 21:55:09 +00:00
|
|
|
def is_git_patch(base_dir: str, nickname: str, domain: str,
|
2022-01-02 14:59:05 +00:00
|
|
|
message_type: str,
|
2021-12-29 21:55:09 +00:00
|
|
|
subject: str, content: str,
|
2022-01-02 14:59:05 +00:00
|
|
|
check_project_name: bool = True) -> bool:
|
2020-05-02 10:07:50 +00:00
|
|
|
"""Is the given post content a git patch?
|
|
|
|
"""
|
2022-01-02 14:59:05 +00:00
|
|
|
if message_type not in ('Note', 'Page', 'Patch'):
|
2020-05-03 10:56:29 +00:00
|
|
|
return False
|
2020-05-02 10:07:50 +00:00
|
|
|
# must have a subject line
|
2020-05-02 10:19:24 +00:00
|
|
|
if not subject:
|
2020-05-02 10:07:50 +00:00
|
|
|
return False
|
|
|
|
if '[PATCH]' not in content:
|
|
|
|
return False
|
|
|
|
if '---' not in content:
|
|
|
|
return False
|
|
|
|
if 'diff ' not in content:
|
|
|
|
return False
|
2020-05-03 12:39:54 +00:00
|
|
|
if 'From ' not in content:
|
|
|
|
return False
|
2020-05-02 10:07:50 +00:00
|
|
|
if 'From:' not in content:
|
|
|
|
return False
|
|
|
|
if 'Date:' not in content:
|
|
|
|
return False
|
|
|
|
if 'Subject:' not in content:
|
|
|
|
return False
|
2020-05-02 14:47:30 +00:00
|
|
|
if '<br>' not in content:
|
|
|
|
if '<br />' not in content:
|
|
|
|
return False
|
2022-01-02 14:59:05 +00:00
|
|
|
if check_project_name:
|
|
|
|
project_name = \
|
2021-12-29 21:55:09 +00:00
|
|
|
_get_git_project_name(base_dir, nickname, domain, subject)
|
2022-01-02 14:59:05 +00:00
|
|
|
if not project_name:
|
2020-05-03 12:39:54 +00:00
|
|
|
return False
|
2020-05-02 11:08:38 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2022-01-02 14:59:05 +00:00
|
|
|
def _get_git_hash(patch_str: str) -> str:
|
2020-05-03 11:38:09 +00:00
|
|
|
"""Returns the commit hash from a given patch
|
|
|
|
"""
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_lines = patch_str.split('\n')
|
|
|
|
for line in patch_lines:
|
2020-05-03 11:38:09 +00:00
|
|
|
if line.startswith('From '):
|
|
|
|
words = line.split(' ')
|
|
|
|
if len(words) > 1:
|
|
|
|
if len(words[1]) > 20:
|
|
|
|
return words[1]
|
|
|
|
break
|
|
|
|
return None
|
|
|
|
|
|
|
|
|
2022-01-02 14:59:05 +00:00
|
|
|
def _get_patch_description(patch_str: str) -> str:
|
2020-05-03 13:50:01 +00:00
|
|
|
"""Returns the description from a given patch
|
|
|
|
"""
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_lines = patch_str.split('\n')
|
2020-05-03 13:50:01 +00:00
|
|
|
description = ''
|
|
|
|
started = False
|
2022-01-02 14:59:05 +00:00
|
|
|
for line in patch_lines:
|
2020-05-03 13:50:01 +00:00
|
|
|
if started:
|
|
|
|
if line.strip() == '---':
|
|
|
|
break
|
|
|
|
description += line + '\n'
|
|
|
|
if line.startswith('Subject:'):
|
|
|
|
started = True
|
|
|
|
return description
|
|
|
|
|
|
|
|
|
2021-12-29 21:55:09 +00:00
|
|
|
def convert_post_to_patch(base_dir: str, nickname: str, domain: str,
|
|
|
|
post_json_object: {}) -> bool:
|
2020-05-03 12:39:54 +00:00
|
|
|
"""Detects whether the given post contains a patch
|
2020-05-03 12:52:13 +00:00
|
|
|
and if so then converts it to a Patch ActivityPub type
|
2020-05-03 12:39:54 +00:00
|
|
|
"""
|
2021-12-26 17:12:07 +00:00
|
|
|
if not has_object_stringType(post_json_object, False):
|
2020-05-03 12:39:54 +00:00
|
|
|
return False
|
2021-12-25 22:09:19 +00:00
|
|
|
if post_json_object['object']['type'] == 'Patch':
|
2020-05-03 12:39:54 +00:00
|
|
|
return True
|
2021-12-25 22:09:19 +00:00
|
|
|
if not post_json_object['object'].get('summary'):
|
2020-05-03 12:39:54 +00:00
|
|
|
return False
|
2021-12-25 22:09:19 +00:00
|
|
|
if not post_json_object['object'].get('content'):
|
2020-05-03 12:39:54 +00:00
|
|
|
return False
|
2021-12-25 22:09:19 +00:00
|
|
|
if not post_json_object['object'].get('attributedTo'):
|
2020-05-03 12:39:54 +00:00
|
|
|
return False
|
2021-12-25 22:09:19 +00:00
|
|
|
if not isinstance(post_json_object['object']['attributedTo'], str):
|
2020-08-06 16:21:46 +00:00
|
|
|
return False
|
2021-12-29 21:55:09 +00:00
|
|
|
if not is_git_patch(base_dir, nickname, domain,
|
|
|
|
post_json_object['object']['type'],
|
|
|
|
post_json_object['object']['summary'],
|
|
|
|
post_json_object['object']['content'],
|
|
|
|
False):
|
2020-05-03 12:39:54 +00:00
|
|
|
return False
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_str = _git_format_content(post_json_object['object']['content'])
|
|
|
|
commit_hash = _get_git_hash(patch_str)
|
|
|
|
if not commit_hash:
|
2020-05-03 12:41:05 +00:00
|
|
|
return False
|
2021-12-25 22:09:19 +00:00
|
|
|
post_json_object['object']['type'] = 'Patch'
|
2020-05-03 12:39:54 +00:00
|
|
|
# add a commitedBy parameter
|
2021-12-25 22:09:19 +00:00
|
|
|
if not post_json_object['object'].get('committedBy'):
|
|
|
|
post_json_object['object']['committedBy'] = \
|
|
|
|
post_json_object['object']['attributedTo']
|
2022-01-02 14:59:05 +00:00
|
|
|
post_json_object['object']['hash'] = commit_hash
|
2021-12-25 22:09:19 +00:00
|
|
|
post_json_object['object']['description'] = {
|
2020-05-03 12:39:54 +00:00
|
|
|
"mediaType": "text/plain",
|
2022-01-02 14:59:05 +00:00
|
|
|
"content": _get_patch_description(patch_str)
|
2020-05-03 12:39:54 +00:00
|
|
|
}
|
2020-05-03 12:52:13 +00:00
|
|
|
# remove content map
|
2021-12-25 22:09:19 +00:00
|
|
|
if post_json_object['object'].get('contentMap'):
|
|
|
|
del post_json_object['object']['contentMap']
|
2020-05-03 12:52:13 +00:00
|
|
|
print('Converted post to Patch ActivityPub type')
|
2020-05-03 12:39:54 +00:00
|
|
|
return True
|
|
|
|
|
|
|
|
|
2022-01-02 14:59:05 +00:00
|
|
|
def _git_add_from_handle(patch_str: str, handle: str) -> str:
|
2020-05-03 09:48:12 +00:00
|
|
|
"""Adds the activitypub handle of the sender to the patch
|
|
|
|
"""
|
2022-01-02 14:59:05 +00:00
|
|
|
from_str = 'AP-signed-off-by: '
|
|
|
|
if from_str in patch_str:
|
|
|
|
return patch_str
|
|
|
|
|
|
|
|
patch_lines = patch_str.split('\n')
|
|
|
|
patch_str = ''
|
|
|
|
for line in patch_lines:
|
|
|
|
patch_str += line + '\n'
|
2020-05-03 09:48:12 +00:00
|
|
|
if line.startswith('From:'):
|
2022-01-02 14:59:05 +00:00
|
|
|
if from_str not in patch_str:
|
|
|
|
patch_str += from_str + handle + '\n'
|
|
|
|
return patch_str
|
2020-05-03 09:48:12 +00:00
|
|
|
|
|
|
|
|
2021-12-29 21:55:09 +00:00
|
|
|
def receive_git_patch(base_dir: str, nickname: str, domain: str,
|
2022-01-02 14:59:05 +00:00
|
|
|
message_type: str, subject: str, content: str,
|
|
|
|
from_nickname: str, from_domain: str) -> bool:
|
2020-05-02 11:08:38 +00:00
|
|
|
"""Receive a git patch
|
|
|
|
"""
|
2021-12-29 21:55:09 +00:00
|
|
|
if not is_git_patch(base_dir, nickname, domain,
|
2022-01-02 14:59:05 +00:00
|
|
|
message_type, subject, content):
|
2020-05-02 11:08:38 +00:00
|
|
|
return False
|
2020-05-02 16:05:28 +00:00
|
|
|
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_str = _git_format_content(content)
|
2020-05-02 16:05:28 +00:00
|
|
|
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_lines = patch_str.split('\n')
|
|
|
|
patch_filename = None
|
|
|
|
project_dir = None
|
|
|
|
patches_dir = acct_dir(base_dir, nickname, domain) + '/patches'
|
2020-05-02 10:07:50 +00:00
|
|
|
# get the subject line and turn it into a filename
|
2022-01-02 14:59:05 +00:00
|
|
|
for line in patch_lines:
|
2020-05-02 10:07:50 +00:00
|
|
|
if line.startswith('Subject:'):
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_subject = \
|
2020-05-02 16:44:46 +00:00
|
|
|
line.replace('Subject:', '').replace('/', '|')
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_subject = patch_subject.replace('[PATCH]', '').strip()
|
|
|
|
patch_subject = patch_subject.replace(' ', '_')
|
|
|
|
project_name = \
|
2021-12-29 21:55:09 +00:00
|
|
|
_get_git_project_name(base_dir, nickname, domain, subject)
|
2022-01-02 14:59:05 +00:00
|
|
|
if not os.path.isdir(patches_dir):
|
|
|
|
os.mkdir(patches_dir)
|
|
|
|
project_dir = patches_dir + '/' + project_name
|
|
|
|
if not os.path.isdir(project_dir):
|
|
|
|
os.mkdir(project_dir)
|
|
|
|
patch_filename = \
|
|
|
|
project_dir + '/' + patch_subject + '.patch'
|
2020-05-02 10:07:50 +00:00
|
|
|
break
|
2022-01-02 14:59:05 +00:00
|
|
|
if not patch_filename:
|
2020-05-02 10:07:50 +00:00
|
|
|
return False
|
2022-01-02 14:59:05 +00:00
|
|
|
patch_str = \
|
|
|
|
_git_add_from_handle(patch_str,
|
|
|
|
'@' + from_nickname + '@' + from_domain)
|
2021-11-25 21:18:53 +00:00
|
|
|
try:
|
2022-01-02 14:59:05 +00:00
|
|
|
with open(patch_filename, 'w+') as patch_file:
|
|
|
|
patch_file.write(patch_str)
|
|
|
|
patch_notify_filename = \
|
2021-12-26 12:02:29 +00:00
|
|
|
acct_dir(base_dir, nickname, domain) + '/.newPatchContent'
|
2022-01-02 14:59:05 +00:00
|
|
|
with open(patch_notify_filename, 'w+') as patch_file:
|
|
|
|
patch_file.write(patch_str)
|
2021-11-25 21:18:53 +00:00
|
|
|
return True
|
2021-12-25 15:28:52 +00:00
|
|
|
except OSError as ex:
|
2022-01-02 14:59:05 +00:00
|
|
|
print('EX: receive_git_patch ' + patch_filename + ' ' + str(ex))
|
2020-05-02 16:44:46 +00:00
|
|
|
return False
|