Create a User SignUp Form With Email Using Ruby on Rails
Last Updated :
23 Jul, 2025
Ruby on Rails, often just called Rails, is a web application framework written in the Ruby programming language. It was created by David Heinemeier Hansson and was first released in 2005. Rails are known for its "Convention over Configuration" approach, which means that it provides sensible defaults for many aspects of web development, allowing developers to focus on writing the code specific to their application. It also has a strong emphasis on "DRY" (Don't Repeat Yourself) principles, which encourage the reuse of code and promote a more maintainable and scalable application structure.
The article focuses on discussing the steps to create a user signup form using the Ruby on Rails framework. The signup form allows users to signup by entering their email id and by creating a password. Access to the data of signed-up users can be enabled through the backend.
Prerequisites
These are the following prerequisites that should be preinstalled on your system:
- Ruby (Version - 3.1.2).
- Node (Version - 12.22.9).
- Yarn (Version - 1.22.9).
- Rails (Version - 7.0.4.2).
Implementation
Model-view-controller Architecture will be used for creating the form. Follow the steps below to create a signup form:
Step 1: Open the terminal and start a new rails application.
$ rails new User_Signup
This process will create all the necessary files and folders in the directory and will install all the essential gems for the application. In Ruby on Rails, a gem is a packaged library or plugin that provides additional functionality to a Rails application. It can extend the core framework, add new features, or simplify common development tasks.
Run the rails server from the terminal to start the development server.
$ rails s
Open http://localhost:3000/ in your browser and it will show a page like this:
Step 2: Goto config/routes.rb and add some routes such as root route and sign_up route.
A route in Ruby on Rails is a URL pattern that maps to a specific action in a controller. The routes file (config/routes.rb) is where you define the mapping between URLs and the corresponding actions in your controllers.
root to: "home#index"
get "sign_up", to: "user#new"
post "sign_up", to: "user#create"
In Ruby on Rails, the get and post methods are used to specify the HTTP request method in routing. get method is used to retrieve a resource and display it on the page. post method is used to submit data to the server to create a new resource.
Step 3: Generate a home controller from the terminal.
A controller is a central part of the application that handles user requests and manages the flow of data between the model and the view. Each controller in Rails corresponds to a specific set of related actions and is named after the model it works with. For example, you might have a PostsController for working with blog posts or a UsersController for working with user accounts.
rails generate controller Home
This generated action will create a file named home_controller.rb in app/controllers and a folder named the home in app/views. Now, Open app/controllers/home_controller.rb and add the index method to it
def index
end
There is no need of writing anything to the index method because we're just going to create our home page.
Step 4: In the folder app/views/home create a new file named index.html.erb and add the following HTML code to it.
<h1> This is Home Page </h1>
Run the rails server again and this time when you will open http://localhost:3000/ the page will be like as shown below.
Step 5: An user will sign up through the form. So, we have to create a model user from the terminal.
A Model is a class that represents a specific table in the database and is used to interact with the data stored in that table. Models are the "M" in the MVC (Model-View-Controller) architecture that Rails follows and are responsible for representing and manipulating the data in the application. For example, you might have a 'Post' model that represents a single blog post, or a 'User' model that represents a user account.
$ rails generate model User
This process will create a file user.rb in app/models and a migration file in db/migrate.
Open the migration file of create_users and add two columns to the user's table email and password_digest.
t.string :email, null: false
t.string :password_digest
After this run the following command:rails db:migrate
The rails db:migrate command is used to run database migrations. A migration in Rails is a way to make changes to the database schema, such as creating new tables, adding columns, or modifying existing columns. Migrations are stored as files in the db/migrate directory and each migration file corresponds to a specific set of changes to the database.
And add some validation for an email in app/models/user.rb and for authentication, we have to show that a user will contain its secure password.
The "valid_email_regex" is a regular expression used to validate the format of an email address in a model. This regex pattern can be used in the validation method to ensure that the email entered by a user matches the correct format before being saved to the database. The default pattern used in Ruby on Rails is the /\A[\w+\-.]+@[a-z\d\-.]+\.[a-z]+\z/i regex.
has_secure_password
VALID_EMAIL_REGEX = /\A[\w+\-.]+@[a-z\d\-.]+\.[a-z]+\z/i
validates :email, presence: true,
uniqueness: { case_sensitive: false },
length: { maximum: 105 },
format: { with: VALID_EMAIL_REGEX }
Step 7: Install bcrypt gem. Goto gemfile and activate the bcrypt gem
gem "bcrypt", "~> 3.1.7"
And after the activation run bundle install to install the gem. bcrypt is a Ruby gem used for securely hashing and storing passwords. Bcrypt is a recommended method for storing passwords because it is slow and provides a salt to protect against dictionary attacks and rainbow table attacks.
$ bundle install
Step 7: Generate a controller user from the terminal.
rails generate controller User
Go to app/controllers/user_controller.rb and add a new and create method to it.
def new
@user = User.new
end
def create
@user = User.new(user_params)
if @user.save
redirect_to root_path, notice: "Successfully created account"
else
render :new
end
end
private
def user_params
params.require(:user).permit(:email, :password, :password_confirmation)
end
In Ruby on Rails, "params" is a special object that holds the parameters that are passed from a web browser to the server, either through the query string of a URL or through form data. The "params" hash allows you to access the values of these parameters in your Rails controller actions and views.
Step 8: Goto the app/views and create a new file new.html.erb in the user folder and write the following code for what is going to be displayed on the signup page.
<h1> Sign Up </h1>
<%= form_with model: @user, url: sign_up_path do |form| %>
<div class="mb-3">
<%= form.label :email, "Email" %>
<%= form.text_field :email, class: "form-control",
placeholder: "Enter your email address" %>
</div>
<div class="mb-3">
<%= form.label :password, "Password"%>
<%= form.password_field :password, class: "form-control",
placeholder: "Enter your password" %>
</div>
<div class="mb-3">
<%= form.label :password_confirmation, "Confirm password" %>
<%= form.password_field :password_confirmation, class: "form-control",
placeholder: "Please enter the password again" %>
</div>
<div class="mb-3">
<%= form.submit "Sign Up", class: "btn btn-primary" %>
</div>
<% end %>
Save all the changes and now open http://localhost:3000/sign_up in the browser and it will look like this:
This signup page doesn't look good.
Step 9: For a better look and feel of the page, add some Bootstrap styling to the application. Bootstrap is a free front-end framework for faster and easier web development. It provides pre-designed HTML, CSS, and JavaScript components, such as buttons, forms, navigation, typography, and more, which can be easily customized and used to build responsive, mobile-first websites.
To activate bootstrap in our application go to app/views/layouts/application.html.erb and add the following code to it.
<link href="https://cdn.jsdelivr.net/npm/[email protected]/dist/css/bootstrap.min.css" rel="stylesheet" crossorigin="anonymous">
<script src="https://cdn.jsdelivr.net/npm/[email protected]/dist/js/bootstrap.bundle.min.js" crossorigin="anonymous"></script>
And add some styling to app/vies/user/new.html.erb
<div class="container">
<div class="row justify-content-center">
<div class="col-5">
<div class="mt-3">
<h1> Sign Up </h1>
</div>
<%= form_with model: @user, url: sign_up_path do |form| %>
<div class="form-group row mt-3">
<%= form.label :email, "Email" %>
<%= form.text_field :email, class: "form-control",
placeholder: "Enter your email address" %>
</div>
<div class="form-group row mt-3">
<%= form.label :password, "Password"%>
<%= form.password_field :password, class: "form-control",
placeholder: "Enter your password" %>
</div>
<div class="form-group row mt-3">
<%= form.label :password_confirmation, "Confirm password" %>
<%= form.password_field :password_confirmation, class: "form-control",
placeholder: "Please enter the password again" %>
</div>
<div class=" form-group row mt-3">
<%= form.submit "Sign Up", class: "btn btn-primary" %>
</div>
<% end %>
</div>
</div>
</div>
Refresh the page and now it the page will look good as shown below.
Step 10: Setup flash messages to show alerts and notices.
Make a new folder 'shared' in app/views and in the shared folder create a partial named _flash.html.erb. Write the below codes in _flash.html.erb.
<% if flash[:notice] %>
<div class="alert alert-info mt-4">
<%= flash[:notice] %>
</div>
<% end %>
<% if flash[:alert] %>
<div class="alert alert-warning mt-4">
<%= flash[:alert] %>
</div>
<% end %>
To access these flash messages go to app/views/layouts/application.html.erb and write the below line in <boody> element.
<%= render partial: "shared/flash" %>
Step 11: Creating Our first user.
Start the rails server and refresh the signup page and choose an email and enter the password.
As you will hit the Sign Up button the user will be created and you will be at your homepage with notice as shown below.
Step 12: Checking whether the user is saved or not.
Open the terminal and run the following command:
$ rails console
OR
$ rails c
Something like below will open up.
And at that point, if you will write User.all it will show all the saved users.
irb(main):001:0> User.all
As you can see there are two users. I created the first one for testing.
Step 13: Showing the errors.
When we will hit signup button without filling in the email or if there is a mismatch of passwords or the email is not according to the validation it will show the errors. Go to app/views/user/new.html.erb add the following lines
<% if @user.errors.any?%>
<div class="alert alert-danger">
<% @user.errors.full_messages.each do |message| %>
<div><%= message %></div>
<% end %>
</div>
<% end %>
When you try to hit signup button some errors following will happen.
If you will try to sign up without entering any data.
If you will try to sign up with an email that is not according to the validation.
If you will try to sign up with not matching passwords.That's it. You've created a Signup page using Rails successfully.
Similar Reads
Ruby Programming Language Ruby is a dynamic, reflective, object-oriented, general-purpose programming language. Ruby is a pure Object-Oriented language developed by Yukihiro Matsumoto. Everything in Ruby is an object except the blocks but there are replacements too for it i.e procs and lambda. The objective of Rubyâs develop
2 min read
Overview
Ruby For BeginnersRuby is a dynamic, reflective, object-oriented, general-purpose programming language. It was designed and developed in the mid-1990s by Yukihiro "Matz" Matsumoto in Japan. This article will cover its basic syntax and some basic programs. This article is divided into various sections for various topi
3 min read
Ruby Programming Language (Introduction)Ruby is a pure Object-Oriented language developed by Yukihiro Matsumoto (also known as Matz in the Ruby community) in the mid 1990âs in Japan. Everything in Ruby is an object except the blocks but there are replacements too for it i.e procs and lambda. The objective of Ruby's development was to make
4 min read
Comparison of Java with Other Programming LanguagesJava is one of the most popular and widely used programming languages and platforms. A platform is an environment that helps to develop and run programs written in any programming language. Java is fast, reliable, and secure. From desktop to web applications, scientific supercomputers to gaming cons
4 min read
Similarities and Differences between Ruby and C languageSimilarities between Ruby and C There are many similarities between C and Ruby, and some of them are: Like C, in Ruby also⦠A programmer is able to program procedurally if they like to do. But still, behind the scenes, it will be object-oriented.Both the languages have the same operators, for exampl
3 min read
Similarities and Differences between Ruby and C++There are many similarities between C++ and Ruby, some of them are: Just like C++, in Ruby⦠As in C++, public, private, and protected works similarly in Ruby also .Inheritance syntax is still only one character, but itâs < instead of : in Ruby.The way ânamespaceâ is used in C++, in the similar wa
3 min read
Environment Setup in RubyRuby is an interpreted, high-level, general-purpose programming language. Ruby is dynamically typed and uses garbage collection. It supports multiple programming paradigms, object-oriented, including procedural and functional programming. Ruby is based on many other languages like Perl, Lisp, Smallt
3 min read
How to install Ruby on Linux?Prerequisite: Ruby Programming Language Before we start with the installation of Ruby on Linux, we must have first-hand knowledge of what Ruby is?. Ruby is a pure Object-Oriented language developed by Yukihiro Matsumoto (also known as Matz in the Ruby community) in the mid-1990s in Japan. Everything
2 min read
How to install Ruby on Windows?Prerequisite: Ruby Programming Language Before we start with the installation of Ruby on Windows, we must have first-hand knowledge of what Ruby is?. Ruby is a pure Object-Oriented language developed by Yukihiro Matsumoto (also known as Matz in the Ruby community) in the mid-1990s in Japan. Everythi
2 min read
Interesting facts about Ruby Programming LanguageRuby is an interpreted, high-level, dynamic, general-purpose, open source programming language which focuses on simplicity and productivity. It was designed and developed in the mid-1990s by Yukihiro Matsumoto (also known as Matz in the Ruby community) in Japan. Here are some interesting facts about
2 min read
Basics
Ruby | KeywordsKeywords or Reserved words are the words in a language that are used for some internal process or represent some predefined actions. These words are therefore not allowed to use as variable names or objects or as constants. Doing this may result in compile-time error. Example: Ruby # Ruby program to
4 min read
Ruby | Data TypesData types in Ruby represents different types of data like text, string, numbers, etc. All data types are based on classes because it is a pure Object-Oriented language. There are different data types in Ruby as follows: NumbersBooleanStringsHashesArraysSymbols Numbers: Generally a number is defined
3 min read
Ruby Basic SyntaxRuby is a pure Object-Oriented language developed by Yukihiro Matsumoto (also known as Matz in the Ruby community) in the mid 1990âs in Japan. To program in Ruby is easy to learn because of its similar syntax to already widely used languages. Here, we will learn the basic syntax of Ruby language. Le
3 min read
Hello World in RubyRuby is a dynamic, reflective, object-oriented, general-purpose programming language. Hello World the program is the most basic and first program when we start a new programming language. This simply prints Hello World on the screen. Below is the program to write hello world". How to run a Ruby Prog
2 min read
Ruby | Types of VariablesThere are different types of variables in Ruby: Local variables Instance variables Class variables Global variables Each variable in Ruby is declared by using a special character at the start of the variable name which is mentioned in the following table: Symbol Type of Variable [a-z] or _ Local Var
4 min read
Global Variable in RubyGlobal Variable has global scope and accessible from anywhere in the program. Assigning to global variables from any point in the program has global implications. Global variable are always prefixed with a dollar sign ($). If we want to have a single variable, which is available across classes, we n
2 min read
Comments in RubyStatements that are not executed by the compiler and interpreter are called Comments. During coding proper use of comments makes maintenance easier and finding bugs easily.In Ruby, there are two types of comments:Â Â Single â line comments.Multi â line comments. Here, we are going to explain both typ
2 min read
Ruby | RangesPrerequisite: Ruby Range Operator Ruby ranges depict a set of values with a beginning and an end. Values of a range can be numbers, characters, strings or objects. It is constructed using start_point..end_point, start_point...endpoint literals, or with ::new. It provides the flexibility to the code
4 min read
Ruby LiteralsAny constant value which can be assigned to the variable is called as literal/constant. we use literal every time when typing an object in the ruby code. Ruby Literals are same as other programming languages, just a few adjustments, and differences here. These are following literals in Ruby. Boolean
4 min read
Ruby DirectoriesA directory is a location where files can be stored. For Ruby, the Dir class and the FileUtils module manages directories and the File class handles the files. Double dot (..) refers to the parent directory for directories and single dot(.)refers to the directory itself.The Dir Class The Dir class p
5 min read
Ruby | OperatorsAn operator is a symbol that represents an operation to be performed with one or more operand. Operators are the foundation of any programming language. Operators allow us to perform different kinds of operations on operands. There are different types of operators used in Ruby as follows: Arithmetic
11 min read
Operator Precedence in RubyOperators are used to perform different kinds of operations on operands. Which operator is performed first in an expression with more than one operators with different precedence is determined by operator precedence. when two operators of the same precedence appear in expression associativity is use
2 min read
Operator Overloading in RubyRuby permits operator overloading, allowing one to define how an operator shall be used in a particular program. For example a '+' operator can be define in such a way to perform subtraction instead addition and vice versa. The operators that can be overloaded are +, -, /, *, **, %, etc and some ope
5 min read
Ruby | Pre-define Variables & ConstantsRuby Predefine Variables Ruby contains a wide range of predefined variables. Every predefined variable has its own specification. You can use predefine variables to perform a specific task like when dealing with interpreter parameters or regular expressions. The list of predefined variables in Ruby
5 min read
Ruby | unless Statement and unless ModifierRuby provides a special statement which is referred as unless statement. This statement is executed when the given condition is false. It is opposite of if statement. In if statement, the block executes once the given condition is true, however in unless statement, the block of code executes once th
2 min read
Control Statements
Ruby | Decision Making (if, if-else, if-else-if, ternary) | Set - 1Decision Making in programming is similar to decision making in real life. In programming too, a certain block of code needs to be executed when some condition is fulfilled. A programming language uses control statements to control the flow of execution of the program based on certain conditions. Th
3 min read
Ruby | Loops (for, while, do..while, until)Looping is a fundamental concept in programming that allows for the repeated execution of a block of code based on a condition. Ruby, being a flexible and dynamic language, provides various types of loops that can be used to handle condition-based iterations. These loops simplify tasks that require
5 min read
Ruby | Case StatementThe case statement is a multiway branch statement just like a switch statement in other languages. It provides an easy way to forward execution to different parts of code based on the value of the expression. There are 3 important keywords which are used in the case statement: case: It is similar to
3 min read
Ruby | Control Flow AlterationPrerequisite : Decision Making , Loops Ruby programming language provides some statements in addition to loops, conditionals, and iterators, which are used to change the flow of control in a program. In other words, these statements are a piece of code that executes one after another until the condi
7 min read
Ruby Break and Next StatementIn Ruby, we use a break statement to break the execution of the loop in the program. It is mostly used in while loop, where value is printed till the condition, is true, then break statement terminates the loop. Syntax : Break Example : Ruby # Ruby program to use break statement #!/usr/bin/ruby -w i
2 min read
Ruby redo and retry StatementIn Ruby, Redo statement is used to repeat the current iteration of the loop. redo always used inside the loop. The redo statement restarts the loop without evaluating the condition again. Ruby # Ruby program of using redo statement #!/usr/bin/ruby restart = false # Using for loop for x in 2..20 if x
2 min read
BEGIN and END Blocks In RubyEvery Ruby source file can run as the BEGIN blocks when the file is being loaded and runs the END blocks after the program has finished executing. The BEGIN and END statements are different from each other. A program may contain multiple BEGIN and END blocks. If there is more than one BEGIN statemen
2 min read
File Handling in RubyIt is a way of processing a file such as creating a new file, reading content in a file, writing content to a file, appending content to a file, renaming the file and deleting the file. Common modes for File Handling "r" : Read-only mode for a file. "r+" : Read-Write mode for a file. "w" : Write-onl
4 min read
Methods
OOP Concepts
Object-Oriented Programming in Ruby | Set 1When we say object-oriented programming, we mean that our code is centered on objects. Objects are real-life instances that are classified into various types. Letâs take an example to understand this better. If we consider a rose as an object, then the class of the rose will be flower. A class is li
9 min read
Object Oriented Programming in Ruby | Set-2Prerequisite: Object Oriented Programming in Ruby | Set-1 Inheritance Inheritance is one of the solid fundamental characteristics of object-oriented programming. sometimes we might need certain features of a class to be replicated into another class. Instead of creating that attribute again, we can
8 min read
Ruby | Class & ObjectRuby is an ideal object-oriented programming language. The features of an object-oriented programming language include data encapsulation, polymorphism, inheritance, data abstraction, operator overloading etc. In object-oriented programming classes and objects plays an important role. A class is a b
4 min read
Private Classes in RubyThe concept of private, protected and public methods in Ruby is a bit different than it other languages like Java. In Ruby, it is all about which class the person is calling, as classes are objects in ruby. Private Class When a constant is declared private in Ruby, it means this constant can never b
3 min read
Freezing Objects | RubyAny object can be frozen by invoking Object#freeze. A frozen object can not be modified: we can't change its instance variables, we can't associate singleton methods with it, and, if it is a class or module, we can't add, delete, or modify its methods. To test if an object is frozen we can use Objec
2 min read
Ruby | InheritanceRuby is the ideal object-oriented language. In an object-oriented programming language, inheritance is one of the most important features. Inheritance allows the programmer to inherit the characteristics of one class into another class. Ruby supports only single class inheritance, it does not suppor
4 min read
Polymorphism in RubyIn Ruby, one does not have anything like the variable types as there is in other programming languages. Every variable is an "object" which can be individually modified. One can easily add methods and functions on every object. So here, the Object Oriented Programming plays a major role. There are m
3 min read
Ruby | ConstructorsA constructor is a special method of the class that gets automatically invoked whenever an instance of the class is created. Like methods, a constructor may also contain a group of instructions or a method that will execute at the time of object creation. Important points to remember about Construct
2 min read
Ruby | Access ControlAccess control is a very important part of the object-oriented programming language which is used to restrict the visibility of methods and member fields to protect data from the accidental modification. In terms of access control, Ruby is different from all other Object Oriented Programming languag
8 min read
Ruby | EncapsulationEncapsulation is defined as the wrapping up of data under a single unit. It is the mechanism that binds together code and the data it manipulates. In a different way, encapsulation is a protective shield that prevents the data from being accessed by the code outside this shield. Technically in encap
2 min read
Ruby MixinsBefore studying about Ruby Mixins, we should have the knowledge about Object Oriented Concepts. If we don't, go through Object Oriented Concepts in Ruby . When a class can inherit features from more than one parent class, the class is supposed to have multiple inheritance. But Ruby does not support
3 min read
Instance Variables in RubyThere are four different types of variables in Ruby- Local variables, Instance variables, Class variables and Global variables. An instance variable in ruby has a name starting with @ symbol, and its content is restricted to whatever the object itself refers to. Two separate objects, even though the
3 min read
Data Abstraction in RubyThe idea of representing significant details and hiding details of functionality is called data abstraction. The interface and the implementation are isolated by this programming technique. Data abstraction is one of the object oriented programming features as well. Abstraction is trying to minimize
3 min read
Ruby Static MembersIn Programming, static keywords are primarily used for memory management. The static keyword is used to share the same method or variable of a class across the objects of that class. There are various members of a class in Ruby. Once an object is created in Ruby, the methods and variables for that o
3 min read
Exceptions
Ruby | ExceptionsA good program(or programmer) predict error and arrange to handle them in an effective manner. This is not as easy as it sounds. Exceptions are the errors that occur at runtime. It halts the execution of a program. They are caused by an extensive variety of exceptional circumstances, such as running
4 min read
Ruby | Exception handlingIn Ruby, exception handling is a process which describes a way to handle the error raised in a program. Here, error means an unwanted or unexpected event, which occurs during the execution of a program, i.e. at run time, that disrupts the normal flow of the program's instructions. So these types of
6 min read
Catch and Throw Exception In RubyAn exception is an object of class Exception or a child of that class. Exceptions occurs when the program reaches a state in its execution that's not defined. Now the program does not know what to do so it raises an exception. This can be done automatically by Ruby or manually. Catch and Throw is si
3 min read
Raising Exceptions in RubyAn exception is an unwanted or unexpected event, which occurs during the execution of a program i.e at runtime, that disrupts the normal flow of the programâs instructions. As we know, the code enclosed between begin and end block is totally secured for handling Exceptions and the rescue block tells
4 min read
Ruby | Exception Handling in Threads | Set - 1Threads can also contain exceptions. In Ruby threads, the only exception arose in the main thread is handled but if an exception arises in the thread(other than main thread) cause the termination of the thread. The arising of an exception in a thread other than the main thread depends upon abort_on_
2 min read
Ruby | Exception Class and its MethodsAn exception is an unwanted or unexpected event, which occurs during the execution of a program, i.e. at runtime, that disrupts the normal flow of the programâs instructions. In Ruby, descendants of an Exception class are used to interface between raise methods and rescue statements in the begin or
3 min read
Ruby Regex
Ruby Classes
Ruby Module
Ruby | ModuleA Module is a collection of methods, constants, and class variables. Modules are defined as a class, but with the module keyword not with class keyword. Important Points about Modules: You cannot inherit modules or you can't create a subclass of a module. Objects cannot be created from a module. Mod
4 min read
Ruby | Comparable ModuleIn Ruby, the mixin of Comparable is used by the class whose objects may be ordered. The class must be defined using an operator which compare the receiver against another object. It will return -1, 0, and 1 depending upon the receiver. If the receiver is less than another object, then it returns -1,
3 min read
Ruby | Math ModuleIn Ruby, Modules are defined as a collection of methods, classes, and constants together. Math module consists of the module methods for basic trigonometric and transcendental functions. Module ConstantsNameDescriptionEDefine the value of base of natural logarithm e.PIDefine the value of Ï. Example:
4 min read
Include v/s Extend in RubyInclude is used to importing module code. Ruby will throw an error when we try to access the methods of import module with the class directly because it gets imported as a subclass for the superclass. So, the only way is to access it through the instance of the class. Extend is also used to importin
2 min read