Rails - передача случайно сгенерированной переменной экземпляра между маршрутами - PullRequest
0 голосов
/ 29 апреля 2018

Я создаю приложение, которое случайным образом отображает «карточку» для языковой практики, которой пользователь должен предоставить перевод. У меня есть две таблицы: карточки (содержащие данные карточки) и практики (для управления практическими занятиями и хранения предоставленных ответов).

Мне удалось получить флэш-карту для случайного отображения, когда пользователь начинает тренировочный сеанс, но я не могу понять, как настроить код так, чтобы, когда пользователь предоставляет ответ, программа запоминала / знала, какой случайный карта была отображена, чтобы он мог правильно POST для практики. Мой текущий код генерирует новую карту между показом карты и сохранением ее в базе данных, поскольку @flashcard не сохраняется между маршрутами.

# practices_controller.rb

def index
  call_random
end

def create
  call_random
  @practice = Practice.create(practice_params.merge(session_id: @session_id, flashcard_id: @flashcard.id))
  if @practice.save
    redirect_to @practice
  else
    render 'index'
  end
end

def show
end

private

def find_practice
  @practice = Practice.find(params[:id])
end

def random_flashcard
  @flashcard = Flashcard.order("RANDOM()").first
end

def call_random
  @flashcard ||= random_flashcard
end

Я пытался установить и прочитать атрибуты класса и другие частные методы, чтобы попытаться сохранить случайно сгенерированную карточку (либо весь объект, либо идентификатор), но ни одно решение не помогло мне. Один из методов, который, как мне показалось, мог бы приблизить меня, заключался в следующем, но он все равно не удался, поскольку @flashcards все еще находился на nil в маршруте создания.

# practice_controller.rb

def index
  random_flashcard(:generate)
end

def create
  random_flashcard(:recall)
  @practice = Practice.create(practice_params.merge(session_id: @session_id, flashcard_id: @flashcard.id))
  if @practice.save
    redirect_to @practice
  else
    render 'index'
  end
end

private

def random_flashcard(action = :generate)
  if action == :generate
    @flashcard = Flashcard.order("RANDOM()").first
  elsif action == :recall
    @flashcard
  end
end

1 Ответ

0 голосов
/ 30 апреля 2018

Я думаю, что вам не хватает концепции, как работает цикл запрос / ответ. В Интернете есть много информации об этом, включая Руководства по Rails, поэтому я буду кратким и упрощенным:

  1. Ваш браузер отправляет запрос в приложение Rails (запрос GET будет выглядеть как «покажи мне этот URL», а запрос POST будет выглядеть как «сделать это, представленное этим URL, с использованием прикрепленных данных») .
  2. Ваше приложение получает этот запрос, маршрутизатор Rails находит соответствующий контроллер / действие, а ваше действие выполняет некоторые действия и отображает шаблон, который Rails помогает превратить в ответ.
  3. Ваш браузер получает этот HTML-ответ и отображает его в браузере.

Вот и все. HTTP не имеет состояния. По умолчанию ваше приложение не сохраняет состояния от запроса к запросу. Вы когда-нибудь видели 50 первых дат? Веб-серверы - это персонаж Дрю Бэрримора, и каждый запрос - это день. Ваше приложение запоминает все, что произошло в этом запросе, но оно начинается с чистого листа при следующем запросе без памяти о последнем. Есть способы обойти это, например, сохранение состояния в файлах cookie, URL-адрес или данные формы, но вы должны сделать это самостоятельно. По умолчанию каждый запрос не имеет связи с последним.

В вашем случае вы имеете дело с двумя запросами. Первый - это GET для вашего index, второй - POST для вашего create действия. Каждый запрос приводит к созданию нового экземпляра контроллера со свежим состоянием. Ваше приложение по умолчанию не видит связи между этими двумя запросами. Всегда ли действие create вызывается после индекса? Ваше приложение не известно или не заботится. Это зависит от вас, разработчик.

Я думаю, вы ожидаете, что ваша переменная @flashcard сохранится между вашим начальным запросом index и последующим POST до create. Но представьте, как это будет работать с несколькими пользователями. Два пользователя загружают индекс, затем пользователь A отправляет сообщения create, за которым следует пользователь B. Как приложение различает их? Как бы он узнал, какую карту каждый просматривал? Ну, это зависит от вас.

Существует несколько способов сохранения состояния между запросами - вы часто будете видеть долговременное состояние, отслеживаемое как часть «сеанса» (cookie), который выходит за рамки ответа - но я Я рекомендую вам прочитать их, так как они очень важны для веб-разработки.

Однако для кратковременного состояния - такого как постоянное состояние между одним запросом -> ответ -> запрос - особенно когда последний запрос является формой POST - вы можете использовать поле формы - и это именно то, как я бы решить вашу проблему.

  1. Не устанавливайте случайную карточку в вашем действии create.
  2. Добавьте flashcard_id скрытое поле к форме, которую вы визуализируете в вашем index.
  3. Когда эта форма будет отправлена, ваше действие create получит параметр flashcard_id. Используйте этот идентификатор для создания вашей записи Practice.

На простом английском это работает следующим образом: ваш пользователь запрашивает страницу со случайной картой (GET index). Ваше приложение отвечает с помощью некоторого HTML-кода, который будет отображать флэш-карту, а также скрытое поле, содержащее его идентификатор. Пользователь отправляет форму, инициируя новый запрос (POST create), и идентификатор карточки отправляется обратно на сервер как часть данных формы этого запроса. Ваше приложение получает этот запрос, ожидает, что этот идентификатор будет включен в данные формы (в хэше params в Rails), и создает Practice с использованием этого идентификатора.

...