Devise
Devise is one of the most popular gems for authentication. Find out more about Devise here.
In order to integrate it fully in Matestack apps and pages we need to adjust a few things. This guide explains what exactly needs to be adjusted.
Devise helpers
We can access devise helper methods inside our controllers, apps, pages and components like we would normally do. In case of our user model this means we could access current_user
or user_signed_in?
in apps, pages and components.
For example:
In our controller we also use devise like it is described by devise documentation. For example checking a user is authenticated before he can access a specific controller by calling authenticate_user!
in a before action.
Devise sign in
Using the default devise sign in views should work without a problem, but they will not be integrated with a Matestack app. Let's assume we have a profile Matestack app called Profile::App
. If we want to take advantage of Matestack's transitions features (not reloading our app layout between page transitions) we can not use devise views, because we would need to redirect to them and therefore need to reload the whole page. Requiring us for example to implement our navigation twice. In our Profile::App
and also in our devise sign in view.
Therefore we need to adjust a few things and create some pages. First we create a custom sign in page containing a form with email and password inputs.
app/matestack/profile/pages/sessions/sign_in.rb
This page displays a form with an email and password input. The default required parameters for a devise sign in. It also contains a toggle
component which gets shown when the event sign_in_failure
is emitted. This event gets emitted in case our form submit was unsuccessful as we specified it in our form_config
hash. If the form is successful our app will make a transition to the page the server would redirect to.
In order to render our sign in page when someone tries to access a route which needs authentication or visits the sign in page we must override devise session controller in order to render our page. We do this by configuring our routes to use a custom controller.
app/config/routes.rb
Override the new
action in order to render our sign in page and set the correct Matestack app in the controller. Also remember to include the components registry. This is necessary if you use custom components in your app or page, because without it Matestack can't resolve them.
app/controllers/users/sessions_controller.rb
Now our sign in is nearly complete. Logging in with correct credentials works fine, but logging in with incorrect credentials triggers a page reload and doesn't show our error message.
Devise usually responds with a 401 for wrong credentials but intercepts this response and redirects to the new action. This means our form
component recieves the response of the new
action, which would have a success status. Therefore it redirects you resulting in a rerendering of the sign in page. So our form
component needs to recieve a error code in order to work as expected. To achieve this we need to provide a custom failure app.
Create the custom failure app under lib/devise/json_failure_app.rb
containing following code:
We only want to overwrite the behavior of the failure app for request with application/json
as content type, setting the status to a 401 unauthorized error and the content_type to json.
There is only one thing left, telling devise to use our custom failure app. Therefore add/update the following lines in config/initializers/devise.rb
.
That's it. When we now try to sign in with incorrect credentials the matestack_form
component triggers the sign_in_failure
event, which sets off our toggle
component resulting in displaying the error message.
Wrap Up That's it. Now you have a working sign in with devise fully integrated into Matestack. All we needed to do was creating a sign in page, updating our routes to use a custom session controller, overriding the new action, creating a custom failure app and updating the devise config.
Devise sign out
Creating a sign out button in Matestack is very straight forward. We use matestacks action
component to create a sign out button. See the example below:
Notice the method: :get
in the configuration hash. We use a http GET request to sign out, because the browser will follow the redirect send from devise session controller and then Matestack tries to load the page where we have been redirected to. When we would use a DELETE request the action we would be redirected to from the browser will be also requested with a http DELETE request, which will result in a rails routing error. Therefore we use GET and need to configure devise accordingly by changing the sign_out_via
configuration parameter.
That's all we have to do.
Last updated