=============================== Converting a fedmsg application =============================== .. highlight:: python Converting publishers ===================== Converting a Flask app ---------------------- .. elections, fedocal Let's use the `elections`_ app as an example. Clone the code using the following command:: git clone https://pagure.io/elections.git And change to this directory. .. _elections: https://pagure.io/elections/ In the ``elections`` app, all calls to publish messages on fedmsg are going through the ``fedora_elections.fedmsgshim.publish`` wrapper function. We can thus modify this function to make it call Fedora Messaging instead of fedmsg. JSON schema ~~~~~~~~~~~ First, you will need a Message schema. To write this schema you must know what kind of messages are sent on the bus. A ``git grep`` command will reveal that all calls are made from the ``admin.py`` file. Open that file and examine those calls. In parallel, copy the ``docs/sample_schema_package/`` directory from the ``fedora-messaging`` git clone to your app directory. Rename it to ``elections-message-schemas``. Edit the ``setup.py`` file like you did before, to change the package metadata (including the entry point). Use ``fedora_elections_message_schemas`` for the name. Rename the ``mailman_schema`` directory to ``fedora_elections_message_schemas`` and adapt the ``setup.py`` metadata. Edit the ``schema.py`` file and write the basic structure for the elections message schema. According to the different calls in ``admin.py``, it could be something like:: { 'id': 'http://fedoraproject.org/message-schema/elections#', '$schema': 'http://json-schema.org/draft-04/schema#', 'description': 'Schema for Fedora Elections', 'type': 'object', 'properties': { 'agent': {'type': 'string'}, 'election': {'type': 'object'}, 'candidate': {'type': 'object'}, }, 'required': ['agent', 'election'], } This could be sufficient, but it would be best to list what properties are available in the ``election`` and ``candidate`` keys. Unfortunately, those are just JSON dumps of the database model, so you'll have to look further to know the structure. Examining the ``to_json()`` methods in ``models.py`` shows which keys are dumped to JSON. The schema could be written as:: { 'id': 'http://fedoraproject.org/message-schema/elections#', '$schema': 'http://json-schema.org/draft-04/schema#', 'description': 'Schema for Fedora Elections', 'type': 'object', 'properties': { 'agent': {'type': 'string'}, 'election': { 'type': 'object', 'properties': { 'shortdesc': {'type': 'string'}, 'alias': {'type': 'string'}, 'description': {'type': 'string'}, 'url': {'type': 'string', 'format': 'uri'}, 'start_date': {'type': 'string'}, 'end_date': {'type': 'string'}, 'embargoed': {'type': 'number'}, 'voting_type': {'type': 'string'}, }, 'required': [ 'shortdesc', 'alias', 'description', 'url', 'start_date', 'end_date', 'embargoed', 'voting_type', ], }, 'candidate': { 'type': 'object', 'properties': { 'name': {'type': 'string'}, 'url': {'type': 'string', 'format': 'uri'}, }, 'required': ['name', 'url'], }, }, 'required': ['agent', 'election'], } Use this schema and adapt the ``__str__()`` method and the ``summary`` property. Since the schema is distributed in a separate python package, it must be added to the ``election`` app's dependencies in ``requirements.txt``. Wrapper function ~~~~~~~~~~~~~~~~ Now you can import this class in ``fedora_elections/fedmsgshim.py`` and use it to encapsulate the messages. The wrapper could look like:: import logging from fedora_elections_message_schemas.schema import Message from fedora_messaging.api import publish as fm_publish from fedora_messaging.exceptions import PublishReturned, ConnectionException LOGGER = logging.getLogger(__name__) def publish(topic, msg): try: fm_publish(Message( topic="fedora.elections." + topic, body=msg, )) except PublishReturned as e: LOGGER.warning( "Fedora Messaging broker rejected message %s: %s", msg.id, e ) except ConnectionException as e: LOGGER.warning("Error sending the message %s: %s", msg.id, e) With this you'll get a couple of nice features over the previous state of things: - the message format is validated, so it's your responsability to update the schema when you decide to change the format, and not the receiver's responsability to handle any database schema changes you may make that may bleed into the message dictionary. And you'll know during development if you break compatibility. - you may handle messaging errors in anyway you deem relevant. Here we're just logging them but you could choose to re-send the messages, store them for further analysis, etc. - when there are no exceptions, you know that the message has reached the broker and has been distributed. Testing ~~~~~~~ Let's start the election app and make sure messages are properly sent on the bus. First, we'll create a virtualenv, and install election and fedora-messaging with the following commands:: virtualenv venv source ./venv/bin/activate pushd elections-message-schemas python setup.py develop popd pip install -r requirements.txt python setup.py develop Make sure the Fedora Messaging configuration file is correct in ``/etc/fedora-messaging/config.toml``. We will add a queue binding to route messages with the ``fedora.elections`` topic to the ``tutorial`` queue. Add this entry in the ``bindings`` list:: [[bindings]] queue = "tutorial" exchange = "amq.topic" routing_keys = ["fedora.elections.#"] You could also add ``"fedora.elections.#"`` to the ``"routing_keys"`` value in the existing entry. Now make sure that RabbitMQ is still running, and run the ``consume.py`` script :ref:`we used before `. Make sure it is not systematically raising exceptions in the callback function (as we did before). Now we'll run the election app, but first we need to create a configuration file. Create a file called ``config.py`` with the following content:: FEDORA_ELECTIONS_ADMIN_GROUP = "" This will allow any Fedora account to be an admin on your instance, which is good enough for this tutorial. Now start the app with:: python createdb.py python runserver.py -c config.py Open your browser to http://localhost:5000/admin/new. Login with FAS, then create an election. Check the terminal where the ``consume.py`` script is running. You should see the message that the ``elections`` app has sent on election creation. Edit the election, and you should see the corresponding message in the terminal where ``consume.py`` is running. Converting a Pyramid app ------------------------ Let's use the `github2fedmsg`_ app as an example. It is a Pyramid webapp that registers a webhook with Github on all subscribed projects, and then broadcasts actions (commits, pull-request, tickets) received on this webhook to the message bus. .. _github2fedmsg: https://github.com/fedora-infra/github2fedmsg Clone the code using the following command:: git clone git@github.com:fedora-infra/github2fedmsg.git And change to this directory. JSON Schema ~~~~~~~~~~~ The only call to fedmsg is in ``github2fedmsg/views/webhooks.py``. Since the app transmits the webhook payload almost transparently to the message bus, the structure isn't obvious, so it's harder to define a schema. Fortunately, the Github documentation has a `comprehensive list`_ of payload formats. .. _comprehensive list: https://developer.github.com/v3/activity/events/types/ It would be to long to define precise JSON schemas for each event type, so we'll just use the generic schema. Sending the messages ~~~~~~~~~~~~~~~~~~~~ Now you can replace the current call to fedmsg with a call to :py:func:`fedora_messaging.api.publish `. Add these lines in the ``github2fedmsg.views.webhook`` module:: import logging from fedora_messaging.api import Message, publish from fedora_messaging.exceptions import PublishReturned, ConnectionException LOGGER = logging.getLogger(__name__) And replace the call to ``fedmsg.publish`` with:: try: msg = Message( topic="github." + event_type, body=payload, ) publish(msg) except PublishReturned as e: LOGGER.warning( "Fedora Messaging broker rejected message %s: %s", msg.id, e ) except ConnectionException as e: LOGGER.warning("Error sending message %s: %s", msg.id, e) Testing it ~~~~~~~~~~ Make sure the Fedora Messaging configuration file is correct in ``/etc/fedora-messaging/config.toml``. We will add a queue binding to route messages with the ``github`` topic to the ``tutorial`` queue. Add this entry in the ``bindings`` list:: [[bindings]] queue = "tutorial" exchange = "amq.topic" routing_keys = ["github.#"] You could also add ``"github.#"`` to the ``"routing_keys"`` value in the existing entry. Now make sure that RabbitMQ is still running, and run the ``consume.py`` script :ref:`we used before `. Make sure it is not systematically raising exceptions in the callback function (as we did before). To setup the ``github2fedmsg`` application, follow the ``README.rst`` file:: virtualenv venv source ./venv/bin/activate python setup.py develop pip install waitress Go off and `register your development application with GitHub `_. Save the oauth tokens and add the secret one to a new file you create called ``secret.ini``. Use the example ``secret.ini.example`` file. Create the database and start the application:: initialize_github2fedmsg_db development.ini pserve development.ini --reload Converting consumers ==================== TODO the-new-hotness