Skip to content

Commit

Permalink
Merge branch 'batch-aggregation' of github.com:zooniverse/panoptes in…
Browse files Browse the repository at this point in the history
…to batch-aggregation
  • Loading branch information
zwolf committed May 28, 2024
2 parents b90bdcc + 419dc65 commit c350406
Show file tree
Hide file tree
Showing 16 changed files with 146 additions and 62 deletions.
4 changes: 3 additions & 1 deletion Gemfile.lock
Original file line number Diff line number Diff line change
Expand Up @@ -402,7 +402,8 @@ GEM
http-cookie (>= 1.0.2, < 2.0)
mime-types (>= 1.16, < 4.0)
netrc (~> 0.8)
rexml (3.2.5)
rexml (3.2.8)
strscan (>= 3.0.9)
rspec (3.12.0)
rspec-core (~> 3.12.0)
rspec-expectations (~> 3.12.0)
Expand Down Expand Up @@ -487,6 +488,7 @@ GEM
stringex (2.8.6)
strong_migrations (1.4.4)
activerecord (>= 5.2)
strscan (3.1.0)
ten_years_rails (0.2.0)
actionview
activesupport
Expand Down
2 changes: 1 addition & 1 deletion app/controllers/api/v1/memberships_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ class Api::V1::MembershipsController < Api::ApiController
resource_actions :index, :show, :create, :update, :deactivate
schema_type :strong_params

allowed_params :update, :state
allowed_params :update, :state, roles: []

def create
resources = resource_class.transaction(requires_new: true) do
Expand Down
4 changes: 2 additions & 2 deletions app/controllers/api/v1/user_groups_controller.rb
Original file line number Diff line number Diff line change
Expand Up @@ -9,8 +9,8 @@ class Api::V1::UserGroupsController < Api::ApiController

alias_method :user_group, :controlled_resource

allowed_params :create, :name, :display_name, :stats_visibility, links: [users: []]
allowed_params :update, :name, :stats_visibility, :display_name
allowed_params :create, :name, :display_name, :private, :stats_visibility, links: [users: []]
allowed_params :update, :name, :private, :stats_visibility, :display_name

search_by do |name, query|
search_names = name.join(' ').downcase
Expand Down
2 changes: 1 addition & 1 deletion app/models/user.rb
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ class User < ApplicationRecord
validates :login, presence: true, format: { with: USER_LOGIN_REGEX }
validates_uniqueness_of :login, case_sensitive: false
validates :display_name, presence: true
validates :unsubscribe_token, presence: true, uniqueness: true
validates :unsubscribe_token, presence: true, uniqueness: true, on: :create
validates_inclusion_of :valid_email, in: [true, false], message: "must be true or false"

validates_with IdentityGroupNameValidator
Expand Down
2 changes: 1 addition & 1 deletion app/policies/membership_policy.rb
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,7 @@ def resolve(action)
when :show, :index
public_memberships = scope.where(identity: false, state: Membership.states[:active])
.joins(:user_group).merge(UserGroup.where(private: false))
Membership.union_all(query, public_memberships)
Membership.union_all(query, public_memberships).distinct
else
query
end
Expand Down
2 changes: 1 addition & 1 deletion app/serializers/workflow_serializer.rb
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@ class WorkflowSerializer

can_filter_by :active, :mobile_friendly

can_sort_by :completeness
can_sort_by :completeness, :id

preload :subject_sets, :attached_images, :classifications_export, :published_version

Expand Down
4 changes: 2 additions & 2 deletions lib/project_copier.rb
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ class ProjectCopier
launched_row_order
beta_row_order
].freeze

