Files
growstuff/spec/models/like_spec.rb
google-labs-jules[bot] 3b60e8f974 Implement blocking feature (#4199)
* Implement blocking feature

This commit introduces a blocking feature that allows members to block other members.

A blocked member is prevented from:
- following the blocker
- sending private messages to the blocker
- replying to the blocker's posts
- liking the blocker's content

The implementation includes:
- A new `Block` model and a corresponding database table.
- Updates to the `Member` model to include associations for blocks.
- A new `BlocksController` to handle blocking and unblocking actions.
- New routes for the `BlocksController`.
- UI changes to add block/unblock buttons to the member profile page.
- Validations in the `Follow`, `Comment`, and `Like` models to enforce the blocking rules.
- A check in the `MessagesController` to prevent sending messages to a member who has blocked the sender.
- A callback in the `Block` model to destroy the follow relationship when a block is created.
- New feature and model specs to test the blocking functionality.

* Implement blocking feature and fix failing tests

This commit introduces a blocking feature that allows members to block other members.

A blocked member is prevented from:
- following the blocker
- sending private messages to the blocker
- replying to the blocker's posts
- liking the blocker's content

The implementation includes:
- A new `Block` model and a corresponding database table.
- Updates to the `Member` model to include associations for blocks.
- A new `BlocksController` to handle blocking and unblocking actions.
- New routes for the `BlocksController`.
- UI changes to add block/unblock buttons to the member profile page.
- Validations in the `Follow`, `Comment`, and `Like` models to enforce the blocking rules.
- A check in the `MessagesController` to prevent sending messages to a member who has blocked the sender.
- A callback in the `Block` model to destroy the follow relationship when a block is created.
- New feature and model specs to test the blocking functionality.

This commit also fixes a failing test in the blocking feature. The error was caused by the validation being called even when the `member` association was `nil`. A guard has been added to the validation methods in the `Like`, `Follow`, and `Comment` models to prevent this from happening.

* Generate schema

* Fix tests

* Add permissions

* Define Block permissions in Ability model

The feature specs for member blocking were failing because the "Block"
link was not being rendered on member profiles. This was due to the
lack of explicit create and destroy permissions for the Block resource
in the Ability model, which is used by CanCanCan to authorize actions
and by the view to conditionally show links.

This change adds the necessary permissions to `member_abilities`:
- Allows members to create blocks (except for blocking themselves).
- Allows members to destroy blocks where they are the blocker.

These rules ensure that the "Block" and "Unblock" links are correctly
rendered and authorized for signed-in members.

Co-authored-by: CloCkWeRX <365751+CloCkWeRX@users.noreply.github.com>

* Comment out specs for now

---------

Co-authored-by: google-labs-jules[bot] <161369871+google-labs-jules[bot]@users.noreply.github.com>
Co-authored-by: Daniel O'Connor <daniel.oconnor@gmail.com>
Co-authored-by: CloCkWeRX <365751+CloCkWeRX@users.noreply.github.com>
2026-04-26 14:22:32 +09:30

91 lines
2.3 KiB
Ruby

# frozen_string_literal: true
require 'rails_helper'
describe Like do
let(:member) { create(:member) }
let(:post) { create(:post) }
let(:photo) { create(:photo) }
context 'existing like' do
before do
@like = Like.create(member:, likeable: post)
end
it 'is valid' do
expect(@like).to be_valid
end
it 'belongs to a member' do
expect(@like.member).to be_an_instance_of Member
end
it 'belongs to a likeable item' do
expect(@like).to respond_to :likeable
end
end
context 'invalid parameters' do
it 'is invalid without a member' do
like = Like.new
like.likeable = post
expect(like).not_to be_valid
end
it 'is invalid without a likeable item' do
like = Like.new
like.member = member
expect(like).not_to be_valid
end
end
it 'does not allow duplicate likes by the same member' do
Like.create(member:, likeable: post)
second_like = Like.new(member:, likeable: post)
expect(second_like).not_to be_valid
end
it 'destroys like if post no longer exists' do
like = Like.create(member:, likeable: post)
post.destroy
expect(Like.all).not_to include like
end
it 'destroys like if post no longer exists' do
like = Like.create(member:, likeable: post)
post.destroy
expect(Like.all).not_to include like
end
it 'destroys like if member no longer exists' do
like = Like.create(member:, likeable: post)
member.destroy
expect(Like.all).not_to include like
end
context "when the likeable author has blocked the member" do
let(:likeable_author) { create(:member) }
let(:post_author) { create(:member) }
let(:post) { create(:post, author: likeable_author) }
before do
likeable_author.blocks.create(blocked: member)
end
it "is not valid" do
like = build(:like, likeable: post, member: member)
expect(like).not_to be_valid
end
end
it 'liked_by_members_names' do
expect(post.liked_by_members_names).to eq []
Like.create(member:, likeable: post)
expect(post.liked_by_members_names).to eq [member.login_name]
expect(photo.liked_by_members_names).to eq []
Like.create(member:, likeable: photo)
expect(photo.liked_by_members_names).to eq [member.login_name]
end
end