简体   繁体   English

Rails - 活动记录“RuntimeError:无法修改冻结的字符串”以某种方式与表单生成器相关?

[英]Rails - Active Record “RuntimeError: can't modify frozen String” somehow related to form builder?

When I submit the form to update my models I'm getting this error in the browser: 当我提交表单来更新我的模型时,我在浏览器中收到此错误:

ActiveRecord::StatementInvalid in CoachesController#update

RuntimeError: can't modify frozen String: INSERT INTO "availabilities" ("coach_id", "created_at", "day", "hour", "updated_at") VALUES (?, ?, ?, ?, ?)

The rails console says this: rails控制台说:

(0.0ms)  begin transaction
Binary data inserted for `string` type on column `day`
SQL (0.5ms)  INSERT INTO "availabilities" ("coach_id", "created_at", "day", "hour", "updated_at") VALUES (?, ?, ?, ?, ?)  [["coach_id", 14], ["created_at", Mon, 27 Feb 2012 21:59:05 UTC +00:00], ["day", "Monday"], ["hour", 20], ["updated_at", Mon, 27 Feb 2012 21:59:05 UTC +00:00]]
RuntimeError: can't modify frozen String: INSERT INTO "availabilities" ("coach_id", "created_at", "day", "hour", "updated_at") VALUES (?, ?, ?, ?, ?)
(0.1ms)  rollback transaction
Completed 500 Internal Server Error in 25ms

