Add more instance stats APIs (#6125)
* Add GET /api/v1/instance/peers API to reveal known domains * Add GET /api/v1/instance/activity API * Make new APIs disableable, exclude private statuses from activity stats * Fix code style issue * Fix week timestamps
This commit is contained in:
		
							parent
							
								
									511c6f9625
								
							
						
					
					
						commit
						38fc1b498d
					
				
					 14 changed files with 144 additions and 8 deletions
				
			
		| 
						 | 
				
			
			@ -17,6 +17,8 @@ module Admin
 | 
			
		|||
      bootstrap_timeline_accounts
 | 
			
		||||
      thumbnail
 | 
			
		||||
      min_invite_role
 | 
			
		||||
      activity_api_enabled
 | 
			
		||||
      peers_api_enabled
 | 
			
		||||
    ).freeze
 | 
			
		||||
 | 
			
		||||
    BOOLEAN_SETTINGS = %w(
 | 
			
		||||
| 
						 | 
				
			
			@ -24,6 +26,8 @@ module Admin
 | 
			
		|||
      open_deletion
 | 
			
		||||
      timeline_preview
 | 
			
		||||
      show_staff_badge
 | 
			
		||||
      activity_api_enabled
 | 
			
		||||
      peers_api_enabled
 | 
			
		||||
    ).freeze
 | 
			
		||||
 | 
			
		||||
    UPLOAD_SETTINGS = %w(
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										36
									
								
								app/controllers/api/v1/instances/activity_controller.rb
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								app/controllers/api/v1/instances/activity_controller.rb
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,36 @@
 | 
			
		|||
# frozen_string_literal: true
 | 
			
		||||
 | 
			
		||||
class Api::V1::Instances::ActivityController < Api::BaseController
 | 
			
		||||
  before_action :require_enabled_api!
 | 
			
		||||
 | 
			
		||||
  respond_to :json
 | 
			
		||||
 | 
			
		||||
  def show
 | 
			
		||||
    render_cached_json('api:v1:instances:activity:show', expires_in: 1.day) { activity }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def activity
 | 
			
		||||
    weeks = []
 | 
			
		||||
 | 
			
		||||
    12.times do |i|
 | 
			
		||||
      day     = i.weeks.ago.to_date
 | 
			
		||||
      week_id = day.cweek
 | 
			
		||||
      week    = Date.commercial(day.cwyear, week_id)
 | 
			
		||||
 | 
			
		||||
      weeks << {
 | 
			
		||||
        week: week.to_time.to_i.to_s,
 | 
			
		||||
        statuses: Redis.current.get("activity:statuses:local:#{week_id}") || 0,
 | 
			
		||||
        logins: Redis.current.pfcount("activity:logins:#{week_id}"),
 | 
			
		||||
        registrations: Redis.current.get("activity:accounts:local:#{week_id}") || 0,
 | 
			
		||||
      }
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    weeks
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def require_enabled_api!
 | 
			
		||||
    head 404 unless Setting.activity_api_enabled
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
							
								
								
									
										17
									
								
								app/controllers/api/v1/instances/peers_controller.rb
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								app/controllers/api/v1/instances/peers_controller.rb
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,17 @@
 | 
			
		|||
# frozen_string_literal: true
 | 
			
		||||
 | 
			
		||||
class Api::V1::Instances::PeersController < Api::BaseController
 | 
			
		||||
  before_action :require_enabled_api!
 | 
			
		||||
 | 
			
		||||
  respond_to :json
 | 
			
		||||
 | 
			
		||||
  def index
 | 
			
		||||
    render_cached_json('api:v1:instances:peers:index', expires_in: 1.day) { Account.remote.domains }
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  private
 | 
			
		||||
 | 
			
		||||
  def require_enabled_api!
 | 
			
		||||
    head 404 unless Setting.peers_api_enabled
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
| 
						 | 
				
			
			@ -121,4 +121,13 @@ class ApplicationController < ActionController::Base
 | 
			
		|||
      end
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def render_cached_json(cache_key, **options)
 | 
			
		||||
    data = Rails.cache.fetch(cache_key, { raw: true }.merge(options)) do
 | 
			
		||||
      yield.to_json
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    expires_in options[:expires_in], public: true
 | 
			
		||||
    render json: data
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -2,10 +2,4 @@
 | 
			
		|||
 | 
			
		||||
class Auth::ConfirmationsController < Devise::ConfirmationsController
 | 
			
		||||
  layout 'auth'
 | 
			
		||||
 | 
			
		||||
  def show
 | 
			
		||||
    super do |user|
 | 
			
		||||
      BootstrapTimelineWorker.perform_async(user.account_id) if user.errors.empty?
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -17,6 +17,7 @@ module UserTrackingConcern
 | 
			
		|||
 | 
			
		||||
    # Mark as signed-in today
 | 
			
		||||
    current_user.update_tracked_fields!(request)
 | 
			
		||||
    ActivityTracker.record('activity:logins', current_user.id)
 | 
			
		||||
 | 
			
		||||
    # Regenerate feed if needed
 | 
			
		||||
    regenerate_feed! if user_needs_feed_update?
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										31
									
								
								app/lib/activity_tracker.rb
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								app/lib/activity_tracker.rb
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,31 @@
 | 
			
		|||
# frozen_string_literal: true
 | 
			
		||||
 | 
			
		||||
class ActivityTracker
 | 
			
		||||
  EXPIRE_AFTER = 90.days.seconds
 | 
			
		||||
 | 
			
		||||
  class << self
 | 
			
		||||
    def increment(prefix)
 | 
			
		||||
      key = [prefix, current_week].join(':')
 | 
			
		||||
 | 
			
		||||
      redis.incrby(key, 1)
 | 
			
		||||
      redis.expire(key, EXPIRE_AFTER)
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def record(prefix, value)
 | 
			
		||||
      key = [prefix, current_week].join(':')
 | 
			
		||||
 | 
			
		||||
      redis.pfadd(key, value)
 | 
			
		||||
      redis.expire(key, value)
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    private
 | 
			
		||||
 | 
			
		||||
    def redis
 | 
			
		||||
      Redis.current
 | 
			
		||||
    end
 | 
			
		||||
 | 
			
		||||
    def current_week
 | 
			
		||||
      Time.zone.today.cweek
 | 
			
		||||
    end
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
| 
						 | 
				
			
			@ -30,6 +30,10 @@ class Form::AdminSettings
 | 
			
		|||
    :bootstrap_timeline_accounts=,
 | 
			
		||||
    :min_invite_role,
 | 
			
		||||
    :min_invite_role=,
 | 
			
		||||
    :activity_api_enabled,
 | 
			
		||||
    :activity_api_enabled=,
 | 
			
		||||
    :peers_api_enabled,
 | 
			
		||||
    :peers_api_enabled=,
 | 
			
		||||
    to: Setting
 | 
			
		||||
  )
 | 
			
		||||
