UniverseCompiler ================ - [Overview](#overview) - [Installation](#installation) - [Core Concepts](#core-concepts) - [Entities](#entities) - [Overview](#overview-1) - [Special directives](#special-directives) - [Constraints and relationships directives](#constraints-and-relationships-directives) - [Compilation](#compilation) - [Inheritance](#inheritance) - [Overrides](#overrides) - [Generate a graph of entities using `Graphviz`](#generate-a-graph-of-entities-using-graphviz) - [Development](#development) - [Contributing](#contributing) - [License](#license) - [Code of Conduct](#code-of-conduct) # Overview The goal of this gem is to provide a simple way to manage a consistent highly complex configuration. The configuration can be split into lot of objects (or `entities`) and complex relations and constraints can be defined between them using a-la-ActiveRecord relationships like `has_many` or `is_array` (see complete list in `lib/universe_compiler/entity/field_constraint_management.rb`). These entities are added to a so-called `universe`. See a universe as a kind of sandbox where entities exist. A `universe` could be _persisted_ to any kind of backend by writing an persistence engine, yet only a yaml persistence engine is available by default in the gem. A `universe` can be _compiled_ in order to produce a new `universe` where all constraints and relations defined by these entities have been resolved. # Installation Add this line to your application's Gemfile: ```ruby gem 'universe_compiler' ``` And then execute: $ bundle Or install it yourself as: $ gem install universe_compiler # Core Concepts ## Entities ### Overview Any `entity` you will create will basically inherit from `UniverseCompiler::Entity::Base`. This will allow the following kind of code: ```ruby class EntityA < UniverseCompiler::Entity::Base entity_type :some_entity field :some_data, :is_hash field :bar, :not_null, should_match: /^Y/ field :stupid has_one EntityA, name: :master end class EntityB < UniverseCompiler::Entity::Base has_many EntityA, name: :some_entities not_empty :some_entities end ``` The core concept is that every entity has a hash property named `fields`. And any field declared using the declaration mechanism as above just adds some content validations mechanisms and direct accessors to the internal `fields` hash. For example: ```ruby a = EntityA.new fields: { stupid: :foo} a.stupid # => :foo a.stupid == a[:stupid] # => true a.stupid == a.fields[:stupid] # => true a.stupid == a['stupid'] # => false, a String is not a Symbol ``` Therefore the _pseudo schema_ you defined when declaring the class is not limiting... you can always do ```ruby a[:non_explicitely_declared_property] = :bar a[:non_explicitely_declared_property] # => :bar # but a.non_explicitely_declared_property # => NoMethodError: undefined method `non_existing_property' for... ``` Every entity has a `valid?` method which performs various checks. For example here we have said that an instance of `EntityB` has many entities of type `EntityA`. You can really see a `has_many` relationship an the definition of an Array which content is validated: ```ruby b = EntityB.new b.valid? # => false b.some_entities << :foo b.valid? # => false, :foo is not of the expected type b.clear b.valid? # => false b.some_entities << b b.valid? # => false, b is not of the expected type b.clear b.some_entities << a b.valid? # => true, a is ok ``` In the same vein: ```ruby a.valid? # => false a.some_entity = a a.valid? # => false, still false as requiring a non null :bar property a.bar = 'hey man' a.valid? # => false, not compliant with regexp specified a.bar = 'Yo man' a.valid? # => true ``` #### Special directives By default every entity has a `type`. It is available using the `#type` instance method or the `::entity_type` class method. The default value for the entity type is coming from the class name but it can be overridden using the `entity_type` directive: ```ruby EntityA.entity_type # => :some_entity EntityB.entity_type # => "entity_b" a = EntityA.new # => # a.type # => :some_entity b = EntityB.new # => # b.type # => "entity_b" ``` The `name` of an entity can be automatically generated using the `auto_named_entity_type` directive optionally providing a seed: ```ruby class EntityC < UniverseCompiler::Entity::Base auto_named_entity_type end class EntityD < UniverseCompiler::Entity::Base auto_named_entity_type :my_seed end EntityC.new # => # EntityD.new # => # EntityD.new # => # ``` #### Constraints and relationships directives The generic form to declare a field is the `field` statement. Any constraint can be declared using the `field` method. Here is the signature: ```ruby def field(field_name, *options) ``` Then other _constraint_ methods that can be used when describing an entity can be grouped into two. The switches: * not_null * not_empty * is_array * is_hash Then some methods taking parameter: * should_match * class_name You have as well relationship methods: * has_one * has_many So for each of these methods can be used either as "real" methods or as `field` parameter. For example: ```ruby class MyEntity < UniverseCompiler::Entity::Base field :my_field, :not_null, class_name: AClass end ``` Is strictly equivalent to: ```ruby class MyEntity < UniverseCompiler::Entity::Base not_null :my_field class_name :my_field, AClass end ``` Notice the fact that in the latter form `my_field` is "declared" more than once. ### Compilation The compilation mechanism is related to universes. When compiling a universe it actually: * Creates a __new universe__ containing __deep copies__ of its original entities. * Applies entities inheritance defined by the special field `extends`. * Applies overrides defined by the `:entity_overide` special entity type. Here is an example ```ruby u = UniverseCompiler.new_universe # Adding entities to universe requires they have a name a = EntityA.new fields: { name: :a, bar: 'Yo man', stupid: :yeah } # a is valid b = EntityA.new fields: { name: :b, extends: a } # Notice b is not valid but extends a u << a << b v = u.compile # v is a new universe result of the "compilation" of u u.name # => "Unnamed Universe" v.name # => "Unnamed Universe - COMPILED #47332840258780" compiled_b = v.get_entity :some_entity, :b compiled_b == b # => true, b and compiled_b although different represent the same entity compiled_b.eql? b # => false, b and compiled_b are in different universe compiled_b.equal? b # => false, b and compiled_b have different object_id b.valid? # => false, in the universe u, b is still not valid compiled_b.valid? # => true, thanks to the fact b extends a a.fields # => {:name=>:a, :bar=>"Yo man", :stupid=>:yeah, :some_data=>{}} b.fields # => {:name=>:b, :extends=>#, :some_data=>{}} compiled_b.fields # => {:name=>:b, :bar=>"Yo man", :stupid=>:yeah, :some_data=>{}, :extends=>#} ``` And each entity in the new universe will have the flag `compiled` set to `true`. ```ruby u.get_entities.map do |entity| {name: entity.name, compiled: entity.compiled} end # => [{name: :a, compiled: false},{name: :b, compiled: false},{name: :c, compiled: false}] v.get_entities.map do |entity| {name: entity.name, compiled: entity.compiled} end # => [{name: :a, compiled: true},{name: :b, compiled: true},{name: :c, compiled: true}] ``` ## Inheritance To be clear, here we talk about __entities (instances) inheritance, NOT classes !__ Each entity can potentially extend (using the `extends` field) one entity... which itself could extend as well another entity. __Circular references are detected and compilation may fail.__ When you `extends` another entity, it means that when the universe "compiles", it will perform some merge operations. e.g. for the the following inheritance definition: ``` u1.a --extends--> u1.b --extends--> u1.c ``` It means that if you have a universe u1 containing these entities a, b, c and you compile it, the resulting universe, let's call it u2, will contain 3 new entities a, b and c which content will be (all content is duplicated): * u2.c content is the __same as u1.c__. * u2.b content will be __the merge of u1.b into u2.c__. * u2.a content will be __the merge of u1.a into u2.b__. Of course the compilation process keeps the initial relationships. ``` u2.a --extends--> u2.b --extends--> u2.c ``` You can see an example of inheritance in previous paragraph. ## Overrides Overrides are actually a special type of entities. They have a special array called `overrides` which contains a list of entities you want to inject content into. When you override entity `a` with override `o`, it means that the content (fields) of `o` will be _injected_ into `a` (fields). This is why an override can override multiple objects of multiple types, because this is just about content injection. Of course as already said, it occurs during the compilation process and only in the "compiled" universe. The original universe is meant to remain unmodified. Overrides are only applied in the context of a `scenario` ```ruby u = UniverseCompiler.new_universe a = EntityA.new fields: { name: :a, bar: 'Yo man', stupid: :yeah } b = EntityA.new fields: { name: :b, extends: a } o = UniverseCompiler.new_override fields: { name: :my_override, scenario: :test_overrides, a_new_stuff: :hey, overrides: [a, b] } u << a << b << o o.type # => :entity_override v = u.compile scenario: :test_overrides v.get_entities.map &:fields # => [{:name=>:a, # :bar=>"Yo man", # :stupid=>:yeah, # :some_data=>{}, # :a_new_stuff=>:hey}, # {:name=>:b, # :bar=>"Yo man", # :stupid=>:yeah, # :some_data=>{}, # :extends=> # #, # :a_new_stuff=>:hey}, # {:name=>:my_override, # :scenario=>:test_overrides, # :a_new_stuff=>:hey, # :overrides=> # [#, # #]}] # ``` You can see there that the compiled version of `b` contains both data coming from the inheritance mechanism as well as those coming from the override... # Generate a graph of entities using `Graphviz` Provides a [Graphviz] graph of relations using the [ruby-graphviz] gem between a set of entities and allows callback mechanisms for the graph customization. All you need is to: ```ruby include UniverseCompiler::Utils::Graphviz ``` In any class, and then simply call: ```ruby graph_entities_to_file(an_array_of_entities, set_of_options, &customization_block) ``` Calls to the `graph_entities_to_file` method returns either the filename of the created file or the graphiz graph object depending on method parameters. A block can be passed to the method, it will yield the graph object, and optionnally a bi-directional structure of entities and graphwiz nodes, allowing for complete graph display customization (you may actually completely change the graph, but maybe in this case you may want to actually completely build the graph. It is normally intended to change/customize display attributes there...). # Development After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment. To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org). # Contributing Bug reports and pull requests are welcome on Gitlab at https://gitlab.com/tools4devops/power_stencil. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct. ## License The gem is available as open source under the terms of the [MIT License]. ## Code of Conduct Everyone interacting in the PowerStencil project’s codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct]. [code of conduct]: CODE_OF_CONDUCT.md [MIT License]: http://opensource.org/licenses/MIT "The MIT license" [Ruby]: https://www.ruby-lang.org "The powerful Ruby language" [Graphviz]: (https://www.graphviz.org/) "Graph Visualization Software" [ruby-graphviz]: (https://rubygems.org/gems/ruby-graphviz) "Ruby interface to graphviz"