ruby
34 lines · 7 steps
How a Rails model builds unique slugs
A Post generates a URL-friendly, collision-free slug before validation and exposes it as the route param.
Explained by
highlit
1class Post < ApplicationRecord
2 before_validation :generate_slug, on: :create
3
4 validates :slug, presence: true, uniqueness: true
5
6 def to_param
7 slug
8 end
9
10 private
11
12 def generate_slug
13 return if title.blank?
14
15 base = normalize(title)
16 candidate = base
17 suffix = 2
18
19 while self.class.where.not(id: id).exists?(slug: candidate)
20 candidate = "#{base}-#{suffix}"
21 suffix += 1
22 end
23
24 self.slug = candidate
25 end
26
27 def normalize(string)
28 I18n.transliterate(string)
29 .downcase
30 .gsub(/[^a-z0-9\s-]/, "")
31 .strip
32 .gsub(/[\s-]+/, "-")
33 end
34end
01 / 01
STEP 01
‹ swipe to step through ›
Walkthrough
Space play
←→ step
click any line
Three takeaways
- 1A before_validation callback lets you populate derived fields before the same record's validations run.
- 2Probing the database for existing values and appending a counter guarantees uniqueness without raising errors.
- 3Overriding to_param swaps the integer id for a human-readable slug in generated URLs.
Related explainers
ruby
require "csv" class SalesReport def initialize(path)
Aggregating CSV sales data in Ruby
data-aggregation
memoization
group_by
Intermediate
6 steps
ruby
module DurationFormatter UNITS = [ ['week', 604_800], ['day', 86_400],
Turning seconds into human-readable durations in Ruby
greedy-decomposition
modular-arithmetic
formatting
Intermediate
7 steps
javascript
const express = require('express'); const v1 = express.Router();
Versioning an API with Express Routers
api versioning
routing
modularity
Intermediate
10 steps
ruby
class Comment < ApplicationRecord belongs_to :post belongs_to :author, class_name: "User"
Live-updating comments with Turbo in Rails
turbo-streams
callbacks
associations
Intermediate
8 steps
ruby
require 'json' require 'set' class SensitiveScrubber
Recursively scrubbing secrets from JSON
recursion
data-masking
pattern-matching
Intermediate
7 steps
ruby
class ReportBatcher BATCH_SIZE = 500 def initialize(account)
Batching monthly email summaries in Rails
batching
service-object
background-jobs
Intermediate
7 steps
Share this explainer
Here's the card — post it anywhere.
Made with highlit — turn any snippet into a walkthrough like this in about a minute.
Explain your code
Embed this explainer
Drop the interactive walkthrough into a blog or docs. Views never cost a credit.
<iframe src="https://highlit.co/explainers/how-a-rails-model-builds-unique-slugs-explained-ruby-6473/embed?autoplay=1" width="100%" height="520" loading="lazy" style="border:0"></iframe>
Autoplay is on by default — add ?autoplay=0 to start paused.