Merge remote-tracking branch 'gitlabhq/master' into git_commit_fix
Conflicts: doc/install/installation.md
This commit is contained in:
commit
93f0a8c9b3
369 changed files with 102667 additions and 1369 deletions
|
@ -18,5 +18,6 @@ module Gitlab
|
|||
mount Issues
|
||||
mount Milestones
|
||||
mount Session
|
||||
mount MergeRequests
|
||||
end
|
||||
end
|
||||
|
|
|
@ -63,5 +63,15 @@ module Gitlab
|
|||
class SSHKey < Grape::Entity
|
||||
expose :id, :title, :key
|
||||
end
|
||||
|
||||
class MergeRequest < Grape::Entity
|
||||
expose :id, :target_branch, :source_branch, :project_id, :title, :closed, :merged
|
||||
expose :author, :assignee, using: Entities::UserBasic
|
||||
end
|
||||
|
||||
class Note < Grape::Entity
|
||||
expose :author, using: Entities::UserBasic
|
||||
expose :note
|
||||
end
|
||||
end
|
||||
end
|
||||
|
|
118
lib/api/merge_requests.rb
Normal file
118
lib/api/merge_requests.rb
Normal file
|
@ -0,0 +1,118 @@
|
|||
module Gitlab
|
||||
# MergeRequest API
|
||||
class MergeRequests < Grape::API
|
||||
before { authenticate! }
|
||||
|
||||
resource :projects do
|
||||
|
||||
# List merge requests
|
||||
#
|
||||
# Parameters:
|
||||
# id (required) - The ID or code name of a project
|
||||
#
|
||||
# Example:
|
||||
# GET /projects/:id/merge_requests
|
||||
#
|
||||
get ":id/merge_requests" do
|
||||
authorize! :read_merge_request, user_project
|
||||
|
||||
present paginate(user_project.merge_requests), with: Entities::MergeRequest
|
||||
end
|
||||
|
||||
# Show MR
|
||||
#
|
||||
# Parameters:
|
||||
# id (required) - The ID or code name of a project
|
||||
# merge_request_id (required) - The ID of MR
|
||||
#
|
||||
# Example:
|
||||
# GET /projects/:id/merge_request/:merge_request_id
|
||||
#
|
||||
get ":id/merge_request/:merge_request_id" do
|
||||
merge_request = user_project.merge_requests.find(params[:merge_request_id])
|
||||
|
||||
authorize! :read_merge_request, merge_request
|
||||
|
||||
present merge_request, with: Entities::MergeRequest
|
||||
end
|
||||
|
||||
# Create MR
|
||||
#
|
||||
# Parameters:
|
||||
#
|
||||
# id (required) - The ID or code name of a project
|
||||
# source_branch (required) - The source branch
|
||||
# target_branch (required) - The target branch
|
||||
# assignee_id - Assignee user ID
|
||||
# title (required) - Title of MR
|
||||
#
|
||||
# Example:
|
||||
# POST /projects/:id/merge_requests
|
||||
#
|
||||
post ":id/merge_requests" do
|
||||
authorize! :write_merge_request, user_project
|
||||
|
||||
attrs = attributes_for_keys [:source_branch, :target_branch, :assignee_id, :title]
|
||||
merge_request = user_project.merge_requests.new(attrs)
|
||||
merge_request.author = current_user
|
||||
|
||||
if merge_request.save
|
||||
merge_request.reload_code
|
||||
present merge_request, with: Entities::MergeRequest
|
||||
else
|
||||
not_found!
|
||||
end
|
||||
end
|
||||
|
||||
# Update MR
|
||||
#
|
||||
# Parameters:
|
||||
# id (required) - The ID or code name of a project
|
||||
# merge_request_id (required) - ID of MR
|
||||
# source_branch - The source branch
|
||||
# target_branch - The target branch
|
||||
# assignee_id - Assignee user ID
|
||||
# title - Title of MR
|
||||
# closed - Status of MR. true - closed
|
||||
# Example:
|
||||
# PUT /projects/:id/merge_request/:merge_request_id
|
||||
#
|
||||
put ":id/merge_request/:merge_request_id" do
|
||||
attrs = attributes_for_keys [:source_branch, :target_branch, :assignee_id, :title, :closed]
|
||||
merge_request = user_project.merge_requests.find(params[:merge_request_id])
|
||||
|
||||
authorize! :modify_merge_request, merge_request
|
||||
|
||||
if merge_request.update_attributes attrs
|
||||
merge_request.reload_code
|
||||
merge_request.mark_as_unchecked
|
||||
present merge_request, with: Entities::MergeRequest
|
||||
else
|
||||
not_found!
|
||||
end
|
||||
end
|
||||
|
||||
# Post comment to merge request
|
||||
#
|
||||
# Parameters:
|
||||
# id (required) - The ID or code name of a project
|
||||
# merge_request_id (required) - ID of MR
|
||||
# note (required) - Text of comment
|
||||
# Examples:
|
||||
# POST /projects/:id/merge_request/:merge_request_id/comments
|
||||
#
|
||||
post ":id/merge_request/:merge_request_id/comments" do
|
||||
merge_request = user_project.merge_requests.find(params[:merge_request_id])
|
||||
note = merge_request.notes.new(note: params[:note], project_id: user_project.id)
|
||||
note.author = current_user
|
||||
|
||||
if note.save
|
||||
present note, with: Entities::Note
|
||||
else
|
||||
not_found!
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
end
|
||||
end
|
|
@ -147,7 +147,7 @@ module Gitlab
|
|||
@hooks = paginate user_project.hooks
|
||||
present @hooks, with: Entities::Hook
|
||||
end
|
||||
|
||||
|
||||
# Get a project hook
|
||||
#
|
||||
# Parameters:
|
||||
|
@ -159,7 +159,7 @@ module Gitlab
|
|||
@hook = user_project.hooks.find(params[:hook_id])
|
||||
present @hook, with: Entities::Hook
|
||||
end
|
||||
|
||||
|
||||
|
||||
# Add hook to project
|
||||
#
|
||||
|
@ -177,7 +177,7 @@ module Gitlab
|
|||
error!({'message' => '404 Not found'}, 404)
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
# Update an existing project hook
|
||||
#
|
||||
# Parameters:
|
||||
|
@ -382,13 +382,7 @@ module Gitlab
|
|||
tree = Tree.new commit.tree, user_project, ref, params[:filepath]
|
||||
not_found! "File" unless tree.try(:tree)
|
||||
|
||||
if tree.text?
|
||||
encoding = Gitlab::Encode.detect_encoding(tree.data)
|
||||
content_type encoding ? "text/plain; charset=#{encoding}" : "text/plain"
|
||||
else
|
||||
content_type tree.mime_type
|
||||
end
|
||||
|
||||
content_type tree.mime_type
|
||||
present tree.data
|
||||
end
|
||||
|
||||
|
|
|
@ -23,24 +23,23 @@ module Gitlab
|
|||
@user = User.find(params[:id])
|
||||
present @user, with: Entities::User
|
||||
end
|
||||
|
||||
|
||||
# Create user. Available only for admin
|
||||
#
|
||||
# Parameters:
|
||||
# email (required) - Email
|
||||
# name (required) - Name
|
||||
# password (required) - Password
|
||||
# password_confirmation (required) - Password confirmation
|
||||
# name - Name
|
||||
# skype - Skype ID
|
||||
# linkedin - Linkedin
|
||||
# twitter - Twitter account
|
||||
# projects_limit - Limit projects wich user can create
|
||||
# projects_limit - Number of projects user can create
|
||||
# Example Request:
|
||||
# POST /users
|
||||
post do
|
||||
authenticated_as_admin!
|
||||
attrs = attributes_for_keys [:email, :name, :password, :password_confirmation, :skype, :linkedin, :twitter, :projects_limit]
|
||||
user = User.new attrs
|
||||
attrs = attributes_for_keys [:email, :name, :password, :skype, :linkedin, :twitter, :projects_limit]
|
||||
user = User.new attrs, as: :admin
|
||||
if user.save
|
||||
present user, with: Entities::User
|
||||
else
|
||||
|
|
68
lib/event_filter.rb
Normal file
68
lib/event_filter.rb
Normal file
|
@ -0,0 +1,68 @@
|
|||
class EventFilter
|
||||
attr_accessor :params
|
||||
|
||||
class << self
|
||||
def default_filter
|
||||
%w{ push issues merge_requests team}
|
||||
end
|
||||
|
||||
def push
|
||||
'push'
|
||||
end
|
||||
|
||||
def merged
|
||||
'merged'
|
||||
end
|
||||
|
||||
def comments
|
||||
'comments'
|
||||
end
|
||||
|
||||
def team
|
||||
'team'
|
||||
end
|
||||
end
|
||||
|
||||
def initialize params
|
||||
@params = if params
|
||||
params.dup
|
||||
else
|
||||
[]#EventFilter.default_filter
|
||||
end
|
||||
end
|
||||
|
||||
def apply_filter events
|
||||
return events unless params.present?
|
||||
|
||||
filter = params.dup
|
||||
|
||||
actions = []
|
||||
actions << Event::Pushed if filter.include? 'push'
|
||||
actions << Event::Merged if filter.include? 'merged'
|
||||
|
||||
if filter.include? 'team'
|
||||
actions << Event::Joined
|
||||
actions << Event::Left
|
||||
end
|
||||
|
||||
actions << Event::Commented if filter.include? 'comments'
|
||||
|
||||
events = events.where(action: actions)
|
||||
end
|
||||
|
||||
def options key
|
||||
filter = params.dup
|
||||
|
||||
if filter.include? key
|
||||
filter.delete key
|
||||
else
|
||||
filter << key
|
||||
end
|
||||
|
||||
filter
|
||||
end
|
||||
|
||||
def active? key
|
||||
params.include? key
|
||||
end
|
||||
end
|
|
@ -14,7 +14,10 @@ module Gitlab
|
|||
end
|
||||
|
||||
def ga_repo
|
||||
@ga_repo ||= ::Gitolite::GitoliteAdmin.new(File.join(config_tmp_dir,'gitolite'))
|
||||
@ga_repo ||= ::Gitolite::GitoliteAdmin.new(
|
||||
File.join(config_tmp_dir,'gitolite'),
|
||||
conf: Gitlab.config.gitolite_config_file
|
||||
)
|
||||
end
|
||||
|
||||
def apply
|
||||
|
|
|
@ -18,7 +18,7 @@ module Grack
|
|||
@env['SCRIPT_NAME'] = ""
|
||||
|
||||
# Find project by PATH_INFO from env
|
||||
if m = /^\/([\w-]+).git/.match(@request.path_info).to_a
|
||||
if m = /^\/([\w\.-]+)\.git/.match(@request.path_info).to_a
|
||||
self.project = Project.find_by_path(m.last)
|
||||
return false unless project
|
||||
end
|
||||
|
@ -65,7 +65,7 @@ module Grack
|
|||
end
|
||||
# Need to reset seek point
|
||||
@request.body.rewind
|
||||
/refs\/heads\/([\w-]+)/.match(input).to_a.first
|
||||
/refs\/heads\/([\w\.-]+)/.match(input).to_a.first
|
||||
end
|
||||
|
||||
protected
|
||||
|
|
|
@ -1,41 +0,0 @@
|
|||
# Patch Strings to enable detect_encoding! on views
|
||||
require 'charlock_holmes/string'
|
||||
module Gitlab
|
||||
module Encode
|
||||
extend self
|
||||
|
||||
def utf8 message
|
||||
# return nil if message is nil
|
||||
return nil unless message
|
||||
|
||||
message.force_encoding("utf-8")
|
||||
# return message if message type is binary
|
||||
detect = CharlockHolmes::EncodingDetector.detect(message)
|
||||
return message if detect[:type] == :binary
|
||||
|
||||
# if message is utf-8 encoding, just return it
|
||||
return message if message.valid_encoding?
|
||||
|
||||
# if message is not utf-8 encoding, convert it
|
||||
if detect[:encoding]
|
||||
message.force_encoding(detect[:encoding])
|
||||
message.encode!("utf-8", detect[:encoding], undef: :replace, replace: "", invalid: :replace)
|
||||
end
|
||||
|
||||
# ensure message encoding is utf8
|
||||
message.valid_encoding? ? message : raise
|
||||
|
||||
# Prevent app from crash cause of encoding errors
|
||||
rescue
|
||||
encoding = detect ? detect[:encoding] : "unknown"
|
||||
"--broken encoding: #{encoding}"
|
||||
end
|
||||
|
||||
def detect_encoding message
|
||||
return nil unless message
|
||||
|
||||
hash = CharlockHolmes::EncodingDetector.detect(message) rescue {}
|
||||
return hash[:encoding] ? hash[:encoding] : nil
|
||||
end
|
||||
end
|
||||
end
|
|
@ -1,58 +0,0 @@
|
|||
module Gitlab
|
||||
# GitLab file editor
|
||||
#
|
||||
# It gives you ability to make changes to files
|
||||
# & commit this changes from GitLab UI.
|
||||
class FileEditor
|
||||
attr_accessor :user, :project, :ref
|
||||
|
||||
def initialize(user, project, ref)
|
||||
self.user = user
|
||||
self.project = project
|
||||
self.ref = ref
|
||||
end
|
||||
|
||||
def update(path, content, commit_message, last_commit)
|
||||
return false unless can_edit?(path, last_commit)
|
||||
|
||||
Grit::Git.with_timeout(10.seconds) do
|
||||
lock_file = Rails.root.join("tmp", "#{project.path}.lock")
|
||||
|
||||
File.open(lock_file, "w+") do |f|
|
||||
f.flock(File::LOCK_EX)
|
||||
|
||||
unless project.satellite.exists?
|
||||
raise "Satellite doesn't exist"
|
||||
end
|
||||
|
||||
project.satellite.clear
|
||||
|
||||
Dir.chdir(project.satellite.path) do
|
||||
r = Grit::Repo.new('.')
|
||||
r.git.sh "git reset --hard"
|
||||
r.git.sh "git fetch origin"
|
||||
r.git.sh "git config user.name \"#{user.name}\""
|
||||
r.git.sh "git config user.email \"#{user.email}\""
|
||||
r.git.sh "git checkout -b #{ref} origin/#{ref}"
|
||||
File.open(path, 'w'){|f| f.write(content)}
|
||||
r.git.sh "git add ."
|
||||
r.git.sh "git commit -am '#{commit_message}'"
|
||||
output = r.git.sh "git push origin #{ref}"
|
||||
|
||||
if output =~ /reject/
|
||||
return false
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
true
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def can_edit?(path, last_commit)
|
||||
current_last_commit = @project.last_commit_for(ref, path).sha
|
||||
last_commit == current_last_commit
|
||||
end
|
||||
end
|
||||
end
|
76
lib/gitlab/git_stats.rb
Normal file
76
lib/gitlab/git_stats.rb
Normal file
|
@ -0,0 +1,76 @@
|
|||
module Gitlab
|
||||
class GitStats
|
||||
attr_accessor :repo, :ref
|
||||
|
||||
def initialize repo, ref
|
||||
@repo, @ref = repo, ref
|
||||
end
|
||||
|
||||
def authors
|
||||
@authors ||= collect_authors
|
||||
end
|
||||
|
||||
def commits_count
|
||||
@commits_count ||= repo.commit_count(ref)
|
||||
end
|
||||
|
||||
def files_count
|
||||
repo.git.sh("git ls-tree -r --name-only #{ref} | wc -l").first.to_i
|
||||
end
|
||||
|
||||
def authors_count
|
||||
authors.size
|
||||
end
|
||||
|
||||
def graph
|
||||
@graph ||= build_graph
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def collect_authors
|
||||
shortlog = repo.git.shortlog({e: true, s: true }, ref)
|
||||
|
||||
authors = []
|
||||
|
||||
lines = shortlog.split("\n")
|
||||
|
||||
lines.each do |line|
|
||||
data = line.split("\t")
|
||||
commits = data.first
|
||||
author = Grit::Actor.from_string(data.last)
|
||||
|
||||
authors << OpenStruct.new(
|
||||
name: author.name,
|
||||
email: author.email,
|
||||
commits: commits.to_i
|
||||
)
|
||||
end
|
||||
|
||||
authors.sort_by(&:commits).reverse
|
||||
end
|
||||
|
||||
def build_graph n = 4
|
||||
from, to = (Date.today - n.weeks), Date.today
|
||||
|
||||
format = "--pretty=format:'%h|%at|%ai|%aE'"
|
||||
commits_strings = repo.git.sh("git rev-list --since #{from.to_s(:date)} #{format} #{ref} | grep -v commit")[0].split("\n")
|
||||
|
||||
commits_dates = commits_strings.map do |string|
|
||||
data = string.split("|")
|
||||
date = data[2]
|
||||
Time.parse(date).to_date.to_s(:date)
|
||||
end
|
||||
|
||||
commits_per_day = from.upto(to).map do |day|
|
||||
commits_dates.count(day.to_date.to_s(:date))
|
||||
end
|
||||
|
||||
OpenStruct.new(
|
||||
labels: from.upto(to).map { |day| day.stamp('Aug 23') },
|
||||
commits: commits_per_day,
|
||||
weeks: n
|
||||
)
|
||||
end
|
||||
end
|
||||
end
|
48
lib/gitlab/graph/commit.rb
Normal file
48
lib/gitlab/graph/commit.rb
Normal file
|
@ -0,0 +1,48 @@
|
|||
require "grit"
|
||||
|
||||
module Gitlab
|
||||
module Graph
|
||||
class Commit
|
||||
include ActionView::Helpers::TagHelper
|
||||
|
||||
attr_accessor :time, :space, :refs
|
||||
|
||||
def initialize(commit)
|
||||
@_commit = commit
|
||||
@time = -1
|
||||
@space = 0
|
||||
end
|
||||
|
||||
def method_missing(m, *args, &block)
|
||||
@_commit.send(m, *args, &block)
|
||||
end
|
||||
|
||||
def to_graph_hash
|
||||
h = {}
|
||||
h[:parents] = self.parents.collect do |p|
|
||||
[p.id,0,0]
|
||||
end
|
||||
h[:author] = author.name
|
||||
h[:time] = time
|
||||
h[:space] = space
|
||||
h[:refs] = refs.collect{|r|r.name}.join(" ") unless refs.nil?
|
||||
h[:id] = sha
|
||||
h[:date] = date
|
||||
h[:message] = escape_once(message)
|
||||
h[:login] = author.email
|
||||
h
|
||||
end
|
||||
|
||||
def add_refs(ref_cache, repo)
|
||||
if ref_cache.empty?
|
||||
repo.refs.each do |ref|
|
||||
ref_cache[ref.commit.id] ||= []
|
||||
ref_cache[ref.commit.id] << ref
|
||||
end
|
||||
end
|
||||
@refs = ref_cache[@_commit.id] if ref_cache.include?(@_commit.id)
|
||||
@refs ||= []
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
172
lib/gitlab/graph/json_builder.rb
Normal file
172
lib/gitlab/graph/json_builder.rb
Normal file
|
@ -0,0 +1,172 @@
|
|||
require "grit"
|
||||
|
||||
module Gitlab
|
||||
module Graph
|
||||
class JsonBuilder
|
||||
attr_accessor :days, :commits, :ref_cache, :repo
|
||||
|
||||
def self.max_count
|
||||
@max_count ||= 650
|
||||
end
|
||||
|
||||
def initialize project
|
||||
@project = project
|
||||
@repo = project.repo
|
||||
@ref_cache = {}
|
||||
|
||||
@commits = collect_commits
|
||||
@days = index_commits
|
||||
end
|
||||
|
||||
def days_json
|
||||
@days_json = @days.compact.map { |d| [d.day, d.strftime("%b")] }.to_json
|
||||
end
|
||||
|
||||
def commits_json
|
||||
@commits_json = @commits.map(&:to_graph_hash).to_json
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
# Get commits from repository
|
||||
#
|
||||
def collect_commits
|
||||
@commits = Grit::Commit.find_all(repo, nil, {max_count: self.class.max_count}).dup
|
||||
|
||||
# Decorate with app/models/commit.rb
|
||||
@commits.map! { |commit| ::Commit.new(commit) }
|
||||
|
||||
# Decorate with lib/gitlab/graph/commit.rb
|
||||
@commits.map! { |commit| Gitlab::Graph::Commit.new(commit) }
|
||||
|
||||
# add refs to each commit
|
||||
@commits.each { |commit| commit.add_refs(ref_cache, repo) }
|
||||
|
||||
@commits
|
||||
end
|
||||
|
||||
# Method is adding time and space on the
|
||||
# list of commits. As well as returns date list
|
||||
# corelated with time set on commits.
|
||||
#
|
||||
# @param [Array<Graph::Commit>] comits to index
|
||||
#
|
||||
# @return [Array<TimeDate>] list of commit dates corelated with time on commits
|
||||
def index_commits
|
||||
days, heads = [], []
|
||||
map = {}
|
||||
|
||||
commits.reverse.each_with_index do |c,i|
|
||||
c.time = i
|
||||
days[i] = c.committed_date
|
||||
map[c.id] = c
|
||||
heads += c.refs unless c.refs.nil?
|
||||
end
|
||||
|
||||
heads.select!{|h| h.is_a? Grit::Head or h.is_a? Grit::Remote}
|
||||
# sort heads so the master is top and current branches are closer
|
||||
heads.sort! do |a,b|
|
||||
if a.name == "master"
|
||||
-1
|
||||
elsif b.name == "master"
|
||||
1
|
||||
else
|
||||
b.commit.committed_date <=> a.commit.committed_date
|
||||
end
|
||||
end
|
||||
|
||||
@_reserved = {}
|
||||
days.each_index do |i|
|
||||
@_reserved[i] = []
|
||||
end
|
||||
|
||||
heads.each do |h|
|
||||
if map.include? h.commit.id then
|
||||
place_chain(map[h.commit.id], map)
|
||||
end
|
||||
end
|
||||
|
||||
days
|
||||
end
|
||||
|
||||
# Add space mark on commit and its parents
|
||||
#
|
||||
# @param [Graph::Commit] the commit object.
|
||||
# @param [Hash<String,Graph::Commit>] map of commits
|
||||
def place_chain(commit, map, parent_time = nil)
|
||||
leaves = take_left_leaves(commit, map)
|
||||
if leaves.empty?
|
||||
return
|
||||
end
|
||||
space = find_free_space(leaves.last.time..leaves.first.time)
|
||||
leaves.each{|l| l.space = space}
|
||||
# and mark it as reserved
|
||||
min_time = leaves.last.time
|
||||
parents = leaves.last.parents.collect
|
||||
parents.each do |p|
|
||||
if map.include? p.id
|
||||
parent = map[p.id]
|
||||
if parent.time < min_time
|
||||
min_time = parent.time
|
||||
end
|
||||
end
|
||||
end
|
||||
if parent_time.nil?
|
||||
max_time = leaves.first.time
|
||||
else
|
||||
max_time = parent_time - 1
|
||||
end
|
||||
mark_reserved(min_time..max_time, space)
|
||||
|
||||
# Visit branching chains
|
||||
leaves.each do |l|
|
||||
parents = l.parents.collect.select{|p| map.include? p.id and map[p.id].space == 0}
|
||||
for p in parents
|
||||
place_chain(map[p.id], map, l.time)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def mark_reserved(time_range, space)
|
||||
for day in time_range
|
||||
@_reserved[day].push(space)
|
||||
end
|
||||
end
|
||||
|
||||
def find_free_space(time_range)
|
||||
reserved = []
|
||||
for day in time_range
|
||||
reserved += @_reserved[day]
|
||||
end
|
||||
space = 1
|
||||
while reserved.include? space do
|
||||
space += 1
|
||||
end
|
||||
space
|
||||
end
|
||||
|
||||
# Takes most left subtree branch of commits
|
||||
# which don't have space mark yet.
|
||||
#
|
||||
# @param [Graph::Commit] the commit object.
|
||||
# @param [Hash<String,Graph::Commit>] map of commits
|
||||
#
|
||||
# @return [Array<Graph::Commit>] list of branch commits
|
||||
def take_left_leaves(commit, map)
|
||||
leaves = []
|
||||
leaves.push(commit) if commit.space.zero?
|
||||
|
||||
while true
|
||||
parent = commit.parents.collect.select do |p|
|
||||
map.include? p.id and map[p.id].space == 0
|
||||
end
|
||||
|
||||
return leaves if parent.count.zero?
|
||||
|
||||
commit = map[parent.first.id]
|
||||
leaves.push(commit)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
|
@ -1,185 +0,0 @@
|
|||
require "grit"
|
||||
|
||||
module Gitlab
|
||||
class GraphCommit
|
||||
attr_accessor :time, :space
|
||||
attr_accessor :refs
|
||||
|
||||
include ActionView::Helpers::TagHelper
|
||||
|
||||
def self.to_graph(project)
|
||||
@repo = project.repo
|
||||
commits = Grit::Commit.find_all(@repo, nil, {max_count: 650})
|
||||
|
||||
ref_cache = {}
|
||||
|
||||
commits.map! {|c| GraphCommit.new(Commit.new(c))}
|
||||
commits.each { |commit| commit.add_refs(ref_cache, @repo) }
|
||||
|
||||
days = GraphCommit.index_commits(commits)
|
||||
@days_json = days.compact.collect{|d| [d.day, d.strftime("%b")] }.to_json
|
||||
@commits_json = commits.map(&:to_graph_hash).to_json
|
||||
|
||||
return @days_json, @commits_json
|
||||
end
|
||||
|
||||
# Method is adding time and space on the
|
||||
# list of commits. As well as returns date list
|
||||
# corelated with time set on commits.
|
||||
#
|
||||
# @param [Array<GraphCommit>] comits to index
|
||||
#
|
||||
# @return [Array<TimeDate>] list of commit dates corelated with time on commits
|
||||
def self.index_commits(commits)
|
||||
days, heads = [], []
|
||||
map = {}
|
||||
|
||||
commits.reverse.each_with_index do |c,i|
|
||||
c.time = i
|
||||
days[i] = c.committed_date
|
||||
map[c.id] = c
|
||||
heads += c.refs unless c.refs.nil?
|
||||
end
|
||||
|
||||
heads.select!{|h| h.is_a? Grit::Head or h.is_a? Grit::Remote}
|
||||
# sort heads so the master is top and current branches are closer
|
||||
heads.sort! do |a,b|
|
||||
if a.name == "master"
|
||||
-1
|
||||
elsif b.name == "master"
|
||||
1
|
||||
else
|
||||
b.commit.committed_date <=> a.commit.committed_date
|
||||
end
|
||||
end
|
||||
|
||||
@_reserved = {}
|
||||
days.each_index do |i|
|
||||
@_reserved[i] = []
|
||||
end
|
||||
|
||||
heads.each do |h|
|
||||
if map.include? h.commit.id then
|
||||
place_chain(map[h.commit.id], map)
|
||||
end
|
||||
end
|
||||
days
|
||||
end
|
||||
|
||||
# Add space mark on commit and its parents
|
||||
#
|
||||
# @param [GraphCommit] the commit object.
|
||||
# @param [Hash<String,GraphCommit>] map of commits
|
||||
def self.place_chain(commit, map, parent_time = nil)
|
||||
leaves = take_left_leaves(commit, map)
|
||||
if leaves.empty? then
|
||||
return
|
||||
end
|
||||
space = find_free_space(leaves.last.time..leaves.first.time)
|
||||
leaves.each{|l| l.space = space}
|
||||
# and mark it as reserved
|
||||
min_time = leaves.last.time
|
||||
parents = leaves.last.parents.collect
|
||||
parents.each do |p|
|
||||
if map.include? p.id then
|
||||
parent = map[p.id]
|
||||
if parent.time < min_time then
|
||||
min_time = parent.time
|
||||
end
|
||||
end
|
||||
end
|
||||
if parent_time.nil? then
|
||||
max_time = leaves.first.time
|
||||
else
|
||||
max_time = parent_time - 1
|
||||
end
|
||||
mark_reserved(min_time..max_time, space)
|
||||
# Visit branching chains
|
||||
leaves.each do |l|
|
||||
parents = l.parents.collect
|
||||
.select{|p| map.include? p.id and map[p.id].space == 0}
|
||||
for p in parents
|
||||
place_chain(map[p.id], map, l.time)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def self.mark_reserved(time_range, space)
|
||||
for day in time_range
|
||||
@_reserved[day].push(space)
|
||||
end
|
||||
end
|
||||
|
||||
def self.find_free_space(time_range)
|
||||
reserved = []
|
||||
for day in time_range
|
||||
reserved += @_reserved[day]
|
||||
end
|
||||
space = 1
|
||||
while reserved.include? space do
|
||||
space += 1
|
||||
end
|
||||
space
|
||||
end
|
||||
|
||||
# Takes most left subtree branch of commits
|
||||
# which don't have space mark yet.
|
||||
#
|
||||
# @param [GraphCommit] the commit object.
|
||||
# @param [Hash<String,GraphCommit>] map of commits
|
||||
#
|
||||
# @return [Array<GraphCommit>] list of branch commits
|
||||
def self.take_left_leaves(commit, map)
|
||||
leaves = []
|
||||
leaves.push(commit) if commit.space == 0
|
||||
while true
|
||||
parent = commit.parents.collect
|
||||
.select{|p| map.include? p.id and map[p.id].space == 0}
|
||||
if parent.count == 0 then
|
||||
return leaves
|
||||
else
|
||||
commit = map[parent.first.id]
|
||||
leaves.push(commit)
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
|
||||
def initialize(commit)
|
||||
@_commit = commit
|
||||
@time = -1
|
||||
@space = 0
|
||||
end
|
||||
|
||||
def method_missing(m, *args, &block)
|
||||
@_commit.send(m, *args, &block)
|
||||
end
|
||||
|
||||
def to_graph_hash
|
||||
h = {}
|
||||
h[:parents] = self.parents.collect do |p|
|
||||
[p.id,0,0]
|
||||
end
|
||||
h[:author] = Gitlab::Encode.utf8(author.name)
|
||||
h[:time] = time
|
||||
h[:space] = space
|
||||
h[:refs] = refs.collect{|r|r.name}.join(" ") unless refs.nil?
|
||||
h[:id] = sha
|
||||
h[:date] = date
|
||||
h[:message] = escape_once(Gitlab::Encode.utf8(message))
|
||||
h[:login] = author.email
|
||||
h
|
||||
end
|
||||
|
||||
def add_refs(ref_cache, repo)
|
||||
if ref_cache.empty?
|
||||
repo.refs.each do |ref|
|
||||
ref_cache[ref.commit.id] ||= []
|
||||
ref_cache[ref.commit.id] << ref
|
||||
end
|
||||
end
|
||||
@refs = ref_cache[@_commit.id] if ref_cache.include?(@_commit.id)
|
||||
@refs ||= []
|
||||
end
|
||||
end
|
||||
end
|
|
@ -1,106 +0,0 @@
|
|||
module Gitlab
|
||||
class Merge
|
||||
attr_accessor :merge_request, :project, :user
|
||||
|
||||
def initialize(merge_request, user)
|
||||
@merge_request = merge_request
|
||||
@project = merge_request.project
|
||||
@user = user
|
||||
end
|
||||
|
||||
def can_be_merged?
|
||||
in_locked_and_timed_satellite do |merge_repo|
|
||||
merge_in_satellite!(merge_repo)
|
||||
end
|
||||
end
|
||||
|
||||
# Merges the source branch into the target branch in the satellite and
|
||||
# pushes it back to Gitolite.
|
||||
# It also removes the source branch if requested in the merge request.
|
||||
#
|
||||
# Returns false if the merge produced conflicts
|
||||
# Returns false if pushing from the satallite to Gitolite failed or was rejected
|
||||
# Returns true otherwise
|
||||
def merge!
|
||||
in_locked_and_timed_satellite do |merge_repo|
|
||||
if merge_in_satellite!(merge_repo)
|
||||
# push merge back to Gitolite
|
||||
# will raise CommandFailed when push fails
|
||||
merge_repo.git.push({raise: true}, :origin, merge_request.target_branch)
|
||||
|
||||
# remove source branch
|
||||
if merge_request.should_remove_source_branch && !project.root_ref?(merge_request.source_branch)
|
||||
# will raise CommandFailed when push fails
|
||||
merge_repo.git.push({raise: true}, :origin, ":#{merge_request.source_branch}")
|
||||
end
|
||||
|
||||
# merge, push and branch removal successful
|
||||
true
|
||||
end
|
||||
end
|
||||
rescue Grit::Git::CommandFailed
|
||||
false
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
# * Sets a 30s timeout for Git
|
||||
# * Locks the satellite repo
|
||||
# * Yields the prepared satallite repo
|
||||
def in_locked_and_timed_satellite
|
||||
Grit::Git.with_timeout(30.seconds) do
|
||||
lock_file = Rails.root.join("tmp", "#{project.path}.lock")
|
||||
|
||||
File.open(lock_file, "w+") do |f|
|
||||
f.flock(File::LOCK_EX)
|
||||
|
||||
unless project.satellite.exists?
|
||||
raise "Satellite doesn't exist"
|
||||
end
|
||||
|
||||
Dir.chdir(project.satellite.path) do
|
||||
repo = Grit::Repo.new('.')
|
||||
|
||||
return yield repo
|
||||
end
|
||||
end
|
||||
end
|
||||
rescue Errno::ENOMEM => ex
|
||||
Gitlab::GitLogger.error(ex.message)
|
||||
rescue Grit::Git::GitTimeout
|
||||
return false
|
||||
end
|
||||
|
||||
# Merges the source_branch into the target_branch in the satellite.
|
||||
#
|
||||
# Note: it will clear out the satellite before doing anything
|
||||
#
|
||||
# Returns false if the merge produced conflicts
|
||||
# Returns true otherwise
|
||||
def merge_in_satellite!(repo)
|
||||
prepare_satelite!(repo)
|
||||
|
||||
# create target branch in satellite at the corresponding commit from Gitolite
|
||||
repo.git.checkout({b: true}, merge_request.target_branch, "origin/#{merge_request.target_branch}")
|
||||
|
||||
# merge the source branch from Gitolite into the satellite
|
||||
# will raise CommandFailed when merge fails
|
||||
repo.git.pull({no_ff: true, raise: true}, :origin, merge_request.source_branch)
|
||||
rescue Grit::Git::CommandFailed
|
||||
false
|
||||
end
|
||||
|
||||
# * Clears the satellite
|
||||
# * Updates the satellite from Gitolite
|
||||
# * Sets up Git variables for the user
|
||||
def prepare_satelite!(repo)
|
||||
project.satellite.clear
|
||||
|
||||
repo.git.reset(hard: true)
|
||||
repo.git.fetch({}, :origin)
|
||||
|
||||
repo.git.config({}, "user.name", user.name)
|
||||
repo.git.config({}, "user.email", user.email)
|
||||
end
|
||||
end
|
||||
end
|
|
@ -1,41 +0,0 @@
|
|||
module Gitlab
|
||||
class Satellite
|
||||
|
||||
PARKING_BRANCH = "__parking_branch"
|
||||
|
||||
attr_accessor :project
|
||||
|
||||
def initialize project
|
||||
self.project = project
|
||||
end
|
||||
|
||||
def create
|
||||
`git clone #{project.url_to_repo} #{path}`
|
||||
end
|
||||
|
||||
def path
|
||||
Rails.root.join("tmp", "repo_satellites", project.path)
|
||||
end
|
||||
|
||||
def exists?
|
||||
File.exists? path
|
||||
end
|
||||
|
||||
#will be deleted all branches except PARKING_BRANCH
|
||||
def clear
|
||||
Dir.chdir(path) do
|
||||
heads = Grit::Repo.new(".").heads.map{|head| head.name}
|
||||
if heads.include? PARKING_BRANCH
|
||||
`git checkout #{PARKING_BRANCH}`
|
||||
else
|
||||
`git checkout -b #{PARKING_BRANCH}`
|
||||
end
|
||||
heads.delete(PARKING_BRANCH)
|
||||
heads.each do |head|
|
||||
`git branch -D #{head}`
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
end
|
46
lib/gitlab/satellite/action.rb
Normal file
46
lib/gitlab/satellite/action.rb
Normal file
|
@ -0,0 +1,46 @@
|
|||
module Gitlab
|
||||
module Satellite
|
||||
class Action
|
||||
DEFAULT_OPTIONS = { git_timeout: 30.seconds }
|
||||
|
||||
attr_accessor :options, :project, :user
|
||||
|
||||
def initialize(user, project, options = {})
|
||||
@options = DEFAULT_OPTIONS.merge(options)
|
||||
@project = project
|
||||
@user = user
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
# * Sets a 30s timeout for Git
|
||||
# * Locks the satellite repo
|
||||
# * Yields the prepared satellite repo
|
||||
def in_locked_and_timed_satellite
|
||||
Grit::Git.with_timeout(options[:git_timeout]) do
|
||||
project.satellite.lock do
|
||||
return yield project.satellite.repo
|
||||
end
|
||||
end
|
||||
rescue Errno::ENOMEM => ex
|
||||
Gitlab::GitLogger.error(ex.message)
|
||||
return false
|
||||
rescue Grit::Git::GitTimeout => ex
|
||||
Gitlab::GitLogger.error(ex.message)
|
||||
return false
|
||||
end
|
||||
|
||||
# * Clears the satellite
|
||||
# * Updates the satellite from Gitolite
|
||||
# * Sets up Git variables for the user
|
||||
#
|
||||
# Note: use this within #in_locked_and_timed_satellite
|
||||
def prepare_satellite!(repo)
|
||||
project.satellite.clear_and_update!
|
||||
|
||||
repo.git.config({}, "user.name", user.name)
|
||||
repo.git.config({}, "user.email", user.email)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
57
lib/gitlab/satellite/edit_file_action.rb
Normal file
57
lib/gitlab/satellite/edit_file_action.rb
Normal file
|
@ -0,0 +1,57 @@
|
|||
module Gitlab
|
||||
module Satellite
|
||||
# GitLab server-side file update and commit
|
||||
class EditFileAction < Action
|
||||
attr_accessor :file_path, :ref
|
||||
|
||||
def initialize(user, project, ref, file_path)
|
||||
super user, project, git_timeout: 10.seconds
|
||||
@file_path = file_path
|
||||
@ref = ref
|
||||
end
|
||||
|
||||
# Updates the files content and creates a new commit for it
|
||||
#
|
||||
# Returns false if the ref has been updated while editing the file
|
||||
# Returns false if commiting the change fails
|
||||
# Returns false if pushing from the satellite to Gitolite failed or was rejected
|
||||
# Returns true otherwise
|
||||
def commit!(content, commit_message, last_commit)
|
||||
return false unless can_edit?(last_commit)
|
||||
|
||||
in_locked_and_timed_satellite do |repo|
|
||||
prepare_satellite!(repo)
|
||||
|
||||
# create target branch in satellite at the corresponding commit from Gitolite
|
||||
repo.git.checkout({raise: true, timeout: true, b: true}, ref, "origin/#{ref}")
|
||||
|
||||
# update the file in the satellite's working dir
|
||||
file_path_in_satellite = File.join(repo.working_dir, file_path)
|
||||
File.open(file_path_in_satellite, 'w') { |f| f.write(content) }
|
||||
|
||||
# commit the changes
|
||||
# will raise CommandFailed when commit fails
|
||||
repo.git.commit(raise: true, timeout: true, a: true, m: commit_message)
|
||||
|
||||
|
||||
# push commit back to Gitolite
|
||||
# will raise CommandFailed when push fails
|
||||
repo.git.push({raise: true, timeout: true}, :origin, ref)
|
||||
|
||||
# everything worked
|
||||
true
|
||||
end
|
||||
rescue Grit::Git::CommandFailed => ex
|
||||
Gitlab::GitLogger.error(ex.message)
|
||||
false
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def can_edit?(last_commit)
|
||||
current_last_commit = @project.last_commit_for(ref, file_path).sha
|
||||
last_commit == current_last_commit
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
71
lib/gitlab/satellite/merge_action.rb
Normal file
71
lib/gitlab/satellite/merge_action.rb
Normal file
|
@ -0,0 +1,71 @@
|
|||
module Gitlab
|
||||
module Satellite
|
||||
# GitLab server-side merge
|
||||
class MergeAction < Action
|
||||
attr_accessor :merge_request
|
||||
|
||||
def initialize(user, merge_request)
|
||||
super user, merge_request.project
|
||||
@merge_request = merge_request
|
||||
end
|
||||
|
||||
# Checks if a merge request can be executed without user interaction
|
||||
def can_be_merged?
|
||||
in_locked_and_timed_satellite do |merge_repo|
|
||||
merge_in_satellite!(merge_repo)
|
||||
end
|
||||
end
|
||||
|
||||
# Merges the source branch into the target branch in the satellite and
|
||||
# pushes it back to Gitolite.
|
||||
# It also removes the source branch if requested in the merge request.
|
||||
#
|
||||
# Returns false if the merge produced conflicts
|
||||
# Returns false if pushing from the satellite to Gitolite failed or was rejected
|
||||
# Returns true otherwise
|
||||
def merge!
|
||||
in_locked_and_timed_satellite do |merge_repo|
|
||||
if merge_in_satellite!(merge_repo)
|
||||
# push merge back to Gitolite
|
||||
# will raise CommandFailed when push fails
|
||||
merge_repo.git.push({raise: true, timeout: true}, :origin, merge_request.target_branch)
|
||||
|
||||
# remove source branch
|
||||
if merge_request.should_remove_source_branch && !project.root_ref?(merge_request.source_branch)
|
||||
# will raise CommandFailed when push fails
|
||||
merge_repo.git.push({raise: true, timeout: true}, :origin, ":#{merge_request.source_branch}")
|
||||
end
|
||||
|
||||
# merge, push and branch removal successful
|
||||
true
|
||||
end
|
||||
end
|
||||
rescue Grit::Git::CommandFailed => ex
|
||||
Gitlab::GitLogger.error(ex.message)
|
||||
false
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
# Merges the source_branch into the target_branch in the satellite.
|
||||
#
|
||||
# Note: it will clear out the satellite before doing anything
|
||||
#
|
||||
# Returns false if the merge produced conflicts
|
||||
# Returns true otherwise
|
||||
def merge_in_satellite!(repo)
|
||||
prepare_satellite!(repo)
|
||||
|
||||
# create target branch in satellite at the corresponding commit from Gitolite
|
||||
repo.git.checkout({raise: true, timeout: true, b: true}, merge_request.target_branch, "origin/#{merge_request.target_branch}")
|
||||
|
||||
# merge the source branch from Gitolite into the satellite
|
||||
# will raise CommandFailed when merge fails
|
||||
repo.git.pull({raise: true, timeout: true, no_ff: true}, :origin, merge_request.source_branch)
|
||||
rescue Grit::Git::CommandFailed => ex
|
||||
Gitlab::GitLogger.error(ex.message)
|
||||
false
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
92
lib/gitlab/satellite/satellite.rb
Normal file
92
lib/gitlab/satellite/satellite.rb
Normal file
|
@ -0,0 +1,92 @@
|
|||
module Gitlab
|
||||
module Satellite
|
||||
class Satellite
|
||||
PARKING_BRANCH = "__parking_branch"
|
||||
|
||||
attr_accessor :project
|
||||
|
||||
def initialize(project)
|
||||
@project = project
|
||||
end
|
||||
|
||||
def clear_and_update!
|
||||
raise "Satellite doesn't exist" unless exists?
|
||||
|
||||
delete_heads!
|
||||
clear_working_dir!
|
||||
update_from_source!
|
||||
end
|
||||
|
||||
def create
|
||||
`git clone #{project.url_to_repo} #{path}`
|
||||
end
|
||||
|
||||
def exists?
|
||||
File.exists? path
|
||||
end
|
||||
|
||||
# * Locks the satellite
|
||||
# * Changes the current directory to the satellite's working dir
|
||||
# * Yields
|
||||
def lock
|
||||
raise "Satellite doesn't exist" unless exists?
|
||||
|
||||
File.open(lock_file, "w+") do |f|
|
||||
f.flock(File::LOCK_EX)
|
||||
|
||||
Dir.chdir(path) do
|
||||
return yield
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def lock_file
|
||||
Rails.root.join("tmp", "#{project.path}.lock")
|
||||
end
|
||||
|
||||
def path
|
||||
Rails.root.join("tmp", "repo_satellites", project.path)
|
||||
end
|
||||
|
||||
def repo
|
||||
raise "Satellite doesn't exist" unless exists?
|
||||
|
||||
@repo ||= Grit::Repo.new(path)
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
# Clear the working directory
|
||||
def clear_working_dir!
|
||||
repo.git.reset(hard: true)
|
||||
end
|
||||
|
||||
# Deletes all branches except the parking branch
|
||||
#
|
||||
# This ensures we have no name clashes or issues updating branches when
|
||||
# working with the satellite.
|
||||
def delete_heads!
|
||||
heads = repo.heads.map(&:name)
|
||||
|
||||
# update or create the parking branch
|
||||
if heads.include? PARKING_BRANCH
|
||||
repo.git.checkout({}, PARKING_BRANCH)
|
||||
else
|
||||
repo.git.checkout({b: true}, PARKING_BRANCH)
|
||||
end
|
||||
|
||||
# remove the parking branch from the list of heads ...
|
||||
heads.delete(PARKING_BRANCH)
|
||||
# ... and delete all others
|
||||
heads.each { |head| repo.git.branch({D: true}, head) }
|
||||
end
|
||||
|
||||
# Updates the satellite from Gitolite
|
||||
#
|
||||
# Note: this will only update remote branches (i.e. origin/*)
|
||||
def update_from_source!
|
||||
repo.git.fetch({timeout: true}, :origin)
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
|
@ -10,10 +10,12 @@ class Redcarpet::Render::GitlabHTML < Redcarpet::Render::HTML
|
|||
end
|
||||
|
||||
def block_code(code, language)
|
||||
options = { options: {encoding: 'utf-8'} }
|
||||
|
||||
if Pygments::Lexer.find(language)
|
||||
Pygments.highlight(code, lexer: language, options: {encoding: 'utf-8'})
|
||||
Pygments.highlight(code, options.merge(lexer: language.downcase))
|
||||
else
|
||||
Pygments.highlight(code, options: {encoding: 'utf-8'})
|
||||
Pygments.highlight(code, options)
|
||||
end
|
||||
end
|
||||
|
||||
|
|
|
@ -1,83 +0,0 @@
|
|||
desc "Imports existing Git repos from a directory into new projects in git_base_path"
|
||||
task :import_projects, [:directory,:email] => :environment do |t, args|
|
||||
user_email, import_directory = args.email, args.directory
|
||||
repos_to_import = Dir.glob("#{import_directory}/*")
|
||||
git_base_path = Gitlab.config.git_base_path
|
||||
imported_count, skipped_count, failed_count = 0
|
||||
|
||||
puts "Found #{repos_to_import.size} repos to import"
|
||||
|
||||
repos_to_import.each do |repo_path|
|
||||
repo_name = File.basename repo_path
|
||||
clone_path = "#{git_base_path}#{repo_name}.git"
|
||||
|
||||
puts " Processing #{repo_name}"
|
||||
|
||||
if Dir.exists? clone_path
|
||||
if Project.find_by_code(repo_name)
|
||||
puts " INFO: #{clone_path} already exists in repositories directory, skipping."
|
||||
skipped_count += 1
|
||||
next
|
||||
else
|
||||
puts " INFO: Project doesn't exist for #{repo_name} (but the repo does)."
|
||||
end
|
||||
else
|
||||
# Clone the repo
|
||||
unless clone_bare_repo_as_git(repo_path, clone_path)
|
||||
failed_count += 1
|
||||
next
|
||||
end
|
||||
end
|
||||
|
||||
# Create the project and repo
|
||||
if create_repo_project(repo_name, user_email)
|
||||
imported_count += 1
|
||||
else
|
||||
failed_count += 1
|
||||
end
|
||||
end
|
||||
|
||||
puts "Finished importing #{imported_count} projects (skipped #{skipped_count}, failed #{failed_count})."
|
||||
end
|
||||
|
||||
# Clones a repo as bare git repo using the git_user
|
||||
def clone_bare_repo_as_git(existing_path, new_path)
|
||||
git_user = Gitlab.config.ssh_user
|
||||
begin
|
||||
sh "sudo -u #{git_user} -i git clone --bare '#{existing_path}' #{new_path}"
|
||||
rescue Exception => msg
|
||||
puts " ERROR: Failed to clone #{existing_path} to #{new_path}"
|
||||
puts " Make sure #{git_user} can reach #{existing_path}"
|
||||
puts " Exception-MSG: #{msg}"
|
||||
end
|
||||
end
|
||||
|
||||
# Creates a project in GitLab given a `project_name` to use
|
||||
# (for name, web url, and code url) and a `user_email` that will be
|
||||
# assigned as the owner of the project.
|
||||
def create_repo_project(project_name, user_email)
|
||||
if user = User.find_by_email(user_email)
|
||||
# Using find_by_code since that's the most important identifer to be unique
|
||||
if Project.find_by_code(project_name)
|
||||
puts " INFO: Project #{project_name} already exists in Gitlab, skipping."
|
||||
else
|
||||
project = Project.create(
|
||||
name: project_name,
|
||||
code: project_name,
|
||||
path: project_name,
|
||||
owner: user,
|
||||
description: "Automatically created from 'import_projects' rake task on #{Time.now}"
|
||||
)
|
||||
|
||||
if project.valid?
|
||||
# Add user as admin for project
|
||||
project.users_projects.create!(:project_access => UsersProject::MASTER, :user => user)
|
||||
project.update_repository
|
||||
else
|
||||
puts " ERROR: Failed to create project #{project} because #{project.errors.first}"
|
||||
end
|
||||
end
|
||||
else
|
||||
puts " ERROR: user with #{user_email} not found, skipping"
|
||||
end
|
||||
end
|
54
lib/tasks/gitlab/import.rake
Normal file
54
lib/tasks/gitlab/import.rake
Normal file
|
@ -0,0 +1,54 @@
|
|||
namespace :gitlab do
|
||||
namespace :import do
|
||||
# How to use:
|
||||
#
|
||||
# 1. copy your bare repos under git base_path
|
||||
# 2. run bundle exec rake gitlab:import:repos RAILS_ENV=production
|
||||
#
|
||||
# Notes:
|
||||
# * project owner will be a first admin
|
||||
# * existing projects will be skipped
|
||||
#
|
||||
desc "GITLAB | Import bare repositories from git_host -> base_path into GitLab project instance"
|
||||
task :repos => :environment do
|
||||
|
||||
git_base_path = Gitlab.config.git_base_path
|
||||
repos_to_import = Dir.glob(git_base_path + '/*')
|
||||
|
||||
repos_to_import.each do |repo_path|
|
||||
repo_name = File.basename repo_path
|
||||
|
||||
# skip gitolite admin
|
||||
next if repo_name == 'gitolite-admin.git'
|
||||
|
||||
path = repo_name.sub(/\.git$/, '')
|
||||
|
||||
project = Project.find_by_path(path)
|
||||
|
||||
puts "Processing #{repo_name}".yellow
|
||||
|
||||
if project
|
||||
puts " * #{project.name} (#{repo_name}) exists"
|
||||
else
|
||||
user = User.admins.first
|
||||
|
||||
project_params = {
|
||||
:name => path,
|
||||
:code => path,
|
||||
:path => path,
|
||||
}
|
||||
|
||||
project = Project.create_by_user(project_params, user)
|
||||
|
||||
if project.valid?
|
||||
puts " * Created #{project.name} (#{repo_name})".green
|
||||
else
|
||||
puts " * Failed trying to create #{project.name} (#{repo_name})".red
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
puts "Done!".green
|
||||
end
|
||||
end
|
||||
end
|
|
@ -65,7 +65,7 @@ namespace :gitlab do
|
|||
end
|
||||
|
||||
print "UMASK for .gitolite.rc is 0007? ............"
|
||||
if open("#{git_base_path}/../.gitolite.rc").grep(/UMASK([ \t]*)=([ \t>]*)0007/).any?
|
||||
if open(File.absolute_path("#{git_base_path}/../.gitolite.rc")).grep(/UMASK([ \t]*)=([ \t>]*)0007/).any?
|
||||
puts "YES".green
|
||||
else
|
||||
puts "NO".red
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue