| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  | # redMine - project management software | 
					
						
							|  |  |  | # Copyright (C) 2006-2007  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 'cgi' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module Redmine | 
					
						
							|  |  |  |   module Scm | 
					
						
							|  |  |  |     module Adapters     | 
					
						
							|  |  |  |       class CommandFailed < StandardError #:nodoc: | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       class AbstractAdapter #:nodoc: | 
					
						
							|  |  |  |         def initialize(url, root_url=nil, login=nil, password=nil) | 
					
						
							|  |  |  |           @url = url | 
					
						
							|  |  |  |           @login = login if login && !login.empty? | 
					
						
							|  |  |  |           @password = (password || "") if @login | 
					
						
							|  |  |  |           @root_url = root_url.blank? ? retrieve_root_url : root_url | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def adapter_name | 
					
						
							|  |  |  |           'Abstract' | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2007-06-24 19:30:38 +00:00
										 |  |  |         def supports_cat? | 
					
						
							|  |  |  |           true | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2007-12-02 20:58:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def supports_annotate? | 
					
						
							|  |  |  |           respond_to?('annotate') | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2007-06-24 19:30:38 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |         def root_url | 
					
						
							|  |  |  |           @root_url | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |         def url | 
					
						
							|  |  |  |           @url | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |         # get info about the svn repository | 
					
						
							|  |  |  |         def info | 
					
						
							|  |  |  |           return nil | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         # Returns the entry identified by path and revision identifier | 
					
						
							|  |  |  |         # or nil if entry doesn't exist in the repository | 
					
						
							|  |  |  |         def entry(path=nil, identifier=nil) | 
					
						
							| 
									
										
										
										
											2008-04-27 10:12:15 +00:00
										 |  |  |           parts = path.to_s.split(%r{[\/\\]}).select {|n| !n.blank?} | 
					
						
							|  |  |  |           search_path = parts[0..-2].join('/') | 
					
						
							|  |  |  |           search_name = parts[-1] | 
					
						
							|  |  |  |           if search_path.blank? && search_name.blank? | 
					
						
							|  |  |  |             # Root entry | 
					
						
							|  |  |  |             Entry.new(:path => '', :kind => 'dir') | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             # Search for the entry in the parent directory | 
					
						
							|  |  |  |             es = entries(search_path, identifier) | 
					
						
							|  |  |  |             es ? es.detect {|e| e.name == search_name} : nil | 
					
						
							|  |  |  |           end | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         # Returns an Entries collection | 
					
						
							|  |  |  |         # or nil if the given path doesn't exist in the repository | 
					
						
							|  |  |  |         def entries(path=nil, identifier=nil) | 
					
						
							|  |  |  |           return nil | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |         def revisions(path=nil, identifier_from=nil, identifier_to=nil, options={}) | 
					
						
							|  |  |  |           return nil | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2008-06-08 16:28:42 +00:00
										 |  |  |         def diff(path, identifier_from, identifier_to=nil) | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |           return nil | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def cat(path, identifier=nil) | 
					
						
							|  |  |  |           return nil | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2007-12-02 20:58:02 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |         def with_leading_slash(path) | 
					
						
							|  |  |  |           path ||= '' | 
					
						
							|  |  |  |           (path[0,1]!="/") ? "/#{path}" : path | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2008-06-07 09:19:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         def with_trailling_slash(path) | 
					
						
							|  |  |  |           path ||= '' | 
					
						
							|  |  |  |           (path[-1,1] == "/") ? path : "#{path}/" | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2007-10-19 17:59:59 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2008-06-08 15:40:24 +00:00
										 |  |  |         def without_leading_slash(path) | 
					
						
							|  |  |  |           path ||= '' | 
					
						
							|  |  |  |           path.gsub(%r{^/+}, '') | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         def without_trailling_slash(path) | 
					
						
							|  |  |  |           path ||= '' | 
					
						
							|  |  |  |           (path[-1,1] == "/") ? path[0..-2] : path | 
					
						
							|  |  |  |          end | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2007-10-19 17:59:59 +00:00
										 |  |  |         def shell_quote(str) | 
					
						
							|  |  |  |           if RUBY_PLATFORM =~ /mswin/ | 
					
						
							|  |  |  |             '"' + str.gsub(/"/, '\\"') + '"' | 
					
						
							|  |  |  |           else | 
					
						
							|  |  |  |             "'" + str.gsub(/'/, "'\"'\"'") + "'" | 
					
						
							|  |  |  |           end | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2008-06-07 09:19:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |       private | 
					
						
							|  |  |  |         def retrieve_root_url | 
					
						
							|  |  |  |           info = self.info | 
					
						
							|  |  |  |           info ? info.root_url : nil | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def target(path) | 
					
						
							| 
									
										
										
										
											2008-03-12 23:00:11 +00:00
										 |  |  |           path ||= '' | 
					
						
							|  |  |  |           base = path.match(/^\//) ? root_url : url | 
					
						
							|  |  |  |           shell_quote("#{base}/#{path}".gsub(/[?<>\*]/, '')) | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |         end | 
					
						
							|  |  |  |              | 
					
						
							|  |  |  |         def logger | 
					
						
							|  |  |  |           RAILS_DEFAULT_LOGGER | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2008-06-06 14:37:49 +00:00
										 |  |  |          | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |         def shellout(cmd, &block) | 
					
						
							|  |  |  |           logger.debug "Shelling out: #{cmd}" if logger && logger.debug? | 
					
						
							| 
									
										
										
										
											2007-12-15 12:14:40 +00:00
										 |  |  |           begin | 
					
						
							|  |  |  |             IO.popen(cmd, "r+") do |io| | 
					
						
							|  |  |  |               io.close_write | 
					
						
							|  |  |  |               block.call(io) if block_given? | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           rescue Errno::ENOENT => e | 
					
						
							| 
									
										
										
										
											2008-06-06 14:37:49 +00:00
										 |  |  |             msg = strip_credential(e.message) | 
					
						
							| 
									
										
										
										
											2007-12-15 12:14:40 +00:00
										 |  |  |             # The command failed, log it and re-raise | 
					
						
							| 
									
										
										
										
											2008-06-06 14:37:49 +00:00
										 |  |  |             logger.error("SCM command failed: #{strip_credential(cmd)}\n  with: #{msg}") | 
					
						
							|  |  |  |             raise CommandFailed.new(msg) | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |           end | 
					
						
							|  |  |  |         end   | 
					
						
							| 
									
										
										
										
											2008-06-06 14:37:49 +00:00
										 |  |  |          | 
					
						
							|  |  |  |         # Hides username/password in a given command | 
					
						
							|  |  |  |         def self.hide_credential(cmd) | 
					
						
							|  |  |  |           q = (RUBY_PLATFORM =~ /mswin/ ? '"' : "'") | 
					
						
							|  |  |  |           cmd.to_s.gsub(/(\-\-(password|username))\s+(#{q}[^#{q}]+#{q}|[^#{q}]\S+)/, '\\1 xxxx') | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def strip_credential(cmd) | 
					
						
							|  |  |  |           self.class.hide_credential(cmd) | 
					
						
							|  |  |  |         end | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |       end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       class Entries < Array | 
					
						
							|  |  |  |         def sort_by_name | 
					
						
							|  |  |  |           sort {|x,y|  | 
					
						
							|  |  |  |             if x.kind == y.kind | 
					
						
							|  |  |  |               x.name <=> y.name | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |               x.kind <=> y.kind | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           }    | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def revisions | 
					
						
							|  |  |  |           revisions ||= Revisions.new(collect{|entry| entry.lastrev}.compact) | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       class Info | 
					
						
							|  |  |  |         attr_accessor :root_url, :lastrev | 
					
						
							|  |  |  |         def initialize(attributes={}) | 
					
						
							|  |  |  |           self.root_url = attributes[:root_url] if attributes[:root_url] | 
					
						
							|  |  |  |           self.lastrev = attributes[:lastrev] | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       class Entry | 
					
						
							|  |  |  |         attr_accessor :name, :path, :kind, :size, :lastrev | 
					
						
							|  |  |  |         def initialize(attributes={}) | 
					
						
							|  |  |  |           self.name = attributes[:name] if attributes[:name] | 
					
						
							|  |  |  |           self.path = attributes[:path] if attributes[:path] | 
					
						
							|  |  |  |           self.kind = attributes[:kind] if attributes[:kind] | 
					
						
							|  |  |  |           self.size = attributes[:size].to_i if attributes[:size] | 
					
						
							|  |  |  |           self.lastrev = attributes[:lastrev] | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def is_file? | 
					
						
							|  |  |  |           'file' == self.kind | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def is_dir? | 
					
						
							|  |  |  |           'dir' == self.kind | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def is_text? | 
					
						
							|  |  |  |           Redmine::MimeType.is_type?('text', name) | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       class Revisions < Array | 
					
						
							|  |  |  |         def latest | 
					
						
							|  |  |  |           sort {|x,y| | 
					
						
							|  |  |  |             unless x.time.nil? or y.time.nil? | 
					
						
							|  |  |  |               x.time <=> y.time | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |               0
 | 
					
						
							|  |  |  |             end | 
					
						
							|  |  |  |           }.last | 
					
						
							|  |  |  |         end  | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |        | 
					
						
							|  |  |  |       class Revision | 
					
						
							|  |  |  |         attr_accessor :identifier, :scmid, :name, :author, :time, :message, :paths, :revision, :branch | 
					
						
							|  |  |  |         def initialize(attributes={}) | 
					
						
							|  |  |  |           self.identifier = attributes[:identifier] | 
					
						
							|  |  |  |           self.scmid = attributes[:scmid] | 
					
						
							|  |  |  |           self.name = attributes[:name] || self.identifier | 
					
						
							|  |  |  |           self.author = attributes[:author] | 
					
						
							|  |  |  |           self.time = attributes[:time] | 
					
						
							|  |  |  |           self.message = attributes[:message] || "" | 
					
						
							|  |  |  |           self.paths = attributes[:paths] | 
					
						
							|  |  |  |           self.revision = attributes[:revision] | 
					
						
							|  |  |  |           self.branch = attributes[:branch] | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |      | 
					
						
							|  |  |  |       end | 
					
						
							|  |  |  |          | 
					
						
							| 
									
										
										
										
											2007-12-02 20:58:02 +00:00
										 |  |  |       class Annotate | 
					
						
							|  |  |  |         attr_reader :lines, :revisions | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def initialize | 
					
						
							|  |  |  |           @lines = [] | 
					
						
							|  |  |  |           @revisions = [] | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def add_line(line, revision) | 
					
						
							|  |  |  |           @lines << line | 
					
						
							|  |  |  |           @revisions << revision | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def content | 
					
						
							|  |  |  |           content = lines.join("\n") | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |          | 
					
						
							|  |  |  |         def empty? | 
					
						
							|  |  |  |           lines.empty? | 
					
						
							|  |  |  |         end | 
					
						
							|  |  |  |       end | 
					
						
							| 
									
										
										
										
											2007-06-12 20:12:05 +00:00
										 |  |  |     end | 
					
						
							|  |  |  |   end | 
					
						
							|  |  |  | end |