• Stars
    star
    498
  • Rank 88,494 (Top 2 %)
  • Language
    Ruby
  • License
    MIT License
  • Created almost 17 years ago
  • Updated about 13 years ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Rails RESTful controller abstraction plugin.

Resource Controller¶ ↑

resource_controller makes RESTful controllers easier, more maintainable, and super readable. With the RESTful controller pattern hidden away, you can focus on what makes your controller special.

Get It¶ ↑

Install it as a plugin:

script/plugin install git://github.com/giraffesoft/resource_controller.git

Or grab the source

git clone git://github.com/giraffesoft/resource_controller.git

Usage¶ ↑

Creating a basic RESTful controller is as easy as…

class PostsController < ResourceController::Base
end

…or if you prefer, you can use the method-call syntax. If you need to inherit from some other class, this syntax is definitely for you:

class PostsController < ApplicationController
  resource_controller
end

Both syntaxes are identical in their behavior. Just make sure you call resource_controller before you use any other r_c functionality in your controller.

Nobody just uses the default RESTful controller, though. resource_controller provides a simple API for customizations.

Action Lifecycle¶ ↑

It’s really easy to make changes to the lifecycle of your actions.

Note: We had to call the new accessor "new_action", since new is somewhat reserved in ruby.

Before and After¶ ↑

class ProjectsController < ResourceController::Base

  new_action.before do
    3.times { object.tasks.build }
  end

  create.after do
    object.creator = current_user
  end

end

Flash¶ ↑

class ProjectsController < ResourceController::Base
  create.flash "Can you believe how easy it is to use resource_controller?  Neither could I!"
end

respond_to¶ ↑

You can add to what’s already there…

class ProjectsController < ResourceController::Base      
  create.wants.js { render :template => "show.rjs" }
end

Or you can create a whole new block. This syntax destroys everything that’s there, and starts again…

class ProjectsController < ResourceController::Base      
  create.response do |wants|
    wants.html
    wants.js { render :template => "show.rjs" }
  end
end

If you have a nested resource and want to redirect to the parent after create/update and destroy you can do this in the object controller

class CommentsController < ResourceController::Base
   belongs_to :post

   create.wants.html { redirect_to smart_url(parent_url_options) } 
   update.wants.html { redirect_to smart_url(parent_url_options) } 
   destroy.wants.html { redirect_to smart_url(parent_url_options) }
 end

Scoping¶ ↑

Because sometimes you want to make a bunch of customizations at once, most of the helpers accept blocks that make grouping calls really easy. Is it a DSL? Maybe; maybe not. But, it’s definitely awesome.

With actions that can fail, the scoping defaults to success. That means that create.flash == create.success.flash.

class ProjectsController < ResourceController::Base

  create do
    flash "Object successfully created!"
    wants.js { render :template => "show.rjs" }

    failure.wants.js { render :template => "display_errors.rjs" }
  end

  destroy do
    flash "You destroyed your project.  Good work."

    failure do
      flash "You cannot destroy that project.  Stop trying!"
      wants.js { render :template => "display_errors.rjs" }
    end
  end

end

Singleton Resource¶ ↑

If you want to create a singleton RESTful controller inherit from ResourceController::Singleton.

class AccountsController < ResourceController::Singleton
end

…or if need to inherit from some other class:

class AccountsController < ApplicationController
  resource_controller :singleton
end

Note: This type of controllers handle a single resource only so the index action and all the collection helpers (collection_url, collection_path…) are not available for them.

Loading objects in singletons is similar to plural controllers with one exception. For non-nested singleton controllers you should override the object method as it defaults to nil for them.

class AccountsController < ResourceController::Singleton
  private
    def object
      @object ||= Account.find(session[:account_id])
    end
 end

In other cases you can use the default logic and override it only if you use permalinks or anything special.

Singleton nesting with both :has_many and :has_one associations is provided…

map.resource :account, :has_many => :options  # /account/options, account is a singleton parent
map.resources :users, :has_one => :image  # /users/1/image, image is a singleton child

If you have the :has_many association with a singleton parent remember to override parent_object for your :has_many controller as it returns nil by default in this case.

class OptionsController < ResourceController::Base
  belongs_to :account

  protected
  def parent_object
    Account.find(session[:account_id])
  end
end

Helpers (ResourceController::Helpers)¶ ↑

Loading objects¶ ↑

You want to add something like pagination to your controller…

class PostsController < ResourceController::Base
  private
    def collection
      @collection ||= end_of_association_chain.find(:all, :page => {:size => 10, :current => params[:page]})
    end
end

Or maybe you used a permalink…

class PostsController < ResourceController::Base
  private
    def object
      @object ||= end_of_association_chain.find_by_permalink(param)
    end
end

Building objects¶ ↑

Maybe you have some alternative way of building objects…

class PostsController < ResourceController::Base
  private
    def build_object
      @object ||= end_of_association_chain.build_my_object_some_funky_way object_params
    end
end

…and there are tons more helpers in the ResourceController::Helpers

Nested Resources¶ ↑

