Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Extension integrating with Rails #36

Merged
merged 1 commit into from
Nov 15, 2019
Merged

Extension integrating with Rails #36

merged 1 commit into from
Nov 15, 2019

Conversation

waiting-for-dev
Copy link
Owner

The first two things to keep in mind in order to integrate with Rails is
that WebPipe instances are Rack applications and that rails router can
perfectly dispatch to a rack application. For example:

 # config/routes.rb
get '/my_route', to: MyRoute.new

 # app/controllers/my_route.rb
class MyRoute
  include WebPipe

  plug :set_response_body

  private

  def set_response_body(conn)
    conn.set_response_body('Hello, World!')
  end
end

In order to do something like the previous example you don't need to enable
this extension. Notice that rails took care of dispatching the request to our
WebPipe rack application, which was then responsible for generating the
response. In this case, it used a simple call to #set_response_body.

It's quite possible that you don't need more than that in terms of rails
integration. Of course, surely you want something more elaborate to generate
responses. For that, you can use the view or template system you like. One
option that will play specially well here is dry-view, which
integrates
itself easily with Rails. Furthermore, we have a tailored dry_view extension.

You need to use this extension if:

  • You want to use action_view as rendering system.
  • You want to use rails url helpers from your WebPipe application.
  • You want to use controller helpers from your WebPipe application.

Rails responsibilities for controlling the request/response cycle and the
rendering process are a little bit tangled. For this reason, even if you
want to use WebPipe applications instead of Rails controller actions you
still have to use the typical top ApplicationController in order to define
some behaviour for the view layer:

  • Which layout is applied to the template.
  • Which helpers will become available to the templates.
  • Where within app/views/ templates are looked up.

By default, the controller in use is ActionController::Base, which means that
no layout is applied and only built-in helpers (for example,
number_as_currency) are available. You can change it via the
:rails_controller configuration option.

The main method that this extension adds to WebPipe::Conn is #render,
which just delegates to the Rails
implementation

as you'd do in a typical rails controller. Remember that you can provide
template instance variables through the keyword :assigns.

 # config/routes.rb
get '/articles', to: ArticlesIndex.new

 # app/controllers/application_controller.rb
class ApplicationController < ActionController::Base
  # By default uses the layout in `layouts/application`
end

 # app/controllers/articles_index.rb
require 'web_pipe/plugs/config'

WebPipe.load_extensions(:rails) # You can put it in an initializer

class ArticlesIndex
  include WebPipe

  plug :config, WebPipe::Plugs::Config.(
    rails_controller: ApplicationController
  )

  def render(conn)
    conn.render(
      template: 'articles/index',
      assigns: { articles: Article.all }
    )
  end
end

Notice that we used the keyword template: instead of taking advantage of
automatic template lookup. We did that way so that we don't have to create also
an ArticlesController, but it's up to you.

Besides, this extension provides with two other methods:

In all the examples we have supposed that we are putting WebPipe applications
within app/controllers/ directory. However, remember you can put them
wherever you like as long as you respect rails autoload_paths.

Here you have a link to a very simple and contrived example of a rails
application integrating web_pipe:

https://github.com/waiting-for-dev/rails-web_pipe

The first two things to keep in mind in order to integrate with Rails is
that `WebPipe` instances are Rack applications and that rails router can
perfectly [dispatch to a rack application](https://guides.rubyonrails.org/routing.html#routing-to-rack-applications). For example:

```ruby
 # config/routes.rb
get '/my_route', to: MyRoute.new

 # app/controllers/my_route.rb
class MyRoute
  include WebPipe

  plug :set_response_body

  private

  def set_response_body(conn)
    conn.set_response_body('Hello, World!')
  end
end
```

In order to do something like the previous example you don't need to enable
this extension. Notice that rails took care of dispatching the request to our
`WebPipe` rack application, which was then responsible for generating the
response. In this case, it used a simple call to `#set_response_body`.

It's quite possible that you don't need more than that in terms of rails
integration. Of course, surely you want something more elaborate to generate
responses. For that, you can use the view or template system you like. One
option that will play specially well here is
[`dry-view`](https://dry-rb.org/gems/dry-view/), which
[integrates](https://github.com/dry-rb/dry-view/tree/master/examples/rails)
itself easily with Rails. Furthermore, we have a tailored `dry_view`
[extension](https://waiting-for-dev.github.io/web_pipe/docs/extensions/dry_view.html).

You need to use this extension if:

- You want to use `action_view` as rendering system.
- You want to use rails url helpers from your `WebPipe` application.
- You want to use controller helpers from your `WebPipe` application.

Rails responsibilities for controlling the request/response cycle and the
rendering process are a little bit tangled. For this reason, even if you
want to use `WebPipe` applications instead of Rails controller actions you
still have to use the typical top `ApplicationController` in order to define
some behaviour for the view layer:

- Which layout is applied to the template.
- Which helpers will become available to the templates.
- Where within `app/views/` templates are looked up.

By default, the controller in use is `ActionController::Base`, which means that
no layout is applied and only built-in helpers (for example,
`number_as_currency`) are available. You can change it via the
`:rails_controller` configuration option.

The main method that this extension adds to `WebPipe::Conn` is `#render`,
which just delegates to the [Rails
implementation](https://api.rubyonrails.org/v6.0.1/classes/ActionController/Renderer.html)
as you'd do in a typical rails controller. Remember that you can provide
template instance variables through the keyword `:assigns`.

```ruby
 # config/routes.rb
get '/articles', to: ArticlesIndex.new

 # app/controllers/application_controller.rb
class ApplicationController < ActionController::Base
  # By default uses the layout in `layouts/application`
end

 # app/controllers/articles_index.rb
require 'web_pipe/plugs/config'

WebPipe.load_extensions(:rails) # You can put it in an initializer

class ArticlesIndex
  include WebPipe

  plug :config, WebPipe::Plugs::Config.(
    rails_controller: ApplicationController
  )

  def render(conn)
    conn.render(
      template: 'articles/index',
      assigns: { articles: Article.all }
    )
  end
end
```

Notice that we used the keyword `template:` instead of taking advantage of
automatic template lookup. We did that way so that we don't have to create also
an `ArticlesController`, but it's up to you.

Besides, this extension provides with two other methods:

- `url_helpers` returns Rails router [url
  helpers](https://api.rubyonrails.org/v6.0.1/classes/ActionView/Helpers/UrlHelper.html).
- `helpers` returns the associated [controller
  helpers](https://api.rubyonrails.org/classes/ActionController/Helpers.html).

In all the examples we have supposed that we are putting `WebPipe` applications
within `app/controllers/` directory. However, remember you can put them
wherever you like as long as you respect rails [`autoload_paths`](https://guides.rubyonrails.org/autoloading_and_reloading_constants.html#autoload-paths).

Here you have a link to a very simple and contrived example of a rails
application integrating `web_pipe`:

https://github.com/waiting-for-dev/rails-web_pipe
@waiting-for-dev waiting-for-dev added the enhancement New feature or request label Nov 15, 2019
@waiting-for-dev waiting-for-dev merged commit ffd134f into master Nov 15, 2019
@waiting-for-dev waiting-for-dev deleted the rails branch November 15, 2019 18:29
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
enhancement New feature or request
Projects
None yet
Development

Successfully merging this pull request may close these issues.

1 participant