Commit 046a5e39 authored by Stan Hu's avatar Stan Hu

More work towards supporting Bitbucket Server

parent c9deb7ce
...@@ -20,7 +20,7 @@ class Import::BitbucketServerController < Import::BaseController ...@@ -20,7 +20,7 @@ class Import::BitbucketServerController < Import::BaseController
target_namespace = find_or_create_namespace(namespace_path, current_user) target_namespace = find_or_create_namespace(namespace_path, current_user)
if current_user.can?(:create_projects, target_namespace) if current_user.can?(:create_projects, target_namespace)
project = Gitlab::BitbucketImport::ProjectCreator.new(repo, project_name, target_namespace, current_user, credentials).execute project = Gitlab::BitbucketServerImport::ProjectCreator.new(project_slug, repo_slug, repo, project_name, target_namespace, current_user, credentials).execute
if project.persisted? if project.persisted?
render json: ProjectSerializer.new.represent(project) render json: ProjectSerializer.new.represent(project)
...@@ -86,7 +86,7 @@ class Import::BitbucketServerController < Import::BaseController ...@@ -86,7 +86,7 @@ class Import::BitbucketServerController < Import::BaseController
{ {
base_uri: session[bitbucket_server_url_key], base_uri: session[bitbucket_server_url_key],
username: session[bitbucket_server_username_key], username: session[bitbucket_server_username_key],
personal_access_token: session[personal_access_token_key] password: session[personal_access_token_key]
} }
end end
end end
...@@ -16,8 +16,8 @@ module BitbucketServer ...@@ -16,8 +16,8 @@ module BitbucketServer
get_collection(path, :comment) get_collection(path, :comment)
end end
def pull_requests(repo) def pull_requests(project_key, repository_slug)
path = "/repositories/#{repo}/pullrequests?state=ALL" path = "/projects/#{project}/repos/#{repo}/pull-requests?state=ALL"
get_collection(path, :pull_request) get_collection(path, :pull_request)
end end
......
...@@ -8,7 +8,7 @@ module BitbucketServer ...@@ -8,7 +8,7 @@ module BitbucketServer
@api_version = options.fetch(:api_version, DEFAULT_API_VERSION) @api_version = options.fetch(:api_version, DEFAULT_API_VERSION)
@base_uri = options[:base_uri] @base_uri = options[:base_uri]
@username = options[:username] @username = options[:username]
@token = options[:personal_access_token] @token = options[:password]
end end
def get(path, extra_query = {}) def get(path, extra_query = {})
......
module Bitbucket
module Representation
class Issue < Representation::Base
CLOSED_STATUS = %w(resolved invalid duplicate wontfix closed).freeze
def iid
raw['id']
end
def kind
raw['kind']
end
def author
raw.dig('reporter', 'username')
end
def description
raw.fetch('content', {}).fetch('raw', nil)
end
def state
closed? ? 'closed' : 'opened'
end
def title
raw['title']
end
def milestone
raw['milestone']['name'] if raw['milestone'].present?
end
def created_at
raw['created_on']
end
def updated_at
raw['edited_on']
end
def to_s
iid
end
private
def closed?
CLOSED_STATUS.include?(raw['state'])
end
end
end
end
...@@ -2,7 +2,7 @@ module BitbucketServer ...@@ -2,7 +2,7 @@ module BitbucketServer
module Representation module Representation
class PullRequest < Representation::Base class PullRequest < Representation::Base
def author def author
raw.fetch('author', {}).fetch('username', nil) raw.fetch('author', {}).fetch('user', {}).fetch('name')
end end
def description def description
...@@ -24,11 +24,11 @@ module BitbucketServer ...@@ -24,11 +24,11 @@ module BitbucketServer
end end
def created_at def created_at
raw['created_on'] raw['createdDate']
end end
def updated_at def updated_at
raw['updated_on'] raw['updatedDate']
end end
def title def title
...@@ -36,29 +36,29 @@ module BitbucketServer ...@@ -36,29 +36,29 @@ module BitbucketServer
end end
def source_branch_name def source_branch_name
source_branch.fetch('branch', {}).fetch('name', nil) source_branch['id']
end end
def source_branch_sha def source_branch_sha
source_branch.fetch('commit', {}).fetch('hash', nil) # XXX Not implemented?
end end
def target_branch_name def target_branch_name
target_branch.fetch('branch', {}).fetch('name', nil) target_branch['id']
end end
def target_branch_sha def target_branch_sha
target_branch.fetch('commit', {}).fetch('hash', nil) # XXX Not implemented?
end end
private private
def source_branch def source_branch
raw['source'] raw['fromRef'] || {}
end end
def target_branch def target_branch
raw['destination'] raw['toRef'] || {}
end end
end end
end end
......
...@@ -15,16 +15,8 @@ module BitbucketServer ...@@ -15,16 +15,8 @@ module BitbucketServer
raw['slug'] raw['slug']
end end
def clone_url(token = nil) def clone_url
url = raw['links']['clone'].find { |link| link['name'].starts_with?('http') }.fetch('href') raw['links']['clone'].find { |link| link['name'].starts_with?('http') }.fetch('href')
if token.present?
clone_url = URI.parse(url)
clone_url.user = "x-token-auth:#{token}"
clone_url.to_s
else
url
end
end end
def description def description
......
...@@ -8,10 +8,12 @@ module Gitlab ...@@ -8,10 +8,12 @@ module Gitlab
{ title: 'proposal', color: '#69D100' }, { title: 'proposal', color: '#69D100' },
{ title: 'task', color: '#7F8C8D' }].freeze { title: 'task', color: '#7F8C8D' }].freeze
attr_reader :project, :client, :errors, :users attr_reader :project_key, :repository_slug, :client, :errors, :users
def initialize(project) def initialize(project)
@project = project @project = project
@project_key = project.import_data.data['project_key']
@repository_slug = project.import_data.data['repo_slug']
@client = BitbucketServer::Client.new(project.import_data.credentials) @client = BitbucketServer::Client.new(project.import_data.credentials)
@formatter = Gitlab::ImportFormatter.new @formatter = Gitlab::ImportFormatter.new
@labels = {} @labels = {}
...@@ -20,7 +22,6 @@ module Gitlab ...@@ -20,7 +22,6 @@ module Gitlab
end end
def execute def execute
import_issues
import_pull_requests import_pull_requests
handle_errors handle_errors
...@@ -49,7 +50,7 @@ module Gitlab ...@@ -49,7 +50,7 @@ module Gitlab
users[username] = User.select(:id) users[username] = User.select(:id)
.joins(:identities) .joins(:identities)
.find_by("identities.extern_uid = ? AND identities.provider = 'bitbucket'", username) .find_by("identities.extern_uid = ? AND identities.provider = 'bitbucket_server'", username)
.try(:id) .try(:id)
end end
...@@ -57,80 +58,8 @@ module Gitlab ...@@ -57,80 +58,8 @@ module Gitlab
@repo ||= client.repo(project.import_source) @repo ||= client.repo(project.import_source)
end end
def import_issues
return unless repo.issues_enabled?
create_labels
client.issues(repo).each do |issue|
begin
description = ''
description += @formatter.author_line(issue.author) unless find_user_id(issue.author)
description += issue.description
label_name = issue.kind
milestone = issue.milestone ? project.milestones.find_or_create_by(title: issue.milestone) : nil
gitlab_issue = project.issues.create!(
iid: issue.iid,
title: issue.title,
description: description,
state: issue.state,
author_id: gitlab_user_id(project, issue.author),
milestone: milestone,
created_at: issue.created_at,
updated_at: issue.updated_at
)
gitlab_issue.labels << @labels[label_name]
import_issue_comments(issue, gitlab_issue) if gitlab_issue.persisted?
rescue StandardError => e
errors << { type: :issue, iid: issue.iid, errors: e.message }
end
end
end
def import_issue_comments(issue, gitlab_issue)
client.issue_comments(repo, issue.iid).each do |comment|
# The note can be blank for issue service messages like "Changed title: ..."
# We would like to import those comments as well but there is no any
# specific parameter that would allow to process them, it's just an empty comment.
# To prevent our importer from just crashing or from creating useless empty comments
# we do this check.
next unless comment.note.present?
note = ''
note += @formatter.author_line(comment.author) unless find_user_id(comment.author)
note += comment.note
begin
gitlab_issue.notes.create!(
project: project,
note: note,
author_id: gitlab_user_id(project, comment.author),
created_at: comment.created_at,
updated_at: comment.updated_at
)
rescue StandardError => e
errors << { type: :issue_comment, iid: issue.iid, errors: e.message }
end
end
end
def create_labels
LABELS.each do |label_params|
label = ::Labels::CreateService.new(label_params).execute(project: project)
if label.valid?
@labels[label_params[:title]] = label
else
raise "Failed to create label \"#{label_params[:title]}\" for project \"#{project.full_name}\""
end
end
end
def import_pull_requests def import_pull_requests
pull_requests = client.pull_requests(repo) pull_requests = client.pull_requests(project_key, repository_slug)
pull_requests.each do |pull_request| pull_requests.each do |pull_request|
begin begin
......
module Gitlab module Gitlab
module BitbucketServerImport module BitbucketServerImport
class ProjectCreator class ProjectCreator
attr_reader :repo, :name, :namespace, :current_user, :session_data attr_reader :project_key, :repo_slug, :repo, :name, :namespace, :current_user, :session_data
def initialize(repo, name, namespace, current_user, session_data) def initialize(project_key, repo_slug, repo, name, namespace, current_user, session_data)
@project_key = project_key
@repo_slug = repo_slug
@repo = repo @repo = repo
@name = name @name = name
@namespace = namespace @namespace = namespace
...@@ -19,19 +21,17 @@ module Gitlab ...@@ -19,19 +21,17 @@ module Gitlab
description: repo.description, description: repo.description,
namespace_id: namespace.id, namespace_id: namespace.id,
visibility_level: repo.visibility_level, visibility_level: repo.visibility_level,
import_type: 'bitbucket', import_type: 'bitbucket_server',
import_source: repo.full_name, import_source: repo.full_name,
import_url: repo.clone_url(session_data[:token]), import_url: repo.clone_url,
import_data: { credentials: session_data }, import_data: {
skip_wiki: skip_wiki credentials: session_data,
data: { project_key: project_key,
repo_slug: repo_slug },
},
skip_wiki: true
).execute ).execute
end end
private
def skip_wiki
repo.has_wiki?
end
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