2022-06-04 23:59:10 +02:00
|
|
|
<?php
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
class GitlabIssueBridge extends BridgeAbstract
|
|
|
|
{
|
|
|
|
const MAINTAINER = 'Mynacol';
|
2023-08-27 12:13:44 +02:00
|
|
|
const NAME = 'Gitlab Issue/Merge Request/Epic';
|
2022-06-04 23:59:10 +02:00
|
|
|
const URI = 'https://gitlab.com/';
|
|
|
|
const CACHE_TIMEOUT = 1800; // 30min
|
2023-08-27 12:13:44 +02:00
|
|
|
const DESCRIPTION = 'Returns comments of an issue/MR/Epic of a gitlab project';
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
const PARAMETERS = [
|
|
|
|
'global' => [
|
|
|
|
'h' => [
|
|
|
|
'name' => 'Gitlab instance host name',
|
|
|
|
'exampleValue' => 'gitlab.com',
|
|
|
|
'defaultValue' => 'gitlab.com',
|
|
|
|
'required' => true
|
|
|
|
],
|
|
|
|
'u' => [
|
|
|
|
'name' => 'User/Organization name',
|
2023-08-27 13:14:21 +02:00
|
|
|
'exampleValue' => 'gitlab-org',
|
2022-06-04 23:59:10 +02:00
|
|
|
'required' => true
|
|
|
|
],
|
|
|
|
'p' => [
|
|
|
|
'name' => 'Project name',
|
2023-08-27 13:14:21 +02:00
|
|
|
'exampleValue' => 'gitlab-foss',
|
2022-06-04 23:59:10 +02:00
|
|
|
'required' => true
|
2022-07-01 15:10:30 +02:00
|
|
|
]
|
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
],
|
|
|
|
'Issue comments' => [
|
|
|
|
'i' => [
|
|
|
|
'name' => 'Issue number',
|
|
|
|
'type' => 'number',
|
2023-08-27 13:14:21 +02:00
|
|
|
'exampleValue' => '1',
|
2022-06-04 23:59:10 +02:00
|
|
|
'required' => true
|
2022-07-01 15:10:30 +02:00
|
|
|
]
|
2022-06-04 23:59:10 +02:00
|
|
|
],
|
|
|
|
'Merge Request comments' => [
|
|
|
|
'i' => [
|
|
|
|
'name' => 'Merge Request number',
|
|
|
|
'type' => 'number',
|
2023-08-27 13:14:21 +02:00
|
|
|
'exampleValue' => '1',
|
2022-06-04 23:59:10 +02:00
|
|
|
'required' => true
|
2022-07-01 15:10:30 +02:00
|
|
|
]
|
2023-08-27 12:13:44 +02:00
|
|
|
],
|
|
|
|
'Epic comments' => [
|
|
|
|
'i' => [
|
|
|
|
'name' => 'Epic number',
|
|
|
|
'type' => 'number',
|
2023-08-27 13:14:21 +02:00
|
|
|
'exampleValue' => '1',
|
2023-08-27 12:13:44 +02:00
|
|
|
'required' => true
|
|
|
|
]
|
2022-07-01 15:10:30 +02:00
|
|
|
]
|
2022-06-04 23:59:10 +02:00
|
|
|
];
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
public function getName()
|
|
|
|
{
|
|
|
|
$name = $this->getInput('h') . '/' . $this->getInput('u') . '/' . $this->getInput('p');
|
|
|
|
switch ($this->queriedContext) {
|
|
|
|
case 'Issue comments':
|
|
|
|
$name .= ' Issue #' . $this->getInput('i');
|
|
|
|
break;
|
|
|
|
case 'Merge Request comments':
|
|
|
|
$name .= ' MR !' . $this->getInput('i');
|
|
|
|
break;
|
2023-08-27 12:13:44 +02:00
|
|
|
case 'Epic comments':
|
|
|
|
$name .= ' Epic &' . $this->getInput('i');
|
|
|
|
break;
|
2022-06-04 23:59:10 +02:00
|
|
|
default:
|
|
|
|
return parent::getName();
|
|
|
|
}
|
|
|
|
return $name;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-05 18:39:54 +02:00
|
|
|
public function getURI()
|
|
|
|
{
|
2022-06-04 23:59:10 +02:00
|
|
|
$host = $this->getInput('h') ?? 'gitlab.com';
|
2022-06-05 18:39:54 +02:00
|
|
|
$uri = 'https://' . $host . '/' . $this->getInput('u') . '/'
|
2022-06-04 23:59:10 +02:00
|
|
|
. $this->getInput('p') . '/';
|
|
|
|
switch ($this->queriedContext) {
|
|
|
|
case 'Issue comments':
|
|
|
|
$uri .= '-/issues';
|
|
|
|
break;
|
|
|
|
case 'Merge Request comments':
|
|
|
|
$uri .= '-/merge_requests';
|
|
|
|
break;
|
2023-08-27 12:13:44 +02:00
|
|
|
case 'Epic comments':
|
2023-08-27 13:14:21 +02:00
|
|
|
$uri = 'https://' . $host . '/groups/' . $this->getInput('u') . '/-/epics';
|
2023-08-27 12:13:44 +02:00
|
|
|
break;
|
2022-06-04 23:59:10 +02:00
|
|
|
default:
|
|
|
|
return $uri;
|
|
|
|
}
|
|
|
|
$uri .= '/' . $this->getInput('i');
|
|
|
|
return $uri;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
public function getIcon()
|
|
|
|
{
|
|
|
|
return 'https://' . $this->getInput('h') . '/favicon.ico';
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
public function collectData()
|
|
|
|
{
|
|
|
|
switch ($this->queriedContext) {
|
|
|
|
case 'Issue comments':
|
|
|
|
$this->items[] = $this->parseIssueDescription();
|
|
|
|
break;
|
|
|
|
case 'Merge Request comments':
|
2022-06-05 18:39:54 +02:00
|
|
|
$this->items[] = $this->parseMergeRequestDescription();
|
2022-06-04 23:59:10 +02:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
/* parse issue/MR comments */
|
|
|
|
$comments_uri = $this->getURI() . '/discussions.json';
|
|
|
|
$comments = getContents($comments_uri);
|
|
|
|
$comments = json_decode($comments, false);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
foreach ($comments as $value) {
|
|
|
|
foreach ($value->notes as $comment) {
|
|
|
|
$item = [];
|
2023-08-27 12:13:44 +02:00
|
|
|
if ($comment->noteable_note_url !== null) {
|
|
|
|
$item['uri'] = $comment->noteable_note_url;
|
|
|
|
$item['uid'] = $item['uri'];
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
// TODO fix invalid timestamps (fdroid bot)
|
|
|
|
$item['timestamp'] = $comment->created_at ?? $comment->updated_at ?? $comment->last_edited_at;
|
|
|
|
$author = $comment->author ?? $comment->last_edited_by;
|
|
|
|
$item['author'] = '<img src="' . $author->avatar_url . '" width=24></img> <a href="https://' .
|
|
|
|
$this->getInput('h') . $author->path . '">' . $author->name . ' @' . $author->username . '</a>';
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
$content = '';
|
|
|
|
if ($comment->system) {
|
|
|
|
$content = $comment->note_html;
|
|
|
|
if ($comment->type === 'StateNote') {
|
|
|
|
$content .= ' the issue';
|
|
|
|
} elseif ($comment->type === null) {
|
|
|
|
// e.g. "added 900 commits\n800 from master\n175h4d - commit message\n..."
|
|
|
|
$content = str_get_html($comment->note_html)->find('p', 0);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// no switch-case to do strict comparison
|
|
|
|
if ($comment->type === null || $comment->type === 'DiscussionNote') {
|
|
|
|
$content = 'commented';
|
|
|
|
} elseif ($comment->type === 'DiffNote') {
|
|
|
|
$content = 'commented on a thread';
|
|
|
|
} else {
|
|
|
|
$content = $comment->note_html;
|
2022-07-01 15:10:30 +02:00
|
|
|
}
|
2022-06-04 23:59:10 +02:00
|
|
|
}
|
|
|
|
$item['title'] = $author->name . " $content";
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
$content = $this->fixImgSrc($comment->note_html);
|
|
|
|
$item['content'] = defaultLinkTo($content, 'https://' . $this->getInput('h') . '/');
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
$this->items[] = $item;
|
2022-07-01 15:10:30 +02:00
|
|
|
}
|
2022-06-04 23:59:10 +02:00
|
|
|
}
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
private function parseIssueDescription()
|
|
|
|
{
|
|
|
|
$description_uri = $this->getURI() . '.json';
|
|
|
|
$description = getContents($description_uri);
|
|
|
|
$description = json_decode($description, false);
|
|
|
|
$description_html = getSimpleHtmlDomCached($this->getURI());
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
$item = [];
|
|
|
|
$item['uri'] = $this->getURI();
|
|
|
|
$item['uid'] = $item['uri'];
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
$item['timestamp'] = $description->created_at ?? $description->updated_at;
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-11-15 03:01:27 +01:00
|
|
|
$author = $this->parseAuthor($description_html);
|
|
|
|
if ($author) {
|
|
|
|
$item['author'] = $author;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
$item['title'] = $description->title;
|
|
|
|
$item['content'] = markdownToHtml($description->description);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
return $item;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-05 18:39:54 +02:00
|
|
|
private function parseMergeRequestDescription()
|
|
|
|
{
|
|
|
|
$description_uri = $this->getURI() . '/cached_widget.json';
|
|
|
|
$description = getContents($description_uri);
|
|
|
|
$description = json_decode($description, false);
|
|
|
|
$description_html = getSimpleHtmlDomCached($this->getURI());
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-05 18:39:54 +02:00
|
|
|
$item = [];
|
|
|
|
$item['uri'] = $this->getURI();
|
|
|
|
$item['uid'] = $item['uri'];
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-05 18:39:54 +02:00
|
|
|
$item['timestamp'] = $description_html->find('.merge-request-details time', 0)->datetime;
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-11-15 03:01:27 +01:00
|
|
|
$author = $this->parseAuthor($description_html);
|
|
|
|
if ($author) {
|
|
|
|
$item['author'] = $author;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-05 18:39:54 +02:00
|
|
|
$item['title'] = 'Merge Request ' . $description->title;
|
|
|
|
$item['content'] = markdownToHtml($description->description);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-05 18:39:54 +02:00
|
|
|
return $item;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
private function fixImgSrc($html)
|
|
|
|
{
|
|
|
|
if (is_string($html)) {
|
|
|
|
$html = str_get_html($html);
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
foreach ($html->find('img') as $img) {
|
|
|
|
$img->src = $img->getAttribute('data-src');
|
|
|
|
}
|
|
|
|
return $html;
|
|
|
|
}
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
private function parseAuthor($description_html)
|
|
|
|
{
|
|
|
|
$description_html = $this->fixImgSrc($description_html);
|
2022-07-01 15:10:30 +02:00
|
|
|
|
2022-06-04 23:59:10 +02:00
|
|
|
$authors = $description_html->find('.issuable-meta a.author-link, .merge-request a.author-link');
|
|
|
|
$editors = $description_html->find('.edited-text a.author-link');
|
2022-11-15 03:01:27 +01:00
|
|
|
if ($authors === [] && $editors === []) {
|
|
|
|
return null;
|
|
|
|
}
|
2022-06-04 23:59:10 +02:00
|
|
|
$author_str = implode(' ', $authors);
|
|
|
|
if ($editors) {
|
|
|
|
$author_str .= ', ' . implode(' ', $editors);
|
|
|
|
}
|
|
|
|
return defaultLinkTo($author_str, 'https://' . $this->getInput('h') . '/');
|
|
|
|
}
|
|
|
|
}
|