end
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -135,6 +135,7 @@ class Status < ApplicationRecord
 | 
			
		|||
  end
 | 
			
		||||
 | 
			
		||||
  after_create_commit :store_uri, if: :local?
 | 
			
		||||
  after_create_commit :update_statistics, if: :local?
 | 
			
		||||
 | 
			
		||||
  around_create Mastodon::Snowflake::Callbacks
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -308,4 +309,9 @@ class Status < ApplicationRecord
 | 
			
		|||
  def set_local
 | 
			
		||||
    self.local = account.local?
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def update_statistics
 | 
			
		||||
    return unless public_visibility? || unlisted_visibility?
 | 
			
		||||
    ActivityTracker.increment('activity:statuses:local')
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -122,9 +122,19 @@ class User < ApplicationRecord
 | 
			
		|||
    update!(disabled: false)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def confirm
 | 
			
		||||
    return if confirmed?
 | 
			
		||||
 | 
			
		||||
    super
 | 
			
		||||
    update_statistics!
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def confirm!
 | 
			
		||||
    return if confirmed?
 | 
			
		||||
 | 
			
		||||
    skip_confirmation!
 | 
			
		||||
    save!
 | 
			
		||||
    update_statistics!
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def promote!
 | 
			
		||||
| 
						 | 
				
			
			@ -202,4 +212,9 @@ class User < ApplicationRecord
 | 
			
		|||
  def sanitize_languages
 | 
			
		||||
    filtered_languages.reject!(&:blank?)
 | 
			
		||||
  end
 | 
			
		||||
 | 
			
		||||
  def update_statistics!
 | 
			
		||||
    BootstrapTimelineWorker.perform_async(account_id)
 | 
			
		||||
    ActivityTracker.increment('activity:accounts:local')
 | 
			
		||||
  end
 | 
			
		||||
