Build your own event system in Python

Event system is not hard to build on your own. There are plenty of libraries ready to use, but for better understanding I want to implement it by myself.

The idea of this post comes from this video. Today I was searching the internet for tips how to …

Event system is not hard to build on your own. There are plenty of libraries ready to use, but for better understanding I want to implement it by myself.

The idea of this post comes from this video. Today I was searching the internet for tips how to manage my project which has a lot of different modules and I wanted to do it nice and clean. One of the ideas I found is Observer pattern from “Gang of four” book. This should be very simple implementation of it.

First thing is that you have subscribers. They subscribe to different types of events. Every time the event is posted, you have to notify the subscriber about it.

First create two methods for subscribing the event, and for posting.

# src/event.py

from collections import defaultdict

subscribers = defaultdict(list)

def subscribe(event_type, fn):
    subscribers[event_type].append(fn)

def post_event(event_type, data): 
    if event_type in subscribers:
        for fn in subscribers[event_type]:
            fn(data)

In this example I will be creating the new user which will invoke event new_user_created. Subscriber for this event will be module for sending welcome message by email.

Let’s create a user. It will be simple dictionary, but it can be an object too.

# src/user.py

from .event import post_event

def register_new_user(name, password, email ):
    user = dict( name = name, 
                password = password, 
                email = email)

    post_event("new_user_created", user)

Now handle sending emails. We will need two things here. Some kind of email provider and some kind of handler for it. I want my email provider class to handle all ‘business logic’ only, so I created modules folder for such.

# src/modules/email.py 

class Email:
    def sendEmail( email, subject, message ):
        print("==========================")
        print(f"From: {email}")
        print(f"Subject: {subject}")
        print(message)
        print("==========================")

And handler for email.

# src/email_handler.py

from .modules.email import Email 
from .event import subscribe

def handle_user_registered_event(user):
    Email.sendEmail(user['email'], 'Welcome!', 'Some welcome message')

def setup_email_event_handlers():
    subscribe('new_user_created', handle_user_registered_event)

Now let’s connect everything together. Make an app file.

# src/app.py

from src.user import register_new_user
from src.email_handler import setup_email_event_handlers

setup_email_event_handlers()

register_new_user('Jakub', 'secret', 'name@domain.com')

First thing to do is to set email handler events up. That means, subscribe to event new_user_created.

If all the subscriptions are done, register new user. Output should be as follows.

==========================
From: name@domain.com
Subject: Welcome!
Some welcome message
==========================



Extend the system

What if we want to extend this. Let’s add some database. In modules, add database.py file and sample implementation of database as a list of users.

# src/modules/database.py 

class DB:
    users = [ ]
    def register_new_user( user ):
        DB.users.append(user)
        print('=======================')
        print(DB.users)
        print('=======================')

And handler for it.

# src/database_handler.py

from .modules.database import DB
from .event import subscribe

def handle_user_registered_event(user):
    DB.register_new_user(user)

def setup_database_event_handlers():
    subscribe('new_user_created', handle_user_registered_event)

Now the only thing to do is set database handler up in app file.

# src/app.py

from src.user import register_new_user
from src.email_handler import setup_email_event_handlers
from src.database_handler import setup_database_event_handlers 

setup_email_event_handlers()
setup_database_event_handlers()

register_new_user('Jakub', 'secret', 'name@domain.com')

Now the result should be:

==========================
From: name@domain.com
Subject: Welcome!
Some welcome message
==========================
==========================
[{'name': 'Jakub', 'password': 'secret', 'email': 'name@domain.com'}]
==========================

Nice thing about it, is that this way keeps your modules independent of each other and keeps their cohesion strong. Besides that, it allows you to manipulate order of listeners in event very quickly. No more ‘ctrl + c | ctrl + v’ whole code ?.

Comment what do you think about it! How do you build workflow in your apps between multiple modules?


Print Share Comment Cite Upload Translate
APA
Kuba | Sciencx (2024-03-28T16:35:31+00:00) » Build your own event system in Python. Retrieved from https://www.scien.cx/2021/04/22/build-your-own-event-system-in-python/.
MLA
" » Build your own event system in Python." Kuba | Sciencx - Thursday April 22, 2021, https://www.scien.cx/2021/04/22/build-your-own-event-system-in-python/
HARVARD
Kuba | Sciencx Thursday April 22, 2021 » Build your own event system in Python., viewed 2024-03-28T16:35:31+00:00,<https://www.scien.cx/2021/04/22/build-your-own-event-system-in-python/>
VANCOUVER
Kuba | Sciencx - » Build your own event system in Python. [Internet]. [Accessed 2024-03-28T16:35:31+00:00]. Available from: https://www.scien.cx/2021/04/22/build-your-own-event-system-in-python/
CHICAGO
" » Build your own event system in Python." Kuba | Sciencx - Accessed 2024-03-28T16:35:31+00:00. https://www.scien.cx/2021/04/22/build-your-own-event-system-in-python/
IEEE
" » Build your own event system in Python." Kuba | Sciencx [Online]. Available: https://www.scien.cx/2021/04/22/build-your-own-event-system-in-python/. [Accessed: 2024-03-28T16:35:31+00:00]
rf:citation
» Build your own event system in Python | Kuba | Sciencx | https://www.scien.cx/2021/04/22/build-your-own-event-system-in-python/ | 2024-03-28T16:35:31+00:00
https://github.com/addpipe/simple-recorderjs-demo