SlideShare une entreprise Scribd logo
CONVERTING TO NODE
How I converted a medium sized Rails project to Node.js
BACKGROUND
•

CTO of Ideal Candidate	


•

MVP existed when I joined	


•

Author of Haraka and a few
other Node.js libraries	


•

Background in Perl and
Anti-Spam
THE RAILS APP
•

Unicorn + Thin + Rails	


•

CoffeeScript with Angular.JS	


•

SASS+Compass for stylesheets	


•

HAML for templates	


•

MySQL + MongoDB	


•

Sidekiq for background tasks
WHY?
APP SIZE
•

5700 Lines of Ruby (170 files)	


•

2500 Lines of CoffeeScript (119 files)	


•

3800 Lines of SASS (42 files)	


•

1500 Lines of HAML (100 files)
FINAL GOAL
•

Express + minimal middleware	


•

PostgreSQL hand coded with referential integrity	


•

Jade templates	


•

SCSS for CSS	


•

Javascript + Angular.js frontend
HOW?
It’s all about the information, Marty
RAILS GIVES YOU
•

Routes	


•

Middleware	


•

DB abstraction layer (aka Model)	


•

Views	


•

Controllers	


•

Configuration	


•

Plugins	


•

A directory structure	


•

Unit tests
EXPRESS GIVES YOU
•

Routes	


•

Middleware	


•

That’s it	


•

(This scares a lot of people)
DIRECTORY STRUCTURE
lib
- general JS libraries	

⌞ model - database accessors	

public
- files that get served to the front end	

⌞ assets - files that are considered part of the app	

⌞ javascripts	

⌞ stylesheets	

⌞ images	

routes
- files that provide express routes for HTML	

⌞ api/v1 - files providing REST API endpoints (JSON)	

views
- Jade templates	

app.js
- entry point
RAILS ASSET PIPELINE
•

Rails does a lot for you, but it’s f ’ing confusing	


•

HAML Template Example:	

•

= javascript_include_tag :application	


•

You’d think this adds <script
src=“application.js”>, but no.
RAILS ASSET PIPELINE
•

Easy if you understand it	


•

Too much magic for my liking	


•

But… the overall effect is good. So I copied some
of it.
JADE EQUIVALENT
	

	