Nested controllers can be a pain, especially if routing is such that you may or may not have a parent. Not so with Resource Controller.

class CommentsController < ResourceController::Base
  belongs_to :post
end

All of the finding, and creation, and everything will be done at the scope of the post automatically.

Namespaced Resources¶ ↑

…are handled automatically, and any namespaces are always available, symbolized, in array form @ ResourceController::Helpers#namespaces

Polymorphic Resources¶ ↑

Everything, including url generation is handled completely automatically. Take this example…

## comment.rb
class Comment
  belongs_to :commentable, :polymorphic => true
end

## comments_controller.rb
class CommentsController < ResourceController::Base
  belongs_to :post, :product, :user
end
*Note:* Your model doesn't have to be polymorphic in the ActiveRecord sense.  It can be associated in whichever way you want.

## routes.rb
map.resources :posts, :has_many => :comments
map.resources :products, :has_many => :comments
map.resources :users, :has_many => :comments

All you have to do is that, and r_c will infer whichever relationship is present, and perform all the actions at the scope of the parent object.

Parent Helpers¶ ↑

You also get some helpers for reflecting on your parent.

parent?       # => true/false is there a parent present?
parent_type   # => :post
parent_model  # => Post
parent_object # => @post

Non-standard resource names¶ ↑

resource_controller supports overrides for every non-standard configuration of resources.

The most common example is where the resource has a different name than the associated model. Simply overriding the model_name helper will get resource_controller working with your model.

map.resources :tags
...
class PhotoTag < ActiveRecord::Base
...
class TagsController < ResourceController::Base
  private
    def model_name
      'photo_tag'
    end
end

In the above example, the variable, and params will be set to @tag, @tags, and params. If you’d like to change that, override object_name.

def object_name
  'photo_tag'
end

If you’re using a non-standard controller name, but everything else is standard, overriding resource_name will propagate through all of the other helpers.

map.resources :tags, :controller => "somethings"
...
class Tag < ActiveRecord::Base
...
class SomethingsController < ResourceController::Base
  private
    def resource_name
      'tag'
    end
end

Finally, the route_name helper is used by Urligence to determine which url helper to call, so if you have non-standard route names, override it.

map.resources :tags, :controller => "taggings"
...
class Taggings < ActiveRecord::Base
...
class TaggingsController < ResourceController::Base
  private
    def route_name
      'tag'
    end
end

Url Helpers¶ ↑

Thanks to Urligence, you also get some free url helpers.

No matter what your controller looks like…

[edit_|new_]object_url # is the equivalent of saying [edit_|new_]post_url(@post)
[edit_|new_]object_url(some_other_object) # allows you to specify an object, but still maintain any paths or namespaces that are present

collection_url # is like saying posts_url

Url helpers are especially useful when working with polymorphic controllers.

# /posts/1/comments
object_url          # => /posts/1/comments/#{@comment.to_param}
object_url(comment) # => /posts/1/comments/#{comment.to_param}
edit_object_url     # => /posts/1/comments/#{@comment.to_param}/edit
collection_url      # => /posts/1/comments

# /products/1/comments
object_url          # => /products/1/comments/#{@comment.to_param}
object_url(comment) # => /products/1/comments/#{comment.to_param}
edit_object_url     # => /products/1/comments/#{@comment.to_param}/edit
collection_url      # => /products/1/comments

# /comments
object_url          # => /comments/#{@comment.to_param}
object_url(comment) # => /comments/#{comment.to_param}
edit_object_url     # => /comments/#{@comment.to_param}/edit
collection_url      # => /comments

Or with namespaced, nested controllers…

# /admin/products/1/options
object_url          # => /admin/products/1/options/#{@option.to_param}
object_url(option)  # => /admin/products/1/options/#{option.to_param}
edit_object_url     # => /admin/products/1/options/#{@option.to_param}/edit
collection_url      # => /admin/products/1/options

You get the idea. Everything is automagical! All parameters are inferred.

Credits¶ ↑

resource_controller was created, and is maintained by James Golick.

License¶ ↑

resource_controller is available under the MIT License

More Repositories

1

timeline_fu

Ruby
410
star
2

degrade

Keep track of service error rates using redis. Degrade functionality if they're too high.
Ruby
310
star
3

active_presenter

The presenter library you already know.
Ruby
302
star
4

is_taggable

Tagging that doesn't want to be on steroids. It's skinny and happy to stay that way
Ruby
213
star
5

trample

A Better Load Simulator
Ruby
160
star
6

observational

Use the observer pattern to better divide your objects' responsibilities.
Ruby
116
star
7

attribute_fu

a better version of this functionality is in rails 2.3. use that!
Ruby
99
star
8

always_verify_ssl_certificates

Ruby's net/http is setup to never verify SSL certificates by default. Most ruby libraries do the same. That means that you're not verifying the identity of the server you're communicating with and are therefore exposed to man in the middle attacks. This gem monkey-patches net/http to force certificate verification and make turning it off impossible.
Ruby
97
star
9

blank

