12
.
02
.
2025
12
.
02
.
2025
Ruby
Ruby on Rails

Active Record - store vs store_accessor

Michał Łęcicki
Ruby Developer
store-vs-store_accessor by Michał Łęcicki

Imagine this: you get a task to add a Questionnaire feature to the app. It needs to support a variety of questions and must be delivered fast. A JSON column seems perfect for flexible data storage, but how do you integrate it with Rails?

Active Record store

After a quick research, you discover Active Record store method. It adds reader and writer accessors for a defined collection of keys. It also provides dirty tracking and prefix/suffix if needed. Let's try it:

# db/migrate/20250109092657_create_questionnaires.rb

class CreateQuestionnaires < ActiveRecord::Migration[8.0]
  def change
    create_table :questionnaires do |t|
      t.jsonb :questions
      t.string :title

      t.timestamps
    end
  end
end

The model looks like:

# app/models/questionnaire.rb

class Questionnaire < ApplicationRecord
  QUESTION_LIST = %w[question1 question2]

  store :questions, *QUESTION_LIST
end

And you can test it:

require "test_helper"

class QuestionnaireTest < ActiveSupport::TestCase
  test "stores questions in jsonb column" do
    questionnaire = Questionnaire.new
    questionnaire.question1 = "How are you?"
    questionnaire.question2 = "How old are you?"

    questionnaire.save!

    questionnaire.reload
    assert_equal questionnaire.question1, "How are you?"
    assert_equal questionnaire.question2, "How old are you?"
  end
end

Fast and easy, job done. 🎉

But a few weeks after the release, the data team guy comes to you and says: "We need to analyze the questionnaire table, but it contains invalid JSON. Please, fix it."
Sure. 🕵️

You go to the database and see the content of the column. It looks like YAML:

"---\\nquestion1: How are you?\\nquestion2: How old are you?\\n"

Oh! By default, Rails uses the YAML format for data serialization. Let's explicitly add the coder option to the store method (as per documentation). Updating the model should help:

# app/models/questionnaire.rb

class Questionnaire < ApplicationRecord
  QUESTION_LIST = %w[question1 question2]

  store :questions, *QUESTION_LIST, coder: JSON
end

How does the content of the database look now?

"{\\"question1\\":\\"How are you?\\",\\"question2\\":\\"How old are you?\\"}"

It's not a JSON object, it's a JSON string. 🤔
The test is green, which means Rails can serialize it properly, but the content stored in the database is still not a high-quality JSON. We can tell because searching with JSON operators still does not work.

Store_accessor

Confused at this point, let's go back to documentation. There is a NOTE:

NOTE: If you are using structured database data types (e.g. PostgreSQL hstore/json, MySQL 5.7+ json, or SQLite 3.38+ json) there is no need for the serialization provided by .store. Use .store_accessor instead to generate the accessor methods. Be aware that these columns use a string keyed hash and do not allow access using a symbol.

Right! 🤦
You do use structured database type. This means you should not use the store method. Just stick to store_accessor instead:

# app/models/questionnaire.rb

class Questionnaire < ApplicationRecord
  QUESTION_LIST = %w[question1 question2]

  store_accessor :questions, *QUESTION_LIST
end

And test if searching with JSON operators works:

require "test_helper"

class QuestionnaireTest < ActiveSupport::TestCase
  test "stores questions in jsonb column" do
    questionnaire = Questionnaire.new
    questionnaire.question1 = "How are you?"

    questionnaire.save!

    assert_equal questionnaire.reload.question1, "How are you?"
    assert_equal questionnaire.id,
      Questionnaire.where("questions ->> 'question1' = 'How are you?'").first.id
  end
end

Eventually, beautiful JSON resides in the database column:

{
  "question1": "How are you?",
  "question2": "How old are you?"
}

Conclusion

When working with structured data types like JSON or JSONB columns, use the store_accessor method. On the other hand, Active Record store method works fine for older use cases: simple key-value store in plain text columns. If in doubt, read the documentation carefully.

P. S. There is a simple Rails app repo with code examples.

Michał Łęcicki
Ruby Developer

Check my Twitter

Check my Linkedin

Did you like it? 

Sign up To VIsuality newsletter

READ ALSO

Why we chose Ruby on Rails and React.js for our main technologies? (FAQ).

02
.
10
.
2024
Michał Krochecki
Ruby on Rails
Backend
Frontend
Visuality

Branding: How to style your Jira?

14
.
11
.
2023
Lukasz Jackiewicz
Tutorial
Design
Project Management

How to start your UX/UI designer career

14
.
11
.
2023
Bartłomiej Bednarski
Design
Tutorial
HR

WebUSB - Bridge between USB devices and web browsers

14
.
11
.
2023
Burak Aybar
Ruby on Rails
Frontend
Backend
Tutorial

Visuality comes to town - this time it's Poznań

14
.
11
.
2023
Michał Piórkowski
Visuality
HR

How to choose a software house.

14
.
11
.
2023
Michał Piórkowski
Ruby on Rails
Business
Visuality

CSS Modules in Rails

14
.
11
.
2023
Adam Król
Ruby on Rails
Tutorial
Backend
Frontend

JSON API versus the NIH syndrome

14
.
11
.
2023
Nadia Miętkiewicz
Backend
Frontend
Tutorial

From Idea to Concept

02
.
10
.
2024
Michał Krochecki
Ruby on Rails
Business
Startups

Styling React Components

14
.
11
.
2023
Umit Naimian
Ruby on Rails
Frontend
Tutorial

How good design can help your business grow

14
.
11
.
2023
Lukasz Jackiewicz
Design
Business
Marketing

TODO not. Do, or do not.

29
.
11
.
2023
Stanisław Zawadzki
Ruby on Rails
Software

CS Lessons #003: Density map in three ways

14
.
11
.
2023
Michał Młoźniak
Ruby
Backend
Tutorial
Software

Clean code for the win

14
.
11
.
2023
Michał Piórkowski
Ruby on Rails
Backend
Frontend
Business

Crowd-operated Christmas Lights

14
.
11
.
2023
Nadia Miętkiewicz
Ruby on Rails
Backend

How to startup and be mature about it

14
.
11
.
2023
Rafał Maliszewski
Ruby on Rails
Startups
Business

A journey of a thousand miles begins with workshops

14
.
11
.
2023
Michał Piórkowski
Software
HR

CS Lessons #002: Data structures

14
.
11
.
2023
Michał Młoźniak
Ruby
Software

Summary of Phoenix workshop at Visuality

14
.
11
.
2023
Karol Słuszniak
Ruby on Rails
Visuality
Backend

CS Lessons #001: Working with binary files

14
.
11
.
2023
Michał Młoźniak
Ruby
Software

CS Lessons #000: Introduction and motivation

14
.
11
.
2023
Michał Młoźniak
Ruby
Software

Working with 40-minute intervals

14
.
11
.
2023
Sakir Temel
Software
HR

THE MATURE TECH STARTUP DILEMMA: WHAT'S NEXT

14
.
11
.
2023
Susanna Romantsova
Startups

Win MVP workshop!

14
.
11
.
2023
Susanna Romantsova
Startups

FINTECH WEEK IN OSLO: WHATs & WHYs

14
.
11
.
2023
Susanna Romantsova
Conferences