##// END OF EJS Templates
merge with stable
Benoit Boissinot -
r9898:b5170b8b merge default
parent child Browse files
Show More
@@ -1,859 +1,864
1 # server.py - inotify status server
1 # server.py - inotify status server
2 #
2 #
3 # Copyright 2006, 2007, 2008 Bryan O'Sullivan <bos@serpentine.com>
3 # Copyright 2006, 2007, 2008 Bryan O'Sullivan <bos@serpentine.com>
4 # Copyright 2007, 2008 Brendan Cully <brendan@kublai.com>
4 # Copyright 2007, 2008 Brendan Cully <brendan@kublai.com>
5 #
5 #
6 # This software may be used and distributed according to the terms of the
6 # This software may be used and distributed according to the terms of the
7 # GNU General Public License version 2, incorporated herein by reference.
7 # GNU General Public License version 2, incorporated herein by reference.
8
8
9 from mercurial.i18n import _
9 from mercurial.i18n import _
10 from mercurial import cmdutil, osutil, util
10 from mercurial import cmdutil, osutil, util
11 import common
11 import common
12 import errno, os, select, socket, stat, struct, sys, tempfile, time
12 import errno, os, select, socket, stat, struct, sys, tempfile, time
13
13
14 try:
14 try:
15 import linux as inotify
15 import linux as inotify
16 from linux import watcher
16 from linux import watcher
17 except ImportError:
17 except ImportError:
18 raise
18 raise
19
19
20 class AlreadyStartedException(Exception): pass
20 class AlreadyStartedException(Exception): pass
21
21
22 def join(a, b):
22 def join(a, b):
23 if a:
23 if a:
24 if a[-1] == '/':
24 if a[-1] == '/':
25 return a + b
25 return a + b
26 return a + '/' + b
26 return a + '/' + b
27 return b
27 return b
28
28
29 def split(path):
29 def split(path):
30 c = path.rfind('/')
30 c = path.rfind('/')
31 if c == -1:
31 if c == -1:
32 return '', path
32 return '', path
33 return path[:c], path[c+1:]
33 return path[:c], path[c+1:]
34
34
35 walk_ignored_errors = (errno.ENOENT, errno.ENAMETOOLONG)
35 walk_ignored_errors = (errno.ENOENT, errno.ENAMETOOLONG)
36
36
37 def walkrepodirs(dirstate, absroot):
37 def walkrepodirs(dirstate, absroot):
38 '''Iterate over all subdirectories of this repo.
38 '''Iterate over all subdirectories of this repo.
39 Exclude the .hg directory, any nested repos, and ignored dirs.'''
39 Exclude the .hg directory, any nested repos, and ignored dirs.'''
40 def walkit(dirname, top):
40 def walkit(dirname, top):
41 fullpath = join(absroot, dirname)
41 fullpath = join(absroot, dirname)
42 try:
42 try:
43 for name, kind in osutil.listdir(fullpath):
43 for name, kind in osutil.listdir(fullpath):
44 if kind == stat.S_IFDIR:
44 if kind == stat.S_IFDIR:
45 if name == '.hg':
45 if name == '.hg':
46 if not top:
46 if not top:
47 return
47 return
48 else:
48 else:
49 d = join(dirname, name)
49 d = join(dirname, name)
50 if dirstate._ignore(d):
50 if dirstate._ignore(d):
51 continue
51 continue
52 for subdir in walkit(d, False):
52 for subdir in walkit(d, False):
53 yield subdir
53 yield subdir
54 except OSError, err:
54 except OSError, err:
55 if err.errno not in walk_ignored_errors:
55 if err.errno not in walk_ignored_errors:
56 raise
56 raise
57 yield fullpath
57 yield fullpath
58
58
59 return walkit('', True)
59 return walkit('', True)
60
60
61 def walk(dirstate, absroot, root):
61 def walk(dirstate, absroot, root):
62 '''Like os.walk, but only yields regular files.'''
62 '''Like os.walk, but only yields regular files.'''
63
63
64 # This function is critical to performance during startup.
64 # This function is critical to performance during startup.
65
65
66 def walkit(root, reporoot):
66 def walkit(root, reporoot):
67 files, dirs = [], []
67 files, dirs = [], []
68
68
69 try:
69 try:
70 fullpath = join(absroot, root)
70 fullpath = join(absroot, root)
71 for name, kind in osutil.listdir(fullpath):
71 for name, kind in osutil.listdir(fullpath):
72 if kind == stat.S_IFDIR:
72 if kind == stat.S_IFDIR:
73 if name == '.hg':
73 if name == '.hg':
74 if not reporoot:
74 if not reporoot:
75 return
75 return
76 else:
76 else:
77 dirs.append(name)
77 dirs.append(name)
78 path = join(root, name)
78 path = join(root, name)
79 if dirstate._ignore(path):
79 if dirstate._ignore(path):
80 continue
80 continue
81 for result in walkit(path, False):
81 for result in walkit(path, False):
82 yield result
82 yield result
83 elif kind in (stat.S_IFREG, stat.S_IFLNK):
83 elif kind in (stat.S_IFREG, stat.S_IFLNK):
84 files.append(name)
84 files.append(name)
85 yield fullpath, dirs, files
85 yield fullpath, dirs, files
86
86
87 except OSError, err:
87 except OSError, err:
88 if err.errno == errno.ENOTDIR:
88 if err.errno == errno.ENOTDIR:
89 # fullpath was a directory, but has since been replaced
89 # fullpath was a directory, but has since been replaced
90 # by a file.
90 # by a file.
91 yield fullpath, dirs, files
91 yield fullpath, dirs, files
92 elif err.errno not in walk_ignored_errors:
92 elif err.errno not in walk_ignored_errors:
93 raise
93 raise
94
94
95 return walkit(root, root == '')
95 return walkit(root, root == '')
96
96
97 def _explain_watch_limit(ui, dirstate, rootabs):
97 def _explain_watch_limit(ui, dirstate, rootabs):
98 path = '/proc/sys/fs/inotify/max_user_watches'
98 path = '/proc/sys/fs/inotify/max_user_watches'
99 try:
99 try:
100 limit = int(file(path).read())
100 limit = int(file(path).read())
101 except IOError, err:
101 except IOError, err:
102 if err.errno != errno.ENOENT:
102 if err.errno != errno.ENOENT:
103 raise
103 raise
104 raise util.Abort(_('this system does not seem to '
104 raise util.Abort(_('this system does not seem to '
105 'support inotify'))
105 'support inotify'))
106 ui.warn(_('*** the current per-user limit on the number '
106 ui.warn(_('*** the current per-user limit on the number '
107 'of inotify watches is %s\n') % limit)
107 'of inotify watches is %s\n') % limit)
108 ui.warn(_('*** this limit is too low to watch every '
108 ui.warn(_('*** this limit is too low to watch every '
109 'directory in this repository\n'))
109 'directory in this repository\n'))
110 ui.warn(_('*** counting directories: '))
110 ui.warn(_('*** counting directories: '))
111 ndirs = len(list(walkrepodirs(dirstate, rootabs)))
111 ndirs = len(list(walkrepodirs(dirstate, rootabs)))
112 ui.warn(_('found %d\n') % ndirs)
112 ui.warn(_('found %d\n') % ndirs)
113 newlimit = min(limit, 1024)
113 newlimit = min(limit, 1024)
114 while newlimit < ((limit + ndirs) * 1.1):
114 while newlimit < ((limit + ndirs) * 1.1):
115 newlimit *= 2
115 newlimit *= 2
116 ui.warn(_('*** to raise the limit from %d to %d (run as root):\n') %
116 ui.warn(_('*** to raise the limit from %d to %d (run as root):\n') %
117 (limit, newlimit))
117 (limit, newlimit))
118 ui.warn(_('*** echo %d > %s\n') % (newlimit, path))
118 ui.warn(_('*** echo %d > %s\n') % (newlimit, path))
119 raise util.Abort(_('cannot watch %s until inotify watch limit is raised')
119 raise util.Abort(_('cannot watch %s until inotify watch limit is raised')
120 % rootabs)
120 % rootabs)
121
121
122 class pollable(object):
122 class pollable(object):
123 """
123 """
124 Interface to support polling.
124 Interface to support polling.
125 The file descriptor returned by fileno() is registered to a polling
125 The file descriptor returned by fileno() is registered to a polling
126 object.
126 object.
127 Usage:
127 Usage:
128 Every tick, check if an event has happened since the last tick:
128 Every tick, check if an event has happened since the last tick:
129 * If yes, call handle_events
129 * If yes, call handle_events
130 * If no, call handle_timeout
130 * If no, call handle_timeout
131 """
131 """
132 poll_events = select.POLLIN
132 poll_events = select.POLLIN
133 instances = {}
133 instances = {}
134 poll = select.poll()
134 poll = select.poll()
135
135
136 def fileno(self):
136 def fileno(self):
137 raise NotImplementedError
137 raise NotImplementedError
138
138
139 def handle_events(self, events):
139 def handle_events(self, events):
140 raise NotImplementedError
140 raise NotImplementedError
141
141
142 def handle_timeout(self):
142 def handle_timeout(self):
143 raise NotImplementedError
143 raise NotImplementedError
144
144
145 def shutdown(self):
145 def shutdown(self):
146 raise NotImplementedError
146 raise NotImplementedError
147
147
148 def register(self, timeout):
148 def register(self, timeout):
149 fd = self.fileno()
149 fd = self.fileno()
150
150
151 pollable.poll.register(fd, pollable.poll_events)
151 pollable.poll.register(fd, pollable.poll_events)
152 pollable.instances[fd] = self
152 pollable.instances[fd] = self
153
153
154 self.registered = True
154 self.registered = True
155 self.timeout = timeout
155 self.timeout = timeout
156
156
157 def unregister(self):
157 def unregister(self):
158 pollable.poll.unregister(self)
158 pollable.poll.unregister(self)
159 self.registered = False
159 self.registered = False
160
160
161 @classmethod
161 @classmethod
162 def run(cls):
162 def run(cls):
163 while True:
163 while True:
164 timeout = None
164 timeout = None
165 timeobj = None
165 timeobj = None
166 for obj in cls.instances.itervalues():
166 for obj in cls.instances.itervalues():
167 if obj.timeout is not None and (timeout is None or obj.timeout < timeout):
167 if obj.timeout is not None and (timeout is None or obj.timeout < timeout):
168 timeout, timeobj = obj.timeout, obj
168 timeout, timeobj = obj.timeout, obj
169 try:
169 try:
170 events = cls.poll.poll(timeout)
170 events = cls.poll.poll(timeout)
171 except select.error, err:
171 except select.error, err:
172 if err[0] == errno.EINTR:
172 if err[0] == errno.EINTR:
173 continue
173 continue
174 raise
174 raise
175 if events:
175 if events:
176 by_fd = {}
176 by_fd = {}
177 for fd, event in events:
177 for fd, event in events:
178 by_fd.setdefault(fd, []).append(event)
178 by_fd.setdefault(fd, []).append(event)
179
179
180 for fd, events in by_fd.iteritems():
180 for fd, events in by_fd.iteritems():
181 cls.instances[fd].handle_pollevents(events)
181 cls.instances[fd].handle_pollevents(events)
182
182
183 elif timeobj:
183 elif timeobj:
184 timeobj.handle_timeout()
184 timeobj.handle_timeout()
185
185
186 def eventaction(code):
186 def eventaction(code):
187 """
187 """
188 Decorator to help handle events in repowatcher
188 Decorator to help handle events in repowatcher
189 """
189 """
190 def decorator(f):
190 def decorator(f):
191 def wrapper(self, wpath):
191 def wrapper(self, wpath):
192 if code == 'm' and wpath in self.lastevent and \
192 if code == 'm' and wpath in self.lastevent and \
193 self.lastevent[wpath] in 'cm':
193 self.lastevent[wpath] in 'cm':
194 return
194 return
195 self.lastevent[wpath] = code
195 self.lastevent[wpath] = code
196 self.timeout = 250
196 self.timeout = 250
197
197
198 f(self, wpath)
198 f(self, wpath)
199
199
200 wrapper.func_name = f.func_name
200 wrapper.func_name = f.func_name
201 return wrapper
201 return wrapper
202 return decorator
202 return decorator
203
203
204 class directory(object):
204 class directory(object):
205 """
205 """
206 Representing a directory
206 Representing a directory
207
207
208 * path is the relative path from repo root to this directory
208 * path is the relative path from repo root to this directory
209 * files is a dict listing the files in this directory
209 * files is a dict listing the files in this directory
210 - keys are file names
210 - keys are file names
211 - values are file status
211 - values are file status
212 * dirs is a dict listing the subdirectories
212 * dirs is a dict listing the subdirectories
213 - key are subdirectories names
213 - key are subdirectories names
214 - values are directory objects
214 - values are directory objects
215 """
215 """
216 def __init__(self, relpath=''):
216 def __init__(self, relpath=''):
217 self.path = relpath
217 self.path = relpath
218 self.files = {}
218 self.files = {}
219 self.dirs = {}
219 self.dirs = {}
220
220
221 def dir(self, relpath):
221 def dir(self, relpath):
222 """
222 """
223 Returns the directory contained at the relative path relpath.
223 Returns the directory contained at the relative path relpath.
224 Creates the intermediate directories if necessary.
224 Creates the intermediate directories if necessary.
225 """
225 """
226 if not relpath:
226 if not relpath:
227 return self
227 return self
228 l = relpath.split('/')
228 l = relpath.split('/')
229 ret = self
229 ret = self
230 while l:
230 while l:
231 next = l.pop(0)
231 next = l.pop(0)
232 try:
232 try:
233 ret = ret.dirs[next]
233 ret = ret.dirs[next]
234 except KeyError:
234 except KeyError:
235 d = directory(join(ret.path, next))
235 d = directory(join(ret.path, next))
236 ret.dirs[next] = d
236 ret.dirs[next] = d
237 ret = d
237 ret = d
238 return ret
238 return ret
239
239
240 def walk(self, states, visited=None):
240 def walk(self, states, visited=None):
241 """
241 """
242 yield (filename, status) pairs for items in the trees
242 yield (filename, status) pairs for items in the trees
243 that have status in states.
243 that have status in states.
244 filenames are relative to the repo root
244 filenames are relative to the repo root
245 """
245 """
246 for file, st in self.files.iteritems():
246 for file, st in self.files.iteritems():
247 if st in states:
247 if st in states:
248 yield join(self.path, file), st
248 yield join(self.path, file), st
249 for dir in self.dirs.itervalues():
249 for dir in self.dirs.itervalues():
250 if visited is not None:
250 if visited is not None:
251 visited.add(dir.path)
251 visited.add(dir.path)
252 for e in dir.walk(states):
252 for e in dir.walk(states):
253 yield e
253 yield e
254
254
255 def lookup(self, states, path, visited):
255 def lookup(self, states, path, visited):
256 """
256 """
257 yield root-relative filenames that match path, and whose
257 yield root-relative filenames that match path, and whose
258 status are in states:
258 status are in states:
259 * if path is a file, yield path
259 * if path is a file, yield path
260 * if path is a directory, yield directory files
260 * if path is a directory, yield directory files
261 * if path is not tracked, yield nothing
261 * if path is not tracked, yield nothing
262 """
262 """
263 if path[-1] == '/':
263 if path[-1] == '/':
264 path = path[:-1]
264 path = path[:-1]
265
265
266 paths = path.split('/')
266 paths = path.split('/')
267
267
268 # we need to check separately for last node
268 # we need to check separately for last node
269 last = paths.pop()
269 last = paths.pop()
270
270
271 tree = self
271 tree = self
272 try:
272 try:
273 for dir in paths:
273 for dir in paths:
274 tree = tree.dirs[dir]
274 tree = tree.dirs[dir]
275 except KeyError:
275 except KeyError:
276 # path is not tracked
276 # path is not tracked
277 visited.add(tree.path)
277 visited.add(tree.path)
278 return
278 return
279
279
280 try:
280 try:
281 # if path is a directory, walk it
281 # if path is a directory, walk it
282 target = tree.dirs[last]
282 target = tree.dirs[last]
283 visited.add(target.path)
283 visited.add(target.path)
284 for file, st in target.walk(states, visited):
284 for file, st in target.walk(states, visited):
285 yield file
285 yield file
286 except KeyError:
286 except KeyError:
287 try:
287 try:
288 if tree.files[last] in states:
288 if tree.files[last] in states:
289 # path is a file
289 # path is a file
290 visited.add(tree.path)
290 visited.add(tree.path)
291 yield path
291 yield path
292 except KeyError:
292 except KeyError:
293 # path is not tracked
293 # path is not tracked
294 pass
294 pass
295
295
296 class repowatcher(pollable):
296 class repowatcher(pollable):
297 """
297 """
298 Watches inotify events
298 Watches inotify events
299 """
299 """
300 statuskeys = 'almr!?'
300 statuskeys = 'almr!?'
301 mask = (
301 mask = (
302 inotify.IN_ATTRIB |
302 inotify.IN_ATTRIB |
303 inotify.IN_CREATE |
303 inotify.IN_CREATE |
304 inotify.IN_DELETE |
304 inotify.IN_DELETE |
305 inotify.IN_DELETE_SELF |
305 inotify.IN_DELETE_SELF |
306 inotify.IN_MODIFY |
306 inotify.IN_MODIFY |
307 inotify.IN_MOVED_FROM |
307 inotify.IN_MOVED_FROM |
308 inotify.IN_MOVED_TO |
308 inotify.IN_MOVED_TO |
309 inotify.IN_MOVE_SELF |
309 inotify.IN_MOVE_SELF |
310 inotify.IN_ONLYDIR |
310 inotify.IN_ONLYDIR |
311 inotify.IN_UNMOUNT |
311 inotify.IN_UNMOUNT |
312 0)
312 0)
313
313
314 def __init__(self, ui, dirstate, root):
314 def __init__(self, ui, dirstate, root):
315 self.ui = ui
315 self.ui = ui
316 self.dirstate = dirstate
316 self.dirstate = dirstate
317
317
318 self.wprefix = join(root, '')
318 self.wprefix = join(root, '')
319 self.prefixlen = len(self.wprefix)
319 self.prefixlen = len(self.wprefix)
320 try:
320 try:
321 self.watcher = watcher.watcher()
321 self.watcher = watcher.watcher()
322 except OSError, err:
322 except OSError, err:
323 raise util.Abort(_('inotify service not available: %s') %
323 raise util.Abort(_('inotify service not available: %s') %
324 err.strerror)
324 err.strerror)
325 self.threshold = watcher.threshold(self.watcher)
325 self.threshold = watcher.threshold(self.watcher)
326 self.fileno = self.watcher.fileno
326 self.fileno = self.watcher.fileno
327
327
328 self.tree = directory()
328 self.tree = directory()
329 self.statcache = {}
329 self.statcache = {}
330 self.statustrees = dict([(s, directory()) for s in self.statuskeys])
330 self.statustrees = dict([(s, directory()) for s in self.statuskeys])
331
331
332 self.last_event = None
332 self.last_event = None
333
333
334 self.lastevent = {}
334 self.lastevent = {}
335
335
336 self.register(timeout=None)
336 self.register(timeout=None)
337
337
338 self.ds_info = self.dirstate_info()
338 self.ds_info = self.dirstate_info()
339 self.handle_timeout()
339 self.handle_timeout()
340 self.scan()
340 self.scan()
341
341
342 def event_time(self):
342 def event_time(self):
343 last = self.last_event
343 last = self.last_event
344 now = time.time()
344 now = time.time()
345 self.last_event = now
345 self.last_event = now
346
346
347 if last is None:
347 if last is None:
348 return 'start'
348 return 'start'
349 delta = now - last
349 delta = now - last
350 if delta < 5:
350 if delta < 5:
351 return '+%.3f' % delta
351 return '+%.3f' % delta
352 if delta < 50:
352 if delta < 50:
353 return '+%.2f' % delta
353 return '+%.2f' % delta
354 return '+%.1f' % delta
354 return '+%.1f' % delta
355
355
356 def dirstate_info(self):
356 def dirstate_info(self):
357 try:
357 try:
358 st = os.lstat(self.wprefix + '.hg/dirstate')
358 st = os.lstat(self.wprefix + '.hg/dirstate')
359 return st.st_mtime, st.st_ino
359 return st.st_mtime, st.st_ino
360 except OSError, err:
360 except OSError, err:
361 if err.errno != errno.ENOENT:
361 if err.errno != errno.ENOENT:
362 raise
362 raise
363 return 0, 0
363 return 0, 0
364
364
365 def add_watch(self, path, mask):
365 def add_watch(self, path, mask):
366 if not path:
366 if not path:
367 return
367 return
368 if self.watcher.path(path) is None:
368 if self.watcher.path(path) is None:
369 if self.ui.debugflag:
369 if self.ui.debugflag:
370 self.ui.note(_('watching %r\n') % path[self.prefixlen:])
370 self.ui.note(_('watching %r\n') % path[self.prefixlen:])
371 try:
371 try:
372 self.watcher.add(path, mask)
372 self.watcher.add(path, mask)
373 except OSError, err:
373 except OSError, err:
374 if err.errno in (errno.ENOENT, errno.ENOTDIR):
374 if err.errno in (errno.ENOENT, errno.ENOTDIR):
375 return
375 return
376 if err.errno != errno.ENOSPC:
376 if err.errno != errno.ENOSPC:
377 raise
377 raise
378 _explain_watch_limit(self.ui, self.dirstate, self.wprefix)
378 _explain_watch_limit(self.ui, self.dirstate, self.wprefix)
379
379
380 def setup(self):
380 def setup(self):
381 self.ui.note(_('watching directories under %r\n') % self.wprefix)
381 self.ui.note(_('watching directories under %r\n') % self.wprefix)
382 self.add_watch(self.wprefix + '.hg', inotify.IN_DELETE)
382 self.add_watch(self.wprefix + '.hg', inotify.IN_DELETE)
383 self.check_dirstate()
383 self.check_dirstate()
384
384
385 def filestatus(self, fn, st):
385 def filestatus(self, fn, st):
386 try:
386 try:
387 type_, mode, size, time = self.dirstate._map[fn][:4]
387 type_, mode, size, time = self.dirstate._map[fn][:4]
388 except KeyError:
388 except KeyError:
389 type_ = '?'
389 type_ = '?'
390 if type_ == 'n':
390 if type_ == 'n':
391 st_mode, st_size, st_mtime = st
391 st_mode, st_size, st_mtime = st
392 if size == -1:
392 if size == -1:
393 return 'l'
393 return 'l'
394 if size and (size != st_size or (mode ^ st_mode) & 0100):
394 if size and (size != st_size or (mode ^ st_mode) & 0100):
395 return 'm'
395 return 'm'
396 if time != int(st_mtime):
396 if time != int(st_mtime):
397 return 'l'
397 return 'l'
398 return 'n'
398 return 'n'
399 if type_ == '?' and self.dirstate._ignore(fn):
399 if type_ == '?' and self.dirstate._ignore(fn):
400 return 'i'
400 return 'i'
401 return type_
401 return type_
402
402
403 def updatefile(self, wfn, osstat):
403 def updatefile(self, wfn, osstat):
404 '''
404 '''
405 update the file entry of an existing file.
405 update the file entry of an existing file.
406
406
407 osstat: (mode, size, time) tuple, as returned by os.lstat(wfn)
407 osstat: (mode, size, time) tuple, as returned by os.lstat(wfn)
408 '''
408 '''
409
409
410 self._updatestatus(wfn, self.filestatus(wfn, osstat))
410 self._updatestatus(wfn, self.filestatus(wfn, osstat))
411
411
412 def deletefile(self, wfn, oldstatus):
412 def deletefile(self, wfn, oldstatus):
413 '''
413 '''
414 update the entry of a file which has been deleted.
414 update the entry of a file which has been deleted.
415
415
416 oldstatus: char in statuskeys, status of the file before deletion
416 oldstatus: char in statuskeys, status of the file before deletion
417 '''
417 '''
418 if oldstatus == 'r':
418 if oldstatus == 'r':
419 newstatus = 'r'
419 newstatus = 'r'
420 elif oldstatus in 'almn':
420 elif oldstatus in 'almn':
421 newstatus = '!'
421 newstatus = '!'
422 else:
422 else:
423 newstatus = None
423 newstatus = None
424
424
425 self.statcache.pop(wfn, None)
425 self.statcache.pop(wfn, None)
426 self._updatestatus(wfn, newstatus)
426 self._updatestatus(wfn, newstatus)
427
427
428 def _updatestatus(self, wfn, newstatus):
428 def _updatestatus(self, wfn, newstatus):
429 '''
429 '''
430 Update the stored status of a file.
430 Update the stored status of a file.
431
431
432 newstatus: - char in (statuskeys + 'ni'), new status to apply.
432 newstatus: - char in (statuskeys + 'ni'), new status to apply.
433 - or None, to stop tracking wfn
433 - or None, to stop tracking wfn
434 '''
434 '''
435 root, fn = split(wfn)
435 root, fn = split(wfn)
436 d = self.tree.dir(root)
436 d = self.tree.dir(root)
437
437
438 oldstatus = d.files.get(fn)
438 oldstatus = d.files.get(fn)
439 # oldstatus can be either:
439 # oldstatus can be either:
440 # - None : fn is new
440 # - None : fn is new
441 # - a char in statuskeys: fn is a (tracked) file
441 # - a char in statuskeys: fn is a (tracked) file
442
442
443 if self.ui.debugflag and oldstatus != newstatus:
443 if self.ui.debugflag and oldstatus != newstatus:
444 self.ui.note(_('status: %r %s -> %s\n') %
444 self.ui.note(_('status: %r %s -> %s\n') %
445 (wfn, oldstatus, newstatus))
445 (wfn, oldstatus, newstatus))
446
446
447 if oldstatus and oldstatus in self.statuskeys \
447 if oldstatus and oldstatus in self.statuskeys \
448 and oldstatus != newstatus:
448 and oldstatus != newstatus:
449 del self.statustrees[oldstatus].dir(root).files[fn]
449 del self.statustrees[oldstatus].dir(root).files[fn]
450
450
451 if newstatus in (None, 'i'):
451 if newstatus in (None, 'i'):
452 d.files.pop(fn, None)
452 d.files.pop(fn, None)
453 elif oldstatus != newstatus:
453 elif oldstatus != newstatus:
454 d.files[fn] = newstatus
454 d.files[fn] = newstatus
455 if newstatus != 'n':
455 if newstatus != 'n':
456 self.statustrees[newstatus].dir(root).files[fn] = newstatus
456 self.statustrees[newstatus].dir(root).files[fn] = newstatus
457
457
458
458
459 def check_deleted(self, key):
459 def check_deleted(self, key):
460 # Files that had been deleted but were present in the dirstate
460 # Files that had been deleted but were present in the dirstate
461 # may have vanished from the dirstate; we must clean them up.
461 # may have vanished from the dirstate; we must clean them up.
462 nuke = []
462 nuke = []
463 for wfn, ignore in self.statustrees[key].walk(key):
463 for wfn, ignore in self.statustrees[key].walk(key):
464 if wfn not in self.dirstate:
464 if wfn not in self.dirstate:
465 nuke.append(wfn)
465 nuke.append(wfn)
466 for wfn in nuke:
466 for wfn in nuke:
467 root, fn = split(wfn)
467 root, fn = split(wfn)
468 del self.statustrees[key].dir(root).files[fn]
468 del self.statustrees[key].dir(root).files[fn]
469 del self.tree.dir(root).files[fn]
469 del self.tree.dir(root).files[fn]
470
470
471 def scan(self, topdir=''):
471 def scan(self, topdir=''):
472 ds = self.dirstate._map.copy()
472 ds = self.dirstate._map.copy()
473 self.add_watch(join(self.wprefix, topdir), self.mask)
473 self.add_watch(join(self.wprefix, topdir), self.mask)
474 for root, dirs, files in walk(self.dirstate, self.wprefix, topdir):
474 for root, dirs, files in walk(self.dirstate, self.wprefix, topdir):
475 for d in dirs:
475 for d in dirs:
476 self.add_watch(join(root, d), self.mask)
476 self.add_watch(join(root, d), self.mask)
477 wroot = root[self.prefixlen:]
477 wroot = root[self.prefixlen:]
478 for fn in files:
478 for fn in files:
479 wfn = join(wroot, fn)
479 wfn = join(wroot, fn)
480 self.updatefile(wfn, self.getstat(wfn))
480 self.updatefile(wfn, self.getstat(wfn))
481 ds.pop(wfn, None)
481 ds.pop(wfn, None)
482 wtopdir = topdir
482 wtopdir = topdir
483 if wtopdir and wtopdir[-1] != '/':
483 if wtopdir and wtopdir[-1] != '/':
484 wtopdir += '/'
484 wtopdir += '/'
485 for wfn, state in ds.iteritems():
485 for wfn, state in ds.iteritems():
486 if not wfn.startswith(wtopdir):
486 if not wfn.startswith(wtopdir):
487 continue
487 continue
488 try:
488 try:
489 st = self.stat(wfn)
489 st = self.stat(wfn)
490 except OSError:
490 except OSError:
491 status = state[0]
491 status = state[0]
492 self.deletefile(wfn, status)
492 self.deletefile(wfn, status)
493 else:
493 else:
494 self.updatefile(wfn, st)
494 self.updatefile(wfn, st)
495 self.check_deleted('!')
495 self.check_deleted('!')
496 self.check_deleted('r')
496 self.check_deleted('r')
497
497
498 def check_dirstate(self):
498 def check_dirstate(self):
499 ds_info = self.dirstate_info()
499 ds_info = self.dirstate_info()
500 if ds_info == self.ds_info:
500 if ds_info == self.ds_info:
501 return
501 return
502 self.ds_info = ds_info
502 self.ds_info = ds_info
503 if not self.ui.debugflag:
503 if not self.ui.debugflag:
504 self.last_event = None
504 self.last_event = None
505 self.ui.note(_('%s dirstate reload\n') % self.event_time())
505 self.ui.note(_('%s dirstate reload\n') % self.event_time())
506 self.dirstate.invalidate()
506 self.dirstate.invalidate()
507 self.handle_timeout()
507 self.handle_timeout()
508 self.scan()
508 self.scan()
509 self.ui.note(_('%s end dirstate reload\n') % self.event_time())
509 self.ui.note(_('%s end dirstate reload\n') % self.event_time())
510
510
511 def update_hgignore(self):
511 def update_hgignore(self):
512 # An update of the ignore file can potentially change the
512 # An update of the ignore file can potentially change the
513 # states of all unknown and ignored files.
513 # states of all unknown and ignored files.
514
514
515 # XXX If the user has other ignore files outside the repo, or
515 # XXX If the user has other ignore files outside the repo, or
516 # changes their list of ignore files at run time, we'll
516 # changes their list of ignore files at run time, we'll
517 # potentially never see changes to them. We could get the
517 # potentially never see changes to them. We could get the
518 # client to report to us what ignore data they're using.
518 # client to report to us what ignore data they're using.
519 # But it's easier to do nothing than to open that can of
519 # But it's easier to do nothing than to open that can of
520 # worms.
520 # worms.
521
521
522 if '_ignore' in self.dirstate.__dict__:
522 if '_ignore' in self.dirstate.__dict__:
523 delattr(self.dirstate, '_ignore')
523 delattr(self.dirstate, '_ignore')
524 self.ui.note(_('rescanning due to .hgignore change\n'))
524 self.ui.note(_('rescanning due to .hgignore change\n'))
525 self.handle_timeout()
525 self.handle_timeout()
526 self.scan()
526 self.scan()
527
527
528 def getstat(self, wpath):
528 def getstat(self, wpath):
529 try:
529 try:
530 return self.statcache[wpath]
530 return self.statcache[wpath]
531 except KeyError:
531 except KeyError:
532 try:
532 try:
533 return self.stat(wpath)
533 return self.stat(wpath)
534 except OSError, err:
534 except OSError, err:
535 if err.errno != errno.ENOENT:
535 if err.errno != errno.ENOENT:
536 raise
536 raise
537
537
538 def stat(self, wpath):
538 def stat(self, wpath):
539 try:
539 try:
540 st = os.lstat(join(self.wprefix, wpath))
540 st = os.lstat(join(self.wprefix, wpath))
541 ret = st.st_mode, st.st_size, st.st_mtime
541 ret = st.st_mode, st.st_size, st.st_mtime
542 self.statcache[wpath] = ret
542 self.statcache[wpath] = ret
543 return ret
543 return ret
544 except OSError:
544 except OSError:
545 self.statcache.pop(wpath, None)
545 self.statcache.pop(wpath, None)
546 raise
546 raise
547
547
548 @eventaction('c')
548 @eventaction('c')
549 def created(self, wpath):
549 def created(self, wpath):
550 if wpath == '.hgignore':
550 if wpath == '.hgignore':
551 self.update_hgignore()
551 self.update_hgignore()
552 try:
552 try:
553 st = self.stat(wpath)
553 st = self.stat(wpath)
554 if stat.S_ISREG(st[0]):
554 if stat.S_ISREG(st[0]):
555 self.updatefile(wpath, st)
555 self.updatefile(wpath, st)
556 except OSError:
556 except OSError:
557 pass
557 pass
558
558
559 @eventaction('m')
559 @eventaction('m')
560 def modified(self, wpath):
560 def modified(self, wpath):
561 if wpath == '.hgignore':
561 if wpath == '.hgignore':
562 self.update_hgignore()
562 self.update_hgignore()
563 try:
563 try:
564 st = self.stat(wpath)
564 st = self.stat(wpath)
565 if stat.S_ISREG(st[0]):
565 if stat.S_ISREG(st[0]):
566 if self.dirstate[wpath] in 'lmn':
566 if self.dirstate[wpath] in 'lmn':
567 self.updatefile(wpath, st)
567 self.updatefile(wpath, st)
568 except OSError:
568 except OSError:
569 pass
569 pass
570
570
571 @eventaction('d')
571 @eventaction('d')
572 def deleted(self, wpath):
572 def deleted(self, wpath):
573 if wpath == '.hgignore':
573 if wpath == '.hgignore':
574 self.update_hgignore()
574 self.update_hgignore()
575 elif wpath.startswith('.hg/'):
575 elif wpath.startswith('.hg/'):
576 if wpath == '.hg/wlock':
576 if wpath == '.hg/wlock':
577 self.check_dirstate()
577 self.check_dirstate()
578 return
578 return
579
579
580 self.deletefile(wpath, self.dirstate[wpath])
580 self.deletefile(wpath, self.dirstate[wpath])
581
581
582 def process_create(self, wpath, evt):
582 def process_create(self, wpath, evt):
583 if self.ui.debugflag:
583 if self.ui.debugflag:
584 self.ui.note(_('%s event: created %s\n') %
584 self.ui.note(_('%s event: created %s\n') %
585 (self.event_time(), wpath))
585 (self.event_time(), wpath))
586
586
587 if evt.mask & inotify.IN_ISDIR:
587 if evt.mask & inotify.IN_ISDIR:
588 self.scan(wpath)
588 self.scan(wpath)
589 else:
589 else:
590 self.created(wpath)
590 self.created(wpath)
591
591
592 def process_delete(self, wpath, evt):
592 def process_delete(self, wpath, evt):
593 if self.ui.debugflag:
593 if self.ui.debugflag:
594 self.ui.note(_('%s event: deleted %s\n') %
594 self.ui.note(_('%s event: deleted %s\n') %
595 (self.event_time(), wpath))
595 (self.event_time(), wpath))
596
596
597 if evt.mask & inotify.IN_ISDIR:
597 if evt.mask & inotify.IN_ISDIR:
598 tree = self.tree.dir(wpath)
598 tree = self.tree.dir(wpath)
599 todelete = [wfn for wfn, ignore in tree.walk('?')]
599 todelete = [wfn for wfn, ignore in tree.walk('?')]
600 for fn in todelete:
600 for fn in todelete:
601 self.deletefile(fn, '?')
601 self.deletefile(fn, '?')
602 self.scan(wpath)
602 self.scan(wpath)
603 else:
603 else:
604 self.deleted(wpath)
604 self.deleted(wpath)
605
605
606 def process_modify(self, wpath, evt):
606 def process_modify(self, wpath, evt):
607 if self.ui.debugflag:
607 if self.ui.debugflag:
608 self.ui.note(_('%s event: modified %s\n') %
608 self.ui.note(_('%s event: modified %s\n') %
609 (self.event_time(), wpath))
609 (self.event_time(), wpath))
610
610
611 if not (evt.mask & inotify.IN_ISDIR):
611 if not (evt.mask & inotify.IN_ISDIR):
612 self.modified(wpath)
612 self.modified(wpath)
613
613
614 def process_unmount(self, evt):
614 def process_unmount(self, evt):
615 self.ui.warn(_('filesystem containing %s was unmounted\n') %
615 self.ui.warn(_('filesystem containing %s was unmounted\n') %
616 evt.fullpath)
616 evt.fullpath)
617 sys.exit(0)
617 sys.exit(0)
618
618
619 def handle_pollevents(self, events):
619 def handle_pollevents(self, events):
620 if self.ui.debugflag:
620 if self.ui.debugflag:
621 self.ui.note(_('%s readable: %d bytes\n') %
621 self.ui.note(_('%s readable: %d bytes\n') %
622 (self.event_time(), self.threshold.readable()))
622 (self.event_time(), self.threshold.readable()))
623 if not self.threshold():
623 if not self.threshold():
624 if self.registered:
624 if self.registered:
625 if self.ui.debugflag:
625 if self.ui.debugflag:
626 self.ui.note(_('%s below threshold - unhooking\n') %
626 self.ui.note(_('%s below threshold - unhooking\n') %
627 (self.event_time()))
627 (self.event_time()))
628 self.unregister()
628 self.unregister()
629 self.timeout = 250
629 self.timeout = 250
630 else:
630 else:
631 self.read_events()
631 self.read_events()
632
632
633 def read_events(self, bufsize=None):
633 def read_events(self, bufsize=None):
634 events = self.watcher.read(bufsize)
634 events = self.watcher.read(bufsize)
635 if self.ui.debugflag:
635 if self.ui.debugflag:
636 self.ui.note(_('%s reading %d events\n') %
636 self.ui.note(_('%s reading %d events\n') %
637 (self.event_time(), len(events)))
637 (self.event_time(), len(events)))
638 for evt in events:
638 for evt in events:
639 assert evt.fullpath.startswith(self.wprefix)
639 assert evt.fullpath.startswith(self.wprefix)
640 wpath = evt.fullpath[self.prefixlen:]
640 wpath = evt.fullpath[self.prefixlen:]
641
641
642 # paths have been normalized, wpath never ends with a '/'
642 # paths have been normalized, wpath never ends with a '/'
643
643
644 if wpath.startswith('.hg/') and evt.mask & inotify.IN_ISDIR:
644 if wpath.startswith('.hg/') and evt.mask & inotify.IN_ISDIR:
645 # ignore subdirectories of .hg/ (merge, patches...)
645 # ignore subdirectories of .hg/ (merge, patches...)
646 continue
646 continue
647
647
648 if evt.mask & inotify.IN_UNMOUNT:
648 if evt.mask & inotify.IN_UNMOUNT:
649 self.process_unmount(wpath, evt)
649 self.process_unmount(wpath, evt)
650 elif evt.mask & (inotify.IN_MODIFY | inotify.IN_ATTRIB):
650 elif evt.mask & (inotify.IN_MODIFY | inotify.IN_ATTRIB):
651 self.process_modify(wpath, evt)
651 self.process_modify(wpath, evt)
652 elif evt.mask & (inotify.IN_DELETE | inotify.IN_DELETE_SELF |
652 elif evt.mask & (inotify.IN_DELETE | inotify.IN_DELETE_SELF |
653 inotify.IN_MOVED_FROM):
653 inotify.IN_MOVED_FROM):
654 self.process_delete(wpath, evt)
654 self.process_delete(wpath, evt)
655 elif evt.mask & (inotify.IN_CREATE | inotify.IN_MOVED_TO):
655 elif evt.mask & (inotify.IN_CREATE | inotify.IN_MOVED_TO):
656 self.process_create(wpath, evt)
656 self.process_create(wpath, evt)
657
657
658 self.lastevent.clear()
658 self.lastevent.clear()
659
659
660 def handle_timeout(self):
660 def handle_timeout(self):
661 if not self.registered:
661 if not self.registered:
662 if self.ui.debugflag:
662 if self.ui.debugflag:
663 self.ui.note(_('%s hooking back up with %d bytes readable\n') %
663 self.ui.note(_('%s hooking back up with %d bytes readable\n') %
664 (self.event_time(), self.threshold.readable()))
664 (self.event_time(), self.threshold.readable()))
665 self.read_events(0)
665 self.read_events(0)
666 self.register(timeout=None)
666 self.register(timeout=None)
667
667
668 self.timeout = None
668 self.timeout = None
669
669
670 def shutdown(self):
670 def shutdown(self):
671 self.watcher.close()
671 self.watcher.close()
672
672
673 def debug(self):
673 def debug(self):
674 """
674 """
675 Returns a sorted list of relatives paths currently watched,
675 Returns a sorted list of relatives paths currently watched,
676 for debugging purposes.
676 for debugging purposes.
677 """
677 """
678 return sorted(tuple[0][self.prefixlen:] for tuple in self.watcher)
678 return sorted(tuple[0][self.prefixlen:] for tuple in self.watcher)
679
679
680 class server(pollable):
680 class server(pollable):
681 """
681 """
682 Listens for client queries on unix socket inotify.sock
682 Listens for client queries on unix socket inotify.sock
683 """
683 """
684 def __init__(self, ui, root, repowatcher, timeout):
684 def __init__(self, ui, root, repowatcher, timeout):
685 self.ui = ui
685 self.ui = ui
686 self.repowatcher = repowatcher
686 self.repowatcher = repowatcher
687 self.sock = socket.socket(socket.AF_UNIX)
687 self.sock = socket.socket(socket.AF_UNIX)
688 self.sockpath = join(root, '.hg/inotify.sock')
688 self.sockpath = join(root, '.hg/inotify.sock')
689 self.realsockpath = None
689 self.realsockpath = None
690 try:
690 try:
691 self.sock.bind(self.sockpath)
691 self.sock.bind(self.sockpath)
692 except socket.error, err:
692 except socket.error, err:
693 if err[0] == errno.EADDRINUSE:
693 if err[0] == errno.EADDRINUSE:
694 raise AlreadyStartedException(_('could not start server: %s')
694 raise AlreadyStartedException(_('could not start server: %s')
695 % err[1])
695 % err[1])
696 if err[0] == "AF_UNIX path too long":
696 if err[0] == "AF_UNIX path too long":
697 tempdir = tempfile.mkdtemp(prefix="hg-inotify-")
697 tempdir = tempfile.mkdtemp(prefix="hg-inotify-")
698 self.realsockpath = os.path.join(tempdir, "inotify.sock")
698 self.realsockpath = os.path.join(tempdir, "inotify.sock")
699 try:
699 try:
700 self.sock.bind(self.realsockpath)
700 self.sock.bind(self.realsockpath)
701 os.symlink(self.realsockpath, self.sockpath)
701 os.symlink(self.realsockpath, self.sockpath)
702 except (OSError, socket.error), inst:
702 except (OSError, socket.error), inst:
703 try:
703 try:
704 os.unlink(self.realsockpath)
704 os.unlink(self.realsockpath)
705 except:
705 except:
706 pass
706 pass
707 os.rmdir(tempdir)
707 os.rmdir(tempdir)
708 if inst.errno == errno.EEXIST:
708 if inst.errno == errno.EEXIST:
709 raise AlreadyStartedException(_('could not start server: %s')
709 raise AlreadyStartedException(_('could not start server: %s')
710 % inst.strerror)
710 % inst.strerror)
711 raise
711 raise
712 else:
712 else:
713 raise
713 raise
714 self.sock.listen(5)
714 self.sock.listen(5)
715 self.fileno = self.sock.fileno
715 self.fileno = self.sock.fileno
716 self.register(timeout=timeout)
716 self.register(timeout=timeout)
717
717
718 def handle_timeout(self):
718 def handle_timeout(self):
719 pass
719 pass
720
720
721 def answer_stat_query(self, cs):
721 def answer_stat_query(self, cs):
722 names = cs.read().split('\0')
722 names = cs.read().split('\0')
723
723
724 states = names.pop()
724 states = names.pop()
725
725
726 self.ui.note(_('answering query for %r\n') % states)
726 self.ui.note(_('answering query for %r\n') % states)
727
727
728 if self.repowatcher.timeout:
728 if self.repowatcher.timeout:
729 # We got a query while a rescan is pending. Make sure we
729 # We got a query while a rescan is pending. Make sure we
730 # rescan before responding, or we could give back a wrong
730 # rescan before responding, or we could give back a wrong
731 # answer.
731 # answer.
732 self.repowatcher.handle_timeout()
732 self.repowatcher.handle_timeout()
733
733
734 visited = set()
734 visited = set()
735 if not names:
735 if not names:
736 def genresult(states, tree):
736 def genresult(states, tree):
737 for fn, state in tree.walk(states):
737 for fn, state in tree.walk(states):
738 yield fn
738 yield fn
739 else:
739 else:
740 def genresult(states, tree):
740 def genresult(states, tree):
741 for fn in names:
741 for fn in names:
742 for f in tree.lookup(states, fn, visited):
742 for f in tree.lookup(states, fn, visited):
743 yield f
743 yield f
744
744
745 return ['\0'.join(r) for r in [
745 return ['\0'.join(r) for r in [
746 genresult('l', self.repowatcher.statustrees['l']),
746 genresult('l', self.repowatcher.statustrees['l']),
747 genresult('m', self.repowatcher.statustrees['m']),
747 genresult('m', self.repowatcher.statustrees['m']),
748 genresult('a', self.repowatcher.statustrees['a']),
748 genresult('a', self.repowatcher.statustrees['a']),
749 genresult('r', self.repowatcher.statustrees['r']),
749 genresult('r', self.repowatcher.statustrees['r']),
750 genresult('!', self.repowatcher.statustrees['!']),
750 genresult('!', self.repowatcher.statustrees['!']),
751 '?' in states
751 '?' in states
752 and genresult('?', self.repowatcher.statustrees['?'])
752 and genresult('?', self.repowatcher.statustrees['?'])
753 or [],
753 or [],
754 [],
754 [],
755 'c' in states and genresult('n', self.repowatcher.tree) or [],
755 'c' in states and genresult('n', self.repowatcher.tree) or [],
756 visited
756 visited
757 ]]
757 ]]
758
758
759 def answer_dbug_query(self):
759 def answer_dbug_query(self):
760 return ['\0'.join(self.repowatcher.debug())]
760 return ['\0'.join(self.repowatcher.debug())]
761
761
762 def handle_pollevents(self, events):
762 def handle_pollevents(self, events):
763 for e in events:
763 for e in events:
764 self.handle_pollevent()
764 self.handle_pollevent()
765
765
766 def handle_pollevent(self):
766 def handle_pollevent(self):
767 sock, addr = self.sock.accept()
767 sock, addr = self.sock.accept()
768
768
769 cs = common.recvcs(sock)
769 cs = common.recvcs(sock)
770 version = ord(cs.read(1))
770 version = ord(cs.read(1))
771
771
772 if version != common.version:
772 if version != common.version:
773 self.ui.warn(_('received query from incompatible client '
773 self.ui.warn(_('received query from incompatible client '
774 'version %d\n') % version)
774 'version %d\n') % version)
775 try:
775 try:
776 # try to send back our version to the client
776 # try to send back our version to the client
777 # this way, the client too is informed of the mismatch
777 # this way, the client too is informed of the mismatch
778 sock.sendall(chr(common.version))
778 sock.sendall(chr(common.version))
779 except:
779 except:
780 pass
780 pass
781 return
781 return
782
782
783 type = cs.read(4)
783 type = cs.read(4)
784
784
785 if type == 'STAT':
785 if type == 'STAT':
786 results = self.answer_stat_query(cs)
786 results = self.answer_stat_query(cs)
787 elif type == 'DBUG':
787 elif type == 'DBUG':
788 results = self.answer_dbug_query()
788 results = self.answer_dbug_query()
789 else:
789 else:
790 self.ui.warn(_('unrecognized query type: %s\n') % type)
790 self.ui.warn(_('unrecognized query type: %s\n') % type)
791 return
791 return
792
792
793 try:
793 try:
794 try:
794 try:
795 v = chr(common.version)
795 v = chr(common.version)
796
796
797 sock.sendall(v + type + struct.pack(common.resphdrfmts[type],
797 sock.sendall(v + type + struct.pack(common.resphdrfmts[type],
798 *map(len, results)))
798 *map(len, results)))
799 sock.sendall(''.join(results))
799 sock.sendall(''.join(results))
800 finally:
800 finally:
801 sock.shutdown(socket.SHUT_WR)
801 sock.shutdown(socket.SHUT_WR)
802 except socket.error, err:
802 except socket.error, err:
803 if err[0] != errno.EPIPE:
803 if err[0] != errno.EPIPE:
804 raise
804 raise
805
805
806 def shutdown(self):
806 def shutdown(self):
807 self.sock.close()
807 self.sock.close()
808 try:
808 try:
809 os.unlink(self.sockpath)
809 os.unlink(self.sockpath)
810 if self.realsockpath:
810 if self.realsockpath:
811 os.unlink(self.realsockpath)
811 os.unlink(self.realsockpath)
812 os.rmdir(os.path.dirname(self.realsockpath))
812 os.rmdir(os.path.dirname(self.realsockpath))
813 except OSError, err:
813 except OSError, err:
814 if err.errno != errno.ENOENT:
814 if err.errno != errno.ENOENT:
815 raise
815 raise
816
816
817 class master(object):
817 class master(object):
818 def __init__(self, ui, dirstate, root, timeout=None):
818 def __init__(self, ui, dirstate, root, timeout=None):
819 self.ui = ui
819 self.ui = ui
820 self.repowatcher = repowatcher(ui, dirstate, root)
820 self.repowatcher = repowatcher(ui, dirstate, root)
821 self.server = server(ui, root, self.repowatcher, timeout)
821 self.server = server(ui, root, self.repowatcher, timeout)
822
822
823 def shutdown(self):
823 def shutdown(self):
824 for obj in pollable.instances.itervalues():
824 for obj in pollable.instances.itervalues():
825 obj.shutdown()
825 obj.shutdown()
826
826
827 def run(self):
827 def run(self):
828 self.repowatcher.setup()
828 self.repowatcher.setup()
829 self.ui.note(_('finished setup\n'))
829 self.ui.note(_('finished setup\n'))
830 if os.getenv('TIME_STARTUP'):
830 if os.getenv('TIME_STARTUP'):
831 sys.exit(0)
831 sys.exit(0)
832 pollable.run()
832 pollable.run()
833
833
834 def start(ui, dirstate, root, opts):
834 def start(ui, dirstate, root, opts):
835 timeout = opts.get('timeout')
835 timeout = opts.get('timeout')
836 if timeout:
836 if timeout:
837 timeout = float(timeout) * 1e3
837 timeout = float(timeout) * 1e3
838
838
839 class service(object):
839 class service(object):
840 def init(self):
840 def init(self):
841 try:
841 try:
842 self.master = master(ui, dirstate, root, timeout)
842 self.master = master(ui, dirstate, root, timeout)
843 except AlreadyStartedException, inst:
843 except AlreadyStartedException, inst:
844 raise util.Abort(str(inst))
844 raise util.Abort(str(inst))
845
845
846 def run(self):
846 def run(self):
847 try:
847 try:
848 self.master.run()
848 self.master.run()
849 finally:
849 finally:
850 self.master.shutdown()
850 self.master.shutdown()
851
851
852 runargs = None
853 if 'inserve' not in sys.argv:
852 if 'inserve' not in sys.argv:
854 runargs = [sys.argv[0], 'inserve', '-R', root]
853 runargs = [sys.argv[0], 'inserve', '-R', root]
854 else:
855 runargs = sys.argv[:]
856
857 pidfile = ui.config('inotify', 'pidfile')
858 if opts['daemon'] and pidfile is not None and 'pid-file' not in runargs:
859 runargs.append("--pid-file=%s" % pidfile)
855
860
856 service = service()
861 service = service()
857 logfile = ui.config('inotify', 'log')
862 logfile = ui.config('inotify', 'log')
858 cmdutil.service(opts, initfn=service.init, runfn=service.run,
863 cmdutil.service(opts, initfn=service.init, runfn=service.run,
859 logfile=logfile, runargs=runargs)
864 logfile=logfile, runargs=runargs)
@@ -1,1282 +1,1282
1 # cmdutil.py - help for command processing in mercurial
1 # cmdutil.py - help for command processing in mercurial
2 #
2 #
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
3 # Copyright 2005-2007 Matt Mackall <mpm@selenic.com>
4 #
4 #
5 # This software may be used and distributed according to the terms of the
5 # This software may be used and distributed according to the terms of the
6 # GNU General Public License version 2, incorporated herein by reference.
6 # GNU General Public License version 2, incorporated herein by reference.
7
7
8 from node import hex, nullid, nullrev, short
8 from node import hex, nullid, nullrev, short
9 from i18n import _
9 from i18n import _
10 import os, sys, errno, re, glob
10 import os, sys, errno, re, glob
11 import mdiff, bdiff, util, templater, patch, error, encoding
11 import mdiff, bdiff, util, templater, patch, error, encoding
12 import match as _match
12 import match as _match
13
13
14 revrangesep = ':'
14 revrangesep = ':'
15
15
16 def findpossible(cmd, table, strict=False):
16 def findpossible(cmd, table, strict=False):
17 """
17 """
18 Return cmd -> (aliases, command table entry)
18 Return cmd -> (aliases, command table entry)
19 for each matching command.
19 for each matching command.
20 Return debug commands (or their aliases) only if no normal command matches.
20 Return debug commands (or their aliases) only if no normal command matches.
21 """
21 """
22 choice = {}
22 choice = {}
23 debugchoice = {}
23 debugchoice = {}
24 for e in table.keys():
24 for e in table.keys():
25 aliases = e.lstrip("^").split("|")
25 aliases = e.lstrip("^").split("|")
26 found = None
26 found = None
27 if cmd in aliases:
27 if cmd in aliases:
28 found = cmd
28 found = cmd
29 elif not strict:
29 elif not strict:
30 for a in aliases:
30 for a in aliases:
31 if a.startswith(cmd):
31 if a.startswith(cmd):
32 found = a
32 found = a
33 break
33 break
34 if found is not None:
34 if found is not None:
35 if aliases[0].startswith("debug") or found.startswith("debug"):
35 if aliases[0].startswith("debug") or found.startswith("debug"):
36 debugchoice[found] = (aliases, table[e])
36 debugchoice[found] = (aliases, table[e])
37 else:
37 else:
38 choice[found] = (aliases, table[e])
38 choice[found] = (aliases, table[e])
39
39
40 if not choice and debugchoice:
40 if not choice and debugchoice:
41 choice = debugchoice
41 choice = debugchoice
42
42
43 return choice
43 return choice
44
44
45 def findcmd(cmd, table, strict=True):
45 def findcmd(cmd, table, strict=True):
46 """Return (aliases, command table entry) for command string."""
46 """Return (aliases, command table entry) for command string."""
47 choice = findpossible(cmd, table, strict)
47 choice = findpossible(cmd, table, strict)
48
48
49 if cmd in choice:
49 if cmd in choice:
50 return choice[cmd]
50 return choice[cmd]
51
51
52 if len(choice) > 1:
52 if len(choice) > 1:
53 clist = choice.keys()
53 clist = choice.keys()
54 clist.sort()
54 clist.sort()
55 raise error.AmbiguousCommand(cmd, clist)
55 raise error.AmbiguousCommand(cmd, clist)
56
56
57 if choice:
57 if choice:
58 return choice.values()[0]
58 return choice.values()[0]
59
59
60 raise error.UnknownCommand(cmd)
60 raise error.UnknownCommand(cmd)
61
61
62 def bail_if_changed(repo):
62 def bail_if_changed(repo):
63 if repo.dirstate.parents()[1] != nullid:
63 if repo.dirstate.parents()[1] != nullid:
64 raise util.Abort(_('outstanding uncommitted merge'))
64 raise util.Abort(_('outstanding uncommitted merge'))
65 modified, added, removed, deleted = repo.status()[:4]
65 modified, added, removed, deleted = repo.status()[:4]
66 if modified or added or removed or deleted:
66 if modified or added or removed or deleted:
67 raise util.Abort(_("outstanding uncommitted changes"))
67 raise util.Abort(_("outstanding uncommitted changes"))
68
68
69 def logmessage(opts):
69 def logmessage(opts):
70 """ get the log message according to -m and -l option """
70 """ get the log message according to -m and -l option """
71 message = opts.get('message')
71 message = opts.get('message')
72 logfile = opts.get('logfile')
72 logfile = opts.get('logfile')
73
73
74 if message and logfile:
74 if message and logfile:
75 raise util.Abort(_('options --message and --logfile are mutually '
75 raise util.Abort(_('options --message and --logfile are mutually '
76 'exclusive'))
76 'exclusive'))
77 if not message and logfile:
77 if not message and logfile:
78 try:
78 try:
79 if logfile == '-':
79 if logfile == '-':
80 message = sys.stdin.read()
80 message = sys.stdin.read()
81 else:
81 else:
82 message = open(logfile).read()
82 message = open(logfile).read()
83 except IOError, inst:
83 except IOError, inst:
84 raise util.Abort(_("can't read commit message '%s': %s") %
84 raise util.Abort(_("can't read commit message '%s': %s") %
85 (logfile, inst.strerror))
85 (logfile, inst.strerror))
86 return message
86 return message
87
87
88 def loglimit(opts):
88 def loglimit(opts):
89 """get the log limit according to option -l/--limit"""
89 """get the log limit according to option -l/--limit"""
90 limit = opts.get('limit')
90 limit = opts.get('limit')
91 if limit:
91 if limit:
92 try:
92 try:
93 limit = int(limit)
93 limit = int(limit)
94 except ValueError:
94 except ValueError:
95 raise util.Abort(_('limit must be a positive integer'))
95 raise util.Abort(_('limit must be a positive integer'))
96 if limit <= 0: raise util.Abort(_('limit must be positive'))
96 if limit <= 0: raise util.Abort(_('limit must be positive'))
97 else:
97 else:
98 limit = sys.maxint
98 limit = sys.maxint
99 return limit
99 return limit
100
100
101 def remoteui(src, opts):
101 def remoteui(src, opts):
102 'build a remote ui from ui or repo and opts'
102 'build a remote ui from ui or repo and opts'
103 if hasattr(src, 'baseui'): # looks like a repository
103 if hasattr(src, 'baseui'): # looks like a repository
104 dst = src.baseui.copy() # drop repo-specific config
104 dst = src.baseui.copy() # drop repo-specific config
105 src = src.ui # copy target options from repo
105 src = src.ui # copy target options from repo
106 else: # assume it's a global ui object
106 else: # assume it's a global ui object
107 dst = src.copy() # keep all global options
107 dst = src.copy() # keep all global options
108
108
109 # copy ssh-specific options
109 # copy ssh-specific options
110 for o in 'ssh', 'remotecmd':
110 for o in 'ssh', 'remotecmd':
111 v = opts.get(o) or src.config('ui', o)
111 v = opts.get(o) or src.config('ui', o)
112 if v:
112 if v:
113 dst.setconfig("ui", o, v)
113 dst.setconfig("ui", o, v)
114 # copy bundle-specific options
114 # copy bundle-specific options
115 r = src.config('bundle', 'mainreporoot')
115 r = src.config('bundle', 'mainreporoot')
116 if r:
116 if r:
117 dst.setconfig('bundle', 'mainreporoot', r)
117 dst.setconfig('bundle', 'mainreporoot', r)
118
118
119 return dst
119 return dst
120
120
121 def revpair(repo, revs):
121 def revpair(repo, revs):
122 '''return pair of nodes, given list of revisions. second item can
122 '''return pair of nodes, given list of revisions. second item can
123 be None, meaning use working dir.'''
123 be None, meaning use working dir.'''
124
124
125 def revfix(repo, val, defval):
125 def revfix(repo, val, defval):
126 if not val and val != 0 and defval is not None:
126 if not val and val != 0 and defval is not None:
127 val = defval
127 val = defval
128 return repo.lookup(val)
128 return repo.lookup(val)
129
129
130 if not revs:
130 if not revs:
131 return repo.dirstate.parents()[0], None
131 return repo.dirstate.parents()[0], None
132 end = None
132 end = None
133 if len(revs) == 1:
133 if len(revs) == 1:
134 if revrangesep in revs[0]:
134 if revrangesep in revs[0]:
135 start, end = revs[0].split(revrangesep, 1)
135 start, end = revs[0].split(revrangesep, 1)
136 start = revfix(repo, start, 0)
136 start = revfix(repo, start, 0)
137 end = revfix(repo, end, len(repo) - 1)
137 end = revfix(repo, end, len(repo) - 1)
138 else:
138 else:
139 start = revfix(repo, revs[0], None)
139 start = revfix(repo, revs[0], None)
140 elif len(revs) == 2:
140 elif len(revs) == 2:
141 if revrangesep in revs[0] or revrangesep in revs[1]:
141 if revrangesep in revs[0] or revrangesep in revs[1]:
142 raise util.Abort(_('too many revisions specified'))
142 raise util.Abort(_('too many revisions specified'))
143 start = revfix(repo, revs[0], None)
143 start = revfix(repo, revs[0], None)
144 end = revfix(repo, revs[1], None)
144 end = revfix(repo, revs[1], None)
145 else:
145 else:
146 raise util.Abort(_('too many revisions specified'))
146 raise util.Abort(_('too many revisions specified'))
147 return start, end
147 return start, end
148
148
149 def revrange(repo, revs):
149 def revrange(repo, revs):
150 """Yield revision as strings from a list of revision specifications."""
150 """Yield revision as strings from a list of revision specifications."""
151
151
152 def revfix(repo, val, defval):
152 def revfix(repo, val, defval):
153 if not val and val != 0 and defval is not None:
153 if not val and val != 0 and defval is not None:
154 return defval
154 return defval
155 return repo.changelog.rev(repo.lookup(val))
155 return repo.changelog.rev(repo.lookup(val))
156
156
157 seen, l = set(), []
157 seen, l = set(), []
158 for spec in revs:
158 for spec in revs:
159 if revrangesep in spec:
159 if revrangesep in spec:
160 start, end = spec.split(revrangesep, 1)
160 start, end = spec.split(revrangesep, 1)
161 start = revfix(repo, start, 0)
161 start = revfix(repo, start, 0)
162 end = revfix(repo, end, len(repo) - 1)
162 end = revfix(repo, end, len(repo) - 1)
163 step = start > end and -1 or 1
163 step = start > end and -1 or 1
164 for rev in xrange(start, end+step, step):
164 for rev in xrange(start, end+step, step):
165 if rev in seen:
165 if rev in seen:
166 continue
166 continue
167 seen.add(rev)
167 seen.add(rev)
168 l.append(rev)
168 l.append(rev)
169 else:
169 else:
170 rev = revfix(repo, spec, None)
170 rev = revfix(repo, spec, None)
171 if rev in seen:
171 if rev in seen:
172 continue
172 continue
173 seen.add(rev)
173 seen.add(rev)
174 l.append(rev)
174 l.append(rev)
175
175
176 return l
176 return l
177
177
178 def make_filename(repo, pat, node,
178 def make_filename(repo, pat, node,
179 total=None, seqno=None, revwidth=None, pathname=None):
179 total=None, seqno=None, revwidth=None, pathname=None):
180 node_expander = {
180 node_expander = {
181 'H': lambda: hex(node),
181 'H': lambda: hex(node),
182 'R': lambda: str(repo.changelog.rev(node)),
182 'R': lambda: str(repo.changelog.rev(node)),
183 'h': lambda: short(node),
183 'h': lambda: short(node),
184 }
184 }
185 expander = {
185 expander = {
186 '%': lambda: '%',
186 '%': lambda: '%',
187 'b': lambda: os.path.basename(repo.root),
187 'b': lambda: os.path.basename(repo.root),
188 }
188 }
189
189
190 try:
190 try:
191 if node:
191 if node:
192 expander.update(node_expander)
192 expander.update(node_expander)
193 if node:
193 if node:
194 expander['r'] = (lambda:
194 expander['r'] = (lambda:
195 str(repo.changelog.rev(node)).zfill(revwidth or 0))
195 str(repo.changelog.rev(node)).zfill(revwidth or 0))
196 if total is not None:
196 if total is not None:
197 expander['N'] = lambda: str(total)
197 expander['N'] = lambda: str(total)
198 if seqno is not None:
198 if seqno is not None:
199 expander['n'] = lambda: str(seqno)
199 expander['n'] = lambda: str(seqno)
200 if total is not None and seqno is not None:
200 if total is not None and seqno is not None:
201 expander['n'] = lambda: str(seqno).zfill(len(str(total)))
201 expander['n'] = lambda: str(seqno).zfill(len(str(total)))
202 if pathname is not None:
202 if pathname is not None:
203 expander['s'] = lambda: os.path.basename(pathname)
203 expander['s'] = lambda: os.path.basename(pathname)
204 expander['d'] = lambda: os.path.dirname(pathname) or '.'
204 expander['d'] = lambda: os.path.dirname(pathname) or '.'
205 expander['p'] = lambda: pathname
205 expander['p'] = lambda: pathname
206
206
207 newname = []
207 newname = []
208 patlen = len(pat)
208 patlen = len(pat)
209 i = 0
209 i = 0
210 while i < patlen:
210 while i < patlen:
211 c = pat[i]
211 c = pat[i]
212 if c == '%':
212 if c == '%':
213 i += 1
213 i += 1
214 c = pat[i]
214 c = pat[i]
215 c = expander[c]()
215 c = expander[c]()
216 newname.append(c)
216 newname.append(c)
217 i += 1
217 i += 1
218 return ''.join(newname)
218 return ''.join(newname)
219 except KeyError, inst:
219 except KeyError, inst:
220 raise util.Abort(_("invalid format spec '%%%s' in output filename") %
220 raise util.Abort(_("invalid format spec '%%%s' in output filename") %
221 inst.args[0])
221 inst.args[0])
222
222
223 def make_file(repo, pat, node=None,
223 def make_file(repo, pat, node=None,
224 total=None, seqno=None, revwidth=None, mode='wb', pathname=None):
224 total=None, seqno=None, revwidth=None, mode='wb', pathname=None):
225
225
226 writable = 'w' in mode or 'a' in mode
226 writable = 'w' in mode or 'a' in mode
227
227
228 if not pat or pat == '-':
228 if not pat or pat == '-':
229 return writable and sys.stdout or sys.stdin
229 return writable and sys.stdout or sys.stdin
230 if hasattr(pat, 'write') and writable:
230 if hasattr(pat, 'write') and writable:
231 return pat
231 return pat
232 if hasattr(pat, 'read') and 'r' in mode:
232 if hasattr(pat, 'read') and 'r' in mode:
233 return pat
233 return pat
234 return open(make_filename(repo, pat, node, total, seqno, revwidth,
234 return open(make_filename(repo, pat, node, total, seqno, revwidth,
235 pathname),
235 pathname),
236 mode)
236 mode)
237
237
238 def expandpats(pats):
238 def expandpats(pats):
239 if not util.expandglobs:
239 if not util.expandglobs:
240 return list(pats)
240 return list(pats)
241 ret = []
241 ret = []
242 for p in pats:
242 for p in pats:
243 kind, name = _match._patsplit(p, None)
243 kind, name = _match._patsplit(p, None)
244 if kind is None:
244 if kind is None:
245 try:
245 try:
246 globbed = glob.glob(name)
246 globbed = glob.glob(name)
247 except re.error:
247 except re.error:
248 globbed = [name]
248 globbed = [name]
249 if globbed:
249 if globbed:
250 ret.extend(globbed)
250 ret.extend(globbed)
251 continue
251 continue
252 ret.append(p)
252 ret.append(p)
253 return ret
253 return ret
254
254
255 def match(repo, pats=[], opts={}, globbed=False, default='relpath'):
255 def match(repo, pats=[], opts={}, globbed=False, default='relpath'):
256 if not globbed and default == 'relpath':
256 if not globbed and default == 'relpath':
257 pats = expandpats(pats or [])
257 pats = expandpats(pats or [])
258 m = _match.match(repo.root, repo.getcwd(), pats,
258 m = _match.match(repo.root, repo.getcwd(), pats,
259 opts.get('include'), opts.get('exclude'), default)
259 opts.get('include'), opts.get('exclude'), default)
260 def badfn(f, msg):
260 def badfn(f, msg):
261 repo.ui.warn("%s: %s\n" % (m.rel(f), msg))
261 repo.ui.warn("%s: %s\n" % (m.rel(f), msg))
262 m.bad = badfn
262 m.bad = badfn
263 return m
263 return m
264
264
265 def matchall(repo):
265 def matchall(repo):
266 return _match.always(repo.root, repo.getcwd())
266 return _match.always(repo.root, repo.getcwd())
267
267
268 def matchfiles(repo, files):
268 def matchfiles(repo, files):
269 return _match.exact(repo.root, repo.getcwd(), files)
269 return _match.exact(repo.root, repo.getcwd(), files)
270
270
271 def findrenames(repo, added, removed, threshold):
271 def findrenames(repo, added, removed, threshold):
272 '''find renamed files -- yields (before, after, score) tuples'''
272 '''find renamed files -- yields (before, after, score) tuples'''
273 ctx = repo['.']
273 ctx = repo['.']
274 for a in added:
274 for a in added:
275 aa = repo.wread(a)
275 aa = repo.wread(a)
276 bestname, bestscore = None, threshold
276 bestname, bestscore = None, threshold
277 for r in removed:
277 for r in removed:
278 if r not in ctx:
278 if r not in ctx:
279 continue
279 continue
280 rr = ctx.filectx(r).data()
280 rr = ctx.filectx(r).data()
281
281
282 # bdiff.blocks() returns blocks of matching lines
282 # bdiff.blocks() returns blocks of matching lines
283 # count the number of bytes in each
283 # count the number of bytes in each
284 equal = 0
284 equal = 0
285 alines = mdiff.splitnewlines(aa)
285 alines = mdiff.splitnewlines(aa)
286 matches = bdiff.blocks(aa, rr)
286 matches = bdiff.blocks(aa, rr)
287 for x1,x2,y1,y2 in matches:
287 for x1,x2,y1,y2 in matches:
288 for line in alines[x1:x2]:
288 for line in alines[x1:x2]:
289 equal += len(line)
289 equal += len(line)
290
290
291 lengths = len(aa) + len(rr)
291 lengths = len(aa) + len(rr)
292 if lengths:
292 if lengths:
293 myscore = equal*2.0 / lengths
293 myscore = equal*2.0 / lengths
294 if myscore >= bestscore:
294 if myscore >= bestscore:
295 bestname, bestscore = r, myscore
295 bestname, bestscore = r, myscore
296 if bestname:
296 if bestname:
297 yield bestname, a, bestscore
297 yield bestname, a, bestscore
298
298
299 def addremove(repo, pats=[], opts={}, dry_run=None, similarity=None):
299 def addremove(repo, pats=[], opts={}, dry_run=None, similarity=None):
300 if dry_run is None:
300 if dry_run is None:
301 dry_run = opts.get('dry_run')
301 dry_run = opts.get('dry_run')
302 if similarity is None:
302 if similarity is None:
303 similarity = float(opts.get('similarity') or 0)
303 similarity = float(opts.get('similarity') or 0)
304 # we'd use status here, except handling of symlinks and ignore is tricky
304 # we'd use status here, except handling of symlinks and ignore is tricky
305 added, unknown, deleted, removed = [], [], [], []
305 added, unknown, deleted, removed = [], [], [], []
306 audit_path = util.path_auditor(repo.root)
306 audit_path = util.path_auditor(repo.root)
307 m = match(repo, pats, opts)
307 m = match(repo, pats, opts)
308 for abs in repo.walk(m):
308 for abs in repo.walk(m):
309 target = repo.wjoin(abs)
309 target = repo.wjoin(abs)
310 good = True
310 good = True
311 try:
311 try:
312 audit_path(abs)
312 audit_path(abs)
313 except:
313 except:
314 good = False
314 good = False
315 rel = m.rel(abs)
315 rel = m.rel(abs)
316 exact = m.exact(abs)
316 exact = m.exact(abs)
317 if good and abs not in repo.dirstate:
317 if good and abs not in repo.dirstate:
318 unknown.append(abs)
318 unknown.append(abs)
319 if repo.ui.verbose or not exact:
319 if repo.ui.verbose or not exact:
320 repo.ui.status(_('adding %s\n') % ((pats and rel) or abs))
320 repo.ui.status(_('adding %s\n') % ((pats and rel) or abs))
321 elif repo.dirstate[abs] != 'r' and (not good or not util.lexists(target)
321 elif repo.dirstate[abs] != 'r' and (not good or not util.lexists(target)
322 or (os.path.isdir(target) and not os.path.islink(target))):
322 or (os.path.isdir(target) and not os.path.islink(target))):
323 deleted.append(abs)
323 deleted.append(abs)
324 if repo.ui.verbose or not exact:
324 if repo.ui.verbose or not exact:
325 repo.ui.status(_('removing %s\n') % ((pats and rel) or abs))
325 repo.ui.status(_('removing %s\n') % ((pats and rel) or abs))
326 # for finding renames
326 # for finding renames
327 elif repo.dirstate[abs] == 'r':
327 elif repo.dirstate[abs] == 'r':
328 removed.append(abs)
328 removed.append(abs)
329 elif repo.dirstate[abs] == 'a':
329 elif repo.dirstate[abs] == 'a':
330 added.append(abs)
330 added.append(abs)
331 if not dry_run:
331 if not dry_run:
332 repo.remove(deleted)
332 repo.remove(deleted)
333 repo.add(unknown)
333 repo.add(unknown)
334 if similarity > 0:
334 if similarity > 0:
335 for old, new, score in findrenames(repo, added + unknown,
335 for old, new, score in findrenames(repo, added + unknown,
336 removed + deleted, similarity):
336 removed + deleted, similarity):
337 if repo.ui.verbose or not m.exact(old) or not m.exact(new):
337 if repo.ui.verbose or not m.exact(old) or not m.exact(new):
338 repo.ui.status(_('recording removal of %s as rename to %s '
338 repo.ui.status(_('recording removal of %s as rename to %s '
339 '(%d%% similar)\n') %
339 '(%d%% similar)\n') %
340 (m.rel(old), m.rel(new), score * 100))
340 (m.rel(old), m.rel(new), score * 100))
341 if not dry_run:
341 if not dry_run:
342 repo.copy(old, new)
342 repo.copy(old, new)
343
343
344 def copy(ui, repo, pats, opts, rename=False):
344 def copy(ui, repo, pats, opts, rename=False):
345 # called with the repo lock held
345 # called with the repo lock held
346 #
346 #
347 # hgsep => pathname that uses "/" to separate directories
347 # hgsep => pathname that uses "/" to separate directories
348 # ossep => pathname that uses os.sep to separate directories
348 # ossep => pathname that uses os.sep to separate directories
349 cwd = repo.getcwd()
349 cwd = repo.getcwd()
350 targets = {}
350 targets = {}
351 after = opts.get("after")
351 after = opts.get("after")
352 dryrun = opts.get("dry_run")
352 dryrun = opts.get("dry_run")
353
353
354 def walkpat(pat):
354 def walkpat(pat):
355 srcs = []
355 srcs = []
356 m = match(repo, [pat], opts, globbed=True)
356 m = match(repo, [pat], opts, globbed=True)
357 for abs in repo.walk(m):
357 for abs in repo.walk(m):
358 state = repo.dirstate[abs]
358 state = repo.dirstate[abs]
359 rel = m.rel(abs)
359 rel = m.rel(abs)
360 exact = m.exact(abs)
360 exact = m.exact(abs)
361 if state in '?r':
361 if state in '?r':
362 if exact and state == '?':
362 if exact and state == '?':
363 ui.warn(_('%s: not copying - file is not managed\n') % rel)
363 ui.warn(_('%s: not copying - file is not managed\n') % rel)
364 if exact and state == 'r':
364 if exact and state == 'r':
365 ui.warn(_('%s: not copying - file has been marked for'
365 ui.warn(_('%s: not copying - file has been marked for'
366 ' remove\n') % rel)
366 ' remove\n') % rel)
367 continue
367 continue
368 # abs: hgsep
368 # abs: hgsep
369 # rel: ossep
369 # rel: ossep
370 srcs.append((abs, rel, exact))
370 srcs.append((abs, rel, exact))
371 return srcs
371 return srcs
372
372
373 # abssrc: hgsep
373 # abssrc: hgsep
374 # relsrc: ossep
374 # relsrc: ossep
375 # otarget: ossep
375 # otarget: ossep
376 def copyfile(abssrc, relsrc, otarget, exact):
376 def copyfile(abssrc, relsrc, otarget, exact):
377 abstarget = util.canonpath(repo.root, cwd, otarget)
377 abstarget = util.canonpath(repo.root, cwd, otarget)
378 reltarget = repo.pathto(abstarget, cwd)
378 reltarget = repo.pathto(abstarget, cwd)
379 target = repo.wjoin(abstarget)
379 target = repo.wjoin(abstarget)
380 src = repo.wjoin(abssrc)
380 src = repo.wjoin(abssrc)
381 state = repo.dirstate[abstarget]
381 state = repo.dirstate[abstarget]
382
382
383 # check for collisions
383 # check for collisions
384 prevsrc = targets.get(abstarget)
384 prevsrc = targets.get(abstarget)
385 if prevsrc is not None:
385 if prevsrc is not None:
386 ui.warn(_('%s: not overwriting - %s collides with %s\n') %
386 ui.warn(_('%s: not overwriting - %s collides with %s\n') %
387 (reltarget, repo.pathto(abssrc, cwd),
387 (reltarget, repo.pathto(abssrc, cwd),
388 repo.pathto(prevsrc, cwd)))
388 repo.pathto(prevsrc, cwd)))
389 return
389 return
390
390
391 # check for overwrites
391 # check for overwrites
392 exists = os.path.exists(target)
392 exists = os.path.exists(target)
393 if not after and exists or after and state in 'mn':
393 if not after and exists or after and state in 'mn':
394 if not opts['force']:
394 if not opts['force']:
395 ui.warn(_('%s: not overwriting - file exists\n') %
395 ui.warn(_('%s: not overwriting - file exists\n') %
396 reltarget)
396 reltarget)
397 return
397 return
398
398
399 if after:
399 if after:
400 if not exists:
400 if not exists:
401 return
401 return
402 elif not dryrun:
402 elif not dryrun:
403 try:
403 try:
404 if exists:
404 if exists:
405 os.unlink(target)
405 os.unlink(target)
406 targetdir = os.path.dirname(target) or '.'
406 targetdir = os.path.dirname(target) or '.'
407 if not os.path.isdir(targetdir):
407 if not os.path.isdir(targetdir):
408 os.makedirs(targetdir)
408 os.makedirs(targetdir)
409 util.copyfile(src, target)
409 util.copyfile(src, target)
410 except IOError, inst:
410 except IOError, inst:
411 if inst.errno == errno.ENOENT:
411 if inst.errno == errno.ENOENT:
412 ui.warn(_('%s: deleted in working copy\n') % relsrc)
412 ui.warn(_('%s: deleted in working copy\n') % relsrc)
413 else:
413 else:
414 ui.warn(_('%s: cannot copy - %s\n') %
414 ui.warn(_('%s: cannot copy - %s\n') %
415 (relsrc, inst.strerror))
415 (relsrc, inst.strerror))
416 return True # report a failure
416 return True # report a failure
417
417
418 if ui.verbose or not exact:
418 if ui.verbose or not exact:
419 if rename:
419 if rename:
420 ui.status(_('moving %s to %s\n') % (relsrc, reltarget))
420 ui.status(_('moving %s to %s\n') % (relsrc, reltarget))
421 else:
421 else:
422 ui.status(_('copying %s to %s\n') % (relsrc, reltarget))
422 ui.status(_('copying %s to %s\n') % (relsrc, reltarget))
423
423
424 targets[abstarget] = abssrc
424 targets[abstarget] = abssrc
425
425
426 # fix up dirstate
426 # fix up dirstate
427 origsrc = repo.dirstate.copied(abssrc) or abssrc
427 origsrc = repo.dirstate.copied(abssrc) or abssrc
428 if abstarget == origsrc: # copying back a copy?
428 if abstarget == origsrc: # copying back a copy?
429 if state not in 'mn' and not dryrun:
429 if state not in 'mn' and not dryrun:
430 repo.dirstate.normallookup(abstarget)
430 repo.dirstate.normallookup(abstarget)
431 else:
431 else:
432 if repo.dirstate[origsrc] == 'a' and origsrc == abssrc:
432 if repo.dirstate[origsrc] == 'a' and origsrc == abssrc:
433 if not ui.quiet:
433 if not ui.quiet:
434 ui.warn(_("%s has not been committed yet, so no copy "
434 ui.warn(_("%s has not been committed yet, so no copy "
435 "data will be stored for %s.\n")
435 "data will be stored for %s.\n")
436 % (repo.pathto(origsrc, cwd), reltarget))
436 % (repo.pathto(origsrc, cwd), reltarget))
437 if repo.dirstate[abstarget] in '?r' and not dryrun:
437 if repo.dirstate[abstarget] in '?r' and not dryrun:
438 repo.add([abstarget])
438 repo.add([abstarget])
439 elif not dryrun:
439 elif not dryrun:
440 repo.copy(origsrc, abstarget)
440 repo.copy(origsrc, abstarget)
441
441
442 if rename and not dryrun:
442 if rename and not dryrun:
443 repo.remove([abssrc], not after)
443 repo.remove([abssrc], not after)
444
444
445 # pat: ossep
445 # pat: ossep
446 # dest ossep
446 # dest ossep
447 # srcs: list of (hgsep, hgsep, ossep, bool)
447 # srcs: list of (hgsep, hgsep, ossep, bool)
448 # return: function that takes hgsep and returns ossep
448 # return: function that takes hgsep and returns ossep
449 def targetpathfn(pat, dest, srcs):
449 def targetpathfn(pat, dest, srcs):
450 if os.path.isdir(pat):
450 if os.path.isdir(pat):
451 abspfx = util.canonpath(repo.root, cwd, pat)
451 abspfx = util.canonpath(repo.root, cwd, pat)
452 abspfx = util.localpath(abspfx)
452 abspfx = util.localpath(abspfx)
453 if destdirexists:
453 if destdirexists:
454 striplen = len(os.path.split(abspfx)[0])
454 striplen = len(os.path.split(abspfx)[0])
455 else:
455 else:
456 striplen = len(abspfx)
456 striplen = len(abspfx)
457 if striplen:
457 if striplen:
458 striplen += len(os.sep)
458 striplen += len(os.sep)
459 res = lambda p: os.path.join(dest, util.localpath(p)[striplen:])
459 res = lambda p: os.path.join(dest, util.localpath(p)[striplen:])
460 elif destdirexists:
460 elif destdirexists:
461 res = lambda p: os.path.join(dest,
461 res = lambda p: os.path.join(dest,
462 os.path.basename(util.localpath(p)))
462 os.path.basename(util.localpath(p)))
463 else:
463 else:
464 res = lambda p: dest
464 res = lambda p: dest
465 return res
465 return res
466
466
467 # pat: ossep
467 # pat: ossep
468 # dest ossep
468 # dest ossep
469 # srcs: list of (hgsep, hgsep, ossep, bool)
469 # srcs: list of (hgsep, hgsep, ossep, bool)
470 # return: function that takes hgsep and returns ossep
470 # return: function that takes hgsep and returns ossep
471 def targetpathafterfn(pat, dest, srcs):
471 def targetpathafterfn(pat, dest, srcs):
472 if _match.patkind(pat):
472 if _match.patkind(pat):
473 # a mercurial pattern
473 # a mercurial pattern
474 res = lambda p: os.path.join(dest,
474 res = lambda p: os.path.join(dest,
475 os.path.basename(util.localpath(p)))
475 os.path.basename(util.localpath(p)))
476 else:
476 else:
477 abspfx = util.canonpath(repo.root, cwd, pat)
477 abspfx = util.canonpath(repo.root, cwd, pat)
478 if len(abspfx) < len(srcs[0][0]):
478 if len(abspfx) < len(srcs[0][0]):
479 # A directory. Either the target path contains the last
479 # A directory. Either the target path contains the last
480 # component of the source path or it does not.
480 # component of the source path or it does not.
481 def evalpath(striplen):
481 def evalpath(striplen):
482 score = 0
482 score = 0
483 for s in srcs:
483 for s in srcs:
484 t = os.path.join(dest, util.localpath(s[0])[striplen:])
484 t = os.path.join(dest, util.localpath(s[0])[striplen:])
485 if os.path.exists(t):
485 if os.path.exists(t):
486 score += 1
486 score += 1
487 return score
487 return score
488
488
489 abspfx = util.localpath(abspfx)
489 abspfx = util.localpath(abspfx)
490 striplen = len(abspfx)
490 striplen = len(abspfx)
491 if striplen:
491 if striplen:
492 striplen += len(os.sep)
492 striplen += len(os.sep)
493 if os.path.isdir(os.path.join(dest, os.path.split(abspfx)[1])):
493 if os.path.isdir(os.path.join(dest, os.path.split(abspfx)[1])):
494 score = evalpath(striplen)
494 score = evalpath(striplen)
495 striplen1 = len(os.path.split(abspfx)[0])
495 striplen1 = len(os.path.split(abspfx)[0])
496 if striplen1:
496 if striplen1:
497 striplen1 += len(os.sep)
497 striplen1 += len(os.sep)
498 if evalpath(striplen1) > score:
498 if evalpath(striplen1) > score:
499 striplen = striplen1
499 striplen = striplen1
500 res = lambda p: os.path.join(dest,
500 res = lambda p: os.path.join(dest,
501 util.localpath(p)[striplen:])
501 util.localpath(p)[striplen:])
502 else:
502 else:
503 # a file
503 # a file
504 if destdirexists:
504 if destdirexists:
505 res = lambda p: os.path.join(dest,
505 res = lambda p: os.path.join(dest,
506 os.path.basename(util.localpath(p)))
506 os.path.basename(util.localpath(p)))
507 else:
507 else:
508 res = lambda p: dest
508 res = lambda p: dest
509 return res
509 return res
510
510
511
511
512 pats = expandpats(pats)
512 pats = expandpats(pats)
513 if not pats:
513 if not pats:
514 raise util.Abort(_('no source or destination specified'))
514 raise util.Abort(_('no source or destination specified'))
515 if len(pats) == 1:
515 if len(pats) == 1:
516 raise util.Abort(_('no destination specified'))
516 raise util.Abort(_('no destination specified'))
517 dest = pats.pop()
517 dest = pats.pop()
518 destdirexists = os.path.isdir(dest) and not os.path.islink(dest)
518 destdirexists = os.path.isdir(dest) and not os.path.islink(dest)
519 if not destdirexists:
519 if not destdirexists:
520 if len(pats) > 1 or _match.patkind(pats[0]):
520 if len(pats) > 1 or _match.patkind(pats[0]):
521 raise util.Abort(_('with multiple sources, destination must be an '
521 raise util.Abort(_('with multiple sources, destination must be an '
522 'existing directory'))
522 'existing directory'))
523 if util.endswithsep(dest):
523 if util.endswithsep(dest):
524 raise util.Abort(_('destination %s is not a directory') % dest)
524 raise util.Abort(_('destination %s is not a directory') % dest)
525
525
526 tfn = targetpathfn
526 tfn = targetpathfn
527 if after:
527 if after:
528 tfn = targetpathafterfn
528 tfn = targetpathafterfn
529 copylist = []
529 copylist = []
530 for pat in pats:
530 for pat in pats:
531 srcs = walkpat(pat)
531 srcs = walkpat(pat)
532 if not srcs:
532 if not srcs:
533 continue
533 continue
534 copylist.append((tfn(pat, dest, srcs), srcs))
534 copylist.append((tfn(pat, dest, srcs), srcs))
535 if not copylist:
535 if not copylist:
536 raise util.Abort(_('no files to copy'))
536 raise util.Abort(_('no files to copy'))
537
537
538 errors = 0
538 errors = 0
539 for targetpath, srcs in copylist:
539 for targetpath, srcs in copylist:
540 for abssrc, relsrc, exact in srcs:
540 for abssrc, relsrc, exact in srcs:
541 if copyfile(abssrc, relsrc, targetpath(abssrc), exact):
541 if copyfile(abssrc, relsrc, targetpath(abssrc), exact):
542 errors += 1
542 errors += 1
543
543
544 if errors:
544 if errors:
545 ui.warn(_('(consider using --after)\n'))
545 ui.warn(_('(consider using --after)\n'))
546
546
547 return errors
547 return errors
548
548
549 def service(opts, parentfn=None, initfn=None, runfn=None, logfile=None,
549 def service(opts, parentfn=None, initfn=None, runfn=None, logfile=None,
550 runargs=None):
550 runargs=None):
551 '''Run a command as a service.'''
551 '''Run a command as a service.'''
552
552
553 if opts['daemon'] and not opts['daemon_pipefds']:
553 if opts['daemon'] and not opts['daemon_pipefds']:
554 rfd, wfd = os.pipe()
554 rfd, wfd = os.pipe()
555 if not runargs:
555 if not runargs:
556 runargs = sys.argv[:]
556 runargs = sys.argv[:]
557 runargs.append('--daemon-pipefds=%d,%d' % (rfd, wfd))
557 runargs.append('--daemon-pipefds=%d,%d' % (rfd, wfd))
558 # Don't pass --cwd to the child process, because we've already
558 # Don't pass --cwd to the child process, because we've already
559 # changed directory.
559 # changed directory.
560 for i in xrange(1,len(runargs)):
560 for i in xrange(1,len(runargs)):
561 if runargs[i].startswith('--cwd='):
561 if runargs[i].startswith('--cwd='):
562 del runargs[i]
562 del runargs[i]
563 break
563 break
564 elif runargs[i].startswith('--cwd'):
564 elif runargs[i].startswith('--cwd'):
565 del runargs[i:i+2]
565 del runargs[i:i+2]
566 break
566 break
567 pid = os.spawnvp(os.P_NOWAIT | getattr(os, 'P_DETACH', 0),
567 pid = os.spawnvp(os.P_NOWAIT | getattr(os, 'P_DETACH', 0),
568 runargs[0], runargs)
568 runargs[0], runargs)
569 os.close(wfd)
569 os.close(wfd)
570 os.read(rfd, 1)
570 os.read(rfd, 1)
571 if parentfn:
571 if parentfn:
572 return parentfn(pid)
572 return parentfn(pid)
573 else:
573 else:
574 os._exit(0)
574 return
575
575
576 if initfn:
576 if initfn:
577 initfn()
577 initfn()
578
578
579 if opts['pid_file']:
579 if opts['pid_file']:
580 fp = open(opts['pid_file'], 'w')
580 fp = open(opts['pid_file'], 'w')
581 fp.write(str(os.getpid()) + '\n')
581 fp.write(str(os.getpid()) + '\n')
582 fp.close()
582 fp.close()
583
583
584 if opts['daemon_pipefds']:
584 if opts['daemon_pipefds']:
585 rfd, wfd = [int(x) for x in opts['daemon_pipefds'].split(',')]
585 rfd, wfd = [int(x) for x in opts['daemon_pipefds'].split(',')]
586 os.close(rfd)
586 os.close(rfd)
587 try:
587 try:
588 os.setsid()
588 os.setsid()
589 except AttributeError:
589 except AttributeError:
590 pass
590 pass
591 os.write(wfd, 'y')
591 os.write(wfd, 'y')
592 os.close(wfd)
592 os.close(wfd)
593 sys.stdout.flush()
593 sys.stdout.flush()
594 sys.stderr.flush()
594 sys.stderr.flush()
595
595
596 nullfd = os.open(util.nulldev, os.O_RDWR)
596 nullfd = os.open(util.nulldev, os.O_RDWR)
597 logfilefd = nullfd
597 logfilefd = nullfd
598 if logfile:
598 if logfile:
599 logfilefd = os.open(logfile, os.O_RDWR | os.O_CREAT | os.O_APPEND)
599 logfilefd = os.open(logfile, os.O_RDWR | os.O_CREAT | os.O_APPEND)
600 os.dup2(nullfd, 0)
600 os.dup2(nullfd, 0)
601 os.dup2(logfilefd, 1)
601 os.dup2(logfilefd, 1)
602 os.dup2(logfilefd, 2)
602 os.dup2(logfilefd, 2)
603 if nullfd not in (0, 1, 2):
603 if nullfd not in (0, 1, 2):
604 os.close(nullfd)
604 os.close(nullfd)
605 if logfile and logfilefd not in (0, 1, 2):
605 if logfile and logfilefd not in (0, 1, 2):
606 os.close(logfilefd)
606 os.close(logfilefd)
607
607
608 if runfn:
608 if runfn:
609 return runfn()
609 return runfn()
610
610
611 class changeset_printer(object):
611 class changeset_printer(object):
612 '''show changeset information when templating not requested.'''
612 '''show changeset information when templating not requested.'''
613
613
614 def __init__(self, ui, repo, patch, diffopts, buffered):
614 def __init__(self, ui, repo, patch, diffopts, buffered):
615 self.ui = ui
615 self.ui = ui
616 self.repo = repo
616 self.repo = repo
617 self.buffered = buffered
617 self.buffered = buffered
618 self.patch = patch
618 self.patch = patch
619 self.diffopts = diffopts
619 self.diffopts = diffopts
620 self.header = {}
620 self.header = {}
621 self.hunk = {}
621 self.hunk = {}
622 self.lastheader = None
622 self.lastheader = None
623
623
624 def flush(self, rev):
624 def flush(self, rev):
625 if rev in self.header:
625 if rev in self.header:
626 h = self.header[rev]
626 h = self.header[rev]
627 if h != self.lastheader:
627 if h != self.lastheader:
628 self.lastheader = h
628 self.lastheader = h
629 self.ui.write(h)
629 self.ui.write(h)
630 del self.header[rev]
630 del self.header[rev]
631 if rev in self.hunk:
631 if rev in self.hunk:
632 self.ui.write(self.hunk[rev])
632 self.ui.write(self.hunk[rev])
633 del self.hunk[rev]
633 del self.hunk[rev]
634 return 1
634 return 1
635 return 0
635 return 0
636
636
637 def show(self, ctx, copies=(), **props):
637 def show(self, ctx, copies=(), **props):
638 if self.buffered:
638 if self.buffered:
639 self.ui.pushbuffer()
639 self.ui.pushbuffer()
640 self._show(ctx, copies, props)
640 self._show(ctx, copies, props)
641 self.hunk[ctx.rev()] = self.ui.popbuffer()
641 self.hunk[ctx.rev()] = self.ui.popbuffer()
642 else:
642 else:
643 self._show(ctx, copies, props)
643 self._show(ctx, copies, props)
644
644
645 def _show(self, ctx, copies, props):
645 def _show(self, ctx, copies, props):
646 '''show a single changeset or file revision'''
646 '''show a single changeset or file revision'''
647 changenode = ctx.node()
647 changenode = ctx.node()
648 rev = ctx.rev()
648 rev = ctx.rev()
649
649
650 if self.ui.quiet:
650 if self.ui.quiet:
651 self.ui.write("%d:%s\n" % (rev, short(changenode)))
651 self.ui.write("%d:%s\n" % (rev, short(changenode)))
652 return
652 return
653
653
654 log = self.repo.changelog
654 log = self.repo.changelog
655 date = util.datestr(ctx.date())
655 date = util.datestr(ctx.date())
656
656
657 hexfunc = self.ui.debugflag and hex or short
657 hexfunc = self.ui.debugflag and hex or short
658
658
659 parents = [(p, hexfunc(log.node(p)))
659 parents = [(p, hexfunc(log.node(p)))
660 for p in self._meaningful_parentrevs(log, rev)]
660 for p in self._meaningful_parentrevs(log, rev)]
661
661
662 self.ui.write(_("changeset: %d:%s\n") % (rev, hexfunc(changenode)))
662 self.ui.write(_("changeset: %d:%s\n") % (rev, hexfunc(changenode)))
663
663
664 branch = ctx.branch()
664 branch = ctx.branch()
665 # don't show the default branch name
665 # don't show the default branch name
666 if branch != 'default':
666 if branch != 'default':
667 branch = encoding.tolocal(branch)
667 branch = encoding.tolocal(branch)
668 self.ui.write(_("branch: %s\n") % branch)
668 self.ui.write(_("branch: %s\n") % branch)
669 for tag in self.repo.nodetags(changenode):
669 for tag in self.repo.nodetags(changenode):
670 self.ui.write(_("tag: %s\n") % tag)
670 self.ui.write(_("tag: %s\n") % tag)
671 for parent in parents:
671 for parent in parents:
672 self.ui.write(_("parent: %d:%s\n") % parent)
672 self.ui.write(_("parent: %d:%s\n") % parent)
673
673
674 if self.ui.debugflag:
674 if self.ui.debugflag:
675 mnode = ctx.manifestnode()
675 mnode = ctx.manifestnode()
676 self.ui.write(_("manifest: %d:%s\n") %
676 self.ui.write(_("manifest: %d:%s\n") %
677 (self.repo.manifest.rev(mnode), hex(mnode)))
677 (self.repo.manifest.rev(mnode), hex(mnode)))
678 self.ui.write(_("user: %s\n") % ctx.user())
678 self.ui.write(_("user: %s\n") % ctx.user())
679 self.ui.write(_("date: %s\n") % date)
679 self.ui.write(_("date: %s\n") % date)
680
680
681 if self.ui.debugflag:
681 if self.ui.debugflag:
682 files = self.repo.status(log.parents(changenode)[0], changenode)[:3]
682 files = self.repo.status(log.parents(changenode)[0], changenode)[:3]
683 for key, value in zip([_("files:"), _("files+:"), _("files-:")],
683 for key, value in zip([_("files:"), _("files+:"), _("files-:")],
684 files):
684 files):
685 if value:
685 if value:
686 self.ui.write("%-12s %s\n" % (key, " ".join(value)))
686 self.ui.write("%-12s %s\n" % (key, " ".join(value)))
687 elif ctx.files() and self.ui.verbose:
687 elif ctx.files() and self.ui.verbose:
688 self.ui.write(_("files: %s\n") % " ".join(ctx.files()))
688 self.ui.write(_("files: %s\n") % " ".join(ctx.files()))
689 if copies and self.ui.verbose:
689 if copies and self.ui.verbose:
690 copies = ['%s (%s)' % c for c in copies]
690 copies = ['%s (%s)' % c for c in copies]
691 self.ui.write(_("copies: %s\n") % ' '.join(copies))
691 self.ui.write(_("copies: %s\n") % ' '.join(copies))
692
692
693 extra = ctx.extra()
693 extra = ctx.extra()
694 if extra and self.ui.debugflag:
694 if extra and self.ui.debugflag:
695 for key, value in sorted(extra.items()):
695 for key, value in sorted(extra.items()):
696 self.ui.write(_("extra: %s=%s\n")
696 self.ui.write(_("extra: %s=%s\n")
697 % (key, value.encode('string_escape')))
697 % (key, value.encode('string_escape')))
698
698
699 description = ctx.description().strip()
699 description = ctx.description().strip()
700 if description:
700 if description:
701 if self.ui.verbose:
701 if self.ui.verbose:
702 self.ui.write(_("description:\n"))
702 self.ui.write(_("description:\n"))
703 self.ui.write(description)
703 self.ui.write(description)
704 self.ui.write("\n\n")
704 self.ui.write("\n\n")
705 else:
705 else:
706 self.ui.write(_("summary: %s\n") %
706 self.ui.write(_("summary: %s\n") %
707 description.splitlines()[0])
707 description.splitlines()[0])
708 self.ui.write("\n")
708 self.ui.write("\n")
709
709
710 self.showpatch(changenode)
710 self.showpatch(changenode)
711
711
712 def showpatch(self, node):
712 def showpatch(self, node):
713 if self.patch:
713 if self.patch:
714 prev = self.repo.changelog.parents(node)[0]
714 prev = self.repo.changelog.parents(node)[0]
715 chunks = patch.diff(self.repo, prev, node, match=self.patch,
715 chunks = patch.diff(self.repo, prev, node, match=self.patch,
716 opts=patch.diffopts(self.ui, self.diffopts))
716 opts=patch.diffopts(self.ui, self.diffopts))
717 for chunk in chunks:
717 for chunk in chunks:
718 self.ui.write(chunk)
718 self.ui.write(chunk)
719 self.ui.write("\n")
719 self.ui.write("\n")
720
720
721 def _meaningful_parentrevs(self, log, rev):
721 def _meaningful_parentrevs(self, log, rev):
722 """Return list of meaningful (or all if debug) parentrevs for rev.
722 """Return list of meaningful (or all if debug) parentrevs for rev.
723
723
724 For merges (two non-nullrev revisions) both parents are meaningful.
724 For merges (two non-nullrev revisions) both parents are meaningful.
725 Otherwise the first parent revision is considered meaningful if it
725 Otherwise the first parent revision is considered meaningful if it
726 is not the preceding revision.
726 is not the preceding revision.
727 """
727 """
728 parents = log.parentrevs(rev)
728 parents = log.parentrevs(rev)
729 if not self.ui.debugflag and parents[1] == nullrev:
729 if not self.ui.debugflag and parents[1] == nullrev:
730 if parents[0] >= rev - 1:
730 if parents[0] >= rev - 1:
731 parents = []
731 parents = []
732 else:
732 else:
733 parents = [parents[0]]
733 parents = [parents[0]]
734 return parents
734 return parents
735
735
736
736
737 class changeset_templater(changeset_printer):
737 class changeset_templater(changeset_printer):
738 '''format changeset information.'''
738 '''format changeset information.'''
739
739
740 def __init__(self, ui, repo, patch, diffopts, mapfile, buffered):
740 def __init__(self, ui, repo, patch, diffopts, mapfile, buffered):
741 changeset_printer.__init__(self, ui, repo, patch, diffopts, buffered)
741 changeset_printer.__init__(self, ui, repo, patch, diffopts, buffered)
742 formatnode = ui.debugflag and (lambda x: x) or (lambda x: x[:12])
742 formatnode = ui.debugflag and (lambda x: x) or (lambda x: x[:12])
743 self.t = templater.templater(mapfile, {'formatnode': formatnode},
743 self.t = templater.templater(mapfile, {'formatnode': formatnode},
744 cache={
744 cache={
745 'parent': '{rev}:{node|formatnode} ',
745 'parent': '{rev}:{node|formatnode} ',
746 'manifest': '{rev}:{node|formatnode}',
746 'manifest': '{rev}:{node|formatnode}',
747 'filecopy': '{name} ({source})'})
747 'filecopy': '{name} ({source})'})
748 # Cache mapping from rev to a tuple with tag date, tag
748 # Cache mapping from rev to a tuple with tag date, tag
749 # distance and tag name
749 # distance and tag name
750 self._latesttagcache = {-1: (0, 0, 'null')}
750 self._latesttagcache = {-1: (0, 0, 'null')}
751
751
752 def use_template(self, t):
752 def use_template(self, t):
753 '''set template string to use'''
753 '''set template string to use'''
754 self.t.cache['changeset'] = t
754 self.t.cache['changeset'] = t
755
755
756 def _meaningful_parentrevs(self, ctx):
756 def _meaningful_parentrevs(self, ctx):
757 """Return list of meaningful (or all if debug) parentrevs for rev.
757 """Return list of meaningful (or all if debug) parentrevs for rev.
758 """
758 """
759 parents = ctx.parents()
759 parents = ctx.parents()
760 if len(parents) > 1:
760 if len(parents) > 1:
761 return parents
761 return parents
762 if self.ui.debugflag:
762 if self.ui.debugflag:
763 return [parents[0], self.repo['null']]
763 return [parents[0], self.repo['null']]
764 if parents[0].rev() >= ctx.rev() - 1:
764 if parents[0].rev() >= ctx.rev() - 1:
765 return []
765 return []
766 return parents
766 return parents
767
767
768 def _latesttaginfo(self, rev):
768 def _latesttaginfo(self, rev):
769 '''return date, distance and name for the latest tag of rev'''
769 '''return date, distance and name for the latest tag of rev'''
770 todo = [rev]
770 todo = [rev]
771 while todo:
771 while todo:
772 rev = todo.pop()
772 rev = todo.pop()
773 if rev in self._latesttagcache:
773 if rev in self._latesttagcache:
774 continue
774 continue
775 ctx = self.repo[rev]
775 ctx = self.repo[rev]
776 tags = [t for t in ctx.tags() if self.repo.tagtype(t) == 'global']
776 tags = [t for t in ctx.tags() if self.repo.tagtype(t) == 'global']
777 if tags:
777 if tags:
778 self._latesttagcache[rev] = ctx.date()[0], 0, ':'.join(sorted(tags))
778 self._latesttagcache[rev] = ctx.date()[0], 0, ':'.join(sorted(tags))
779 continue
779 continue
780 try:
780 try:
781 # The tuples are laid out so the right one can be found by comparison.
781 # The tuples are laid out so the right one can be found by comparison.
782 pdate, pdist, ptag = max(
782 pdate, pdist, ptag = max(
783 self._latesttagcache[p.rev()] for p in ctx.parents())
783 self._latesttagcache[p.rev()] for p in ctx.parents())
784 except KeyError:
784 except KeyError:
785 # Cache miss - recurse
785 # Cache miss - recurse
786 todo.append(rev)
786 todo.append(rev)
787 todo.extend(p.rev() for p in ctx.parents())
787 todo.extend(p.rev() for p in ctx.parents())
788 continue
788 continue
789 self._latesttagcache[rev] = pdate, pdist + 1, ptag
789 self._latesttagcache[rev] = pdate, pdist + 1, ptag
790 return self._latesttagcache[rev]
790 return self._latesttagcache[rev]
791
791
792 def _show(self, ctx, copies, props):
792 def _show(self, ctx, copies, props):
793 '''show a single changeset or file revision'''
793 '''show a single changeset or file revision'''
794
794
795 def showlist(name, values, plural=None, **args):
795 def showlist(name, values, plural=None, **args):
796 '''expand set of values.
796 '''expand set of values.
797 name is name of key in template map.
797 name is name of key in template map.
798 values is list of strings or dicts.
798 values is list of strings or dicts.
799 plural is plural of name, if not simply name + 's'.
799 plural is plural of name, if not simply name + 's'.
800
800
801 expansion works like this, given name 'foo'.
801 expansion works like this, given name 'foo'.
802
802
803 if values is empty, expand 'no_foos'.
803 if values is empty, expand 'no_foos'.
804
804
805 if 'foo' not in template map, return values as a string,
805 if 'foo' not in template map, return values as a string,
806 joined by space.
806 joined by space.
807
807
808 expand 'start_foos'.
808 expand 'start_foos'.
809
809
810 for each value, expand 'foo'. if 'last_foo' in template
810 for each value, expand 'foo'. if 'last_foo' in template
811 map, expand it instead of 'foo' for last key.
811 map, expand it instead of 'foo' for last key.
812
812
813 expand 'end_foos'.
813 expand 'end_foos'.
814 '''
814 '''
815 if plural: names = plural
815 if plural: names = plural
816 else: names = name + 's'
816 else: names = name + 's'
817 if not values:
817 if not values:
818 noname = 'no_' + names
818 noname = 'no_' + names
819 if noname in self.t:
819 if noname in self.t:
820 yield self.t(noname, **args)
820 yield self.t(noname, **args)
821 return
821 return
822 if name not in self.t:
822 if name not in self.t:
823 if isinstance(values[0], str):
823 if isinstance(values[0], str):
824 yield ' '.join(values)
824 yield ' '.join(values)
825 else:
825 else:
826 for v in values:
826 for v in values:
827 yield dict(v, **args)
827 yield dict(v, **args)
828 return
828 return
829 startname = 'start_' + names
829 startname = 'start_' + names
830 if startname in self.t:
830 if startname in self.t:
831 yield self.t(startname, **args)
831 yield self.t(startname, **args)
832 vargs = args.copy()
832 vargs = args.copy()
833 def one(v, tag=name):
833 def one(v, tag=name):
834 try:
834 try:
835 vargs.update(v)
835 vargs.update(v)
836 except (AttributeError, ValueError):
836 except (AttributeError, ValueError):
837 try:
837 try:
838 for a, b in v:
838 for a, b in v:
839 vargs[a] = b
839 vargs[a] = b
840 except ValueError:
840 except ValueError:
841 vargs[name] = v
841 vargs[name] = v
842 return self.t(tag, **vargs)
842 return self.t(tag, **vargs)
843 lastname = 'last_' + name
843 lastname = 'last_' + name
844 if lastname in self.t:
844 if lastname in self.t:
845 last = values.pop()
845 last = values.pop()
846 else:
846 else:
847 last = None
847 last = None
848 for v in values:
848 for v in values:
849 yield one(v)
849 yield one(v)
850 if last is not None:
850 if last is not None:
851 yield one(last, tag=lastname)
851 yield one(last, tag=lastname)
852 endname = 'end_' + names
852 endname = 'end_' + names
853 if endname in self.t:
853 if endname in self.t:
854 yield self.t(endname, **args)
854 yield self.t(endname, **args)
855
855
856 def showbranches(**args):
856 def showbranches(**args):
857 branch = ctx.branch()
857 branch = ctx.branch()
858 if branch != 'default':
858 if branch != 'default':
859 branch = encoding.tolocal(branch)
859 branch = encoding.tolocal(branch)
860 return showlist('branch', [branch], plural='branches', **args)
860 return showlist('branch', [branch], plural='branches', **args)
861
861
862 def showparents(**args):
862 def showparents(**args):
863 parents = [[('rev', p.rev()), ('node', p.hex())]
863 parents = [[('rev', p.rev()), ('node', p.hex())]
864 for p in self._meaningful_parentrevs(ctx)]
864 for p in self._meaningful_parentrevs(ctx)]
865 return showlist('parent', parents, **args)
865 return showlist('parent', parents, **args)
866
866
867 def showtags(**args):
867 def showtags(**args):
868 return showlist('tag', ctx.tags(), **args)
868 return showlist('tag', ctx.tags(), **args)
869
869
870 def showextras(**args):
870 def showextras(**args):
871 for key, value in sorted(ctx.extra().items()):
871 for key, value in sorted(ctx.extra().items()):
872 args = args.copy()
872 args = args.copy()
873 args.update(dict(key=key, value=value))
873 args.update(dict(key=key, value=value))
874 yield self.t('extra', **args)
874 yield self.t('extra', **args)
875
875
876 def showcopies(**args):
876 def showcopies(**args):
877 c = [{'name': x[0], 'source': x[1]} for x in copies]
877 c = [{'name': x[0], 'source': x[1]} for x in copies]
878 return showlist('file_copy', c, plural='file_copies', **args)
878 return showlist('file_copy', c, plural='file_copies', **args)
879
879
880 files = []
880 files = []
881 def getfiles():
881 def getfiles():
882 if not files:
882 if not files:
883 files[:] = self.repo.status(ctx.parents()[0].node(),
883 files[:] = self.repo.status(ctx.parents()[0].node(),
884 ctx.node())[:3]
884 ctx.node())[:3]
885 return files
885 return files
886 def showfiles(**args):
886 def showfiles(**args):
887 return showlist('file', ctx.files(), **args)
887 return showlist('file', ctx.files(), **args)
888 def showmods(**args):
888 def showmods(**args):
889 return showlist('file_mod', getfiles()[0], **args)
889 return showlist('file_mod', getfiles()[0], **args)
890 def showadds(**args):
890 def showadds(**args):
891 return showlist('file_add', getfiles()[1], **args)
891 return showlist('file_add', getfiles()[1], **args)
892 def showdels(**args):
892 def showdels(**args):
893 return showlist('file_del', getfiles()[2], **args)
893 return showlist('file_del', getfiles()[2], **args)
894 def showmanifest(**args):
894 def showmanifest(**args):
895 args = args.copy()
895 args = args.copy()
896 args.update(dict(rev=self.repo.manifest.rev(ctx.changeset()[0]),
896 args.update(dict(rev=self.repo.manifest.rev(ctx.changeset()[0]),
897 node=hex(ctx.changeset()[0])))
897 node=hex(ctx.changeset()[0])))
898 return self.t('manifest', **args)
898 return self.t('manifest', **args)
899
899
900 def showdiffstat(**args):
900 def showdiffstat(**args):
901 diff = patch.diff(self.repo, ctx.parents()[0].node(), ctx.node())
901 diff = patch.diff(self.repo, ctx.parents()[0].node(), ctx.node())
902 files, adds, removes = 0, 0, 0
902 files, adds, removes = 0, 0, 0
903 for i in patch.diffstatdata(util.iterlines(diff)):
903 for i in patch.diffstatdata(util.iterlines(diff)):
904 files += 1
904 files += 1
905 adds += i[1]
905 adds += i[1]
906 removes += i[2]
906 removes += i[2]
907 return '%s: +%s/-%s' % (files, adds, removes)
907 return '%s: +%s/-%s' % (files, adds, removes)
908
908
909 def showlatesttag(**args):
909 def showlatesttag(**args):
910 return self._latesttaginfo(ctx.rev())[2]
910 return self._latesttaginfo(ctx.rev())[2]
911 def showlatesttagdistance(**args):
911 def showlatesttagdistance(**args):
912 return self._latesttaginfo(ctx.rev())[1]
912 return self._latesttaginfo(ctx.rev())[1]
913
913
914 defprops = {
914 defprops = {
915 'author': ctx.user(),
915 'author': ctx.user(),
916 'branches': showbranches,
916 'branches': showbranches,
917 'date': ctx.date(),
917 'date': ctx.date(),
918 'desc': ctx.description().strip(),
918 'desc': ctx.description().strip(),
919 'file_adds': showadds,
919 'file_adds': showadds,
920 'file_dels': showdels,
920 'file_dels': showdels,
921 'file_mods': showmods,
921 'file_mods': showmods,
922 'files': showfiles,
922 'files': showfiles,
923 'file_copies': showcopies,
923 'file_copies': showcopies,
924 'manifest': showmanifest,
924 'manifest': showmanifest,
925 'node': ctx.hex(),
925 'node': ctx.hex(),
926 'parents': showparents,
926 'parents': showparents,
927 'rev': ctx.rev(),
927 'rev': ctx.rev(),
928 'tags': showtags,
928 'tags': showtags,
929 'extras': showextras,
929 'extras': showextras,
930 'diffstat': showdiffstat,
930 'diffstat': showdiffstat,
931 'latesttag': showlatesttag,
931 'latesttag': showlatesttag,
932 'latesttagdistance': showlatesttagdistance,
932 'latesttagdistance': showlatesttagdistance,
933 }
933 }
934 props = props.copy()
934 props = props.copy()
935 props.update(defprops)
935 props.update(defprops)
936
936
937 # find correct templates for current mode
937 # find correct templates for current mode
938
938
939 tmplmodes = [
939 tmplmodes = [
940 (True, None),
940 (True, None),
941 (self.ui.verbose, 'verbose'),
941 (self.ui.verbose, 'verbose'),
942 (self.ui.quiet, 'quiet'),
942 (self.ui.quiet, 'quiet'),
943 (self.ui.debugflag, 'debug'),
943 (self.ui.debugflag, 'debug'),
944 ]
944 ]
945
945
946 types = {'header': '', 'changeset': 'changeset'}
946 types = {'header': '', 'changeset': 'changeset'}
947 for mode, postfix in tmplmodes:
947 for mode, postfix in tmplmodes:
948 for type in types:
948 for type in types:
949 cur = postfix and ('%s_%s' % (type, postfix)) or type
949 cur = postfix and ('%s_%s' % (type, postfix)) or type
950 if mode and cur in self.t:
950 if mode and cur in self.t:
951 types[type] = cur
951 types[type] = cur
952
952
953 try:
953 try:
954
954
955 # write header
955 # write header
956 if types['header']:
956 if types['header']:
957 h = templater.stringify(self.t(types['header'], **props))
957 h = templater.stringify(self.t(types['header'], **props))
958 if self.buffered:
958 if self.buffered:
959 self.header[ctx.rev()] = h
959 self.header[ctx.rev()] = h
960 else:
960 else:
961 self.ui.write(h)
961 self.ui.write(h)
962
962
963 # write changeset metadata, then patch if requested
963 # write changeset metadata, then patch if requested
964 key = types['changeset']
964 key = types['changeset']
965 self.ui.write(templater.stringify(self.t(key, **props)))
965 self.ui.write(templater.stringify(self.t(key, **props)))
966 self.showpatch(ctx.node())
966 self.showpatch(ctx.node())
967
967
968 except KeyError, inst:
968 except KeyError, inst:
969 msg = _("%s: no key named '%s'")
969 msg = _("%s: no key named '%s'")
970 raise util.Abort(msg % (self.t.mapfile, inst.args[0]))
970 raise util.Abort(msg % (self.t.mapfile, inst.args[0]))
971 except SyntaxError, inst:
971 except SyntaxError, inst:
972 raise util.Abort(_('%s: %s') % (self.t.mapfile, inst.args[0]))
972 raise util.Abort(_('%s: %s') % (self.t.mapfile, inst.args[0]))
973
973
974 def show_changeset(ui, repo, opts, buffered=False, matchfn=False):
974 def show_changeset(ui, repo, opts, buffered=False, matchfn=False):
975 """show one changeset using template or regular display.
975 """show one changeset using template or regular display.
976
976
977 Display format will be the first non-empty hit of:
977 Display format will be the first non-empty hit of:
978 1. option 'template'
978 1. option 'template'
979 2. option 'style'
979 2. option 'style'
980 3. [ui] setting 'logtemplate'
980 3. [ui] setting 'logtemplate'
981 4. [ui] setting 'style'
981 4. [ui] setting 'style'
982 If all of these values are either the unset or the empty string,
982 If all of these values are either the unset or the empty string,
983 regular display via changeset_printer() is done.
983 regular display via changeset_printer() is done.
984 """
984 """
985 # options
985 # options
986 patch = False
986 patch = False
987 if opts.get('patch'):
987 if opts.get('patch'):
988 patch = matchfn or matchall(repo)
988 patch = matchfn or matchall(repo)
989
989
990 tmpl = opts.get('template')
990 tmpl = opts.get('template')
991 style = None
991 style = None
992 if tmpl:
992 if tmpl:
993 tmpl = templater.parsestring(tmpl, quoted=False)
993 tmpl = templater.parsestring(tmpl, quoted=False)
994 else:
994 else:
995 style = opts.get('style')
995 style = opts.get('style')
996
996
997 # ui settings
997 # ui settings
998 if not (tmpl or style):
998 if not (tmpl or style):
999 tmpl = ui.config('ui', 'logtemplate')
999 tmpl = ui.config('ui', 'logtemplate')
1000 if tmpl:
1000 if tmpl:
1001 tmpl = templater.parsestring(tmpl)
1001 tmpl = templater.parsestring(tmpl)
1002 else:
1002 else:
1003 style = ui.config('ui', 'style')
1003 style = ui.config('ui', 'style')
1004
1004
1005 if not (tmpl or style):
1005 if not (tmpl or style):
1006 return changeset_printer(ui, repo, patch, opts, buffered)
1006 return changeset_printer(ui, repo, patch, opts, buffered)
1007
1007
1008 mapfile = None
1008 mapfile = None
1009 if style and not tmpl:
1009 if style and not tmpl:
1010 mapfile = style
1010 mapfile = style
1011 if not os.path.split(mapfile)[0]:
1011 if not os.path.split(mapfile)[0]:
1012 mapname = (templater.templatepath('map-cmdline.' + mapfile)
1012 mapname = (templater.templatepath('map-cmdline.' + mapfile)
1013 or templater.templatepath(mapfile))
1013 or templater.templatepath(mapfile))
1014 if mapname: mapfile = mapname
1014 if mapname: mapfile = mapname
1015
1015
1016 try:
1016 try:
1017 t = changeset_templater(ui, repo, patch, opts, mapfile, buffered)
1017 t = changeset_templater(ui, repo, patch, opts, mapfile, buffered)
1018 except SyntaxError, inst:
1018 except SyntaxError, inst:
1019 raise util.Abort(inst.args[0])
1019 raise util.Abort(inst.args[0])
1020 if tmpl: t.use_template(tmpl)
1020 if tmpl: t.use_template(tmpl)
1021 return t
1021 return t
1022
1022
1023 def finddate(ui, repo, date):
1023 def finddate(ui, repo, date):
1024 """Find the tipmost changeset that matches the given date spec"""
1024 """Find the tipmost changeset that matches the given date spec"""
1025
1025
1026 df = util.matchdate(date)
1026 df = util.matchdate(date)
1027 m = matchall(repo)
1027 m = matchall(repo)
1028 results = {}
1028 results = {}
1029
1029
1030 def prep(ctx, fns):
1030 def prep(ctx, fns):
1031 d = ctx.date()
1031 d = ctx.date()
1032 if df(d[0]):
1032 if df(d[0]):
1033 results[ctx.rev()] = d
1033 results[ctx.rev()] = d
1034
1034
1035 for ctx in walkchangerevs(repo, m, {'rev': None}, prep):
1035 for ctx in walkchangerevs(repo, m, {'rev': None}, prep):
1036 rev = ctx.rev()
1036 rev = ctx.rev()
1037 if rev in results:
1037 if rev in results:
1038 ui.status(_("Found revision %s from %s\n") %
1038 ui.status(_("Found revision %s from %s\n") %
1039 (rev, util.datestr(results[rev])))
1039 (rev, util.datestr(results[rev])))
1040 return str(rev)
1040 return str(rev)
1041
1041
1042 raise util.Abort(_("revision matching date not found"))
1042 raise util.Abort(_("revision matching date not found"))
1043
1043
1044 def walkchangerevs(repo, match, opts, prepare):
1044 def walkchangerevs(repo, match, opts, prepare):
1045 '''Iterate over files and the revs in which they changed.
1045 '''Iterate over files and the revs in which they changed.
1046
1046
1047 Callers most commonly need to iterate backwards over the history
1047 Callers most commonly need to iterate backwards over the history
1048 in which they are interested. Doing so has awful (quadratic-looking)
1048 in which they are interested. Doing so has awful (quadratic-looking)
1049 performance, so we use iterators in a "windowed" way.
1049 performance, so we use iterators in a "windowed" way.
1050
1050
1051 We walk a window of revisions in the desired order. Within the
1051 We walk a window of revisions in the desired order. Within the
1052 window, we first walk forwards to gather data, then in the desired
1052 window, we first walk forwards to gather data, then in the desired
1053 order (usually backwards) to display it.
1053 order (usually backwards) to display it.
1054
1054
1055 This function returns an iterator yielding contexts. Before
1055 This function returns an iterator yielding contexts. Before
1056 yielding each context, the iterator will first call the prepare
1056 yielding each context, the iterator will first call the prepare
1057 function on each context in the window in forward order.'''
1057 function on each context in the window in forward order.'''
1058
1058
1059 def increasing_windows(start, end, windowsize=8, sizelimit=512):
1059 def increasing_windows(start, end, windowsize=8, sizelimit=512):
1060 if start < end:
1060 if start < end:
1061 while start < end:
1061 while start < end:
1062 yield start, min(windowsize, end-start)
1062 yield start, min(windowsize, end-start)
1063 start += windowsize
1063 start += windowsize
1064 if windowsize < sizelimit:
1064 if windowsize < sizelimit:
1065 windowsize *= 2
1065 windowsize *= 2
1066 else:
1066 else:
1067 while start > end:
1067 while start > end:
1068 yield start, min(windowsize, start-end-1)
1068 yield start, min(windowsize, start-end-1)
1069 start -= windowsize
1069 start -= windowsize
1070 if windowsize < sizelimit:
1070 if windowsize < sizelimit:
1071 windowsize *= 2
1071 windowsize *= 2
1072
1072
1073 follow = opts.get('follow') or opts.get('follow_first')
1073 follow = opts.get('follow') or opts.get('follow_first')
1074
1074
1075 if not len(repo):
1075 if not len(repo):
1076 return []
1076 return []
1077
1077
1078 if follow:
1078 if follow:
1079 defrange = '%s:0' % repo['.'].rev()
1079 defrange = '%s:0' % repo['.'].rev()
1080 else:
1080 else:
1081 defrange = '-1:0'
1081 defrange = '-1:0'
1082 revs = revrange(repo, opts['rev'] or [defrange])
1082 revs = revrange(repo, opts['rev'] or [defrange])
1083 wanted = set()
1083 wanted = set()
1084 slowpath = match.anypats() or (match.files() and opts.get('removed'))
1084 slowpath = match.anypats() or (match.files() and opts.get('removed'))
1085 fncache = {}
1085 fncache = {}
1086 change = util.cachefunc(repo.changectx)
1086 change = util.cachefunc(repo.changectx)
1087
1087
1088 if not slowpath and not match.files():
1088 if not slowpath and not match.files():
1089 # No files, no patterns. Display all revs.
1089 # No files, no patterns. Display all revs.
1090 wanted = set(revs)
1090 wanted = set(revs)
1091 copies = []
1091 copies = []
1092
1092
1093 if not slowpath:
1093 if not slowpath:
1094 # Only files, no patterns. Check the history of each file.
1094 # Only files, no patterns. Check the history of each file.
1095 def filerevgen(filelog, node):
1095 def filerevgen(filelog, node):
1096 cl_count = len(repo)
1096 cl_count = len(repo)
1097 if node is None:
1097 if node is None:
1098 last = len(filelog) - 1
1098 last = len(filelog) - 1
1099 else:
1099 else:
1100 last = filelog.rev(node)
1100 last = filelog.rev(node)
1101 for i, window in increasing_windows(last, nullrev):
1101 for i, window in increasing_windows(last, nullrev):
1102 revs = []
1102 revs = []
1103 for j in xrange(i - window, i + 1):
1103 for j in xrange(i - window, i + 1):
1104 n = filelog.node(j)
1104 n = filelog.node(j)
1105 revs.append((filelog.linkrev(j),
1105 revs.append((filelog.linkrev(j),
1106 follow and filelog.renamed(n)))
1106 follow and filelog.renamed(n)))
1107 for rev in reversed(revs):
1107 for rev in reversed(revs):
1108 # only yield rev for which we have the changelog, it can
1108 # only yield rev for which we have the changelog, it can
1109 # happen while doing "hg log" during a pull or commit
1109 # happen while doing "hg log" during a pull or commit
1110 if rev[0] < cl_count:
1110 if rev[0] < cl_count:
1111 yield rev
1111 yield rev
1112 def iterfiles():
1112 def iterfiles():
1113 for filename in match.files():
1113 for filename in match.files():
1114 yield filename, None
1114 yield filename, None
1115 for filename_node in copies:
1115 for filename_node in copies:
1116 yield filename_node
1116 yield filename_node
1117 minrev, maxrev = min(revs), max(revs)
1117 minrev, maxrev = min(revs), max(revs)
1118 for file_, node in iterfiles():
1118 for file_, node in iterfiles():
1119 filelog = repo.file(file_)
1119 filelog = repo.file(file_)
1120 if not len(filelog):
1120 if not len(filelog):
1121 if node is None:
1121 if node is None:
1122 # A zero count may be a directory or deleted file, so
1122 # A zero count may be a directory or deleted file, so
1123 # try to find matching entries on the slow path.
1123 # try to find matching entries on the slow path.
1124 if follow:
1124 if follow:
1125 raise util.Abort(_('cannot follow nonexistent file: "%s"') % file_)
1125 raise util.Abort(_('cannot follow nonexistent file: "%s"') % file_)
1126 slowpath = True
1126 slowpath = True
1127 break
1127 break
1128 else:
1128 else:
1129 continue
1129 continue
1130 for rev, copied in filerevgen(filelog, node):
1130 for rev, copied in filerevgen(filelog, node):
1131 if rev <= maxrev:
1131 if rev <= maxrev:
1132 if rev < minrev:
1132 if rev < minrev:
1133 break
1133 break
1134 fncache.setdefault(rev, [])
1134 fncache.setdefault(rev, [])
1135 fncache[rev].append(file_)
1135 fncache[rev].append(file_)
1136 wanted.add(rev)
1136 wanted.add(rev)
1137 if follow and copied:
1137 if follow and copied:
1138 copies.append(copied)
1138 copies.append(copied)
1139 if slowpath:
1139 if slowpath:
1140 if follow:
1140 if follow:
1141 raise util.Abort(_('can only follow copies/renames for explicit '
1141 raise util.Abort(_('can only follow copies/renames for explicit '
1142 'filenames'))
1142 'filenames'))
1143
1143
1144 # The slow path checks files modified in every changeset.
1144 # The slow path checks files modified in every changeset.
1145 def changerevgen():
1145 def changerevgen():
1146 for i, window in increasing_windows(len(repo) - 1, nullrev):
1146 for i, window in increasing_windows(len(repo) - 1, nullrev):
1147 for j in xrange(i - window, i + 1):
1147 for j in xrange(i - window, i + 1):
1148 yield change(j)
1148 yield change(j)
1149
1149
1150 for ctx in changerevgen():
1150 for ctx in changerevgen():
1151 matches = filter(match, ctx.files())
1151 matches = filter(match, ctx.files())
1152 if matches:
1152 if matches:
1153 fncache[ctx.rev()] = matches
1153 fncache[ctx.rev()] = matches
1154 wanted.add(ctx.rev())
1154 wanted.add(ctx.rev())
1155
1155
1156 class followfilter(object):
1156 class followfilter(object):
1157 def __init__(self, onlyfirst=False):
1157 def __init__(self, onlyfirst=False):
1158 self.startrev = nullrev
1158 self.startrev = nullrev
1159 self.roots = []
1159 self.roots = []
1160 self.onlyfirst = onlyfirst
1160 self.onlyfirst = onlyfirst
1161
1161
1162 def match(self, rev):
1162 def match(self, rev):
1163 def realparents(rev):
1163 def realparents(rev):
1164 if self.onlyfirst:
1164 if self.onlyfirst:
1165 return repo.changelog.parentrevs(rev)[0:1]
1165 return repo.changelog.parentrevs(rev)[0:1]
1166 else:
1166 else:
1167 return filter(lambda x: x != nullrev,
1167 return filter(lambda x: x != nullrev,
1168 repo.changelog.parentrevs(rev))
1168 repo.changelog.parentrevs(rev))
1169
1169
1170 if self.startrev == nullrev:
1170 if self.startrev == nullrev:
1171 self.startrev = rev
1171 self.startrev = rev
1172 return True
1172 return True
1173
1173
1174 if rev > self.startrev:
1174 if rev > self.startrev:
1175 # forward: all descendants
1175 # forward: all descendants
1176 if not self.roots:
1176 if not self.roots:
1177 self.roots.append(self.startrev)
1177 self.roots.append(self.startrev)
1178 for parent in realparents(rev):
1178 for parent in realparents(rev):
1179 if parent in self.roots:
1179 if parent in self.roots:
1180 self.roots.append(rev)
1180 self.roots.append(rev)
1181 return True
1181 return True
1182 else:
1182 else:
1183 # backwards: all parents
1183 # backwards: all parents
1184 if not self.roots:
1184 if not self.roots:
1185 self.roots.extend(realparents(self.startrev))
1185 self.roots.extend(realparents(self.startrev))
1186 if rev in self.roots:
1186 if rev in self.roots:
1187 self.roots.remove(rev)
1187 self.roots.remove(rev)
1188 self.roots.extend(realparents(rev))
1188 self.roots.extend(realparents(rev))
1189 return True
1189 return True
1190
1190
1191 return False
1191 return False
1192
1192
1193 # it might be worthwhile to do this in the iterator if the rev range
1193 # it might be worthwhile to do this in the iterator if the rev range
1194 # is descending and the prune args are all within that range
1194 # is descending and the prune args are all within that range
1195 for rev in opts.get('prune', ()):
1195 for rev in opts.get('prune', ()):
1196 rev = repo.changelog.rev(repo.lookup(rev))
1196 rev = repo.changelog.rev(repo.lookup(rev))
1197 ff = followfilter()
1197 ff = followfilter()
1198 stop = min(revs[0], revs[-1])
1198 stop = min(revs[0], revs[-1])
1199 for x in xrange(rev, stop-1, -1):
1199 for x in xrange(rev, stop-1, -1):
1200 if ff.match(x):
1200 if ff.match(x):
1201 wanted.discard(x)
1201 wanted.discard(x)
1202
1202
1203 def iterate():
1203 def iterate():
1204 if follow and not match.files():
1204 if follow and not match.files():
1205 ff = followfilter(onlyfirst=opts.get('follow_first'))
1205 ff = followfilter(onlyfirst=opts.get('follow_first'))
1206 def want(rev):
1206 def want(rev):
1207 return ff.match(rev) and rev in wanted
1207 return ff.match(rev) and rev in wanted
1208 else:
1208 else:
1209 def want(rev):
1209 def want(rev):
1210 return rev in wanted
1210 return rev in wanted
1211
1211
1212 for i, window in increasing_windows(0, len(revs)):
1212 for i, window in increasing_windows(0, len(revs)):
1213 change = util.cachefunc(repo.changectx)
1213 change = util.cachefunc(repo.changectx)
1214 nrevs = [rev for rev in revs[i:i+window] if want(rev)]
1214 nrevs = [rev for rev in revs[i:i+window] if want(rev)]
1215 for rev in sorted(nrevs):
1215 for rev in sorted(nrevs):
1216 fns = fncache.get(rev)
1216 fns = fncache.get(rev)
1217 ctx = change(rev)
1217 ctx = change(rev)
1218 if not fns:
1218 if not fns:
1219 def fns_generator():
1219 def fns_generator():
1220 for f in ctx.files():
1220 for f in ctx.files():
1221 if match(f):
1221 if match(f):
1222 yield f
1222 yield f
1223 fns = fns_generator()
1223 fns = fns_generator()
1224 prepare(ctx, fns)
1224 prepare(ctx, fns)
1225 for rev in nrevs:
1225 for rev in nrevs:
1226 yield change(rev)
1226 yield change(rev)
1227 return iterate()
1227 return iterate()
1228
1228
1229 def commit(ui, repo, commitfunc, pats, opts):
1229 def commit(ui, repo, commitfunc, pats, opts):
1230 '''commit the specified files or all outstanding changes'''
1230 '''commit the specified files or all outstanding changes'''
1231 date = opts.get('date')
1231 date = opts.get('date')
1232 if date:
1232 if date:
1233 opts['date'] = util.parsedate(date)
1233 opts['date'] = util.parsedate(date)
1234 message = logmessage(opts)
1234 message = logmessage(opts)
1235
1235
1236 # extract addremove carefully -- this function can be called from a command
1236 # extract addremove carefully -- this function can be called from a command
1237 # that doesn't support addremove
1237 # that doesn't support addremove
1238 if opts.get('addremove'):
1238 if opts.get('addremove'):
1239 addremove(repo, pats, opts)
1239 addremove(repo, pats, opts)
1240
1240
1241 return commitfunc(ui, repo, message, match(repo, pats, opts), opts)
1241 return commitfunc(ui, repo, message, match(repo, pats, opts), opts)
1242
1242
1243 def commiteditor(repo, ctx, subs):
1243 def commiteditor(repo, ctx, subs):
1244 if ctx.description():
1244 if ctx.description():
1245 return ctx.description()
1245 return ctx.description()
1246 return commitforceeditor(repo, ctx, subs)
1246 return commitforceeditor(repo, ctx, subs)
1247
1247
1248 def commitforceeditor(repo, ctx, subs):
1248 def commitforceeditor(repo, ctx, subs):
1249 edittext = []
1249 edittext = []
1250 modified, added, removed = ctx.modified(), ctx.added(), ctx.removed()
1250 modified, added, removed = ctx.modified(), ctx.added(), ctx.removed()
1251 if ctx.description():
1251 if ctx.description():
1252 edittext.append(ctx.description())
1252 edittext.append(ctx.description())
1253 edittext.append("")
1253 edittext.append("")
1254 edittext.append("") # Empty line between message and comments.
1254 edittext.append("") # Empty line between message and comments.
1255 edittext.append(_("HG: Enter commit message."
1255 edittext.append(_("HG: Enter commit message."
1256 " Lines beginning with 'HG:' are removed."))
1256 " Lines beginning with 'HG:' are removed."))
1257 edittext.append(_("HG: Leave message empty to abort commit."))
1257 edittext.append(_("HG: Leave message empty to abort commit."))
1258 edittext.append("HG: --")
1258 edittext.append("HG: --")
1259 edittext.append(_("HG: user: %s") % ctx.user())
1259 edittext.append(_("HG: user: %s") % ctx.user())
1260 if ctx.p2():
1260 if ctx.p2():
1261 edittext.append(_("HG: branch merge"))
1261 edittext.append(_("HG: branch merge"))
1262 if ctx.branch():
1262 if ctx.branch():
1263 edittext.append(_("HG: branch '%s'")
1263 edittext.append(_("HG: branch '%s'")
1264 % encoding.tolocal(ctx.branch()))
1264 % encoding.tolocal(ctx.branch()))
1265 edittext.extend([_("HG: subrepo %s") % s for s in subs])
1265 edittext.extend([_("HG: subrepo %s") % s for s in subs])
1266 edittext.extend([_("HG: added %s") % f for f in added])
1266 edittext.extend([_("HG: added %s") % f for f in added])
1267 edittext.extend([_("HG: changed %s") % f for f in modified])
1267 edittext.extend([_("HG: changed %s") % f for f in modified])
1268 edittext.extend([_("HG: removed %s") % f for f in removed])
1268 edittext.extend([_("HG: removed %s") % f for f in removed])
1269 if not added and not modified and not removed:
1269 if not added and not modified and not removed:
1270 edittext.append(_("HG: no files changed"))
1270 edittext.append(_("HG: no files changed"))
1271 edittext.append("")
1271 edittext.append("")
1272 # run editor in the repository root
1272 # run editor in the repository root
1273 olddir = os.getcwd()
1273 olddir = os.getcwd()
1274 os.chdir(repo.root)
1274 os.chdir(repo.root)
1275 text = repo.ui.edit("\n".join(edittext), ctx.user())
1275 text = repo.ui.edit("\n".join(edittext), ctx.user())
1276 text = re.sub("(?m)^HG:.*\n", "", text)
1276 text = re.sub("(?m)^HG:.*\n", "", text)
1277 os.chdir(olddir)
1277 os.chdir(olddir)
1278
1278
1279 if not text.strip():
1279 if not text.strip():
1280 raise util.Abort(_("empty commit message"))
1280 raise util.Abort(_("empty commit message"))
1281
1281
1282 return text
1282 return text
@@ -1,83 +1,96
1 #!/bin/sh
1 #!/bin/sh
2
2
3 "$TESTDIR/hghave" inotify || exit 80
3 "$TESTDIR/hghave" inotify || exit 80
4
4
5 hg init
5 hg init repo1
6 cd repo1
6
7
7 touch a b c d e
8 touch a b c d e
8 mkdir dir
9 mkdir dir
9 mkdir dir/bar
10 mkdir dir/bar
10 touch dir/x dir/y dir/bar/foo
11 touch dir/x dir/y dir/bar/foo
11
12
12 hg ci -Am m
13 hg ci -Am m
14 cd ..
15 hg clone repo1 repo2
13
16
14 echo "[extensions]" >> $HGRCPATH
17 echo "[extensions]" >> $HGRCPATH
15 echo "inotify=" >> $HGRCPATH
18 echo "inotify=" >> $HGRCPATH
16
19
20 cd repo2
21 echo b >> a
22 # check that daemon started automatically works correctly
23 # and make sure that inotify.pidfile works
24 hg --config "inotify.pidfile=../hg2.pid" status
25
26 # make sure that pidfile worked. Output should be silent.
27 kill `cat ../hg2.pid`
28
29 cd ../repo1
17 echo % inserve
30 echo % inserve
18 hg inserve -d --pid-file=hg.pid
31 hg inserve -d --pid-file=hg.pid
19 cat hg.pid >> "$DAEMON_PIDS"
32 cat hg.pid >> "$DAEMON_PIDS"
20
33
21 # let the daemon finish its stuff
34 # let the daemon finish its stuff
22 sleep 1
35 sleep 1
23 # issue907
36 # issue907
24 hg status
37 hg status
25 echo % clean
38 echo % clean
26 hg status -c
39 hg status -c
27 echo % all
40 echo % all
28 hg status -A
41 hg status -A
29
42
30 echo '% path patterns'
43 echo '% path patterns'
31 echo x > dir/x
44 echo x > dir/x
32 hg status .
45 hg status .
33 hg status dir
46 hg status dir
34 cd dir
47 cd dir
35 hg status .
48 hg status .
36 cd ..
49 cd ..
37
50
38 #issue 1375
51 #issue 1375
39 #Testing that we can remove a folder and then add a file with the same name
52 #Testing that we can remove a folder and then add a file with the same name
40 echo % issue 1375
53 echo % issue 1375
41
54
42 mkdir h
55 mkdir h
43 echo h > h/h
56 echo h > h/h
44 hg ci -Am t
57 hg ci -Am t
45 hg rm h
58 hg rm h
46
59
47 echo h >h
60 echo h >h
48 hg add h
61 hg add h
49
62
50 hg status
63 hg status
51 hg ci -m0
64 hg ci -m0
52
65
53 # Test for issue1735: inotify watches files in .hg/merge
66 # Test for issue1735: inotify watches files in .hg/merge
54 hg st
67 hg st
55
68
56 echo a > a
69 echo a > a
57
70
58 hg ci -Am a
71 hg ci -Am a
59 hg st
72 hg st
60
73
61 echo b >> a
74 echo b >> a
62 hg ci -m ab
75 hg ci -m ab
63 hg st
76 hg st
64
77
65 echo c >> a
78 echo c >> a
66 hg st
79 hg st
67
80
68 hg up 0
81 hg up 0
69 hg st
82 hg st
70
83
71 HGMERGE=internal:local hg up
84 HGMERGE=internal:local hg up
72 hg st
85 hg st
73
86
74 # Test for 1844: "hg ci folder" will not commit all changes beneath "folder"
87 # Test for 1844: "hg ci folder" will not commit all changes beneath "folder"
75 mkdir 1844
88 mkdir 1844
76 echo a > 1844/foo
89 echo a > 1844/foo
77 hg add 1844
90 hg add 1844
78 hg ci -m 'working'
91 hg ci -m 'working'
79
92
80 echo b >> 1844/foo
93 echo b >> 1844/foo
81 hg ci 1844 -m 'broken'
94 hg ci 1844 -m 'broken'
82
95
83 kill `cat hg.pid`
96 kill `cat hg.pid`
@@ -1,71 +1,71
1 #!/bin/sh
1 #!/bin/sh
2
2
3 # issues when status queries are issued when dirstate is dirty
3 # issues when status queries are issued when dirstate is dirty
4
4
5 "$TESTDIR/hghave" inotify || exit 80
5 "$TESTDIR/hghave" inotify || exit 80
6
6
7 echo "[extensions]" >> $HGRCPATH
7 echo "[extensions]" >> $HGRCPATH
8 echo "inotify=" >> $HGRCPATH
8 echo "inotify=" >> $HGRCPATH
9 echo "fetch=" >> $HGRCPATH
9 echo "fetch=" >> $HGRCPATH
10
10
11 echo % issue1810: inotify and fetch
11 echo % issue1810: inotify and fetch
12 mkdir test; cd test
12 mkdir test; cd test
13 hg init
13 hg init
14 hg inserve -d --pid-file=../hg.pid
14 hg inserve -d --pid-file=../hg.pid
15 cat ../hg.pid >> "$DAEMON_PIDS"
15 cat ../hg.pid >> "$DAEMON_PIDS"
16
16
17 echo foo > foo
17 echo foo > foo
18 hg add
18 hg add
19 hg ci -m foo
19 hg ci -m foo
20
20
21 cd ..
21 cd ..
22
22
23 hg --config "extensions.inotify=!" clone test test2
23 hg --config "inotify.pidfile=../hg2.pid" clone test test2
24 cat ../hg2.pid >> "$DAEMON_PIDS"
25
24 cd test2
26 cd test2
25 hg inserve -d --pid-file=../hg2.pid
26 cat ../hg2.pid >> "$DAEMON_PIDS"
27 echo bar > bar
27 echo bar > bar
28 hg add
28 hg add
29 hg ci -m bar
29 hg ci -m bar
30 cd ../test
30 cd ../test
31 echo spam > spam
31 echo spam > spam
32 hg add
32 hg add
33 hg ci -m spam
33 hg ci -m spam
34 cd ../test2
34 cd ../test2
35 hg st
35 hg st
36
36
37 # abort, outstanding changes
37 # abort, outstanding changes
38 hg fetch -q
38 hg fetch -q
39 hg st
39 hg st
40 cd ..
40 cd ..
41
41
42
42
43 echo % issue1719: inotify and mq
43 echo % issue1719: inotify and mq
44
44
45 echo "mq=" >> $HGRCPATH
45 echo "mq=" >> $HGRCPATH
46
46
47 hg init test-1719
47 hg init test-1719
48 cd test-1719
48 cd test-1719
49
49
50 echo % inserve
50 echo % inserve
51 hg inserve -d --pid-file=../hg-test-1719.pid
51 hg inserve -d --pid-file=../hg-test-1719.pid
52 cat ../hg-test-1719.pid >> "$DAEMON_PIDS"
52 cat ../hg-test-1719.pid >> "$DAEMON_PIDS"
53
53
54 echo content > file
54 echo content > file
55 hg add file
55 hg add file
56
56
57 hg qnew -f test.patch
57 hg qnew -f test.patch
58
58
59 hg status
59 hg status
60 hg qpop
60 hg qpop
61
61
62 echo % st should not output anything
62 echo % st should not output anything
63 hg status
63 hg status
64
64
65 hg qpush
65 hg qpush
66
66
67 echo % st should not output anything
67 echo % st should not output anything
68 hg status
68 hg status
69
69
70 hg qrefresh
70 hg qrefresh
71 hg status
71 hg status
@@ -1,6 +1,7
1 % fail
1 % fail
2 abort: could not start server: File exists
2 abort: could not start server: File exists
3 could not talk to new inotify server: No such file or directory
3 abort: could not start server: File exists
4 abort: could not start server: File exists
4 % inserve
5 % inserve
5 % status
6 % status
6 ? hg.pid
7 ? hg.pid
@@ -1,47 +1,50
1 adding a
1 adding a
2 adding b
2 adding b
3 adding c
3 adding c
4 adding d
4 adding d
5 adding dir/bar/foo
5 adding dir/bar/foo
6 adding dir/x
6 adding dir/x
7 adding dir/y
7 adding dir/y
8 adding e
8 adding e
9 updating to branch default
10 8 files updated, 0 files merged, 0 files removed, 0 files unresolved
11 M a
9 % inserve
12 % inserve
10 ? hg.pid
13 ? hg.pid
11 % clean
14 % clean
12 C a
15 C a
13 C b
16 C b
14 C c
17 C c
15 C d
18 C d
16 C dir/bar/foo
19 C dir/bar/foo
17 C dir/x
20 C dir/x
18 C dir/y
21 C dir/y
19 C e
22 C e
20 % all
23 % all
21 ? hg.pid
24 ? hg.pid
22 C a
25 C a
23 C b
26 C b
24 C c
27 C c
25 C d
28 C d
26 C dir/bar/foo
29 C dir/bar/foo
27 C dir/x
30 C dir/x
28 C dir/y
31 C dir/y
29 C e
32 C e
30 % path patterns
33 % path patterns
31 M dir/x
34 M dir/x
32 ? hg.pid
35 ? hg.pid
33 M dir/x
36 M dir/x
34 M x
37 M x
35 % issue 1375
38 % issue 1375
36 adding h/h
39 adding h/h
37 adding hg.pid
40 adding hg.pid
38 removing h/h
41 removing h/h
39 A h
42 A h
40 R h/h
43 R h/h
41 M a
44 M a
42 merging a
45 merging a
43 1 files updated, 1 files merged, 2 files removed, 0 files unresolved
46 1 files updated, 1 files merged, 2 files removed, 0 files unresolved
44 M a
47 M a
45 3 files updated, 1 files merged, 0 files removed, 0 files unresolved
48 3 files updated, 1 files merged, 0 files removed, 0 files unresolved
46 M a
49 M a
47 adding 1844/foo
50 adding 1844/foo
General Comments 0
You need to be logged in to leave comments. Login now