Add admin API for managing canonical e-mail blocks (#19067)
This commit is contained in:
		@@ -0,0 +1,99 @@
 | 
			
		||||
# frozen_string_literal: true
 | 
			
		||||
 | 
			
		||||
class Api::V1::Admin::CanonicalEmailBlocksController < Api::BaseController
 | 
			
		||||
  include Authorization
 | 
			
		||||
  include AccountableConcern
 | 
			
		||||
 | 
			
		||||
  LIMIT = 100
 | 
			
		||||
 | 
			
		||||
  before_action -> { authorize_if_got_token! :'admin:read', :'admin:read:canonical_email_blocks' }, only: [:index, :show, :test]
 | 
			
		||||
  before_action -> { authorize_if_got_token! :'admin:write', :'admin:write:canonical_email_blocks' }, except: [:index, :show, :test]
 | 
			
		||||
 | 
			
		||||
  before_action :set_canonical_email_blocks, only: :index
 | 
			
		||||
  before_action :set_canonical_email_blocks_from_test, only: [:test]
 | 
			
		||||
  before_action :set_canonical_email_block, only: [:show, :destroy]
 | 
			
		||||
 | 
			
		||||
  after_action :verify_authorized
 | 
			
		||||
  after_action :insert_pagination_headers, only: :index
 | 
			
		||||
 | 
			
		||||
  PAGINATION_PARAMS = %i(limit).freeze
 | 
			
		||||
 | 
			
		||||
  def index
 | 
			
		||||
    authorize :canonical_email_block, :index?
 | 
			
		||||
    render json: @canonical_email_blocks, each_serializer: REST::Admin::CanonicalEmailBlockSerializer
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def show
 | 
			
		||||
    authorize @canonical_email_block, :show?
 | 
			
		||||
    render json: @canonical_email_block, serializer: REST::Admin::CanonicalEmailBlockSerializer
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def test
 | 
			
		||||
    authorize :canonical_email_block, :test?
 | 
			
		||||
    render json: @canonical_email_blocks, each_serializer: REST::Admin::CanonicalEmailBlockSerializer
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def create
 | 
			
		||||
    authorize :canonical_email_block, :create?
 | 
			
		||||
 | 
			
		||||
    @canonical_email_block = CanonicalEmailBlock.create!(resource_params)
 | 
			
		||||
    log_action :create, @canonical_email_block
 | 
			
		||||
 | 
			
		||||
    render json: @canonical_email_block, serializer: REST::Admin::CanonicalEmailBlockSerializer
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def destroy
 | 
			
		||||
    authorize @canonical_email_block, :destroy?
 | 
			
		||||
 | 
			
		||||
    @canonical_email_block.destroy!
 | 
			
		||||
    log_action :destroy, @canonical_email_block
 | 
			
		||||
 | 
			
		||||
    render json: @canonical_email_block, serializer: REST::Admin::CanonicalEmailBlockSerializer
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def resource_params
 | 
			
		||||
    params.permit(:canonical_email_hash, :email)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_canonical_email_blocks
 | 
			
		||||
    @canonical_email_blocks = CanonicalEmailBlock.order(id: :desc).to_a_paginated_by_id(limit_param(LIMIT), params_slice(:max_id, :since_id, :min_id))
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_canonical_email_blocks_from_test
 | 
			
		||||
    @canonical_email_blocks = CanonicalEmailBlock.matching_email(params[:email])
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def set_canonical_email_block
 | 
			
		||||
    @canonical_email_block = CanonicalEmailBlock.find(params[:id])
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def insert_pagination_headers
 | 
			
		||||
    set_pagination_headers(next_path, prev_path)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def next_path
 | 
			
		||||
    api_v1_admin_canonical_email_blocks_url(pagination_params(max_id: pagination_max_id)) if records_continue?
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def prev_path
 | 
			
		||||
    api_v1_admin_canonical_email_blocks_url(pagination_params(min_id: pagination_since_id)) unless @canonical_email_blocks.empty?
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def pagination_max_id
 | 
			
		||||
    @canonical_email_blocks.last.id
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def pagination_since_id
 | 
			
		||||
    @canonical_email_blocks.first.id
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def records_continue?
 | 
			
		||||
    @canonical_email_blocks.size == limit_param(LIMIT)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def pagination_params(core_params)
 | 
			
		||||
    params.slice(*PAGINATION_PARAMS).permit(*PAGINATION_PARAMS).merge(core_params)
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -9,8 +9,6 @@ module Admin::ActionLogsHelper
 | 
			
		||||
      link_to log.human_identifier, admin_account_path(log.route_param)
 | 
			
		||||
    when 'UserRole'
 | 
			
		||||
      link_to log.human_identifier, admin_roles_path(log.target_id)
 | 
			
		||||
    when 'CustomEmoji'
 | 
			
		||||
      log.human_identifier
 | 
			
		||||
    when 'Report'
 | 
			
		||||
      link_to "##{log.human_identifier}", admin_report_path(log.target_id)
 | 
			
		||||
    when 'DomainBlock', 'DomainAllow', 'EmailDomainBlock', 'UnavailableDomain'
 | 
			
		||||
@@ -21,10 +19,10 @@ module Admin::ActionLogsHelper
 | 
			
		||||
      link_to log.human_identifier, admin_account_path(log.target_id)
 | 
			
		||||
    when 'Announcement'
 | 
			
		||||
      link_to truncate(log.human_identifier), edit_admin_announcement_path(log.target_id)
 | 
			
		||||
    when 'IpBlock'
 | 
			
		||||
      log.human_identifier
 | 
			
		||||
    when 'Instance'
 | 
			
		||||
    when 'IpBlock', 'Instance', 'CustomEmoji'
 | 
			
		||||
      log.human_identifier
 | 
			
		||||
    when 'CanonicalEmailBlock'
 | 
			
		||||
      content_tag(:samp, log.human_identifier[0...7], title: log.human_identifier)
 | 
			
		||||
    when 'Appeal'
 | 
			
		||||
      link_to log.human_identifier, disputes_strike_path(log.route_param)
 | 
			
		||||
    end
 | 
			
		||||
 
 | 
			
		||||
@@ -22,18 +22,22 @@ class Admin::ActionLogFilter
 | 
			
		||||
    create_domain_allow: { target_type: 'DomainAllow', action: 'create' }.freeze,
 | 
			
		||||
    create_domain_block: { target_type: 'DomainBlock', action: 'create' }.freeze,
 | 
			
		||||
    create_email_domain_block: { target_type: 'EmailDomainBlock', action: 'create' }.freeze,
 | 
			
		||||
    create_ip_block: { target_type: 'IpBlock', action: 'create' }.freeze,
 | 
			
		||||
    create_unavailable_domain: { target_type: 'UnavailableDomain', action: 'create' }.freeze,
 | 
			
		||||
    create_user_role: { target_type: 'UserRole', action: 'create' }.freeze,
 | 
			
		||||
    create_canonical_email_block: { target_type: 'CanonicalEmailBlock', action: 'create' }.freeze,
 | 
			
		||||
    demote_user: { target_type: 'User', action: 'demote' }.freeze,
 | 
			
		||||
    destroy_announcement: { target_type: 'Announcement', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_custom_emoji: { target_type: 'CustomEmoji', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_domain_allow: { target_type: 'DomainAllow', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_domain_block: { target_type: 'DomainBlock', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_ip_block: { target_type: 'IpBlock', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_email_domain_block: { target_type: 'EmailDomainBlock', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_instance: { target_type: 'Instance', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_unavailable_domain: { target_type: 'UnavailableDomain', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_status: { target_type: 'Status', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_user_role: { target_type: 'UserRole', action: 'destroy' }.freeze,
 | 
			
		||||
    destroy_canonical_email_block: { target_type: 'CanonicalEmailBlock', action: 'destroy' }.freeze,
 | 
			
		||||
    disable_2fa_user: { target_type: 'User', action: 'disable' }.freeze,
 | 
			
		||||
    disable_custom_emoji: { target_type: 'CustomEmoji', action: 'disable' }.freeze,
 | 
			
		||||
    disable_user: { target_type: 'User', action: 'disable' }.freeze,
 | 
			
		||||
@@ -56,6 +60,7 @@ class Admin::ActionLogFilter
 | 
			
		||||
    update_custom_emoji: { target_type: 'CustomEmoji', action: 'update' }.freeze,
 | 
			
		||||
    update_status: { target_type: 'Status', action: 'update' }.freeze,
 | 
			
		||||
    update_user_role: { target_type: 'UserRole', action: 'update' }.freeze,
 | 
			
		||||
    update_ip_block: { target_type: 'IpBlock', action: 'update' }.freeze,
 | 
			
		||||
    unblock_email_account: { target_type: 'Account', action: 'unblock_email' }.freeze,
 | 
			
		||||
  }.freeze
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -5,27 +5,30 @@
 | 
			
		||||
#
 | 
			
		||||
#  id                   :bigint(8)        not null, primary key
 | 
			
		||||
#  canonical_email_hash :string           default(""), not null
 | 
			
		||||
#  reference_account_id :bigint(8)        not null
 | 
			
		||||
#  reference_account_id :bigint(8)
 | 
			
		||||
#  created_at           :datetime         not null
 | 
			
		||||
#  updated_at           :datetime         not null
 | 
			
		||||
#
 | 
			
		||||
 | 
			
		||||
class CanonicalEmailBlock < ApplicationRecord
 | 
			
		||||
  include EmailHelper
 | 
			
		||||
  include Paginable
 | 
			
		||||
 | 
			
		||||
  belongs_to :reference_account, class_name: 'Account'
 | 
			
		||||
  belongs_to :reference_account, class_name: 'Account', optional: true
 | 
			
		||||
 | 
			
		||||
  validates :canonical_email_hash, presence: true, uniqueness: true
 | 
			
		||||
 | 
			
		||||
  scope :matching_email, ->(email) { where(canonical_email_hash: email_to_canonical_email_hash(email)) }
 | 
			
		||||
 | 
			
		||||
  def to_log_human_identifier
 | 
			
		||||
    canonical_email_hash
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def email=(email)
 | 
			
		||||
    self.canonical_email_hash = email_to_canonical_email_hash(email)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def self.block?(email)
 | 
			
		||||
    where(canonical_email_hash: email_to_canonical_email_hash(email)).exists?
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def self.find_blocks(email)
 | 
			
		||||
    where(canonical_email_hash: email_to_canonical_email_hash(email))
 | 
			
		||||
    matching_email(email).exists?
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
 
 | 
			
		||||
							
								
								
									
										23
									
								
								app/policies/canonical_email_block_policy.rb
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										23
									
								
								app/policies/canonical_email_block_policy.rb
									
									
									
									
									
										Normal file
									
								
							@@ -0,0 +1,23 @@
 | 
			
		||||
# frozen_string_literal: true
 | 
			
		||||
 | 
			
		||||
class CanonicalEmailBlockPolicy < ApplicationPolicy
 | 
			
		||||
  def index?
 | 
			
		||||
    role.can?(:manage_blocks)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def show?
 | 
			
		||||
    role.can?(:manage_blocks)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def test?
 | 
			
		||||
    role.can?(:manage_blocks)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def create?
 | 
			
		||||
    role.can?(:manage_blocks)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def destroy?
 | 
			
		||||
    role.can?(:manage_blocks)
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -0,0 +1,9 @@
 | 
			
		||||
# frozen_string_literal: true
 | 
			
		||||
 | 
			
		||||
class REST::Admin::CanonicalEmailBlockSerializer < ActiveModel::Serializer
 | 
			
		||||
  attributes :id, :canonical_email_hash
 | 
			
		||||
 | 
			
		||||
  def id
 | 
			
		||||
    object.id.to_s
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -239,6 +239,7 @@ en:
 | 
			
		||||
        confirm_user: Confirm User
 | 
			
		||||
        create_account_warning: Create Warning
 | 
			
		||||
        create_announcement: Create Announcement
 | 
			
		||||
        create_canonical_email_block: Create E-mail Block
 | 
			
		||||
        create_custom_emoji: Create Custom Emoji
 | 
			
		||||
        create_domain_allow: Create Domain Allow
 | 
			
		||||
        create_domain_block: Create Domain Block
 | 
			
		||||
@@ -248,6 +249,7 @@ en:
 | 
			
		||||
        create_user_role: Create Role
 | 
			
		||||
        demote_user: Demote User
 | 
			
		||||
        destroy_announcement: Delete Announcement
 | 
			
		||||
        destroy_canonical_email_block: Delete E-mail Block
 | 
			
		||||
        destroy_custom_emoji: Delete Custom Emoji
 | 
			
		||||
        destroy_domain_allow: Delete Domain Allow
 | 
			
		||||
        destroy_domain_block: Delete Domain Block
 | 
			
		||||
@@ -283,6 +285,7 @@ en:
 | 
			
		||||
        update_announcement: Update Announcement
 | 
			
		||||
        update_custom_emoji: Update Custom Emoji
 | 
			
		||||
        update_domain_block: Update Domain Block
 | 
			
		||||
        update_ip_block: Update IP rule
 | 
			
		||||
        update_status: Update Post
 | 
			
		||||
        update_user_role: Update Role
 | 
			
		||||
      actions:
 | 
			
		||||
@@ -294,6 +297,7 @@ en:
 | 
			
		||||
        confirm_user_html: "%{name} confirmed e-mail address of user %{target}"
 | 
			
		||||
        create_account_warning_html: "%{name} sent a warning to %{target}"
 | 
			
		||||
        create_announcement_html: "%{name} created new announcement %{target}"
 | 
			
		||||
        create_canonical_email_block_html: "%{name} blocked e-mail with the hash %{target}"
 | 
			
		||||
        create_custom_emoji_html: "%{name} uploaded new emoji %{target}"
 | 
			
		||||
        create_domain_allow_html: "%{name} allowed federation with domain %{target}"
 | 
			
		||||
        create_domain_block_html: "%{name} blocked domain %{target}"
 | 
			
		||||
@@ -303,6 +307,7 @@ en:
 | 
			
		||||
        create_user_role_html: "%{name} created %{target} role"
 | 
			
		||||
        demote_user_html: "%{name} demoted user %{target}"
 | 
			
		||||
        destroy_announcement_html: "%{name} deleted announcement %{target}"
 | 
			
		||||
        destroy_canonical_email_block_html: "%{name} unblocked e-mail with the hash %{target}"
 | 
			
		||||
        destroy_custom_emoji_html: "%{name} deleted emoji %{target}"
 | 
			
		||||
        destroy_domain_allow_html: "%{name} disallowed federation with domain %{target}"
 | 
			
		||||
        destroy_domain_block_html: "%{name} unblocked domain %{target}"
 | 
			
		||||
@@ -338,6 +343,7 @@ en:
 | 
			
		||||
        update_announcement_html: "%{name} updated announcement %{target}"
 | 
			
		||||
        update_custom_emoji_html: "%{name} updated emoji %{target}"
 | 
			
		||||
        update_domain_block_html: "%{name} updated domain block for %{target}"
 | 
			
		||||
        update_ip_block_html: "%{name} changed rule for IP %{target}"
 | 
			
		||||
        update_status_html: "%{name} updated post by %{target}"
 | 
			
		||||
        update_user_role_html: "%{name} changed %{target} role"
 | 
			
		||||
      empty: No logs found.
 | 
			
		||||
 
 | 
			
		||||
@@ -602,6 +602,12 @@ Rails.application.routes.draw do
 | 
			
		||||
        post :measures, to: 'measures#create'
 | 
			
		||||
        post :dimensions, to: 'dimensions#create'
 | 
			
		||||
        post :retention, to: 'retention#create'
 | 
			
		||||
 | 
			
		||||
        resources :canonical_email_blocks, only: [:index, :create, :show, :destroy] do
 | 
			
		||||
          collection do
 | 
			
		||||
            post :test
 | 
			
		||||
          end
 | 
			
		||||
        end
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
 
 | 
			
		||||
@@ -0,0 +1,5 @@
 | 
			
		||||
class ChangeCanonicalEmailBlocksNullable < ActiveRecord::Migration[6.1]
 | 
			
		||||
  def change
 | 
			
		||||
    safety_assured { change_column :canonical_email_blocks, :reference_account_id, :bigint, null: true, default: nil }
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
@@ -10,7 +10,7 @@
 | 
			
		||||
#
 | 
			
		||||
# It's strongly recommended that you check this file into your version control system.
 | 
			
		||||
 | 
			
		||||
ActiveRecord::Schema.define(version: 2022_08_24_164532) do
 | 
			
		||||
ActiveRecord::Schema.define(version: 2022_08_27_195229) do
 | 
			
		||||
 | 
			
		||||
  # These are extensions that must be enabled in order to support this database
 | 
			
		||||
  enable_extension "plpgsql"
 | 
			
		||||
@@ -296,7 +296,7 @@ ActiveRecord::Schema.define(version: 2022_08_24_164532) do
 | 
			
		||||
 | 
			
		||||
  create_table "canonical_email_blocks", force: :cascade do |t|
 | 
			
		||||
    t.string "canonical_email_hash", default: "", null: false
 | 
			
		||||
    t.bigint "reference_account_id", null: false
 | 
			
		||||
    t.bigint "reference_account_id"
 | 
			
		||||
    t.datetime "created_at", precision: 6, null: false
 | 
			
		||||
    t.datetime "updated_at", precision: 6, null: false
 | 
			
		||||
    t.index ["canonical_email_hash"], name: "index_canonical_email_blocks_on_canonical_email_hash", unique: true
 | 
			
		||||
 
 | 
			
		||||
@@ -18,17 +18,15 @@ module Mastodon
 | 
			
		||||
      When suspending a local user, a hash of a "canonical" version of their e-mail
 | 
			
		||||
      address is stored to prevent them from signing up again.
 | 
			
		||||
 | 
			
		||||
      This command can be used to find whether a known email address is blocked,
 | 
			
		||||
      and if so, which account it was attached to.
 | 
			
		||||
      This command can be used to find whether a known email address is blocked.
 | 
			
		||||
    LONG_DESC
 | 
			
		||||
    def find(email)
 | 
			
		||||
      accts = CanonicalEmailBlock.find_blocks(email).map(&:reference_account).map(&:acct).to_a
 | 
			
		||||
      accts = CanonicalEmailBlock.matching_email(email)
 | 
			
		||||
 | 
			
		||||
      if accts.empty?
 | 
			
		||||
        say("#{email} is not blocked", :yellow)
 | 
			
		||||
        say("#{email} is not blocked", :green)
 | 
			
		||||
      else
 | 
			
		||||
        accts.each do |acct|
 | 
			
		||||
          say(acct, :white)
 | 
			
		||||
        end
 | 
			
		||||
        say("#{email} is blocked", :red)
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
@@ -40,24 +38,13 @@ module Mastodon
 | 
			
		||||
      This command allows removing a canonical email block.
 | 
			
		||||
    LONG_DESC
 | 
			
		||||
    def remove(email)
 | 
			
		||||
      blocks = CanonicalEmailBlock.find_blocks(email)
 | 
			
		||||
      blocks = CanonicalEmailBlock.matching_email(email)
 | 
			
		||||
 | 
			
		||||
      if blocks.empty?
 | 
			
		||||
        say("#{email} is not blocked", :yellow)
 | 
			
		||||
        say("#{email} is not blocked", :green)
 | 
			
		||||
      else
 | 
			
		||||
        blocks.destroy_all
 | 
			
		||||
        say("Removed canonical email block for #{email}", :green)
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    private
 | 
			
		||||
 | 
			
		||||
    def color(processed, failed)
 | 
			
		||||
      if !processed.zero? && failed.zero?
 | 
			
		||||
        :green
 | 
			
		||||
      elsif failed.zero?
 | 
			
		||||
        :yellow
 | 
			
		||||
      else
 | 
			
		||||
        :red
 | 
			
		||||
        say("Unblocked #{email}", :green)
 | 
			
		||||
      end
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user