A library for managing a standardized set of seed data for applications in a non-production environment.
- Rails
- Postgresql (This library currently only works with the Postgresql database)
group :development do
gem 'application_seeds'
end
You should add this gem to any environment group that would need access to the seed data (like an "integration" environment, for example).
This library operates on a set of YAML files that represent your shared data set. The dataset can be provided two different ways.
In order to easily share the seed data between apps, you can package the YAML files into a gem. The gem should have the following directory structure:
lib/
+-- seeds/
|-- seed_data_set_1/
| |-- some_data.yml
| +-- some_other_data.yml
+-- seed_data_set_2/
|-- some_data.yml
+-- some_other_data.yml
The gem may contain any number of datasets. The above example has two datasets,
seed_data_set_1
and seed_data_set_2
. The YAML files are located in the
dataset directories.
You will need to include the gem containing your seed data in your Gemfile
.
Use the data_gem_name
API method to specify where your seed data is located.
ApplicationSeeds.data_gem_name = "my-seed-data-gem"
The dataset may also exist on the filesystem. The directory structure should
be identical to what is described above in the "Via a gem" section, but the lib
diretory is not required.
seeds/
|-- seed_data_set_1/
| |-- some_data.yml
| +-- some_other_data.yml
+-- seed_data_set_2/
|-- some_data.yml
+-- some_other_data.yml
Use the data_directory
API method to specify the path to your seed data on the filesystem.
ApplicationSeeds.data_directory = "/path/to/seeds"
The application needs to create objects from the common seed data. To do this, you will need to create a Rake task (such as the one below) for your application that reads the seed data, and uses it to create the objects in the application's own data model.
ApplicationSeeds
provides an API to allow for the easy retrieveal of
seed data. See below for more information about the API.
namespace :application_seeds do
desc 'Dump the development database and load it with standardized application seed data'
task :load, [:dataset] => ['db:drop', 'db:create', 'db:migrate', :environment] do |t, args|
ApplicationSeeds.data_gem_name = "my-seed-data-gem"
ApplicationSeeds.dataset = args[:dataset]
seed_campaigns
seed_line_items
seed_some_other_objects
end
def seed_campaigns
# If we do not need to change the attirbute hash, we can just create
# the object with the attributes that are specified in the seed data
# file.
ApplicationSeeds.campaigns.each do |id, attributes|
ApplicationSeeds.create_object!(Campaign, id, attributes)
end
end
def seed_line_items
# If we need to reject attributes from the attribute hash, or
# only use specific attributes, we can use the select_attributes or
# the reject_attributes helper methods.
ApplicationSeeds.line_items.each do |id, attributes|
ApplicationSeeds.create_object!(LineItem, id, attributes.reject_attributes(:some_unused_attribute))
end
end
def seed_some_objects
# If we need to modify attribute names, we can do so using the
# map_attributes helper method.
ApplicationSeeds.some_objects.each do |id, attributes|
ApplicationSeeds.create_object!(SomeObject, id, attributes.map_attributes(
:old_name1 => :new_name1, :old_name2 => :new_name2))
end
end
def seed_some_other_objects
# If we need tighter control over how the object is created, we can
# simply create it ourselves.
ApplicationSeeds.some_other_objects.each do |id, attributes|
x = SomeOtherObject.new(param1: attributes['param1'],
param2: attributes['param2'],
param3: attributes['param3'])
x.id = id
x.save!
end
end
end
bundle exec rake application_seeds:load[your_data_set]
You must specify the seed data set that you would like to use. The dataset name is simply the name of the directory containing the seed YAML files.
Add the following line to your deploy.rb file:
require "application_seeds/capistrano"
Then, you can seed a remote database by running the following:
bundle exec cap <environment> deploy:application_seeds -s dataset=your_data_set
The application_seeds
library supports multiple datasets within the
same source (place on the file system, gem, etc). The user specifies
which dataset to load when beginning to work with the seed data.
ApplicationSeeds.dataset = "my_data_set"
Datasets can be structured so that child directories can inherit the seed data files that are stored in the parent directories. For example, let's look at the following directory structure:
seeds/
+-- parent_data_set/
|-- companies.yml
+-- child_data_set/
|-- departments.yml
+-- grandchild_data_set/
+-- people.yml
In this example, if the grandchild_data_set
is loaded, you will have
access to the seed data files in grandchild_data_set
,
child_data_set
, and parent_data_set
. Because of this, data from
people.yml
, departments.yml
, and companies.yml
can be loaded.
If child_data_set
is loaded, you will have access to the seed data
files in child_data_set
and parent_data_set
, but not
grandchild_data_set
. This includes the departments.yml
and
companies.yml
data files.
It is possible to have files for the same data type scattered throughout the dataset hierarchy.
seeds/
+-- parent_data_set/
|-- companies.yml
|-- people.yml
+-- child_data_set/
|-- departments.yml
|-- people.yml
+-- grandchild_data_set/
+-- people.yml
In this example, when data is loaded from the people
dataset via call
to ApplicationSeeds.people
, then the result will contain the data from
all three files.
If the files contain conflicting labels, then precedence is given to
data at the lowest level (grandchild_data_set
in this example).
The seed files contain the data that the Rake task works with to populate the database. The seed files look and work much like Rails fixtures files.
Here is an example from the people.yml
file in this library's test
suite:
joe_smith:
first_name: Joe
last_name: Smith
company_id: mega_corp
start_date: <%= 2.months.ago.to_date %>
jane_doe:
first_name: Jane
last_name: Doe
company_id: mega_corp
start_date: <%= 10.months.ago.to_date %>
Seed data must contain a label that is unique to the file.
Seed files may contain ERB snippets to support more dynamic data, or data that may change over time.
Relationships can be established between seed data files using labels.
One piece of seed data can specify a belongs_to
relationship with
another piece of seed data by specifying the other data's label in the
_id
field.
In this example, ApplicationSeeds
will look in the companies.yml
file for a seed data element with the label mega_corp
.
company_id: mega_corp
If the _id
field does not share a name with the file that the
corresponding seed data can be found, you can specify the name of the
seed file, like so:
employer_id: mega_corp (companies)
Many to many relationships can be specified using arrays. The name of
the field must end in ids
or uuids
.
company_ids: [mega_corp, ma_and_pa]
If the _ids
field does not share a name with the file that the
corresponding seed data can be found, you can specify the name of the
seed file, like so:
employer_ids: "[mega_corp, ma_and_pa] (companies)"
Here, the array must be enclosed in a string, to prevent the YAML parser from erroring out due to invalid YAML syntax.
By default, ApplicationSeeds
will generate a unique ID for each piece
of seed data based on the name of the file containing the data and the
data's label. The IDs will not change, as long as the name of the file
containing the seed data and the labels do not change.
If you need to specify a specific id for a piece of seed data, you can specify the id in the list of attributes.
joe_smith:
id: 123
first_name: Joe
last_name: Smith
company_id: mega_corp
start_date: <%= 2.months.ago.to_date %>
ApplicationSeeds
will not generate an ID for you if one has been
specified.
See spec/seed_data/test_data_set
for more examples of seed data files.
The ApplicationSeeds
module can generate integer or UUID ids. You can
use the config
method to tell ApplicationSeeds
which id type you would
like to use.
ID types can be specified at the global level (to be applied to all seed data types)...
ApplicationSeeds.config = { :id_type => :uuid }
...at the data type level (if some types have UUID primary keys and other have integer primary keys)...
ApplicationSeeds.config = { :people_id_type => :uuid, :companies_id_type => :integer }
...or a combination of both (if every type uses integer primary keys except for one, for example)
ApplicationSeeds.config = { :id_type => :uuid, :companies_id_type => :integer }
integer
is the default id type.
config
needs to be called before the dataset is specified using dataset=
The ApplicationSeeds
module provides an API that enables the programmatic retrieval of seed data,
so the rake task can easily access all the seed data necessary to build the data object.
ApplicationSeeds.data_directory = "/path/to/seeds/directory"
Specify the name of the directory that contains the application seed data.
ApplicationSeeds.data_gem_name = "my-seed-data-gem"
Specify the name of the gem that contains the application seed data.
Defaults to application_seed_data
if this method is not called.
ApplicationSeeds.dataset = "name_of_your_dataset"
Specify the name of the dataset to use. An exception will be raised if the dataset could not be found.
ApplicationSeeds.dataset
Returns the name of the dataset that has been loaded, or nil if not running an application_seeds dataset.
ApplicationSeeds.seed_data_exists?(:campaigns)
Returns true
if campaigns.yml
exists in this dataset, false
if it
does not.
ApplicationSeeds.campaigns # where "campaigns" is the name of the seed file
This call returns a hash with one or more entries (depending on the contentes of the seed file). The IDs of the object are the keys, and a hash containing the object's attributes are the values. An exception is raised if no seed data could be with the given name.
ApplicationSeeds.campaigns(:some_campaign) # where "campaigns" is the name of the seed file, and :some_campaign is the label of the campaign
This call returns a hash containing the object's attributes. An exception is raised if no seed data could be found with the given label.
ApplicationSeeds.campaigns(1) # where "campaigns" is the name of the seed file, and 1 is the ID of the campaign
This call returns a hash containing the object's attributes. An exception is raised if no seed data could be found with the given ID.
ApplicationSeeds.campaigns(foo: 'bar', name: 'John') # where "campaigns" is the name of the seed file
This call returns the seed data that contains the specified attributes, and the specified attribute values. It returns a hash with zero or more entries. The IDs of the object are the keys of the hash, and a hash containing the object's attributes are the values. Any empty hash will be returned if no seed data could be found with the given attribute names and values.
ApplicationSeeds.create_object!(Campaign, id, attributes)
This call will create a new instance of the Campaign
class, with the
specified id and attributes.
ApplicationSeeds.create_object!(Campaign, id, attributes.reject_attributes(:unused_attribute))
This call will create a new instance of the Campaign
class without the
unused_attribute
attribute.
ApplicationSeeds.create_object!(Campaign, id, attributes.select_attributes(:attribute1, :attribute2))
This call will create a new instance of the Campaign
class with only the
attribute1
and attribute2
attributes.
ApplicationSeeds.create_object!(Campaign, id, attributes.map_attributes(
:old_name1 => :new_name1, :old_name2 => :new_name2))
This call will create a new instance of the Campaign
class, using the
seed data for old_name1 as the attribute value for new_name1, and the
seed data for old_name2 as the attribute value for new_name2. This
method let's you easly account for slight differences is attribute names
across applications.
ApplicationSeeds.reset_sequence_numbers
This method will reset the sequence numbers on id columns for all tables in the database with an id column. If you are having issues where you are unable to insert new data into the databse after your dataset has been imported, then this should correct them.
Applications in a service oriented architecture (SOA) are often interconnected. One of the challenges with a SOA is that, since the applications are (and must be to some extent) all interconnected, the data sets used by the different applications must be in sync.
Applications will need to store keys to data in other applications that can be used to fetch more detailed information from the services that own that data. In order for one application to lookup data owned by another application, the key specified by the client must be in the server's data set, along with the other data associated with the key that the client is requesting.
Often, each application will have its own, siloed seed data, making inter-app communication impossible. In order to get all of the application data in sync, developers will often resort to populating their development databases with production data. Production data on a developer machine (especially a laptop) is bad business. Do you want to send the email to all of your customers telling them that their sensitive data was on a stolen laptop? I didn't think so.
The goal of this project is to create a common set of seed data that can be used by all applications running in development. Re-seeding the applications in development with this shared seed data would put them all "on the same page", preparing them for inter-app communication.
The seed data would be in a general format, not formatted to any application's data model. Each application will have a script that mutates this seed data to confirm to its data model, and then persist it to its database.
Easier said than done :) Yes, it would be fantastic if we could run an application in isolation, and everything just works. But maintaining the stubs can be difficult. Also, when you stub out service calls, you're not really testing the inter-app communication process. More importantly, stubbing out the calls really only works for read-only APIs. For APIs that create or mutate data, stubbing isn't an ideal strategy. What happens when the app tries to fetch data that it just created/updated on a remote service? How will you see the data you created/updated?
Not really. But, you will need to be running the applications that service API calls for whatever it is that you are developing/testing. This is where POW comes in. POW is a zero-config Rack server for OSX. After installing POW, your apps will be accessible via a .dev url, like http://myapp.dev No more remembering to start an application before you use one of its services. No more remembering which applications run on which ports. If your application is not currently running, POW will start it automatically on the fly.
Making it easier for our applications to talk to one another does have some disadvantages. One being that it makes it easier to couple applications. The goal of a service oriented architecture is to prevent this. With great power comes great responsibility. Carefully consider the trade offs any time you introduce an API call to fetch data from a remote service.