The Monster Role Abilities
Unlike the passive monster abilities, of which there were 218, we only have 57 role abilities to worry about, and they can be found in this FAQ from Krystal109 on GameFAQs.com. Since these abilities are already in two HTML tables, it's easy to copy-and-paste them directly into a spreadsheet and tweak it the way we want it. I used Google sheets, and removed the "Best Infusion Source," "Learned by," and "Comments" columns because we don't need them. Then I split the merged cells in the "Role" column and copied the role names as necessary. Finally, I added a column for whether the ability is infusable or not, depending on which of the two tables it came from, before exporting the sheet to a .csv file.But wait! There are actually 70 infusable role abilities in the Monster Infusion FAQ that we've been using, plus the non-infusable role abilities from the second HTML table, so we're missing a few. We are dealing with the common occurrence of incomplete data here. I had to cross-check the lists and add in the missing role abilities from the Monster Infusion FAQ. I ended up with 87 abilities in the end. This manual process is still probably faster than writing a script, but we're not sure, yet, if we have all of the non-infusable role abilities. We'll find out in a minute when we try to link up this data with the monster table.
Before we can add these associations, we have to generate a model of the role abilities table in Rails, like so:
$ rails g model RoleAbility name:string role:string infusable:booleanclass CreateRoleAbilities < ActiveRecord::Migration[6.0]
  def change
    create_table :role_abilities do |t|
      t.string :name
      t.string :role
      t.boolean :infusable
      t.timestamps
    end
  end
end# passive abilities import from monster_abilities.csv
# ...
csv_file_path = 'db/monster_role_abilities.csv'
CSV.foreach(csv_file_path, {headers: true}) do |row|
  role_ability = row.to_hash
  role_ability['infusable'] = (role_ability['infusable'] == 'true')
  RoleAbility.create!(role_ability)
  puts "#{row['name']} added!"
end
# monster data import from monsters.csv
# ...CSV.foreach(csv_file_path, {headers: true}) do |row|
  monster = row.to_hash
  # ... associate passive abilities ...
  monster.keys.select { |key| key.include? '_skill'  }.each do |key|
    if monster[key]
      monster[key] = RoleAbility.find_by(name: monster[key])
      if monster[key].nil?
        puts "ERROR: monster #{monster['name']} #{key} not found!"
        return
      end
      puts "Found #{key} #{monster[key].name}"
    end
  end
  Monster.create!(monster)
  puts "#{row['name']} added!"
endclass CreateMonsters < ActiveRecord::Migration[6.0]
  def change
    create_table :monsters do |t|
      # ... all of the other monster attributes ...
      t.references :default_skill1
      t.references :default_skill2
      t.references :default_skill3
      t.references :default_skill4
      t.references :default_skill5
      t.references :default_skill6
      t.references :default_skill7
      t.references :default_skill8
      t.references :lv_02_skill
      t.references :lv_03_skill
      # ... etc ...
    end
    # ... passive ability foreign keys ...
    
    add_foreign_key :monsters, :role_abilities, column: :default_skill1_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :default_skill2_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :default_skill3_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :default_skill4_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :default_skill5_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :default_skill6_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :default_skill7_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :default_skill8_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :lv_02_skill_id, primary_key: :id
    add_foreign_key :monsters, :role_abilities, column: :lv_03_skill_id, primary_key: :id
    # ... etc ...
  end
endclass Monster < ApplicationRecord
  # ... passive ability belongs_to declarations
  belongs_to :default_skill1, class_name: 'RoleAbility', optional: true
  belongs_to :default_skill2, class_name: 'RoleAbility', optional: true
  belongs_to :default_skill3, class_name: 'RoleAbility', optional: true
  belongs_to :default_skill4, class_name: 'RoleAbility', optional: true
  belongs_to :default_skill5, class_name: 'RoleAbility', optional: true
  belongs_to :default_skill6, class_name: 'RoleAbility', optional: true
  belongs_to :default_skill7, class_name: 'RoleAbility', optional: true
  belongs_to :default_skill8, class_name: 'RoleAbility', optional: true
  belongs_to :lv_02_skill, class_name: 'RoleAbility', optional: true
  belongs_to :lv_03_skill, class_name: 'RoleAbility', optional: true
  # ... etc ...
