Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/mongoid/mongoid-rspec
RSpec matchers and macros for Mongoid.
https://github.com/mongoid/mongoid-rspec
mongoid rspec-matchers ruby
Last synced: 6 days ago
JSON representation
RSpec matchers and macros for Mongoid.
- Host: GitHub
- URL: https://github.com/mongoid/mongoid-rspec
- Owner: mongoid
- License: mit
- Created: 2010-04-14T08:06:50.000Z (over 14 years ago)
- Default Branch: master
- Last Pushed: 2024-05-12T16:31:22.000Z (8 months ago)
- Last Synced: 2024-05-23T05:43:50.996Z (8 months ago)
- Topics: mongoid, rspec-matchers, ruby
- Language: Ruby
- Homepage:
- Size: 354 KB
- Stars: 500
- Watchers: 20
- Forks: 179
- Open Issues: 21
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
- License: LICENSE
Awesome Lists containing this project
- awesome-rspec - mongoid-rspec - RSpec matchers for Mongoid. (Matchers)
README
# [mongoid-rspec](https://github.com/mongoid/mongoid-rspec "A collection of RSpec-compatible matchers that help to test Mongoid documents.")
[![Gem Version](https://badge.fury.io/rb/mongoid-rspec.svg)](https://badge.fury.io/rb/mongoid-rspec)
[![Test Status](https://github.com/mongoid/mongoid-rspec/workflows/Test/badge.svg)](https://github.com/mongoid/mongoid-rspec/actions)
[![Rubocop Status](https://github.com/mongoid/mongoid-rspec/workflows/Rubocop/badge.svg)](https://github.com/mongoid/mongoid-rspec/actions)The mongoid-rspec library provides a collection of RSpec-compatible matchers that help to test Mongoid documents.
[Tested](https://github.com/mongoid/mongoid-locker/actions) against:
- MRI: `2.6.x`, `2.7.x`, `3.0.x`, `3.1.x`, `3.2.x`
- Mongoid: `4`, `5`, `6`, `7`, `8`, `9`See [.github/workflows/rspec.yml](.github/workflows/rspec.yml) for the latest test matrix.
## Installation
Drop this line into your Gemfile:
```ruby
group :test do
gem 'mongoid-rspec'
end```
## Compatibility
This gem is compatible with Mongoid 3, 4, 5, 6, 7, 8, 9.
## Configuration
### Rails
Add to your `rails_helper.rb` file
```ruby
require 'mongoid-rspec'RSpec.configure do |config|
config.include Mongoid::Matchers, type: :model
end
```### Other
Add to your `spec_helper.rb` file.
```ruby
require 'mongoid-rspec'RSpec.configure do |config|
config.include Mongoid::Matchers
end
```## Matchers
### be_mongoid_document
```ruby
class Post
include Mongoid::Document
endRSpec.describe Post, type: :model do
it { is_expected.to be_mongoid_document }
end
```### be_dynamic_document
```ruby
class User
include Mongoid::Document
include Mongoid::Attributes::Dynamic
endRSpec.describe User, type: :model do
it { is_expected.to be_dynamic_document }
end
```### have_timestamps
With full timestamps.
```ruby
class Log
include Mongoid::Document
include Mongoid::Timestamps
endRSpec.describe Log, type: :model do
it { is_expected.to have_timestamps }
end
```With short timestamps.
```ruby
class User
include Mongoid::Document
include Mongoid::Timestamps::Short
endRSpec.describe User, type: :model do
it { is_expected.to have_timestamps.shortened }
end
```With only creating or updating timestamps.
```ruby
class Admin
include Mongoid::Document
include Mongoid::Timestamps::Create
include Mongoid::Timestamps::Update
endRSpec.describe Admin, type: :model do
it { is_expected.to have_timestamps.for(:creating) }
it { is_expected.to have_timestamps.for(:updating) }
end
```With short creating or updating timestamps.
```ruby
class Post
include Mongoid::Document
include Mongoid::Timestamps::Create::Short
endRSpec.describe Short, type: :model do
it { is_expected.to have_timestamps.for(:creating).shortened }
end
```### be_stored_in
```ruby
class Post
include Mongoid::Documentstore_in database: 'db1', collection: 'messages', client: 'secondary'
endRSpec.describe Post, type: :model do
it { is_expected.to be_stored_in(database: 'db1', collection: 'messages', client: 'secondary') }
end
```It checks only those options, that you specify. For instance, test in example below will pass, even though expectation contains only `database` option.
```ruby
class Comment
include Mongoid::Documentstore_in database: 'db2', collection: 'messages'
endRSpec.describe Comment, type: :model do
it { is_expected.to be_stored_in(database: 'db2') }
end
```It works fine with lambdas and procs.
```ruby
class User
include Mongoid::Documentstore_in database: ->{ Thread.current[:database] }
endRSpec.describe Post, type: :model do
it do
Thread.current[:database] = 'db3'
is_expected.to be_stored_in(database: 'db3')Thread.current[:database] = 'db4'
is_expected.to be_stored_in(database: 'db4')
end
end
```### have_index_for
```ruby
class Article
index({ title: 1 }, { unique: true, background: true })
index({ title: 1, created_at: -1 })
index({ category: 1 })
endRSpec.describe Article, type: :model do
it do
is_expected
.to have_index_for(title: 1)
.with_options(unique: true, background: true)
end
it { is_expected.to have_index_for(title: 1, created_at: -1) }
it { is_expected.to have_index_for(category: 1) }
end
```### Field Matchers
```ruby
RSpec.describe Article do
it { is_expected.to have_field(:published).of_type(Mongoid::Boolean).with_default_value_of(false) }
it { is_expected.to have_field(:allow_comments).of_type(Mongoid::Boolean).with_default_value_of(true) }
it { is_expected.not_to have_field(:allow_comments).of_type(Mongoid::Boolean).with_default_value_of(false) }
it { is_expected.not_to have_field(:number_of_comments).of_type(Integer).with_default_value_of(1) }
endRSpec.describe User do
it { is_expected.to have_fields(:email, :login) }
it { is_expected.to have_field(:s).with_alias(:status) }
it { is_expected.to have_fields(:birthdate, :registered_at).of_type(DateTime) }
end
```### Association Matchers
```ruby
RSpec.describe User do
it { is_expected.to have_many(:articles).with_foreign_key(:author_id).ordered_by(:title) }it { is_expected.to have_one(:record) }
# can verify autobuild is set to true
it { is_expected.to have_one(:record).with_autobuild }it { is_expected.to have_many :comments }
# can also specify with_dependent to test if :dependent => :destroy/:destroy_all/:delete is set
it { is_expected.to have_many(:comments).with_dependent(:destroy) }# can verify autosave is set to true
it { is_expected.to have_many(:comments).with_autosave }it { is_expected.to embed_one :profile }
it { is_expected.to have_and_belong_to_many(:children) }
it { is_expected.to have_and_belong_to_many(:children).of_type(User) }
endRSpec.describe Profile do
it { is_expected.to be_embedded_in(:user).as_inverse_of(:profile) }
endRSpec.describe Article do
it { is_expected.to belong_to(:author).of_type(User).as_inverse_of(:articles) }
it { is_expected.to belong_to(:author).of_type(User).as_inverse_of(:articles).with_index }
it { is_expected.to embed_many(:comments) }
end# when the touch option is provided, then we can also verify that it is set
# by default, with_touch matches true when no parameters are provided
describe Article do
it { is_expected.to belong_to(:author).of_type(User).as_inverse_of(:articles).with_index.with_touch }
end# parameters are supported for explicit matching
describe Comment do
it { is_expected.to be_embedded_in(:article).as_inverse_of(:comments).with_polymorphism.with_touch(true) }
enddescribe Permalink do
it { is_expected.to be_embedded_in(:linkable).as_inverse_of(:link).with_touch(false) }
end# touch can also take a symbol representing a field on the parent to touch
describe Record do
it { is_expected.to belong_to(:user).as_inverse_of(:record).with_touch(:record_updated_at) }
endRSpec.describe Comment do
it { is_expected.to be_embedded_in(:article).as_inverse_of(:comments) }
it { is_expected.to belong_to(:user).as_inverse_of(:comments) }
endRSpec.describe Record do
it { is_expected.to belong_to(:user).as_inverse_of(:record) }
endRSpec.describe Site do
it { is_expected.to have_many(:users).as_inverse_of(:site).ordered_by(:email.asc).with_counter_cache }
endRSpec.describe Message do
it { is_expected.to belong_to(:user).with_optional }
end
```### Validation Matchers
```ruby
RSpec.describe Site do
it { is_expected.to validate_presence_of(:name) }
it { is_expected.to validate_uniqueness_of(:name) }
endRSpec.describe User do
it { is_expected.to validate_presence_of(:login) }
it { is_expected.to validate_uniqueness_of(:login).scoped_to(:site) }
it { is_expected.to validate_uniqueness_of(:email).case_insensitive.with_message("is already taken") }
it { is_expected.to validate_format_of(:login).to_allow("valid_login").not_to_allow("invalid login") }
it { is_expected.to validate_associated(:profile) }
it { is_expected.to validate_exclusion_of(:login).to_not_allow("super", "index", "edit") }
it { is_expected.to validate_inclusion_of(:role).to_allow("admin", "member") }
it { is_expected.to validate_confirmation_of(:email) }
it { is_expected.to validate_presence_of(:age).on(:create, :update) }
it { is_expected.to validate_numericality_of(:age).on(:create, :update) }
it { is_expected.to validate_inclusion_of(:age).to_allow(23..42).on([:create, :update]) }
it { is_expected.to validate_presence_of(:password).on(:create) }
it { is_expected.to validate_presence_of(:provider_uid).on(:create) }
it { is_expected.to validate_inclusion_of(:locale).to_allow([:en, :ru]) }
endRSpec.describe Article do
it { is_expected.to validate_length_of(:title).within(8..16) }
it { is_expected.to validate_absence_of(:deletion_date) }
endRSpec.describe Visitor do
it { is_expected.to validate_length_of(:name).with_maximum(160).with_minimum(1) }
endRSpec.describe Profile do
it { is_expected.to validate_numericality_of(:age).greater_than(0) }
endRSpec.describe MovieArticle do
it { is_expected.to validate_numericality_of(:rating).to_allow(:greater_than => 0).less_than_or_equal_to(5) }
it { is_expected.to validate_numericality_of(:classification).to_allow(:even => true, :only_integer => true, :nil => false) }
endRSpec.describe Person do
# in order to be able to use the custom_validate matcher, the custom validator class (in this case SsnValidator)
# should redefine the kind method to return :custom, i.e. "def self.kind() :custom end"
it { is_expected.to custom_validate(:ssn).with_validator(SsnValidator) }
end# If you're using validators with if/unless conditionals, spec subject must be object instance
# This is supported on Mongoid 4 and newer
Rspec.describe User do
context 'when user has `admin` role' do
subject { User.new(role: 'admin') }it { is_expected.to validate_length_of(:password).greater_than(20) }
endcontext 'when user does not have `admin` role' do
subject { User.new(role: 'member') }it { is_expected.not_to validate_length_of(:password) }
end
end
```### Mass Assignment Matcher
```ruby
RSpec.describe User do
it { is_expected.to allow_mass_assignment_of(:login) }
it { is_expected.to allow_mass_assignment_of(:email) }
it { is_expected.to allow_mass_assignment_of(:age) }
it { is_expected.to allow_mass_assignment_of(:password) }
it { is_expected.to allow_mass_assignment_of(:password) }
it { is_expected.to allow_mass_assignment_of(:role).as(:admin) }it { is_expected.not_to allow_mass_assignment_of(:role) }
end
```### Accepts Nested Attributes Matcher
```ruby
RSpec.describe User do
it { is_expected.to accept_nested_attributes_for(:articles) }
it { is_expected.to accept_nested_attributes_for(:comments) }
endRSpec.describe Article do
it { is_expected.to accept_nested_attributes_for(:permalink) }
end
```## Contributing
You're encouraged to contribute to this library. See [CONTRIBUTING](CONTRIBUTING.md) for details.
## Copyright and License
Copyright (c) 2009-2018 Evan Sagge and Contributors.
MIT License. See [LICENSE](LICENSE) for details.