Skip to content

Latest commit

 

History

History
268 lines (187 loc) · 7.89 KB

README.md

File metadata and controls

268 lines (187 loc) · 7.89 KB

MERN-Cljs

What is it?

For now, this repository serves to share the knowledge on how to achieve above and I don't intend to release a library or framework :)

Motivation

I wanted to build web application very quickly. I find predominantly functional language such as Clojure to be a very powerful choice for speeding up the application development. Meanwhile, I wanted to leverage the development by the tools other people built and tested well already. So I chose to combine ClojureScript and node.js. This way, I can take advantage of very rich set of JavaScript libraries managed by npm. I don't have to switch language (e.g. Python <-> JavaScript) to develop backend and frontend.

An example web app

Selquote: Highlight a web page and click the bookmarklet to create a beautiful quote text image. The async worker generate image as API server put the tasks to AMQP.

What you need to install

  1. Node.js
  2. MongoDB (or DynamoDB)
  3. Leiningen

How to run locally

Run MongoDB

brew services start mongodb

Run the example API server

cd example/api
lein compile
lein run

Run the web app

cd example/www
lein compile
lein run

Point the browser to http://localhost:1337/me (API server is served on port 5000 by default.)

About the source code

This section shows where you should look at when you want to extend/develop your application.

example/common

It contains configurations and object models:

Configuration file

example/common/src/common/config.cljs example/common/src/common/frontend_config.cljs

Object models

This web stack uses Mongoose for object modeling (Vogels when used with DyanmoDB). See their docs for the Schema and Model generation.

In the example app, you can find the user schema at example/common/src/common/models. Also check out example/common/src/common/model.cljs.

example/api

API app.

  • example/api/src/api/core.cljs: Logic to start Express.js.
  • example/api/src/api/handlers.cljs: Handlers for each API endpoint.

example/www

WWW app. It contains backend and frontend code.

Backend

  • example/www/src/backend/server/core.cljs: Logic to start Express.js.
  • example/www/src/backend/server/handlers.cljs: Handlers for each web page. Triggers server side rendering.
  • example/www/src/backend/server/views.cljs: Static view template with kioo.

Frontend

  • example/www/src/frontend/app/core.cljs: Frontend logic. Trace from Ln 73 (def route) to find how profile is pulled from API server.
  • example/www/src/frontend/app/atom.cljs: Application state.
  • example/www/src/frontend/app/views.cljs: Dynamic view implementation with kioo.

HTML templates

  • example/resources/public

example/worker

Worker instance that executes async tasks. This is turned off by default since it requires RabbitMQ. See Async tasks section to find how to turn it on.

  • example/worker/src/worker/core.cljs: Core async task logic and example task ("log" function)

Authentication

Social Login

The example app is set to login via Facebook account. For your convenience, an Facebook application is made for demonstration purpose. The demo application's FACEBOOK-CLIENT-ID and FACEBOOK-CLIENT-SECRET are set in example/common/src/common/config.cljs. When you create your own application, you will need to set up your own application. Easy Node Authentication: Facebook has a good explanation on how to set up for node.js applications.

API Authentication process

Frontend app tries to fetch me resource. If it fails, it will redirect to Social auth. WWW backend authenticates via Passport.js and write information on MongoDB with a one time API token. Token is passed on to the frontend and it is used to API auth. In the example app, it will be redirected to me page to fetch me resource with proper authentication.

Logging

Bunyan is used for logging. Please refer to the doc.

Async tasks

If you run async task, install and run RabbitMQ first:

brew install rabbitmq
brew services start rabbitmq

Then set USE-RABBITMQ to true in example/common/src/common/config.cljs:

Recompile and run api instance:

cd example/api
lein compile
lein run

Start worker instance:

cd example/worker
lein compile
lein run

Activate RabbitMQ management dashboard plugin:

rabbitmq-plugins enable rabbitmq_management

Point your browser to http://localhost:15672/#/connections to see the connections. There should be one from api instance and one from worker.

Point your browser to http://localhost:5000/task to trigger a trivial task. Check the console of worker instance to see if the task was received. This should look like:

 [*] Waiting for messages. To exit press CTRL+C
 [x] Received message hello

When http://localhost:5000/task is requested (GET), the api server code (handle_task function in example/api/src/api/handlers.cljs) send the message to RabbitMQ. The worker app takes the message to execute the log function in example/worker/src/worker/core.cljs.

The log function is not a good example of async task. The idea is to avoid API server to process time consuming tasks. In production, we can deploy multiple worker instances to do pararell task execution.

DynamoDB

You can use DynamoDB instead of MongoDB.

To test locally, install and start DynamoDB local.

Then edit example/common/src/common/config.cljs:

; mongodb or dynamodb
(def DATABASE "mongodb")
; (def DATABASE "dynamodb")

DynamoDB local's default port is 8000. The config variable DYNAMODB-PORT is also set as 8000 in the example app.

Recompile and start www and api (and worker) apps.

Deployment to AWS

The production build of an application based on MERN-cljs has been deployed on Docker containers on EC2 instances. The application uses DyanmoDB on AWS and CloudAMQP for the database and RabbitMQ.

Compiling production code

You should compile ClojureScript with prod profile when building the production version. For example,

cd example/www
lein with-profile +prod compile

Don't forget + before prod.

The compilation takes much longer than building development version, but the code is optimized. Front-end code does not even run on development version on Safari browser due to the number of dependencies that needs to be loaded. The production version bundles all the dependencies and minifies. It loads and runs much faster.

Docker files

I put these example Docker files. They do not work as they are and you need to set up proper path and environmental variables, but will give you the idea how to set up. Please refer to Docker document for how to use them.

  • example/Dockerfile: Build a Docker image that has node.js, npm packages, and lein. Dockerfiles under each component picks up from the above Docker image and set up the image for each container:
    • example/api/Dockerfile
    • example/www/Dockerfile
    • example/worker/Dockerfile

If you want to run all the component in one EC2 instance, Docker Compose may be handy and here is the yml file:

  • example/docker-compose.yml