Multiple Foreign Keys
I’d like to document a problem that I’ve never had before, and the solution I arrived at. It was not a real complicated problem, but I got hung up on an even simpler problem.
Say I want to store directories in the database, and the files within those directories in an associated table. The actual problem is much more specific, but I’ve made it generic. Here are some sample entries:
id: 1, name: home id: 2, name: example
a_dir_id: 1, filename: 'test', content: 'some content'
Looking at this, it seemed fairly basic. The catch was that the directory id and filename would have to be a composite primary key, which I haven’t really used all that much. On top of that, the directory id also needs to be a foreign key. I read a bit and decided that this was probably doable, so this is where I got to:
require 'data_mapper' DataMapper.setup(:default, 'mysql://[email protected]/sample') class ADir include DataMapper::Resource property :id, Serial property :name, String, :unique => true has n, :a_files end class AFile include DataMapper::Resource property :a_dir_id, Integer, :key => true property :filename, String, :key => true property :content, Text belongs_to :a_dir end DataMapper.auto_migrate!
Running this, I ended up with this cryptic error:
...Can't create table 'sample.#sql-1ed_113' (errno: 150) (DataObjects::SQLError)
It was not a helpful error message. I assumed that what I was trying was simply not possible, but I blazed through and started writing the raw SQL to acheive what I wanted, and it turns out that it certainly is possible in MySQL.
After many rabbit trails, it dawned on me that
Serial columns are unsigned and
Integer columns are not. Adding a
:min argument to this one line made it
work like a charm:
property :a_dir_id, Integer, :min => 0, :key => true