FriendlyID уникальные слагы по объему и активному администратору
У меня есть модельное шоу, и у меня есть модельный эпизод. У шоу есть много эпизодов, как вы можете себе представить. Я настроил friendlyID и в модели эпизода:
friendly_id: slug_candidates, use:: slugged,: use =>: scoped,: scope =>: show
Я полагаю, что у меня проблема в том, что я не работаю с областью ресурсов в Active Admin и не настраиваю поиск вложенных ресурсов, но я не уверен, как это сделать: вот что происходит.
Допустим, у меня есть шоу под названием "Show 1", и я создаю эпизод с "Episode 1" в качестве заголовка. Когда я создаю новое шоу и у него есть "Эпизод 1" внутри "Шоу 2", его можно будет сохранить, поскольку оно ограничено заголовком шоу.
2014-02-07T00:38:28.009413+00:00 app[web.1]: Parameters: {"utf8"=>"✓", "authenticity_token"=>"yhP8K6jUVhWHHzqWhXvaABXDzh8eACYcHhkEhMKJt+8=", "show"=>{"creator_id"=>"132", "title"=>"Menuda Noche", "subtitle"=>"", "description"=>"", "position"=>"", "age_range"=>"", "single"=>"0", "skiplist"=>"0", "promote"=>"0", "approved"=>"1", "created_at(1i)"=>"", "created_at(2i)"=>"", "created_at(3i)"=>"", "created_at(4i)"=>"", "created_at(5i)"=>"", "updated_at(1i)"=>"", "updated_at(2i)"=>"", "updated_at(3i)"=>"", "updated_at(4i)"=>"", "updated_at(5i)"=>"", "episodes_attributes"=>{"1391733477006"=>{"episode"=>"1", "title"=>"Episode 1", "description"=>"", "duration"=>"6:11", "age_range"=>"", "embed_id"=>"3", "video"=>"b3f85199", "approved"=>"1", "tag_list"=>""}}, "channel_ids"=>["", "3"]}, "commit"=>"Create Show"}
2014-02-07T00:38:28.089273+00:00 app[web.1]: PG::UniqueViolation: ERROR: duplicate key value violates unique constraint "index_episodes_on_slug"
2014-02-07T00:38:28.089273+00:00 app[web.1]: DETAIL: Key (slug)=(episode-1) already exists.
2014-02-07T00:38:28.093312+00:00 app[web.1]: Completed 500 Internal Server Error in 84ms
2014-02-07T00:38:28.099544+00:00 app[web.1]: ActiveRecord::RecordNotUnique (PG::UniqueViolation: ERROR: duplicate key value violates unique constraint "index_episodes_on_slug"
Мой вопрос заключается в следующем: что мне нужно сделать в admin / show.rb или admin / episode.rb, чтобы не только сохранить эпизод с уже использованным заголовком в шоу, но и уникальным для этого шоу, и как это сделать? Я также получаю это?
Обновить
Я все еще пытаюсь выяснить, где я должен переопределить метод создания / обновления; так как я использую контроллер admin / show для создания нового шоу и вложенных эпизодов, я предполагаю, что мне нужно переопределить его там, но я все еще не уверен в этом:
В admin / show.rb я помещаю это в:
controller do
def find_resource
scoped_collection.friendly.find(params[:id])
end
def create
puts " ACCCCCTTTIVE AADDDDMMMIIIN CREEEEATTTTTTE "
params.inspect
end
end
Что, конечно, ОСТАНАВЛИВАЕТ что-нибудь от сохранения, но я подумал, что это будет хороший тупой тест. Затем я прокомментировал это и вставил нечто похожее в admin / episode.rb, но, кажется, его никогда не вызывали - что позволяет мне думать, что функции сохранения episode.rb там не происходят...
Started POST "/admin/shows" for 127.0.0.1 at 2014-02-06 23:03:57 -0500
Processing by Admin::ShowsController#create as HTML
Parameters: {"utf8"=>"✓", "authenticity_token"=>"S5sXpSWZCLG3lg3x2QSw9p1xuHyGxIJndAqYWQWrC40=", "show"=>{"creator_id"=>"1", "title"=>"new new new", "subtitle"=>"sdf", "description"=>"", "position"=>"", "age_range"=>"", "single"=>"0", "skiplist"=>"0", "promote"=>"0", "approved"=>"0", "created_at(1i)"=>"", "created_at(2i)"=>"", "created_at(3i)"=>"", "created_at(4i)"=>"", "created_at(5i)"=>"", "updated_at(1i)"=>"", "updated_at(2i)"=>"", "updated_at(3i)"=>"", "updated_at(4i)"=>"", "updated_at(5i)"=>"", "episodes_attributes"=>{"1391745822502"=>{"episode"=>"", "title"=>"Part 1", "description"=>"", "duration"=>"", "age_range"=>"", "embed_id"=>"1", "video"=>"1234", "approved"=>"0", "tag_list"=>""}}, "channel_ids"=>[""]}, "commit"=>"Create Show"}
AdminUser Load (38.1ms) SELECT "admin_users".* FROM "admin_users" WHERE "admin_users"."id" = 1 ORDER BY "admin_users"."id" ASC LIMIT 1
(149.2ms) BEGIN
Show Exists (18.7ms) SELECT 1 AS one FROM "shows" WHERE "shows"."slug" = 'new-new-new' LIMIT 1
Episode Exists (23.1ms) SELECT 1 AS one FROM "episodes" WHERE "episodes"."show_id" IS NULL AND "episodes"."slug" = 'part-1' LIMIT 1
SQL (205.8ms) INSERT INTO "shows" ("age_range", "approved", "created_at", "creator_id", "description", "promote", "single", "skiplist", "slug", "subtitle", "title", "updated_at") VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11, $12) RETURNING "id" [["age_range", ""], ["approved", false], ["created_at", Thu, 06 Feb 2014 23:03:57 EST -05:00], ["creator_id", 1], ["description", ""], ["promote", false], ["single", false], ["skiplist", false], ["slug", "new-new-new"], ["subtitle", "sdf"], ["title", "new new new"], ["updated_at", Thu, 06 Feb 2014 23:03:57 EST -05:00]]
SQL (22.4ms) INSERT INTO "episodes" ("age_range", "approved", "created_at", "description", "duration", "embed_id", "show_id", "slug", "title", "updated_at", "video") VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11) RETURNING "id" [["age_range", ""], ["approved", false], ["created_at", Thu, 06 Feb 2014 23:03:57 EST -05:00], ["description", ""], ["duration", ""], ["embed_id", "1"], ["show_id", 188], ["slug", "part-1"], ["title", "Part 1"], ["updated_at", Thu, 06 Feb 2014 23:03:57 EST -05:00], ["video", "1234"]]
PG::UniqueViolation: ERROR: duplicate key value violates unique constraint "index_episodes_on_slug"
DETAIL: Key (slug)=(part-1) already exists.
: INSERT INTO "episodes" ("age_range", "approved", "created_at", "description", "duration", "embed_id", "show_id", "slug", "title", "updated_at", "video") VALUES ($1, $2, $3, $4, $5, $6, $7, $8, $9, $10, $11) RETURNING "id"
(1.6ms) ROLLBACK
Completed 500 Internal Server Error in 573ms
PG::UniqueViolation - ERROR: duplicate key value violates unique constraint "index_episodes_on_slug"
DETAIL: Key (slug)=(part-1) already exists.
Обновление № 2
Все еще немного смущен; Я вижу, что он ломается внутри admin / show в контроллере do / def create, но я не могу понять, как лучше всего остановить автоматический процесс сохранения шоу и вложенных эпизодов.
Show Exists (4.3ms) SELECT 1 AS one FROM "shows" WHERE "shows"."slug" = 'blah-black' LIMIT 1
Episode Exists (2.5ms) SELECT 1 AS one FROM "episodes" WHERE "episodes"."show_id" IS NULL AND "episodes"."slug" = 'part-1' LIMIT 1
Episode Exists (2.9ms) SELECT 1 AS one FROM "episodes" WHERE "episodes"."show_id" IS NULL AND "episodes"."slug" = 'part-2' LIMIT 1
Поскольку эпизоды находятся в стадии показа, все равно кажется, что это должно отличаться, поскольку нет других шоу с этими именами эпизодов, которые имеют нулевой идентификатор.
Единственное, о чем я могу думать, - это сделать вручную: сначала сохранить шоу, получить show_id, затем попытаться сохранить каждый вложенный эпизод с этим show_id, но не уверен, правильно ли я думаю об этом...
3 ответа
PG::UniqueViolation - ERROR: duplicate key value violates unique constraint "index_episodes_on_slug"
Похоже, эта ошибка исходит от вашей базы данных, а не от активного администратора. Возможно, вам придется изменить уникальный индекс в вашей миграции, чтобы включить show_id.
Так что в вашей миграции вместо:
add_index :episodes, :slug, :unique => true
Вы должны иметь что-то вроде
add_index :episodes, [:show_id, :slug], :unique => true
Рассматривали ли вы использование belongs_to
?
приложение / администратор /shows.rb:
ActiveAdmin.register Show do
...
end
приложение / администратор / episodes.rb:
ActiveAdmin.register Episode do
belongs_to :show
...
end
ActiveAdmin зарегистрирует вложенные маршруты для ваших ресурсов, аля:
http://server.com/shows/1/episodes/2
Приложение / модели /episode.rb:
class Episode < ActiveRecord::Base
....
friendly_id :title, :use => [:slugged, :scoped], :scope => :show
....
end
приложение / администратор /episodes.rb:
ActiveAdmin.register Episode do
belongs_to :show, :optional => true
...
end
Работал нормально для меня.