mirror of
				https://github.com/redmine/redmine.git
				synced 2025-10-26 07:46:17 +01:00 
			
		
		
		
	git-svn-id: https://svn.redmine.org/redmine/trunk@23698 e93f8b46-1217-0410-a6f0-8f06a7374b81
		
			
				
	
	
		
			219 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
			
		
		
	
	
			219 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			Ruby
		
	
	
	
	
	
| # frozen_string_literal: true
 | |
| 
 | |
| # Redmine - project management software
 | |
| # Copyright (C) 2006-  Jean-Philippe Lang
 | |
| #
 | |
| # This program is free software; you can redistribute it and/or
 | |
| # modify it under the terms of the GNU General Public License
 | |
| # as published by the Free Software Foundation; either version 2
 | |
| # of the License, or (at your option) any later version.
 | |
| #
 | |
| # This program is distributed in the hope that it will be useful,
 | |
| # but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| # GNU General Public License for more details.
 | |
| #
 | |
| # You should have received a copy of the GNU General Public License
 | |
| # along with this program; if not, write to the Free Software
 | |
| # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
 | |
| 
 | |
| require 'redmine/scm/adapters/mercurial_adapter'
 | |
| 
 | |
| class Repository::Mercurial < Repository
 | |
|   # sort changesets by revision number
 | |
|   has_many :changesets,
 | |
|            lambda {order("#{Changeset.table_name}.id DESC")},
 | |
|            :foreign_key => 'repository_id'
 | |
| 
 | |
|   validates_presence_of :url
 | |
| 
 | |
|   # number of changesets to fetch at once
 | |
|   FETCH_AT_ONCE = 100
 | |
| 
 | |
|   def self.human_attribute_name(attribute_key_name, *)
 | |
|     attr_name = attribute_key_name.to_s
 | |
|     if attr_name == "url"
 | |
|       attr_name = "path_to_repository"
 | |
|     end
 | |
|     super(attr_name, *)
 | |
|   end
 | |
| 
 | |
|   def self.scm_adapter_class
 | |
|     Redmine::Scm::Adapters::MercurialAdapter
 | |
|   end
 | |
| 
 | |
|   def self.scm_name
 | |
|     'Mercurial'
 | |
|   end
 | |
| 
 | |
|   def supports_directory_revisions?
 | |
|     true
 | |
|   end
 | |
| 
 | |
|   def supports_revision_graph?
 | |
|     true
 | |
|   end
 | |
| 
 | |
|   def repo_log_encoding
 | |
|     'UTF-8'
 | |
|   end
 | |
| 
 | |
|   # Returns the readable identifier for the given mercurial changeset
 | |
|   def self.format_changeset_identifier(changeset)
 | |
|     "#{changeset.revision}:#{changeset.scmid[0, 12]}"
 | |
|   end
 | |
| 
 | |
|   # Returns the identifier for the given Mercurial changeset
 | |
|   def self.changeset_identifier(changeset)
 | |
|     changeset.scmid
 | |
|   end
 | |
| 
 | |
|   def diff_format_revisions(cs, cs_to, sep=':')
 | |
|     super(cs, cs_to, ' ')
 | |
|   end
 | |
| 
 | |
|   def modify_entry_lastrev_identifier(entry)
 | |
|     if entry.lastrev && entry.lastrev.identifier
 | |
|       entry.lastrev.identifier = scmid_for_inserting_db(entry.lastrev.identifier)
 | |
|     end
 | |
|   end
 | |
|   private :modify_entry_lastrev_identifier
 | |
| 
 | |
|   def entry(path=nil, identifier=nil)
 | |
|     entry = scm.entry(path, identifier)
 | |
|     return nil if entry.nil?
 | |
| 
 | |
|     modify_entry_lastrev_identifier(entry)
 | |
|     entry
 | |
|   end
 | |
| 
 | |
|   def scm_entries(path=nil, identifier=nil)
 | |
|     entries = scm.entries(path, identifier)
 | |
|     return nil if entries.nil?
 | |
| 
 | |
|     entries.each {|entry| modify_entry_lastrev_identifier(entry)}
 | |
|     entries
 | |
|   end
 | |
|   protected :scm_entries
 | |
| 
 | |
|   # Finds and returns a revision with a number or the beginning of a hash
 | |
|   def find_changeset_by_name(name)
 | |
|     return nil if name.blank?
 | |
| 
 | |
|     s = name.to_s
 | |
|     if /[^\d]/.match?(s) || s.size > 8
 | |
|       cs = changesets.where(:scmid => s).first
 | |
|     else
 | |
|       cs = changesets.find_by(:revision => s)
 | |
|     end
 | |
|     return cs if cs
 | |
| 
 | |
|     changesets.where('scmid LIKE ?', "#{s}%").first
 | |
|   end
 | |
| 
 | |
|   # Returns the latest changesets for +path+; sorted by revision number
 | |
|   #
 | |
|   # Because :order => 'id DESC' is defined at 'has_many',
 | |
|   # there is no need to set 'order'.
 | |
|   # But, MySQL test fails.
 | |