endclass RoleAbility < ApplicationRecord
  has_many :default_skill1_monsters, :class_name => 'Monster', :foreign_key => 'default_skill1'
  has_many :default_skill2_monsters, :class_name => 'Monster', :foreign_key => 'default_skill2'
  has_many :default_skill3_monsters, :class_name => 'Monster', :foreign_key => 'default_skill3'
  has_many :default_skill4_monsters, :class_name => 'Monster', :foreign_key => 'default_skill4'
  has_many :default_skill5_monsters, :class_name => 'Monster', :foreign_key => 'default_skill5'
  has_many :default_skill6_monsters, :class_name => 'Monster', :foreign_key => 'default_skill6'
  has_many :default_skill7_monsters, :class_name => 'Monster', :foreign_key => 'default_skill7'
  has_many :default_skill8_monsters, :class_name => 'Monster', :foreign_key => 'default_skill8'
  has_many :lv_02_skill_monsters, :class_name => 'Monster', :foreign_key => 'lv_02_skill'
  has_many :lv_03_skill_monsters, :class_name => 'Monster', :foreign_key => 'lv_03_skill'
  # ... etc ...
end$ mv <old monster migration name> <new monster migration name>
$ rails db:purge
$ rails db:migrate
$ rails db:seedMaking a Game Location Graph
You start the game in New Bodhum 003 AF. Actually, you start in Valhalla, but after getting through the intro segment with Lightning and a bunch of over-the-top cut scenes, the game really starts in New Bodhum with Serah. As you progress through the game, you unlock more and more new areas by jumping through time gates. Some locations have multiple time gates that lead to multiple new locations. Different monsters live in different locations, and which locations they live in is noted as one or more of the monsters' location attributes.We want to capture the graph of these location dependencies in a database table and link the locations to the monster attributes so that we can figure out the earliest possible times in the game that we can acquire monsters with certain abilities. In order to do that, we're going to build a table to represent that graph. We could build this table in a couple of ways. One way is to make an adjacency matrix, but this representation requires a row and column for each node in the graph. In this case such a matrix would be sparsely populated, so we're going to use an adjacency list instead. Each row in the table will have one location's name, and which location leads to this location. Since each location has only one other location as a source, we can get away with this simple table structure. There are only a small number of locations, so the .csv file for this table can be created by hand. Plus, I couldn't find a good list on the internet, so here it is, created from scratch:
name,source
New Bodhum 003 AF,nil
Bresha Ruins 005 AF,New Bodhum 003 AF
Bresha Ruins 300 AF,Bresha Ruins 005 AF
Yaschas Massif 110 AF,Bresha Ruins 300 AF
Yaschas Massif 010 AF,Bresha Ruins 005 AF
Oerba 200 AF,Yaschas Massif 010 AF
Yaschas Massif 01X AF,Oerba 200 AF
Augusta Tower 300 AF,Yaschas Massif 01X AF
Serendipity Year Unknown,Yaschas Massif 01X AF
Sunleth Waterscape 300 AF,Bresha Ruins 005 AF
Coliseum ??? AF,Sunleth Waterscape 300 AF
Archylte Steppe ??? AF,Sunleth Waterscape 300 AF
Vile Peaks 200 AF,Archylte Steppe ??? AF
Academia 400 AF,Sunleth Waterscape 300 AF
Yaschas Massif 100 AF,Academia 400 AF
Sunleth Waterscape 400 AF,Yaschas Massif 100 AF
Augusta Tower 200 AF,Academia 400 AF
Oerba 300 AF,Augusta Tower 200 AF
Oerba 400 AF,Oerba 300 AF
Academia 4XX AF,Augusta Tower 200 AF
The Void Beyond ??? AF,Academia 4XX AF
Vile Peaks 010 AF,Academia 4XX AF
A Dying World 700 AF,Academia 4XX AF
Bresha Ruins 100 AF,A Dying World 700 AF
New Bodhum 700 AF,A Dying World 700 AF
Academia 500 AF,New Bodhum 700 AF
Valhalla ??? AF,Academia 500 AF
Coliseum ??? AF (DLC),New Bodhum 003 AF
Valhalla ??? AF (DLC),New Bodhum 003 AF
Serendipity ??? AF (DLC),New Bodhum 003 AF$ rails g model Location name:string source:referencesclass CreateLocations < ActiveRecord::Migration[6.0]
  def change
    create_table :locations do |t|
      t.string :name
      t.references :source
      t.timestamps
    end
  end
