Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/jasl/activeentity
Active Record without Database
https://github.com/jasl/activeentity
activemodel activerecord rails ruby
Last synced: 3 months ago
JSON representation
Active Record without Database
- Host: GitHub
- URL: https://github.com/jasl/activeentity
- Owner: jasl
- License: other
- Created: 2019-01-01T19:10:58.000Z (about 6 years ago)
- Default Branch: master
- Last Pushed: 2023-03-09T01:38:14.000Z (almost 2 years ago)
- Last Synced: 2024-10-12T03:53:08.368Z (3 months ago)
- Topics: activemodel, activerecord, rails, ruby
- Language: Ruby
- Homepage:
- Size: 305 KB
- Stars: 52
- Watchers: 4
- Forks: 8
- Open Issues: 7
-
Metadata Files:
- Readme: README.md
- License: MIT-LICENSE
Awesome Lists containing this project
README
Active Entity
====Active Entity is a Rails virtual model solution based on ActiveModel and it's design for Rails 6+.
Active Entity is forked from Active Record by removing all database relates codes, so it nearly no need to learn how to use.
## About Virtual Model
Virtual Model is the model not backed by a database table, usually used as "form model" or "presenter", because it's implement interfaces of Active Model, so you can use it like a normal Active Record model in your Rails app.
## Features
### Attribute declaration
```ruby
class Book < ActiveEntity::Base
attribute :title, :string
attribute :tags, :string, array: true, default: []
end
```Same usage with Active Record, [Learn more](https://api.rubyonrails.org/classes/ActiveRecord/Attributes/ClassMethods.html#method-i-attribute).
One enhancement is `array: true` that transform the attribute to an array that can accept multiple values.
### Nested attributes
Active Entity supports its own variant of nested attributes via the `embeds_one` / `embeds_many` macros. The intention is to be mostly compatible with ActiveRecord's `accepts_nested_attributes_for` functionality.
```ruby
class Holiday < ActiveEntity::Base
attribute :date, :date
validates :date, presence: true
endclass HolidaysForm < ActiveEntity::Base
embeds_many :holidays
accepts_nested_attributes_for :holidays, reject_if: :all_blank
end
```### Validations
```ruby
class Book < ActiveEntity::Base
attribute :title, :string
validates :title, presence: true
end
```Supported Active Record validations:
- [acceptance](https://guides.rubyonrails.org/active_record_validations.html#acceptance)
- [confirmation](https://guides.rubyonrails.org/active_record_validations.html#confirmation)
- [exclusion](https://guides.rubyonrails.org/active_record_validations.html#exclusion)
- [format](https://guides.rubyonrails.org/active_record_validations.html#format)
- [inclusion](https://guides.rubyonrails.org/active_record_validations.html#inclusion)
- [length](https://guides.rubyonrails.org/active_record_validations.html#length)
- [numericality](https://guides.rubyonrails.org/active_record_validations.html#numericality)
- [presence](https://guides.rubyonrails.org/active_record_validations.html#presence)
- [absence](https://guides.rubyonrails.org/active_record_validations.html#absence)[Common validation options](https://guides.rubyonrails.org/active_record_validations.html#common-validation-options) supported too.
#### `subset` validation
Because Active Entity supports array attribute, for some reason, you may want to test values of an array attribute are all included in a given set.
Active Entity provides `subset` validation to achieve that, it usage similar to `inclusion` or `exclusion`
```ruby
class Steak < ActiveEntity::Base
attribute :side_dishes, :string, array: true, default: []
validates :side_dishes, subset: { in: %w(chips mashed_potato salad) }
end
```#### `uniqueness_in_embeds` validation
Active Entity provides `uniqueness_in_embeds` validation to test duplicate nesting virtual record.
Argument `key` is attribute name of nested model, it also supports multiple attributes by given an array.
```ruby
class Category < ActiveEntity::Base
attribute :name, :string
endclass Reviewer < ActiveEntity::Base
attribute :first_name, :string
attribute :last_name, :string
endclass Book < ActiveEntity::Base
embeds_many :categories, index_errors: true
validates :categories, uniqueness_in_embeds: {key: :name}embeds_many :reviewers
validates :reviewers, uniqueness_in_embeds: {key: [:first_name, :last_name]}
end
```#### `uniqueness_in_active_record` validation
Active Entity provides `uniqueness_in_active_record` validation to test given `scope` doesn't present in ActiveRecord model.
The usage same as [uniqueness](https://guides.rubyonrails.org/active_record_validations.html#uniqueness) in addition you must give a AR model `class_name`
```ruby
class Candidate < ActiveEntity::Base
attribute :name, :stringvalidates :name,
uniqueness_on_active_record: {
class_name: "Staff"
}
end
```### Others
These Active Record feature also available in Active Entity
- [`composed_of`](https://api.rubyonrails.org/classes/ActiveRecord/Aggregations/ClassMethods.html)
- [`serializable_hash`](https://api.rubyonrails.org/classes/ActiveModel/Serialization.html#method-i-serializable_hash)
- [`serialize`](https://api.rubyonrails.org/classes/ActiveRecord/AttributeMethods/Serialization/ClassMethods.html#method-i-serialize)
- [`store`](https://api.rubyonrails.org/classes/ActiveRecord/Store.html)#### I18n
Same to [Active Record I18n](https://guides.rubyonrails.org/i18n.html#translations-for-active-record-models), the only different is the root of locale YAML is `active_entity` instead of `activerecord`
#### Enum
You can use the `enum` class method to define a set of possible values for an attribute. It is similar to the `enum` functionality in Active Model, but has significant enough quirks that you should think of them as distinct.
```rb
class Example < ActiveEntity::Base
attribute :steve, :integer
enum steve: [:martin, :carell, :buscemi]
endexample = Example.new
example.attributes # => {"steve"=>nil}
example.steve = :carell
example.carell? # => true
example.attributes # => {"steve"=>"carell"}
example.steve = 2
example.attributes # => {"steve"=>"buscemi"}# IMPORTANT: the next line will only work if you implement an update! method
example.martin! # => {"steve"=>"martin"}example.steve = :bannon # ArgumentError ('bannon' is not a valid steve)
```The first thing you'll notice about the `:steve` attribute is that it is an "Integer", even though it might seem logical to define it as a String... TL;DR: don't do this. Internally enum tracks the possible values based on their index position in the array.
It's also possible to provide a Hash of possible values:
```rb
class Example < ActiveEntity::Base
attribute :steve, :integer, default: 9
enum steve: {martin: 5, carell: 12, buscemi: 9}
endexample = Example.new
example.attributes # => {"steve"=>"buscemi"}
```The other quirk of this implementation is that you must create your attribute before you call enum.
enum does not create the search scopes that might be familar to Active Model users, since there is no search or where concept in Active Entity. You can, however, access the mapping directly to obtain the index number for a given value:```rb
Example.steves[:buscemi] # => 9
```You can define prefixes and suffixes for your `enum` attributes. Note the underscores:
```rb
class Conversation < ActiveEntity::Base
attribute :status, :integer
attribute :comments_status, :integer
enum status: [ :active, :archived ], _suffix: true
enum comments_status: [ :active, :inactive ], _prefix: :comments
endconversation = Conversation.new
conversation.active_status! # only if you have an update! method
conversation.archived_status? # => falseconversation.comments_inactive! # only if you have an update! method
conversation.comments_active? # => false
```#### Read-only attributes
You can use `attr_readonly :title, :author` to prevent assign value to attribute after initialized.
You can use `enable_readonly!` and `disable_readonly!` to control the behavior.
**Important: It's no effect with embeds or array attributes !!!**
## Extending
Most of Active Model plugins are compatible with Active Entity.
You need to include them manually.
Tested extensions:
- [adzap/validates_timeliness](https://github.com/adzap/validates_timeliness)
## Installation
Add this line to your application's Gemfile:
```ruby
gem 'activeentity', require: "active_entity/railtie"
```And then execute:
```bash
$ bundle
```Or install it yourself as:
```bash
$ gem install activeentity
```## Other awesome gems
- [makandra/active_type](https://github.com/makandra/active_type)
## Contributing
- Fork the project.
- Make your feature addition or bug fix.
- Add tests for it. This is important so I don't break it in a future version unintentionally.
- Commit, do not mess with Rakefile or version (if you want to have your own version, that is fine but bump version in a commit by itself I can ignore when I pull)
- Send me a pull request. Bonus points for topic branches.## License
The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).