ActiveRecord::StatementInvalid (RuntimeError: can't modify frozen String: INSERT INTO "availabilities" ("coach_id", "created_at", "day", "hour", "updated_at") VALUES (?, ?, ?, ?, ?)):
app/models/coach.rb:93:in `block (2 levels) in update_general_availability'
app/models/coach.rb:92:in `each'
app/models/coach.rb:92:in `block in update_general_availability'
app/models/coach.rb:91:in `each'
app/models/coach.rb:91:in `update_general_availability'
app/controllers/coaches_controller.rb:25:in `update'

After much experimentation I've found how to work around this error, but not why I'm getting it in the first place. 经过多次实验,我发现如何解决这个错误,但不是为什么我首先得到它。

I have two models: Coach and Availabilities , with has_many and belongs_to associations. 我有两个模型: CoachAvailabilities ,有has_manybelongs_to关联。 This is the schema for the availability table: 这是可用性表的架构:

# Table name: availabilities
#  id         :integer         not null, primary key
#  coach_id   :integer
#  day        :string(255)
#  hour       :integer

It stores a day of the week and hour during the day that a coach is free. 它存储了教练免费的一天中的一天和一小时。

I wrote two methods in the Coach model to more easily deal with a coach's weekly availability. 我在Coach模型中编写了两种方法,以便更轻松地处理教练的每周可用性。 They use a nested hash table, so you can query whether a coach is free at a given time. 它们使用嵌套哈希表,因此您可以查询教练在给定时间是否空闲。 (Eg: general_availability["Thursday"]["12"] #=> true ) (例如: general_availability["Thursday"]["12"] #=> true

#coach.rb
class Coach < ActiveRecord::Base
  ...

  # Creates a hash table mapping day and hour to true if available then, false otherwise
  # Form is general_availability["day"]["hr"]. Per Availability model, "0" = midnight, and
  # day of the week is of the form "Monday" or "Tuesday".
  def general_availability
    h = Hash.new()
    %w(Monday Tuesday Wednesday Thursday Friday Saturday Sunday).each { |day| h[day] = Hash.new(false) }

    self.availabilities.each do |a|
      h[a.day][a.hour.to_s] = true
    end

    return h
  end

  # Takes a hash table of the kind returned by general_availability and updates
  # this coach's records in the Availabilities table
  def update_general_availability(ga_hash_table)
    self.availabilities.clear

    ga_hash_table.each do |day, hrs|
      hrs.each do |hr, val|
        self.availabilities.create({day: day, hour: hr.to_i})
      end
    end
  end

This is the partial to display the coach's weekly availability as a table. 这是部分显示教练的每周可用性作为表格。 Each day/hour cell is a checkbox which a coach can check or uncheck to indicate if they're free. 每天/每小时的单元格是一个复选框,教练可以选中或取消选中以指示它们是否空闲。

<!-- availabilities/_scheduler.html.erb -->
<h2>General Availability</h2>
Please check the times below that you would generally be available for a training session.
<table class="table" id="availabilities_table">
  <tr>
    <th>Time</th>
    <% days_of_the_week.each do |day| %> 
      <th><%= day %></th> 
    <% end %>
  </tr>
  <% (6..21).each do |hr| %>
    <tr>
      <td><%= format_as_time hr %></td>
      <% days_of_the_week.each do |day| %>
        <% is_checked = @general_availability[day][hr.to_s] %>
        <td class="availabilities_cell">
          <%= check_box_tag "availability[#{day}][#{hr}]", true, is_checked, :class => 'availabilities_check_box' %>
        </td>
      <% end %>
    </tr>
  <% end %>
</table>

This is the controller: 这是控制器:

# coaches_controller.rb
...
def edit
  @coach = current_user.coach
  @general_availability = @coach.general_availability
end

def update
  @coach = Coach.find(params[:id])

  @coach.update_attributes(params[:coach])  
  if @coach.save
    @coach.update_general_availability(params[:availability])
    redirect_to @coach
  end
  # ...
end

It is the line 这是线

@coach.update_general_availability(params[:availability])

that causes the error. 这会导致错误。

Now, here is my question. 现在,这是我的问题。 Why does this view cause the above error? 为什么此视图会导致上述错误?

<!-- edit.html.erb version 1 -->
<h1><%= @coach.user.first_name %>  </h1>
<%= form_for @coach, :html => { :multipart => true } do |f| %> 

  <%= f.label :profile_photo %> 
  <%= f.file_field :profile_photo %> 

  <div class="field">
    <%= f.label :phone_number %>
    <%= f.text_field :phone_number %>
  </div>  

  ... More Form Fields Here ...

  <%= render 'availabilities/scheduler' %>
  <%= f.button %> 

<% end %> 

While this view does not? 虽然这种观点没有?

<!-- edit.html.erb version 2 -->
<h1><%=  @coach.user.first_name %>  </h1>
<%= form_for @coach, :html => { :multipart => true } do |f| %> 

  <%= f.label :profile_photo %> 
  <%= f.file_field :profile_photo %> 

  <div class="field">
    <%= f.label :phone_number %>
    <%= f.text_field :phone_number %>
  </div>  

  ... More Form Fields Here ...

  <%= f.button %> 

<% end %> 

<%= form_for @coach, :class => "form-vertical" do |f| %>
  <%= render 'availabilities/scheduler' %>
  <%= submit_tag "Update Schedule" %>
<% end %>

Note that in the former the partial is inside the form builder form, whereas in the second case the partial is rendered as its own form_for below. 请注意,在前者中,partial在表单构建器表单中,而在第二种情况下,partial在下面呈现为自己的form_for

The part that jumps out at me from the log I pasted above is this line: 从上面粘贴的日志中跳出来的部分是这一行:

Binary data inserted for `string` type on column `day`

which does not appear when the form works (eg, in version 2 of the form). 表单工作时不会出现(例如,在表单的第2版中)。 It seems important, but I don't know what it means or why it's happening. 这似乎很重要,但我不知道它意味着什么或为什么会发生。

Thanks so much! 非常感谢!

Figured it out. 弄清楚了。 Ruby hash table keys are frozen. Ruby哈希表键被冻结。 So my params looked like: 所以我的参数看起来像:

params[:availability][:Thursday][:10] = "true"

When my update_general_availability method did this: 当我的update_general_availability方法执行此操作时:

self.availabilities.create({day: day, hour: hr.to_i})

:day was "Thursday" , but the SQLite Adapter understood it to have Encoding::ASCII_8BIT (aka "binary"), and tried do encode! 'utf-8' :day"Thursday" ,但SQLite Adapter知道它有Encoding::ASCII_8BIT (又名“binary”),并尝试encode! 'utf-8' encode! 'utf-8' it. encode! 'utf-8'它。 However, since it was frozen this threw the Runtime Frozen String error. 但是,由于它被冻结,因此抛出了Runtime Frozen String错误。 The problem was solved by adding these lines to the update_general_availability method: 通过将这些行添加到update_general_availability方法来解决该问题:

day = day.dup
hr = hr.dup

Now, since they're duplicated and not the hash keys themselves, they can get encoded to utf-8. 现在,由于它们是重复的而不是哈希键本身,它们可以编码为utf-8。

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM