Originally posted at Serverless on September 12th, 2017
In my previous post on Anatomy of a Serverless Application, I lay the foundation for building a very simple application with an email service using the Serverless Framework, deployed to AWS Lambda.
In this post, we will build mailman, an event-driven serverless application. The application has a simple frontend using curl that calls into a couple of backend services: a users service and an email service. The post will highlight event-driven application development with focus on full local development experience. We will look at services emitting, subscribing and reacting to events in a seamless manner using the Serverless Application Platform.
You will learn how to:
Setup the development environment
Create an application project
Create a users service
Create an email service
Write an event-driven application
Run the serverless services locally
Run the full application locally
Getting Started
We will look at an important aspect of writing serverless applications, i.e. local development support. We touched upon the fact that Serverless Framework helps with local testing of a serverless service by using sls invoke. But, the real productivity gain is the ability to write a serverless, event-driven application with full local development support, with provider emulation and running it without the need to deploy it to a cloud provider.
My choices:
- Programming language: NodeJS
2. The Serverless Application Platform: It consists of:
Serverless Framework (v1.20.0 or higher)
Serverless Event Gateway, the central hub of event communication
Serverless Emulator, the local serverless provider emulator
Serverless Functions Development Kit (FDK), to enhance developer experience for writing severless applications
Setup
Let’s install and setup the toolsets required for development.
Install NodeJS (v6.10.3): download or using package manager
Install the Serverless Framework (v1.20.0 or higher): npm install -g serverless
-
No need for setting up any cloud provider account. YEAH!
Without the step (read hurdle) of setting up a cloud provider account you have already saved so much time and hassle.
Note: You might wonder what happened to the Event Gateway and the Emulator setup. That’s where the Serverless Framework makes it all come together. Keep reading.
With Serverless Framework already installed, we will see how to use the Event Gateway and the Emulator in order to provide a centralized event hub and a much needed local development environment. The newly added serverless run command downloads and runs the necessary components.
Before we use the serverless run command, we need to first login to the Serverless Application Platform using the serverless login command:
Event-driven Serverless Application
Let’s look at three core capabilities that we need to write, test and run a serverless event-driven application: an event-driven workflow, programmatic access, and full local development experience.
Event-driven Workflow
The tenet of an event-driven application is that the components interact with each other asynchronously via events. The components are not aware of each other and rely on a central event communication hub for relaying events across the application.
The Event Gateway serves as the central event communication fabric for serverless applications. It acts as the broker for all event communication and allows services to publish and subscribe to events. Additionally, it acts as an API Gateway for all HTTP communication.
The Event Gateway uses a special http event-type to recognize standard HTTP requests. It standardizes both pub/sub and HTTP communication to be represented as events, combining them into a single experience. Services can send custom events and the Event Gateway wraps them up in a standard event envelope passing the payload along as-is. Events from well-known SaaS providers will be recognized as first-class event-types in the near future. "Everything as events" is the mantra.
Read Event Gateway — The Missing Piece of Serverless Architectures for more details.
Programmatic Access
To write event-driven serverless applications, its individual services need to programmatically take advantage of all the goodness of the Event Gateway features. The Serverless SDK hands that capability to the developers. The developers have access to the Event Gateway API via code for registering & invoking functions, subscribing to and emitting events, and configuration.
Checkout the SDK source for more details.
Local Development Support
One of the biggest challenges with developing serverless applications is to run and test the application locally. Since serverless applications are hosted on the cloud, it makes it very tedious to debug, test and develop code in an iterative manner. Although all cloud providers and the Serverless Framework allows invoking one function at a time locally, the developer wishes to run the full application locally on their machine.
The Serverless Emulator emulates different cloud provider FaaS offerings (currently AWS Lambda or Google Cloud Functions) on your local machine in an offline-focused manner. It provides that missing piece of tooling that makes application development with serverless so productive and exciting. It enables deploying and invoking serverless functions without the requirement of setting up and deploying your serverless application to the cloud provider.
Check out the Emulator source for more details.
Seamless Developer Experience
To empower the developer with a great development experience, the Serverless Framework brings it all together in a very simple and intuitive interface: the serverless run command.
The serverless run command detects and installs or spins up the latest version of the Event Gateway and the Emulator. It provides a unified visual interface for the event-driven workflow for the application as they happen. It provides an intuitive and decoupled interface for the services of an application to communicate with.
In the sections that follow, we will use the example application to demonstrate these killer features.
Creating the Application
A basic project structure shown below will work:
where,
frontend folder holds the frontend
-
services folder holds the serverless service(s)
With sls invoke you could test one single function at a time. But with the Serverless Application Platform, you can run the full event-driven serverless application, with events interacting across the application, locally on your machine, without signup up for any cloud provider. YEAH!
Let’s create a couple of the backend services and then we will explore the local experience when we run the application. The way the non-serverless frontend portion of the application is written is totally your choice. In this post, I will primarily focus on the serverless portion of the application inside the services folder, and use a basic curl script to simulate the frontend.
Goals
The application has a few business goals:
register a user via an HTTP POST API call from the frontend
on registration, send a welcome email to the user
The application has a few technical goals as well:
run and test the application locally without deploying to the cloud
have loosely-coupled services interact with the application
visualize the event-driven workflow as it happens
Building the Services
We will not get into the details of creating the services, but we will explore the code and dig deep into the functionality that is offered. Also, the guts of the services are mocked up, but the intention of the example is to showcase the event-driven nature of the services, so we will just focus on that.
Building the Users Service
The workflow of the users service is:
expose an HTTP endpoint for user registration,
register a user, and finally
emit an event user.registered
Let’s look at the functions section of the serverless.yml:
Nothing different than what you are used to.
The important aspect to note is that the Serverless Application Platform with all its new products and enhancements is still compatible with existing serverless applications that you have built.
Behind the scenes
Actually, there are a couple of things that the Serverless Application Platform does behind the scenes. The serverless.yml is parsed, and the register function from the users service is registered with the Event Gateway. Next, a subscription is created for the http event and the register function.
We will visualize this when we run the service and talk about it in the next section.
Now, let’s look at the handler code:
In this portion of the code, we use the Serverless FDK to initialize the Event Gateway.
This portion shows the register function, that first checks if an email was passed in the event body, and then registers the user. Next, on successful registration of the user, it emits an event user.registered using the FDK, passing in the user data.
Building the Email Service
The workflow of the email service is:
send a welcome email to the registered user, and
emit an event email.sent
Let’s look at the functions section of the serverless.yml:
Behind the scenes
There are a couple of things that the Serverless Application Platform does behind the scenes. The serverless.yml is parsed, and the sendWelcomeEmail function from the email service is registered with the Event Gateway. Next, a subscription is created for the user.registered event and the sendWelcomeEmail function.
We will visualize this when we run the service and talk about it in the next section.
Now, let’s look at the handler code, specifically the sendWelcomeEmail function:
This portion shows the sendWelcomeEmail function, that would send out a welcome email to the user that is passed in via the event data. Next, it emits an event email.sent using the FDK, passing in the event data.
Running the Application
Now that we understand what the application is meant to do, and having looked at the code, let’s run the application. Everything will run locally on our machine. We will run one service at a time, emit and respond to events, visualize the workflow and then call the services via curl simulating an application user interface.
Two new awesome commands: serverless run and serverless emit implemented as core plugins to the Serverless Framework were released recently.
The Local Development Experience
The first time you run any service using serverless run, you will get a slightly different experience. Let's run the users service, and look at the visualization on the terminal:
The required components of the Serverless Application Platform, namely the Event Gateway and the Emulator, are downloaded and installed locally on your machine unless already installed. Also, note that the individual components advertise their API endpoints. Then, these components stay in the background listening for events and ready for action.
I want to point out that this is how the local development experience starts for a developer. It is all inclusive, seamless and is just there for you.
Running the Users Service
With the local development setup done, we are ready to run our users service.
There are a couple of things that are happening here:
Here the Serverless Framework detects the register function in the users service and deploys it to the local Emulator.
Then the register function from the users service is registered with the Event Gateway. And, a subscription is created for the http event and the register function.
Running the Email Service
Let’s run the email service now.
There are a couple of things that are happening here:
Here the Serverless Framework detects the sendWelcomeEmail function in the email service and deploys it to the local Emulator.
Then the sendWelcomeEmail function from the email service is registered with the Event Gateway. Further, a subscription is created for the user.registered event and the sendWelcomeEmail function.
Calling the Services
To simulate an application user interface, we will use curl to call the user service API for registering a user. We have the current serverless run session running on one pane and we will run the frontend curl commands in another pane.
Run the following curl command on the other pane:
Let’s break it up and discuss what just happened.
- The http event triggered the register function from the users service. The register function emitted an user.registered event, which was received by the Event Gateway. You can see the data payload for the event as well.
- The http event triggered the register function from the users service. The register function emitted an user.registered event, which was received by the Event Gateway. You can see the data payload for the event as well.
- The user.registered event triggered the sendWelcomeEmail function from the email service. Then, the register function which had started async in Step 2, finishes.
- Then the email.sent event emitted by sendWelcomeEmail function is received by the Event Gateway. Then, the sendWelcomeEmail function which had started async in Step 3, finishes.
- The http event triggered the register function from the users service. The register function emitted an user.registered event, which was received by the Event Gateway. You can see the data payload for the event as well.
Hopefully, you followed the workflow I laid out and can see how easy it is to write an event-driven serverless application that can be tested and run locally.
The terminal visualization during development is crucial to debugging and tracing async event-driven applications, and can greatly help speed up application development cycles.
Code: Get the full source code to the mailman application project on Github.
Summary
The serverless event-driven application that we created was limited to a couple of services to keep it simple and easy to follow. We have a reference example application that showcases many other event-driven use cases utilizing a variety of services, spanning multiple cloud providers.
Originally published at https://www.serverless.com.
Top comments (0)