Commit d3c5ff7b authored by Hordur Freyr Yngvason's avatar Hordur Freyr Yngvason Committed by Michael Kozono

Squash project templates on update

As per https://gitlab.com/gitlab-org/gitlab-ce/issues/46043, project
templates should be squashed before updating, so that repositories
created from these templates don't include the full history of the
backing repository.
parent 44717d26
...@@ -24,6 +24,14 @@ module Gitlab ...@@ -24,6 +24,14 @@ module Gitlab
"#{preview}.git" "#{preview}.git"
end end
def project_path
URI.parse(preview).path.sub(%r{\A/}, '')
end
def uri_encoded_project_path
ERB::Util.url_encode(project_path)
end
def ==(other) def ==(other)
name == other.name && title == other.title name == other.name && title == other.title
end end
......
...@@ -40,7 +40,6 @@ namespace :gitlab do ...@@ -40,7 +40,6 @@ namespace :gitlab do
templates.each do |template| templates.each do |template|
params = { params = {
import_url: template.clone_url,
namespace_id: tmp_namespace.id, namespace_id: tmp_namespace.id,
path: template.name, path: template.name,
skip_wiki: true skip_wiki: true
...@@ -53,21 +52,45 @@ namespace :gitlab do ...@@ -53,21 +52,45 @@ namespace :gitlab do
raise "Failed to create project: #{project.errors.messages}" raise "Failed to create project: #{project.errors.messages}"
end end
loop do uri_encoded_project_path = template.uri_encoded_project_path
if project.import_finished?
puts "Import finished for #{template.name}"
break
end
if project.import_failed? # extract a concrete commit for signing off what we actually downloaded
raise "Failed to import from #{project_params[:import_url]}" # this way we do the right thing even if the repository gets updated in the meantime
end get_commits_response = Gitlab::HTTP.get("https://gitlab.com/api/v4/projects/#{uri_encoded_project_path}/repository/commits",
query: { page: 1, per_page: 1 }
)
raise "Failed to retrieve latest commit for template '#{template.name}'" unless get_commits_response.success?
puts "Waiting for the import to finish" commit_sha = get_commits_response.parsed_response.dig(0, 'id')
sleep(5) project_archive_uri = "https://gitlab.com/api/v4/projects/#{uri_encoded_project_path}/repository/archive.tar.gz?sha=#{commit_sha}"
project.reset commit_message = <<~MSG
Initialized from '#{template.title}' project template
Template repository: #{template.preview}
Commit SHA: #{commit_sha}
MSG
Dir.mktmpdir do |tmpdir|
Dir.chdir(tmpdir) do
Gitlab::TaskHelpers.run_command!(['wget', project_archive_uri, '-O', 'archive.tar.gz'])
Gitlab::TaskHelpers.run_command!(['tar', 'xf', 'archive.tar.gz'])
extracted_project_basename = Dir['*/'].first
Dir.chdir(extracted_project_basename) do
Gitlab::TaskHelpers.run_command!(%w(git init))
Gitlab::TaskHelpers.run_command!(%w(git add .))
Gitlab::TaskHelpers.run_command!(['git', 'commit', '--author', 'GitLab <root@localhost>', '--message', commit_message])
# Hacky workaround to push to the project in a way that works with both GDK and the test environment
Gitlab::GitalyClient::StorageSettings.allow_disk_access do
Gitlab::TaskHelpers.run_command!(['git', 'remote', 'add', 'origin', "file://#{project.repository.raw.path}"])
end
Gitlab::TaskHelpers.run_command!(['git', 'push', '-u', 'origin', 'master'])
end
end end
end
project.reset
Projects::ImportExport::ExportService.new(project, admin).execute Projects::ImportExport::ExportService.new(project, admin).execute
downloader.call(project.export_file, template.archive_path) downloader.call(project.export_file, template.archive_path)
......
...@@ -28,6 +28,18 @@ describe Gitlab::ProjectTemplate do ...@@ -28,6 +28,18 @@ describe Gitlab::ProjectTemplate do
end end
end end
describe '#project_path' do
subject { described_class.new('name', 'title', 'description', 'https://gitlab.com/some/project/path').project_path }
it { is_expected.to eq 'some/project/path' }
end
describe '#uri_encoded_project_path' do
subject { described_class.new('name', 'title', 'description', 'https://gitlab.com/some/project/path').uri_encoded_project_path }
it { is_expected.to eq 'some%2Fproject%2Fpath' }
end
describe '.find' do describe '.find' do
subject { described_class.find(query) } subject { described_class.find(query) }
......
...@@ -8,9 +8,18 @@ describe 'gitlab:update_project_templates rake task' do ...@@ -8,9 +8,18 @@ describe 'gitlab:update_project_templates rake task' do
before do before do
Rake.application.rake_require 'tasks/gitlab/update_templates' Rake.application.rake_require 'tasks/gitlab/update_templates'
create(:admin) create(:admin)
allow(Gitlab::ProjectTemplate) allow(Gitlab::ProjectTemplate)
.to receive(:archive_directory) .to receive(:archive_directory)
.and_return(Pathname.new(tmpdir)) .and_return(Pathname.new(tmpdir))
# Gitlab::HTTP resolves the domain to an IP prior to WebMock taking effect, hence the wildcard
stub_request(:get, %r{^https://.*/api/v4/projects/gitlab-org%2Fproject-templates%2Frails/repository/commits\?page=1&per_page=1})
.to_return(
status: 200,
body: [{ id: '67812735b83cb42710f22dc98d73d42c8bf4d907' }].to_json,
headers: { 'Content-Type' => 'application/json' }
)
end end
after do after do
......
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