##// END OF EJS Templates
Added 2 log messages in SubversionAdapter#entries....
Jean-Philippe Lang -
r732:83eed09109ba
parent child
Show More
@@ -1,173 +1,175
1 # redMine - project management software
1 # redMine - project management software
2 # Copyright (C) 2006-2007 Jean-Philippe Lang
2 # Copyright (C) 2006-2007 Jean-Philippe Lang
3 #
3 #
4 # This program is free software; you can redistribute it and/or
4 # This program is free software; you can redistribute it and/or
5 # modify it under the terms of the GNU General Public License
5 # modify it under the terms of the GNU General Public License
6 # as published by the Free Software Foundation; either version 2
6 # as published by the Free Software Foundation; either version 2
7 # of the License, or (at your option) any later version.
7 # of the License, or (at your option) any later version.
8 #
8 #
9 # This program is distributed in the hope that it will be useful,
9 # This program is distributed in the hope that it will be useful,
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 # GNU General Public License for more details.
12 # GNU General Public License for more details.
13 #
13 #
14 # You should have received a copy of the GNU General Public License
14 # You should have received a copy of the GNU General Public License
15 # along with this program; if not, write to the Free Software
15 # along with this program; if not, write to the Free Software
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
17
17
18 require 'redmine/scm/adapters/abstract_adapter'
18 require 'redmine/scm/adapters/abstract_adapter'
19 require 'rexml/document'
19 require 'rexml/document'
20
20
21 module Redmine
21 module Redmine
22 module Scm
22 module Scm
23 module Adapters
23 module Adapters
24 class SubversionAdapter < AbstractAdapter
24 class SubversionAdapter < AbstractAdapter
25
25
26 # SVN executable name
26 # SVN executable name
27 SVN_BIN = "svn"
27 SVN_BIN = "svn"
28
28
29 # Get info about the svn repository
29 # Get info about the svn repository
30 def info
30 def info
31 cmd = "#{SVN_BIN} info --xml #{target('')}"
31 cmd = "#{SVN_BIN} info --xml #{target('')}"
32 cmd << " --username #{@login} --password #{@password}" if @login
32 cmd << " --username #{@login} --password #{@password}" if @login
33 info = nil
33 info = nil
34 shellout(cmd) do |io|
34 shellout(cmd) do |io|
35 begin
35 begin
36 doc = REXML::Document.new(io)
36 doc = REXML::Document.new(io)
37 #root_url = doc.elements["info/entry/repository/root"].text
37 #root_url = doc.elements["info/entry/repository/root"].text
38 info = Info.new({:root_url => doc.elements["info/entry/repository/root"].text,
38 info = Info.new({:root_url => doc.elements["info/entry/repository/root"].text,
39 :lastrev => Revision.new({
39 :lastrev => Revision.new({
40 :identifier => doc.elements["info/entry/commit"].attributes['revision'],
40 :identifier => doc.elements["info/entry/commit"].attributes['revision'],
41 :time => Time.parse(doc.elements["info/entry/commit/date"].text).localtime,
41 :time => Time.parse(doc.elements["info/entry/commit/date"].text).localtime,
42 :author => (doc.elements["info/entry/commit/author"] ? doc.elements["info/entry/commit/author"].text : "")
42 :author => (doc.elements["info/entry/commit/author"] ? doc.elements["info/entry/commit/author"].text : "")
43 })
43 })
44 })
44 })
45 rescue
45 rescue
46 end
46 end
47 end
47 end
48 return nil if $? && $?.exitstatus != 0
48 return nil if $? && $?.exitstatus != 0
49 info
49 info
50 rescue Errno::ENOENT => e
50 rescue Errno::ENOENT => e
51 return nil
51 return nil
52 end
52 end
53
53
54 # Returns the entry identified by path and revision identifier
54 # Returns the entry identified by path and revision identifier
55 # or nil if entry doesn't exist in the repository
55 # or nil if entry doesn't exist in the repository
56 def entry(path=nil, identifier=nil)
56 def entry(path=nil, identifier=nil)
57 e = entries(path, identifier)
57 e = entries(path, identifier)
58 e ? e.first : nil
58 e ? e.first : nil
59 end
59 end
60
60
61 # Returns an Entries collection
61 # Returns an Entries collection
62 # or nil if the given path doesn't exist in the repository
62 # or nil if the given path doesn't exist in the repository
63 def entries(path=nil, identifier=nil)
63 def entries(path=nil, identifier=nil)
64 path ||= ''
64 path ||= ''
65 identifier = 'HEAD' unless identifier and identifier > 0
65 identifier = 'HEAD' unless identifier and identifier > 0
66 entries = Entries.new
66 entries = Entries.new
67 cmd = "#{SVN_BIN} list --xml #{target(path)}@#{identifier}"
67 cmd = "#{SVN_BIN} list --xml #{target(path)}@#{identifier}"
68 cmd << " --username #{@login} --password #{@password}" if @login
68 cmd << " --username #{@login} --password #{@password}" if @login
69 shellout(cmd) do |io|
69 shellout(cmd) do |io|
70 begin
70 begin
71 doc = REXML::Document.new(io)
71 doc = REXML::Document.new(io)
72 doc.elements.each("lists/list/entry") do |entry|
72 doc.elements.each("lists/list/entry") do |entry|
73 entries << Entry.new({:name => entry.elements['name'].text,
73 entries << Entry.new({:name => entry.elements['name'].text,
74 :path => ((path.empty? ? "" : "#{path}/") + entry.elements['name'].text),
74 :path => ((path.empty? ? "" : "#{path}/") + entry.elements['name'].text),
75 :kind => entry.attributes['kind'],
75 :kind => entry.attributes['kind'],
76 :size => (entry.elements['size'] and entry.elements['size'].text).to_i,
76 :size => (entry.elements['size'] and entry.elements['size'].text).to_i,
77 :lastrev => Revision.new({
77 :lastrev => Revision.new({
78 :identifier => entry.elements['commit'].attributes['revision'],
78 :identifier => entry.elements['commit'].attributes['revision'],
79 :time => Time.parse(entry.elements['commit'].elements['date'].text).localtime,
79 :time => Time.parse(entry.elements['commit'].elements['date'].text).localtime,
80 :author => (entry.elements['commit'].elements['author'] ? entry.elements['commit'].elements['author'].text : "")
80 :author => (entry.elements['commit'].elements['author'] ? entry.elements['commit'].elements['author'].text : "")
81 })
81 })
82 })
82 })
83 end
83 end
84 rescue
84 rescue Exception => e
85 logger.info("Error parsing svn output: #{e.message}")
85 end
86 end
86 end
87 end
87 return nil if $? && $?.exitstatus != 0
88 return nil if $? && $?.exitstatus != 0
89 logger.debug("Found #{entries.size} entries in the repository for #{target(path)}") if logger && logger.debug?
88 entries.sort_by_name
90 entries.sort_by_name
89 rescue Errno::ENOENT => e
91 rescue Errno::ENOENT => e
90 raise CommandFailed
92 raise CommandFailed
91 end
93 end
92
94
93 def revisions(path=nil, identifier_from=nil, identifier_to=nil, options={})
95 def revisions(path=nil, identifier_from=nil, identifier_to=nil, options={})
94 path ||= ''
96 path ||= ''
95 identifier_from = 'HEAD' unless identifier_from and identifier_from.to_i > 0
97 identifier_from = 'HEAD' unless identifier_from and identifier_from.to_i > 0
96 identifier_to = 1 unless identifier_to and identifier_to.to_i > 0
98 identifier_to = 1 unless identifier_to and identifier_to.to_i > 0
97 revisions = Revisions.new
99 revisions = Revisions.new
98 cmd = "#{SVN_BIN} log --xml -r #{identifier_from}:#{identifier_to}"
100 cmd = "#{SVN_BIN} log --xml -r #{identifier_from}:#{identifier_to}"
99 cmd << " --username #{@login} --password #{@password}" if @login
101 cmd << " --username #{@login} --password #{@password}" if @login
100 cmd << " --verbose " if options[:with_paths]
102 cmd << " --verbose " if options[:with_paths]
101 cmd << target(path)
103 cmd << target(path)
102 shellout(cmd) do |io|
104 shellout(cmd) do |io|
103 begin
105 begin
104 doc = REXML::Document.new(io)
106 doc = REXML::Document.new(io)
105 doc.elements.each("log/logentry") do |logentry|
107 doc.elements.each("log/logentry") do |logentry|
106 paths = []
108 paths = []
107 logentry.elements.each("paths/path") do |path|
109 logentry.elements.each("paths/path") do |path|
108 paths << {:action => path.attributes['action'],
110 paths << {:action => path.attributes['action'],
109 :path => path.text,
111 :path => path.text,
110 :from_path => path.attributes['copyfrom-path'],
112 :from_path => path.attributes['copyfrom-path'],
111 :from_revision => path.attributes['copyfrom-rev']
113 :from_revision => path.attributes['copyfrom-rev']
112 }
114 }
113 end
115 end
114 paths.sort! { |x,y| x[:path] <=> y[:path] }
116 paths.sort! { |x,y| x[:path] <=> y[:path] }
115
117
116 revisions << Revision.new({:identifier => logentry.attributes['revision'],
118 revisions << Revision.new({:identifier => logentry.attributes['revision'],
117 :author => (logentry.elements['author'] ? logentry.elements['author'].text : ""),
119 :author => (logentry.elements['author'] ? logentry.elements['author'].text : ""),
118 :time => Time.parse(logentry.elements['date'].text).localtime,
120 :time => Time.parse(logentry.elements['date'].text).localtime,
119 :message => logentry.elements['msg'].text,
121 :message => logentry.elements['msg'].text,
120 :paths => paths
122 :paths => paths
121 })
123 })
122 end
124 end
123 rescue
125 rescue
124 end
126 end
125 end
127 end
126 return nil if $? && $?.exitstatus != 0
128 return nil if $? && $?.exitstatus != 0
127 revisions
129 revisions
128 rescue Errno::ENOENT => e
130 rescue Errno::ENOENT => e
129 raise CommandFailed
131 raise CommandFailed
130 end
132 end
131
133
132 def diff(path, identifier_from, identifier_to=nil, type="inline")
134 def diff(path, identifier_from, identifier_to=nil, type="inline")
133 path ||= ''
135 path ||= ''
134 if identifier_to and identifier_to.to_i > 0
136 if identifier_to and identifier_to.to_i > 0
135 identifier_to = identifier_to.to_i
137 identifier_to = identifier_to.to_i
136 else
138 else
137 identifier_to = identifier_from.to_i - 1
139 identifier_to = identifier_from.to_i - 1
138 end
140 end
139 cmd = "#{SVN_BIN} diff -r "
141 cmd = "#{SVN_BIN} diff -r "
140 cmd << "#{identifier_to}:"
142 cmd << "#{identifier_to}:"
141 cmd << "#{identifier_from}"
143 cmd << "#{identifier_from}"
142 cmd << "#{target(path)}@#{identifier_from}"
144 cmd << "#{target(path)}@#{identifier_from}"
143 cmd << " --username #{@login} --password #{@password}" if @login
145 cmd << " --username #{@login} --password #{@password}" if @login
144 diff = []
146 diff = []
145 shellout(cmd) do |io|
147 shellout(cmd) do |io|
146 io.each_line do |line|
148 io.each_line do |line|
147 diff << line
149 diff << line
148 end
150 end
149 end
151 end
150 return nil if $? && $?.exitstatus != 0
152 return nil if $? && $?.exitstatus != 0
151 DiffTableList.new diff, type
153 DiffTableList.new diff, type
152 rescue Errno::ENOENT => e
154 rescue Errno::ENOENT => e
153 raise CommandFailed
155 raise CommandFailed
154 end
156 end
155
157
156 def cat(path, identifier=nil)
158 def cat(path, identifier=nil)
157 identifier = (identifier and identifier.to_i > 0) ? identifier.to_i : "HEAD"
159 identifier = (identifier and identifier.to_i > 0) ? identifier.to_i : "HEAD"
158 cmd = "#{SVN_BIN} cat #{target(path)}@#{identifier}"
160 cmd = "#{SVN_BIN} cat #{target(path)}@#{identifier}"
159 cmd << " --username #{@login} --password #{@password}" if @login
161 cmd << " --username #{@login} --password #{@password}" if @login
160 cat = nil
162 cat = nil
161 shellout(cmd) do |io|
163 shellout(cmd) do |io|
162 io.binmode
164 io.binmode
163 cat = io.read
165 cat = io.read
164 end
166 end
165 return nil if $? && $?.exitstatus != 0
167 return nil if $? && $?.exitstatus != 0
166 cat
168 cat
167 rescue Errno::ENOENT => e
169 rescue Errno::ENOENT => e
168 raise CommandFailed
170 raise CommandFailed
169 end
171 end
170 end
172 end
171 end
173 end
172 end
174 end
173 end
175 end
General Comments 0
You need to be logged in to leave comments. Login now