Integrate Stripe Checkout with Rails app - ruby-on-rails

I have an existing Rails application where users can create classifieds/ads for boats. Currently there is no payment integrated and once user submits the form, it gets posted. Now I am looking to add Spripe Checkout (as it seems the easiest, correct me if i'm wrong) to my app. I am not sure what would be the best way to approach this. Ideally I would like the user to submit the form and then be asked to pay.
Also, I am not sure which approach I should take: 1) the ad gets created in the database first and then with the payment a Boolean column value (for example "payment") gets changed or 2) the ad's database record gets created only with a successful payment.
Should I create a new Charges controller like Stripe tutorial suggests or incorporate it in my existing AdsController?
ads_controller.rb
class AdsController < ApplicationController
before_action :logged_in_user, only: [:edit, :update, :new]
before_action :correct_user, only: [:edit, :update, :destroy]
def index
if params[:category].blank? && params[:state].blank?
#ads = Ad.all.order("created_at DESC")
elsif params[:category].blank?
#state_id = State.find_by(name: params[:state]).id
#ads = Ad.where(state_id: #state_id).order("created_at DESC")
else
#category_id = Category.find_by(name: params[:category]).id
#ads = Ad.where(category_id: #category_id).order("created_at DESC")
end
#ads = #ads.paginate(page: params[:page], :per_page => 21)
end
def new
#ad = Ad.new
end
def create
#ad = current_user.ads.build(ad_params)
if #ad.save
flash[:success] = "Ad successfully created"
redirect_to #ad
else
render 'new'
end
end
def show
#ad = Ad.find(params[:id])
end
def edit
#ad = Ad.find(params[:id])
end
def update
#ad = Ad.find(params[:id])
if #ad.update_attributes(ad_params)
flash[:success] = "Ad successfully updated"
redirect_to #ad
else
render 'edit'
end
end
def destroy
#ad.destroy
flash[:success] = "Ad deleted"
redirect_to request.referrer || root_url
end
private
def ad_params
params.require(:ad).permit(:title, :price, :year, :location, :description, :contact_name, :contact_number, :category_id, :picture, :picture2, :picture3, :state_id)
end
def correct_user
#ad = current_user.ads.find_by(id: params[:id])
redirect_to root_url if #ad.nil?
end
def logged_in_user
unless logged_in?
flash[:danger] = "Please log in"
redirect_to login_path
end
end
my current form:
<div class="row">
<div class="col-md-6 col-md-offset-3">
<%= form_for(#ad, html: { multipart: true }) do |f| %>
<%= render 'shared/error_messages', object: f.object %>
<%= f.label "Category:" %>
<%= f.collection_select :category_id, Category.all, :id, :name, { prompt: "Select Category" }, class: 'form-control' %>
# Lots of other form fields
<%= f.submit "Submit", class: "btn btn-primary" %>
<% end %>
</div>
Any thoughts on this will be highly appreciated.

It would be more railsesque to create a separate controller than build it into the ads controller. You should definitely wrap the entire db process in what is known as a 'transaction' and look to save the add only on successful payment. If any part fails throughout the 'transaction' process it will roll back your database and maintain conformity.
The jstripe tutorials are very in depth but if you're looking for more information and specific examples I read a great book I bought on this subject that walked me around a lot of the pitfalls and gotchas you may run into:
https://www.masteringmodernpayments.com/

Related

Display User with "most completed" in Views using Rails counter_cache

I am building my first RoR application and it involves Assignments with Tasks. The user has the ability to Create Assignments, add Tasks to those assignments, and also mark assignments complete. I have an "Accomplishments" page that show cases a featured User with "most completed" assignments (User that has more than 5 assignments completed). Where my problem lies is that the User doesn't change as the assignments completed count does. Currently I only have 2 users. User1 has completed 8 assignments, User2 has completed 7 assignments. User2 is being displayed as the "Featured User with most completed assignments".
My code is as follows:
Assignments Controller
class AssignmentsController < ApplicationController
before_action :authenticate_user!, except: [:index, :show]
before_action :set_assignment, only: [:show, :edit, :update,
:destroy]
#GET/ASSIGNMENTS
def index
user = User.find params[:user_id]
#incomplete_assignments = user.assignments.incomplete
#complete_assignments = user.assignments.complete
end#index
def new
#assignment = current_user.assignments.build
#assignment.tasks.build
end#new
#POST
def create
#assignment = current_user.assignments.build(assignment_params)
if #assignment.save
redirect_to [current_user, #assignment], notice: "Assignment was created successfully!"
else
render :new
end
end #create
def show
##user = current_user.assignments
#task = Task.new
end#show
#PATCH
def edit
end
def update
#assignment.update(assignment_params)
redirect_to [current_user, #assignment], notice: "Assignment updated successfully!"
end#update
#DELETE
def destroy
#assignment.destroy
redirect_to [current_user, #assignment], notice: "Assignment was deleted successfully!"
end #destroy
def completed
Assignment.where(id: params[:assignment_ids]).update_all(status: true)
redirect_to user_assignments_path(current_user.id)
end#completed
private
#STRONG PARAMS
def set_assignment
#assignment = Assignment.find(params[:id])
end
def assignment_params
params.require(:assignment).permit(:name, :due_date, task_attributes: [:name])
end
end#class
Users Controller
class UsersController < ApplicationController
before_action :authenticate_user!
def show
#user = User.find(params[:id])
#assignment = Assignment.new
redirect_to user_path(#user)
end
def show_completed
#users = User.all
end
end
My view with Accomplishments
<h3> Status of who has done what</h3>
<br>
<% for user in #users %>
<%= user.email %> has completed <%= user.assignments.count %> assignments. <br>
<% end %>
<% if user.assignments.complete.count > 5 %>
<h1> **Featured User** </h1>
<h2><%= user.email %> has completed the most assignments!</h2>
<% else %>
<p> Not enough completed Assignments to be featured!</p>
<% end %>
<%= link_to 'Back to your asssignments', user_assignments_path(current_user, #assignment) %>
This may be a matter of a misunderstanding on how to properly use .count in a method, or something as simple as that in my View where I say display the user with assignments greater than 5 my application is doing just that.

Nested Ticket Edit Action Comes Up Blank Inside Project

I am building a rails app to learn about nested controllers actions. Hence I have Projects to which Tickets are nested underneath. It's basically a rip-off from Rails Action. Now, I am trying to add edit action in Ticket's controller.
Everything works fine in Projects, ie, all the actions.
Everything seems to work fine in Tickets. As stated earlier, it's a rip-off from Rails Action. So trying learn how to put in correct stuff within blank actions, ie, Edit, Show, Update, and Destroy
When I added edit action in Tickets show page, and clicked on it, it showed the page, but the form fields are blank.
The url path is correct: http://127.0.0.1:3000/projects/2/tickets/2/edit so that means the resources are set up correctly.
How do I fix this so it shows the previous content? My vague understanding is that it may have to do with the form?
Current Tickets Controller
class TicketsController < ApplicationController
#which actions are necessary? learn how to not need set_project set_ticket
before_action :set_project
before_action :set_ticket, only: [:show, :edit, :update, :destroy]
def create
#ticket = #project.tickets.build(ticket_params)
if #ticket.save
flash[:notice] = "Ticket has been created."
redirect_to project_path(#project)
else
flash.now[:alert] = "Ticket has not been created."
render "new"
end
end
def new
#ticket = #project.tickets.build
end
def show
end
def edit
#ticket = #project.tickets.find(params[:id])
end
def update
end
def destroy
end
private
def ticket_params
params.require(:ticket).permit(:title, :description)
end
def set_project
#project = Project.find(params[:project_id])
end
def set_ticket
#ticket = #project.tickets.find(params[:id])
end
end
show.html.erb - Ticket
<h1><%= #project.title %></h1>
<h3><%= #ticket.title %></h3>
<%= link_to "All Projects", projects_path %>
<%= link_to "Edit Ticket", edit_project_ticket_path %>
_form.html.erb - Ticket
<%= form_for([#project, #project.tickets.build]) do |f| %>
<p>
<%= f.label :title %><br/>
<%= f.text_field :title %>
</p>
<p>
<%= f.label :description %><br />
<%= f.text_area :description %>
</p>
<p>
<%= f.submit %>
</p>
<% end %>
EDIT, I've updated the _form.html.erb to the following code beneath, and it seems to work, ie, it shows the contents to modify. Yay. But when I click on Update Ticket button, it doesn't move beyond the page.
<%= form_for [#project,#ticket] do |f| %>
EDIT AGAIN
I updated the ticket's controller actions for Show and Update, and now it seems to work! Including it for feedback in case this was done incorrectly.
class TicketsController < ApplicationController
#which actions are necessary? learn how to not need set_project set_ticket
before_action :set_project
before_action :set_ticket, only: [:show, :edit, :update, :destroy]
def create
#ticket = #project.tickets.build(ticket_params)
if #ticket.save
flash[:notice] = "Ticket has been created."
redirect_to project_path(#project)
else
flash.now[:alert] = "Ticket has not been created."
render "new"
end
end
def new
#ticket = #project.tickets.build
end
def show
#ticket = #project.tickets.find(params[:id])
end
def edit
#ticket = #project.tickets.find(params[:id])
end
def update
if #ticket.update(ticket_params)
flash[:notice] = "Ticket has been updated."
redirect_to [#project, #ticket]
else
flash.now[:alert] = "Ticket has not been updated."
render "edit"
end
end
def destroy
end
private
def ticket_params
params.require(:ticket).permit(:title, :description)
end
def set_project
#project = Project.find(params[:project_id])
end
def set_ticket
#ticket = #project.tickets.find(params[:id])
end
end

Prefill form fields as values in Rails

TL;DR Is it possible to prefill from one model in the same form builder to another model? Is there a correct way of solving this problem?
I have a form that takes several methods that requires to be passed in: name, email, bio, location, homepage, work.
Email method will need to be retrieved via Devise user model.
In my new.html.erb, it works fine as normal but it returns nil from the email field whenever I do the following:
<%= form_for :profile, url: profiles_path do |f| %>
...
<div><% if current_user.email.present? %><%= f.label :email %><br>
<%= f.email_field current_user.email %><% end %></div>
<% end %>
ProfilesController.rb:
class ProfilesController < ApplicationController
before_action :authenticate_user!, except: [:show]
before_action :find_profile_by_id, only: [:show, :edit, :update]
after_filter :destroy_user!, only: [:destroy]
def index
if #profile.blank?
render :new
else
render :show
end
end
def new
#profile = current_user.profiles.build
#user = current_user.where('email=?')
end
def create
#profile = current_user.profiles.build(profile_params)
if #profile.save
redirect_to #profile
else
render :new
end
end
def show
end
def edit
end
def update
if #profile.update(profile_params)
redirect_to #profile
else
render :edit
end
end
def destroy
#user.destroy
#profile.destroy
redirect_to root_path
end
private
def profile_params
params.require(:profile).permit(:name, :email, :bio, :location, :homepage, :work)
end
def find_profile_by_id
#profile = Profile.find(params[:id])
end
def destroy_user!
#user = User.find(params[:id])
end
end
Is this the correct solution for this?
Your approach seems generally okay but try changing
<%= f.email_field current_user.email %>
To
<%= f.email_field(current_user, current_user.email) %>

Rails - Updating and deleting content?

EDIT: I managed to delete! i had to define teh instance variable #movies = Movie.find(params[:id]) to the delete method in the controller.
I still can't update though. I get "param is missing or the value is empty: movie"
I forgot to add my contrller! sorry!
I'm trying to replicate one of my in class exercises into another app, for practice.
The idea is to be able to add new movies into a database, and get the option to update their info, and delete them as well. I can add new content, but I can't update them or delete them.
Appreciate any inputs I can get.
Routes.rb
Rails.application.routes.draw do
root "movies#index"
get "/movies", to: "movies#index"
get "/movies/new", to: "movies#new", as: "new_movie"
get '/movies/:id', to: "movies#movie_page", as: "movie_page"
post "/movies", to: "movies#add"
get "/movies/:id/edit", to: "movies#edit", as: "movie_edit"
put "/movies/:id", to: "movies#update"
patch "/movies/:id", to: "movies#update", as: "update_movie"
delete "/movies/:id", to: "movies#delete", as: "delete_movie"
end
Controller
class MoviesController < ApplicationController
def index
#movies = Movie.all
end
def movie_page
#movies = Movie.find(params[:id])
end
def new
#movies = Movie.new
end
def add
#movies = Movie.create(movie_params)
redirect_to movies_path
end
def edit
#movies = Movie.find(params[:id])
end
def update
#movies.update(movie_params)
redirect_to #movies, notice: "Shirt was updated."
end
def delete
#movies = Movie.find(params[:id])
#movies.destroy
# flash[:notice] = "Shirt was deleted."
redirect_to root_path, notice: "Shirt was deleted."
end
def movie_params
params.require(:movie).permit(:title, :description, :year_released)
end
# def set_movie
# #movies = Movie.find(params[:id])
# end
end
Form partial
<%= form_for #movies do |m| %>
<p>
<%= m.label :title %><br>
<%= m.text_field :title %>
</p>
<p>
<%= m.label :description %><br>
<%= m.text_field :description %>
</p>
<p>
<%= m.label :year_released %><br>
<%= m.text_field :year_released %>
</p>
<p>
<%= m.submit %>
</p>
<% end %>
Movie page html (individual movies, labeled by IDs)**I can't update or Delete, no route matches Delete.
When I press Update - I get param is missing or the value is empty: movie
<h1><%= #movies.title %></h1>
<h2>Released on : <%= #movies.year_released %> </h2>
<p> <%= #movies.description %> </p>
<%= link_to "Update", movie_edit_path(#movies) %>
<%= link_to "Delete", movies_path, method: :delete %
Edit page *I cant access this link. the form is the problem
<h1>Edit <%= #movies.title %> Info </h1>
<%= render "form" %>
<%= link_to "Cancel Edit", movie_edit_path(#movies) %>
Many thanks guys
def update
#movie = Move.find(params[:id])
#movie.update(movie_params)
redirect_to movie_path(#movie)
end
on your routes. all you need is resources :movies
you are getting param is empty because you have to pass in the id of the movie to update.
The major issue is that you do not load the variable #movies from the DB before you use it.
def update
#movies.update(movie_params)
redirect_to #movies, notice: "Shirt was updated."
end
def update
#movies.find(params[:id])
#movie.update(movie_params)
redirect_to #movies, notice: "Shirt was updated."
end
Aside from that you have tons of duplication and quite a few idiosyncrasies.
Rails uses these naming conventions for actions:
index
show (not movie_page)
new
create (not add)
edit
update
destroy (not delete)
You should follow them unless you have a damn good reason not to.
class MoviesController < ApplicationController
# cuts the duplication
before_filter :set_movie, except: [:new, :index]
def index
#movies = Movie.all
end
# GET /movies/:id
def show
end
# GET /movies/new
def new
#movie = Movie.new
end
# POST /movies
def create
#movie = Movie.create(movie_params)
redirect_to movies_path
end
# GET /movies/edit
def edit
end
# PUT|PATCH /movies/:id
def update
#movie.update(movie_params)
redirect_to #movie, notice: "Shirt was updated."
end
# DELETE /movies/:id
def destroy
#movie.destroy
redirect_to action: :index
end
private
def movie_params
params.require(:movie).permit(:title, :description, :year_released)
end
def set_movie
# Use the singular form when naming a variable with a single record
# failure to do so may result in tarring and feathering
#movie = Movie.find(params[:id])
end
end

Edit nested in a form

Thanks to help on stackoverflow I got my create nested models form working the other day but I can't for the life of me get the corresponding update form to work. I have read a lot and tried out as many solutions as I can find.
The form looks fine but the nested attributes of Manufacturer and Scale, which are select via drop down, don't have their current values. All non nested elements of the form work fine.
Whatever changes you make to the two nested drop downs, pressing save changes creates NEW lines in the corresponding tables and doesn't alter the existing.
Ultimately what I want is for the attributes to be editable and then i'll have an "add manufacturer" and "add scale" button or link for miniatures that need multiple listings.
Here are my form fields where I've tried and failed to pass a hidden field.
Form
<%= render 'shared/error_messages', object: f.object %>
<%= f.label :name %>
<%= f.text_field :name %>
<%= f.label :material %>
<%= f.select 'material', options_from_collection_for_select(Miniature.select("DISTINCT material"), :material, 'material', #miniature.material) %>
<%= f.fields_for :sizes do |size_fields| %>
<%= size_fields.label :scale_id, "Scale".pluralize %>
<%= hidden_field "Miniature Scale", #miniature.sizes %>
<%= size_fields.select :scale_id, options_from_collection_for_select(Scale.all, :id, :name) %>
<% end %>
<%= f.fields_for :productions do |production_fields| %>
<%= production_fields.label :manufacturer_id, "Manufacturer".pluralize %>
<%= hidden_field "Miniature Manufacturer", #miniature.productions %>
<%= production_fields.select :manufacturer_id, options_from_collection_for_select(Manufacturer.all, :id, :name, #miniature.manufacturers) %>
<% end %>
<%= f.label :release_date %>
<%= f.date_select :release_date, :start_year => Date.current.year, :end_year => 1970, :include_blank => true %>
Here is the miniatures controller where I'm pretty sure I've filled the 'def update' with too much/the wrong stuff.
Miniatures Controller
class MiniaturesController < ApplicationController
before_action :signed_in_user, only: [:new, :create, :edit, :update]
before_action :admin_user, only: :destroy
def show
#miniature = Miniature.find(params[:id])
end
def new
#miniature = Miniature.new
#miniature.productions.build
#miniature.sizes.build
end
def create
#miniature = Miniature.new(miniature_params)
#production = #miniature.productions.build
#size = #miniature.sizes.build
if #miniature.save
redirect_to #miniature
else
render 'new'
end
end
def edit
#miniature = Miniature.find(params[:id])
end
def update
#miniature = Miniature.find(params[:id])
#production = #miniature.productions.find(params[:id])
#size = #miniature.sizes.find(params[:id])
if #miniature.update_attributes(miniature_params)
#production = #miniature.productions.update_attributes(:manufacturer_id)
#size = #miniature.sizes.update_attributes(:scale_id)
flash[:success] = "Miniature updated"
redirect_to #miniature
else
render 'edit'
end
end
def index
#miniatures = Miniature.paginate(page: params[:page])
end
def destroy
Miniature.find(params[:id]).destroy
flash[:success] = "Miniature destroyed."
redirect_to miniatures_url
end
private
def miniature_params
params.require(:miniature).permit(:name, :release_date, :material, productions_attributes: [:manufacturer_id], sizes_attributes: [:scale_id])
end
def admin_user
redirect_to(root_url) unless current_user.admin?
end
def signed_in_user
unless signed_in?
store_location
redirect_to signin_url, notice: "Please sign in."
end
end
end
I won't attach the models as I'm pretty sure the relationships are all correct since they work fine for creating new nested models.
Miniatures have_many scales and manufactures through sizes and productions.
Any help or pointers very much appreciated.
Thanks to an answer on this Q I've solved it. What I had already was fine for CREATING but didn't work for UPDATES because I hadn't whitelisted the JOIN model ids in the 'miniature_params, so they couldn't retrieve the existing info.
Now I have productions_attributes: [:id, :manufacturer_id] instead of just productions_attributes: [:manufacturer_id]
as below
def miniature_params
params.require(:miniature).permit(:name, :release_date, :material, productions_attributes: [:id, :manufacturer_id], sizes_attributes: [:id, :scale_id])
end
I can also strip ALL of the references to the nested models out of my miniatures controller update method as it 'just works'.
def update
#miniature = Miniature.find(params[:id])
if #miniature.update_attributes(miniature_params)
flash[:success] = "Miniature updated"
redirect_to #miniature
else
render 'edit'
end
end
Hope this is useful to someone in future.

Resources