endclass CreateMonsters < ActiveRecord::Migration[6.0]
  def change
    create_table :monsters do |t|
      t.string :name
      t.string :role
      t.references :location, foreign_key: true
      t.references :location2, foreign_key: {to_table: :locations}
      t.references :location3, foreign_key: {to_table: :locations}
      # ... the mess of other attribute declarations ...
    end
    # ... a mess of foreign key declarations ...
  end
end
The next step is to update the seeds.rb script for the location and monster models. Let's do the models first this time. The location model will need a self-reference for the source attribute and declare that it has many monsters for the location attributes in the monster model, like so:
class Location < ApplicationRecord
  belongs_to :source, class_name: 'Location', optional: true
  has_many :location_monsters, :class_name => 'Monster', :foreign_key => 'location'
  has_many :location2_monsters, :class_name => 'Monster', :foreign_key => 'location2'
  has_many :location3_monsters, :class_name => 'Monster', :foreign_key => 'location3'
endclass Monster < ApplicationRecord
  belongs_to :location, class_name: 'Location', optional: true
  belongs_to :location2, class_name: 'Location', optional: true
  belongs_to :location3, class_name: 'Location', optional: true
  # ... a mess of other reference declarations ...
end# ... role abilities import from monster_role_abilities.csv ...
csv_file_path = 'db/monster_locations.csv'
CSV.foreach(csv_file_path, {headers: true}) do |row|
  location = row.to_hash
  location['source'] = Location.find_by(name: location['source'])
  if location['source'].nil? && location['name'] != 'New Bodhum 003 AF'
    puts "ERROR: location #{location['source']} not found!"
    return
  end
  Location.create!(location)
  puts "Location #{location['name']} added!"
end
# ... monster data import from monsters.csv ...CSV.foreach(csv_file_path, {headers: true}) do |row|
  monster = row.to_hash
  # ... associate role abilities ...
  monster.keys.select { |key| key.starts_with? 'location'  }.each do |key|
    if monster[key]
      monster[key] = Location.find_by(name: monster[key])
      if monster[key].nil?
        puts "ERROR: monster #{monster['name']} #{key} not found!"
        return
      end
      puts "Found #{key} '#{monster[key].name}'"
    end
  end
  Monster.create!(monster)
  puts "#{row['name']} added!"
end$ mv <old monster migration name> <new monster migration name>
$ rails db:purge
$ rails db:migrate
$ rails db:seedWrapping up Monster Materials and Characteristics
This article is getting pretty long, so I'm going to finish off the monster materials and characteristics tables quickly. The characteristics is a straightforward table that doesn't require figuring out anything new. The materials table can also be straightforward, at least for now. The monster materials themselves constitute a simple table with a name, a grade of 1-5, and a type of either biological or mechanical. The trick is that we'll eventually want to know when in the game we can get each grade of monster material because they're used to upgrade our tamed monsters. The way to acquire those materials is by defeating other monsters in battle. The different materials are some of the loot that the monsters drop. This information is all in the Monster Infusion FAQ, but it would require another parser to extract it and import it into the monster table. We can do that later in the series, but right now I want to complete the tables and get on with starting to view the data so we'll just go with making the simple material table.A table of the materials can be found in the same HTML FAQ from Krystal109. Like the first set of role abilities, we can copy this table into a spreadsheet and then export it to a .csv file. Then we run through the same steps of importing this data into our Rails database, starting with generating a material model:
$ rails g model Material name:string grade:integer material_type:string# ... location import from monster_locations.csv ...
csv_file_path = 'db/monster_materials.csv'
CSV.foreach(csv_file_path, {headers: true}) do |row|
  Material.create!(row.to_hash)
  puts "Material #{row['name']} added!"
end
# ... monster data import from monsters.csv ...$ rails g model Characteristic name:string description:string# ... material import from monster_materials.csv ...
csv_file_path = 'db/monster_characteristics.csv'
CSV.foreach(csv_file_path, {headers: true}) do |row|
  Characteristic.create!(row.to_hash)
  puts "Characteristic #{row['name']} added!"
end
# ... monster data import from monsters.csv ...$ rails db:purge
$ rails db:migrate
$ rails db:seedThat last step will wrap up the database design and building, for now. We'll have a bit of work to do later when we want to connect the monster materials to the monsters that drop them, but this is good enough to get started with creating views of this data. After spending multiple articles parsing, exporting, importing, and creating data for the database, I'm anxious to get moving on those views, so that's what we'll start looking at next time.
Exploring Monster Taming Mechanics Table of Contents:

No comments:
Post a Comment