each script in javascripts	
script(src=script, type=“text/javascript”)
APP.JS:
var js = find.fileSync(/.(coffee|js)$/, __dirname + 	
	 ‘/public/assets/javascripts/controllers')	
	 .map(function (i) {	
	 	 return i.replace(/^.*/assets/, ‘/assets')	
	 	 	 	 .replace(/.coffee$/, ‘.js')	
	 });	
app.all('*', function (req, res, next) {	
res.locals.javascripts = js;	
next();	
});
SERVING COFFEESCRIPT
// development	
app.use(require('coffee-middleware')({	
src: __dirname + '/public',	
compress: false,	
}));	

!

// Production	
var all_js = coffee_compiler.generate(js, true);	

!

var shasum = crypto.createHash('sha256');	
shasum.update(all_js);	
var js_sha = shasum.digest('hex');	

!

js = ['/assets/javascripts/application-' + js_sha + '.js'];	

!

app.get('/assets/javascripts/application-' + js_sha + '.js', function (req, res) {	
res.type('js');	
res.setHeader('Cache-Control', 'public, max-age=31557600');	
res.end(all_js);	
});
SERVING SCSS
•

Node-sass module serves .scss files	


•

Doesn’t serve up .sass files (weird, huh?)	


!
# mass-convert.bash	
for F in `find . -name *.sass`; do	
O=“${F/.sass/.scss}”	
sass-convert -F sass -T scss “$F” “$O”	
git rm -f “$F”	
git add “$O”	
done
SCSS APP.JS
function compile_css () {	
console.log("Recompiling CSS");	
resources.watchers.forEach(function (w) {	
w.close();	
});	
	
resources.watchers = [];	

!

!
!
!
}

resources.css = sass.renderSync({	
file: __dirname + '/public/assets/stylesheets/application.scss',	
includePaths: [__dirname + '/public/assets/stylesheets'],	
});	
var shasum = crypto.createHash('sha256');	
shasum.update(resources.css);	
var css_sha = shasum.digest('hex');	
resources.stylesheets[0] = '/assets/stylesheets/application-' + css_sha + '.css';	
resources.stylesheets[1] = '//fonts.googleapis.com/css?family=Open+Sans:400,700,800,300,600';	
find.fileSync(cssregexp, __dirname + '/public/assets/stylesheets').forEach(function (f) {	
resources.watchers.push(fs.watch(f, compile_css));	
})
SCSS APP.JS

compile_css();	

!
// Dev and Production the same	
app.get(//assets/stylesheets/application-(w+).css/, function (req, res) {	
res.type('css');	
res.setHeader('Cache-Control', 'public, max-age=31557600');	
res.end(resources.css);	
});
ADDING COMPASS
•

Ruby CSS framework	


•

Luckily only the CSS3 part used	


•

CSS3 code is just SASS files	


•

Once I figured this out, copied them into my
project, et voila!
CONVERTING HAML TO JADE
•

Both indent based	


•

HAML:	


	
	
	
	
	

•
	
	
	

•

	
	
	
	
	

%tag{attr: “Value”}	
	
Some Text	
.person	
	
.name	
	
	
Bob	

JADE:	

	
	
	

tag(attr=“Value”) Some Text	
.person	
	
.name Bob	

Other subtle differences too
CONVERSION TOOL: SUBLIME
TEXT
•

Afterthought: I should have written something in Perl	


•

Regexp Find/Replace	

•

^(s*)%

=> $1 (fix tags)	


•

(w){(.*)} => $1($2) (attribute curlys)	


•

(w):s*([“‘]) => $1=$2 (attribute key: value)
THINGS LEFT TO FIX
•

Helpers: = some_helper	


•

Text on a line on its own - Jade treats these as tags	


•

Nested attributes:	


	
	

•

	
->

%tag{ng:{style:’…’,click:’…’},class:’foo’}	
tag(ng-style=‘…’, ng-click=‘…’, class=‘foo’)	

Making sure the output matched the Ruby/HAML
version was HARD - HTML Diff tools suck
HELPERS BECAME MIXINS
•

Standard Rails Helpers:	


= form_for @person do |f|	
f.label :first_name	
f.text_field :first_name	
%br	

!
f.label :last_name	
f.text_field :last_name	
%br	

!
f.submit	

•

Custom Rails helpers stored in app/helpers/ folder	


•

http://api.rubyonrails.org/classes/ActionView/Helpers/FormHelper.html
JADE MIXINS
•

Very powerful. Poorly documented.	


•

Standalone or can have block contents	

// implementation	
mixin form(action)	
form(accept-charset="UTF-8", action=action, method="POST")	
input(name="utf8" type="hidden" value="&#x2713;")	
block	
// usage:	
+form(‘/post/to/here’)	
input(type=“text”,name=“hello”,value=“world”)	

•

Supports js code	

mixin app_error(field)	
if (errors && errors[field])	
div(class="err_msg " + field)	
each error in errors[field]	
span= error
JADE: NO DYNAMIC
INCLUDES
•

HAML/Helpers can do:	


	 	

= popup ‘roles/new'	

•

Jade needed:	


	
	

	
	

+popup(‘roles/new’)	
	
include ../roles/new
CREATING THE MODEL
•

Hand coded db.js layer developed over time from previous
projects	


•

One file per table (generally) in lib/model/*.js	


	
	
	
	
	

"use strict";	
	
var db = require('./db');	
	
exports.get = function get (id, cb) {	
	
	
db.get_one_row("SELECT * FROM Answers WHERE id=$1", [id], cb);	
}	
	
exports.get_by_submission_id = function (submission_id, cb) {	
	
	
db.query("SELECT * FROM Answers	
	
WHERE submission_id=$1	
	
	
ORDER BY question_id", [submission_id], cb);	
}
WHY CONTROL THE SQL?
exports.get_avg_team_size = function (company_id, role_id, months, cb) {	
var role_sql = role_id ? ' AND role_id=$2' : ' AND $2=$2';	

!
!

}	

if (months == 0 || months == '0') {	
months = '9000'; // OVER NINE THOUSAND	
}	
var sql = "SELECT avg(c) as value	
FROM (	
SELECT g.month, count(e.*)	
FROM generate_series(	
date_trunc('month', now() - CAST($3 AS INTERVAL)),	
now(),	
INTERVAL '1 month') g(month)	
LEFT JOIN employees e ON e.company_id=$1" + role_sql + "	
AND (start_date, COALESCE(end_date, 'infinity'))	
OVERLAPS	
(g.month, INTERVAL '1 month')	
GROUP BY g.month	
HAVING count(e.*) > 0	
) av(month,c)";	
db.get_one_row(sql, [company_id, role_id, months + " months"], cb);
AND FINALLY…
•

Routes	

•

Run Rails version of App	


•

Open Chrome Dev Console Network tools	


•

Hit record	


•

Find all routes and implement them
CREATING ROUTES/MODELS
•

While I glossed over this, it was the bulk of the
work	


•

Each endpoint was painstakingly re-created	


•

This allowed me to get a view of the DB layout	

•

And fix design bugs in the DB layout

find.fileSync(/.js$/, __dirname + '/routes').forEach(function (route_file) {	
require(route_file);	
});
BACKGROUND TASKS
•

Currently using Sidekiq, which uses Redis as a queue	

•

Used for downloading slow data feeds	


•

Node.js doesn’t care if downloads are slow	


•

So I punted on background tasks for now	


•

If I need them later I will use Kue (see npmjs.org)
DEPLOYMENT
•

Linode + Nginx + Postgres 9.3.1 + Runit +
Memcached	


•

/var/apps and deploy_to_runit for github autodeploy	


•

Monitoring via Zabbix	


•

60M used vs 130M for Rails
NEXT STEPS
•

Convert coffeescript code to plain JS - I find
coffeescript too much of a pain	


•

Implement graceful restarts using cluster	


•

Consider porting CSS to Bootstrap so we get
mobile support

Contenu connexe

Tendances

Artificial Intelligence (AI) in African Education
Artificial Intelligence (AI) in African EducationArtificial Intelligence (AI) in African Education
Artificial Intelligence (AI) in African Education
'Dimeji Falana
 
African Undersea Cables -- A History
African Undersea Cables -- A HistoryAfrican Undersea Cables -- A History
African Undersea Cables -- A History
Steve Song
 
E60_Power_Module.pdf
E60_Power_Module.pdfE60_Power_Module.pdf
E60_Power_Module.pdf
DanielNitu9
 
資策會網路系統整合工程師就業養成班
資策會網路系統整合工程師就業養成班資策會網路系統整合工程師就業養成班
資策會網路系統整合工程師就業養成班Ryan Chung
 
Artificial intelligence & machine learning landscape
Artificial intelligence & machine learning landscapeArtificial intelligence & machine learning landscape
Artificial intelligence & machine learning landscape
MOBConf
 
Incumbencias e interdisciplina
Incumbencias e interdisciplinaIncumbencias e interdisciplina
Incumbencias e interdisciplina
clarawebers
 
Top 10 Strategic Technologies in 2024: AI and Automation
Top 10 Strategic Technologies in 2024: AI and AutomationTop 10 Strategic Technologies in 2024: AI and Automation
Top 10 Strategic Technologies in 2024: AI and Automation
AutomationEdge Technologies
 
Doing The Trans-Siberian Railway In Style
Doing The Trans-Siberian Railway In StyleDoing The Trans-Siberian Railway In Style
Doing The Trans-Siberian Railway In Style
The Trans-Siberian Travel Company
 
Artificial Intelligence: Case studies (what can you build)
Artificial Intelligence: Case studies (what can you build)Artificial Intelligence: Case studies (what can you build)
Artificial Intelligence: Case studies (what can you build)
Rudradeb Mitra
 
300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán
300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán
300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán
haic2hv.net
 
14th Annual Parents, Kids & Money Survey
14th Annual Parents, Kids & Money Survey14th Annual Parents, Kids & Money Survey
14th Annual Parents, Kids & Money Survey
T. Rowe Price
 

Tendances (11)

Artificial Intelligence (AI) in African Education
Artificial Intelligence (AI) in African EducationArtificial Intelligence (AI) in African Education
Artificial Intelligence (AI) in African Education
 
African Undersea Cables -- A History
African Undersea Cables -- A HistoryAfrican Undersea Cables -- A History
African Undersea Cables -- A History
 
E60_Power_Module.pdf
E60_Power_Module.pdfE60_Power_Module.pdf
E60_Power_Module.pdf
 
資策會網路系統整合工程師就業養成班
資策會網路系統整合工程師就業養成班資策會網路系統整合工程師就業養成班
資策會網路系統整合工程師就業養成班
 
Artificial intelligence & machine learning landscape
Artificial intelligence & machine learning landscapeArtificial intelligence & machine learning landscape
Artificial intelligence & machine learning landscape
 
Incumbencias e interdisciplina
Incumbencias e interdisciplinaIncumbencias e interdisciplina
Incumbencias e interdisciplina
 
Top 10 Strategic Technologies in 2024: AI and Automation
Top 10 Strategic Technologies in 2024: AI and AutomationTop 10 Strategic Technologies in 2024: AI and Automation
Top 10 Strategic Technologies in 2024: AI and Automation
 
Doing The Trans-Siberian Railway In Style
Doing The Trans-Siberian Railway In StyleDoing The Trans-Siberian Railway In Style
Doing The Trans-Siberian Railway In Style
 
Artificial Intelligence: Case studies (what can you build)
Artificial Intelligence: Case studies (what can you build)Artificial Intelligence: Case studies (what can you build)
Artificial Intelligence: Case studies (what can you build)
 
300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán
300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán
300 bài toán lớp 4 ôn luyện thi học sinh giỏi và violympic toán
 
14th Annual Parents, Kids & Money Survey
14th Annual Parents, Kids & Money Survey14th Annual Parents, Kids & Money Survey
14th Annual Parents, Kids & Money Survey
 

Similaire à Converting a Rails application to Node.js

Intro to Scala.js - Scala UG Cologne
Intro to Scala.js - Scala UG CologneIntro to Scala.js - Scala UG Cologne
Intro to Scala.js - Scala UG Cologne
Marius Soutier
 
Couchbas for dummies
Couchbas for dummiesCouchbas for dummies
Couchbas for dummies
Qureshi Tehmina
 
Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology
Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology
Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology
Ayes Chinmay
 
前端MVC之BackboneJS
前端MVC之BackboneJS前端MVC之BackboneJS
前端MVC之BackboneJS
Zhang Xiaoxue
 
Dependency Management with RequireJS
Dependency Management with RequireJSDependency Management with RequireJS
Dependency Management with RequireJS
Aaronius
 
Amazon Web Services for PHP Developers
Amazon Web Services for PHP DevelopersAmazon Web Services for PHP Developers
Amazon Web Services for PHP Developers
Jeremy Lindblom
 
Wider than rails
Wider than railsWider than rails
Wider than rails
Alexey Nayden
 
(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...
(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...
(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...
Amazon Web Services
 
[Coscup 2012] JavascriptMVC
[Coscup 2012] JavascriptMVC[Coscup 2012] JavascriptMVC
[Coscup 2012] JavascriptMVC
Alive Kuo
 
Solutions for bi-directional Integration between Oracle RDMBS & Apache Kafka
Solutions for bi-directional Integration between Oracle RDMBS & Apache KafkaSolutions for bi-directional Integration between Oracle RDMBS & Apache Kafka
Solutions for bi-directional Integration between Oracle RDMBS & Apache Kafka
Guido Schmutz
 
Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...
Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...
Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...
confluent
 
3 Dundee-Spark Overview for C* developers
3 Dundee-Spark Overview for C* developers3 Dundee-Spark Overview for C* developers
3 Dundee-Spark Overview for C* developers
Christopher Batey
 
Manchester Hadoop Meetup: Spark Cassandra Integration
Manchester Hadoop Meetup: Spark Cassandra IntegrationManchester Hadoop Meetup: Spark Cassandra Integration
Manchester Hadoop Meetup: Spark Cassandra Integration
Christopher Batey
 
Full Stack Development with Node.js and NoSQL
Full Stack Development with Node.js and NoSQLFull Stack Development with Node.js and NoSQL
Full Stack Development with Node.js and NoSQL
All Things Open
 
Full stack development with node and NoSQL - All Things Open - October 2017
Full stack development with node and NoSQL - All Things Open - October 2017Full stack development with node and NoSQL - All Things Open - October 2017
Full stack development with node and NoSQL - All Things Open - October 2017
Matthew Groves
 
Intro to Spark and Spark SQL
Intro to Spark and Spark SQLIntro to Spark and Spark SQL
Intro to Spark and Spark SQL
jeykottalam
 
PHP and Rich Internet Applications
PHP and Rich Internet ApplicationsPHP and Rich Internet Applications
PHP and Rich Internet Applications
elliando dias
 
Solid And Sustainable Development in Scala
Solid And Sustainable Development in ScalaSolid And Sustainable Development in Scala
Solid And Sustainable Development in Scala
Kazuhiro Sera
 
Reading Cassandra Meetup Feb 2015: Apache Spark
Reading Cassandra Meetup Feb 2015: Apache SparkReading Cassandra Meetup Feb 2015: Apache Spark
Reading Cassandra Meetup Feb 2015: Apache Spark
Christopher Batey
 
Understanding backbonejs
Understanding backbonejsUnderstanding backbonejs
Understanding backbonejs
Nick Lee
 

Similaire à Converting a Rails application to Node.js (20)

Intro to Scala.js - Scala UG Cologne
Intro to Scala.js - Scala UG CologneIntro to Scala.js - Scala UG Cologne
Intro to Scala.js - Scala UG Cologne
 
Couchbas for dummies
Couchbas for dummiesCouchbas for dummies
Couchbas for dummies
 
Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology
Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology
Internet and Web Technology (CLASS-10) [Node.js] | NIC/NIELIT Web Technology
 
前端MVC之BackboneJS
前端MVC之BackboneJS前端MVC之BackboneJS
前端MVC之BackboneJS
 
Dependency Management with RequireJS
Dependency Management with RequireJSDependency Management with RequireJS
Dependency Management with RequireJS
 
Amazon Web Services for PHP Developers
Amazon Web Services for PHP DevelopersAmazon Web Services for PHP Developers
Amazon Web Services for PHP Developers
 
Wider than rails
Wider than railsWider than rails
Wider than rails
 
(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...
(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...
(BDT401) Big Data Orchestra - Harmony within Data Analysis Tools | AWS re:Inv...
 
[Coscup 2012] JavascriptMVC
[Coscup 2012] JavascriptMVC[Coscup 2012] JavascriptMVC
[Coscup 2012] JavascriptMVC
 
Solutions for bi-directional Integration between Oracle RDMBS & Apache Kafka
Solutions for bi-directional Integration between Oracle RDMBS & Apache KafkaSolutions for bi-directional Integration between Oracle RDMBS & Apache Kafka
Solutions for bi-directional Integration between Oracle RDMBS & Apache Kafka
 
Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...
Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...
Solutions for bi-directional integration between Oracle RDBMS and Apache Kafk...
 
3 Dundee-Spark Overview for C* developers
3 Dundee-Spark Overview for C* developers3 Dundee-Spark Overview for C* developers
3 Dundee-Spark Overview for C* developers
 
Manchester Hadoop Meetup: Spark Cassandra Integration
Manchester Hadoop Meetup: Spark Cassandra IntegrationManchester Hadoop Meetup: Spark Cassandra Integration
Manchester Hadoop Meetup: Spark Cassandra Integration
 
Full Stack Development with Node.js and NoSQL
Full Stack Development with Node.js and NoSQLFull Stack Development with Node.js and NoSQL
Full Stack Development with Node.js and NoSQL
 
Full stack development with node and NoSQL - All Things Open - October 2017
Full stack development with node and NoSQL - All Things Open - October 2017Full stack development with node and NoSQL - All Things Open - October 2017
Full stack development with node and NoSQL - All Things Open - October 2017
 
Intro to Spark and Spark SQL
Intro to Spark and Spark SQLIntro to Spark and Spark SQL
Intro to Spark and Spark SQL
 
PHP and Rich Internet Applications
PHP and Rich Internet ApplicationsPHP and Rich Internet Applications
PHP and Rich Internet Applications
 
Solid And Sustainable Development in Scala
Solid And Sustainable Development in ScalaSolid And Sustainable Development in Scala
Solid And Sustainable Development in Scala
 
Reading Cassandra Meetup Feb 2015: Apache Spark
Reading Cassandra Meetup Feb 2015: Apache SparkReading Cassandra Meetup Feb 2015: Apache Spark
Reading Cassandra Meetup Feb 2015: Apache Spark
 
Understanding backbonejs
Understanding backbonejsUnderstanding backbonejs
Understanding backbonejs
 

Dernier

AWS Cloud Cost Optimization Presentation.pptx
AWS Cloud Cost Optimization Presentation.pptxAWS Cloud Cost Optimization Presentation.pptx
AWS Cloud Cost Optimization Presentation.pptx
HarisZaheer8
 
5th LF Energy Power Grid Model Meet-up Slides
5th LF Energy Power Grid Model Meet-up Slides5th LF Energy Power Grid Model Meet-up Slides
5th LF Energy Power Grid Model Meet-up Slides
DanBrown980551
 
HCL Notes and Domino License Cost Reduction in the World of DLAU
HCL Notes and Domino License Cost Reduction in the World of DLAUHCL Notes and Domino License Cost Reduction in the World of DLAU
HCL Notes and Domino License Cost Reduction in the World of DLAU
panagenda
 
Serial Arm Control in Real Time Presentation
Serial Arm Control in Real Time PresentationSerial Arm Control in Real Time Presentation
Serial Arm Control in Real Time Presentation
tolgahangng
 
Columbus Data & Analytics Wednesdays - June 2024
Columbus Data & Analytics Wednesdays - June 2024Columbus Data & Analytics Wednesdays - June 2024
Columbus Data & Analytics Wednesdays - June 2024
Jason Packer
 
Nunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdf
Nunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdfNunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdf
Nunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdf
flufftailshop
 
HCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAU
HCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAUHCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAU
HCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAU
panagenda
 
Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...
Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...
Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...
saastr
 
Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...
Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...
Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...
saastr
 
How to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdf
How to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdfHow to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdf
How to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdf
Chart Kalyan
 
Unlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdf
Unlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdfUnlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdf
Unlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdf
Malak Abu Hammad
 
Driving Business Innovation: Latest Generative AI Advancements & Success Story
Driving Business Innovation: Latest Generative AI Advancements & Success StoryDriving Business Innovation: Latest Generative AI Advancements & Success Story
Driving Business Innovation: Latest Generative AI Advancements & Success Story
Safe Software
 
Recommendation System using RAG Architecture
Recommendation System using RAG ArchitectureRecommendation System using RAG Architecture
Recommendation System using RAG Architecture
fredae14
 
Operating System Used by Users in day-to-day life.pptx
Operating System Used by Users in day-to-day life.pptxOperating System Used by Users in day-to-day life.pptx
Operating System Used by Users in day-to-day life.pptx
Pravash Chandra Das
 
Best 20 SEO Techniques To Improve Website Visibility In SERP
Best 20 SEO Techniques To Improve Website Visibility In SERPBest 20 SEO Techniques To Improve Website Visibility In SERP
Best 20 SEO Techniques To Improve Website Visibility In SERP
Pixlogix Infotech
 
UI5 Controls simplified - UI5con2024 presentation
UI5 Controls simplified - UI5con2024 presentationUI5 Controls simplified - UI5con2024 presentation
UI5 Controls simplified - UI5con2024 presentation
Wouter Lemaire
 
Nordic Marketo Engage User Group_June 13_ 2024.pptx
Nordic Marketo Engage User Group_June 13_ 2024.pptxNordic Marketo Engage User Group_June 13_ 2024.pptx
Nordic Marketo Engage User Group_June 13_ 2024.pptx
MichaelKnudsen27
 
Taking AI to the Next Level in Manufacturing.pdf
Taking AI to the Next Level in Manufacturing.pdfTaking AI to the Next Level in Manufacturing.pdf
Taking AI to the Next Level in Manufacturing.pdf
ssuserfac0301
 
dbms calicut university B. sc Cs 4th sem.pdf
dbms  calicut university B. sc Cs 4th sem.pdfdbms  calicut university B. sc Cs 4th sem.pdf
dbms calicut university B. sc Cs 4th sem.pdf
Shinana2
 
leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...
leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...
leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...
alexjohnson7307
 

Dernier (20)

AWS Cloud Cost Optimization Presentation.pptx
AWS Cloud Cost Optimization Presentation.pptxAWS Cloud Cost Optimization Presentation.pptx
AWS Cloud Cost Optimization Presentation.pptx
 
5th LF Energy Power Grid Model Meet-up Slides
5th LF Energy Power Grid Model Meet-up Slides5th LF Energy Power Grid Model Meet-up Slides
5th LF Energy Power Grid Model Meet-up Slides
 
HCL Notes and Domino License Cost Reduction in the World of DLAU
HCL Notes and Domino License Cost Reduction in the World of DLAUHCL Notes and Domino License Cost Reduction in the World of DLAU
HCL Notes and Domino License Cost Reduction in the World of DLAU
 
Serial Arm Control in Real Time Presentation
Serial Arm Control in Real Time PresentationSerial Arm Control in Real Time Presentation
Serial Arm Control in Real Time Presentation
 
Columbus Data & Analytics Wednesdays - June 2024
Columbus Data & Analytics Wednesdays - June 2024Columbus Data & Analytics Wednesdays - June 2024
Columbus Data & Analytics Wednesdays - June 2024
 
Nunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdf
Nunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdfNunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdf
Nunit vs XUnit vs MSTest Differences Between These Unit Testing Frameworks.pdf
 
HCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAU
HCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAUHCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAU
HCL Notes und Domino Lizenzkostenreduzierung in der Welt von DLAU
 
Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...
Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...
Deep Dive: AI-Powered Marketing to Get More Leads and Customers with HyperGro...
 
Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...
Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...
Overcoming the PLG Trap: Lessons from Canva's Head of Sales & Head of EMEA Da...
 
How to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdf
How to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdfHow to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdf
How to Interpret Trends in the Kalyan Rajdhani Mix Chart.pdf
 
Unlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdf
Unlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdfUnlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdf
Unlock the Future of Search with MongoDB Atlas_ Vector Search Unleashed.pdf
 
Driving Business Innovation: Latest Generative AI Advancements & Success Story
Driving Business Innovation: Latest Generative AI Advancements & Success StoryDriving Business Innovation: Latest Generative AI Advancements & Success Story
Driving Business Innovation: Latest Generative AI Advancements & Success Story
 
Recommendation System using RAG Architecture
Recommendation System using RAG ArchitectureRecommendation System using RAG Architecture
Recommendation System using RAG Architecture
 
Operating System Used by Users in day-to-day life.pptx
Operating System Used by Users in day-to-day life.pptxOperating System Used by Users in day-to-day life.pptx
Operating System Used by Users in day-to-day life.pptx
 
Best 20 SEO Techniques To Improve Website Visibility In SERP
Best 20 SEO Techniques To Improve Website Visibility In SERPBest 20 SEO Techniques To Improve Website Visibility In SERP
Best 20 SEO Techniques To Improve Website Visibility In SERP
 
UI5 Controls simplified - UI5con2024 presentation
UI5 Controls simplified - UI5con2024 presentationUI5 Controls simplified - UI5con2024 presentation
UI5 Controls simplified - UI5con2024 presentation
 
Nordic Marketo Engage User Group_June 13_ 2024.pptx
Nordic Marketo Engage User Group_June 13_ 2024.pptxNordic Marketo Engage User Group_June 13_ 2024.pptx
Nordic Marketo Engage User Group_June 13_ 2024.pptx
 
Taking AI to the Next Level in Manufacturing.pdf
Taking AI to the Next Level in Manufacturing.pdfTaking AI to the Next Level in Manufacturing.pdf
Taking AI to the Next Level in Manufacturing.pdf
 
dbms calicut university B. sc Cs 4th sem.pdf
dbms  calicut university B. sc Cs 4th sem.pdfdbms  calicut university B. sc Cs 4th sem.pdf
dbms calicut university B. sc Cs 4th sem.pdf
 
leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...
leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...
leewayhertz.com-AI in predictive maintenance Use cases technologies benefits ...
 

Converting a Rails application to Node.js

  • 1. CONVERTING TO NODE How I converted a medium sized Rails project to Node.js
  • 2. BACKGROUND • CTO of Ideal Candidate • MVP existed when I joined • Author of Haraka and a few other Node.js libraries • Background in Perl and Anti-Spam
  • 3. THE RAILS APP • Unicorn + Thin + Rails • CoffeeScript with Angular.JS • SASS+Compass for stylesheets • HAML for templates • MySQL + MongoDB • Sidekiq for background tasks
  • 5. APP SIZE • 5700 Lines of Ruby (170 files) • 2500 Lines of CoffeeScript (119 files) • 3800 Lines of SASS (42 files) • 1500 Lines of HAML (100 files)
  • 6. FINAL GOAL • Express + minimal middleware • PostgreSQL hand coded with referential integrity • Jade templates • SCSS for CSS • Javascript + Angular.js frontend
  • 7. HOW? It’s all about the information, Marty
  • 8. RAILS GIVES YOU • Routes • Middleware • DB abstraction layer (aka Model) • Views • Controllers • Configuration • Plugins • A directory structure • Unit tests
  • 9. EXPRESS GIVES YOU • Routes • Middleware • That’s it • (This scares a lot of people)
  • 10. DIRECTORY STRUCTURE lib - general JS libraries ⌞ model - database accessors public - files that get served to the front end ⌞ assets - files that are considered part of the app ⌞ javascripts ⌞ stylesheets ⌞ images routes - files that provide express routes for HTML ⌞ api/v1 - files providing REST API endpoints (JSON) views - Jade templates app.js - entry point
  • 11. RAILS ASSET PIPELINE • Rails does a lot for you, but it’s f ’ing confusing • HAML Template Example: • = javascript_include_tag :application • You’d think this adds <script src=“application.js”>, but no.
  • 12. RAILS ASSET PIPELINE • Easy if you understand it • Too much magic for my liking • But… the overall effect is good. So I copied some of it.
  • 13. JADE EQUIVALENT each script in javascripts script(src=script, type=“text/javascript”)
  • 14. APP.JS: var js = find.fileSync(/.(coffee|js)$/, __dirname + ‘/public/assets/javascripts/controllers') .map(function (i) { return i.replace(/^.*/assets/, ‘/assets') .replace(/.coffee$/, ‘.js') }); app.all('*', function (req, res, next) { res.locals.javascripts = js; next(); });
  • 15. SERVING COFFEESCRIPT // development app.use(require('coffee-middleware')({ src: __dirname + '/public', compress: false, })); ! // Production var all_js = coffee_compiler.generate(js, true); ! var shasum = crypto.createHash('sha256'); shasum.update(all_js); var js_sha = shasum.digest('hex'); ! js = ['/assets/javascripts/application-' + js_sha + '.js']; ! app.get('/assets/javascripts/application-' + js_sha + '.js', function (req, res) { res.type('js'); res.setHeader('Cache-Control', 'public, max-age=31557600'); res.end(all_js); });
  • 16. SERVING SCSS • Node-sass module serves .scss files • Doesn’t serve up .sass files (weird, huh?) ! # mass-convert.bash for F in `find . -name *.sass`; do O=“${F/.sass/.scss}” sass-convert -F sass -T scss “$F” “$O” git rm -f “$F” git add “$O” done
  • 17. SCSS APP.JS function compile_css () { console.log("Recompiling CSS"); resources.watchers.forEach(function (w) { w.close(); }); resources.watchers = []; ! ! ! ! } resources.css = sass.renderSync({ file: __dirname + '/public/assets/stylesheets/application.scss', includePaths: [__dirname + '/public/assets/stylesheets'], }); var shasum = crypto.createHash('sha256'); shasum.update(resources.css); var css_sha = shasum.digest('hex'); resources.stylesheets[0] = '/assets/stylesheets/application-' + css_sha + '.css'; resources.stylesheets[1] = '//fonts.googleapis.com/css?family=Open+Sans:400,700,800,300,600'; find.fileSync(cssregexp, __dirname + '/public/assets/stylesheets').forEach(function (f) { resources.watchers.push(fs.watch(f, compile_css)); })
  • 18. SCSS APP.JS compile_css(); ! // Dev and Production the same app.get(//assets/stylesheets/application-(w+).css/, function (req, res) { res.type('css'); res.setHeader('Cache-Control', 'public, max-age=31557600'); res.end(resources.css); });
  • 19. ADDING COMPASS • Ruby CSS framework • Luckily only the CSS3 part used • CSS3 code is just SASS files • Once I figured this out, copied them into my project, et voila!
  • 20. CONVERTING HAML TO JADE • Both indent based • HAML: • • %tag{attr: “Value”} Some Text .person .name Bob JADE: tag(attr=“Value”) Some Text .person .name Bob Other subtle differences too
  • 21. CONVERSION TOOL: SUBLIME TEXT • Afterthought: I should have written something in Perl • Regexp Find/Replace • ^(s*)% => $1 (fix tags) • (w){(.*)} => $1($2) (attribute curlys) • (w):s*([“‘]) => $1=$2 (attribute key: value)
  • 22. THINGS LEFT TO FIX • Helpers: = some_helper • Text on a line on its own - Jade treats these as tags • Nested attributes: • -> %tag{ng:{style:’…’,click:’…’},class:’foo’} tag(ng-style=‘…’, ng-click=‘…’, class=‘foo’) Making sure the output matched the Ruby/HAML version was HARD - HTML Diff tools suck
  • 23. HELPERS BECAME MIXINS • Standard Rails Helpers: = form_for @person do |f| f.label :first_name f.text_field :first_name %br ! f.label :last_name f.text_field :last_name %br ! f.submit • Custom Rails helpers stored in app/helpers/ folder • http://api.rubyonrails.org/classes/ActionView/Helpers/FormHelper.html
  • 24. JADE MIXINS • Very powerful. Poorly documented. • Standalone or can have block contents // implementation mixin form(action) form(accept-charset="UTF-8", action=action, method="POST") input(name="utf8" type="hidden" value="&#x2713;") block // usage: +form(‘/post/to/here’) input(type=“text”,name=“hello”,value=“world”) • Supports js code mixin app_error(field) if (errors && errors[field]) div(class="err_msg " + field) each error in errors[field] span= error
  • 25. JADE: NO DYNAMIC INCLUDES • HAML/Helpers can do: = popup ‘roles/new' • Jade needed: +popup(‘roles/new’) include ../roles/new
  • 26. CREATING THE MODEL • Hand coded db.js layer developed over time from previous projects • One file per table (generally) in lib/model/*.js "use strict"; var db = require('./db'); exports.get = function get (id, cb) { db.get_one_row("SELECT * FROM Answers WHERE id=$1", [id], cb); } exports.get_by_submission_id = function (submission_id, cb) { db.query("SELECT * FROM Answers WHERE submission_id=$1 ORDER BY question_id", [submission_id], cb); }
  • 27. WHY CONTROL THE SQL? exports.get_avg_team_size = function (company_id, role_id, months, cb) { var role_sql = role_id ? ' AND role_id=$2' : ' AND $2=$2'; ! ! } if (months == 0 || months == '0') { months = '9000'; // OVER NINE THOUSAND } var sql = "SELECT avg(c) as value FROM ( SELECT g.month, count(e.*) FROM generate_series( date_trunc('month', now() - CAST($3 AS INTERVAL)), now(), INTERVAL '1 month') g(month) LEFT JOIN employees e ON e.company_id=$1" + role_sql + " AND (start_date, COALESCE(end_date, 'infinity')) OVERLAPS (g.month, INTERVAL '1 month') GROUP BY g.month HAVING count(e.*) > 0 ) av(month,c)"; db.get_one_row(sql, [company_id, role_id, months + " months"], cb);
  • 28. AND FINALLY… • Routes • Run Rails version of App • Open Chrome Dev Console Network tools • Hit record • Find all routes and implement them
  • 29. CREATING ROUTES/MODELS • While I glossed over this, it was the bulk of the work • Each endpoint was painstakingly re-created • This allowed me to get a view of the DB layout • And fix design bugs in the DB layout find.fileSync(/.js$/, __dirname + '/routes').forEach(function (route_file) { require(route_file); });
  • 30. BACKGROUND TASKS • Currently using Sidekiq, which uses Redis as a queue • Used for downloading slow data feeds • Node.js doesn’t care if downloads are slow • So I punted on background tasks for now • If I need them later I will use Kue (see npmjs.org)
  • 31. DEPLOYMENT • Linode + Nginx + Postgres 9.3.1 + Runit + Memcached • /var/apps and deploy_to_runit for github autodeploy • Monitoring via Zabbix • 60M used vs 130M for Rails
  • 32. NEXT STEPS • Convert coffeescript code to plain JS - I find coffeescript too much of a pain • Implement graceful restarts using cluster • Consider porting CSS to Bootstrap so we get mobile support

Notes de l'éditeur

  1. Knowledge Hiring Lack of “Magic” Performance Queryability PostgreSQL Referential integrity Pre-Customers so zero downtime
  2. “Little ones and zeros, it’s all just electrons”. Web apps are a flow of information. HTML+JS goes to the client, XHR queries come back, XHR responses get sent, POST and PUT requests modify data. Monitor all these, watch the flow, and you have your app. The backend is unimportant at that level.
  3. And probably more
  4. Let’s start with a solid directory structure. I always do this when I’m creating express apps as it helps me stay grounded and not create too many hacks. My app.js has evolved a bit over time to include many things. Happy to share it.
  5. What it actually does: Looks for your application.js. If it doesn’t find it, looks for application.coffee. It then scans that file for comments containing =require or =require_tree In production it loads all those files, coffee script compiles them if necessary, minimises, and delivers via a &lt;script src=“application-$md5.js”&gt;. In dev, it puts in as many &lt;script&gt; tags as required for each file and delivers them, coffee script compiled. Rails does the same for SASS stylesheets. FML.
  6. I wrote coffee_compiler - but it’s very simple
  7. Did this by installing the gem, finding the .sass files, and copying them over to my project.
  8. Helpers are functions to simplify HTML generation. Rails comes with a bunch that help with creating forms (and some others). These had to be ported to become Jade mixins.
  9. HAML version creates some HTML and loads the template ‘views/roles/new’ to be part of the content Jade cannot do that, so we manually include the template
  10. Slightly more code than ActiveRecord, but gives you full control over SQL Only returns plain objects, doesn’t support dynamic updates Considering using getters/setters for next iteration, but not sure how they serialize