2007-06-13 00:12:05 +04: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 'redmine/scm/adapters/abstract_adapter'
|
|
|
|
require 'rexml/document'
|
|
|
|
|
|
|
|
module Redmine
|
|
|
|
module Scm
|
|
|
|
module Adapters
|
|
|
|
class SubversionAdapter < AbstractAdapter
|
|
|
|
|
|
|
|
# SVN executable name
|
|
|
|
SVN_BIN = "svn"
|
|
|
|
|
|
|
|
# Get info about the svn repository
|
|
|
|
def info
|
|
|
|
cmd = "#{SVN_BIN} info --xml #{target('')}"
|
2007-10-19 21:59:59 +04:00
|
|
|
cmd << credentials_string
|
2007-06-13 00:12:05 +04:00
|
|
|
info = nil
|
|
|
|
shellout(cmd) do |io|
|
|
|
|
begin
|
|
|
|
doc = REXML::Document.new(io)
|
|
|
|
#root_url = doc.elements["info/entry/repository/root"].text
|
|
|
|
info = Info.new({:root_url => doc.elements["info/entry/repository/root"].text,
|
|
|
|
:lastrev => Revision.new({
|
|
|
|
:identifier => doc.elements["info/entry/commit"].attributes['revision'],
|
2007-07-14 18:53:18 +04:00
|
|
|
:time => Time.parse(doc.elements["info/entry/commit/date"].text).localtime,
|
2007-06-13 00:12:05 +04:00
|
|
|
:author => (doc.elements["info/entry/commit/author"] ? doc.elements["info/entry/commit/author"].text : "")
|
|
|
|
})
|
|
|
|
})
|
|
|
|
rescue
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil if $? && $?.exitstatus != 0
|
|
|
|
info
|
2007-12-15 15:14:40 +03:00
|
|
|
rescue CommandFailed
|
2007-06-13 00:12:05 +04:00
|
|
|
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)
|
|
|
|
e = entries(path, identifier)
|
|
|
|
e ? e.first : nil
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns an Entries collection
|
|
|
|
# or nil if the given path doesn't exist in the repository
|
|
|
|
def entries(path=nil, identifier=nil)
|
|
|
|
path ||= ''
|
|
|
|
identifier = 'HEAD' unless identifier and identifier > 0
|
|
|
|
entries = Entries.new
|
|
|
|
cmd = "#{SVN_BIN} list --xml #{target(path)}@#{identifier}"
|
2007-10-19 21:59:59 +04:00
|
|
|
cmd << credentials_string
|
2007-09-30 21:38:07 +04:00
|
|
|
cmd << " 2>&1"
|
2007-06-13 00:12:05 +04:00
|
|
|
shellout(cmd) do |io|
|
2007-09-30 21:38:07 +04:00
|
|
|
output = io.read
|
2007-06-13 00:12:05 +04:00
|
|
|
begin
|
2007-09-30 21:38:07 +04:00
|
|
|
doc = REXML::Document.new(output)
|
2007-06-13 00:12:05 +04:00
|
|
|
doc.elements.each("lists/list/entry") do |entry|
|
|
|
|
entries << Entry.new({:name => entry.elements['name'].text,
|
|
|
|
:path => ((path.empty? ? "" : "#{path}/") + entry.elements['name'].text),
|
|
|
|
:kind => entry.attributes['kind'],
|
|
|
|
:size => (entry.elements['size'] and entry.elements['size'].text).to_i,
|
|
|
|
:lastrev => Revision.new({
|
|
|
|
:identifier => entry.elements['commit'].attributes['revision'],
|
2007-07-14 18:53:18 +04:00
|
|
|
:time => Time.parse(entry.elements['commit'].elements['date'].text).localtime,
|
2007-06-13 00:12:05 +04:00
|
|
|
:author => (entry.elements['commit'].elements['author'] ? entry.elements['commit'].elements['author'].text : "")
|
|
|
|
})
|
|
|
|
})
|
|
|
|
end
|
2007-09-19 01:22:28 +04:00
|
|
|
rescue Exception => e
|
2007-09-30 21:38:07 +04:00
|
|
|
logger.error("Error parsing svn output: #{e.message}")
|
|
|
|
logger.error("Output was:\n #{output}")
|
2007-06-13 00:12:05 +04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil if $? && $?.exitstatus != 0
|
2007-09-19 01:22:28 +04:00
|
|
|
logger.debug("Found #{entries.size} entries in the repository for #{target(path)}") if logger && logger.debug?
|
2007-06-13 00:12:05 +04:00
|
|
|
entries.sort_by_name
|
|
|
|
end
|
|
|
|
|
|
|
|
def revisions(path=nil, identifier_from=nil, identifier_to=nil, options={})
|
|
|
|
path ||= ''
|
|
|
|
identifier_from = 'HEAD' unless identifier_from and identifier_from.to_i > 0
|
|
|
|
identifier_to = 1 unless identifier_to and identifier_to.to_i > 0
|
|
|
|
revisions = Revisions.new
|
|
|
|
cmd = "#{SVN_BIN} log --xml -r #{identifier_from}:#{identifier_to}"
|
2007-10-19 21:59:59 +04:00
|
|
|
cmd << credentials_string
|
2007-06-13 00:12:05 +04:00
|
|
|
cmd << " --verbose " if options[:with_paths]
|
|
|
|
cmd << target(path)
|
|
|
|
shellout(cmd) do |io|
|
|
|
|
begin
|
|
|
|
doc = REXML::Document.new(io)
|
|
|
|
doc.elements.each("log/logentry") do |logentry|
|
|
|
|
paths = []
|
|
|
|
logentry.elements.each("paths/path") do |path|
|
|
|
|
paths << {:action => path.attributes['action'],
|
|
|
|
:path => path.text,
|
|
|
|
:from_path => path.attributes['copyfrom-path'],
|
|
|
|
:from_revision => path.attributes['copyfrom-rev']
|
|
|
|
}
|
|
|
|
end
|
|
|
|
paths.sort! { |x,y| x[:path] <=> y[:path] }
|
|
|
|
|
|
|
|
revisions << Revision.new({:identifier => logentry.attributes['revision'],
|
|
|
|
:author => (logentry.elements['author'] ? logentry.elements['author'].text : ""),
|
2007-07-14 18:53:18 +04:00
|
|
|
:time => Time.parse(logentry.elements['date'].text).localtime,
|
2007-06-13 00:12:05 +04:00
|
|
|
:message => logentry.elements['msg'].text,
|
|
|
|
:paths => paths
|
|
|
|
})
|
|
|
|
end
|
|
|
|
rescue
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil if $? && $?.exitstatus != 0
|
|
|
|
revisions
|
|
|
|
end
|
|
|
|
|
|
|
|
def diff(path, identifier_from, identifier_to=nil, type="inline")
|
|
|
|
path ||= ''
|
|
|
|
if identifier_to and identifier_to.to_i > 0
|
|
|
|
identifier_to = identifier_to.to_i
|
|
|
|
else
|
|
|
|
identifier_to = identifier_from.to_i - 1
|
|
|
|
end
|
|
|
|
cmd = "#{SVN_BIN} diff -r "
|
|
|
|
cmd << "#{identifier_to}:"
|
|
|
|
cmd << "#{identifier_from}"
|
|
|
|
cmd << "#{target(path)}@#{identifier_from}"
|
2007-10-19 21:59:59 +04:00
|
|
|
cmd << credentials_string
|
2007-06-13 00:12:05 +04:00
|
|
|
diff = []
|
|
|
|
shellout(cmd) do |io|
|
|
|
|
io.each_line do |line|
|
|
|
|
diff << line
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil if $? && $?.exitstatus != 0
|
|
|
|
DiffTableList.new diff, type
|
|
|
|
end
|
|
|
|
|
|
|
|
def cat(path, identifier=nil)
|
|
|
|
identifier = (identifier and identifier.to_i > 0) ? identifier.to_i : "HEAD"
|
|
|
|
cmd = "#{SVN_BIN} cat #{target(path)}@#{identifier}"
|
2007-10-19 21:59:59 +04:00
|
|
|
cmd << credentials_string
|
2007-06-13 00:12:05 +04:00
|
|
|
cat = nil
|
|
|
|
shellout(cmd) do |io|
|
|
|
|
io.binmode
|
|
|
|
cat = io.read
|
|
|
|
end
|
|
|
|
return nil if $? && $?.exitstatus != 0
|
|
|
|
cat
|
|
|
|
end
|
2007-10-19 21:59:59 +04:00
|
|
|
|
2007-12-02 23:58:02 +03:00
|
|
|
def annotate(path, identifier=nil)
|
|
|
|
identifier = (identifier and identifier.to_i > 0) ? identifier.to_i : "HEAD"
|
|
|
|
cmd = "#{SVN_BIN} blame #{target(path)}@#{identifier}"
|
|
|
|
cmd << credentials_string
|
|
|
|
blame = Annotate.new
|
|
|
|
shellout(cmd) do |io|
|
|
|
|
io.each_line do |line|
|
|
|
|
next unless line =~ %r{^\s*(\d+)\s*(\S+)\s(.*)$}
|
|
|
|
blame.add_line($3.rstrip, Revision.new(:identifier => $1.to_i, :author => $2.strip))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
return nil if $? && $?.exitstatus != 0
|
|
|
|
blame
|
|
|
|
end
|
|
|
|
|
2007-10-19 21:59:59 +04:00
|
|
|
private
|
|
|
|
|
|
|
|
def credentials_string
|
|
|
|
str = ''
|
|
|
|
str << " --username #{shell_quote(@login)}" unless @login.blank?
|
|
|
|
str << " --password #{shell_quote(@password)}" unless @login.blank? || @password.blank?
|
|
|
|
str
|
|
|
|
end
|
2007-06-13 00:12:05 +04:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|