|   # Sqlite3 and PostgreSQL pass.
 | |
|   # Is this MySQL bug?
 | |
|   def latest_changesets(path, rev, limit=10)
 | |
|     changesets.
 | |
|       includes(:user).
 | |
|       where(latest_changesets_cond(path, rev, limit)).
 | |
|       references(:user).
 | |
|       limit(limit).
 | |
|       order("#{Changeset.table_name}.id DESC").
 | |
|       to_a
 | |
|   end
 | |
| 
 | |
|   def is_short_id_in_db?
 | |
|     return @is_short_id_in_db unless @is_short_id_in_db.nil?
 | |
| 
 | |
|     cs = changesets.first
 | |
|     @is_short_id_in_db = (!cs.nil? && cs.scmid.length != 40)
 | |
|   end
 | |
|   private :is_short_id_in_db?
 | |
| 
 | |
|   def scmid_for_inserting_db(scmid)
 | |
|     is_short_id_in_db? ? scmid[0, 12] : scmid
 | |
|   end
 | |
| 
 | |
|   def nodes_in_branch(rev, branch_limit)
 | |
|     scm.nodes_in_branch(rev, :limit => branch_limit).collect do |b|
 | |
|       scmid_for_inserting_db(b)
 | |
|     end
 | |
|   end
 | |
| 
 | |
|   def tag_scmid(rev)
 | |
|     scmid = scm.tagmap[rev]
 | |
|     scmid.nil? ? nil : scmid_for_inserting_db(scmid)
 | |
|   end
 | |
| 
 | |
|   def latest_changesets_cond(path, rev, limit)
 | |
|     cond, args = [], []
 | |
|     if scm.branchmap.member? rev
 | |
|       # Mercurial named branch is *stable* in each revision.
 | |
|       # So, named branch can be stored in database.
 | |
|       # Mercurial provides *bookmark* which is equivalent with git branch.
 | |
|       # But, bookmark is not implemented.
 | |
|       cond << "#{Changeset.table_name}.scmid IN (?)"
 | |
|       # Revisions in root directory and sub directory are not equal.
 | |
|       # So, in order to get correct limit, we need to get all revisions.
 | |
|       # But, it is very heavy.
 | |
|       # Mercurial does not treat directory.
 | |
|       # So, "hg log DIR" is very heavy.
 | |
|       branch_limit = path.blank? ? limit : (limit * 5)
 | |
|       args << nodes_in_branch(rev, branch_limit)
 | |
|     elsif last = rev ? find_changeset_by_name(tag_scmid(rev) || rev) : nil
 | |
|       cond << "#{Changeset.table_name}.id <= ?"
 | |
|       args << last.id
 | |
|     end
 | |
|     unless path.blank?
 | |
|       cond << "EXISTS (SELECT * FROM #{Change.table_name}
 | |
|                  WHERE #{Change.table_name}.changeset_id = #{Changeset.table_name}.id
 | |
|                  AND (#{Change.table_name}.path = ?
 | |
|                        OR #{Change.table_name}.path LIKE ? ESCAPE ?))"
 | |
|       args << path.with_leading_slash
 | |
|       args << "#{path.with_leading_slash.gsub(%r{[%_\\]}) {|s| "\\#{s}"}}/%" << '\\'
 | |
|     end
 | |
|     [cond.join(' AND '), *args] unless cond.empty?
 | |
|   end
 | |
|   private :latest_changesets_cond
 | |
| 
 | |
|   def fetch_changesets
 | |
|     return if scm.info.nil?
 | |
| 
 | |
|     scm_rev = scm.info.lastrev.revision.to_i
 | |
|     db_rev  = latest_changeset ? latest_changeset.revision.to_i : -1
 | |
|     return unless db_rev < scm_rev  # already up-to-date
 | |
| 
 | |
|     logger.debug "Fetching changesets for repository #{url}" if logger
 | |
|     (db_rev + 1).step(scm_rev, FETCH_AT_ONCE) do |i|
 | |
|       scm.each_revision('', i, [i + FETCH_AT_ONCE - 1, scm_rev].min) do |re|
 | |
|         transaction do
 | |
|           parents = (re.parents || []).filter_map do |rp|
 | |
|             find_changeset_by_name(scmid_for_inserting_db(rp))
 | |
|           end
 | |
|           cs = Changeset.create(:repository   => self,
 | |
|                                 :revision     => re.revision,
 | |
|                                 :scmid        => scmid_for_inserting_db(re.scmid),
 | |
|                                 :committer    => re.author,
 | |
|                                 :committed_on => re.time,
 | |
|                                 :comments     => re.message,
 | |
|                                 :parents      => parents)
 | |
|           unless cs.new_record?
 | |
|             re.paths.each do |e|
 | |
|               if from_revision = e[:from_revision]
 | |
|                 e[:from_revision] = scmid_for_inserting_db(from_revision)
 | |
|               end
 | |
|               cs.create_change(e)
 | |
|             end
 | |
|           end
 | |
|         end
 | |
|       end
 | |
|     end
 | |
|   end
 | |
| end
 |