module SerializeableSpec
class TestModel
attr_accessor :name, :age
def initialize(name: nil, age: nil)
@name = name
@age = age
end
end
class TestModelMapper < Lutaml::Model::Serializable
model TestModel
attribute :name, Lutaml::Model::Type::String
attribute :age, Lutaml::Model::Type::String
end
class TestMapper < Lutaml::Model::Serializable
attribute :name, Lutaml::Model::Type::String
attribute :age, Lutaml::Model::Type::String
yaml do
map :na, to: :name
map :ag, to: :age
end
end
class KeyValueMapper < Lutaml::Model::Serializable
attribute :first_name, :string
attribute :last_name, :string
attribute :age, :integer
key_value do
map :first_name, to: :first_name
map :last_name, to: :last_name
map :age, to: :age
end
end
### XML root mapping
class RecordDate < Lutaml::Model::Serializable
attribute :content, :string
xml do
root "recordDate"
map_content to: :content
end
end
class OriginInfo < Lutaml::Model::Serializable
attribute :date_issued, RecordDate, collection: true
xml do
root "originInfo"
map_element "dateIssued", to: :date_issued
end
end
### Enumeration
class Ceramic < Lutaml::Model::Serializable
attribute :type, :string
attribute :firing_temperature, :integer
end
class CeramicCollection < Lutaml::Model::Serializable
attribute :featured_piece,
Ceramic,
values: [
Ceramic.new(type: "Porcelain", firing_temperature: 1300),
Ceramic.new(type: "Stoneware", firing_temperature: 1200),
Ceramic.new(type: "Earthenware", firing_temperature: 1000),
]
end
class GlazeTechnique < Lutaml::Model::Serializable
attribute :name, :string, values: ["Celadon", "Raku", "Majolica"]
end
end
RSpec.describe Lutaml::Model::Serializable do
describe ".model" do
it "sets the model for the class" do
expect do
described_class.model(SerializeableSpec::TestModel)
end.to change(
described_class, :model
)
.from(described_class)
.to(SerializeableSpec::TestModel)
end
end
describe ".attribute" do
subject(:mapper) { described_class.new }
it "adds the attribute and getter setter for that attribute" do
expect { described_class.attribute("foo", Lutaml::Model::Type::String) }
.to change { described_class.attributes.keys }.from([]).to(["foo"])
.and change { mapper.respond_to?(:foo) }.from(false).to(true)
.and change { mapper.respond_to?(:foo=) }.from(false).to(true)
end
end
describe ".hash_representation" do
context "when model is separate" do
let(:instance) do
SerializeableSpec::TestModel.new(name: "John", age: 18)
end
let(:expected_hash) do
{
"name" => "John",
"age" => "18",
}
end
it "return hash representation" do
generate_hash = SerializeableSpec::TestModelMapper.hash_representation(
instance, :yaml
)
expect(generate_hash).to eq(expected_hash)
end
end
context "when model is self" do
let(:instance) do
SerializeableSpec::TestMapper.new(name: "John", age: 18)
end
let(:expected_hash) do
{
"na" => "John",
"ag" => "18",
}
end
it "return hash representation" do
generate_hash = SerializeableSpec::TestMapper.hash_representation(
instance, :yaml
)
expect(generate_hash).to eq(expected_hash)
end
end
end
describe ".mappings_for" do
context "when mapping is defined" do
it "returns the defined mapping" do
actual_mappings = SerializeableSpec::TestMapper.mappings_for(:yaml).mappings
expect(actual_mappings[0].name).to eq(:na)
expect(actual_mappings[0].to).to eq(:name)
expect(actual_mappings[1].name).to eq(:ag)
expect(actual_mappings[1].to).to eq(:age)
end
end
context "when mapping is not defined" do
it "maps attributes to mappings" do
allow(SerializeableSpec::TestMapper.mappings).to receive(:[]).with(:yaml).and_return(nil)
actual_mappings = SerializeableSpec::TestMapper.mappings_for(:yaml).mappings
expect(actual_mappings[0].name).to eq("name")
expect(actual_mappings[0].to).to eq(:name)
expect(actual_mappings[1].name).to eq("age")
expect(actual_mappings[1].to).to eq(:age)
end
end
end
describe ".apply_child_mappings" do
let(:child_mappings) do
{
id: :key,
path: %i[path link],
name: %i[path name],
}
end
let(:hash) do
{
"foo" => {
"path" => {
"link" => "link one",
"name" => "one",
},
},
"abc" => {
"path" => {
"link" => "link two",
"name" => "two",
},
},
"hello" => {
"path" => {
"link" => "link three",
"name" => "three",
},
},
}
end
let(:expected_value) do
[
{ id: "foo", path: "link one", name: "one" },
{ id: "abc", path: "link two", name: "two" },
{ id: "hello", path: "link three", name: "three" },
]
end
it "generates hash based on child_mappings" do
expect(described_class.apply_child_mappings(hash,
child_mappings)).to eq(expected_value)
end
end
describe "#key_value" do
let(:model) { SerializeableSpec::KeyValueMapper }
Lutaml::Model::Config::KEY_VALUE_FORMATS.each do |format|
it "defines 3 mappings for #{format}" do
expect(model.mappings_for(format).mappings.count).to eq(3)
end
it "defines mappings correctly for #{format}" do
defined_mappings = model.mappings_for(format).mappings.map(&:name)
expect(defined_mappings).to eq(%i[first_name last_name age])
end
end
end
describe "XML root name override" do
it "uses root name defined at the component class" do
record_date = SerializeableSpec::RecordDate.new(content: "2021-01-01")
expected_xml = "2021-01-01"
expect(record_date.to_xml).to eq(expected_xml)
end
it "uses mapped element name at the aggregating class, overriding root name" do
origin_info = SerializeableSpec::OriginInfo.new(date_issued: [SerializeableSpec::RecordDate.new(content: "2021-01-01")])
expected_xml = <<~XML
2021-01-01
XML
expect(origin_info.to_xml).to be_equivalent_to(expected_xml)
end
end
describe "String enumeration" do
context "when assigning an invalid value" do
it "raises an error after creation after validate" do
glaze = SerializeableSpec::GlazeTechnique.new(name: "Celadon")
glaze.name = "Tenmoku"
expect do
glaze.validate!
end.to raise_error(Lutaml::Model::ValidationError) do |error|
expect(error).to include(Lutaml::Model::InvalidValueError)
expect(error.error_messages).to include("name is `Tenmoku`, must be one of the following [Celadon, Raku, Majolica]")
end
end
end
context "when assigning a valid value" do
it "changes the value after creation" do
glaze = SerializeableSpec::GlazeTechnique.new(name: "Celadon")
glaze.name = "Raku"
expect(glaze.name).to eq("Raku")
end
it "assigns the value during creation" do
glaze = SerializeableSpec::GlazeTechnique.new(name: "Majolica")
expect(glaze.name).to eq("Majolica")
end
end
end
describe "Serializable object enumeration" do
context "when assigning an invalid value" do
it "raises ValidationError containing InvalidValueError after creation" do
glaze = SerializeableSpec::GlazeTechnique.new(name: "Celadon")
glaze.name = "Tenmoku"
expect do
glaze.validate!
end.to raise_error(Lutaml::Model::ValidationError) do |error|
expect(error).to include(Lutaml::Model::InvalidValueError)
expect(error.error_messages).to include(a_string_matching(/name is `Tenmoku`, must be one of the following/))
end
end
it "raises ValidationError containing InvalidValueError during creation" do
expect do
SerializeableSpec::GlazeTechnique.new(name: "Crystalline").validate!
end.to raise_error(Lutaml::Model::ValidationError) do |error|
expect(error).to include(Lutaml::Model::InvalidValueError)
expect(error.error_messages).to include(a_string_matching(/name is `Crystalline`, must be one of the following/))
end
end
end
context "when assigning a valid value" do
it "changes the value after creation" do
collection = SerializeableSpec::CeramicCollection.new(
featured_piece: SerializeableSpec::Ceramic.new(type: "Porcelain",
firing_temperature: 1300),
)
collection.featured_piece = SerializeableSpec::Ceramic.new(type: "Stoneware",
firing_temperature: 1200)
expect(collection.featured_piece.type).to eq("Stoneware")
end
it "assigns the value during creation" do
collection = SerializeableSpec::CeramicCollection.new(
featured_piece: SerializeableSpec::Ceramic.new(type: "Earthenware",
firing_temperature: 1000),
)
expect(collection.featured_piece.type).to eq("Earthenware")
end
end
end
end