Source code for IGitt.GitLab.GitLabNotification
"""
This contains the Notification implementation for GitHub.
Take note that GitHub Notifications are actually available via Todos API.
"""
from functools import lru_cache
from typing import Union
from IGitt.GitLab import BASE_URL
from IGitt.GitLab import GitLabMixin
from IGitt.GitLab import GitLabOAuthToken
from IGitt.GitLab import GitLabPrivateToken
from IGitt.GitLab.GitLabIssue import GitLabIssue
from IGitt.GitLab.GitLabMergeRequest import GitLabMergeRequest
from IGitt.GitLab.GitLabRepository import GitLabRepository
from IGitt.Interfaces import get
from IGitt.Interfaces import post
from IGitt.Interfaces.Notification import Notification
from IGitt.Interfaces.Notification import Reason
[docs]class GitLabNotification(GitLabMixin, Notification):
"""
This class represents a Notification on GitLab.
"""
def __init__(self,
token: Union[GitLabOAuthToken, GitLabPrivateToken],
identifier: Union[str, int]):
"""
Creates a new GitLabNotification object with the given credentials.
"""
self._token = token
self._id = str(identifier)
self._url = '/todos/' + self._id
@property
def identifier(self) -> str:
"""
Returns the id of the notification thread from GitLab.
"""
return self._id
@property
def reason(self) -> Reason:
"""
Returns the reason for notification.
"""
return {
'assigned': Reason.ASSIGNED,
'mentioned': Reason.MENTIONED,
'directly_addressed': Reason.MENTIONED,
'marked': Reason.MARKED,
'build_failed': Reason.BUILD_FAILED,
'approval_required': Reason.APPROVAL_REQUIRED
}[self.data['action_name']]
@property
def subject_type(self) -> type:
"""
Returns the type of the subject the notification.
"""
return {
'MergeRequest': GitLabMergeRequest,
'Issue': GitLabIssue
}[self.data['target_type']]
@property
def subject(self) -> Union[GitLabIssue, GitLabMergeRequest]:
"""
Returns the subject of the notification.
:return: A GitLabIssue or GitLabMergeRequest object.
"""
subject_data = self.data['target']
return self.subject_type.from_data(subject_data, self._token,
self.repository.full_name,
subject_data['iid'])
@property
def repository(self) -> GitLabRepository:
"""
Returns the repository this notification is related to.
"""
return GitLabRepository(self._token,
self.data['project']['path_with_namespace'])
@property
def pending(self) -> bool:
"""
Returns True if the notification is pending/not read.
"""
return self.data['state'] == 'pending'
[docs] def unsubscribe(self):
"""
Unsubscribe from this subject.
"""
url = '{}/unsubscribe'.format(self.subject.url.replace(BASE_URL, ''))
self.data = post(self._token, self.absolute_url(url), {})
[docs] def mark_done(self):
"""
Marks the notification as done/read.
"""
self.data = post(self._token, self.url + '/mark_as_done', {})
def _get_data(self):
try:
return [todo for todo in self._fetch_all(self._token)
if str(todo['id']) == self.identifier][0]
except IndexError:
# Couldn't find the matching notification
raise RuntimeError({'error':'404 Not Found'}, 404)
@staticmethod
@lru_cache(None)
def _fetch_all(token):
return get(token, GitLabNotification.absolute_url('/todos'))
[docs] @staticmethod
def fetch_all(token: Union[GitLabPrivateToken, GitLabOAuthToken]):
"""
Returns the list of notifications for the user bearing the token.
"""
return [GitLabNotification.from_data(data, token, data['id'])
for data in GitLabNotification._fetch_all(token)]