2010-01-23 16:30:17 -05:00
|
|
|
require 'rails/railtie'
|
2010-03-01 22:29:12 -05:00
|
|
|
require 'active_support/core_ext/module/delegation'
|
|
|
|
require 'pathname'
|
2010-04-26 11:25:07 -04:00
|
|
|
require 'rbconfig'
|
2010-06-23 10:10:26 -04:00
|
|
|
require 'rails/engine/railties'
|
2010-01-21 17:14:20 -05:00
|
|
|
|
|
|
|
module Rails
|
2011-01-14 03:34:59 -05:00
|
|
|
# Rails::Engine allows you to wrap a specific Rails application and share it across
|
|
|
|
# different applications. Since Rails 3.0, every <tt>Rails::Application</tt> is nothing
|
2011-01-19 15:30:28 -05:00
|
|
|
# more than an engine, allowing you to share it very easily.
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Any <tt>Rails::Engine</tt> is also a <tt>Rails::Railtie</tt>, so the same methods
|
2011-01-19 15:30:28 -05:00
|
|
|
# (like <tt>rake_tasks</tt> and +generators+) and configuration available in the
|
2011-01-14 03:34:59 -05:00
|
|
|
# latter can also be used in the former.
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
|
|
|
# == Creating an Engine
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# In Rails versions prior to 3.0, your gems automatically behaved as engines, however,
|
2010-02-02 14:05:26 -05:00
|
|
|
# this coupled Rails to Rubygems. Since Rails 3.0, if you want a gem to automatically
|
2011-01-19 15:30:28 -05:00
|
|
|
# behave as an engine, you have to specify an +Engine+ for it somewhere inside
|
|
|
|
# your plugin's +lib+ folder (similar to how we specify a +Railtie+):
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
|
|
|
# # lib/my_engine.rb
|
|
|
|
# module MyEngine
|
|
|
|
# class Engine < Rails::Engine
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Then ensure that this file is loaded at the top of your <tt>config/application.rb</tt>
|
2011-01-19 15:30:28 -05:00
|
|
|
# (or in your +Gemfile+) and it will automatically load models, controllers and helpers
|
|
|
|
# inside +app+, load routes at <tt>config/routes.rb</tt>, load locales at
|
2011-01-14 03:34:59 -05:00
|
|
|
# <tt>config/locales/*</tt>, and load tasks at <tt>lib/tasks/*</tt>.
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
|
|
|
# == Configuration
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# Besides the +Railtie+ configuration which is shared across the application, in a
|
2011-01-14 03:34:59 -05:00
|
|
|
# <tt>Rails::Engine</tt> you can access <tt>autoload_paths</tt>, <tt>eager_load_paths</tt>
|
|
|
|
# and <tt>autoload_once_paths</tt>, which, differently from a <tt>Railtie</tt>, are scoped to
|
2011-01-19 15:30:28 -05:00
|
|
|
# the current engine.
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
|
|
|
# Example:
|
|
|
|
#
|
|
|
|
# class MyEngine < Rails::Engine
|
|
|
|
# # Add a load path for this specific Engine
|
2010-06-22 17:17:20 -04:00
|
|
|
# config.autoload_paths << File.expand_path("../lib/some/path", __FILE__)
|
2010-04-05 18:32:03 -04:00
|
|
|
#
|
|
|
|
# initializer "my_engine.add_middleware" do |app|
|
2010-04-29 14:43:09 -04:00
|
|
|
# app.middleware.use MyEngine::Middleware
|
2010-04-05 18:32:03 -04:00
|
|
|
# end
|
2010-02-02 14:05:26 -05:00
|
|
|
# end
|
2010-05-15 09:08:55 -04:00
|
|
|
#
|
2010-09-30 13:17:36 -04:00
|
|
|
# == Generators
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# You can set up generators for engines with <tt>config.generators</tt> method:
|
2010-09-30 13:17:36 -04:00
|
|
|
#
|
|
|
|
# class MyEngine < Rails::Engine
|
|
|
|
# config.generators do |g|
|
|
|
|
# g.orm :active_record
|
|
|
|
# g.template_engine :erb
|
|
|
|
# g.test_framework :test_unit
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# You can also set generators for an application by using <tt>config.app_generators</tt>:
|
2010-09-30 13:17:36 -04:00
|
|
|
#
|
|
|
|
# class MyEngine < Rails::Engine
|
|
|
|
# # note that you can also pass block to app_generators in the same way you
|
|
|
|
# # can pass it to generators method
|
|
|
|
# config.app_generators.orm :datamapper
|
|
|
|
# end
|
|
|
|
#
|
2010-02-02 14:05:26 -05:00
|
|
|
# == Paths
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# Since Rails 3.0, both your application and engines do not have hardcoded paths.
|
2011-01-14 03:34:59 -05:00
|
|
|
# This means that you are not required to place your controllers at <tt>app/controllers</tt>,
|
2010-02-02 14:05:26 -05:00
|
|
|
# but in any place which you find convenient.
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# For example, let's suppose you want to place your controllers in <tt>lib/controllers</tt>.
|
2011-01-14 03:34:59 -05:00
|
|
|
# All you would need to do is:
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
|
|
|
# class MyEngine < Rails::Engine
|
2010-10-11 04:29:31 -04:00
|
|
|
# paths["app/controllers"] = "lib/controllers"
|
2010-02-02 14:05:26 -05:00
|
|
|
# end
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# You can also have your controllers loaded from both <tt>app/controllers</tt> and
|
|
|
|
# <tt>lib/controllers</tt>:
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
|
|
|
# class MyEngine < Rails::Engine
|
2010-10-11 04:29:31 -04:00
|
|
|
# paths["app/controllers"] << "lib/controllers"
|
2010-02-02 14:05:26 -05:00
|
|
|
# end
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# The available paths in an engine are:
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
|
|
|
# class MyEngine < Rails::Engine
|
2010-10-06 11:18:59 -04:00
|
|
|
# paths["app"] #=> ["app"]
|
|
|
|
# paths["app/controllers"] #=> ["app/controllers"]
|
|
|
|
# paths["app/helpers"] #=> ["app/helpers"]
|
|
|
|
# paths["app/models"] #=> ["app/models"]
|
|
|
|
# paths["app/views"] #=> ["app/views"]
|
|
|
|
# paths["lib"] #=> ["lib"]
|
|
|
|
# paths["lib/tasks"] #=> ["lib/tasks"]
|
|
|
|
# paths["config"] #=> ["config"]
|
|
|
|
# paths["config/initializers"] #=> ["config/initializers"]
|
|
|
|
# paths["config/locales"] #=> ["config/locales"]
|
|
|
|
# paths["config/routes"] #=> ["config/routes.rb"]
|
2010-02-02 14:05:26 -05:00
|
|
|
# end
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# Your <tt>Application</tt> class adds a couple more paths to this set. And as in your
|
|
|
|
# <tt>Application</tt>,all folders under +app+ are automatically added to the load path.
|
2011-01-14 03:34:59 -05:00
|
|
|
# So if you have <tt>app/observers</tt>, it's added by default.
|
2010-02-02 14:05:26 -05:00
|
|
|
#
|
2010-07-22 05:26:36 -04:00
|
|
|
# == Endpoint
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# An engine can be also a rack application. It can be useful if you have a rack application that
|
|
|
|
# you would like to wrap with +Engine+ and provide some of the +Engine+'s features.
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# To do that, use the +endpoint+ method:
|
2011-01-14 03:34:59 -05:00
|
|
|
#
|
2010-07-22 05:26:36 -04:00
|
|
|
# module MyEngine
|
|
|
|
# class Engine < Rails::Engine
|
|
|
|
# endpoint MyRackApplication
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# Now you can mount your engine in application's routes just like that:
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# MyRailsApp::Application.routes.draw do
|
|
|
|
# mount MyEngine::Engine => "/engine"
|
|
|
|
# end
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
|
|
|
# == Middleware stack
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# As an engine can now be rack endpoint, it can also have a middleware stack. The usage is exactly
|
2011-01-14 03:34:59 -05:00
|
|
|
# the same as in <tt>Application</tt>:
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
|
|
|
# module MyEngine
|
|
|
|
# class Engine < Rails::Engine
|
|
|
|
# middleware.use SomeMiddleware
|
|
|
|
# end
|
|
|
|
# end
|
|
|
|
#
|
|
|
|
# == Routes
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# If you don't specify an endpoint, routes will be used as the default endpoint. You can use them
|
|
|
|
# just like you use an application's routes:
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# # ENGINE/config/routes.rb
|
|
|
|
# MyEngine::Engine.routes.draw do
|
|
|
|
# match "/" => "posts#index"
|
|
|
|
# end
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
|
|
|
# == Mount priority
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Note that now there can be more than one router in your application, and it's better to avoid
|
|
|
|
# passing requests through many routers. Consider this situation:
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# MyRailsApp::Application.routes.draw do
|
|
|
|
# mount MyEngine::Engine => "/blog"
|
|
|
|
# match "/blog/omg" => "main#omg"
|
|
|
|
# end
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# +MyEngine+ is mounted at <tt>/blog</tt>, and <tt>/blog/omg</tt> points to application's
|
|
|
|
# controller. In such a situation, requests to <tt>/blog/omg</tt> will go through +MyEngine+,
|
|
|
|
# and if there is no such route in +Engine+'s routes, it will be dispatched to <tt>main#omg</tt>.
|
2011-01-14 03:34:59 -05:00
|
|
|
# It's much better to swap that:
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# MyRailsApp::Application.routes.draw do
|
|
|
|
# match "/blog/omg" => "main#omg"
|
|
|
|
# mount MyEngine::Engine => "/blog"
|
|
|
|
# end
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# Now, +Engine+ will get only requests that were not handled by +Application+.
|
2010-07-22 05:26:36 -04:00
|
|
|
#
|
2010-07-22 16:11:32 -04:00
|
|
|
# == Asset path
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# When you use +Engine+ with its own public directory, you will probably want to copy or symlink it
|
2011-01-14 03:34:59 -05:00
|
|
|
# to application's public directory. To simplify generating paths for assets, you can set <tt>asset_path</tt>
|
2011-01-19 15:30:28 -05:00
|
|
|
# for an engine:
|
2010-07-22 16:11:32 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# module MyEngine
|
|
|
|
# class Engine < Rails::Engine
|
|
|
|
# config.asset_path = "/my_engine/%s"
|
|
|
|
# end
|
2010-08-03 16:54:56 -04:00
|
|
|
# end
|
2010-07-22 16:11:32 -04:00
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# With such a config, asset paths will be automatically modified inside +Engine+:
|
2011-01-14 03:34:59 -05:00
|
|
|
#
|
|
|
|
# image_path("foo.jpg") #=> "/my_engine/images/foo.jpg"
|
2010-07-22 16:11:32 -04:00
|
|
|
#
|
2010-09-06 09:00:31 -04:00
|
|
|
# == Serving static files
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# By default, Rails uses <tt>ActionDispatch::Static</tt> to serve static files in development mode. This is ok
|
2011-01-19 15:30:28 -05:00
|
|
|
# while you develop your application, but when you want to deploy it, assets from an engine will not be
|
2010-10-11 04:29:31 -04:00
|
|
|
# served by default. You should choose one of the two following strategies:
|
2010-09-06 09:00:31 -04:00
|
|
|
#
|
|
|
|
# * enable serving static files by setting config.serve_static_assets to true
|
2011-01-14 03:34:59 -05:00
|
|
|
# * copy engine's public files to application's public folder with <tt>rake ENGINE_NAME:install:assets</tt>, for example
|
|
|
|
# <tt>rake my_engine:install:assets</tt>
|
2010-09-06 09:00:31 -04:00
|
|
|
#
|
2010-08-03 17:52:58 -04:00
|
|
|
# == Engine name
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# There are some places where an Engine's name is used:
|
|
|
|
# * routes: when you mount an Engine with <tt>mount(MyEngine::Engine => '/my_engine')</tt>,
|
|
|
|
# it's used as default :as option
|
|
|
|
# * some of the rake tasks are based on engine name, e.g. <tt>my_engine:install:migrations</tt>,
|
|
|
|
# <tt>my_engine:install:assets</tt>
|
2010-08-03 17:52:58 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Engine name is set by default based on class name. For <tt>MyEngine::Engine</tt> it will be
|
|
|
|
# <tt>my_engine_engine</tt>. You can change it manually it manually using the <tt>engine_name</tt> method:
|
2010-08-03 17:52:58 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# module MyEngine
|
|
|
|
# class Engine < Rails::Engine
|
|
|
|
# engine_name "my_engine"
|
|
|
|
# end
|
2010-08-03 17:52:58 -04:00
|
|
|
# end
|
|
|
|
#
|
2010-12-15 03:09:29 -05:00
|
|
|
# == Isolated Engine
|
2010-08-03 16:54:56 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Normally when you create controllers, helpers and models inside an engine, they are treated
|
|
|
|
# as they were created inside the application. This means all application helpers and named routes
|
|
|
|
# will be available to your engine's controllers.
|
2010-10-11 04:29:31 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# However, sometimes you want to isolate your engine from the application, especially if your engine
|
|
|
|
# has its own router. To do that, you simply need to call +isolate_namespace+. This method requires
|
2010-10-11 04:29:31 -04:00
|
|
|
# you to pass a module where all your controllers, helpers and models should be nested to:
|
2010-08-03 16:54:56 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# module MyEngine
|
|
|
|
# class Engine < Rails::Engine
|
2010-10-09 14:59:20 -04:00
|
|
|
# isolate_namespace MyEngine
|
2010-09-03 18:52:07 -04:00
|
|
|
# end
|
2010-08-03 16:54:56 -04:00
|
|
|
# end
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# With such an engine, everything that is inside the +MyEngine+ module will be isolated from
|
2011-01-14 03:34:59 -05:00
|
|
|
# the application.
|
2010-08-03 16:54:56 -04:00
|
|
|
#
|
|
|
|
# Consider such controller:
|
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# module MyEngine
|
|
|
|
# class FooController < ActionController::Base
|
|
|
|
# end
|
2010-08-03 16:54:56 -04:00
|
|
|
# end
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# If an engine is marked as isolated, +FooController+ has access only to helpers from +Engine+ and
|
2011-01-14 03:34:59 -05:00
|
|
|
# <tt>url_helpers</tt> from <tt>MyEngine::Engine.routes</tt>.
|
2010-08-03 16:54:56 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# The next thing that changes in isolated engines is the behaviour of routes. Normally, when you namespace
|
|
|
|
# your controllers, you also need to do namespace all your routes. With an isolated engine,
|
2010-10-11 04:29:31 -04:00
|
|
|
# the namespace is applied by default, so you can ignore it in routes:
|
|
|
|
#
|
|
|
|
# MyEngine::Engine.routes.draw do
|
|
|
|
# resources :articles
|
|
|
|
# end
|
2011-01-14 03:34:59 -05:00
|
|
|
#
|
|
|
|
# The routes above will automatically point to <tt>MyEngine::ApplicationContoller</tt>. Furthermore, you don't
|
2011-03-04 00:19:58 -05:00
|
|
|
# need to use longer url helpers like <tt>my_engine_articles_path</tt>. Instead, you should simply use
|
2011-01-14 03:34:59 -05:00
|
|
|
# <tt>articles_path</tt> as you would do with your application.
|
|
|
|
#
|
|
|
|
# To make that behaviour consistent with other parts of the framework, an isolated engine also has influence on
|
|
|
|
# <tt>ActiveModel::Naming</tt>. When you use a namespaced model, like <tt>MyEngine::Article</tt>, it will normally
|
2011-03-04 00:19:58 -05:00
|
|
|
# use the prefix "my_engine". In an isolated engine, the prefix will be omitted in url helpers and
|
2010-10-11 04:29:31 -04:00
|
|
|
# form fields for convenience.
|
2010-09-20 15:27:43 -04:00
|
|
|
#
|
|
|
|
# polymorphic_url(MyEngine::Article.new) #=> "articles_path"
|
|
|
|
#
|
|
|
|
# form_for(MyEngine::Article.new) do
|
|
|
|
# text_field :title #=> <input type="text" name="article[title]" id="article_title" />
|
|
|
|
# end
|
|
|
|
#
|
2011-03-04 00:19:58 -05:00
|
|
|
# Additionally isolated engine will set its name according to namespace, so
|
2010-10-11 04:29:31 -04:00
|
|
|
# MyEngine::Engine.engine_name #=> "my_engine". It will also set MyEngine.table_name_prefix
|
|
|
|
# to "my_engine_", changing MyEngine::Article model to use my_engine_article table.
|
2010-08-03 16:54:56 -04:00
|
|
|
#
|
2010-08-03 18:03:26 -04:00
|
|
|
# == Using Engine's routes outside Engine
|
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# Since you can now mount an engine inside application's routes, you do not have direct access to +Engine+'s
|
|
|
|
# <tt>url_helpers</tt> inside +Application+. When you mount an engine in an application's routes, a special helper is
|
2011-01-14 03:34:59 -05:00
|
|
|
# created to allow you to do that. Consider such a scenario:
|
2010-08-03 18:03:26 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# # APP/config/routes.rb
|
|
|
|
# MyApplication::Application.routes.draw do
|
|
|
|
# mount MyEngine::Engine => "/my_engine", :as => "my_engine"
|
|
|
|
# match "/foo" => "foo#index"
|
|
|
|
# end
|
2010-08-03 18:03:26 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Now, you can use the <tt>my_engine</tt> helper inside your application:
|
2010-08-03 18:03:26 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# class FooController < ApplicationController
|
|
|
|
# def index
|
|
|
|
# my_engine.root_url #=> /my_engine/
|
|
|
|
# end
|
2010-08-03 18:03:26 -04:00
|
|
|
# end
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# There is also a <tt>main_app</tt> helper that gives you access to application's routes inside Engine:
|
2010-08-03 18:03:26 -04:00
|
|
|
#
|
2010-09-03 18:52:07 -04:00
|
|
|
# module MyEngine
|
|
|
|
# class BarController
|
2010-10-11 04:29:31 -04:00
|
|
|
# def index
|
|
|
|
# main_app.foo_path #=> /foo
|
|
|
|
# end
|
2010-09-03 18:52:07 -04:00
|
|
|
# end
|
2010-08-03 18:03:26 -04:00
|
|
|
# end
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Note that the <tt>:as</tt> option given to mount takes the <tt>engine_name</tT> as default, so most of the time
|
|
|
|
# you can simply omit it.
|
2010-08-25 13:02:21 -04:00
|
|
|
#
|
2011-01-19 15:30:28 -05:00
|
|
|
# Finally, if you want to generate a url to an engine's route using <tt>polymorphic_url</tt>, you also need
|
2010-10-11 04:29:31 -04:00
|
|
|
# to pass the engine helper. Let's say that you want to create a form pointing to one of the
|
|
|
|
# engine's routes. All you need to do is pass the helper as the first element in array with
|
|
|
|
# attributes for url:
|
2010-08-25 13:02:21 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# form_for([my_engine, @user])
|
2010-08-25 13:02:21 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# This code will use <tt>my_engine.user_path(@user)</tt> to generate the proper route.
|
2010-08-25 13:02:21 -04:00
|
|
|
#
|
2010-09-19 12:47:41 -04:00
|
|
|
# == Migrations & seed data
|
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Engines can have their own migrations. The default path for migrations is exactly the same
|
|
|
|
# as in application: <tt>db/migrate</tt>
|
2010-09-19 12:47:41 -04:00
|
|
|
#
|
|
|
|
# To use engine's migrations in application you can use rake task, which copies them to
|
|
|
|
# application's dir:
|
|
|
|
#
|
2010-10-11 04:29:31 -04:00
|
|
|
# rake ENGINE_NAME:install:migrations
|
2010-09-19 12:47:41 -04:00
|
|
|
#
|
2011-01-14 03:34:59 -05:00
|
|
|
# Note that some of the migrations may be skipped if a migration with the same name already exists
|
|
|
|
# in application. In such a situation you must decide whether to leave that migration or rename the
|
2010-10-13 16:14:15 -04:00
|
|
|
# migration in application and rerun copying migrations.
|
|
|
|
#
|
2010-09-19 12:47:41 -04:00
|
|
|
# If your engine has migrations, you may also want to prepare data for the database in
|
2011-01-14 03:34:59 -05:00
|
|
|
# the <tt>seeds.rb</tt> file. You can load that data using the <tt>load_seed</tt> method, e.g.
|
2010-09-19 12:47:41 -04:00
|
|
|
#
|
|
|
|
# MyEngine::Engine.load_seed
|
|
|
|
#
|
2010-01-21 17:14:20 -05:00
|
|
|
class Engine < Railtie
|
2010-01-23 16:30:17 -05:00
|
|
|
autoload :Configuration, "rails/engine/configuration"
|
2010-11-11 11:30:24 -05:00
|
|
|
autoload :Railties, "rails/engine/railties"
|
2010-01-23 12:41:53 -05:00
|
|
|
|
2010-01-21 17:14:20 -05:00
|
|
|
class << self
|
2010-10-09 14:59:20 -04:00
|
|
|
attr_accessor :called_from, :isolated
|
2010-11-14 20:32:43 -05:00
|
|
|
alias :isolated? :isolated
|
2010-07-26 12:32:42 -04:00
|
|
|
alias :engine_name :railtie_name
|
2010-01-21 17:14:20 -05:00
|
|
|
|
|
|
|
def inherited(base)
|
2010-03-26 09:41:40 -04:00
|
|
|
unless base.abstract_railtie?
|
2010-01-23 12:41:53 -05:00
|
|
|
base.called_from = begin
|
2010-02-18 12:39:39 -05:00
|
|
|
# Remove the line number from backtraces making sure we don't leave anything behind
|
2010-10-15 10:50:55 -04:00
|
|
|
call_stack = caller.map { |p| p.sub(/:\d+.*/, '') }
|
2010-10-15 10:31:00 -04:00
|
|
|
File.dirname(call_stack.detect { |p| p !~ %r[railties[\w.-]*/lib/rails|rack[\w.-]*/lib/rack] })
|
2010-01-23 12:41:53 -05:00
|
|
|
end
|
2010-01-21 17:14:20 -05:00
|
|
|
end
|
2010-01-23 12:41:53 -05:00
|
|
|
|
2010-01-21 17:14:20 -05:00
|
|
|
super
|
|
|
|
end
|
|
|
|
|
2010-07-08 06:07:25 -04:00
|
|
|
def endpoint(endpoint = nil)
|
|
|
|
@endpoint = endpoint if endpoint
|
|
|
|
@endpoint
|
|
|
|
end
|
2010-07-30 01:14:48 -04:00
|
|
|
|
2010-10-09 14:59:20 -04:00
|
|
|
def isolate_namespace(mod)
|
2010-08-03 16:36:12 -04:00
|
|
|
engine_name(generate_railtie_name(mod))
|
|
|
|
|
|
|
|
name = engine_name
|
2010-09-01 13:31:29 -04:00
|
|
|
self.routes.default_scope = {:module => name}
|
2010-10-09 14:59:20 -04:00
|
|
|
self.isolated = true
|
2010-09-25 13:02:14 -04:00
|
|
|
|
|
|
|
unless mod.respond_to?(:_railtie)
|
|
|
|
_railtie = self
|
|
|
|
mod.singleton_class.instance_eval do
|
|
|
|
define_method(:_railtie) do
|
|
|
|
_railtie
|
|
|
|
end
|
|
|
|
|
2010-11-20 20:20:09 -05:00
|
|
|
unless mod.respond_to?(:table_name_prefix)
|
|
|
|
define_method(:table_name_prefix) do
|
|
|
|
"#{name}_"
|
|
|
|
end
|
2010-09-25 13:02:14 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2010-08-23 13:25:04 -04:00
|
|
|
end
|
2010-12-08 17:15:13 -05:00
|
|
|
|
|
|
|
# Finds engine with given path
|
|
|
|
def find(path)
|
2011-03-02 18:17:29 -05:00
|
|
|
path = path.to_s
|
|
|
|
Rails::Engine::Railties.engines.find { |r|
|
|
|
|
File.expand_path(r.root.to_s) == File.expand_path(path)
|
|
|
|
}
|
2010-12-08 17:15:13 -05:00
|
|
|
end
|
2010-01-21 17:14:20 -05:00
|
|
|
end
|
|
|
|
|
2010-07-19 11:53:14 -04:00
|
|
|
delegate :middleware, :root, :paths, :to => :config
|
2010-10-09 14:59:20 -04:00
|
|
|
delegate :engine_name, :isolated?, :to => "self.class"
|
2010-01-23 11:51:48 -05:00
|
|
|
|
|
|
|
def load_tasks
|
2010-02-28 17:39:01 -05:00
|
|
|
super
|
2010-10-06 11:18:59 -04:00
|
|
|
paths["lib/tasks"].existent.sort.each { |ext| load(ext) }
|
2010-01-23 11:51:48 -05:00
|
|
|
end
|
|
|
|
|
2010-05-15 17:48:56 -04:00
|
|
|
def eager_load!
|
|
|
|
config.eager_load_paths.each do |load_path|
|
|
|
|
matcher = /\A#{Regexp.escape(load_path)}\/(.*)\.rb\Z/
|
|
|
|
Dir.glob("#{load_path}/**/*.rb").sort.each do |file|
|
|
|
|
require_dependency file.sub(matcher, '\1')
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-06-23 10:10:26 -04:00
|
|
|
def railties
|
2010-07-19 11:53:14 -04:00
|
|
|
@railties ||= self.class::Railties.new(config)
|
2010-06-23 10:10:26 -04:00
|
|
|
end
|
|
|
|
|
2010-06-22 17:51:28 -04:00
|
|
|
def app
|
2010-07-05 11:40:35 -04:00
|
|
|
@app ||= begin
|
|
|
|
config.middleware = config.middleware.merge_into(default_middleware_stack)
|
|
|
|
config.middleware.build(endpoint)
|
|
|
|
end
|
2010-06-22 17:51:28 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def endpoint
|
2010-06-24 08:32:11 -04:00
|
|
|
self.class.endpoint || routes
|
2010-06-22 17:51:28 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
def call(env)
|
2010-07-22 16:11:32 -04:00
|
|
|
app.call(env.merge!(env_config))
|
|
|
|
end
|
|
|
|
|
|
|
|
def env_config
|
|
|
|
@env_config ||= {
|
|
|
|
'action_dispatch.routes' => routes,
|
|
|
|
'action_dispatch.asset_path' => config.asset_path
|
|
|
|
}
|
2010-06-22 17:51:28 -04:00
|
|
|
end
|
|
|
|
|
2010-10-06 09:54:28 -04:00
|
|
|
def routes
|
2010-06-24 08:32:11 -04:00
|
|
|
@routes ||= ActionDispatch::Routing::RouteSet.new
|
2010-10-06 09:54:28 -04:00
|
|
|
@routes.append(&Proc.new) if block_given?
|
2010-09-25 18:17:06 -04:00
|
|
|
@routes
|
2010-06-24 08:32:11 -04:00
|
|
|
end
|
|
|
|
|
2010-06-23 17:02:02 -04:00
|
|
|
def initializers
|
|
|
|
initializers = []
|
|
|
|
railties.all { |r| initializers += r.initializers }
|
|
|
|
initializers += super
|
|
|
|
initializers
|
|
|
|
end
|
|
|
|
|
2010-07-19 11:53:14 -04:00
|
|
|
def config
|
2010-07-29 10:23:15 -04:00
|
|
|
@config ||= Engine::Configuration.new(find_root_with_flag("lib"))
|
2010-07-19 11:53:14 -04:00
|
|
|
end
|
|
|
|
|
2010-09-19 11:21:16 -04:00
|
|
|
# Load data from db/seeds.rb file. It can be used in to load engines'
|
|
|
|
# seeds, e.g.:
|
|
|
|
#
|
|
|
|
# Blog::Engine.load_seed
|
|
|
|
def load_seed
|
2010-10-06 11:18:59 -04:00
|
|
|
seed_file = paths["db/seeds"].existent.first
|
2010-09-19 11:21:16 -04:00
|
|
|
load(seed_file) if File.exist?(seed_file)
|
|
|
|
end
|
|
|
|
|
2010-01-21 17:14:20 -05:00
|
|
|
# Add configured load paths to ruby load paths and remove duplicates.
|
2010-05-15 09:08:55 -04:00
|
|
|
initializer :set_load_path, :before => :bootstrap_hook do
|
2010-06-27 18:57:47 -04:00
|
|
|
_all_load_paths.reverse_each do |path|
|
2010-01-22 10:24:44 -05:00
|
|
|
$LOAD_PATH.unshift(path) if File.directory?(path)
|
|
|
|
end
|
2010-01-21 17:14:20 -05:00
|
|
|
$LOAD_PATH.uniq!
|
|
|
|
end
|
|
|
|
|
|
|
|
# Set the paths from which Rails will automatically load source files,
|
|
|
|
# and the load_once paths.
|
2010-05-15 09:08:55 -04:00
|
|
|
#
|
|
|
|
# This needs to be an initializer, since it needs to run once
|
|
|
|
# per engine and get the engine as a block parameter
|
|
|
|
initializer :set_autoload_paths, :before => :bootstrap_hook do |app|
|
2010-06-27 18:57:47 -04:00
|
|
|
ActiveSupport::Dependencies.autoload_paths.unshift(*_all_autoload_paths)
|
2010-09-03 18:52:07 -04:00
|
|
|
ActiveSupport::Dependencies.autoload_once_paths.unshift(*_all_autoload_once_paths)
|
2010-01-21 17:14:20 -05:00
|
|
|
|
2010-01-22 19:29:29 -05:00
|
|
|
# Freeze so future modifications will fail rather than do nothing mysteriously
|
2010-06-22 17:17:20 -04:00
|
|
|
config.autoload_paths.freeze
|
2010-06-27 18:57:47 -04:00
|
|
|
config.eager_load_paths.freeze
|
2010-06-22 17:17:20 -04:00
|
|
|
config.autoload_once_paths.freeze
|
2010-01-21 17:14:20 -05:00
|
|
|
end
|
|
|
|
|
2010-01-27 11:39:35 -05:00
|
|
|
initializer :add_routing_paths do |app|
|
2010-10-06 11:18:59 -04:00
|
|
|
paths = self.paths["config/routes"].existent
|
|
|
|
|
|
|
|
if routes? || paths.any?
|
|
|
|
app.routes_reloader.paths.unshift(*paths)
|
2010-10-06 09:54:28 -04:00
|
|
|
app.routes_reloader.route_sets << routes
|
2010-01-22 19:29:29 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-01-25 16:00:07 -05:00
|
|
|
# I18n load paths are a special case since the ones added
|
|
|
|
# later have higher priority.
|
2010-01-22 19:29:29 -05:00
|
|
|
initializer :add_locales do
|
2010-10-06 11:18:59 -04:00
|
|
|
config.i18n.railties_load_path.concat(paths["config/locales"].existent)
|
2010-01-21 17:14:20 -05:00
|
|
|
end
|
|
|
|
|
2010-01-21 19:10:31 -05:00
|
|
|
initializer :add_view_paths do
|
2010-10-06 11:18:59 -04:00
|
|
|
views = paths["app/views"].existent
|
2010-09-03 18:52:07 -04:00
|
|
|
unless views.empty?
|
|
|
|
ActiveSupport.on_load(:action_controller){ prepend_view_path(views) }
|
|
|
|
ActiveSupport.on_load(:action_mailer){ prepend_view_path(views) }
|
2010-03-29 20:08:08 -04:00
|
|
|
end
|
2010-01-21 19:10:31 -05:00
|
|
|
end
|
|
|
|
|
2010-07-18 17:23:54 -04:00
|
|
|
initializer :load_environment_config, :before => :load_environment_hook do
|
2010-10-06 11:18:59 -04:00
|
|
|
environment = paths["config/environments"].existent.first
|
2010-07-18 17:23:54 -04:00
|
|
|
require environment if environment
|
|
|
|
end
|
|
|
|
|
2010-09-01 05:07:08 -04:00
|
|
|
initializer :append_asset_paths do
|
2010-12-09 15:18:41 -05:00
|
|
|
config.asset_path ||= default_asset_path
|
2010-08-04 16:22:06 -04:00
|
|
|
|
2010-10-06 11:18:59 -04:00
|
|
|
public_path = paths["public"].first
|
2010-08-04 16:22:06 -04:00
|
|
|
if config.compiled_asset_path && File.exist?(public_path)
|
|
|
|
config.static_asset_paths[config.compiled_asset_path] = public_path
|
|
|
|
end
|
|
|
|
end
|
2010-08-23 13:25:04 -04:00
|
|
|
|
2010-10-06 11:18:59 -04:00
|
|
|
initializer :prepend_helpers_path do |app|
|
2010-10-09 14:59:20 -04:00
|
|
|
if !isolated? || (app == self)
|
2010-10-06 11:18:59 -04:00
|
|
|
app.config.helpers_paths.unshift(*paths["app/helpers"].existent)
|
2010-08-23 13:25:04 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-09-02 06:09:16 -04:00
|
|
|
initializer :load_config_initializers do
|
2010-10-06 11:18:59 -04:00
|
|
|
config.paths["config/initializers"].existent.sort.each do |initializer|
|
2010-09-02 06:09:16 -04:00
|
|
|
load(initializer)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-09-01 05:07:39 -04:00
|
|
|
initializer :engines_blank_point do
|
|
|
|
# We need this initializer so all extra initializers added in engines are
|
|
|
|
# consistently executed after all the initializers above across all engines.
|
|
|
|
end
|
|
|
|
|
2010-10-09 14:15:45 -04:00
|
|
|
rake_tasks do
|
|
|
|
next if self.is_a?(Rails::Application)
|
|
|
|
|
|
|
|
namespace railtie_name do
|
2010-10-13 16:01:46 -04:00
|
|
|
desc "Shortcut for running both rake #{railtie_name}:install:migrations and #{railtie_name}:install:assets"
|
|
|
|
task :install do
|
|
|
|
Rake::Task["#{railtie_name}:install:migrations"].invoke
|
|
|
|
Rake::Task["#{railtie_name}:install:assets"].invoke
|
|
|
|
end
|
|
|
|
|
2010-10-09 14:15:45 -04:00
|
|
|
namespace :install do
|
2010-10-11 04:29:31 -04:00
|
|
|
# TODO Add assets copying to this list
|
|
|
|
# TODO Skip this if there is no paths["db/migrate"] for the engine
|
2010-10-09 14:15:45 -04:00
|
|
|
desc "Copy migrations from #{railtie_name} to application"
|
|
|
|
task :migrations do
|
|
|
|
ENV["FROM"] = railtie_name
|
|
|
|
Rake::Task["railties:install:migrations"].invoke
|
|
|
|
end
|
2010-10-10 09:03:57 -04:00
|
|
|
|
2010-10-13 09:20:09 -04:00
|
|
|
desc "Copy assets from #{railtie_name} to application"
|
|
|
|
task :assets do
|
|
|
|
ENV["FROM"] = railtie_name
|
|
|
|
Rake::Task["railties:install:assets"].invoke
|
|
|
|
end
|
2010-10-10 09:03:57 -04:00
|
|
|
end
|
2010-10-09 14:15:45 -04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2010-01-23 12:41:53 -05:00
|
|
|
protected
|
2010-12-09 15:18:41 -05:00
|
|
|
|
|
|
|
def default_asset_path
|
|
|
|
"/#{railtie_name}%s"
|
|
|
|
end
|
|
|
|
|
2010-10-06 11:18:59 -04:00
|
|
|
def routes?
|
|
|
|
defined?(@routes)
|
|
|
|
end
|
|
|
|
|
2010-07-19 11:53:14 -04:00
|
|
|
def find_root_with_flag(flag, default=nil)
|
|
|
|
root_path = self.class.called_from
|
|
|
|
|
|
|
|
while root_path && File.directory?(root_path) && !File.exist?("#{root_path}/#{flag}")
|
|
|
|
parent = File.dirname(root_path)
|
|
|
|
root_path = parent != root_path && parent
|
|
|
|
end
|
|
|
|
|
|
|
|
root = File.exist?("#{root_path}/#{flag}") ? root_path : default
|
|
|
|
raise "Could not find root path for #{self}" unless root
|
|
|
|
|
2010-10-05 03:48:32 -04:00
|
|
|
RbConfig::CONFIG['host_os'] =~ /mswin|mingw/ ?
|
2010-07-19 11:53:14 -04:00
|
|
|
Pathname.new(root).expand_path : Pathname.new(root).realpath
|
|
|
|
end
|
|
|
|
|
2010-07-08 06:07:25 -04:00
|
|
|
def default_middleware_stack
|
|
|
|
ActionDispatch::MiddlewareStack.new
|
|
|
|
end
|
2010-01-23 12:41:53 -05:00
|
|
|
|
2010-09-03 18:52:07 -04:00
|
|
|
def _all_autoload_once_paths
|
|
|
|
config.autoload_once_paths
|
|
|
|
end
|
|
|
|
|
2010-06-27 18:57:47 -04:00
|
|
|
def _all_autoload_paths
|
|
|
|
@_all_autoload_paths ||= (config.autoload_paths + config.eager_load_paths + config.autoload_once_paths).uniq
|
|
|
|
end
|
|
|
|
|
|
|
|
def _all_load_paths
|
|
|
|
@_all_load_paths ||= (config.paths.load_paths + _all_autoload_paths).uniq
|
2010-01-23 12:41:53 -05:00
|
|
|
end
|
2010-01-21 17:14:20 -05:00
|
|
|
end
|
2010-02-26 16:29:44 -05:00
|
|
|
end
|