Blank is GiraffeSoft’s blank rails starter app.
Ruby
61
star
10

zebra

One line tests without the smells.
Ruby
56
star
11

enum_field

Ruby
55
star
12

statsd.scala

statsd on the jvm
Scala
48
star
13

conductor

Build clusters in the cloud with conductor.
Ruby
48
star
14

munin_plugin

Ruby
46
star
15

scala-resque-worker

Scala
44
star
16

action_messager

Dead simple IM notifications for your app
Ruby
43
star
17

cassandra-munin-plugins

Munin plugins for apache cassandra.
42
star
18

classy_resources

Think resource_controller, except for sinatra.
Ruby
41
star
19

lexical_uuid

UUIDs that are byte-ordered lamport clocks (timestamp, worker_id). Much simpler than type-1 UUID's crappy, weirdo layout.
Ruby
31
star
20

markaby

Markaby patched to run on rails 2.0.2
Ruby
23
star
21

tweet

Tweet from the command line.
Ruby
22
star
22

timeline_fu-example

Ruby
22
star
23

fetlife-nagios-plugins

Nagios plugins in use at FetLife. Services include: resque, cassandra, etc.
Ruby
22
star
24

lexical_uuid.erl

UUIDs that are byte-ordered lamport clocks (timestamp, worker_id). Much simpler than type-1 UUID's crappy, weirdo layout.
Erlang
21
star
25

openvpn-recipes

Get OpenVPN up and running on Ubuntu, using Chef. Great way to beat geofencing!
Ruby
19
star
26

dotfiles

Vim Script
17
star
27

friendly

http://jamesgolick.com/2009/12/16/introducing-friendly-nosql-with-mysql-in-ruby.html
Ruby
17
star
28

has_browser

has_browser makes it possible to create simple, parameterized browser interfaces to your models. That is, given a set of parameters, return all the models that match.
Ruby
16
star
29

conductor-rails

A rails stack for conductor.
Ruby
12
star
30

scrawny

a lightweight, RESTful persistent queue on top of thin, rack and invisible in < 60 LoC
Ruby
11
star
31

expectations

expectations is a lightweight unit testing framework.
Ruby
11
star
32

starling-on-invisible

Proof of concept that you can build a queueing server and client RESTfully, leveraging starling's persistent queue code, a lightweight web framework, and thin.
Ruby
10
star
33

scala-connection-pool

Connection pool abstractions in scala. Supports load balancing and failover in a thread-safe manner. Defers to commons-pool for the actual pooling.
Scala
9
star
34

spymemcached-gem

A jruby-only memcached client and rails cache store that uses spymemcached under the hood.
Ruby
7
star
35

action_mailer_verp

Send VERP emails with actionmailer. http://en.wikipedia.org/wiki/Variable_envelope_return_path
Ruby
7
star
36

dirty_callbacks

Ruby
6
star
37

per_mailer_smtp_settings

Setup different smtp settings per ActionMailer::Base subclass.
Ruby
6
star
38

memcache-lock

Extraction of the lock from cache-money.
Ruby
6
star
39

what_does_this_error_mean-rails

Rails plugin for integration with whatdoesthiserrormean.com
Ruby
6
star
40

file-sherpa

Upload progress bars for jQuery.
JavaScript
5
star
41

rollout_rest_api

A sinatra app that exposes a REST API for rollout.
Ruby
5
star
42

zfs-mysql-backup

4
star
43

fetlife-packages

Binary FreeBSD packages in use at fetlife.com (NSFW).
4
star
44

what_does_this_error_mean-merb

Ruby
3
star
45

sbt-ruby

sbt plugin for including ruby code and rubygems in the build.
Scala
3
star
46

smemcache

A scala wrapper around spymemcached.
Scala
3
star
47

rest_client-cocoa

Ruby
2
star
48

vim_conf

2
star
49

restful_roles

Ruby
2
star
50

rubyee-freebsd-port

Adapted from the ruby18 port. I'm new to freebsd and ports, so it's probably all wrong!
2
star
51

jgmalloc

C
2
star
52

health

Health checks for rails.
Ruby
2
star
53

activerecordless_migrations

Ruby
2
star
54

memcached.erl

because obviously I wanted to write a fucking memcached client
Erlang
2
star
55

gperftools

C++
2
star
56

r_c-2.1.1

2
star
57

tempo_cli

CLI for sending commands to tempo app.
Ruby
2
star
58

markdownj-gem

(J)Ruby wrapper around markdownj.
Ruby
1
star
59

telephos

A really simple Scala wrapper around Cassandra's thrift bindings. No DSLs here.
Scala
1
star
60

jruby-scala-issue

Ruby
1
star
61

client_proxy

Basic ruby proxy class to use around service clients that handles retrying requests and failover.
Ruby
1
star
62

thrift-exception-notifier

If you run thrift services in production, it's handy to receive an email when shit breaks.
Scala
1
star
63

uriparser

C
1
star
64

rpmspecs

RPM specs that I'm maintaining. I'm not good at this. They're probably all wrong.
1
star