h1. Slugalicious -- Easy and powerful URL slugging for Rails 3 _*(no monkey-patching required)*_ | *Author* | Tim Morgan | | *Version* | 1.1.1 (Aug 13, 2011) | | *License* | Released under the MIT license. | h2. About Slugalicious is an easy-to-use slugging library that helps you generate pretty URLs for your ActiveRecord objects. It's built for Rails 3 and is cordoned off in a monkey patching-free zone. Slugalicious is easy to use and powerful enough to cover all of the most common use-cases for slugging. Slugs are stored in a separate table, meaning you don't have to make schema changes to your models, and you can change slugs while still keeping the old URLs around for redirecting purposes. Slugalicious is an intelligent slug generator: You can specify multiple ways to generate slugs, and Slugalicious will try them all until it finds one that generates a unique slug. If all else fails, Slugalicious will fall back on a less pretty but guaranteed-unique backup slug generation strategy. Slugalicious works with the Stringex Ruby library, meaning you get meaningful slugs via the @String#to_url@ method. Below are two examples of how powerful Stringex is:

"$6 Dollar Burger".to_url #=> "six-dollar-burger"
"新年好".to_url #=> "xin-nian-hao"
h2. Installation *Important Note:* Slugalicious is written for Rails 3.0 and Ruby 1.9 only. Firstly, add the gem to your Rails project's @Gemfile@:

gem 'slugalicious'
Next, use the generator to add the @Slug@ model and its migration to your project:

rails generate slugalicious
Then run the migration to set up your database. h2. Usage For any model you want to slug, include the @Slugalicious@ module and call @slugged@:

class User < ActiveRecord::Base
  include Slugalicious
  slugged ->(user) { "#{user.first_name} #{user.last_name}" }
end
Doing this sets the @to_param@ method, so you can go ahead and start generating URLs using your models. You can use the @find_from_slug@ method to load a record from a slug:

user = User.find_from_slug(params[:id])
h3. Multiple slug generators The @slugged@ method takes a list of method names (as symbols) or @Procs@ that each attempt to generate a slug. Each of these generators is tried in order until a unique slug is generated. (The output of each of these generators is run through the slugifier to convert it to a URL-safe string. The slugifier is by default @String#to_url@, provided by the Stringex gem.) So, if we had our @User@ class, and we first wanted to slug by last name only, but then add in the first name if two people share a last name, we'd call @slugged@ like so:

slugged :last_name, ->(user) { "#{user.first_name} #{user.last_name}" }
In the event that none of these generators manages to make a unique slug, a fallback generator is used. This generator prepends the ID of the record, making it guaranteed unique. Let's use the example generators shown above. If we create a user with the name "Sancho Sample", he will get the slug "sample". Create another user with the same name, and that user will get the slug "sancho-sample;2". The semicolon is the default ID separator (and it can be overridden). h3. Scoped slugs Slugs must normally be unique for a single model type. Thus, if you have a @User@ named Hammer and a @Product@ named hammer, they can both share the "hammer" slug. If you want to decrease the uniqueness scope of a slug, you can do so with the @:scope@ option on the @slugged@ method. Let's say you wanted to limit the scope of a @Product@'s slug to its associated @Department@; that way you could have a product named "keyboard" in both the Computer Supplies and the Music Supplies departments. To do so, override the @:scope@ option with a method name (as symbol) or a @Proc@ that limits the scope of the uniqueness requirement:

class Product < ActiveRecord::Base
  include Slugalicious
  belongs_to :department
  slugged :name, scope: :department_url_component

  private

  def department_url_component
    department.name.to_url + "/"
  end
end
Now, your computer keyboard's slug will be "computer-supplies/keyboard" and your piano keyboard's slug will be "music-supplies/keyboard". There's an important thing to notice here: The method or proc you use to scope the slug must return a proper URL substring. That typically means you need to URL-escape it and add a slash at the end, as shown in the example above. When you call @to_param@ on your piano keyboard, instead of just "keyboard", you will get "music-supplies/keyboard". Likewise, you can use the @find_from_slug_path@ method to find a record from its full path, slug and scope included. You would usually use this method in conjunction with route globbing. For example, we could set up our @routes.rb@ file like so:

get '/products/*path', 'products#show', as: :products
Then, in our @ProductsController@, we load the product from the path slug like so:

def find_product
  @product = Product.find_from_slug_path(params[:path])
end
This is why it's very convenient to have your @:scope@ method/proc not only return the uniqueness constraint, but also the scoped portion of the URL preceding the slug. h3. Altering and expiring slugs When a model is created, it gets one slug, marked as the active slug (by default). This slug is the first generator that produces a unique slug string. If a model is updated, its slug is regenerated. Each of the slug generators is invoked, and if any of them produces an existing slug assigned to the object, that slug is made the active slug. (Priority goes to the first slug generator that produces an existing slug [active or inactive]). If none of the slug generators generates a known, existing slug belonging to the object, then the first unique slug is used. A new @Slug@ instance is created and marked as active, and any other slugs belonging to the object are marked as inactive. Inactive slugs do not act any differently from active slugs. An object can be found by its inactive slug just as well as its active slug. The flag is there so you can alter the behavior of your application depending on whether the slug is current. A common application of this is to have inactive slugs 301-redirect to the active slug, as a way of both updating search engines' indexes and ensuring that people know the URL has changed. As an example of how do this, we alter the @find_product@ method shown above to be like so:

def find_product
  @product = Product.find_from_slug_path(params[:path])
  unless @product.active_slug?(params[:path].split('/').last)
    redirect_to product_url(@product), status: :moved_permanently
    return false
  end
  return true
end
The old URL will remain indefinitely, but users who hit it will be redirected to the new URL. Ideally, links to the old URL will be replaced over time with links to the new URL. The problem is that even though the old slug is inactive, it's still "taken." If you create a product called "Keyboard", but then rename it to "Piano", the product will claim both the "keyboard" and "piano" slugs. If you had renamed it to make room for a different product called "Keyboard" (like a computer keyboard), you'd find its slug is "keyboard;2" or similar. To prevent the slug namespace from becoming more and more polluted over time, websites generally expire inactive slugs after a period of time. To do this in Slugalicious, write a task that periodically checks for and deletes old, inactive @Slug@ records. Such a task could be invoked through a cron job, for instance. An example:

Slug.inactive.where([ "created_at < ?", 30.days.ago ]).delete_all