In this tutorial, we are going to build a basic chat web application using Pusher and Angular, a popular Javascript framework.
What we are going to be building
Using our application, users will be able to view and send messages on a private channel.
To achieve this, we are going to be using Pusher’s API for sending and receiving messages in realtime. So make sure you have your Pusher account at hand, or sign up for free here. We will also write a tiny server-side Node app that will be used by Pusher to authenticate channel subscriptions received from our Angular front end. Let’s look at how these three parts (Pusher, Server, Client) fit together.
Project structure
You can find the full code used in this post in this Github repo
We are going to keep our server-side and client-side apps separate in their respective directories.
Building a Node app
First, let’s get the server app out of the way by creating its directory and initializing npm.
This will create a package.json file for our app. Now let’s install our dependencies.
Note that we are installing the pusher Node library as a dependency.
It’s also a good practice to define a npm start script. In our package.json file, let’s replace the default scripts property with the following.
This maps the command node server.js to an npm start command. However, server.js doesn’t exist as a file yet, so let’s create it next.
The calls to our endpoint will be coming in from a different origin, therefore we need to make sure we include the CORS headers (Access-Control-Allow-Origin). If you are unfamiliar with the concept of CORS headers, you can find more information here.
This all is standard Node application configuration, nothing specific to our app.
API Routes
In our server.js file we are referencing a non-existent file api.js on line 5. That’s the file where we will define our API routes and the logic behind them.
We’ve defined one route for our app: /, that returns an all goodresponse. Now, let’s take a step back and define what we want our API to do based on our requirements.
Pusher Channel Authentication
Considering that we will be triggering Pusher events directly from our client, we need to implement an authentication endpoint. This endpoint will be called by Pusher directly in order to authenticate any subscription it receives from our front end.
Pusher will be making a POST request to an endpoint that we will provide. Therefore, let’s create a POST /pusher/auth endpoint.
Whenever Pusher calls the authentication endpoint, it sends a socket_id and channel_name that we will use to authenticate the incoming subscription.
Next step is to authenticate the subscription using the extracted socketId and channel values from the request. Since we will be telling Pusher to authenticate the subscription, we need to initialize Pusher by instantiating it with our Pusher account credentials first. Let’s do that at the top of our api.js file.
You can find these values by navigating to a specific Pusher application from your personal Pusher dashboard.
We now have an instance of Pusher available to us that we can use for authenticating subscriptions.
That is all the logic we need to write for our Node app. All that is left now is to run it so that it’s ready to start serving requests. We start our app by running npm start.
Building an Angular web app
As you remember, the client app’s responsibility is to send and receive new messages from Pusher in real time. Let’s get going.
Angular app initialization
We are going to use Angular CLI to quickly scaffold our application. You can follow the installation instructions if you don’t have Angular CLI installed already.
After you’ve installed Angular CLI, it’s time to initialize our Angular application. Navigate to the chat-app directory and run the following command to scaffold an application.
After the installation is finished we need to install Pusher’s client library.
Installing the library isn’t enough, as we also want to include it on our page. In order to do that we need to add the library to third party scripts to be loaded by Angular CLI. All CLI config is stored in .angular-cli.json file. Modify the scripts property to include the link to pusher.min.js.
We are now ready to write some more code. However, let’s take a step back and figure out the application structure.
Angular application structure
Nothing is stopping us from writing the whole app inside the AppComponent, however, if we want to follow best practices (and we do), then we need to avoid that. AppComponent is best left for any startup initialization we need to do for our app.
Considering that our app is basic, I can only see one other component we can create – MessagesComponent that will be in charge of displaying existing messages, as well as collecting user input and sending new messages.
There will also be one service that we’ll need to create – a PusherService that deals with Pusher.
With CLI, we can easily generate components and services. In our instance, we want to run the following commands.
Note: g is an alias for a generate command, c is an alias for component and s is an alias for a service
We now have all our components in place. Even though our service has been created, it hasn’t been provided. Therefore we need to manually add it as a provider to the correct application module. However, because we only have one module, AppModule, we can only provide it in there.
Pusher initialization
First, let’s initialize Pusher. When we were initializing Pusher on the server, we needed three properties: appId, key and secret, however, we only need the key on the client.
In a real world application, you are likely to use different Pusher keys depending on the environment you are in (like development or production), therefore it is a good idea to store our Pusher key as a property on the environment constant.
Angular CLI creates an environment.ts file that is used to store environment-specific variables. Let’s store our Pusher key there.
Now we can use the environment variable in our PusherService when initializing an instance of Pusher. Upon initialization, we will store it in a property on the service that any other component can use.
At this point, Typescript will complain about our new Pusher(..) expression, because we haven’t imported Pusher. However, we don’t need to import it, as it exists in the Window object in the browser because we’re including the pusher-js library in our index.html. Therefore, to silence the Typescript compiler, we need to declare Pusher at the top of the file along with other imports.
In addition to the application key, we also need to instantiate Pusher with our authentication endpoint that we created with Node.
localhost:3000 is where our Node application is running.
Now we can have access to our pusher instance through the PusherService class.
Pusher works by publishing events to a specific channel. Whoever is subscribed to that channel will receive the published event. Events can have data associated with them. Our client will be receiving and triggering events to a predefined channel. On top of storing the Pusher instance as a property on the PusherService, let’s also store the channel that we will be receiving and triggering messages on.
It is at this point, during the subscription to a channel that Pusher sends an authentication request to our specified endpoint.
In the snippet above, private-messages is the name of our channel. Triggering events on the front end only works with private or presence channels, therefore we have to prefix our channel’s name with private.
MessagesComponent
Our Angular project was initialized with routing because we passed a --routing flag with the initialization command. However, we need to tell Angular to use our MessagesComponent for the default route. Routing is configured in app-routing.module.ts.
Now that MessagesComponent loads for our default route, let’s work on it.
One of this component’s responsibilities is to display existing messages. We could achieve that with an array of messages that we are looping through in the view and display them.
Note: one of the most powerful Typescript features is (obviously) types. They are there to help you, so make sure you use them. Like in the snippet above, I created an interface for an individual message object. As a result, I can specify that messages property is an array of Message and if I try putting anything else in it – Typescript will not allow me to.
We can now loop over the messages array to display individual messages.
Because the messages property is initialized to an empty array, nothing will be displayed. Now that we have a way of storing all messages, let’s write the logic to populate our storage.
Realtime messages from Pusher
New messages are going to be coming in via Pusher. We have the channel that will be transmitting messages and it’s stored in the PusherService. So what we want to do is start listening for events on that channel and handle whenever an event is received.
Because we want to start listening for events on the channel as soon as the MessagesComponent is initialized, we need to use Angular component lifecycle hooks. Specifically, we are interested in the OnInit lifecycle hook, as it is triggered as soon as the component initialization is finished.
Before we use it, we need to let Angular know that we want to be using this hook by specifying that our component implements the OnInit interface.
ngOnInit is the method that will be run when the OnInit lifecycle hook is triggered by Angular. That makes it a perfect place to run any sort of initialization logic, in this case, start listening for events. For that, we need to get access to the messagesChannel property on the PusherService
Before we can use PusherService inside our component, we need to inject it as a dependency of this component.
Now we can start listening for events on the messagesChannel property of PusherService.
In the snippet above, we are listening for client-new-message events on the messages channel. The second parameter of the bind is the callback function. It is a function that will be called whenever an event is received with the event data. In our case, the data associated with the event will be the message, therefore we are pushing the new message into our array of messages.
Triggering Pusher events
In the MessagesComponent we need to collect the user’s username and message text in order to send it as a new event to Pusher. First, let’s add two input fields in our HTML.
Two things to note here:
We are using Angular’s two-way binding with ngModel to record user input.
We have added a click handler on the button that calls a method sendMessage on our component (this method doesn’t exist yet, we will write it soon) with the userName and messageText values.
Because we will be are triggering events directly from the client, we need to prefix the event name with client.
There are also other restrictions that apply when triggering events from the client. Even though I will be pointing them out as we go in this article, I suggest you read the Pusher documentation on the topic to avoid confusion.
Before we can start triggering events from the client, we need to tell Pusher that we intend to do so. You can do that by navigating to your application’s settings on your Pusher dashboard.
Inside our sendMessage method, we will want to trigger an event named client-new-message with the message assembled from this method’s arguments. On top of that, we need to add this message to our array of messages.
Finally, we can send messages and display realtime messages! <img src="https://s.w.org/images/core/emoji/2/72x72/1f389.png" alt="