INCLUDE_ASSOCIATIONS = [
:tutorials,
:pages,
Expand Down Expand Up @@ -61,7 +61,7 @@ def copy_project
copied_project = project_to_copy.deep_clone include: INCLUDE_ASSOCIATIONS, except: EXCLUDE_ATTRIBUTES
copied_project.owner = user

copied_project.display_name += ' (copy)' if user == project_to_copy.owner
copied_project.display_name += " (copy: #{Time.now.utc.strftime('%Y-%m-%d %H:%M:%S')})"

copied_project.assign_attributes(launch_approved: false, live: false)
# reset the project's configuration but record the source project id
Expand Down
4 changes: 2 additions & 2 deletions lib/subjects/remover.rb
Original file line number Diff line number Diff line change
Expand Up @@ -30,10 +30,10 @@ def cleanup
private

def can_be_removed?
return false if belongs_to_other_subject_set?

return false if has_been_collected_or_classified?

return false if belongs_to_other_subject_set?

return false if has_been_talked_about?

return false if has_been_counted_or_retired?
Expand Down
8 changes: 8 additions & 0 deletions lib/tasks/user.rake
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,14 @@ namespace :user do
end
end

desc 'Scrub unsubscribe_token of deactivated users'
task scrub_inactive_user_unsubscribe_token: :environment do
User.where(activated_state: 'inactive').select(:id).find_in_batches do |users|
ids = users.map(&:id)
User.where(id: ids).update_all(unsubscribe_token: nil)
end
end

desc 'Backfill UX testing emails comms field in batches'
task backfill_ux_testing_email_field: :environment do
User.select(:id).find_in_batches do |users|
Expand Down
1 change: 1 addition & 0 deletions lib/user_info_scrubber.rb
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ def self.scrub_details(user)
user.valid_email = false
user.private_profile = true
user.api_key = nil
user.unsubscribe_token = nil
user.tsv = nil
user.save!
user
Expand Down
17 changes: 17 additions & 0 deletions spec/controllers/api/v1/memberships_controller_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -35,6 +35,23 @@
end

it_behaves_like "is updatable"

context 'when updating roles as group_admin' do
let(:other_user) { create(:user) }
let(:other_user_group) { create(:user_group, admin: authorized_user, private: true) }
let(:other_user_membership) { create(:membership, user: other_user, user_group: other_user_group, roles: ['group_member']) }

it 'allows update of membership roles' do
default_request user_id: authorized_user.id, scopes: scopes
params = {
memberships: { roles: ['group_admin'] },
id: other_user_membership.id
}

put :update, params: params
expect(other_user_membership.reload.roles).to eq(['group_admin'])
end
end
end

describe "#create" do
Expand Down
5 changes: 3 additions & 2 deletions spec/controllers/api/v1/user_groups_controller_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -106,7 +106,8 @@
let(:update_params) do
{
user_groups: {
display_name: 'A-Different-Name'
display_name: 'A-Different-Name',
private: false
}
}
end
Expand Down Expand Up @@ -193,7 +194,7 @@
describe '#create' do
let(:test_attr) { :name }
let(:test_attr_value) { 'Zooniverse' }
let(:create_params) { { user_groups: { name: 'Zooniverse' } } }
let(:create_params) { { user_groups: { name: 'Zooniverse', private: false } } }

it_behaves_like 'is creatable'

Expand Down
10 changes: 3 additions & 7 deletions spec/lib/project_copier_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -16,13 +16,9 @@
expect(copied_project.owner).to eq(copyist)
end

it 'renames a project when the owner is copying their own project' do
new_copy = described_class.new(project.id, project.owner.id).copy
expect(new_copy.display_name).to include('(copy)')
end

it 'has matching attributes' do
expect(copied_project.display_name).to eq(project.display_name)
it 'appends copy and current timestamp to display_name' do
allow(Time).to receive(:now).and_return(Time.utc(2024, 5, 17, 12, 0, 0))
expect(copied_project.display_name).to eq("#{project.display_name} (copy: 2024-05-17 12:00:00)")
end

it 'resets the live attribute' do
Expand Down
32 changes: 24 additions & 8 deletions spec/lib/subjects/remover_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -105,14 +105,30 @@
expect { SetMemberSubject.find(sms_ids) }.to raise_error(ActiveRecord::RecordNotFound)
end

context 'with multiple subject sets' do
let(:alternate_subject_set) { create(:subject_set) }
let(:remover) { described_class.new(subject.id, panoptes_client, alternate_subject_set.id) }
let(:new_sms) { create(:set_member_subject, subject: subject, subject_set: alternate_subject_set) }

it 'does not remove subjects associated with multiple set_member_subjects' do
remover.cleanup
expect(Subject.where(id: subject.id)).to exist
context 'when subject_set_id is param in init' do
let(:remover_with_subject_set) {
described_class.new(subject.id, panoptes_client, subject_set.id)
}

it 'removes a subject that has not been used' do
remover_with_subject_set.cleanup
expect { Subject.find(subject.id) }.to raise_error(ActiveRecord::RecordNotFound)
end

it 'does not remove a subject that has been classified' do
create(:classification, subjects: [subject])
expect(remover_with_subject_set.cleanup).to be_falsey
end

context 'with multiple subject sets' do
let(:alternate_subject_set) { create(:subject_set) }
let(:remover) { described_class.new(subject.id, panoptes_client, alternate_subject_set.id) }
let(:new_sms) { create(:set_member_subject, subject: subject, subject_set: alternate_subject_set) }

it 'does not remove subjects associated with multiple set_member_subjects' do
remover.cleanup
expect(Subject.where(id: subject.id)).to exist
end
end
end

Expand Down
4 changes: 4 additions & 0 deletions spec/lib/user_info_scrubber_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,10 @@ def scrub_user_details(user)(user)
expect { described_class.scrub_personal_info!(user) }.to change(user, :credited_name).to(nil)
end

it 'scrubs the unsubscribe_token' do
expect { described_class.scrub_personal_info!(user) }.to change(user, :unsubscribe_token).to(nil)
end

it 'scrubs the encrypted_password' do
expect { described_class.scrub_personal_info!(user) }.to change(user, :encrypted_password)
end
Expand Down
107 changes: 73 additions & 34 deletions spec/policies/membership_policy_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,11 @@
MembershipPolicy::Scope.new(api_user, Membership)
end

let(:group_admin_public_membership) { Membership.where(user_id: group_admin.id, user_group_id: public_user_group.id, state: 'active')[0] }
let(:group_admin_private_membership) {
Membership.where(user_id: group_admin.id, user_group_id: private_user_group.id, state: 'active')[0]
}

context 'an anonymous user' do
let(:api_user) { ApiUser.new(anonymous_user) }

Expand All @@ -31,44 +36,58 @@

context 'a normal user', :aggregate_failures do
let(:api_user) { ApiUser.new(logged_in_user) }
let(:other_user) { create(:user) }
let(:other_user_public_membership) { create(:membership, user: other_user, user_group: public_user_group) }
let(:other_user_private_membership) { create(:membership, user: other_user, user_group: private_user_group) }
let(:logged_in_user_public_membership) { create(:membership, user: logged_in_user, user_group: public_user_group) }
let(:logged_in_user_private_membership) { create(:membership, user: logged_in_user, user_group: private_user_group) }

it 'can see other people in public groups' do
other_user = create :user
membership1 = create :membership, user: other_user, user_group: public_user_group
membership2 = create :membership, user: other_user, user_group: private_user_group
membership3 = create :membership, user_group: public_user_group, state: :inactive

expect(scope.resolve(:index)).to include(membership1)
expect(scope.resolve(:show)).to include(membership1)
expect(scope.resolve(:index)).not_to include(membership2)
expect(scope.resolve(:show)).not_to include(membership2)
expect(scope.resolve(:index)).not_to include(membership3)
expect(scope.resolve(:show)).not_to include(membership3)
inactive_membership = create :membership, user_group: public_user_group, state: :inactive

expect(scope.resolve(:index)).to contain_exactly(other_user_public_membership, group_admin_public_membership)
expect(scope.resolve(:show)).to contain_exactly(other_user_public_membership, group_admin_public_membership)
expect(scope.resolve(:index)).not_to include(other_user_private_membership)
expect(scope.resolve(:show)).not_to include(other_user_private_membership)
expect(scope.resolve(:index)).not_to include(inactive_membership)
expect(scope.resolve(:show)).not_to include(inactive_membership)
end

it 'can see who are members of private groups they are in' do
other_user = create :user
membership1 = create :membership, user: logged_in_user, user_group: public_user_group
membership2 = create :membership, user: logged_in_user, user_group: private_user_group
membership3 = create :membership, user: other_user, user_group: public_user_group
membership4 = create :membership, user: other_user, user_group: private_user_group

expect(scope.resolve(:index)).to include(membership1, membership2, membership3, membership4)
expect(scope.resolve(:show)).to include(membership1, membership2, membership3, membership4)
expect(scope.resolve(:index)).to contain_exactly(
logged_in_user_public_membership,
logged_in_user_private_membership,
other_user_public_membership,
other_user_private_membership,
group_admin_public_membership,
group_admin_private_membership
)
expect(scope.resolve(:show)).to contain_exactly(
logged_in_user_public_membership,
logged_in_user_private_membership,
other_user_public_membership,
other_user_private_membership,
group_admin_public_membership,
group_admin_private_membership
)
end

it "cannot modify other user's memberships" do
other_user = create :user
membership1 = create :membership, user: logged_in_user, user_group: public_user_group
membership2 = create :membership, user: logged_in_user, user_group: private_user_group
membership3 = create :membership, user: other_user, user_group: public_user_group
membership4 = create :membership, user: other_user, user_group: private_user_group

expect(scope.resolve(:update)).to include(membership1, membership2)
expect(scope.resolve(:destroy)).to include(membership1, membership2)

expect(scope.resolve(:update)).not_to include(membership3, membership4)
expect(scope.resolve(:destroy)).not_to include(membership3, membership4)
expect(scope.resolve(:update)).to include(logged_in_user_public_membership, logged_in_user_private_membership)
expect(scope.resolve(:destroy)).to include(logged_in_user_public_membership, logged_in_user_private_membership)

expect(scope.resolve(:update)).not_to include(
other_user_private_membership,
other_user_public_membership,
group_admin_private_membership,
group_admin_public_membership
)
expect(scope.resolve(:destroy)).not_to include(
other_user_private_membership,
other_user_public_membership,
group_admin_private_membership,
group_admin_public_membership
)
end
end

Expand All @@ -79,10 +98,30 @@
membership1 = create :membership, user: logged_in_user, user_group: public_user_group
membership2 = create :membership, user: logged_in_user, user_group: private_user_group

expect(scope.resolve(:index)).to include(membership1, membership2)
expect(scope.resolve(:show)).to include(membership1, membership2)
expect(scope.resolve(:update)).to include(membership1, membership2)
expect(scope.resolve(:destroy)).to include(membership1, membership2)
expect(scope.resolve(:index)).to contain_exactly(
membership1,
membership2,
group_admin_private_membership,
group_admin_public_membership
)
expect(scope.resolve(:show)).to contain_exactly(
membership1,
membership2,
group_admin_private_membership,
group_admin_public_membership
)
expect(scope.resolve(:update)).to include(
membership1,
membership2,
group_admin_private_membership,
group_admin_public_membership
)
expect(scope.resolve(:destroy)).to include(
membership1,
membership2,
group_admin_private_membership,
group_admin_public_membership
)
end
end
end
Expand Down

0 comments on commit c350406

Please sign in to comment.