It is advised to upgrade you rails app at least to 4.2 and use the original webpacker gem.
If you are stuck in legacy rails apps without choice and/or an upgrade might take long time, this gem might be helpful.
This gem has been tested in:
- ruby 1.9.3 and jruby 1.7.26
- rails 3.2
It may work in rails 4.0 and 4.1
Webpacker makes it easy to use the JavaScript pre-processor and bundler Webpack 2.x.x+ to manage application-like JavaScript in Rails. It coexists with the asset pipeline, as the primary purpose for Webpack is app-like JavaScript, not images, CSS, or even JavaScript Sprinkles (that all continues to live in app/assets).
However, it is possible to use Webpacker for CSS, images and fonts assets as well, in which case you may not even need the asset pipeline. This is mostly relevant when exclusively using component-based JavaScript frameworks.
- Prerequisites
- Features
- Installation
- Integrations
- Binstubs
- Configuration
- Linking Styles, Images and Fonts
- How-tos
- Extending
- Deployment
- Testing
- Troubleshooting
- Wishlist
- License
- Ruby 2.2+
- Rails 4.2+
- Node.js 6.4.0+
- Yarn 0.20.1+
- Webpack 2
- ES6 with babel
- Automatic code splitting using multiple entry points
- Stylesheets - SASS and CSS
- Images and fonts
- PostCSS - Auto-Prefixer
- Asset compression, source-maps, and minification
- CDN support
- React, Angular, Elm and Vue support out-of-the-box
- Rails view helpers
- Extensible and configurable
You can either add Webpacker during setup of a new Rails 5.1+ application
using new --webpack
option:
# Available Rails 5.1+
rails new myapp --webpack
Or add it to your Gemfile
, run bundle and ./bin/rails webpacker:install
or bundle exec rake webpacker:install
(on rails version < 5.0):
# Gemfile
gem 'webpacker', '~> 2.0'
# OR if you prefer to use master
gem 'webpacker', git: 'https://github.com/rails/webpacker.git'
Note: Use rake
instead of rails
if you are using webpacker
with rails version < 5.0
Webpacker by default ships with basic out-of-the-box integration for React, Angular, Vue and Elm. You can see a list of available commands/tasks by running:
# Within rails app
./bin/rails webpacker
or in rails version < 5.0
# Within rails app
./bin/rake webpacker
To use Webpacker with React, create a
new Rails 5.1+ app using --webpack=react
option:
# Rails 5.1+
rails new myapp --webpack=react
(or run ./bin/rails webpacker:install:react
in a existing Rails app already
setup with webpacker).
The installer will add all relevant dependencies using yarn, any changes
to the configuration files and an example React component to your
project in app/javascript/packs
so that you can experiment with React right away.
To use Webpacker with Angular, create a
new Rails 5.1+ app using --webpack=angular
option:
# Rails 5.1+
rails new myapp --webpack=angular
(or run ./bin/rails webpacker:install:angular
on a Rails app already
setup with webpacker).
The installer will add TypeScript and Angular core libraries using yarn plus
any changes to the configuration files. An example component is written in
TypeScript will also be added to your project in app/javascript
so that
you can experiment with Angular right away.
To use Webpacker with Vue, create a
new Rails 5.1+ app using --webpack=vue
option:
# Rails 5.1+
rails new myapp --webpack=vue
(or run ./bin/rails webpacker:install:vue
on a Rails app already setup with webpacker).
The installer will add Vue and required libraries using yarn plus
any changes to the configuration files. An example component will
also be added to your project in app/javascript
so that you can
experiment Vue right away.
To use Webpacker with Elm, create a
new Rails 5.1+ app using --webpack=elm
option:
# Rails 5.1+
rails new myapp --webpack=elm
(or run ./bin/rails webpacker:install:elm
on a Rails app already setup with webpacker).
The Elm library and core packages will be added via Yarn and Elm itself.
An example Main.elm
app will also be added to your project in app/javascript
so that you can experiment with Elm right away.
Webpacker ships with two binstubs: ./bin/webpack
and ./bin/webpack-dev-server
.
Both are thin wrappers around the standard webpack.js
and webpack-dev-server.js
executable to ensure that the right configuration file and environment variables
are loaded depending on your environment.
In development, you'll need to run ./bin/webpack-dev-server
in a separate terminal
from ./bin/rails server
to have your app/javascript/packs/*.js
files compiled
as you make changes.
./bin/webpack-dev-server
launches the Webpack Dev Server, which serves your pack files
on http://localhost:8080/
by default and supports live code reloading in the development environment. You will need to install additional plugins for Webpack if you want
features like Hot Module Replacement.
If you'd rather not have to run the two processes separately by hand, you can use Foreman:
gem install foreman
# Procfile
web: bundle exec rails s
webpacker: ./bin/webpack-dev-server
foreman start
You can also pass CLI options supported by webpack-dev-server. Please note that inline options will always take precedence over the ones already set in the configuration file.
./bin/webpack-dev-server --host 0.0.0.0 --inline true --hot false
We recommend using webpack-dev-server
during development for a better experience,
however, if you don't want that for some reason you can always use webpack
binstub with
watch option, which uses webpack Command Line Interface (CLI). This will use public_output_path
from config/webpacker.yml
directory to serve your packs using configured rails server.
You can pass cli options available with Webpack:
./bin/webpack --watch --progress --colors
Webpacker gives you a default set of configuration files for test, development and
production environments. They all live together with the shared
points in config/webpack/*.js
.
By default, you shouldn't have to make any changes to config/webpack/*.js
files since it's all standard production-ready configuration however
if you do need to customize or add a new loader this is where you would go.
Webpack enables the use of loaders to preprocess files. This allows you to
bundle any static resource way beyond JavaScript. All base loaders
that ships with webpacker are located inside config/webpack/loaders
.
If you want to add a new loader, for example, to process json
files via webpack:
yarn add json-loader
And create a json.js
file inside loaders
directory:
module.exports = {
test: /\.json$/,
use: 'json-loader'
}
Now if you import()
any .json
files inside your javascript
they will be processed using json-loader
. Voila!
By default, webpacker ships with simple conventions for where the javascript
app files and compiled webpack bundles will go in your rails app,
but all these options are configurable from config/webpacker.yml
file.
The configuration for what Webpack is supposed to compile by default rests
on the convention that every file in app/javascript/packs/*
(default)
or whatever path you set for source_entry_path
in the webpacker.yml
configuration
is turned into their own output files (or entry points, as Webpack calls it).
Suppose you want to change the source directory from app/javascript
to frontend
and output to assets/packs
this is how you would do it:
# config/webpacker.yml
source_path: frontend
source_entry_path: packs
public_output_path: assets/packs # outputs to => public/assets/packs
Similary you can also control and configure webpack-dev-server
settings from config/webpacker.yml
file:
# config/webpacker.yml
development:
dev_server:
host: 0.0.0.0
port: 8080
https: false
Webpacker ships with babel - a JavaScript compiler so
you can use next generation JavaScript, today. The Webpacker installer sets up a
standard .babelrc
file in your app root, which will work great in most cases
because of babel-env-preset.
Following ES6/7 features are supported out of the box:
- Async/await.
- Object Rest/Spread Properties.
- Exponentiation Operator.
- Dynamic import() - useful for route level code-splitting
- Class Fields and Static Properties.
We have also included babel polyfill that includes a custom regenerator runtime and core-js.
Webpacker out-of-the-box provides CSS post-processing using
postcss-loader
and the installer sets up a standard .postcssrc.yml
file in your app root with standard plugins.
plugins:
postcss-smart-import: {}
postcss-cssnext: {}
Webpacker out-of-the-box provides CDN support using your Rails app config.action_controller.asset_host
setting. If you already have CDN added in your rails app
you don't need to do anything extra for webpacker, it just works.
You may require the webpack-dev-server
to serve views over HTTPS in development.
To do this, set the https
option for webpack-dev-server
to true
in config/webpacker.yml
, then start the dev server as usual
with ./bin/webpack-dev-server
.
Please note that the webpack-dev-server
will use a self-signed certificate,
so your web browser will display a warning upon accessing the page.
Webpacker out-of-the-box doesn't ship with HMR just yet. You will need to install additional plugins for Webpack if you want to add HMR support.
You can checkout these links on this subject:
- https://webpack.js.org/configuration/dev-server/#devserver-hot
- https://webpack.js.org/guides/hmr-react/
Static assets like images, fonts and stylesheets support is enabled out-of-box and you can link them into your javascript app code and have them compiled automatically.
// app/javascript/hello_react/styles/hello-react.sass
.hello-react
padding: 20px
font-size: 12px
// React component example
// app/javascripts/packs/hello_react.jsx
import React from 'react'
import helloIcon from '../hello_react/images/icon.png'
import '../hello_react/styles/hello-react.sass'
const Hello = props => (
<div className="hello-react">
<img src={helloIcon} alt="hello-icon" />
<p>Hello {props.name}!</p>
</div>
)
Under the hood webpack uses
extract-text-webpack-plugin plugin to extract all the referenced styles within your app and compile it into
a separate [pack_name].css
bundle so that in your view you can use the
stylesheet_pack_tag
helper.
<%= stylesheet_pack_tag 'hello_react' %>
You can also link js/images/styles used within your js app in views using
asset_pack_path
helper. This helper is useful in cases where you just want to
create a <link rel="prefetch">
or <img />
for an asset.
<%= asset_pack_path 'hello_react.css' %>
<%# => "/packs/hello_react.css" %>
<img src="<%= asset_pack_path 'calendar.png' %>" />
<% # => <img src="/packs/calendar.png" /> %>
You can also import styles from node_modules
using the following syntax.
Please note that your styles will always be extracted into [pack_name].css
:
// app/javascript/app-styles.sass
// ~ to tell webpack that this is not a relative import:
@import '~@material/animation/mdc-animation.scss'
@import '~boostrap/dist/bootstrap.css'
// Your main app pack
// app/javascript/packs/app.js
import '../app-styles'
<%# In your views %>
<%= javascript_pack_tag 'app' %>
<%= stylesheet_pack_tag 'app' %>
Let's say you're building a calendar app. Your JS app structure could look like this:
// app/javascript/packs/calendar.js
import 'calendar'
app/javascript/calendar/index.js // gets loaded by import 'calendar'
app/javascript/calendar/components/grid.jsx
app/javascript/calendar/styles/grid.sass
app/javascript/calendar/models/month.js
<%# app/views/layouts/application.html.erb %>
<%= javascript_pack_tag 'calendar' %>
<%= stylesheet_pack_tag 'calendar' %>
But it could also look a million other ways.
You can also namespace your packs using directories similar to a Rails app.
app/javascript/packs/admin/orders.js
app/javascript/packs/shop/orders.js
and reference them in your views like this:
<%# app/views/admin/orders/index.html.erb %>
<%= javascript_pack_tag 'admin/orders' %>
and
<%# app/views/shop/orders/index.html.erb %>
<%= javascript_pack_tag 'shop/orders' %>
You may consider using react-rails or webpacker-react for more advanced react integration. However here is how you can do it yourself:
<%# views/layouts/application.html.erb %>
<%= content_tag :div,
id: "hello-react",
data: {
message: 'Hello!',
name: 'David'
}.to_json do %>
<% end %>
// app/javascript/packs/hello_react.js
const Hello = props => (
<div className='react-app-wrapper'>
<img src={clockIcon} alt="clock" />
<h5 className='hello-react'>
{props.message} {props.name}!
</h5>
</div>
)
// Render component with data
document.addEventListener('DOMContentLoaded', () => {
const node = document.getElementById('hello-react')
const data = JSON.parse(node.getAttribute('data'))
ReactDOM.render(<Hello {...data} />, node)
})
<%= content_tag :div,
id: "hello-vue",
data: {
message: "Hello!",
name: "David"
} do %>
<% end %>
<div id="hello-vue" data-name="David" data-message="Hello!"></div>
// Render component with data
document.addEventListener('DOMContentLoaded', () => {
const node = document.getElementById('hello-vue')
const data = JSON.parse(node.getAttribute('data'))
const app = new Vue({
data: data,
el: '#vue-app',
template: '<App/>',
components: { App }
})
console.log(app)
})
You can follow same steps for Angular too.
The CommonsChunkPlugin is an opt-in feature that creates a separate file (known as a chunk), consisting of common modules shared between multiple entry points. By separating common modules from bundles, the resulting chunked file can be loaded once initially, and stored in the cache for later use. This results in page speed optimizations as the browser can quickly serve the shared code from the cache, rather than being forced to load a larger bundle whenever a new page is visited.
Create a app-config.js
file inside config/webpack
and in that file add:
module.exports = {
plugins: [
// Creates a common vendor.js with all shared modules
new webpack.optimize.CommonsChunkPlugin({
name: 'vendor',
minChunks: (module) => {
// this assumes your vendor imports exist in the node_modules directory
return module.context && module.context.indexOf('node_modules') !== -1;
}
}),
// Webpack code chunk - manifest.js
new webpack.optimize.CommonsChunkPlugin({
name: 'manifest',
minChunks: Infinity
})
]
}
You can add this in shared.js
too but we are doing this to ensure smoother upgrades.
// config/webpack/shared.js
// .... rest of the config
const appConfig = require('./app-config.js')
plugins: appConfig.plugins.concat([
// ...existing plugins
])
Now, add these files to your layouts/application.html.erb
:
<%# Head %>
<%= javascript_pack_tag 'manifest' %>
<%= javascript_pack_tag 'vendor' %>
<%# If importing any styles from node_modules in your JS app %>
<%= stylesheet_pack_tag 'vendor' %>
While you are free to use require()
and module.exports
, we encourage you
to use import
and export
instead since it reads and looks much better.
import Button from 'react-bootstrap/lib/Button'
// or
import { Button } from 'react-bootstrap'
class Foo {
// code...
}
export default Foo
import Foo from './foo'
You can also use named export and import
export const foo = () => console.log('hello world')
import { foo } from './foo'
To add any new JS module you can use yarn
:
yarn add bootstrap material-ui
You can use yarn to add bootstrap or any other modules available on npm:
yarn add bootstrap
Import Bootstrap and theme(optional) CSS in your app/javascript/packs/app.js file:
import 'bootstrap/dist/css/bootstrap.css'
import 'bootstrap/dist/css/bootstrap-theme.css'
Or in your app/javascript/app.sass file:
// ~ to tell that this is not a relative import
@import '~bootstrap/dist/css/bootstrap.css'
@import '~bootstrap/dist/css/bootstrap-theme.css'
- Setup react using webpacker react installer. Then add required depedencies for using typescript with React:
yarn add ts-loader typescript @types/react @types/react-dom
# You don't need this with typescript
yarn remove prop-types
- Add a
tsconfig.json
to project root:
{
"compilerOptions": {
"declaration": false,
"emitDecoratorMetadata": true,
"experimentalDecorators": true,
"lib": ["es6", "dom"],
"module": "es6",
"moduleResolution": "node",
"sourceMap": true,
"jsx": "react",
"target": "es5"
},
"exclude": [
"**/*.spec.ts",
"node_modules",
"public"
],
"compileOnSave": false
}
- Add a new loader
config/webpack/loaders/typescript.js
:
module.exports = {
test: /.(ts|tsx)$/,
loader: 'ts-loader'
}
- Finally add
.tsx
to the list of extensions inconfig/webpacker.yml
and rename your generatedhello_react.js
using react installer tohello_react.tsx
and make it valid typescript and now you can use typescript, JSX with React.
After you have installed angular using angular installer you would need to follow these steps to add HTML templates support:
- Use
yarn
to add html-loader
yarn add html-loader
- Add html-loader to
config/webpacker/loaders/html.js
module.exports = {
test: /\.html$/,
use: [{
loader: 'html-loader',
options: {
minimize: true,
removeAttributeQuotes: false,
caseSensitive: true,
customAttrSurround: [ [/#/, /(?:)/], [/\*/, /(?:)/], [/\[?\(?/, /(?:)/] ],
customAttrAssign: [ /\)?\]?=/ ]
}
}]
}
- Add
.html
toconfig/webpacker.yml
extensions:
- .elm
- .coffee
- .html
- Setup a custom
d.ts
definition
// app/javascript/hello_angular/html.d.ts
declare module "*.html" {
const content: string
export default content
}
- Add a template.html file relative to
app.component.ts
<h1>Hello {{name}}</h1>
- Import template into
app.component.ts
import { Component } from '@angular/core'
import templateString from './template.html'
@Component({
selector: 'hello-angular',
template: templateString
})
export class AppComponent {
name = 'Angular!'
}
That's all. Voila!
To enable CSS modules, you would need to update config/webpack/loaders/sass.js
file, particularly css-loader
:
// Add css-modules
{
loader: 'css-loader',
options: {
minimize: env.NODE_ENV === 'production',
modules: true,
localIdentName: '[path][name]__[local]--[hash:base64:5]'
}
}
That's all. Now, you can use CSS modules within your JS app:
import React from 'react'
import styles from './styles.css'
const Hello = props => (
<div className={styles.wrapper}>
<img src={clockIcon} alt="clock" className={styles.img} />
<h5 className={styles.name}>
{props.message} {props.name}!
</h5>
</div>
)
css-next is supported out-of-box in Webpacker allowing the use of latest CSS features, today.
If you are using vim or emacs and want to ignore certain files you can add ignore-loader
:
yard add ignore-loader
and create a new loader file inside config/webpack/loaders
:
// config/webpack/loaders/ignores.js
// ignores vue~ swap files
module.exports = {
test: /.vue~$/,
loader: 'ignore-loader'
}
And now all files with .vue~
will be ignored by the webpack compiler.
It's possible to link to assets that have been precompiled by sprockets. Add the .erb
extension to your javascript file, then you can use Sprockets' asset helpers:
<%# app/javascript/my_pack/example.js.erb %>
<% helpers = ActionController::Base.helpers %>
var railsImagePath = "<%= helpers.image_path('rails.png') %>"
This is enabled by the rails-erb-loader
loader rule in config/webpack/loaders/erb.js
.
You can also use babel-plugin-module-resolver to reference assets directly from app/assets/**
yarn add babel-plugin-module-resolver
Specify the plugin in your .babelrc
with the custom root or alias. Here's an example:
{
"plugins": [
["module-resolver", {
"root": ["./app"],
"alias": {
"assets": "./assets"
}
}]
]
}
And then within your javascript app code:
// Note: we don't have do any ../../ jazz
import FooImage from 'assets/images/foo-image.png'
import 'assets/stylesheets/bar.sass'
We suggest you don't directly overwrite the provided configuration files and extend instead for smoother upgrades. Here is one way to do it:
Create a app-config.js
file inside config/webpack
, and in that add:
module.exports = {
production: {
plugins: [
// ... Add plugins
]
},
development: {
output: {
// ... Custom output path
}
}
}
// config/webpack/production.js
const { plugins } = require('./app-config.js')
plugins: appConfig.plugins.concat([
// ...existing plugins
])
But this could be done million other ways.
Webpacker hooks up a new webpacker:compile
task to assets:precompile
, which gets run whenever you run assets:precompile
. If you are not using sprockets you
can manually trigger bundle exec rails webpacker:compile
during your app deploy.
The javascript_pack_tag
and stylesheet_pack_tag
helper method will automatically insert the correct HTML tag for compiled pack. Just like the asset pipeline does it.
By default the output will look like this in different environments:
<!-- In development mode with webpack-dev-server -->
<script src="http://localhost:8080/calendar.js"></script>
<link rel="stylesheet" media="screen" href="http://localhost:8080/calendar.css">
<!-- In development mode -->
<script src="/packs/calendar.js"></script>
<link rel="stylesheet" media="screen" href="/packs/calendar.css">
<!-- In production mode -->
<script src="/packs/calendar-0bd141f6d9360cf4a7f5.js"></script>
<link rel="stylesheet" media="screen" href="/packs/calendar-dc02976b5f94b507e3b6.css">
Heroku installs yarn and node by default if you deploy a rails app with Webpacker so all you would need to do:
heroku create shiny-webpacker-app
heroku addons:create heroku-postgresql:hobby-dev
git push heroku master
Webpacker lazily compiles assets in test env so you can write your tests without any extra setup and everything will just work out of the box.
Here is a sample system test case with hello_react example component:
// Example react component
import React from 'react'
import ReactDOM from 'react-dom'
import PropTypes from 'prop-types'
const Hello = props => (
<div>Hello David</div>
)
document.addEventListener('DOMContentLoaded', () => {
ReactDOM.render(
<Hello />,
document.body.appendChild(document.createElement('div')),
)
})
<%# views/pages/home.html.erb %>
<%= javascript_pack_tag "hello_react" %>
# Tests example react component
require "application_system_test_case"
class HomesTest < ApplicationSystemTestCase
test "can see the hello message" do
visit root_url
assert_selector "h5", text: "Hello! David"
end
end
-
If you get this error
ENOENT: no such file or directory - node-sass
on Heroku or elsewhere duringassets:precompile
orbundle exec rails webpacker:compile
then you would need to rebuild node-sass. It's a bit weird error, basically, it can't find thenode-sass
binary. An easy solution is to create a postinstall hook -npm rebuild node-sass
inpackage.json
and that will ensurenode-sass
is rebuild whenever you install any new modules. -
If you get this error
Can't find hello_react.js in manifest.json
when loading a view in the browser it's because Webpack is still compiling packs. Webpacker uses amanifest.json
file to keep track of packs in all environments, however since this file is generated after packs are compiled by webpack. So, if you load a view in browser whilst webpack is compiling you will get this error. Therefore, make sure webpack (i.e./bin/webpack-dev-server
) is running and has completed the compilation successfully before loading a view.
Webpacker is released under the MIT License.