require File.expand_path('../../../spec_helper', __FILE__)
require 'nokogiri'
module NcsNavigator::Mdes
describe Variable do
describe '.from_element' do
def variable(s)
Variable.from_element(schema_element(s), :log => logger)
end
let(:comments) {
variable(<<-XSD)
XSD
}
let(:sc_id) {
variable('')
}
it 'has the correct name' do
comments.name.should == 'comments'
end
describe '#type' do
context 'when embedded' do
it 'is a VariableType' do
comments.type.should be_a VariableType
end
it 'is parsed from the contents' do
comments.type.max_length.should == 8000
end
it 'is not a reference' do
comments.type.should_not be_reference
end
end
context 'when a named type' do
context 'with the XML schema prefix' do
let!(:subject) { variable('') }
it 'is a VariableType' do
subject.type.should be_a VariableType
end
it 'has a matching base type' do
subject.type.base_type.should == :decimal
end
it 'is not a reference' do
subject.type.should_not be_reference
end
end
context 'with another prefix' do
it 'is a VariableType' do
sc_id.type.should be_a VariableType
end
it 'has the name' do
sc_id.type.name.should == 'ncs:study_center_cl1'
end
it 'is a reference' do
sc_id.type.should be_reference
end
end
end
context 'when none present' do
let!(:subject) { variable('') }
it 'is nil' do
subject.type.should be_nil
end
it 'warns' do
logger[:warn].first.should == 'Could not determine a type for variable "bar" on line 2'
end
end
end
describe '#required?' do
it 'is true when not nillable' do
variable('').should be_required
end
it 'is false when nillable' do
comments.should_not be_required
end
it 'is false when not nillable but minOccurs is 0' do
variable('').should_not be_required
end
it 'is true when nillable omitted but minOccurs is > 0' do
variable('').should be_required
end
end
describe '#pii' do
it 'is false when blank' do
variable('').pii.should == false
end
it 'is true when "Y"' do
variable('').pii.should == true
end
it 'is :possible when "P"' do
variable('').pii.should == :possible
end
it 'is the literal value when some other value' do
variable('').pii.should == '7'
end
it 'is :unknown when not set' do
variable('').pii.should == :unknown
end
end
describe '#status' do
it 'is :active for 1' do
variable('').status.should == :active
end
it 'is :new for 2' do
variable('').status.should == :new
end
it 'is :modified for 3' do
variable('').status.should == :modified
end
it 'is :retired for 4' do
variable('').status.should == :retired
end
it 'is the literal value for some other value' do
variable('').status.should == 'P4'
end
it 'is nil when not set' do
variable('').status.should be_nil
end
end
end
describe '#resolve_type!' do
let(:reference_type) { VariableType.reference('ncs:color_cl3') }
let(:actual_type) { VariableType.new('color_cl3') }
let(:types) { [ actual_type ] }
let(:variable) {
Variable.new('hair_color').tap { |v| v.type = reference_type }
}
context 'when the type is available' do
it 'resolves' do
variable.resolve_type!(types)
variable.type.should be actual_type
end
end
context 'when the type is not resolvable' do
before { variable.resolve_type!([], :log => logger) }
it 'leaves the reference alone' do
variable.type.should be reference_type
end
it 'warns' do
logger[:warn].first.should == 'Undefined type ncs:color_cl3 for hair_color.'
end
end
context 'when the reference is of an unknown namespace' do
let(:unknown_kind_of_ref) { VariableType.reference('foo:bar') }
before {
variable.type = unknown_kind_of_ref
variable.resolve_type!(types, :log => logger)
}
it 'leaves it in place' do
variable.type.should be unknown_kind_of_ref
end
it 'warns' do
logger[:warn].first.
should == 'Unknown reference namespace in type "foo:bar" for hair_color'
end
end
context 'when the type is an XML Schema type' do
it 'ignores it' do
variable.type = VariableType.xml_schema_type('decimal')
variable.resolve_type!(types, :log => logger)
logger[:warn].should == []
end
end
context 'when the type is not a reference' do
let(:not_a_ref) { VariableType.new('ncs:color_cl3') }
it 'does nothing' do
variable.type = not_a_ref
variable.resolve_type!(types)
variable.type.should be not_a_ref
end
end
end
describe '#resolve_foreign_key!' do
let(:variable) {
Variable.new('helicopter_id').tap { |v| v.type = variable_type }
}
let(:table) {
TransmissionTable.new('flights').tap do |t|
t.variables = [ variable ]
end
}
let(:table_with_matching_pk) {
TransmissionTable.new('helicopters').tap do |t|
t.variables = [
Variable.new('helicopter_id').tap do |v|
v.type = VariableType.new('primaryKeyType')
end
]
end
}
let(:table_with_nonmatching_pk) {
TransmissionTable.new('frogs').tap do |t|
t.variables = [
Variable.new('frog_id').tap do |v|
v.type = VariableType.new('primaryKeyType')
end
]
end
}
let(:table_with_same_fk) {
TransmissionTable.new('autorotation_events').tap do |t|
t.variables = [
Variable.new('helicopter_id').tap do |v|
v.type = VariableType.new('foreignKeyTypeRequired')
end
]
end
}
let(:all_tables) {
[table, table_with_matching_pk, table_with_nonmatching_pk, table_with_same_fk]
}
# These are overridden in nested contexts as necessary
let(:override) { nil }
let(:tables) { all_tables }
before do
variable.resolve_foreign_key!(tables, override, :log => logger)
end
shared_examples 'a foreign key' do
context 'when there is no table matching' do
context 'and there is no override' do
let(:tables) {
[table, table_with_same_fk, table_with_nonmatching_pk]
}
it 'does not set a table reference' do
variable.table_reference.should be_nil
end
it 'warns' do
logger[:warn].first.
should == 'Foreign key not resolvable: no tables have a primary key named "helicopter_id".'
end
end
include_examples 'for overrides'
end
context 'when there is exactly one matching table' do
let(:tables) { all_tables }
context 'and there is no override' do
it 'sets the table_reference to that matching table' do
variable.table_reference.should be table_with_matching_pk
end
it 'does not warn' do
logger[:warn].should be_empty
end
end
include_examples 'for overrides'
end
context 'when there is more than one matching table' do
let(:other_matches) {
[
TransmissionTable.new('helicopters_2').tap { |t|
t.variables = table_with_matching_pk.variables.dup
},
TransmissionTable.new('choppers').tap { |t|
t.variables = table_with_matching_pk.variables.dup
}
]
}
let(:tables) { all_tables + other_matches }
context 'and there is no override' do
it 'does not set a table reference' do
variable.table_reference.should be_nil
end
it 'warns' do
logger[:warn].first.
should == '3 possible parent tables found for foreign key "helicopter_id": "helicopters", "helicopters_2", "choppers". None used due to ambiguity.'
end
end
include_examples 'for overrides'
end
end
shared_examples 'for overrides' do
context 'and there is an override' do
context 'and the override is to a known table' do
let(:override) { table_with_same_fk.name }
it 'sets the table reference to the override' do
variable.table_reference.should be table_with_same_fk
end
it 'does not warn' do
logger[:warn].should == []
end
end
context 'and the override is to an unknown table' do
let(:override) { 'aircraft' }
it 'does not set a table reference' do
variable.table_reference.should be_nil
end
it 'warns' do
logger[:warn].first.
should == 'Foreign key "helicopter_id" explicitly mapped to unknown table "aircraft".'
end
end
context 'and the override is false' do
let(:override) { false }
it 'does not set a table reference' do
variable.table_reference.should be_nil
end
it 'does not warn' do
logger[:warn].should == []
end
end
end
end
context 'when a nullable FK' do
let(:variable_type) { VariableType.new('foreignKeyTypeNullable') }
it_behaves_like 'a foreign key'
end
context 'when a required FK' do
let(:variable_type) { VariableType.new('foreignKeyTypeRequired') }
it_behaves_like 'a foreign key'
end
context 'when not an FK' do
let(:variable_type) { VariableType.new('confirm_cl7') }
it 'does nothing' do
variable.table_reference.should be_nil
end
it 'does not warn' do
logger[:warn].should be_empty
end
include_examples 'for overrides'
end
end
end
end