summaryrefslogtreecommitdiffstats
path: root/app/models
diff options
context:
space:
mode:
Diffstat (limited to 'app/models')
-rw-r--r--app/models/repository.rb18
-rw-r--r--app/models/repository/git.rb83
2 files changed, 67 insertions, 34 deletions
diff --git a/app/models/repository.rb b/app/models/repository.rb
index bf181bfad..860395b5c 100644
--- a/app/models/repository.rb
+++ b/app/models/repository.rb
@@ -62,6 +62,18 @@ class Repository < ActiveRecord::Base
def entries(path=nil, identifier=nil)
scm.entries(path, identifier)
end
+
+ def branches
+ scm.branches
+ end
+
+ def tags
+ scm.tags
+ end
+
+ def default_branch
+ scm.default_branch
+ end
def properties(path, identifier=nil)
scm.properties(path, identifier)
@@ -92,11 +104,15 @@ class Repository < ActiveRecord::Base
def latest_changeset
@latest_changeset ||= changesets.find(:first)
end
+
+ def latest_changesets(path,rev,limit=10)
+ @latest_changesets ||= changesets.find(:all, limit, :order => "committed_on DESC")
+ end
def scan_changesets_for_issue_ids
self.changesets.each(&:scan_comment_for_issue_ids)
end
-
+
# Returns an array of committers usernames and associated user_id
def committers
@committers ||= Changeset.connection.select_rows("SELECT DISTINCT committer, user_id FROM #{Changeset.table_name} WHERE repository_id = #{id}")
diff --git a/app/models/repository/git.rb b/app/models/repository/git.rb
index f721b938f..b3cdf3643 100644
--- a/app/models/repository/git.rb
+++ b/app/models/repository/git.rb
@@ -29,43 +29,60 @@ class Repository::Git < Repository
'Git'
end
+ def branches
+ scm.branches
+ end
+
+ def tags
+ scm.tags
+ end
+
def changesets_for_path(path, options={})
- Change.find(:all, :include => {:changeset => :user},
- :conditions => ["repository_id = ? AND path = ?", id, path],
- :order => "committed_on DESC, #{Changeset.table_name}.revision DESC",
- :limit => options[:limit]).collect(&:changeset)
+ Change.find(
+ :all,
+ :include => {:changeset => :user},
+ :conditions => ["repository_id = ? AND path = ?", id, path],
+ :order => "committed_on DESC, #{Changeset.table_name}.revision DESC",
+ :limit => options[:limit]
+ ).collect(&:changeset)
end
+ # With SCM's that have a sequential commit numbering, redmine is able to be
+ # clever and only fetch changesets going forward from the most recent one
+ # it knows about. However, with git, you never know if people have merged
+ # commits into the middle of the repository history, so we always have to
+ # parse the entire log.
def fetch_changesets
- scm_info = scm.info
- if scm_info
- # latest revision found in database
- db_revision = latest_changeset ? latest_changeset.revision : nil
- # latest revision in the repository
- scm_revision = scm_info.lastrev.scmid
+ # Save ourselves an expensive operation if we're already up to date
+ return if scm.num_revisions == changesets.count
+
+ revisions = scm.revisions('', nil, nil, :all => true)
+ return if revisions.nil? || revisions.empty?
+
+ # Find revisions that redmine knows about already
+ existing_revisions = changesets.find(:all).map!{|c| c.scmid}
+
+ # Clean out revisions that are no longer in git
+ Changeset.delete_all(["scmid NOT IN (?) AND repository_id = (?)", revisions.map{|r| r.scmid}, self.id])
+
+ # Subtract revisions that redmine already knows about
+ revisions.reject!{|r| existing_revisions.include?(r.scmid)}
+
+ # Save the remaining ones to the database
+ revisions.each{|r| r.save(self)} unless revisions.nil?
+ end
+
+ def latest_changesets(path,rev,limit=10)
+ revisions = scm.revisions(path, nil, rev, :limit => limit, :all => false)
+ return [] if revisions.nil? || revisions.empty?
- unless changesets.find_by_scmid(scm_revision)
- scm.revisions('', db_revision, nil, :reverse => true) do |revision|
- if changesets.find_by_scmid(revision.scmid.to_s).nil?
- transaction do
- changeset = Changeset.create!(:repository => self,
- :revision => revision.identifier,
- :scmid => revision.scmid,
- :committer => revision.author,
- :committed_on => revision.time,
- :comments => revision.message)
-
- revision.paths.each do |change|
- Change.create!(:changeset => changeset,
- :action => change[:action],
- :path => change[:path],
- :from_path => change[:from_path],
- :from_revision => change[:from_revision])
- end
- end
- end
- end
- end
- end
+ changesets.find(
+ :all,
+ :conditions => [
+ "scmid IN (?)",
+ revisions.map!{|c| c.scmid}
+ ],
+ :order => 'committed_on DESC'
+ )
end
end