require "spec_helper" describe Schematic::Serializers::Xsd do before do class EmptyModel < ActiveRecord::Base def self.columns [] end end end describe ".extend" do context "when the model inherits ActiveRecord::Base" do subject { EmptyModel } it "should allow the model to be extended" do lambda { subject.class_eval do extend Schematic::Serializers::Xsd end }.should_not raise_error end end context "when the model does not inherit ActiveRecord::Base" do subject { Object } it "should raise an exception" do lambda { subject.class_eval do extend Schematic::Serializers::Xsd end }.should raise_error(Schematic::InvalidClass) end end end describe ".to_xsd" do context "XSD validation" do context "when the model is not namespaced" do subject { SomeModel.to_xsd } with_model :some_model do table do |t| t.string "some_string" t.float "some_float" t.datetime "some_datetime" t.date "some_date" t.boolean "some_boolean" end model do validates :some_string, :presence => true validates :some_date, :presence => true, :allow_blank => true validates :some_datetime, :presence => true, :allow_blank => false class << self def xsd_methods {:foo => { :bar => { :baz => nil }, :quz => [:qaz] } } end end end end it "should generate a valid XSD" do validate_xsd(subject) end end context "when the model is namespaced" do before do module Namespace; end end subject { Namespace::SomeModel.to_xsd } with_model :some_model do table do |t| t.string "some_string" end model do validates :some_string, :presence => true end end before do class Namespace::SomeModel < SomeModel end end it "should generate a valid XSD" do validate_xsd(subject) end end context "when the model has a nested attribute on a subclass with a reference to the superclass" do with_model :parent do table {} model do has_many :children accepts_nested_attributes_for :children end end with_model :child do table do |t| t.integer :parent_id end model do belongs_to :parent end end before do module Namespace; end class Namespace::Child < Child accepts_nested_attributes_for :parent end end subject { Namespace::Child.to_xsd } it "should generate a valid XSD" do validate_xsd(subject) end end context "when the model has a circular nested attribute reference" do with_model :blog do table {} model do has_many :posts accepts_nested_attributes_for :posts end end with_model :post do table do |t| t.integer :blog_id end model do belongs_to :blog accepts_nested_attributes_for :blog end end subject { Post.to_xsd } it "should generate a valid XSD" do validate_xsd(subject) end end end context "for an empty model with no attributes or validations" do subject { sanitize_xml(EmptyModel.to_xsd) } it "should return an xsd for an array of the model" do xsd = <<-XML XML subject.should == sanitize_xml(xsd) end end context "for a model with attributes" do subject { sanitize_xml(SomeModel.to_xsd) } context "for a any attribute" do with_model :some_model do table :id => false do |t| t.float 'some_float' end end it "should define the correct xsd element" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end subject.should == xsd end end describe "additional methods" do with_model :some_model do table {} end it "should include the additional method" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end sanitize_xml(SomeModel.to_xsd(:methods => {:foo_bar => nil})).should == xsd end end end context "with a model with validations" do subject { sanitize_xml(SomeModel.to_xsd) } context "presence of validation" do context "when allow blank is true" do with_model :some_model do table :id => false do |t| t.string "title" end model do validate :title, :presence => true, :allow_blank => true end end it "should mark that the field minimum occurrences is 0" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end subject.should == xsd end end context "when allow blank is false" do with_model :some_model do table :id => false do |t| t.string "title" end model do validates :title, :presence => true end end it "should mark that the field minimum occurrences is 1" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end subject.should == xsd end end context "when there is a condition" do with_model :some_model do table :id => false do |t| t.string "title" end model do validates :title, :presence => true, :if => lambda { |model| false } end end it "should mark that the field minimum occurrences is 0" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end subject.should == xsd end end end describe "length validation" do end describe "inclusion validation" do end end end describe ".xsd_methods" do context "given a method" do with_model :some_model do table {} model do def self.xsd_methods {:foo_bar => nil} end end end it "should include the additional method" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end sanitize_xml(SomeModel.to_xsd).should eq(xsd) end end context "given a an array of methods" do with_model :some_model do table {} model do def self.xsd_methods {:foo => [:bar]} end end end it "should include the additional methods" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end sanitize_xml(SomeModel.to_xsd).should eq(xsd) end end context "given nested methods" do with_model :some_model do table {} model do def self.xsd_methods { :foo => { :bar => {:baz => nil } } } end end end it "should nested the additional methods" do xsd = generate_xsd_for_model(SomeModel) do <<-XML XML end sanitize_xml(SomeModel.to_xsd).should eq(xsd) end end end describe ".xsd_ignore_methods" do with_model :some_model do table :id => false do |t| t.string :title end model do def self.xsd_ignore_methods [:title] end end end it "should exclude the methods" do xsd = generate_xsd_for_model(SomeModel) do end sanitize_xml(SomeModel.to_xsd).should eq(xsd) end end describe ".xsd_minimum_occurrences_for" do context "given a column with no validations" do with_model :some_model do table :id => false do |t| t.string "title" end model {} end it "should return 0" do SomeModel.xsd_minimum_occurrences_for_column(SomeModel.columns.first).should == "0" end end context "given a column with presence of but allow blank" do with_model :some_model do table :id => false do |t| t.string "title" end model do validates :title, :presence => true, :allow_blank => true end end it "should return 0" do SomeModel.xsd_minimum_occurrences_for_column(SomeModel.columns.first).should == "0" end end context "given a column with presence of and no allow blank" do with_model :some_model do table :id => false do |t| t.string "title" end model do validates :title, :presence => true end end it "should return 1" do SomeModel.xsd_minimum_occurrences_for_column(SomeModel.columns.first).should == "1" end end end private def validate_xsd(xml) xsd_schema_file = File.join(File.dirname(__FILE__), "xsd", "XMLSchema.xsd") meta_xsd = Nokogiri::XML::Schema(File.open(xsd_schema_file)) doc = Nokogiri::XML.parse(xml) meta_xsd.validate(doc).each do |error| error.message.should be_nil end end def sanitize_xml(xml) xml.split("\n").reject(&:blank?).map(&:strip).join("\n") end def generate_xsd_for_model(model) output = <<-XML #{yield} XML sanitize_xml(output) end end