DEV Community

Cover image for Engineering: A Technical Exploration of Argonaut's Notifications System
Argonaut
Argonaut

Posted on • Originally published at argonaut.dev

Engineering: A Technical Exploration of Argonaut's Notifications System

This article is written with massive help from Prajjwal Dimri who built the Notifications feature.

We are excited to add Notifications to Argonaut. Notifications allow users to get instant updates on any build and deploy actions that occur in a user’s org. This blog is a walk-through of our approach, details of our architecture, along with use cases for this feature.

There are three layers in our architecture: The transformation layer, the processing layer, and the fanout layer. We built the transformation and processing layer in house and used Novu for the fanout layer. It helped us save time by avoiding reengineering this component, which is fairly consistent in most notification systems across various products. It provided us with additional benefits:

  1. Easy integrations with communication channels that our users are on.
  2. Web widget out of the box, making the integration into Argonaut seamless.
  3. User level control for subscribe/unsub for in-app and email notifications.

Notifications play a crucial role in keeping you and your team informed about build and deploy stage updates in real-time. We recognize that the effectiveness of notifications hinges on their promptness, delivery through optimal channels, and appropriate frequency. Consequently, we have designed a customizable notifications system that allows you to receive only the desired notifications through your preferred medium, such as Slack, Teams, Discord, or email.

Use cases

You get a lot of useful information at a glance! The notifications look something like this. And clicking on any notification takes you directly to the respective screen.

Notifications in Argoanut UI

We currently provide different kinds of updates for Application Developers. With the addition of events catering to infra coming soon.

Role in your company Get updates on So that you can
Application Developer High-priority notifications in the case of my builds or deployment pipelines failing Fix the errors quickly
Application Developer High-priority notifications in the case of pods failing to start after a new deployment Fix the errors quickly
Application Developer Opt-in to receive notifications when pipelines succeed Be confident that the deployments went through
👇 Coming soon 👇
DevOps Engineer Pods take a very high amount of resources (close to the set limit) in the cluster Get this fixed by allocating higher limits or getting the dev team to lower the resource usage of the pod
DevOps Engineer Pods getting evicted Fix it by allocating higher limits or getting the dev team to lower the resource usage of the pod
DevOps Engineer Cluster spinning up more nodes than the desired nodes config Check why more nodes are getting spun up and how this is affecting our costs in our cloud provider
DevOps Engineer Get high-priority notifications if any infra-CRUD operations fail Fix and restart the infra operation
A Manager Get notified when environments are created or deleted. Be notified of resource usage
A Manager VCS & cloud account integrations are added or deleted Be in sync about users’ actions in the org

Notifications architecture

Here’s a quick overview of our implementation. The process can be broken down into three layers: The transformation layer, the processing layer, and the fanout layer.

Architecture of notifications in Argonaut

Events

Actions by users, such as creating new apps, updating a build or deploy config, and triggering a pipeline, are all considered to be events. We use event driven programming in our backend, where our internal services communicate with each other through publishing and consuming events from a queue. For example, when a build is triggered, running, gets completed or fails, the build service publishes these events to the queue which can be consumed by various other services.

Transformer

The notification transformer listens to a particular topic in the queue, processes the message, and transforms it into a standard format (including information about the org and the user).

Transformers have the capability to deal with various types of events getting generated in the system and will also have the logic to assign priorities to various types of events. (priority queue coming soon)

Standard notification format

Key Name Key Type Key Description
type EventType (Enum) Type of event that caused this payload to be created
priority number Priority for the event.
users Array (User Ids) The users for which this event is relevant
organizations Array (Org Ids) The organizations for which this event is relevant
createdAt Date-Time Event creation timestamp
metadata JSON Additional metadata and information relevant to the event

We use Novu subscriber in the fanout layer. Therefore, we internally map each org/user with the Novu subscriberId.

Processor

The processor takes the information about the user and the org and retrieves the map, and changes the ID to match the Novu subscriber ID, which is required in the next step to connect with the Fan Out layer and ensure the notifications are routed to the right users.

Fanout layer

The final layer is the Fanout layer, which is managed by Novu. Using Novu made our integration with Slack, Teams, email, and Discord a breeze. The fanout layer (Novu) receives the payload and sends it to the subscribers (members in the same workspace).

Note: Every user and org ID is mapped out to respective subscriberIds.


func (d defaultFanOutService) TriggerFanOut(ctx pkgCommons.IContext, eventId string, subscriberIds []string, payload map[string]interface{}) errors.IError {

novuClient := d.getNovuClient()

_, err := novuClient.EventApi.Trigger(context.Background(), eventId, novu.ITriggerPayloadOptions{

To: subscriberIds,

Payload: payload,

})
Enter fullscreen mode Exit fullscreen mode

Journey of a Notification

  • An event occurs and is published in Argonaut’s internal backend service.
  • The event is added to the SNS queue.
  • The appropriate mapped notification transformer retrieves the event.
  • Transformer decides if the event should be processed or not.
  • The transformer fetches all the users and organizations for which this event is relevant. For example, In case of a failed build event, the relevant org and its users will be fetched.
  • A notification processor (goroutine)is spawned and it picks up the notification.
  • Processor loops over the given users and organizations.
  • Every iteration, it fetches the subscription preferences, creates a payload to trigger Novu, and dispatches the payload.
  • Novu does its magic and sends notifications to users through their selected channels.

Experience using Novu

In our setup, Novu forms the fanout layer, and its main responsibility is sending out notifications to various platforms, e.g., Slack, Email, UI, etc. We are currently using Novu Cloud, which has a free tier supporting up to 10K events per month.

We use Templates offered by Novu, which ties in all the different channels under a single entity. This provides a consistent experience for both the developer and the user, regardless of the channel they choose to receive the notification on.

Entry of notifications and preferences of subscribers is stored in Novu’s MongoDB Database. Notification Aggregation (Digest) is also handled by Novu.

We preferred Novu’s solution over Courier & Knock as it was more economical and it is also an Open Source project.

Novu’s active discord community was supportive when we had queries regarding implementation. We were also able to add our contributions to their Go lang SDK.

We’ve also used Novu’s front-end react client to show notifications within the Argonaut UI. It was easy to use and configure according to our brand colors and design, the configuration of which took under 1 hour.

Upcoming features

  • External event integrations where we would add a Webhook processor to receive notifications from external services and third-party tools.
  • Adding a priority queue so that high-priority issues like pod failures and cost escalations are shared with the user instantly.
  • Notifications for more events catering to different types of users. Such as infra updates, cost, pod utilization, billing, etc.

Conclusion

In summary, adding Notifications to Argonaut has greatly improved the user experience by giving quick updates on build and deploy actions. Using Novu, we've created a smooth and personalized notification system with options for updates through channels like Slack, Teams, Discord, or email.

Our implementation includes the Process layer, Transform layer, and Fan out layer, making sure each notification is useful, prompt, and correct. Working with Novu has led to a more cost-effective solution. We're sure our users will appreciate this new feature, and we're eager to keep enhancing and growing our offerings to meet the various needs of our users

Top comments (0)