To check out this repository please hg clone the following URL, or open the URL using EasyMercurial or your preferred Mercurial client.
root / .svn / pristine / 55 / 550aea64d9d567b8fd19eb6154f19d0a52250270.svn-base @ 1297:0a574315af3e
History | View | Annotate | Download (6 KB)
| 1 |
# Redmine - project management software |
|---|---|
| 2 |
# Copyright (C) 2006-2011 Jean-Philippe Lang |
| 3 |
# Copyright (C) 2007 Patrick Aljord patcito@ŋmail.com |
| 4 |
# |
| 5 |
# This program is free software; you can redistribute it and/or |
| 6 |
# modify it under the terms of the GNU General Public License |
| 7 |
# as published by the Free Software Foundation; either version 2 |
| 8 |
# of the License, or (at your option) any later version. |
| 9 |
# |
| 10 |
# This program is distributed in the hope that it will be useful, |
| 11 |
# but WITHOUT ANY WARRANTY; without even the implied warranty of |
| 12 |
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
| 13 |
# GNU General Public License for more details. |
| 14 |
# |
| 15 |
# You should have received a copy of the GNU General Public License |
| 16 |
# along with this program; if not, write to the Free Software |
| 17 |
# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. |
| 18 |
|
| 19 |
require 'redmine/scm/adapters/git_adapter' |
| 20 |
|
| 21 |
class Repository::Git < Repository |
| 22 |
attr_protected :root_url |
| 23 |
validates_presence_of :url |
| 24 |
|
| 25 |
def self.human_attribute_name(attribute_key_name) |
| 26 |
attr_name = attribute_key_name |
| 27 |
if attr_name == "url" |
| 28 |
attr_name = "path_to_repository" |
| 29 |
end |
| 30 |
super(attr_name) |
| 31 |
end |
| 32 |
|
| 33 |
def self.scm_adapter_class |
| 34 |
Redmine::Scm::Adapters::GitAdapter |
| 35 |
end |
| 36 |
|
| 37 |
def self.scm_name |
| 38 |
'Git' |
| 39 |
end |
| 40 |
|
| 41 |
def report_last_commit |
| 42 |
extra_report_last_commit |
| 43 |
end |
| 44 |
|
| 45 |
def extra_report_last_commit |
| 46 |
return false if extra_info.nil? |
| 47 |
v = extra_info["extra_report_last_commit"] |
| 48 |
return false if v.nil? |
| 49 |
v.to_s != '0' |
| 50 |
end |
| 51 |
|
| 52 |
def supports_directory_revisions? |
| 53 |
true |
| 54 |
end |
| 55 |
|
| 56 |
def supports_revision_graph? |
| 57 |
true |
| 58 |
end |
| 59 |
|
| 60 |
def repo_log_encoding |
| 61 |
'UTF-8' |
| 62 |
end |
| 63 |
|
| 64 |
# Returns the identifier for the given git changeset |
| 65 |
def self.changeset_identifier(changeset) |
| 66 |
changeset.scmid |
| 67 |
end |
| 68 |
|
| 69 |
# Returns the readable identifier for the given git changeset |
| 70 |
def self.format_changeset_identifier(changeset) |
| 71 |
changeset.revision[0, 8] |
| 72 |
end |
| 73 |
|
| 74 |
def branches |
| 75 |
scm.branches |
| 76 |
end |
| 77 |
|
| 78 |
def tags |
| 79 |
scm.tags |
| 80 |
end |
| 81 |
|
| 82 |
def default_branch |
| 83 |
scm.default_branch |
| 84 |
rescue Exception => e |
| 85 |
logger.error "git: error during get default branch: #{e.message}"
|
| 86 |
nil |
| 87 |
end |
| 88 |
|
| 89 |
def find_changeset_by_name(name) |
| 90 |
return nil if name.nil? || name.empty? |
| 91 |
e = changesets.find(:first, :conditions => ['revision = ?', name.to_s]) |
| 92 |
return e if e |
| 93 |
changesets.find(:first, :conditions => ['scmid LIKE ?', "#{name}%"])
|
| 94 |
end |
| 95 |
|
| 96 |
def entries(path=nil, identifier=nil) |
| 97 |
scm.entries(path, |
| 98 |
identifier, |
| 99 |
options = {:report_last_commit => extra_report_last_commit})
|
| 100 |
end |
| 101 |
|
| 102 |
# With SCMs that have a sequential commit numbering, |
| 103 |
# such as Subversion and Mercurial, |
| 104 |
# Redmine is able to be clever and only fetch changesets |
| 105 |
# going forward from the most recent one it knows about. |
| 106 |
# |
| 107 |
# However, Git does not have a sequential commit numbering. |
| 108 |
# |
| 109 |
# In order to fetch only new adding revisions, |
| 110 |
# Redmine needs to parse revisions per branch. |
| 111 |
# Branch "last_scmid" is for this requirement. |
| 112 |
# |
| 113 |
# In Git and Mercurial, revisions are not in date order. |
| 114 |
# Redmine Mercurial fixed issues. |
| 115 |
# * Redmine Takes Too Long On Large Mercurial Repository |
| 116 |
# http://www.redmine.org/issues/3449 |
| 117 |
# * Sorting for changesets might go wrong on Mercurial repos |
| 118 |
# http://www.redmine.org/issues/3567 |
| 119 |
# |
| 120 |
# Database revision column is text, so Redmine can not sort by revision. |
| 121 |
# Mercurial has revision number, and revision number guarantees revision order. |
| 122 |
# Redmine Mercurial model stored revisions ordered by database id to database. |
| 123 |
# So, Redmine Mercurial model can use correct ordering revisions. |
| 124 |
# |
| 125 |
# Redmine Mercurial adapter uses "hg log -r 0:tip --limit 10" |
| 126 |
# to get limited revisions from old to new. |
| 127 |
# But, Git 1.7.3.4 does not support --reverse with -n or --skip. |
| 128 |
# |
| 129 |
# The repository can still be fully reloaded by calling #clear_changesets |
| 130 |
# before fetching changesets (eg. for offline resync) |
| 131 |
def fetch_changesets |
| 132 |
scm_brs = branches |
| 133 |
return if scm_brs.nil? || scm_brs.empty? |
| 134 |
h1 = extra_info || {}
|
| 135 |
h = h1.dup |
| 136 |
h["branches"] ||= {}
|
| 137 |
h["db_consistent"] ||= {}
|
| 138 |
if changesets.count == 0 |
| 139 |
h["db_consistent"]["ordering"] = 1 |
| 140 |
merge_extra_info(h) |
| 141 |
self.save |
| 142 |
elsif ! h["db_consistent"].has_key?("ordering")
|
| 143 |
h["db_consistent"]["ordering"] = 0 |
| 144 |
merge_extra_info(h) |
| 145 |
self.save |
| 146 |
end |
| 147 |
scm_brs.each do |br1| |
| 148 |
br = br1.to_s |
| 149 |
from_scmid = nil |
| 150 |
from_scmid = h["branches"][br]["last_scmid"] if h["branches"][br] |
| 151 |
h["branches"][br] ||= {}
|
| 152 |
scm.revisions('', from_scmid, br, {:reverse => true}) do |rev|
|
| 153 |
db_rev = find_changeset_by_name(rev.revision) |
| 154 |
transaction do |
| 155 |
if db_rev.nil? |
| 156 |
db_saved_rev = save_revision(rev) |
| 157 |
parents = {}
|
| 158 |
parents[db_saved_rev] = rev.parents unless rev.parents.nil? |
| 159 |
parents.each do |ch, chparents| |
| 160 |
ch.parents = chparents.collect{|rp| find_changeset_by_name(rp)}.compact
|
| 161 |
end |
| 162 |
end |
| 163 |
h["branches"][br]["last_scmid"] = rev.scmid |
| 164 |
merge_extra_info(h) |
| 165 |
self.save |
| 166 |
end |
| 167 |
end |
| 168 |
end |
| 169 |
end |
| 170 |
|
| 171 |
def save_revision(rev) |
| 172 |
changeset = Changeset.new( |
| 173 |
:repository => self, |
| 174 |
:revision => rev.identifier, |
| 175 |
:scmid => rev.scmid, |
| 176 |
:committer => rev.author, |
| 177 |
:committed_on => rev.time, |
| 178 |
:comments => rev.message |
| 179 |
) |
| 180 |
if changeset.save |
| 181 |
rev.paths.each do |file| |
| 182 |
Change.create( |
| 183 |
:changeset => changeset, |
| 184 |
:action => file[:action], |
| 185 |
:path => file[:path]) |
| 186 |
end |
| 187 |
end |
| 188 |
changeset |
| 189 |
end |
| 190 |
private :save_revision |
| 191 |
|
| 192 |
def latest_changesets(path,rev,limit=10) |
| 193 |
revisions = scm.revisions(path, nil, rev, :limit => limit, :all => false) |
| 194 |
return [] if revisions.nil? || revisions.empty? |
| 195 |
|
| 196 |
changesets.find( |
| 197 |
:all, |
| 198 |
:conditions => [ |
| 199 |
"scmid IN (?)", |
| 200 |
revisions.map!{|c| c.scmid}
|
| 201 |
], |
| 202 |
:order => 'committed_on DESC' |
| 203 |
) |
| 204 |
end |
| 205 |
end |