Chris@0
|
1 # redMine - project management software
|
Chris@0
|
2 # Copyright (C) 2006-2007 Jean-Philippe Lang
|
Chris@0
|
3 #
|
Chris@0
|
4 # This program is free software; you can redistribute it and/or
|
Chris@0
|
5 # modify it under the terms of the GNU General Public License
|
Chris@0
|
6 # as published by the Free Software Foundation; either version 2
|
Chris@0
|
7 # of the License, or (at your option) any later version.
|
Chris@0
|
8 #
|
Chris@0
|
9 # This program is distributed in the hope that it will be useful,
|
Chris@0
|
10 # but WITHOUT ANY WARRANTY; without even the implied warranty of
|
Chris@0
|
11 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
Chris@0
|
12 # GNU General Public License for more details.
|
Chris@0
|
13 #
|
Chris@0
|
14 # You should have received a copy of the GNU General Public License
|
Chris@0
|
15 # along with this program; if not, write to the Free Software
|
Chris@0
|
16 # Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
Chris@0
|
17
|
Chris@0
|
18 require 'redmine/scm/adapters/abstract_adapter'
|
Chris@0
|
19
|
Chris@0
|
20 module Redmine
|
Chris@0
|
21 module Scm
|
Chris@0
|
22 module Adapters
|
Chris@0
|
23 class CvsAdapter < AbstractAdapter
|
Chris@0
|
24
|
Chris@0
|
25 # CVS executable name
|
Chris@210
|
26 CVS_BIN = Redmine::Configuration['scm_cvs_command'] || "cvs"
|
Chris@245
|
27
|
Chris@245
|
28 class << self
|
Chris@245
|
29 def client_command
|
Chris@245
|
30 @@bin ||= CVS_BIN
|
Chris@245
|
31 end
|
Chris@245
|
32
|
Chris@245
|
33 def sq_bin
|
Chris@245
|
34 @@sq_bin ||= shell_quote(CVS_BIN)
|
Chris@245
|
35 end
|
Chris@245
|
36
|
Chris@245
|
37 def client_version
|
Chris@245
|
38 @@client_version ||= (scm_command_version || [])
|
Chris@245
|
39 end
|
Chris@245
|
40
|
Chris@245
|
41 def client_available
|
Chris@245
|
42 client_version_above?([1, 12])
|
Chris@245
|
43 end
|
Chris@245
|
44
|
Chris@245
|
45 def scm_command_version
|
Chris@245
|
46 scm_version = scm_version_from_command_line.dup
|
Chris@245
|
47 if scm_version.respond_to?(:force_encoding)
|
Chris@245
|
48 scm_version.force_encoding('ASCII-8BIT')
|
Chris@245
|
49 end
|
Chris@245
|
50 if m = scm_version.match(%r{\A(.*?)((\d+\.)+\d+)}m)
|
Chris@245
|
51 m[2].scan(%r{\d+}).collect(&:to_i)
|
Chris@245
|
52 end
|
Chris@245
|
53 end
|
Chris@245
|
54
|
Chris@245
|
55 def scm_version_from_command_line
|
Chris@245
|
56 shellout("#{sq_bin} --version") { |io| io.read }.to_s
|
Chris@245
|
57 end
|
Chris@245
|
58 end
|
Chris@245
|
59
|
Chris@0
|
60 # Guidelines for the input:
|
Chris@0
|
61 # url -> the project-path, relative to the cvsroot (eg. module name)
|
Chris@0
|
62 # root_url -> the good old, sometimes damned, CVSROOT
|
Chris@0
|
63 # login -> unnecessary
|
Chris@0
|
64 # password -> unnecessary too
|
Chris@245
|
65 def initialize(url, root_url=nil, login=nil, password=nil,
|
Chris@245
|
66 path_encoding=nil)
|
Chris@0
|
67 @url = url
|
Chris@0
|
68 @login = login if login && !login.empty?
|
Chris@0
|
69 @password = (password || "") if @login
|
Chris@0
|
70 #TODO: better Exception here (IllegalArgumentException)
|
Chris@0
|
71 raise CommandFailed if root_url.blank?
|
Chris@0
|
72 @root_url = root_url
|
Chris@0
|
73 end
|
Chris@245
|
74
|
Chris@0
|
75 def root_url
|
Chris@0
|
76 @root_url
|
Chris@0
|
77 end
|
Chris@245
|
78
|
Chris@0
|
79 def url
|
Chris@0
|
80 @url
|
Chris@0
|
81 end
|
Chris@245
|
82
|
Chris@0
|
83 def info
|
Chris@0
|
84 logger.debug "<cvs> info"
|
Chris@0
|
85 Info.new({:root_url => @root_url, :lastrev => nil})
|
Chris@0
|
86 end
|
Chris@245
|
87
|
Chris@0
|
88 def get_previous_revision(revision)
|
Chris@0
|
89 CvsRevisionHelper.new(revision).prevRev
|
Chris@0
|
90 end
|
Chris@245
|
91
|
Chris@0
|
92 # Returns an Entries collection
|
Chris@0
|
93 # or nil if the given path doesn't exist in the repository
|
Chris@0
|
94 # this method is used by the repository-browser (aka LIST)
|
Chris@0
|
95 def entries(path=nil, identifier=nil)
|
Chris@0
|
96 logger.debug "<cvs> entries '#{path}' with identifier '#{identifier}'"
|
Chris@0
|
97 path_with_project="#{url}#{with_leading_slash(path)}"
|
Chris@0
|
98 entries = Entries.new
|
Chris@245
|
99 cmd = "#{self.class.sq_bin} -d #{shell_quote root_url} rls -e"
|
Chris@0
|
100 cmd << " -D \"#{time_to_cvstime(identifier)}\"" if identifier
|
Chris@0
|
101 cmd << " #{shell_quote path_with_project}"
|
Chris@0
|
102 shellout(cmd) do |io|
|
Chris@0
|
103 io.each_line(){|line|
|
Chris@0
|
104 fields=line.chop.split('/',-1)
|
Chris@0
|
105 logger.debug(">>InspectLine #{fields.inspect}")
|
Chris@245
|
106
|
Chris@0
|
107 if fields[0]!="D"
|
Chris@0
|
108 entries << Entry.new({:name => fields[-5],
|
Chris@0
|
109 #:path => fields[-4].include?(path)?fields[-4]:(path + "/"+ fields[-4]),
|
Chris@0
|
110 :path => "#{path}/#{fields[-5]}",
|
Chris@0
|
111 :kind => 'file',
|
Chris@0
|
112 :size => nil,
|
Chris@0
|
113 :lastrev => Revision.new({
|
Chris@0
|
114 :revision => fields[-4],
|
Chris@0
|
115 :name => fields[-4],
|
Chris@0
|
116 :time => Time.parse(fields[-3]),
|
Chris@0
|
117 :author => ''
|
Chris@0
|
118 })
|
Chris@0
|
119 })
|
Chris@0
|
120 else
|
Chris@0
|
121 entries << Entry.new({:name => fields[1],
|
Chris@0
|
122 :path => "#{path}/#{fields[1]}",
|
Chris@0
|
123 :kind => 'dir',
|
Chris@0
|
124 :size => nil,
|
Chris@0
|
125 :lastrev => nil
|
Chris@0
|
126 })
|
Chris@0
|
127 end
|
Chris@0
|
128 }
|
Chris@0
|
129 end
|
Chris@0
|
130 return nil if $? && $?.exitstatus != 0
|
Chris@0
|
131 entries.sort_by_name
|
Chris@245
|
132 end
|
Chris@0
|
133
|
Chris@0
|
134 STARTLOG="----------------------------"
|
Chris@0
|
135 ENDLOG ="============================================================================="
|
Chris@245
|
136
|
Chris@0
|
137 # Returns all revisions found between identifier_from and identifier_to
|
Chris@0
|
138 # in the repository. both identifier have to be dates or nil.
|
Chris@0
|
139 # these method returns nothing but yield every result in block
|
Chris@0
|
140 def revisions(path=nil, identifier_from=nil, identifier_to=nil, options={}, &block)
|
Chris@0
|
141 logger.debug "<cvs> revisions path:'#{path}',identifier_from #{identifier_from}, identifier_to #{identifier_to}"
|
Chris@245
|
142
|
Chris@0
|
143 path_with_project="#{url}#{with_leading_slash(path)}"
|
Chris@245
|
144 cmd = "#{self.class.sq_bin} -d #{shell_quote root_url} rlog"
|
Chris@210
|
145 cmd << " -d\">#{time_to_cvstime_rlog(identifier_from)}\"" if identifier_from
|
Chris@0
|
146 cmd << " #{shell_quote path_with_project}"
|
Chris@0
|
147 shellout(cmd) do |io|
|
Chris@0
|
148 state="entry_start"
|
Chris@0
|
149
|
Chris@0
|
150 commit_log=String.new
|
Chris@0
|
151 revision=nil
|
Chris@0
|
152 date=nil
|
Chris@0
|
153 author=nil
|
Chris@0
|
154 entry_path=nil
|
Chris@0
|
155 entry_name=nil
|
Chris@0
|
156 file_state=nil
|
Chris@0
|
157 branch_map=nil
|
Chris@0
|
158
|
Chris@245
|
159 io.each_line() do |line|
|
Chris@0
|
160
|
Chris@0
|
161 if state!="revision" && /^#{ENDLOG}/ =~ line
|
Chris@0
|
162 commit_log=String.new
|
Chris@0
|
163 revision=nil
|
Chris@0
|
164 state="entry_start"
|
Chris@0
|
165 end
|
Chris@0
|
166
|
Chris@0
|
167 if state=="entry_start"
|
Chris@0
|
168 branch_map=Hash.new
|
Chris@0
|
169 if /^RCS file: #{Regexp.escape(root_url_path)}\/#{Regexp.escape(path_with_project)}(.+),v$/ =~ line
|
Chris@0
|
170 entry_path = normalize_cvs_path($1)
|
Chris@0
|
171 entry_name = normalize_path(File.basename($1))
|
Chris@0
|
172 logger.debug("Path #{entry_path} <=> Name #{entry_name}")
|
Chris@0
|
173 elsif /^head: (.+)$/ =~ line
|
Chris@0
|
174 entry_headRev = $1 #unless entry.nil?
|
Chris@0
|
175 elsif /^symbolic names:/ =~ line
|
Chris@0
|
176 state="symbolic" #unless entry.nil?
|
Chris@0
|
177 elsif /^#{STARTLOG}/ =~ line
|
Chris@0
|
178 commit_log=String.new
|
Chris@0
|
179 state="revision"
|
Chris@0
|
180 end
|
Chris@0
|
181 next
|
Chris@0
|
182 elsif state=="symbolic"
|
Chris@0
|
183 if /^(.*):\s(.*)/ =~ (line.strip)
|
Chris@0
|
184 branch_map[$1]=$2
|
Chris@0
|
185 else
|
Chris@0
|
186 state="tags"
|
Chris@0
|
187 next
|
Chris@0
|
188 end
|
Chris@0
|
189 elsif state=="tags"
|
Chris@0
|
190 if /^#{STARTLOG}/ =~ line
|
Chris@0
|
191 commit_log = ""
|
Chris@0
|
192 state="revision"
|
Chris@0
|
193 elsif /^#{ENDLOG}/ =~ line
|
Chris@0
|
194 state="head"
|
Chris@0
|
195 end
|
Chris@0
|
196 next
|
Chris@0
|
197 elsif state=="revision"
|
Chris@245
|
198 if /^#{ENDLOG}/ =~ line || /^#{STARTLOG}/ =~ line
|
Chris@0
|
199 if revision
|
Chris@245
|
200
|
Chris@0
|
201 revHelper=CvsRevisionHelper.new(revision)
|
Chris@0
|
202 revBranch="HEAD"
|
Chris@0
|
203
|
Chris@0
|
204 branch_map.each() do |branch_name,branch_point|
|
Chris@0
|
205 if revHelper.is_in_branch_with_symbol(branch_point)
|
Chris@0
|
206 revBranch=branch_name
|
Chris@0
|
207 end
|
Chris@0
|
208 end
|
Chris@0
|
209
|
Chris@0
|
210 logger.debug("********** YIELD Revision #{revision}::#{revBranch}")
|
Chris@0
|
211
|
Chris@245
|
212 yield Revision.new({
|
Chris@0
|
213 :time => date,
|
Chris@0
|
214 :author => author,
|
Chris@0
|
215 :message=>commit_log.chomp,
|
Chris@0
|
216 :paths => [{
|
Chris@0
|
217 :revision => revision,
|
Chris@0
|
218 :branch=> revBranch,
|
Chris@0
|
219 :path=>entry_path,
|
Chris@0
|
220 :name=>entry_name,
|
Chris@0
|
221 :kind=>'file',
|
Chris@0
|
222 :action=>file_state
|
Chris@0
|
223 }]
|
Chris@245
|
224 })
|
Chris@0
|
225 end
|
Chris@245
|
226
|
Chris@0
|
227 commit_log=String.new
|
Chris@0
|
228 revision=nil
|
Chris@0
|
229
|
Chris@0
|
230 if /^#{ENDLOG}/ =~ line
|
Chris@0
|
231 state="entry_start"
|
Chris@0
|
232 end
|
Chris@0
|
233 next
|
Chris@0
|
234 end
|
Chris@245
|
235
|
Chris@0
|
236 if /^branches: (.+)$/ =~ line
|
Chris@0
|
237 #TODO: version.branch = $1
|
Chris@0
|
238 elsif /^revision (\d+(?:\.\d+)+).*$/ =~ line
|
Chris@0
|
239 revision = $1
|
Chris@0
|
240 elsif /^date:\s+(\d+.\d+.\d+\s+\d+:\d+:\d+)/ =~ line
|
Chris@0
|
241 date = Time.parse($1)
|
Chris@0
|
242 author = /author: ([^;]+)/.match(line)[1]
|
Chris@0
|
243 file_state = /state: ([^;]+)/.match(line)[1]
|
Chris@0
|
244 #TODO: linechanges only available in CVS.... maybe a feature our SVN implementation. i'm sure, they are
|
Chris@0
|
245 # useful for stats or something else
|
Chris@0
|
246 # linechanges =/lines: \+(\d+) -(\d+)/.match(line)
|
Chris@0
|
247 # unless linechanges.nil?
|
Chris@0
|
248 # version.line_plus = linechanges[1]
|
Chris@0
|
249 # version.line_minus = linechanges[2]
|
Chris@0
|
250 # else
|
Chris@0
|
251 # version.line_plus = 0
|
Chris@245
|
252 # version.line_minus = 0
|
Chris@245
|
253 # end
|
Chris@245
|
254 else
|
Chris@0
|
255 commit_log << line unless line =~ /^\*\*\* empty log message \*\*\*/
|
Chris@245
|
256 end
|
Chris@245
|
257 end
|
Chris@0
|
258 end
|
Chris@0
|
259 end
|
Chris@245
|
260 end
|
Chris@245
|
261
|
Chris@0
|
262 def diff(path, identifier_from, identifier_to=nil)
|
Chris@0
|
263 logger.debug "<cvs> diff path:'#{path}',identifier_from #{identifier_from}, identifier_to #{identifier_to}"
|
Chris@0
|
264 path_with_project="#{url}#{with_leading_slash(path)}"
|
Chris@245
|
265 cmd = "#{self.class.sq_bin} -d #{shell_quote root_url} rdiff -u -r#{identifier_to} -r#{identifier_from} #{shell_quote path_with_project}"
|
Chris@0
|
266 diff = []
|
Chris@0
|
267 shellout(cmd) do |io|
|
Chris@0
|
268 io.each_line do |line|
|
Chris@0
|
269 diff << line
|
Chris@0
|
270 end
|
Chris@0
|
271 end
|
Chris@0
|
272 return nil if $? && $?.exitstatus != 0
|
Chris@0
|
273 diff
|
Chris@245
|
274 end
|
Chris@245
|
275
|
Chris@0
|
276 def cat(path, identifier=nil)
|
Chris@0
|
277 identifier = (identifier) ? identifier : "HEAD"
|
Chris@0
|
278 logger.debug "<cvs> cat path:'#{path}',identifier #{identifier}"
|
Chris@0
|
279 path_with_project="#{url}#{with_leading_slash(path)}"
|
Chris@245
|
280 cmd = "#{self.class.sq_bin} -d #{shell_quote root_url} co"
|
Chris@0
|
281 cmd << " -D \"#{time_to_cvstime(identifier)}\"" if identifier
|
Chris@0
|
282 cmd << " -p #{shell_quote path_with_project}"
|
Chris@0
|
283 cat = nil
|
Chris@0
|
284 shellout(cmd) do |io|
|
Chris@245
|
285 io.binmode
|
Chris@0
|
286 cat = io.read
|
Chris@0
|
287 end
|
Chris@0
|
288 return nil if $? && $?.exitstatus != 0
|
Chris@0
|
289 cat
|
Chris@245
|
290 end
|
Chris@0
|
291
|
Chris@0
|
292 def annotate(path, identifier=nil)
|
Chris@119
|
293 identifier = (identifier) ? identifier.to_i : "HEAD"
|
Chris@0
|
294 logger.debug "<cvs> annotate path:'#{path}',identifier #{identifier}"
|
Chris@0
|
295 path_with_project="#{url}#{with_leading_slash(path)}"
|
Chris@245
|
296 cmd = "#{self.class.sq_bin} -d #{shell_quote root_url} rannotate -r#{identifier} #{shell_quote path_with_project}"
|
Chris@0
|
297 blame = Annotate.new
|
Chris@0
|
298 shellout(cmd) do |io|
|
Chris@0
|
299 io.each_line do |line|
|
Chris@0
|
300 next unless line =~ %r{^([\d\.]+)\s+\(([^\)]+)\s+[^\)]+\):\s(.*)$}
|
Chris@0
|
301 blame.add_line($3.rstrip, Revision.new(:revision => $1, :author => $2.strip))
|
Chris@0
|
302 end
|
Chris@0
|
303 end
|
Chris@0
|
304 return nil if $? && $?.exitstatus != 0
|
Chris@0
|
305 blame
|
Chris@0
|
306 end
|
Chris@245
|
307
|
Chris@0
|
308 private
|
Chris@245
|
309
|
Chris@0
|
310 # Returns the root url without the connexion string
|
Chris@0
|
311 # :pserver:anonymous@foo.bar:/path => /path
|
Chris@0
|
312 # :ext:cvsservername:/path => /path
|
Chris@0
|
313 def root_url_path
|
Chris@0
|
314 root_url.to_s.gsub(/^:.+:\d*/, '')
|
Chris@0
|
315 end
|
Chris@0
|
316
|
Chris@0
|
317 # convert a date/time into the CVS-format
|
Chris@0
|
318 def time_to_cvstime(time)
|
Chris@0
|
319 return nil if time.nil?
|
Chris@119
|
320 return Time.now if time == 'HEAD'
|
Chris@119
|
321
|
Chris@0
|
322 unless time.kind_of? Time
|
Chris@0
|
323 time = Time.parse(time)
|
Chris@0
|
324 end
|
Chris@0
|
325 return time.strftime("%Y-%m-%d %H:%M:%S")
|
Chris@0
|
326 end
|
Chris@210
|
327
|
Chris@210
|
328 def time_to_cvstime_rlog(time)
|
Chris@210
|
329 return nil if time.nil?
|
Chris@210
|
330 t1 = time.clone.localtime
|
Chris@210
|
331 return t1.strftime("%Y-%m-%d %H:%M:%S")
|
Chris@210
|
332 end
|
Chris@0
|
333
|
Chris@0
|
334 def normalize_cvs_path(path)
|
Chris@0
|
335 normalize_path(path.gsub(/Attic\//,''))
|
Chris@0
|
336 end
|
Chris@0
|
337
|
Chris@0
|
338 def normalize_path(path)
|
Chris@0
|
339 path.sub(/^(\/)*(.*)/,'\2').sub(/(.*)(,v)+/,'\1')
|
Chris@0
|
340 end
|
Chris@0
|
341 end
|
Chris@0
|
342
|
Chris@0
|
343 class CvsRevisionHelper
|
Chris@0
|
344 attr_accessor :complete_rev, :revision, :base, :branchid
|
Chris@0
|
345
|
Chris@0
|
346 def initialize(complete_rev)
|
Chris@0
|
347 @complete_rev = complete_rev
|
Chris@0
|
348 parseRevision()
|
Chris@0
|
349 end
|
Chris@0
|
350
|
Chris@0
|
351 def branchPoint
|
Chris@0
|
352 return @base
|
Chris@0
|
353 end
|
Chris@0
|
354
|
Chris@0
|
355 def branchVersion
|
Chris@0
|
356 if isBranchRevision
|
Chris@0
|
357 return @base+"."+@branchid
|
Chris@0
|
358 end
|
Chris@0
|
359 return @base
|
Chris@0
|
360 end
|
Chris@0
|
361
|
Chris@0
|
362 def isBranchRevision
|
Chris@0
|
363 !@branchid.nil?
|
Chris@0
|
364 end
|
Chris@0
|
365
|
Chris@0
|
366 def prevRev
|
Chris@0
|
367 unless @revision==0
|
Chris@0
|
368 return buildRevision(@revision-1)
|
Chris@0
|
369 end
|
Chris@0
|
370 return buildRevision(@revision)
|
Chris@0
|
371 end
|
Chris@0
|
372
|
Chris@0
|
373 def is_in_branch_with_symbol(branch_symbol)
|
Chris@0
|
374 bpieces=branch_symbol.split(".")
|
Chris@0
|
375 branch_start="#{bpieces[0..-3].join(".")}.#{bpieces[-1]}"
|
Chris@0
|
376 return (branchVersion==branch_start)
|
Chris@0
|
377 end
|
Chris@0
|
378
|
Chris@0
|
379 private
|
Chris@0
|
380 def buildRevision(rev)
|
Chris@0
|
381 if rev== 0
|
Chris@245
|
382 if @branchid.nil?
|
Chris@245
|
383 @base+".0"
|
Chris@245
|
384 else
|
Chris@245
|
385 @base
|
Chris@245
|
386 end
|
Chris@0
|
387 elsif @branchid.nil?
|
Chris@0
|
388 @base+"."+rev.to_s
|
Chris@0
|
389 else
|
Chris@0
|
390 @base+"."+@branchid+"."+rev.to_s
|
Chris@0
|
391 end
|
Chris@0
|
392 end
|
Chris@0
|
393
|
Chris@0
|
394 # Interpretiert die cvs revisionsnummern wie z.b. 1.14 oder 1.3.0.15
|
Chris@0
|
395 def parseRevision()
|
Chris@0
|
396 pieces=@complete_rev.split(".")
|
Chris@0
|
397 @revision=pieces.last.to_i
|
Chris@0
|
398 baseSize=1
|
Chris@0
|
399 baseSize+=(pieces.size/2)
|
Chris@0
|
400 @base=pieces[0..-baseSize].join(".")
|
Chris@0
|
401 if baseSize > 2
|
Chris@0
|
402 @branchid=pieces[-2]
|
Chris@0
|
403 end
|
Chris@0
|
404 end
|
Chris@0
|
405 end
|
Chris@0
|
406 end
|
Chris@0
|
407 end
|
Chris@0
|
408 end
|