Compare commits

...

25 Commits

Author SHA1 Message Date
Daniel O'Connor
bd637c3310 Cache what a crop is harvested for (#4582) 2026-04-27 13:32:45 +09:30
Daniel O'Connor
9abb0d02b9 Merge pull request #4581 from Growstuff/add-rack-attack-protection-3014929071908440304
Add Rack::Attack rate limiting and Fail2Ban protection
2026-04-27 13:23:17 +09:30
Daniel O'Connor
2e56f8cb2f Cache what a crop is harvested for 2026-04-27 03:52:11 +00:00
Daniel O'Connor
3127f45d0f Merge pull request #4578 from Growstuff/member-inactive-delete
Delete inactive members with no activity in 3 years
2026-04-27 02:15:17 +09:30
Daniel O'Connor
15571940f5 Add fragment cache for crop partials (#4577) 2026-04-27 01:48:35 +09:30
Daniel O'Connor
8e7dd25e98 Add rake task to cleanup inactive members (#4574)
* Add members:cleanup_inactive rake task

This task identifies and deletes members who have not logged in for over
24 months and have no gardens, plantings, or other activity (posts,
comments, seeds, harvests, etc).

Includes support for DRY_RUN=true to preview deletions.
Added tests in spec/tasks/members_spec.rb.

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

* Refactor activity check to Member#has_activity? and update rake task

- Added `Member#has_activity?` to encapsulate the check for gardens, plantings, and other activity.
- Updated `members:cleanup_inactive` rake task to use `Member#has_activity?`.
- Maintained `DRY_RUN` support and existing tests.

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

* Apply suggestion from @CloCkWeRX

* Apply suggestions from code review

Co-authored-by: Daniel O'Connor <daniel.oconnor@gmail.com>

---------

Co-authored-by: google-labs-jules[bot] <161369871+google-labs-jules[bot]@users.noreply.github.com>
2026-04-27 01:40:54 +09:30
Daniel O'Connor
2723599f27 Add fragment cache for crop partials 2026-04-26 16:07:29 +00:00
Daniel O'Connor
98c8bdc0bb Merge pull request #4564 from Growstuff/memory-optimization-2149092598558110155
Memory usage optimization
2026-04-27 00:37:51 +09:30
Daniel O'Connor
573daa8c8a Swap to modern expect style (#4571) 2026-04-26 22:58:04 +09:30
Daniel O'Connor
5174b1236e Merge pull request #4567 from Growstuff/memory-optimisation-3
Members - Nearest To - Memory improvements
2026-04-26 22:57:01 +09:30
Daniel O'Connor
5a349f8f1b Swap to modern expect style 2026-04-26 13:21:15 +00:00
Daniel O'Connor
0d850804cf Merge pull request #4570 from Growstuff/rubocop-tweaks
Rubocop fixes
2026-04-26 22:47:46 +09:30
Daniel O'Connor
161a934811 Merge pull request #4569 from Growstuff/plant_part_spec
Rubocop: Fix no expectation errors
2026-04-26 22:44:49 +09:30
Daniel O'Connor
8cfef5ce1a Rubocop fixes 2026-04-26 13:09:00 +00:00
Daniel O'Connor
6dacb0af74 Swap to modern expect style 2026-04-26 13:03:46 +00:00
Daniel O'Connor
7e2d36f99a Swap to modern expect style 2026-04-26 12:55:58 +00:00
Daniel O'Connor
3406d9e7bc Merge pull request #4568 from Growstuff/memory-usage-4
Posts - memory usage
2026-04-26 19:05:56 +09:30
Daniel O'Connor
7a91746f73 Update .dockerignore to remove .ruby-version
Remove .ruby-version from .dockerignore
2026-04-26 19:05:41 +09:30
Daniel O'Connor
209973e72b Memory usage 2026-04-26 09:26:52 +00:00
Daniel O'Connor
4848302eab Merge pull request #4565 from Growstuff/memory-usage-1
Admin - Members - optimise memory usage
2026-04-26 18:45:12 +09:30
Daniel O'Connor
920a28a144 Merge pull request #4566 from Growstuff/memory-usage-2
GBIF - optimise memory usage
2026-04-26 18:44:58 +09:30
Daniel O'Connor
fff7a14635 GBIF - optimise memory usage 2026-04-26 09:03:45 +00:00
Daniel O'Connor
0131c9b531 Admin - Members - optimise memory usage 2026-04-26 09:01:18 +00:00
Daniel O'Connor
1b091b2f6f Merge pull request #4453 from Growstuff/add-mark-as-failed-to-crop-view-13853484652230549508
Add "mark as failed" action to crop view
2026-04-26 14:44:09 +09:30
google-labs-jules[bot]
8bafba7f9d Ensure "mark as failed" option is available when viewing a crop
This change adds the "mark as failed" action to the crop view in two places:
1. In the "Crop Actions" button group, a new "Mark as failed" button is added if the current member has active plantings of that crop. Clicking it opens a modal to select which planting failed.
2. In the "See who's planted" list, an "Actions" dropdown is added to any plantings owned by the current member, which includes the "Mark as failed" option.

A new partial `app/views/plantings/_failed_modal.html.haml` was created to handle the planting selection modal.
`app/views/crops/_actions.html.haml` and `app/views/crops/_plantings.html.haml` were updated to include these new actions.

Co-authored-by: CloCkWeRX <365751+CloCkWeRX@users.noreply.github.com>
2026-02-22 00:08:58 +00:00
53 changed files with 347 additions and 144 deletions

View File

@@ -7,7 +7,6 @@ tmp/*
node_modules
public/assets
.env
.ruby-version
.ruby-gemset
.editorconfig
.esignore

View File

@@ -116,6 +116,8 @@ gem 'xmlrpc' # fixes rake error - can be removed if not needed later
gem 'puma'
gem 'rack-attack'
gem 'loofah', '>= 2.19.1'
gem 'rack-protection', '>= 2.0.1'

View File

@@ -503,6 +503,8 @@ GEM
query_diet (0.7.3)
racc (1.8.1)
rack (2.2.23)
rack-attack (6.8.0)
rack (>= 1.0, < 4)
rack-cors (2.0.2)
rack (>= 2.0.0)
rack-protection (3.2.0)
@@ -841,6 +843,7 @@ DEPENDENCIES
pry
puma
query_diet
rack-attack
rack-cors
rack-protection (>= 2.0.1)
rails (~> 7.2.0)

View File

@@ -8,9 +8,9 @@ module Admin
responders :flash
def index
@members = Member.all
@members = @members.where("login_name ILIKE ?", "%#{search_term}%") unless search_term.nil?
@members = @members.order(:login_name).paginate(page: params[:page])
@members = Member.order(:login_name)
@members = @members.where("login_name ILIKE ?", "%#{search_term}%") if search_term.present?
@members = @members.paginate(page: params[:page])
end
def edit

View File

@@ -12,7 +12,7 @@ class BlocksController < ApplicationController
else
flash[:error] = "Already blocking or error while blocking."
end
redirect_back fallback_location: root_path
redirect_back_or_to(root_path)
end
def destroy

View File

@@ -14,9 +14,12 @@ module Charts
def harvested_for
@crop = Crop.find_by!(slug: params[:crop_slug])
render json: Harvest.joins(:plant_part)
.where(crop: @crop)
.group("plant_parts.name").count(:id)
data = Rails.cache.fetch("#{@crop.cache_key_with_version}/harvested_for", expires_in: 1.day) do
Harvest.joins(:plant_part)
.where(crop: @crop)
.group("plant_parts.name").count(:id)
end
render json: data
end
private

View File

@@ -39,7 +39,7 @@ class MessagesController < ApplicationController
recipient = Member.find(params[:recipient_id])
if recipient.already_blocking?(current_member)
flash[:error] = "You cannot send a message to a member who has blocked you."
redirect_back fallback_location: root_path
redirect_back_or_to(root_path)
return
end
body = params[:body]

View File

@@ -31,8 +31,9 @@ class Comment < ApplicationRecord
def author_is_not_blocked
return unless author
if commentable.author.already_blocking?(author)
errors.add(:base, "You cannot comment on a post of a member who has blocked you.")
end
return unless commentable.author.already_blocking?(author)
errors.add(:base, "You cannot comment on a post of a member who has blocked you.")
end
end

View File

@@ -20,8 +20,9 @@ class Follow < ApplicationRecord
def follower_is_not_blocked
return unless follower
if followed.already_blocking?(follower)
errors.add(:base, "You cannot follow a member who has blocked you.")
end
return unless followed.already_blocking?(follower)
errors.add(:base, "You cannot follow a member who has blocked you.")
end
end

View File

@@ -19,9 +19,10 @@ class Like < ApplicationRecord
def member_is_not_blocked
return unless member
author = likeable_author
if author && author.already_blocking?(member)
errors.add(:base, "You cannot like content of a member who has blocked you.")
end
return unless author&.already_blocking?(member)
errors.add(:base, "You cannot like content of a member who has blocked you.")
end
end

View File

@@ -173,12 +173,12 @@ class Member < ApplicationRecord
end
def self.nearest_to(place)
nearby_members = []
if place
latitude, longitude = Geocoder.coordinates(place, params: { limit: 1 })
nearby_members = Member.located.sort_by { |x| x.distance_from([latitude, longitude]) } if latitude && longitude
end
nearby_members
return [] if place.blank?
latitude, longitude = Geocoder.coordinates(place, params: { limit: 1 })
return [] unless latitude && longitude
Member.located.near([latitude, longitude], 1000)
end
def already_following?(member)
@@ -196,4 +196,25 @@ class Member < ApplicationRecord
def get_block(member)
blocks.find_by(blocked_id: member.id) if already_blocking?(member)
end
def has_activity?
(gardens.exists? && gardens.count > 1) ||
plantings.exists? ||
harvests.exists? ||
seeds.exists? ||
photos.exists? ||
forums.exists? ||
activities.exists? ||
posts.exists? ||
comments.exists? ||
requested_crops.exists? ||
created_crops.exists? ||
likes.exists? ||
created_alternate_names.exists? ||
created_scientific_names.exists? ||
follows.exists? ||
inverse_follows.exists? ||
blocks.exists? ||
inverse_blocks.exists?
end
end

View File

@@ -49,9 +49,10 @@ class Post < ApplicationRecord
# return posts sorted by recent activity
def self.recently_active
Post.order(created_at: :desc).sort do |a, b|
b.recent_activity <=> a.recent_activity
end
left_joins(:comments)
.select('posts.*, COALESCE(MAX(comments.created_at), posts.created_at) AS last_activity_at')
.group('posts.id')
.order(Arel.sql('last_activity_at DESC'))
end
def owner_id

View File

@@ -85,7 +85,7 @@ class GbifService
end
def import!
Crop.order(updated_at: :desc).each do |crop|
Crop.order(updated_at: :desc).find_each do |crop|
Rails.logger.debug { "#{crop.id}, #{crop.name}" }
update_crop(crop) if crop.valid?
rescue ActiveRecord::RecordInvalid

View File

@@ -22,8 +22,8 @@
.form-group
= f.label :crop_id, class: 'control-label col-md-2'
.col-md-8
= collection_select(:alternate_name, :crop_id,
Crop.all, :id, :name,
= select(:alternate_name, :crop_id,
Crop.order(:name).pluck(:name, :id),
{ selected: @alternate_name.crop_id || @crop.id },
class: 'form-control')

View File

@@ -1,6 +1,9 @@
- if crop.approved? && signed_in?
- active_plantings = current_member.plantings.where(crop: crop).active
.btn-group.crop-actions{"aria-label" => "Crop Actions", role: "group"}
= render 'plantings/modal', planting: Planting.new(crop: crop, owner: current_member)
= render 'harvests/modal', harvest: Harvest.new(crop: @crop, owner: current_member)
= render 'seeds/modal', seed: Seed.new(crop: @crop, owner: current_member)
- if active_plantings.any?
= render 'plantings/failed_modal', crop: crop, active_plantings: active_plantings

View File

@@ -85,7 +85,7 @@
-# Only crop wranglers see the crop hierarchy (for now)
- if can? :wrangle, @crop
= f.collection_select(:parent_id, Crop.all.order(:name), :id, :name,
= f.select(:parent_id, Crop.order(:name).pluck(:name, :id),
{ include_blank: true, label: 'Parent crop'})
%span.help-block Optional. For setting up crop hierarchies for varieties etc.

View File

@@ -8,6 +8,8 @@
= link_to planting_path(planting), class: 'card-link' do
= planting_icon
= planting
- if can?(:edit, planting)
.float-right= render 'plantings/actions', planting: planting
.float-right= render 'members/location', member: planting.owner
.card-footer
- if crop.approved?

View File

@@ -73,10 +73,11 @@
= pie_chart crop_harvested_for_path(@crop, format: :json), legend: "bottom"
- if @crop.varieties.any?
%section.varieties
%h2 Varieties
.index-cards
= render 'varieties', crop: @crop
- cache [@crop, 'varieties'] do
%section.varieties
%h2 Varieties
.index-cards
= render 'varieties', crop: @crop
%section.crop-map
%h2
@@ -134,9 +135,11 @@
= render 'harvests', crop: @crop
= render 'find_seeds', crop: @crop
= render 'openfarm_data', crop: @crop
- cache [@crop, 'openfarm_data'] do
= render 'openfarm_data', crop: @crop
= render 'nutritional_data', crop: @crop
- cache [@crop, 'nutritional_data'] do
= render 'nutritional_data', crop: @crop
= cute_icon
.card

View File

@@ -0,0 +1,26 @@
#modelFailedPlantingForm.modal.fade{"aria-hidden" => "true", "aria-labelledby" => "failed-planting-button", role: "dialog", tabindex: "-1"}
.modal-dialog{role: "document"}
.modal-content
.modal-header.text-center
%h4.modal-title.w-100.font-weight-bold Mark #{crop.name} planting as failed
%button.close{"aria-label" => "Close", "data-bs-dismiss" => "modal", type: "button"}
%span{"aria-hidden" => "true"} &#215;
.modal-body
%p Which planting would you like to mark as failed?
%ul.list-group
- active_plantings.each do |planting|
%li.list-group-item
= link_to planting_path(planting, planting: {failed: 1}), method: :put do
.d-flex.justify-content-between
%span
%h4= planting.garden.name
%p Planted #{planting.planted_at}
%span
= finished_icon
.mt-3.text-right
= link_to 'cancel', '', "data-bs-dismiss" => "modal", class: 'btn btn-secondary'
%a.btn#failed-planting-button{"data-bs-target" => "#modelFailedPlantingForm", "data-bs-toggle" => "modal", href: ""}
= finished_icon
Mark as failed

View File

@@ -17,8 +17,8 @@
.form-group
= f.label :crop_id, class: 'control-label col-md-2'
.col-md-8
= collection_select(:scientific_name, :crop_id, Crop.all.order(:name), :id,
:name, { selected: @scientific_name.crop_id || @crop.id },
= select(:scientific_name, :crop_id, Crop.order(:name).pluck(:name, :id),
{ selected: @scientific_name.crop_id || @crop.id },
class: 'form-control')
.form-group
= f.label :name, class: 'control-label col-md-2'

View File

@@ -73,6 +73,8 @@ module Growstuff
config.newsletter_list_id = ENV.fetch('GROWSTUFF_MAILCHIMP_NEWSLETTER_ID', nil)
# config.active_record.raise_in_transactional_callbacks = true
config.middleware.insert_before 0, Rack::Attack
config.middleware.insert_before 0, Rack::Cors do
allow do
origins '*'

View File

@@ -0,0 +1,34 @@
# frozen_string_literal: true
class Rack::Attack
### Throttle Config ###
if Rails.env.production?
# Throttle requests to /plantings, /harvests, and /members to 10 per minute per IP
# Includes API routes
throttle('req/ip/restricted_routes', limit: 20, period: 1.minute) do |req|
if req.path =~ %r{^/(plantings|harvests|members)(/|$)} || req.path =~ %r{^/api/v1/(plantings|harvests|members)(/|$)}
req.ip
end
end
### Fail2Ban Config ###
# Block IPs that make too many requests to suspicious paths
# After 5 "bad" requests in 10 minutes, block the IP for 1 hour
blocklist('fail2ban/pentesters') do |req|
Fail2Ban.filter("pentesters-#{req.ip}", maxretry: 5, findtime: 10.minutes, bantime: 1.hour) do
# The count for the IP is incremented if the return value is truthy.
req.path.include?('wp-admin') ||
req.path.include?('wp-login') ||
req.path.include?('cgi-bin') ||
req.path.end_with?('.php', '.asp', '.aspx', '.jsp', '.exe', '.env', '.git')
end
end
end
### Custom Response Headers ###
# Add Retry-After header to throttled responses
self.throttled_response_retry_after_header = true
end

View File

@@ -8,6 +8,6 @@ class CreateBlocks < ActiveRecord::Migration[6.1]
t.timestamps
end
add_index :blocks, [:blocker_id, :blocked_id], unique: true
add_index :blocks, %i(blocker_id blocked_id), unique: true
end
end

33
lib/tasks/members.rake Normal file
View File

@@ -0,0 +1,33 @@
# frozen_string_literal: true
namespace :members do
desc "Remove inactive members with no activity and last login > 24 months ago"
# usage: rake members:cleanup_inactive
# usage: DRY_RUN=true rake members:cleanup_inactive
task cleanup_inactive: :environment do
limit_date = 3.years.ago
dry_run = ENV.fetch('DRY_RUN', 'false') == 'true'
inactive_members = Member.where("last_sign_in_at < ? OR (last_sign_in_at IS NULL AND created_at < ?)", limit_date, limit_date)
count = 0
inactive_members.find_each do |member|
# Check for activity using the model method
unless member.has_activity?
if dry_run
puts "[DRY RUN] Would delete inactive member: #{member.login_name} (ID: #{member.id}, Last login: #{member.last_sign_in_at || 'Never'}, Created: #{member.created_at})"
else
puts "Deleting inactive member: #{member.login_name} (ID: #{member.id}, Last login: #{member.last_sign_in_at || 'Never'}, Created: #{member.created_at})"
member.destroy
end
count += 1
end
end
if dry_run
puts "Total inactive members that would be deleted: #{count}"
else
puts "Total inactive members deleted: #{count}"
end
end
end

View File

@@ -49,9 +49,9 @@ describe ForumsController do
let(:valid_attributes) { { name: "New Forum", description: "A new forum", owner_id: admin.id } }
it "creates a new Forum" do
expect {
expect do
post :create, params: { forum: valid_attributes }
}.to change(Forum, :count).by(1)
end.to change(Forum, :count).by(1)
end
it "redirects to the created forum" do
@@ -81,9 +81,9 @@ describe ForumsController do
describe "DELETE #destroy" do
it "destroys the requested forum" do
forum # ensure forum exists
expect {
expect do
delete :destroy, params: { id: forum.to_param }
}.to change(Forum, :count).by(-1)
end.to change(Forum, :count).by(-1)
end
it "redirects to the forums list" do

View File

@@ -9,18 +9,18 @@ describe PlacesController do
describe "GET show" do
before do
@member_london = create(:london_member)
@member_south_pole = create(:south_pole_member)
@london_member = create(:london_member)
@edinburgh_member = create(:edinburgh_member)
end
it "assigns place name" do
get :show, params: { place: @member_london.location }
assigns(:place).should eq @member_london.location
get :show, params: { place: @london_member.location }
assigns(:place).should eq @london_member.location
end
it "assigns nearby members" do
get :show, params: { place: @member_london.location }
assigns(:nearby_members).should eq [@member_london, @member_south_pole]
get :show, params: { place: @london_member.location }
assigns(:nearby_members).should eq [@london_member, @edinburgh_member]
end
end

View File

@@ -18,8 +18,8 @@ describe PlantPart do
@h2 = create(:harvest,
crop: @maize,
plant_part: @pp1)
@pp1.crops.should include @tomato
@pp1.crops.should include @maize
expect(@pp1.crops).to include @tomato
expect(@pp1.crops).to include @maize
end
it "doesn't duplicate crops" do
@@ -31,6 +31,6 @@ describe PlantPart do
@h2 = create(:harvest,
crop: @maize,
plant_part: @pp1)
@pp1.crops.should eq [@maize]
expect(@pp1.crops).to eq [@maize]
end
end

View File

@@ -7,35 +7,35 @@ describe Seed do
let(:seed) { build(:seed, owner:) }
it 'saves a basic seed' do
seed.save.should be(true)
expect(seed.save).to be(true)
end
it "has a slug" do
seed.save
seed.slug.should match(/tamateapokaiwhenua-magic-bean/)
expect(seed.slug).to match(/tamateapokaiwhenua-magic-bean/)
end
context 'quantity' do
it 'allows integer quantities' do
@seed = build(:seed, quantity: 99)
@seed.should be_valid
expect(@seed).to be_valid
end
it "doesn't allow decimal quantities" do
@seed = build(:seed, quantity: 99.9)
@seed.should_not be_valid
expect(@seed).not_to be_valid
end
it "doesn't allow non-numeric quantities" do
@seed = build(:seed, quantity: 'foo')
@seed.should_not be_valid
expect(@seed).not_to be_valid
end
it "allows blank quantities" do
@seed = build(:seed, quantity: nil)
@seed.should be_valid
expect(@seed).to be_valid
@seed = build(:seed, quantity: '')
@seed.should be_valid
expect(@seed).to be_valid
end
end
@@ -43,14 +43,14 @@ describe Seed do
it 'all valid tradable_to values should work' do
%w(nowhere locally nationally internationally).each do |t|
@seed = build(:seed, tradable_to: t)
@seed.should be_valid
expect(@seed).to be_valid
end
end
it 'refuses invalid tradable_to values' do
@seed = build(:seed, tradable_to: 'not valid')
@seed.should_not be_valid
@seed.errors[:tradable_to].should include(
expect(@seed).not_to be_valid
expect(@seed.errors[:tradable_to]).to include(
"You may only trade seed nowhere, locally, " \
"nationally, or internationally"
)
@@ -58,35 +58,35 @@ describe Seed do
it 'does not allow nil or blank values' do
@seed = build(:seed, tradable_to: nil)
@seed.should_not be_valid
expect(@seed).not_to be_valid
@seed = build(:seed, tradable_to: '')
@seed.should_not be_valid
expect(@seed).not_to be_valid
end
it 'tradable gives the right answers' do
@seed = create(:seed, tradable_to: 'nowhere')
@seed.tradable.should be false
expect(@seed.tradable).to be false
@seed = create(:seed, tradable_to: 'locally')
@seed.tradable.should be true
expect(@seed.tradable).to be true
@seed = create(:seed, tradable_to: 'nationally')
@seed.tradable.should be true
expect(@seed.tradable).to be true
@seed = create(:seed, tradable_to: 'internationally')
@seed.tradable.should be true
expect(@seed.tradable).to be true
end
it 'recognises a tradable seed' do
create(:tradable_seed).tradable.should == true
expect(create(:tradable_seed).tradable).to be true
end
it 'recognises an untradable seed' do
create(:untradable_seed).tradable.should == false
expect(create(:untradable_seed).tradable).to be false
end
it 'scopes correctly' do
@tradable = create(:tradable_seed)
@untradable = create(:untradable_seed)
described_class.tradable.should include @tradable
described_class.tradable.should_not include @untradable
expect(described_class.tradable).to include @tradable
expect(described_class.tradable).not_to include @untradable
end
end
@@ -95,7 +95,7 @@ describe Seed do
['certified organic', 'non-certified organic',
'conventional/non-organic', 'unknown'].each do |t|
@seed = build(:seed, organic: t)
@seed.should be_valid
expect(@seed).to be_valid
end
end
@@ -103,31 +103,31 @@ describe Seed do
['certified GMO-free', 'non-certified GMO-free',
'GMO', 'unknown'].each do |t|
@seed = build(:seed, gmo: t)
@seed.should be_valid
expect(@seed).to be_valid
end
end
it 'all valid heirloom values should work' do
%w(heirloom hybrid unknown).each do |t|
@seed = build(:seed, heirloom: t)
@seed.should be_valid
expect(@seed).to be_valid
end
end
it 'refuses invalid organic/GMO/heirloom values' do
%i(organic gmo heirloom).each do |field|
@seed = build(:seed, field => 'not valid')
@seed.should_not be_valid
@seed.errors[field].should_not be_empty
expect(@seed).not_to be_valid
expect(@seed.errors[field]).not_to be_empty
end
end
it 'does not allow nil or blank values' do
%i(organic gmo heirloom).each do |field|
@seed = build(:seed, field => nil)
@seed.should_not be_valid
expect(@seed).not_to be_valid
@seed = build(:seed, field => '')
@seed.should_not be_valid
expect(@seed).not_to be_valid
end
end
end
@@ -136,13 +136,13 @@ describe Seed do
it 'returns seeds with a plant_before date in the past' do
expired_seed = create(:seed, plant_before: 1.day.ago)
not_expired_seed = create(:seed, plant_before: 1.day.from_now)
described_class.expired.should include expired_seed
described_class.expired.should_not include not_expired_seed
expect(described_class.expired).to include expired_seed
expect(described_class.expired).not_to include not_expired_seed
end
it 'does not return finished seeds' do
expired_seed = create(:seed, plant_before: 1.day.ago, finished: true)
described_class.expired.should_not include expired_seed
expect(described_class.expired).not_to include expired_seed
end
end
@@ -158,11 +158,11 @@ describe Seed do
@seed3 = create(:tradable_seed)
@seed4 = create(:seed)
described_class.interesting.should include @seed1
described_class.interesting.should_not include @seed2
described_class.interesting.should_not include @seed3
described_class.interesting.should_not include @seed4
described_class.interesting.size.should == 1
expect(described_class.interesting).to include @seed1
expect(described_class.interesting).not_to include @seed2
expect(described_class.interesting).not_to include @seed3
expect(described_class.interesting).not_to include @seed4
expect(described_class.interesting.size).to eq 1
end
end
@@ -172,7 +172,7 @@ describe Seed do
before { seed.photos << create(:photo, owner: seed.owner) }
it 'is found in has_photos scope' do
described_class.has_photos.should include(seed)
expect(described_class.has_photos).to include(seed)
end
end

View File

@@ -37,9 +37,9 @@ describe "Forums" do
before { sign_in admin }
it "creates a new forum" do
expect {
expect do
post forums_path, params: { forum: { name: "New Request Forum", description: "Desc", owner_id: admin.id } }
}.to change(Forum, :count).by(1)
end.to change(Forum, :count).by(1)
expect(response).to redirect_to(forum_path(Forum.last))
end
end

View File

@@ -0,0 +1,68 @@
# frozen_string_literal: true
require 'rails_helper'
require 'rake'
describe 'members:cleanup_inactive' do
before :all do
Rails.application.load_tasks
end
let(:cleanup_task) { Rake::Task['members:cleanup_inactive'] }
before do
cleanup_task.reenable
end
it "deletes inactive members with no gardens and no other activity" do
inactive_no_activity = create(:member, last_sign_in_at: 25.months.ago)
# We must explicitly remove the default garden to test the "no gardens" condition
inactive_no_activity.gardens.destroy_all
expect(inactive_no_activity.gardens.count).to eq(0)
cleanup_task.invoke
expect(Member.exists?(inactive_no_activity.id)).to be_falsey
end
it "does not delete inactive members with a garden (even if empty)" do
inactive_with_garden = create(:member, last_sign_in_at: 25.months.ago)
# They have 1 default garden
expect(inactive_with_garden.gardens.count).to eq(1)
cleanup_task.invoke
expect(Member.exists?(inactive_with_garden.id)).to be_truthy
end
it "does not delete members with recent login" do
recent_member = create(:member, last_sign_in_at: 1.month.ago)
recent_member.gardens.destroy_all
cleanup_task.invoke
expect(Member.exists?(recent_member.id)).to be_truthy
end
it "does not delete inactive members with activity (posts)" do
inactive_with_post = create(:member, last_sign_in_at: 25.months.ago)
inactive_with_post.gardens.destroy_all
create(:post, author: inactive_with_post)
cleanup_task.invoke
expect(Member.exists?(inactive_with_post.id)).to be_truthy
end
it "honors DRY_RUN environment variable" do
inactive_no_activity = create(:member, last_sign_in_at: 25.months.ago)
inactive_no_activity.gardens.destroy_all
ENV['DRY_RUN'] = 'true'
cleanup_task.invoke
ENV['DRY_RUN'] = nil
expect(Member.exists?(inactive_no_activity.id)).to be_truthy
end
end

View File

@@ -15,11 +15,11 @@ describe 'comments/index.rss.haml' do
end
it 'shows RSS feed title' do
rendered.should have_content "Recent comments on all posts"
expect(rendered).to have_content "Recent comments on all posts"
end
it 'shows item title' do
rendered.should have_content "Comment by #{@author.login_name}"
expect(rendered).to have_content "Comment by #{@author.login_name}"
end
it 'escapes html for link to post' do
@@ -28,6 +28,6 @@ describe 'comments/index.rss.haml' do
end
it 'shows content of comments' do
rendered.should have_content "OMG LOL"
expect(rendered).to have_content "OMG LOL"
end
end

View File

@@ -13,7 +13,7 @@ describe "crops/_grown_for" do
it 'shows plant parts' do
render partial: 'crops/grown_for', locals: { crop: }
rendered.should have_content plant_path.name
expect(rendered).to have_content plant_path.name
assert_select "a", href: plant_part_path(plant_path)
end
end

View File

@@ -12,10 +12,10 @@ describe "crops/_popover" do
end
it 'has a scientific name' do
rendered.should have_content 'Solanum lycopersicum'
expect(rendered).to have_content 'Solanum lycopersicum'
end
it 'shows count of plantings' do
rendered.should have_content '1 time'
expect(rendered).to have_content '1 time'
end
end

View File

@@ -33,7 +33,7 @@ describe "crops/index.html.haml" do
context "downloads" do
it "offers data downloads" do
render
rendered.should have_content "The data on this page is available in the following formats:"
expect(rendered).to have_content "The data on this page is available in the following formats:"
assert_select "a", href: crops_path(format: 'csv')
assert_select "a", href: crops_path(format: 'json')
assert_select "a", href: crops_path(format: 'rss')

View File

@@ -13,11 +13,11 @@ describe 'crops/index.rss.haml' do
end
it 'shows RSS feed title' do
rendered.should have_content "Recently added crops"
expect(rendered).to have_content "Recently added crops"
end
it 'shows names of crops' do
rendered.should have_content @tomato.name
rendered.should have_content @maize.name
expect(rendered).to have_content @tomato.name
expect(rendered).to have_content @maize.name
end
end

View File

@@ -10,6 +10,6 @@ describe 'devise/confirmations/new.html.haml', type: "view" do
end
it 'contains a login field' do
rendered.should have_content "Enter either your login name or your email address"
expect(rendered).to have_content "Enter either your login name or your email address"
end
end

View File

@@ -10,11 +10,11 @@ describe 'devise/mailer/confirmation_instructions.html.haml', type: "view" do
end
it 'has a confirmation link' do
rendered.should have_content 'Confirm my account'
expect(rendered).to have_content 'Confirm my account'
end
it 'has a link to the homepage' do
rendered.should have_content root_url
expect(rendered).to have_content root_url
end
end
end

View File

@@ -12,8 +12,8 @@ describe 'devise/mailer/reset_password_instructions.html.haml', type: "view" do
end
it 'has some of the right text' do
rendered.should have_content 'Change my password'
rendered.should have_content 'Someone has requested a link to reset your password'
expect(rendered).to have_content 'Change my password'
expect(rendered).to have_content 'Someone has requested a link to reset your password'
end
end
end

View File

@@ -9,11 +9,11 @@ describe 'devise/mailer/unlock_instructions.html.haml', type: "view" do
end
it "explains what's happened" do
rendered.should have_content "account has been locked"
expect(rendered).to have_content "account has been locked"
end
it "has an unlock link" do
rendered.should have_content "Unlock my account"
expect(rendered).to have_content "Unlock my account"
end
end
end

View File

@@ -16,7 +16,7 @@ describe 'devise/registrations/edit.html.haml', type: "view" do
it 'has some fields' do
render
rendered.should have_content 'Email'
expect(rendered).to have_content 'Email'
end
context 'email section' do

View File

@@ -13,7 +13,7 @@ describe 'devise/registrations/new.html.haml', type: "view" do
end
it 'has some fields' do
rendered.should have_content 'Email'
expect(rendered).to have_content 'Email'
end
it 'has a checkbox for newsletter subscription' do

View File

@@ -13,8 +13,8 @@ describe 'devise/sessions/new.html.haml', type: "view" do
end
it 'has some fields' do
rendered.should have_content 'Remember me'
rendered.should have_content 'Password'
expect(rendered).to have_content 'Remember me'
expect(rendered).to have_content 'Password'
end
end
end

View File

@@ -10,7 +10,7 @@ describe 'home/_blurb.html.haml', type: "view" do
end
it 'has description' do
rendered.should have_content 'is a community of food gardeners'
expect(rendered).to have_content 'is a community of food gardeners'
end
it 'has signup section' do
@@ -19,7 +19,7 @@ describe 'home/_blurb.html.haml', type: "view" do
end
it 'has a link to sign in' do
rendered.should have_content "Or sign in if you already have an account"
expect(rendered).to have_content "Or sign in if you already have an account"
assert_select "a", href: new_member_session_path
end
end

View File

@@ -5,6 +5,6 @@ require 'rails_helper'
describe 'home/_stats.html.haml', type: "view" do
it 'has activity stats' do
render
rendered.should have_content "So far, 0 members have planted 0 crops"
expect(rendered).to have_content "So far, 0 members have planted 0 crops"
end
end

View File

@@ -29,8 +29,8 @@ describe 'home/index.html.haml', type: "view" do
end
it 'show interesting members' do
rendered.should have_content @member.login_name
rendered.should have_content @member.location
expect(rendered).to have_content @member.login_name
expect(rendered).to have_content @member.location
end
end
end

View File

@@ -14,24 +14,24 @@ describe 'layouts/_header.html.haml', type: "view" do
end
it 'has signup/signin links' do
rendered.should have_content 'Sign up'
rendered.should have_content 'Sign in'
expect(rendered).to have_content 'Sign up'
expect(rendered).to have_content 'Sign in'
end
it 'has a Crops link' do
rendered.should have_content "Crops"
expect(rendered).to have_content "Crops"
end
it 'has a Seeds link' do
rendered.should have_content "Seeds"
expect(rendered).to have_content "Seeds"
end
it 'has a Places link' do
rendered.should have_content "Community Map"
expect(rendered).to have_content "Community Map"
end
it 'has a Community section' do
rendered.should have_content "Community"
expect(rendered).to have_content "Community"
end
it 'links to members' do
@@ -62,7 +62,7 @@ describe 'layouts/_header.html.haml', type: "view" do
context "login name" do
it 'has member login name' do
rendered.should have_content @member.login_name.to_s
expect(rendered).to have_content @member.login_name.to_s
end
it "shows link to member's gardens" do
@@ -83,19 +83,19 @@ describe 'layouts/_header.html.haml', type: "view" do
end
it 'shows signout link' do
rendered.should have_content 'Sign out'
expect(rendered).to have_content 'Sign out'
end
it 'shows inbox link' do
rendered.should have_content 'Inbox'
rendered.should_not match(/Inbox \d+/)
expect(rendered).to have_content 'Inbox'
expect(rendered).not_to match(/Inbox \d+/)
end
context 'has notifications' do
it 'shows inbox count' do
create(:notification, recipient: @member)
render
rendered.should have_content 'Inbox 1'
expect(rendered).to have_content 'Inbox 1'
end
end
end

View File

@@ -11,6 +11,6 @@ describe 'layouts/application.html.haml', type: "view" do
Rails.application.config.analytics_code = '<script>console.log("foo!");</script>'
render
assert_select "script", text: 'console.log("foo!");'
rendered.should_not have_content 'script'
expect(rendered).to have_no_content 'script'
end
end

View File

@@ -109,7 +109,7 @@ describe "photos/show" do
end
it "contains the phrase 'All rights reserved'" do
rendered.should have_content "All rights reserved"
expect(rendered).to have_content "All rights reserved"
end
end
end

View File

@@ -22,7 +22,7 @@ describe "places/show" do
it "shows the names of nearby members" do
@nearby_members.each do |m|
rendered.should have_content m.login_name
expect(rendered).to have_content m.login_name
end
end
end

View File

@@ -45,7 +45,7 @@ describe "plantings/index.html.haml" do
it "provides data links" do
render
rendered.should have_content "The data on this page is available in the following formats:"
expect(rendered).to have_content "The data on this page is available in the following formats:"
assert_select "a", href: plantings_path(format: 'csv')
assert_select "a", href: plantings_path(format: 'json')
assert_select "a", href: plantings_path(format: 'rss')

View File

@@ -54,11 +54,11 @@ describe "posts/_single" do
end
it "shows edited at" do
rendered.should have_content "edited at"
expect(rendered).to have_content "edited at"
end
it "shows the updated time" do
rendered.should have_content @post.updated_at.to_fs(:default)
expect(rendered).to have_content @post.updated_at.to_fs(:default)
end
end
@@ -74,11 +74,11 @@ describe "posts/_single" do
end
it "shows edited at time" do
rendered.should have_content "edited at"
expect(rendered).to have_content "edited at"
end
it "shows updated time" do
rendered.should have_content @comment.updated_at
expect(rendered).to have_content @comment.updated_at
end
end
@@ -93,7 +93,7 @@ describe "posts/_single" do
end
it "does not show edited at" do
rendered.should_not have_content "edited at #{@post.updated_at}"
expect(rendered).to have_no_content "edited at #{@post.updated_at}"
end
end
@@ -109,7 +109,7 @@ describe "posts/_single" do
end
it "does not show edited at" do
rendered.should_not have_content "edited at #{@comment.updated_at}"
expect(rendered).to have_no_content "edited at #{@comment.updated_at}"
end
end
end

View File

@@ -27,7 +27,7 @@ describe "posts/edit" do
end
it 'no forum mentioned' do
rendered.should_not have_content "This post will be posted in the forum"
expect(rendered).to have_no_content "This post will be posted in the forum"
end
context "forum specified" do
@@ -44,7 +44,7 @@ describe "posts/edit" do
end
it 'tells the user what forum it will be posted in' do
rendered.should have_content "This post will be posted in the forum #{@forum.name}"
expect(rendered).to have_content "This post will be posted in the forum #{@forum.name}"
end
end
end

View File

@@ -14,19 +14,19 @@ describe 'posts/show.rss.haml' do
end
it 'shows RSS feed title' do
rendered.should have_content "Recent comments on #{@post.subject}"
expect(rendered).to have_content "Recent comments on #{@post.subject}"
end
it 'shows item title' do
rendered.should have_content "Comment by #{@author.login_name}"
expect(rendered).to have_content "Comment by #{@author.login_name}"
end
it 'escapes html for link to post' do
# it's then unescaped by 'render' so we don't actually look for &lt;
rendered.should have_content '<a href='
expect(rendered).to have_content '<a href='
end
it 'shows content of comments' do
rendered.should have_content "OMG LOL"
expect(rendered).to have_content "OMG LOL"
end
end