ActiveRecord::Base без таблицы
-
06-09-2019 - |
Вопрос
Это появилось немного раньше( Атрибуты модели рельсов без соответствующего столбца в базе данных ), но похоже, что упомянутый плагин Rails не поддерживается ( http://agilewebdevelopment.com/plugins/activerecord_base_without_table ).Нет ли возможности сделать это с помощью ActiveRecord как есть?
Если нет, есть ли способ получить правила проверки ActiveRecord без использования ActiveRecord?
ActiveRecord, конечно, хочет, чтобы таблица существовала.
Решение
Это подход, который я использовал в прошлом:
В приложение/модели/tableless.rb
class Tableless < ActiveRecord::Base
def self.columns
@columns ||= [];
end
def self.column(name, sql_type = nil, default = nil, null = true)
columns << ActiveRecord::ConnectionAdapters::Column.new(name.to_s, default,
sql_type.to_s, null)
end
# Override the save method to prevent exceptions.
def save(validate = true)
validate ? valid? : true
end
end
В приложение/модели/foo.rb
class Foo < Tableless
column :bar, :string
validates_presence_of :bar
end
В скрипт/консоль
Loading development environment (Rails 2.2.2)
>> foo = Foo.new
=> #<Foo bar: nil>
>> foo.valid?
=> false
>> foo.errors
=> #<ActiveRecord::Errors:0x235b270 @errors={"bar"=>["can't be blank"]}, @base=#<Foo bar: nil>>
Другие советы
Валидации — это просто модуль ActiveRecord.Пробовали ли вы смешать их со своей моделью, отличной от ActiveRecord?
class MyModel
include ActiveRecord::Validations
# ...
end
Я считаю, что чем больше ответов, тем лучше, поскольку это один из первых результатов в Google при поиске «модели рельсов 3.1 без таблиц».
Я реализовал то же самое без использования ActiveRecord::Base, включая ActiveRecord::Validations.
Основная цель заключалась в том, чтобы заставить все работать в formtastic, и ниже я включил образец платежа, который нигде не будет сохранен, но все же имеет возможность быть проверенным с использованием проверок, которые мы все знаем и любим.
class Payment
include ActiveModel::Validations
attr_accessor :cc_number, :payment_type, :exp_mm, :exp_yy, :card_security, :first_name, :last_name, :address_1, :address_2, :city, :state, :zip_code, :home_telephone, :email, :new_record
validates_presence_of :cc_number, :payment_type, :exp_mm, :exp_yy, :card_security, :first_name, :last_name, :address_1, :address_2, :city, :state
def initialize(options = {})
if options.blank?
new_record = true
else
new_record = false
end
options.each do |key, value|
method_object = self.method((key + "=").to_sym)
method_object.call(value)
end
end
def new_record?
return new_record
end
def to_key
end
def persisted?
return false
end
end
Надеюсь, это кому-то поможет, поскольку сегодня я потратил несколько часов, пытаясь разобраться в этом.
ОБНОВЛЯТЬ: В Rails 3 это можно сделать очень легко.В Rails 3+ вы можете использовать новый ActiveModel
модуль и его подмодули.Это должно работать сейчас:
class Tableless
include ActiveModel::Validations
attr_accessor :name
validates_presence_of :name
end
Для получения дополнительной информации вы можете просмотреть Рейлскаст (или почитайте об этом на AsciiCasts) по теме, а также по этому поводу сообщение в блоге Иегуды Каца.
СТАРЫЙ ОТВЕТ СЛЕДУЕТ:
Возможно, вам придется добавить это к решению, предложенному Джоном Топли в предыдущем комментарии:
class Tableless
class << self
def table_name
self.name.tableize
end
end
end
class Foo < Tableless; end
Foo.table_name # will return "foos"
Это дает вам «поддельное» имя таблицы, если оно вам нужно.Без этого метода Foo::table_name
будет оценен как «без таблиц».
Я нашел эту ссылку, которая работает КРАСИВО.
Просто дополнение к принятому ответу:
Сделайте так, чтобы ваши подклассы наследовали родительские столбцы с помощью:
class FakeAR < ActiveRecord::Base
def self.inherited(subclass)
subclass.instance_variable_set("@columns", columns)
super
end
def self.columns
@columns ||= []
end
def self.column(name, sql_type = nil, default = nil, null = true)
columns << ActiveRecord::ConnectionAdapters::Column.new(name.to_s, default, sql_type.to_s, null)
end
# Overrides save to prevent exceptions.
def save(validate = true)
validate ? valid? : true
end
end
Это форма поиска, которая представляет объект под названием критерии который имеет вложенный период объект с начало и конец атрибуты.
Действие контроллера очень простое, но оно загружает значения из вложенных объектов в форме и при необходимости повторно отображает те же значения с сообщениями об ошибках.
Работает на Rails 3.1.
Модель:
class Criteria < ActiveRecord::Base
class << self
def column_defaults
{}
end
def column_names
[]
end
end # of class methods
attr_reader :period
def initialize values
values ||= {}
@period = Period.new values[:period] || {}
super values
end
def period_attributes
@period
end
def period_attributes= new_values
@period.attributes = new_values
end
end
В контроллере:
def search
@criteria = Criteria.new params[:criteria]
end
В помощнике:
def criteria_index_path ct, options = {}
url_for :action => :search
end
В представлении:
<%= form_for @criteria do |form| %>
<%= form.fields_for :period do |prf| %>
<%= prf.text_field :beginning_as_text %>
<%= prf.text_field :end_as_text %>
<% end %>
<%= form.submit "Search" %>
<% end %>
Создает HTML:
<form action="/admin/search" id="new_criteria" method="post">
<input id="criteria_period_attributes_beginning_as_text" name="criteria[period_attributes][beginning_as_text]" type="text">
<input id="criteria_period_attributes_end_as_text" name="criteria[period_attributes][end_as_text]" type="text">
Примечание:Атрибут действия, предоставляемый помощником, и формат именования вложенных атрибутов, который позволяет контроллеру легко загружать все значения одновременно.
Есть драгоценный камень activerecord-less.Это великолепная возможность создания бестабличных моделей ActiveRecord, поэтому она поддерживает проверки, ассоциации и типы.Он поддерживает Active Record 2.3, 3.0, 3.2.
Рекомендуемый способ сделать это в Rails 3.x (с использованием ActiveModel) не поддерживает ни ассоциации, ни типы.