Mercurial > hg4j
comparison src/org/tmatesoft/hg/core/HgLogCommand.java @ 131:aa1629f36482
Renamed .core classes to start with Hg prefix
author | Artem Tikhomirov <tikhomirov.artem@gmail.com> |
---|---|
date | Wed, 16 Feb 2011 20:47:56 +0100 |
parents | src/org/tmatesoft/hg/core/LogCommand.java@645829962785 |
children | 4a948ec83980 |
comparison
equal
deleted
inserted
replaced
130:7567f4a42fe5 | 131:aa1629f36482 |
---|---|
1 /* | |
2 * Copyright (c) 2011 TMate Software Ltd | |
3 * | |
4 * This program is free software; you can redistribute it and/or modify | |
5 * it under the terms of the GNU General Public License as published by | |
6 * the Free Software Foundation; version 2 of the License. | |
7 * | |
8 * This program is distributed in the hope that it will be useful, | |
9 * but WITHOUT ANY WARRANTY; without even the implied warranty of | |
10 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
11 * GNU General Public License for more details. | |
12 * | |
13 * For information on how to redistribute this software under | |
14 * the terms of a license other than GNU General Public License | |
15 * contact TMate Software at support@hg4j.com | |
16 */ | |
17 package org.tmatesoft.hg.core; | |
18 | |
19 import static org.tmatesoft.hg.repo.HgRepository.TIP; | |
20 | |
21 import java.util.Calendar; | |
22 import java.util.Collections; | |
23 import java.util.ConcurrentModificationException; | |
24 import java.util.LinkedList; | |
25 import java.util.List; | |
26 import java.util.Set; | |
27 import java.util.TreeSet; | |
28 | |
29 import org.tmatesoft.hg.repo.HgChangelog.Changeset; | |
30 import org.tmatesoft.hg.repo.HgChangelog; | |
31 import org.tmatesoft.hg.repo.HgDataFile; | |
32 import org.tmatesoft.hg.repo.HgRepository; | |
33 import org.tmatesoft.hg.repo.HgStatusCollector; | |
34 import org.tmatesoft.hg.util.PathPool; | |
35 | |
36 | |
37 /** | |
38 * Access to changelog, 'hg log' command counterpart. | |
39 * | |
40 * <pre> | |
41 * Usage: | |
42 * new LogCommand().limit(20).branch("maintenance-2.1").user("me").execute(new MyHandler()); | |
43 * </pre> | |
44 * Not thread-safe (each thread has to use own {@link HgLogCommand} instance). | |
45 * | |
46 * @author Artem Tikhomirov | |
47 * @author TMate Software Ltd. | |
48 */ | |
49 public class HgLogCommand implements HgChangelog.Inspector { | |
50 | |
51 private final HgRepository repo; | |
52 private Set<String> users; | |
53 private Set<String> branches; | |
54 private int limit = 0, count = 0; | |
55 private int startRev = 0, endRev = TIP; | |
56 private Handler delegate; | |
57 private Calendar date; | |
58 private Path file; | |
59 private boolean followHistory; // makes sense only when file != null | |
60 private HgChangeset changeset; | |
61 | |
62 public HgLogCommand(HgRepository hgRepo) { | |
63 repo = hgRepo; | |
64 } | |
65 | |
66 /** | |
67 * Limit search to specified user. Multiple user names may be specified. | |
68 * @param user - full or partial name of the user, case-insensitive, non-null. | |
69 * @return <code>this</code> instance for convenience | |
70 */ | |
71 public HgLogCommand user(String user) { | |
72 if (user == null) { | |
73 throw new IllegalArgumentException(); | |
74 } | |
75 if (users == null) { | |
76 users = new TreeSet<String>(); | |
77 } | |
78 users.add(user.toLowerCase()); | |
79 return this; | |
80 } | |
81 | |
82 /** | |
83 * Limit search to specified branch. Multiple branch specification possible (changeset from any of these | |
84 * would be included in result). If unspecified, all branches are considered. | |
85 * @param branch - branch name, case-sensitive, non-null. | |
86 * @return <code>this</code> instance for convenience | |
87 */ | |
88 public HgLogCommand branch(String branch) { | |
89 if (branch == null) { | |
90 throw new IllegalArgumentException(); | |
91 } | |
92 if (branches == null) { | |
93 branches = new TreeSet<String>(); | |
94 } | |
95 branches.add(branch); | |
96 return this; | |
97 } | |
98 | |
99 // limit search to specific date | |
100 // multiple? | |
101 public HgLogCommand date(Calendar date) { | |
102 this.date = date; | |
103 // FIXME implement | |
104 // isSet(field) - false => don't use in detection of 'same date' | |
105 throw HgRepository.notImplemented(); | |
106 } | |
107 | |
108 /** | |
109 * | |
110 * @param num - number of changeset to produce. Pass 0 to clear the limit. | |
111 * @return <code>this</code> instance for convenience | |
112 */ | |
113 public HgLogCommand limit(int num) { | |
114 limit = num; | |
115 return this; | |
116 } | |
117 | |
118 /** | |
119 * Limit to specified subset of Changelog, [min(rev1,rev2), max(rev1,rev2)], inclusive. | |
120 * Revision may be specified with {@link HgRepository#TIP} | |
121 * @param rev1 | |
122 * @param rev2 | |
123 * @return <code>this</code> instance for convenience | |
124 */ | |
125 public HgLogCommand range(int rev1, int rev2) { | |
126 if (rev1 != TIP && rev2 != TIP) { | |
127 startRev = rev2 < rev1 ? rev2 : rev1; | |
128 endRev = startRev == rev2 ? rev1 : rev2; | |
129 } else if (rev1 == TIP && rev2 != TIP) { | |
130 startRev = rev2; | |
131 endRev = rev1; | |
132 } else { | |
133 startRev = rev1; | |
134 endRev = rev2; | |
135 } | |
136 return this; | |
137 } | |
138 | |
139 /** | |
140 * Visit history of a given file only. | |
141 * @param file path relative to repository root. Pass <code>null</code> to reset. | |
142 * @param followCopyRename true to report changesets of the original file(-s), if copy/rename ever occured to the file. | |
143 */ | |
144 public HgLogCommand file(Path file, boolean followCopyRename) { | |
145 // multiple? Bad idea, would need to include extra method into Handler to tell start of next file | |
146 this.file = file; | |
147 followHistory = followCopyRename; | |
148 return this; | |
149 } | |
150 | |
151 /** | |
152 * Similar to {@link #execute(org.tmatesoft.hg.repo.Changeset.Inspector)}, collects and return result as a list. | |
153 */ | |
154 public List<HgChangeset> execute() { | |
155 CollectHandler collector = new CollectHandler(); | |
156 execute(collector); | |
157 return collector.getChanges(); | |
158 } | |
159 | |
160 /** | |
161 * | |
162 * @param inspector | |
163 * @throws IllegalArgumentException when inspector argument is null | |
164 * @throws ConcurrentModificationException if this log command instance is already running | |
165 */ | |
166 public void execute(Handler handler) { | |
167 if (handler == null) { | |
168 throw new IllegalArgumentException(); | |
169 } | |
170 if (delegate != null) { | |
171 throw new ConcurrentModificationException(); | |
172 } | |
173 try { | |
174 delegate = handler; | |
175 count = 0; | |
176 changeset = new HgChangeset(new HgStatusCollector(repo), new PathPool(repo.getPathHelper())); | |
177 if (file == null) { | |
178 repo.getChangelog().range(startRev, endRev, this); | |
179 } else { | |
180 HgDataFile fileNode = repo.getFileNode(file); | |
181 fileNode.history(startRev, endRev, this); | |
182 if (fileNode.isCopy()) { | |
183 // even if we do not follow history, report file rename | |
184 do { | |
185 if (handler instanceof FileHistoryHandler) { | |
186 FileRevision src = new FileRevision(repo, fileNode.getCopySourceRevision(), fileNode.getCopySourceName()); | |
187 FileRevision dst = new FileRevision(repo, fileNode.getRevision(0), fileNode.getPath()); | |
188 ((FileHistoryHandler) handler).copy(src, dst); | |
189 } | |
190 if (limit > 0 && count >= limit) { | |
191 // if limit reach, follow is useless. | |
192 break; | |
193 } | |
194 if (followHistory) { | |
195 fileNode = repo.getFileNode(fileNode.getCopySourceName()); | |
196 fileNode.history(this); | |
197 } | |
198 } while (followHistory && fileNode.isCopy()); | |
199 } | |
200 } | |
201 } finally { | |
202 delegate = null; | |
203 changeset = null; | |
204 } | |
205 } | |
206 | |
207 // | |
208 | |
209 public void next(int revisionNumber, Nodeid nodeid, Changeset cset) { | |
210 if (limit > 0 && count >= limit) { | |
211 return; | |
212 } | |
213 if (branches != null && !branches.contains(cset.branch())) { | |
214 return; | |
215 } | |
216 if (users != null) { | |
217 String csetUser = cset.user().toLowerCase(); | |
218 boolean found = false; | |
219 for (String u : users) { | |
220 if (csetUser.indexOf(u) != -1) { | |
221 found = true; | |
222 break; | |
223 } | |
224 } | |
225 if (!found) { | |
226 return; | |
227 } | |
228 } | |
229 if (date != null) { | |
230 // FIXME | |
231 } | |
232 count++; | |
233 changeset.init(revisionNumber, nodeid, cset); | |
234 delegate.next(changeset); | |
235 } | |
236 | |
237 public interface Handler { | |
238 /** | |
239 * @param changeset not necessarily a distinct instance each time, {@link HgChangeset#clone() clone()} if need a copy. | |
240 */ | |
241 void next(HgChangeset changeset); | |
242 } | |
243 | |
244 /** | |
245 * When {@link HgLogCommand} is executed against file, handler passed to {@link HgLogCommand#execute(Handler)} may optionally | |
246 * implement this interface to get information about file renames. Method {@link #copy(FileRevision, FileRevision)} would | |
247 * get invoked prior any changeset of the original file (if file history being followed) is reported via {@link #next(HgChangeset)}. | |
248 * | |
249 * For {@link HgLogCommand#file(Path, boolean)} with renamed file path and follow argument set to false, | |
250 * {@link #copy(FileRevision, FileRevision)} would be invoked for the first copy/rename in the history of the file, but not | |
251 * followed by any changesets. | |
252 * | |
253 * @author Artem Tikhomirov | |
254 * @author TMate Software Ltd. | |
255 */ | |
256 public interface FileHistoryHandler extends Handler { | |
257 // XXX perhaps, should distinguish copy from rename? And what about merged revisions and following them? | |
258 void copy(FileRevision from, FileRevision to); | |
259 } | |
260 | |
261 public static class CollectHandler implements Handler { | |
262 private final List<HgChangeset> result = new LinkedList<HgChangeset>(); | |
263 | |
264 public List<HgChangeset> getChanges() { | |
265 return Collections.unmodifiableList(result); | |
266 } | |
267 | |
268 public void next(HgChangeset changeset) { | |
269 result.add(changeset.clone()); | |
270 } | |
271 } | |
272 | |
273 public static final class FileRevision { | |
274 private final HgRepository repo; | |
275 private final Nodeid revision; | |
276 private final Path path; | |
277 | |
278 /*package-local*/FileRevision(HgRepository hgRepo, Nodeid rev, Path p) { | |
279 if (hgRepo == null || rev == null || p == null) { | |
280 throw new IllegalArgumentException(); | |
281 } | |
282 repo = hgRepo; | |
283 revision = rev; | |
284 path = p; | |
285 } | |
286 | |
287 public Path getPath() { | |
288 return path; | |
289 } | |
290 public Nodeid getRevision() { | |
291 return revision; | |
292 } | |
293 public byte[] getContent() { | |
294 // XXX Content wrapper, to allow formats other than byte[], e.g. Stream, DataAccess, etc? | |
295 return repo.getFileNode(path).content(revision); | |
296 } | |
297 } | |
298 } |