Commit 392b90cd authored by Alex Buijs's avatar Alex Buijs

Offload tracking onboarding progress to worker

In order to prevent writing to the database inside
an endpoint
parent 6ea26ab1
...@@ -78,11 +78,11 @@ module Repositories ...@@ -78,11 +78,11 @@ module Repositories
def update_fetch_statistics def update_fetch_statistics
return unless project return unless project
return if Gitlab::Database.read_only? return if Gitlab::Database.read_only?
return if Feature.enabled?(:disable_git_http_fetch_writes)
return unless repo_type.project? return unless repo_type.project?
OnboardingProgressService.new(project.namespace).execute(action: :git_read) OnboardingProgressService.async(project.namespace_id).execute(action: :git_pull)
return if Feature.enabled?(:disable_git_http_fetch_writes)
if Feature.enabled?(:project_statistics_sync, project, default_enabled: true) if Feature.enabled?(:project_statistics_sync, project, default_enabled: true)
Projects::FetchStatisticsIncrementService.new(project).execute Projects::FetchStatisticsIncrementService.new(project).execute
......
...@@ -66,6 +66,13 @@ class OnboardingProgress < ApplicationRecord ...@@ -66,6 +66,13 @@ class OnboardingProgress < ApplicationRecord
where(namespace: namespace).where.not(action_column => nil).exists? where(namespace: namespace).where.not(action_column => nil).exists?
end end
def not_completed?(namespace_id, action)
return unless ACTIONS.include?(action)
action_column = column_name(action)
where(namespace_id: namespace_id).where(action_column => nil).exists?
end
def column_name(action) def column_name(action)
:"#{action}_at" :"#{action}_at"
end end
......
# frozen_string_literal: true # frozen_string_literal: true
class OnboardingProgressService class OnboardingProgressService
class Async
attr_reader :namespace_id
def initialize(namespace_id)
@namespace_id = namespace_id
end
def execute(action:)
return unless OnboardingProgress.not_completed?(namespace_id, action)
Namespaces::OnboardingProgressWorker.perform_async(namespace_id, action)
end
end
def self.async(namespace_id)
Async.new(namespace_id)
end
def initialize(namespace) def initialize(namespace)
@namespace = namespace&.root_ancestor @namespace = namespace&.root_ancestor
end end
......
...@@ -1840,6 +1840,14 @@ ...@@ -1840,6 +1840,14 @@
:weight: 1 :weight: 1
:idempotent: true :idempotent: true
:tags: [] :tags: []
- :name: namespaces_onboarding_progress
:feature_category: :product_analytics
:has_external_dependencies:
:urgency: :low
:resource_boundary: :unknown
:weight: 1
:idempotent: true
:tags: []
- :name: namespaces_onboarding_user_added - :name: namespaces_onboarding_user_added
:feature_category: :users :feature_category: :users
:has_external_dependencies: :has_external_dependencies:
......
# frozen_string_literal: true
module Namespaces
class OnboardingProgressWorker
include ApplicationWorker
feature_category :product_analytics
urgency :low
deduplicate :until_executed
idempotent!
def perform(namespace_id, action)
namespace = Namespace.find_by_id(namespace_id)
return unless namespace && action
OnboardingProgressService.new(namespace).execute(action: action.to_sym)
end
end
end
...@@ -212,6 +212,8 @@ ...@@ -212,6 +212,8 @@
- 1 - 1
- - namespaces_onboarding_pipeline_created - - namespaces_onboarding_pipeline_created
- 1 - 1
- - namespaces_onboarding_progress
- 1
- - namespaces_onboarding_user_added - - namespaces_onboarding_user_added
- 1 - 1
- - new_epic - - new_epic
......
...@@ -54,14 +54,17 @@ RSpec.describe Repositories::GitHttpController do ...@@ -54,14 +54,17 @@ RSpec.describe Repositories::GitHttpController do
}.from(0).to(1) }.from(0).to(1)
end end
it_behaves_like 'records an onboarding progress action', :git_read do describe 'recording the onboarding progress', :sidekiq_inline do
let(:namespace) { project.namespace } let_it_be(:namespace) { project.namespace }
subject { send_request }
before do before do
stub_feature_flags(disable_git_http_fetch_writes: false) OnboardingProgress.onboard(namespace)
send_request
end end
subject { OnboardingProgress.completed?(namespace, :git_pull) }
it { is_expected.to be(true) }
end end
context 'when disable_git_http_fetch_writes is enabled' do context 'when disable_git_http_fetch_writes is enabled' do
...@@ -75,12 +78,6 @@ RSpec.describe Repositories::GitHttpController do ...@@ -75,12 +78,6 @@ RSpec.describe Repositories::GitHttpController do
send_request send_request
end end
it 'does not record onboarding progress' do
expect(OnboardingProgressService).not_to receive(:new)
send_request
end
end end
end end
end end
......
...@@ -182,6 +182,30 @@ RSpec.describe OnboardingProgress do ...@@ -182,6 +182,30 @@ RSpec.describe OnboardingProgress do
end end
end end
describe '.not_completed?' do
subject { described_class.not_completed?(namespace.id, action) }
context 'when the namespace has not yet been onboarded' do
it { is_expected.to be(false) }
end
context 'when the namespace has been onboarded but not registered the action yet' do
before do
described_class.onboard(namespace)
end
it { is_expected.to be(true) }
context 'when the action has been registered' do
before do
described_class.register(namespace, action)
end
it { is_expected.to be(false) }
end
end
end
describe '.column_name' do describe '.column_name' do
subject { described_class.column_name(action) } subject { described_class.column_name(action) }
......
...@@ -3,6 +3,47 @@ ...@@ -3,6 +3,47 @@
require 'spec_helper' require 'spec_helper'
RSpec.describe OnboardingProgressService do RSpec.describe OnboardingProgressService do
describe '.async' do
let_it_be(:namespace) { create(:namespace) }
let_it_be(:action) { :git_pull }
subject(:execute_service) { described_class.async(namespace.id).execute(action: action) }
context 'when not onboarded' do
it 'does not schedule a worker' do
expect(Namespaces::OnboardingProgressWorker).not_to receive(:perform_async)
execute_service
end
end
context 'when onboarded' do
before do
OnboardingProgress.onboard(namespace)
end
context 'when action is already completed' do
before do
OnboardingProgress.register(namespace, action)
end
it 'does not schedule a worker' do
expect(Namespaces::OnboardingProgressWorker).not_to receive(:perform_async)
execute_service
end
end
context 'when action is not yet completed' do
it 'schedules a worker' do
expect(Namespaces::OnboardingProgressWorker).to receive(:perform_async)
execute_service
end
end
end
end
describe '#execute' do describe '#execute' do
let(:namespace) { create(:namespace, parent: root_namespace) } let(:namespace) { create(:namespace, parent: root_namespace) }
let(:root_namespace) { nil } let(:root_namespace) { nil }
......
# frozen_string_literal: true
require 'spec_helper'
RSpec.describe Namespaces::OnboardingProgressWorker, '#perform' do
let_it_be(:namespace) { create(:namespace) }
let_it_be(:action) { 'git_pull' }
it_behaves_like 'records an onboarding progress action', :git_pull do
include_examples 'an idempotent worker' do
subject { described_class.new.perform(namespace.id, action) }
end
end
it_behaves_like 'does not record an onboarding progress action' do
subject { described_class.new.perform(namespace.id, nil) }
end
it_behaves_like 'does not record an onboarding progress action' do
subject { described_class.new.perform(nil, action) }
end
end
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment