Commit c3d0947e authored by George Koltsov's avatar George Koltsov

Fix Wiki Pipeline network error if wiki does not exist

  - Add a preliminary check on source to see if there are
    any wiki pages available before doing wiki repository import
  - Update `BulkImports::Clients::HTTP` to have a more detailed
    error message when a non-successful http response is returned

Changelog: fixed
parent 1d835b16
...@@ -20,8 +20,6 @@ ...@@ -20,8 +20,6 @@
class BulkImports::Entity < ApplicationRecord class BulkImports::Entity < ApplicationRecord
self.table_name = 'bulk_import_entities' self.table_name = 'bulk_import_entities'
EXPORT_RELATIONS_URL = '/%{resource}/%{full_path}/export_relations'
belongs_to :bulk_import, optional: false belongs_to :bulk_import, optional: false
belongs_to :parent, class_name: 'BulkImports::Entity', optional: true belongs_to :parent, class_name: 'BulkImports::Entity', optional: true
...@@ -112,14 +110,22 @@ class BulkImports::Entity < ApplicationRecord ...@@ -112,14 +110,22 @@ class BulkImports::Entity < ApplicationRecord
entity_type.pluralize entity_type.pluralize
end end
def base_resource_url_path
"/#{pluralized_name}/#{encoded_source_full_path}"
end
def export_relations_url_path def export_relations_url_path
@export_relations_url_path ||= EXPORT_RELATIONS_URL % { resource: pluralized_name, full_path: encoded_source_full_path } "#{base_resource_url_path}/export_relations"
end end
def relation_download_url_path(relation) def relation_download_url_path(relation)
"#{export_relations_url_path}/download?relation=#{relation}" "#{export_relations_url_path}/download?relation=#{relation}"
end end
def wikis_url_path
"#{base_resource_url_path}/wikis"
end
def project? def project?
source_type == 'project_entity' source_type == 'project_entity'
end end
......
...@@ -123,7 +123,7 @@ module BulkImports ...@@ -123,7 +123,7 @@ module BulkImports
def with_error_handling def with_error_handling
response = yield response = yield
raise ::BulkImports::NetworkError.new(response: response) unless response.success? raise ::BulkImports::NetworkError.new("Unsuccessful response #{response.code} from #{response.request.path.path}", response: response) unless response.success?
response response
rescue *Gitlab::HTTP::HTTP_ERRORS => e rescue *Gitlab::HTTP::HTTP_ERRORS => e
......
...@@ -7,7 +7,9 @@ module BulkImports ...@@ -7,7 +7,9 @@ module BulkImports
include Pipeline include Pipeline
def extract(*) def extract(*)
BulkImports::Pipeline::ExtractedData.new(data: { url: url_from_parent_path(context.entity.source_full_path) }) url = url_from_parent_path(context.entity.source_full_path) if source_wiki_exists?
BulkImports::Pipeline::ExtractedData.new(data: { url: url })
end end
def transform(_, data) def transform(_, data)
...@@ -15,14 +17,15 @@ module BulkImports ...@@ -15,14 +17,15 @@ module BulkImports
end end
def load(context, data) def load(context, data)
return unless context.portable.wiki return unless data&.dig(:url)
wiki = context.portable.wiki
url = data[:url].sub("://", "://oauth2:#{context.configuration.access_token}@") url = data[:url].sub("://", "://oauth2:#{context.configuration.access_token}@")
Gitlab::UrlBlocker.validate!(url, allow_local_network: allow_local_requests?, allow_localhost: allow_local_requests?) Gitlab::UrlBlocker.validate!(url, allow_local_network: allow_local_requests?, allow_localhost: allow_local_requests?)
context.portable.wiki.ensure_repository wiki.ensure_repository
context.portable.wiki.repository.fetch_as_mirror(url) wiki.repository.fetch_as_mirror(url)
end end
private private
...@@ -36,6 +39,16 @@ module BulkImports ...@@ -36,6 +39,16 @@ module BulkImports
def allow_local_requests? def allow_local_requests?
Gitlab::CurrentSettings.allow_local_requests_from_web_hooks_and_services? Gitlab::CurrentSettings.allow_local_requests_from_web_hooks_and_services?
end end
def source_wiki_exists?
wikis = client.get(context.entity.wikis_url_path).parsed_response
wikis.any?
end
def client
BulkImports::Clients::HTTP.new(url: context.configuration.url, token: context.configuration.access_token)
end
end end
end end
end end
......
...@@ -38,11 +38,11 @@ RSpec.describe BulkImports::Clients::HTTP do ...@@ -38,11 +38,11 @@ RSpec.describe BulkImports::Clients::HTTP do
context 'when response is not success' do context 'when response is not success' do
it 'raises BulkImports::Error' do it 'raises BulkImports::Error' do
response_double = double(code: 503, success?: false) response_double = double(code: 503, success?: false, request: double(path: double(path: '/test')))
allow(Gitlab::HTTP).to receive(method).and_return(response_double) allow(Gitlab::HTTP).to receive(method).and_return(response_double)
expect { subject.public_send(method, resource) }.to raise_exception(BulkImports::NetworkError) expect { subject.public_send(method, resource) }.to raise_exception(BulkImports::NetworkError, 'Unsuccessful response 503 from /test')
end end
end end
end end
......
...@@ -282,4 +282,20 @@ RSpec.describe BulkImports::Entity, type: :model do ...@@ -282,4 +282,20 @@ RSpec.describe BulkImports::Entity, type: :model do
expect(project_entity.group?).to eq(false) expect(project_entity.group?).to eq(false)
end end
end end
describe '#base_resource_url_path' do
it 'returns base entity url path' do
entity = build(:bulk_import_entity)
expect(entity.base_resource_url_path).to eq("/groups/#{entity.encoded_source_full_path}")
end
end
describe '#wiki_url_path' do
it 'returns entity wiki url path' do
entity = build(:bulk_import_entity)
expect(entity.wikis_url_path).to eq("/groups/#{entity.encoded_source_full_path}/wikis")
end
end
end end
...@@ -9,7 +9,6 @@ RSpec.shared_examples 'wiki pipeline imports a wiki for an entity' do ...@@ -9,7 +9,6 @@ RSpec.shared_examples 'wiki pipeline imports a wiki for an entity' do
let(:extracted_data) { BulkImports::Pipeline::ExtractedData.new(data: {}) } let(:extracted_data) { BulkImports::Pipeline::ExtractedData.new(data: {}) }
context 'successfully imports wiki for an entity' do
subject { described_class.new(context) } subject { described_class.new(context) }
before do before do
...@@ -18,7 +17,10 @@ RSpec.shared_examples 'wiki pipeline imports a wiki for an entity' do ...@@ -18,7 +17,10 @@ RSpec.shared_examples 'wiki pipeline imports a wiki for an entity' do
end end
end end
context 'when wiki exists' do
it 'imports new wiki into destination project' do it 'imports new wiki into destination project' do
expect(subject).to receive(:source_wiki_exists?).and_return(true)
expect_next_instance_of(Gitlab::GitalyClient::RepositoryService) do |repository_service| expect_next_instance_of(Gitlab::GitalyClient::RepositoryService) do |repository_service|
url = "https://oauth2:token@gitlab.example/#{entity.source_full_path}.wiki.git" url = "https://oauth2:token@gitlab.example/#{entity.source_full_path}.wiki.git"
expect(repository_service).to receive(:fetch_remote).with(url, any_args).and_return 0 expect(repository_service).to receive(:fetch_remote).with(url, any_args).and_return 0
...@@ -27,5 +29,16 @@ RSpec.shared_examples 'wiki pipeline imports a wiki for an entity' do ...@@ -27,5 +29,16 @@ RSpec.shared_examples 'wiki pipeline imports a wiki for an entity' do
subject.run subject.run
end end
end end
context 'when wiki does not exist' do
it 'does not import wiki' do
expect(subject).to receive(:source_wiki_exists?).and_return(false)
expect(parent.wiki).not_to receive(:ensure_repository)
expect(parent.wiki.repository).not_to receive(:ensure_repository)
expect { subject.run }.not_to raise_error
end
end
end end
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