require 'coverage_helper' require 'rasti-db' require 'minitest/autorun' require 'minitest/colorin' require 'minitest/line/describe_track' require 'pry-nav' require 'logger' require 'sequel/extensions/pg_hstore' require 'sequel/extensions/pg_array' require 'sequel/extensions/pg_json' Rasti::DB.configure do |config| config.type_converters = [Rasti::DB::TypeConverters::TimeInZone] end User = Rasti::DB::Model[:id, :name, :posts, :comments, :person, :comments_count] Post = Rasti::DB::Model[:id, :title, :body, :user_id, :user, :comments, :categories, :language_id, :language, :notice, :author] Comment = Rasti::DB::Model[:id, :text, :user_id, :user, :post_id, :post] Category = Rasti::DB::Model[:id, :name, :posts] Person = Rasti::DB::Model[:document_number, :first_name, :last_name, :birth_date, :user_id, :user, :languages, :full_name] Language = Rasti::DB::Model[:id, :name, :people] class Users < Rasti::DB::Collection one_to_many :posts one_to_many :comments one_to_one :person computed_attribute :comments_count do Rasti::DB::ComputedAttribute.new(Sequel[:comments_count][:value]) do |dataset| subquery = dataset.db.from(:comments) .select(Sequel[:user_id], Sequel.function('count', :id).as(:value)) .group(:user_id) .as(:comments_count) dataset.join_table(:inner, subquery, :user_id => :id) end end end class Posts < Rasti::DB::Collection many_to_one :user many_to_one :language many_to_many :categories one_to_many :comments query :created_by, ->(user_id) { where user_id: user_id } query :entitled do |title| where title: title end query :commented_by do |user_id| chainable do dataset.join(qualify(:comments), post_id: :id) .where(Sequel[:comments][:user_id] => user_id) .select_all(:posts) .distinct end end computed_attribute :notice do Rasti::DB::ComputedAttribute.new Sequel.join([:title, ': ', :body]) end computed_attribute :author do Rasti::DB::ComputedAttribute.new Sequel[:user] end end class Comments < Rasti::DB::Collection many_to_one :user many_to_one :post def posts_commented_by(user_id) dataset.where(Sequel[:comments][:user_id] => user_id) .join(qualify(:posts, data_source_name: :default), id: :post_id) .select_all(:posts) .map { |row| Post.new row } end end class Categories < Rasti::DB::Collection many_to_many :posts end class People < Rasti::DB::Collection set_collection_name :people set_primary_key :document_number set_foreign_key :document_number set_model Person many_to_one :user many_to_many :languages computed_attribute :full_name do |db| Rasti::DB::ComputedAttribute.new Sequel.join([:first_name, ' ', :last_name]) end end class Languages < Rasti::DB::Collection set_data_source_name :custom many_to_many :people, collection: People, relation_data_source_name: :default one_to_many :posts end class Minitest::Spec let(:users) { Users.new environment } let(:posts) { Posts.new environment } let(:comments) { Comments.new environment } let(:categories) { Categories.new environment } let(:people) { People.new environment } let(:languages) { Languages.new environment } let(:driver) { (RUBY_ENGINE == 'jruby') ? 'jdbc:sqlite::memory:' : {adapter: :sqlite} } let :environment do Rasti::DB::Environment.new default: Rasti::DB::DataSource.new(db), custom: Rasti::DB::DataSource.new(custom_db) end let :db do Sequel.connect(driver).tap do |db| db.create_table :users do primary_key :id String :name, null: false, unique: true end db.create_table :posts do primary_key :id String :title, null: false, unique: true String :body, null: false Integer :language_id, null: false, index: true foreign_key :user_id, :users, null: false, index: true end db.create_table :comments do primary_key :id String :text, null: false foreign_key :user_id, :users, null: false, index: true foreign_key :post_id, :posts, null: false, index: true end db.create_table :categories do primary_key :id String :name, null: false, unique: true end db.create_table :categories_posts do foreign_key :category_id, :categories, null: false, index: true foreign_key :post_id, :posts, null: false, index: true primary_key [:category_id, :post_id] end db.create_table :people do String :document_number, null: false, primary_key: true String :first_name, null: false String :last_name, null: false Date :birth_date, null: false foreign_key :user_id, :users, null: false, unique: true end db.create_table :languages_people do Integer :language_id, null: false, index: true foreign_key :document_number, :people, type: String, null: false, index: true primary_key [:language_id, :document_number] end end end let :custom_db do Sequel.connect(driver).tap do |db| db.create_table :languages do primary_key :id String :name, null: false, unique: true end end end end