Skip to content

Latest commit

 

History

History
478 lines (381 loc) · 18.9 KB

File metadata and controls

478 lines (381 loc) · 18.9 KB

Step 4: Realtime Meteor Server

Now that we have the initial chats layout and its component, we will take it a step further by providing the chats data from a server instead of having it locally. In this step we will be implementing the API server and we will do so using Meteor.

First make sure that you have Meteor installed. If not, install it by typing the following command:

$ curl https://install.meteor.com/ | sh

We will start by creating the Meteor project which will be placed inside the api dir:

$ meteor create api
$ rm -rf api/node_modules

A Meteor project will contain the following dirs by default:

  • client - A dir containing all client scripts.
  • server - A dir containing all server scripts.

These scripts should be loaded automatically by their alphabetic order on their belonging platform, e.g. a script defined under the client dir should be loaded by Meteor only on the client. A script defined in neither of these folders should be loaded on both. Since we're using Ionic's CLI for the client code we have no need in the client dir in the Meteor project. Let's get rid of it:

api$ rm -rf client

Since we don't wanna have duplicate resources between the client and the server, we will remove the package.json file in the api dir:

api$ rm package.json

And we will add a symbolic link between Ionic's node_modules and Meteor's node_modules:

api$ ln -s ../node_modules

Since we will be writing our app using Typescript, we will need to support it in our Meteor project as well, especially when the client and the server share some of the script files. To add this support we will add the following package to our Meteor project:

api$ meteor add barbatus:typescript

We will also need to add a configuration file for the TypeScript compiler in the Meteor server, which is derived from our Ionic app's config:

Added api/tsconfig.json
@@ -0,0 +1,29 @@
+┊  ┊ 1┊{
+┊  ┊ 2┊  "compilerOptions": {
+┊  ┊ 3┊    "allowSyntheticDefaultImports": true,
+┊  ┊ 4┊    "declaration": false,
+┊  ┊ 5┊    "emitDecoratorMetadata": true,
+┊  ┊ 6┊    "experimentalDecorators": true,
+┊  ┊ 7┊    "lib": [
+┊  ┊ 8┊      "dom",
+┊  ┊ 9┊      "es2015"
+┊  ┊10┊    ],
+┊  ┊11┊    "module": "commonjs",
+┊  ┊12┊    "moduleResolution": "node",
+┊  ┊13┊    "sourceMap": true,
+┊  ┊14┊    "target": "es5",
+┊  ┊15┊    "skipLibCheck": true,
+┊  ┊16┊    "stripInternal": true,
+┊  ┊17┊    "noImplicitAny": false,
+┊  ┊18┊    "types": [
+┊  ┊19┊      "meteor-typings"
+┊  ┊20┊    ]
+┊  ┊21┊  },
+┊  ┊22┊  "exclude": [
+┊  ┊23┊    "node_modules"
+┊  ┊24┊  ],
+┊  ┊25┊  "compileOnSave": false,
+┊  ┊26┊  "atom": {
+┊  ┊27┊    "rewriteTsconfig": false
+┊  ┊28┊  }
+┊  ┊29┊}

Since we're using TypeScript, let's change the main server file extension from .js to .ts:

api$ mv server/main.js server/main.ts

Now we will need to create a symbolic link to the declaration file located in src/declarations.d.ts. This way we can share external TypeScript declarations in both client and server. To create the desired symbolic link, simply type the following command in the command line:

api$ ln -s ../src/declarations.d.ts

The following dependencies are required to be installed so our server can function properly:

$ npm install --save babel-runtime
$ npm install --save meteor-node-stubs

Now we'll have to move our models interfaces to the api dir so the server will have access to them as well:

$ mv src/models.ts api/server/models.ts

This requires us to update its reference in the declarations file as well:

Changed src/pages/chats/chats.ts
@@ -1,7 +1,7 @@
 ┊1┊1┊import { Component } from '@angular/core';
 ┊2┊2┊import { Observable } from 'rxjs';
 ┊3┊3┊import * as moment from 'moment';