end
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -46,5 +46,13 @@
 | 
			
		|||
  .fields-group
 | 
			
		||||
    = f.input :bootstrap_timeline_accounts, wrapper: :with_block_label, label: t('admin.settings.bootstrap_timeline_accounts.title'), hint: t('admin.settings.bootstrap_timeline_accounts.desc_html')
 | 
			
		||||
 | 
			
		||||
  %hr/
 | 
			
		||||
 | 
			
		||||
  .fields-group
 | 
			
		||||
    = f.input :activity_api_enabled, as: :boolean, wrapper: :with_label, label: t('admin.settings.activity_api_enabled.title'), hint: t('admin.settings.activity_api_enabled.desc_html')
 | 
			
		||||
 | 
			
		||||
  .fields-group
 | 
			
		||||
    = f.input :peers_api_enabled, as: :boolean, wrapper: :with_label, label: t('admin.settings.peers_api_enabled.title'), hint: t('admin.settings.peers_api_enabled.desc_html')
 | 
			
		||||
 | 
			
		||||
  .actions
 | 
			
		||||
    = f.button :button, t('generic.save_changes'), type: :submit
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -265,12 +265,18 @@ en:
 | 
			
		|||
      unresolved: Unresolved
 | 
			
		||||
      view: View
 | 
			
		||||
    settings:
 | 
			
		||||
      activity_api_enabled:
 | 
			
		||||
        desc_html: Counts of locally posted statuses, active users, and new registrations in weekly buckets
 | 
			
		||||
        title: Publish aggregate statistics about user activity
 | 
			
		||||
      bootstrap_timeline_accounts:
 | 
			
		||||
        desc_html: Separate multiple usernames by comma. Only local and unlocked accounts will work. Default when empty is all local admins.
 | 
			
		||||
        title: Default follows for new users
 | 
			
		||||
      contact_information:
 | 
			
		||||
        email: Business e-mail
 | 
			
		||||
        username: Contact username
 | 
			
		||||
      peers_api_enabled:
 | 
			
		||||
        desc_html: Domain names this instance has encountered in the fediverse
 | 
			
		||||
        title: Publish list of discovered instances
 | 
			
		||||
      registrations:
 | 
			
		||||
        closed_message:
 | 
			
		||||
          desc_html: Displayed on frontpage when registrations are closed. You can use HTML tags
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -241,7 +241,11 @@ Rails.application.routes.draw do
 | 
			
		|||
 | 
			
		||||
      resources :apps, only: [:create]
 | 
			
		||||
 | 
			
		||||
      resource :instance,      only: [:show]
 | 
			
		||||
      resource :instance, only: [:show] do
 | 
			
		||||
        resources :peers, only: [:index], controller: 'instances/peers'
 | 
			
		||||
        resource :activity, only: [:show], controller: 'instances/activity'
 | 
			
		||||
      end
 | 
			
		||||
 | 
			
		||||
      resource :domain_blocks, only: [:show, :create, :destroy]
 | 
			
		||||
 | 
			
		||||
      resources :follow_requests, only: [:index] do
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -47,7 +47,8 @@ defaults: &defaults
 | 
			
		|||
    - webmaster
 | 
			
		||||
    - administrator
 | 
			
		||||
  bootstrap_timeline_accounts: ''
 | 
			
		||||
 | 
			
		||||
  activity_api_enabled: true
 | 
			
		||||
  peers_api_enabled: true
 | 
			
		||||
development:
 | 
			
		||||
  <<: *defaults
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
		Reference in a new issue