Skip to content

Latest commit

 

History

History
182 lines (131 loc) · 5.85 KB

README.md

File metadata and controls

182 lines (131 loc) · 5.85 KB

🐌 slugifiable - Rails gem to generate SEO-friendly slugs

Gem Version

Automatically generates unique slugs for your Rails' model records, so you can expose SEO-friendly URLs.

Example:

https://myapp.com/products/big-red-backpack-321678

Where big-red-backpack-321678 is the slug.

slugifiable can generate:

  • Slugs like "big-red-backpack" or "big-red-backpack-321678": unique, string-based slugs based on any attribute, such as product.name
  • Slugs like "d4735e3a265": unique hex string slugs
  • Slugs like 321678: unique number-only slugs

Why

When building Rails apps, we usually need to expose something in the URL to identify a record, like:

https://myapp.com/products/123

But exposing IDs (like 123) is not usually good practice. It's not SEO-friendly, it can give away how many records you have in the database, it could also be an attack vector, and it just feels off.

It would be much better to have a random-like string or number instead, while still remaining unique and identifiable:

https://myapp.com/products/d4735e3a265

Or better yet, use other instance attribute (like product.name) to build the slug:

https://myapp.com/products/big-red-backpack

slugifiable takes care of building all these kinds of slugs for you.

Installation

Add this line to your application's Gemfile:

gem 'slugifiable'

Then run bundle install.

After installing the gem, add include Slugifiable::Model to any ActiveRecord model, like this:

class Product < ApplicationRecord
  include Slugifiable::Model # Adding this provides all the required slug-related methods to your model
end

That's it!

Then you can, for example, get the slug for a product like this:

Product.first.slug
=> "4e07408562b"

You can also define how to generate slugs:

class Product < ApplicationRecord
  include Slugifiable::Model
  generate_slug_based_on :name
end

And this will generate slugs based on your Product instance name, like:

Product.first.slug
=> "big-red-backpack"

If your model has a slug attribute in the database, slugifiable will automatically generate a slug for that model upon instance creation, and save it to the DB.

Important

Your slug attribute SHOULD NOT have null: false in the migration / database. If it does, slugifiable will not be able to save the slug to the database, and will raise an error like ERROR: null value in column "slug" of relation "posts" violates not-null constraint (PG::NotNullViolation) This is because records are created without a slug, and the slug is generated later.

If you're generating slugs based off the model id, you can also set a desired length:

class Product < ApplicationRecord
  include Slugifiable::Model
  generate_slug_based_on :id, length: 6
end

Which would return something like:

Product.first.slug
=> "6b86b2"

More details in the "How to use" section.

How to use

Slugs should never change, so it's recommended you save your slugs to the database.

Therefore, all models that include Slugifiable::Model should have a slug attribute that persists the slug in the database. If your model doesn't have a slug attribute yet, just run:

rails g migration addSlugTo<MODEL_NAME> slug:text

where <MODEL_NAME> is your model name in plural, and then run:

rails db:migrate

And your model should now have a slug attribute in the database.

When a model has a slug attribute, slugifiable automatically generates a slug for that model upon instance creation, and saves it to the DB.

slugifiable can also work without persisting slugs to the databse, though: you can always run .slug, and that will give you a valid, unique slug for your record.

Define how slugs are generated

By default, when you include Slugifiable::Model, slugs will be generated as a random-looking string based off the record id (SHA hash)

slugifiable supports both id and uuid.

The default setting is:

generate_slug_based_on id: :hex_string

Which returns slugs like: d4735e3a265

If you don't like hex strings, you can get number-only slugs with:

generate_slug_based_on id: :number

Which will return slugs like: 321678 – nonconsecutive, nonincremental, not a total count.

When you're generating obfuscated slugs (based on id), you can specify a desired slug length:

generate_slug_based_on id: :number, length: 3

The length should be a positive number between 1 and 64.

If instead of obfuscated slugs you want human-readable slugs, you can specify an attribute to base your slugs off of. For example:

generate_slug_based_on :name

Will look for a name attribute in your instance, and use its value to generate the slug. So if you have a product like:

Product.first.name
=> "Big Red Backpack"

then the slug will be computed as:

Product.first.slug
=> "big-red-backpack"

There may be collisions if two records share the same name – but slugs should be unique! To resolve this, when this happens, slugifiable will append a unique string at the end to make the slug unique:

Product.first.slug
=> "big-red-backpack-321678"

Development

After checking out the repo, run bin/setup to install dependencies. Then, run rake spec to run the tests. You can also run bin/console for an interactive prompt that will allow you to experiment.

To install this gem onto your local machine, run bundle exec rake install.

Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/rameerez/slugifiable. Our code of conduct is: just be nice and make your mom proud of what you do and post online.

License

The gem is available as open source under the terms of the MIT License.