Commit f2efbc6d authored by Gabriel Mazetto's avatar Gabriel Mazetto

Improve queries, refactor view and controller, update locale and specs

parent bf27639b
...@@ -221,6 +221,7 @@ ...@@ -221,6 +221,7 @@
color: $gl-text-red; color: $gl-text-red;
} }
} }
@include media-breakpoint-down(xs) { @include media-breakpoint-down(xs) {
.project-status-container + .project-status-container { .project-status-container + .project-status-container {
margin-top: 10px; margin-top: 10px;
......
...@@ -9,20 +9,16 @@ class Admin::GeoProjectsController < Admin::ApplicationController ...@@ -9,20 +9,16 @@ class Admin::GeoProjectsController < Admin::ApplicationController
def index def index
finder = ::Geo::ProjectRegistryStatusFinder.new finder = ::Geo::ProjectRegistryStatusFinder.new
case params[:sync_status] @registries = case params[:sync_status]
when 'never' when 'never'
# This method uses FDW heavily and due to optimizations we need to inject the pagination finder.never_synced_projects.page(params[:page])
# earlier as well, so we need the block to do that when 'failed'
@projects = finder.never_synced_projects do |fdw_relation| finder.failed_projects.page(params[:page])
fdw_relation.page(params[:page]) when 'pending'
end.page(params[:page]) finder.pending_projects.page(params[:page])
when 'failed' else
@registries = finder.failed_projects.page(params[:page]) finder.synced_projects.page(params[:page])
when 'pending' end
@registries = finder.pending_projects.page(params[:page])
else
@registries = finder.synced_projects.page(params[:page])
end
end end
def recheck def recheck
......
...@@ -20,7 +20,7 @@ module Geo ...@@ -20,7 +20,7 @@ module Geo
no_repository_resync no_repository_resync
.and(no_repository_sync_failure) .and(no_repository_sync_failure)
.and(repository_verified) .and(repository_verified)
).includes(:project) ).includes(project: :route)
end end
# Return any project registry which project is pending to update # Return any project registry which project is pending to update
...@@ -40,7 +40,7 @@ module Geo ...@@ -40,7 +40,7 @@ module Geo
.and(flagged_for_resync .and(flagged_for_resync
.or(repository_pending_verification .or(repository_pending_verification
.and(repository_without_verification_failure_before))) .and(repository_without_verification_failure_before)))
).includes(:project) ).includes(project: :route)
end end
# Return any project registry which project has a failure # Return any project registry which project has a failure
...@@ -55,31 +55,15 @@ module Geo ...@@ -55,31 +55,15 @@ module Geo
repository_sync_failed repository_sync_failed
.or(repository_verification_failed) .or(repository_verification_failed)
.or(repository_checksum_mismatch) .or(repository_checksum_mismatch)
).includes(:project) ).includes(project: :route)
end end
# Return projects that has never been fully synced # Return any project registry that has never been fully synced
# #
# We include here both projects without a corresponding ProjectRegistry # We don't include projects without a corresponding ProjectRegistry
# or projects that have never successfully synced. # for performance reasons.
#
# @yield [ActiveRecord::Relation]
# @return [Geo::Fdw::Project] Projects that has never been fully synced
def never_synced_projects def never_synced_projects
no_project_registry = project_registry[:project_id].eq(nil) Geo::ProjectRegistry.where(last_repository_successful_sync_at: nil).includes(project: :route)
no_repository_synced = project_registry[:last_repository_successful_sync_at].eq(nil)
project_ids = Geo::Fdw::Project.select(:id).joins("LEFT OUTER JOIN project_registry ON (project_registry.project_id = #{Geo::Fdw::Project.table_name}.id)")
.where(
no_project_registry
.or(no_repository_synced))
# This allows us to inject pagination
if block_given?
project_ids = yield(project_ids)
end
Project.where(id: project_ids.to_a).includes(:project_registry)
end end
private private
......
...@@ -2,8 +2,6 @@ module Geo ...@@ -2,8 +2,6 @@ module Geo
module Fdw module Fdw
class Project < ::Geo::BaseFdw class Project < ::Geo::BaseFdw
self.table_name = Gitlab::Geo::Fdw.table('projects') self.table_name = Gitlab::Geo::Fdw.table('projects')
has_one :project_registry, class_name: 'Geo::ProjectRegistry'
end end
end end
end end
...@@ -168,6 +168,18 @@ class Geo::ProjectRegistry < Geo::BaseRegistry ...@@ -168,6 +168,18 @@ class Geo::ProjectRegistry < Geo::BaseRegistry
project.wiki_enabled? && (never_synced_wiki? || wiki_sync_needed?(scheduled_time)) project.wiki_enabled? && (never_synced_wiki? || wiki_sync_needed?(scheduled_time))
end end
def repository_verification_pending?
self.repository_verification_checksum_sha.nil?
end
def wiki_verification_pending?
self.wiki_verification_checksum_sha.nil?
end
def verification_pending?
repository_verification_pending? || wiki_verification_pending?
end
def syncs_since_gc def syncs_since_gc
Gitlab::Redis::SharedState.with { |redis| redis.get(fetches_since_gc_redis_key).to_i } Gitlab::Redis::SharedState.with { |redis| redis.get(fetches_since_gc_redis_key).to_i }
end end
...@@ -203,7 +215,7 @@ class Geo::ProjectRegistry < Geo::BaseRegistry ...@@ -203,7 +215,7 @@ class Geo::ProjectRegistry < Geo::BaseRegistry
# Flag the repository to be re-checked # Flag the repository to be re-checked
def flag_repository_for_recheck! def flag_repository_for_recheck!
self.update(repository_verification_checksum_sha: nil, last_repository_verification_failure: nil, repository_checksum_mismatch: nil) self.update(repository_verification_checksum_sha: nil, last_repository_verification_failure: nil, repository_checksum_mismatch: false)
end end
# Flag the repository to be re-synced # Flag the repository to be re-synced
......
- @projects.each do |project| - @registries.each do |project_registry|
.card.project-card.prepend-top-15 .card.project-card.prepend-top-15
.card-header{ id: "project-#{project.id}-header" } .card-header{ id: "project-#{project.id}-header" }
.d-flex .d-flex
%strong.header-text-primary.flex-fill %strong.header-text-primary.flex-fill
= project.full_name = project_registry.project.full_name
.card-body .card-body
.container .container
...@@ -12,30 +12,30 @@ ...@@ -12,30 +12,30 @@
.project-status-title.text-muted .project-status-title.text-muted
= s_('Geo|Next sync scheduled at') = s_('Geo|Next sync scheduled at')
.project-status-content .project-status-content
- if project.project_registry&.repository_retry_at - if project_registry.repository_retry_at
= distance_of_time_in_words(Time.now, project.project_registry.repository_retry_at) = distance_of_time_in_words(Time.now, project_registry.repository_retry_at)
- else - else
= s_('Geo|Never') = s_('Geo|Never')
.col-sm.project-status-container .col-sm.project-status-container
.project-status-title.text-muted .project-status-title.text-muted
= s_('Geo|Last sync attempt') = s_('Geo|Last sync attempt')
.project-status-content .project-status-content
- if project.project_registry&.last_repository_synced_at - if project_registry.last_repository_synced_at
= distance_of_time_in_words(Time.now, project.project_registry.last_repository_synced_at) = distance_of_time_in_words(Time.now, project_registry.last_repository_synced_at)
- else - else
= s_('Geo|Never') = s_('Geo|Never')
.col-sm.project-status-container .col-sm.project-status-container
.project-status-title.text-muted .project-status-title.text-muted
= s_('Geo|Retry counts') = s_('Geo|Retry counts')
.project-status-content .project-status-content
= project.project_registry.repository_retry_count if project.project_registry = project_registry.repository_retry_count
.col-sm.project-status-container .col-sm.project-status-container
.project-status-title.text-muted .project-status-title.text-muted
= s_('Geo|Error message') = s_('Geo|Error message')
.project-status-content.font-weight-bold .project-status-content.font-weight-bold
- if project.project_registry - if project_registry
= project.project_registry.last_repository_sync_failure = project_registry.last_repository_sync_failure
- else - else
= s_('Geo|No errors') = s_('Geo|No errors')
= paginate @projects, theme: 'gitlab' = paginate @registries, theme: 'gitlab'
...@@ -24,7 +24,7 @@ ...@@ -24,7 +24,7 @@
- if project_registry.repository_retry_at - if project_registry.repository_retry_at
= distance_of_time_in_words(Time.now, project_registry.repository_retry_at) = distance_of_time_in_words(Time.now, project_registry.repository_retry_at)
- else - else
= s_('Geo|Never') = s_('Geo|Never')
.col-sm.project-status-container .col-sm.project-status-container
.project-status-title.text-muted .project-status-title.text-muted
= s_('Geo|Last sync attempt') = s_('Geo|Last sync attempt')
......
...@@ -28,4 +28,4 @@ ...@@ -28,4 +28,4 @@
- when 'pending' - when 'pending'
= render(partial: 'pending') = render(partial: 'pending')
- else - else
= render(partial: 'all') = render(partial: 'synced')
# frozen_string_literal: true
require 'spec_helper'
describe Admin::GeoProjectsController, :geo do
set(:admin) { create(:admin) }
let(:synced_registry) { create(:geo_project_registry, :synced) }
before do
sign_in(admin)
end
shared_examples 'license required' do
context 'without a valid license' do
it 'redirects to license page with a flash message' do
expect(subject).to redirect_to(admin_license_path)
expect(flash[:alert]).to include('You need a different license to use Geo replication')
end
end
end
describe '#index' do
subject { get :index }
it_behaves_like 'license required'
context 'with a valid license' do
before do
allow(Gitlab::Geo).to receive(:license_allows?).and_return(true)
end
it 'renders synced template when no extra get params is specified' do
expect(subject).to have_gitlab_http_status(200)
expect(subject).to render_template(:index, partial: :synced)
end
context 'with sync_status=pending' do
subject { get :index, sync_status: 'pending' }
it 'renders pending template' do
expect(subject).to have_gitlab_http_status(200)
expect(subject).to render_template(:index, partial: :pending)
end
end
context 'with sync_status=failed' do
subject { get :index, sync_status: 'failed' }
it 'renders failed template' do
expect(subject).to have_gitlab_http_status(200)
expect(subject).to render_template(:index, partial: :failed)
end
end
context 'with sync_status=never' do
subject { get :index, sync_status: 'never' }
it 'renders failed template' do
expect(subject).to have_gitlab_http_status(200)
expect(subject).to render_template(:index, partial: :never)
end
end
end
end
describe '#recheck' do
subject { post :recheck, id: synced_registry }
it_behaves_like 'license required'
context 'with a valid license' do
before do
allow(Gitlab::Geo).to receive(:license_allows?).and_return(true)
end
it 'flags registry for recheck' do
expect(subject).to redirect_to(admin_geo_projects_path)
expect(flash[:notice]).to include('is scheduled for re-check')
expect(synced_registry.reload.verification_pending?).to be_truthy
end
end
end
describe '#resync' do
subject { post :resync, id: synced_registry }
it_behaves_like 'license required'
context 'with a valid license' do
before do
allow(Gitlab::Geo).to receive(:license_allows?).and_return(true)
end
it 'flags registry for resync' do
expect(subject).to redirect_to(admin_geo_projects_path)
expect(flash[:notice]).to include('is scheduled for re-sync')
expect(synced_registry.reload.resync_repository?).to be_truthy
end
end
end
describe '#force_redownload' do
subject { post :force_redownload, id: synced_registry }
it_behaves_like 'license required'
context 'with a valid license' do
before do
allow(Gitlab::Geo).to receive(:license_allows?).and_return(true)
end
it 'flags registry for re-download' do
expect(subject).to redirect_to(admin_geo_projects_path)
expect(flash[:notice]).to include('is scheduled for forced re-download')
expect(synced_registry.reload.should_be_redownloaded?('repository')).to be_truthy
end
end
end
end
...@@ -17,8 +17,6 @@ describe Geo::ProjectRegistryStatusFinder, :geo do ...@@ -17,8 +17,6 @@ describe Geo::ProjectRegistryStatusFinder, :geo do
set(:never_synced_registry) { create(:geo_project_registry) } set(:never_synced_registry) { create(:geo_project_registry) }
set(:never_synced_registry_with_failure) { create(:geo_project_registry, :repository_sync_failed) } set(:never_synced_registry_with_failure) { create(:geo_project_registry, :repository_sync_failed) }
set(:project_without_registry) { create(:project, name: 'project without registry') }
let(:project_with_never_synced_registry) { never_synced_registry.project }
subject { described_class.new(current_node: secondary) } subject { described_class.new(current_node: secondary) }
...@@ -61,13 +59,12 @@ describe Geo::ProjectRegistryStatusFinder, :geo do ...@@ -61,13 +59,12 @@ describe Geo::ProjectRegistryStatusFinder, :geo do
end end
describe '#never_synced_projects' do describe '#never_synced_projects' do
it 'returns only FDW projects without registry or with never synced registries' do it 'returns only never fully synced registries' do
result = subject.never_synced_projects result = subject.never_synced_projects
expect(result).to contain_exactly( expect(result).to contain_exactly(
project_without_registry, never_synced_registry,
project_with_never_synced_registry, never_synced_registry_with_failure
never_synced_registry_with_failure.project
) )
end end
end end
......
...@@ -3262,6 +3262,9 @@ msgstr "" ...@@ -3262,6 +3262,9 @@ msgstr ""
msgid "Geo|Next sync scheduled at" msgid "Geo|Next sync scheduled at"
msgstr "" msgstr ""
msgid "Geo|No errors"
msgstr ""
msgid "Geo|Pending" msgid "Geo|Pending"
msgstr "" msgstr ""
......
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