12.4 Playtime 12.1 Iteration G1: Capturing an Order
Demonstrate various respond_to/format options, as well as "through" relations and basic authentication.
Define a "who_bought" member action
edit app/controllers/products_controller.rb
def who_bought
@product = Product.find(params[:id])
@latest_order = @product.orders.order(:updated_at).last
if stale?(:etag => @latest_order, :last_modified => @latest_order.created_at.utc)
respond_to do |format|
format.atom
end
end
end
Define an Atom view (using the Atom builder)
edit app/views/products/who_bought.atom.builder
atom_feed do |feed|
feed.title "Who bought #{@product.title}"
feed.updated @latest_order.try(:updated_at)
@product.orders.each do |order|
feed.entry(order) do |entry|
entry.title "Order #{order.id}"
entry.summary :type => 'xhtml' do |xhtml|
xhtml.p "Shipped to #{order.address}"
xhtml.table do
xhtml.tr do
xhtml.th 'Product'
xhtml.th 'Quantity'
xhtml.th 'Total Price'
end
order.line_items.each do |item|
xhtml.tr do
xhtml.td item.product.title
xhtml.td item.quantity
xhtml.td number_to_currency item.total_price
end
end
xhtml.tr do
xhtml.th 'total', :colspan => 2
xhtml.th number_to_currency \
order.line_items.map(&:total_price).sum
end
end
xhtml.p "Paid by #{order.pay_type}"
end
entry.author do |author|
author.name order.name
author.email order.email
end
end
end
end
Add "orders" to the Product class
edit app/models/product.rb
class Product < ActiveRecord::Base
has_many :line_items
has_many :orders, :through => :line_items
#...
end
Add to the routes
edit config/routes.rb
Depot::Application.routes.draw do
resources :orders
resources :line_items
resources :carts
get "store/index"
resources :products do
get :who_bought, :on => :member
end
# The priority is based upon order of creation:
# first created -> highest priority.
# Sample of regular route:
# match 'products/:id' => 'catalog#view'
# Keep in mind you can assign values other than :controller and :action
# Sample of named route:
# match 'products/:id/purchase' => 'catalog#purchase', :as => :purchase
# This route can be invoked with purchase_url(:id => product.id)
# Sample resource route (maps HTTP verbs to controller actions automatically):
# resources :products
# Sample resource route with options:
# resources :products do
# member do
# get 'short'
# post 'toggle'
# end
#
# collection do
# get 'sold'
# end
# end
# Sample resource route with sub-resources:
# resources :products do
# resources :comments, :sales
# resource :seller
# end
# Sample resource route with more complex sub-resources
# resources :products do
# resources :comments
# resources :sales do
# get 'recent', :on => :collection
# end
# end
# Sample resource route within a namespace:
# namespace :admin do
# # Directs /admin/products/* to Admin::ProductsController
# # (app/controllers/admin/products_controller.rb)
# resources :products
# end
# You can have the root of your site routed with "root"
# just remember to delete public/index.html.
root :to => 'store#index', :as => 'store'
# ...
end
Fetch the Atom feed
curl --max-time 15 --silent --user dave:secret http://localhost:3000/products/2/who_bought.atom
<?xml version="1.0" encoding="UTF-8"?>
<feed xml:lang="en-US" xmlns="http://www.w3.org/2005/Atom">
<id>tag:localhost,2005:/products/2/who_bought</id>
<link type="text/html" href="http://localhost:3000" rel="alternate"/>
<link type="application/atom+xml" href="http://localhost:3000/products/2/who_bought.atom" rel="self"/>
<title>Who bought CoffeeScript</title>
<updated>2014-02-04T19:43:26Z</updated>
<entry>
<id>tag:localhost,2005:Order/1</id>
<published>2014-02-04T19:43:26Z</published>
<updated>2014-02-04T19:43:26Z</updated>
<link type="text/html" href="http://localhost:3000/orders/1" rel="alternate"/>
<title>Order 1</title>
<summary type="xhtml">
<div xmlns="http://www.w3.org/1999/xhtml">
<p>Shipped to 123 Main St</p>
<table>
<tr>
<th>Product</th>
<th>Quantity</th>
<th>Total Price</th>
</tr>
<tr>
<td>CoffeeScript</td>
<td>1</td>
<td>$36.00</td>
</tr>
<tr>
<th colspan="2">total</th>
<th>$36.00</th>
</tr>
</table>
<p>Paid by Check</p>
</div>
</summary>
<author>
<name>Dave Thomas</name>
<email>customer@example.com</email>
</author>
</entry>
</feed>
Look at the headers
curl --max-time 15 --silent --dump - --output /dev/null --user dave:secret http://localhost:3000/products/2/who_bought.atom
HTTP/1.1 200 OK
Cache-Control: max-age=0, private, must-revalidate
Last-Modified: Tue, 04 Feb 2014 19:43:26 GMT
Connection: Keep-Alive
Date: Tue, 04 Feb 2014 19:43:27 GMT
Content-Type: application/atom+xml; charset=utf-8
Etag: "7bce6972ddab01a984b0a50ada27c262"
Server: WEBrick/1.3.1 (Ruby/1.8.7/2013-06-27)
Content-Length: 1324
X-Runtime: 0.105118
X-Ua-Compatible: IE=Edge
curl --max-time 15 --silent --dump - --output /dev/null --user dave:secret http://localhost:3000/products/2/who_bought.atom -H 'If-None-Match: "7bce6972ddab01a984b0a50ada27c262"'
HTTP/1.1 304 Not Modified
Cache-Control: max-age=0, private, must-revalidate
Last-Modified: Tue, 04 Feb 2014 19:43:26 GMT
Date: Tue, 04 Feb 2014 19:43:27 GMT
Etag: "7bce6972ddab01a984b0a50ada27c262"
Server: WEBrick/1.3.1 (Ruby/1.8.7/2013-06-27)
X-Runtime: 0.085210
X-Ua-Compatible: IE=Edge
curl --max-time 15 --silent --dump - --output /dev/null --user dave:secret http://localhost:3000/products/2/who_bought.atom -H 'If-Modified-Since: Tue, 04 Feb 2014 19:43:26 GMT'
HTTP/1.1 304 Not Modified
Cache-Control: max-age=0, private, must-revalidate
Last-Modified: Tue, 04 Feb 2014 19:43:26 GMT
Date: Tue, 04 Feb 2014 19:43:28 GMT
Etag: "7bce6972ddab01a984b0a50ada27c262"
Server: WEBrick/1.3.1 (Ruby/1.8.7/2013-06-27)
X-Runtime: 0.091480
X-Ua-Compatible: IE=Edge