-┊4┊ ┊import { Chat, MessageType } from '../../models';
+┊ ┊4┊import { Chat, MessageType } from 'api/models';
 ┊5┊5┊
 ┊6┊6┊@Component({
 ┊7┊7┊  templateUrl: 'chats.html'

We will also install the meteor-rxjs package so we can define collections and data streams and TypeScript definitions for Meteor:

$ npm install --save meteor-rxjs
$ npm install --save-dev @types/meteor

To avoid duplicate identifier errors in Ionic we will have to remove meteor-typings from the root's tsconfig.json types array:

Changed tsconfig.json
@@ -20,7 +20,6 @@
 ┊20┊20┊    "stripInternal": true,
 ┊21┊21┊    "noImplicitAny": false,
 ┊22┊22┊    "types": [
-┊23┊  ┊      "meteor-typings",
 ┊24┊23┊      "@types/underscore"
 ┊25┊24┊    ]
 ┊26┊25┊  },

Collections

This collection is actually a reference to a MongoDB collection, and it is provided to us by a Meteor package called Minimongo, and it shares almost the same API as a native MongoDB collection. In this tutorial we will be wrapping our collections using RxJS's Observables, which is available to us thanks to meteor-rxjs.

Our initial collections are gonna be the chats and messages collection, the one is going to store some chats-models, and the other is going to store messages-models:

Added api/server/collections/chats.ts
@@ -0,0 +1,4 @@
+┊ ┊1┊import { MongoObservable } from 'meteor-rxjs';
+┊ ┊2┊import { Chat } from '../models';
+┊ ┊3┊
+┊ ┊4┊export const Chats = new MongoObservable.Collection<Chat>('chats');
Added api/server/collections/messages.ts
@@ -0,0 +1,4 @@
+┊ ┊1┊import { MongoObservable } from 'meteor-rxjs';
+┊ ┊2┊import { Message } from '../models';
+┊ ┊3┊
+┊ ┊4┊export const Messages = new MongoObservable.Collection<Message>('messages');

We chose to create a dedicated module for each collection, because in the near future there might be more logic added into each one of them. To make importation convenient, we will export all collections from a single file:

Added api/server/collections/index.ts
@@ -0,0 +1,2 @@
+┊ ┊1┊export * from './chats';
+┊ ┊2┊export * from './messages';

Now instead of requiring each collection individually, we can just require them from the index.ts file.

Data fixtures

Since we have real collections now, and not dummy ones, we will need to fill them up with some data in case they are empty, so we can test our application properly. Let's create our data fixtures in the server:

Changed api/server/main.ts
@@ -1,5 +1,71 @@
 ┊ 1┊ 1┊import { Meteor } from 'meteor/meteor';
+┊  ┊ 2┊import { Chats } from './collections/chats';
+┊  ┊ 3┊import { Messages } from './collections/messages';
+┊  ┊ 4┊import * as moment from 'moment';
+┊  ┊ 5┊import { MessageType } from './models';
 ┊ 2┊ 6┊
 ┊ 3┊ 7┊Meteor.startup(() => {
-┊ 4┊  ┊  // code to run on server at startup
+┊  ┊ 8┊  if (Chats.find({}).cursor.count() === 0) {
+┊  ┊ 9┊    let chatId;
+┊  ┊10┊
+┊  ┊11┊    chatId = Chats.collection.insert({
+┊  ┊12┊      title: 'Ethan Gonzalez',
+┊  ┊13┊      picture: 'https://randomuser.me/api/portraits/thumb/men/1.jpg'
+┊  ┊14┊    });
+┊  ┊15┊
+┊  ┊16┊    Messages.collection.insert({
+┊  ┊17┊      chatId: chatId,
+┊  ┊18┊      content: 'You on your way?',
+┊  ┊19┊      createdAt: moment().subtract(1, 'hours').toDate(),
+┊  ┊20┊      type: MessageType.TEXT
+┊  ┊21┊    });
+┊  ┊22┊
+┊  ┊23┊    chatId = Chats.collection.insert({
+┊  ┊24┊      title: 'Bryan Wallace',
+┊  ┊25┊      picture: 'https://randomuser.me/api/portraits/thumb/lego/1.jpg'
+┊  ┊26┊    });
+┊  ┊27┊
+┊  ┊28┊    Messages.collection.insert({
+┊  ┊29┊      chatId: chatId,
+┊  ┊30┊      content: 'Hey, it\'s me',
+┊  ┊31┊      createdAt: moment().subtract(2, 'hours').toDate(),
+┊  ┊32┊      type: MessageType.TEXT
+┊  ┊33┊    });
+┊  ┊34┊
+┊  ┊35┊    chatId = Chats.collection.insert({
+┊  ┊36┊      title: 'Avery Stewart',
+┊  ┊37┊      picture: 'https://randomuser.me/api/portraits/thumb/women/1.jpg'
+┊  ┊38┊    });
+┊  ┊39┊
+┊  ┊40┊    Messages.collection.insert({
+┊  ┊41┊      chatId: chatId,
+┊  ┊42┊      content: 'I should buy a boat',
+┊  ┊43┊      createdAt: moment().subtract(1, 'days').toDate(),
+┊  ┊44┊      type: MessageType.TEXT
+┊  ┊45┊    });
+┊  ┊46┊
+┊  ┊47┊    chatId = Chats.collection.insert({
+┊  ┊48┊      title: 'Katie Peterson',
+┊  ┊49┊      picture: 'https://randomuser.me/api/portraits/thumb/women/2.jpg'
+┊  ┊50┊    });
+┊  ┊51┊
+┊  ┊52┊    Messages.collection.insert({
+┊  ┊53┊      chatId: chatId,
+┊  ┊54┊      content: 'Look at my mukluks!',
+┊  ┊55┊      createdAt: moment().subtract(4, 'days').toDate(),
+┊  ┊56┊      type: MessageType.TEXT
+┊  ┊57┊    });
+┊  ┊58┊
+┊  ┊59┊    chatId = Chats.collection.insert({
+┊  ┊60┊      title: 'Ray Edwards',
+┊  ┊61┊      picture: 'https://randomuser.me/api/portraits/thumb/men/2.jpg'
+┊  ┊62┊    });
+┊  ┊63┊
+┊  ┊64┊    Messages.collection.insert({
+┊  ┊65┊      chatId: chatId,
+┊  ┊66┊      content: 'This is wicked good ice cream.',
+┊  ┊67┊      createdAt: moment().subtract(2, 'weeks').toDate(),
+┊  ┊68┊      type: MessageType.TEXT
+┊  ┊69┊    });
+┊  ┊70┊  }
 ┊ 5┊71┊});

This behavior is not recommended and should be removed once we're ready for production. A conditioned environment variable may also be a great solution

Note how we use the .collection property to get the actual Mongo.Collection instance. In the Meteor server we want to avoid the usage of observables since it uses fibers. More information about fibers can be found here.

Preparing the Meteor client

In order to connect to the Meteor server, we need a client which is capable of doing so. To create a Meteor client, we will use a bundler called meteor-client-bundler. This bundler, bundles all the necessary Meteor client script files into a single module. This is very useful when we want to interact with Atmosphere packages and integrate them into our client. To install meteor-client-bundler, run the following command:

$ sudo npm install -g meteor-client-bundler

Now we'll add a bundling script to the package.json as followed:

Changed package.json
@@ -13,7 +13,8 @@
 ┊13┊13┊    "build": "ionic-app-scripts build",
 ┊14┊14┊    "lint": "ionic-app-scripts lint",
 ┊15┊15┊    "ionic:build": "ionic-app-scripts build",
-┊16┊  ┊    "ionic:serve": "ionic-app-scripts serve"
+┊  ┊16┊    "ionic:serve": "ionic-app-scripts serve",
+┊  ┊17┊    "meteor-client:bundle": "meteor-client bundle -s api"
 ┊17┊18┊  },
 ┊18┊19┊  "dependencies": {
 ┊19┊20┊    "@angular/common": "4.1.2",

We will also need to install the tmp package:

$ sudo npm install --save-dev tmp

To execute the bundler, simply run:

$ npm run meteor-client:bundle

This will generate a file called meteor-client.js under the node_modules dir, which needs to be imported in our application like so:

Changed src/app/main.ts
@@ -1,3 +1,5 @@
+┊ ┊1┊import 'meteor-client';
+┊ ┊2┊
 ┊1┊3┊import { platformBrowserDynamic } from '@angular/platform-browser-dynamic';
 ┊2┊4┊
 ┊3┊5┊import { AppModule } from './app.module';

By default, the client will assume that the server is running at localhost:3000. If you'd like to change that, you can simply specify a --url option in the NPM script. Further information can be found here.

The client we've just imported gives us the ability to interact with the server. Let's replace the local chats-data with a data which is fetched from the Meteor server:

Changed src/pages/chats/chats.ts
@@ -1,71 +1,33 @@
-┊ 1┊  ┊import { Component } from '@angular/core';
+┊  ┊ 1┊import { Component, OnInit } from '@angular/core';
+┊  ┊ 2┊import { Chats, Messages } from 'api/collections';
+┊  ┊ 3┊import { Chat } from 'api/models';
 ┊ 2┊ 4┊import { Observable } from 'rxjs';
-┊ 3┊  ┊import * as moment from 'moment';
-┊ 4┊  ┊import { Chat, MessageType } from 'api/models';
 ┊ 5┊ 5┊
 ┊ 6┊ 6┊@Component({
 ┊ 7┊ 7┊  templateUrl: 'chats.html'
 ┊ 8┊ 8┊})
-┊ 9┊  ┊export class ChatsPage {
-┊10┊  ┊  chats: Observable<Chat[]>;
+┊  ┊ 9┊export class ChatsPage implements OnInit {
+┊  ┊10┊  chats;
 ┊11┊11┊
 ┊12┊12┊  constructor() {
-┊13┊  ┊    this.chats = this.findChats();
 ┊14┊13┊  }
 ┊15┊14┊
-┊16┊  ┊  private findChats(): Observable<Chat[]> {
-┊17┊  ┊    return Observable.of([
-┊18┊  ┊      {
-┊19┊  ┊        _id: '0',
-┊20┊  ┊        title: 'Ethan Gonzalez',
-┊21┊  ┊        picture: 'https://randomuser.me/api/portraits/thumb/men/1.jpg',
-┊22┊  ┊        lastMessage: {
-┊23┊  ┊          content: 'You on your way?',
-┊24┊  ┊          createdAt: moment().subtract(1, 'hours').toDate(),
-┊25┊  ┊          type: MessageType.TEXT
-┊26┊  ┊        }
-┊27┊  ┊      },
-┊28┊  ┊      {
-┊29┊  ┊        _id: '1',
-┊30┊  ┊        title: 'Bryan Wallace',
-┊31┊  ┊        picture: 'https://randomuser.me/api/portraits/thumb/lego/1.jpg',
-┊32┊  ┊        lastMessage: {
-┊33┊  ┊          content: 'Hey, it\'s me',
-┊34┊  ┊          createdAt: moment().subtract(2, 'hours').toDate(),
-┊35┊  ┊          type: MessageType.TEXT
-┊36┊  ┊        }
-┊37┊  ┊      },
-┊38┊  ┊      {
-┊39┊  ┊        _id: '2',
-┊40┊  ┊        title: 'Avery Stewart',
-┊41┊  ┊        picture: 'https://randomuser.me/api/portraits/thumb/women/1.jpg',
-┊42┊  ┊        lastMessage: {
-┊43┊  ┊          content: 'I should buy a boat',
-┊44┊  ┊          createdAt: moment().subtract(1, 'days').toDate(),
-┊45┊  ┊          type: MessageType.TEXT
-┊46┊  ┊        }
-┊47┊  ┊      },
-┊48┊  ┊      {
-┊49┊  ┊        _id: '3',
-┊50┊  ┊        title: 'Katie Peterson',
-┊51┊  ┊        picture: 'https://randomuser.me/api/portraits/thumb/women/2.jpg',
-┊52┊  ┊        lastMessage: {
-┊53┊  ┊          content: 'Look at my mukluks!',
-┊54┊  ┊          createdAt: moment().subtract(4, 'days').toDate(),
-┊55┊  ┊          type: MessageType.TEXT
-┊56┊  ┊        }
-┊57┊  ┊      },
-┊58┊  ┊      {
-┊59┊  ┊        _id: '4',
-┊60┊  ┊        title: 'Ray Edwards',
-┊61┊  ┊        picture: 'https://randomuser.me/api/portraits/thumb/men/2.jpg',
-┊62┊  ┊        lastMessage: {
-┊63┊  ┊          content: 'This is wicked good ice cream.',
-┊64┊  ┊          createdAt: moment().subtract(2, 'weeks').toDate(),
-┊65┊  ┊          type: MessageType.TEXT
-┊66┊  ┊        }
-┊67┊  ┊      }
-┊68┊  ┊    ]);
+┊  ┊15┊  ngOnInit() {
+┊  ┊16┊    this.chats = Chats
+┊  ┊17┊      .find({})
+┊  ┊18┊      .mergeMap((chats: Chat[]) =>
+┊  ┊19┊        Observable.combineLatest(
+┊  ┊20┊          ...chats.map((chat: Chat) =>
+┊  ┊21┊            Messages
+┊  ┊22┊              .find({chatId: chat._id})
+┊  ┊23┊              .startWith(null)
+┊  ┊24┊              .map(messages => {
+┊  ┊25┊                if (messages) chat.lastMessage = messages[0];
+┊  ┊26┊                return chat;
+┊  ┊27┊              })
+┊  ┊28┊          )
+┊  ┊29┊        )
+┊  ┊30┊      ).zone();
 ┊69┊31┊  }
 ┊70┊32┊
 ┊71┊33┊  removeChat(chat: Chat): void {

And re-implement the removeChat method using the actual Meteor collection:

Changed src/pages/chats/chats.ts
@@ -31,13 +31,7 @@
 ┊31┊31┊  }
 ┊32┊32┊
 ┊33┊33┊  removeChat(chat: Chat): void {
-┊34┊  ┊    this.chats = this.chats.map((chatsArray: Chat[]) => {
-┊35┊  ┊      const chatIndex = chatsArray.indexOf(chat);
-┊36┊  ┊      if (chatIndex !== -1) {
-┊37┊  ┊        chatsArray.splice(chatIndex, 1);
-┊38┊  ┊      }
-┊39┊  ┊
-┊40┊  ┊      return chatsArray;
+┊  ┊34┊    Chats.remove({_id: chat._id}).subscribe(() => {
 ┊41┊35┊    });
 ┊42┊36┊  }
 ┊43┊37┊}
< Previous Step Next Step >