Added Travis-Ci integration (closes #104)
This commit is contained in:
parent
f5b010c15c
commit
b920927dab
9 changed files with 221 additions and 0 deletions
|
@ -53,6 +53,7 @@ Backends
|
|||
platypush/backend/sensor.serial.rst
|
||||
platypush/backend/tcp.rst
|
||||
platypush/backend/todoist.rst
|
||||
platypush/backend/travisci.rst
|
||||
platypush/backend/trello.rst
|
||||
platypush/backend/weather.buienradar.rst
|
||||
platypush/backend/weather.darksky.rst
|
||||
|
|
|
@ -46,6 +46,7 @@ Events
|
|||
platypush/events/sound.rst
|
||||
platypush/events/todoist.rst
|
||||
platypush/events/torrent.rst
|
||||
platypush/events/travisci.rst
|
||||
platypush/events/trello.rst
|
||||
platypush/events/video.rst
|
||||
platypush/events/weather.rst
|
||||
|
|
5
docs/source/platypush/backend/travisci.rst
Normal file
5
docs/source/platypush/backend/travisci.rst
Normal file
|
@ -0,0 +1,5 @@
|
|||
``platypush.backend.travisci``
|
||||
==============================
|
||||
|
||||
.. automodule:: platypush.backend.travisci
|
||||
:members:
|
5
docs/source/platypush/events/travisci.rst
Normal file
5
docs/source/platypush/events/travisci.rst
Normal file
|
@ -0,0 +1,5 @@
|
|||
``platypush.message.event.travisci``
|
||||
====================================
|
||||
|
||||
.. automodule:: platypush.message.event.travisci
|
||||
:members:
|
5
docs/source/platypush/plugins/travisci.rst
Normal file
5
docs/source/platypush/plugins/travisci.rst
Normal file
|
@ -0,0 +1,5 @@
|
|||
``platypush.plugins.travisci``
|
||||
==============================
|
||||
|
||||
.. automodule:: platypush.plugins.travisci
|
||||
:members:
|
|
@ -96,6 +96,7 @@ Plugins
|
|||
platypush/plugins/tcp.rst
|
||||
platypush/plugins/todoist.rst
|
||||
platypush/plugins/torrent.rst
|
||||
platypush/plugins/travisci.rst
|
||||
platypush/plugins/trello.rst
|
||||
platypush/plugins/tts.rst
|
||||
platypush/plugins/tts.google.rst
|
||||
|
|
77
platypush/backend/travisci.py
Normal file
77
platypush/backend/travisci.py
Normal file
|
@ -0,0 +1,77 @@
|
|||
from typing import Optional
|
||||
|
||||
from platypush.backend import Backend
|
||||
from platypush.context import get_plugin
|
||||
from platypush.message.event.travisci import TravisciBuildPassedEvent, TravisciBuildFailedEvent
|
||||
|
||||
|
||||
class TravisciBackend(Backend):
|
||||
"""
|
||||
This backend polls for new builds on a `Travis-Ci <https://travis-ci.org>`_ account and triggers an event
|
||||
whenever a new build is completed.
|
||||
|
||||
Requires:
|
||||
|
||||
* The :class:`platypush.plugins.foursquare.FoursquarePlugin` plugin configured and enabled.
|
||||
|
||||
Triggers:
|
||||
|
||||
- :class:`platypush.message.event.foursquare.FoursquareCheckinEvent` when a new check-in occurs.
|
||||
|
||||
"""
|
||||
|
||||
_last_build_finished_at_varname = '_travisci_last_build_finished_at'
|
||||
|
||||
def __init__(self, poll_seconds: Optional[float] = 60.0, *args, **kwargs):
|
||||
"""
|
||||
:param poll_seconds: How often the backend should check for new builds (default: one minute).
|
||||
"""
|
||||
super().__init__(*args, poll_seconds=poll_seconds, **kwargs)
|
||||
self._last_build_finished_at = None
|
||||
|
||||
def __enter__(self):
|
||||
self._last_build_finished_at = int(get_plugin('variable').get(self._last_build_finished_at_varname).
|
||||
output.get(self._last_build_finished_at_varname) or 0)
|
||||
self.logger.info('Started Travis-CI backend')
|
||||
|
||||
def loop(self):
|
||||
builds = get_plugin('travisci').builds(limit=1).output
|
||||
if not builds:
|
||||
return
|
||||
|
||||
last_build = builds[0]
|
||||
last_build_finished_at = last_build.get('finished_at', 0)
|
||||
if self._last_build_finished_at and last_build_finished_at <= self._last_build_finished_at:
|
||||
return
|
||||
|
||||
if last_build.get('state') == 'passed':
|
||||
evt_type = TravisciBuildPassedEvent
|
||||
elif last_build.get('state') == 'failed':
|
||||
evt_type = TravisciBuildFailedEvent
|
||||
else:
|
||||
return
|
||||
|
||||
evt = evt_type(repository_id=last_build.get('repository', {}).get('id'),
|
||||
repository_name=last_build.get('repository', {}).get('name'),
|
||||
repository_slug=last_build.get('repository').get('slug'),
|
||||
build_id=int(last_build.get('id')),
|
||||
build_number=int(last_build.get('number')),
|
||||
duration=last_build.get('duration'),
|
||||
previous_state=last_build.get('previous_state'),
|
||||
private=last_build.get('private'),
|
||||
tag=last_build.get('tag'),
|
||||
branch=last_build.get('branch', {}).get('name'),
|
||||
commit_id=last_build.get('commit', {}).get('id'),
|
||||
commit_sha=last_build.get('commit', {}).get('sha'),
|
||||
commit_message=last_build.get('commit', {}).get('message'),
|
||||
committed_at=last_build.get('commit', {}).get('committed_at'),
|
||||
created_by=last_build.get('created_by', {}).get('login'),
|
||||
started_at=last_build.get('started_at'),
|
||||
finished_at=last_build.get('finished_at'))
|
||||
|
||||
self.bus.post(evt)
|
||||
self._last_build_finished_at = last_build_finished_at
|
||||
get_plugin('variable').set(**{self._last_build_finished_at_varname: self._last_build_finished_at})
|
||||
|
||||
|
||||
# vim:sw=4:ts=4:et:
|
68
platypush/message/event/travisci.py
Normal file
68
platypush/message/event/travisci.py
Normal file
|
@ -0,0 +1,68 @@
|
|||
from typing import Optional
|
||||
|
||||
from platypush.message.event import Event
|
||||
|
||||
|
||||
class TravisciBuildEvent(Event):
|
||||
def __init__(self,
|
||||
repository_id: int,
|
||||
repository_name: str,
|
||||
repository_slug: str,
|
||||
passed: bool,
|
||||
build_id: int,
|
||||
build_number: int,
|
||||
duration: int,
|
||||
previous_state: str,
|
||||
private: bool,
|
||||
tag: Optional[str],
|
||||
branch: str,
|
||||
commit_id: Optional[str],
|
||||
commit_sha: Optional[str],
|
||||
commit_message: Optional[str],
|
||||
committed_at: str,
|
||||
created_by: str,
|
||||
started_at: str,
|
||||
finished_at: str,
|
||||
*args,
|
||||
**kwargs):
|
||||
super().__init__(*args,
|
||||
repository_id=repository_id,
|
||||
repository_name=repository_name,
|
||||
repository_slug=repository_slug,
|
||||
passed=passed,
|
||||
build_id=build_id,
|
||||
build_number=build_number,
|
||||
duration=duration,
|
||||
previous_state=previous_state,
|
||||
private=private,
|
||||
tag=tag,
|
||||
branch=branch,
|
||||
commit_id=commit_id,
|
||||
commit_sha=commit_sha,
|
||||
commit_message=commit_message,
|
||||
committed_at=committed_at,
|
||||
created_by=created_by,
|
||||
started_at=started_at,
|
||||
finished_at=finished_at,
|
||||
**kwargs)
|
||||
|
||||
|
||||
class TravisciBuildPassedEvent(TravisciBuildEvent):
|
||||
"""
|
||||
Event triggered when a Travis-Ci build passes.
|
||||
"""
|
||||
def __init__(self, *args, **kwargs):
|
||||
kwargs['passed'] = True
|
||||
super().__init__(*args, **kwargs)
|
||||
|
||||
|
||||
class TravisciBuildFailedEvent(TravisciBuildEvent):
|
||||
"""
|
||||
Event triggered when a Travis-Ci build fails.
|
||||
"""
|
||||
def __init__(self, *args, **kwargs):
|
||||
kwargs['passed'] = False
|
||||
super().__init__(*args, **kwargs)
|
||||
|
||||
|
||||
# vim:sw=4:ts=4:et:
|
58
platypush/plugins/travisci.py
Normal file
58
platypush/plugins/travisci.py
Normal file
|
@ -0,0 +1,58 @@
|
|||
import requests
|
||||
from typing import Callable, Dict, Any, List, Optional
|
||||
|
||||
from platypush.plugins import Plugin, action
|
||||
|
||||
|
||||
class TravisciPlugin(Plugin):
|
||||
"""
|
||||
`Travis-Ci <https://travis-ci.org/>`_ continuous integration plugin.
|
||||
|
||||
Setup
|
||||
#####
|
||||
|
||||
- Get your API token from your `Travis-Ci account settings page <https://travis-ci.org/account/preferences>`_.
|
||||
|
||||
"""
|
||||
|
||||
api_base_url = 'https://api.travis-ci.org/'
|
||||
|
||||
def __init__(self, token: str, **kwargs):
|
||||
super().__init__(**kwargs)
|
||||
self.headers = {
|
||||
'Travis-API-Version': '3',
|
||||
'Authorization': 'token ' + token,
|
||||
}
|
||||
|
||||
def _make_request(self, method: Callable, endpoint: str, **kwargs):
|
||||
url = self.api_base_url + endpoint
|
||||
response = method(url, headers=self.headers, **kwargs).json()
|
||||
if response.get('@type') == 'error':
|
||||
raise AssertionError('{type}: {message}'.
|
||||
format(type=response.get('error_type'), message=response.get('error_message')))
|
||||
|
||||
return response
|
||||
|
||||
@action
|
||||
def repos(self) -> Dict[str, Dict[str, Any]]:
|
||||
"""
|
||||
Get the repos owned by current user.
|
||||
:return: Repo name -> Repo attributes mapping.
|
||||
"""
|
||||
return {
|
||||
repo['name']: repo
|
||||
for repo in self._make_request(requests.get, endpoint='repos').get('repositories', [])
|
||||
}
|
||||
|
||||
@action
|
||||
def builds(self, limit: int = 100) -> Dict[str, List[Dict[str, Any]]]:
|
||||
"""
|
||||
Get the list of builds triggered on the owned repositories
|
||||
|
||||
:param limit: Maximum number of builds to be retrieved (default: 100).
|
||||
:return: Repo name -> List of builds
|
||||
"""
|
||||
return self._make_request(requests.get, endpoint='builds').get('builds', [])[:limit]
|
||||
|
||||
|
||||
# vim:sw=4:ts=4:et:
|
Loading…
Reference in a new issue