1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44 package org.eclipse.jgit.transport;
45
46 import java.io.IOException;
47 import java.io.InputStream;
48 import java.io.InterruptedIOException;
49 import java.io.OutputStream;
50 import java.net.InetAddress;
51 import java.net.InetSocketAddress;
52 import java.net.ServerSocket;
53 import java.net.Socket;
54 import java.net.SocketAddress;
55
56 import org.eclipse.jgit.errors.RepositoryNotFoundException;
57 import org.eclipse.jgit.internal.JGitText;
58 import org.eclipse.jgit.lib.PersonIdent;
59 import org.eclipse.jgit.lib.Repository;
60 import org.eclipse.jgit.storage.pack.PackConfig;
61 import org.eclipse.jgit.transport.resolver.ReceivePackFactory;
62 import org.eclipse.jgit.transport.resolver.RepositoryResolver;
63 import org.eclipse.jgit.transport.resolver.ServiceNotAuthorizedException;
64 import org.eclipse.jgit.transport.resolver.ServiceNotEnabledException;
65 import org.eclipse.jgit.transport.resolver.UploadPackFactory;
66
67
68
69
70 public static final int DEFAULT_PORT = 9418;
71
72 private static final int BACKLOG = 5;
73
74 private InetSocketAddress myAddress;
75
76 private final DaemonService[] services;
77
78 private final ThreadGroup processors;
79
80 private boolean run;
81
82 private Thread acceptThread;
83
84 private int timeout;
85
86 private PackConfig packConfig;
87
88 private volatile RepositoryResolver<DaemonClient> repositoryResolver;
89
90 private volatile UploadPackFactory<DaemonClient> uploadPackFactory;
91
92 private volatile ReceivePackFactory<DaemonClient> receivePackFactory;
93
94
95 public Daemon() {
96 this(null);
97 }
98
99
100
101
102
103
104
105
106 @SuppressWarnings("unchecked")
107 public Daemon(final InetSocketAddress addr) {
108 myAddress = addr;
109 processors = new ThreadGroup("Git-Daemon");
110
111 repositoryResolver = (RepositoryResolver<DaemonClient>) RepositoryResolver.NONE;
112
113 uploadPackFactory = new UploadPackFactory<DaemonClient>() {
114 public UploadPack create(DaemonClient req, Repository db)
115 throws ServiceNotEnabledException,
116 ServiceNotAuthorizedException {
117 UploadPack up = new UploadPack(db);
118 up.setTimeout(getTimeout());
119 up.setPackConfig(getPackConfig());
120 return up;
121 }
122 };
123
124 receivePackFactory = new ReceivePackFactory<DaemonClient>() {
125 public ReceivePack create(DaemonClient req, Repository db)
126 throws ServiceNotEnabledException,
127 ServiceNotAuthorizedException {
128 ReceivePack rp = new ReceivePack(db);
129
130 InetAddress peer = req.getRemoteAddress();
131 String host = peer.getCanonicalHostName();
132 if (host == null)
133 host = peer.getHostAddress();
134 String name = "anonymous";
135 String email = name + "@" + host;
136 rp.setRefLogIdent(new PersonIdent(name, email));
137 rp.setTimeout(getTimeout());
138
139 return rp;
140 }
141 };
142
143 services = new DaemonService[] {
144 new DaemonService("upload-pack", "uploadpack") {
145 {
146 setEnabled(true);
147 }
148
149 @Override
150 protected void execute(final DaemonClient dc,
151 final Repository db) throws IOException,
152 ServiceNotEnabledException,
153 ServiceNotAuthorizedException {
154 UploadPack up = uploadPackFactory.create(dc, db);
155 InputStream in = dc.getInputStream();
156 OutputStream out = dc.getOutputStream();
157 up.upload(in, out, null);
158 }
159 }, new DaemonService("receive-pack", "receivepack") {
160 {
161 setEnabled(false);
162 }
163
164 @Override
165 protected void execute(final DaemonClient dc,
166 final Repository db) throws IOException,
167 ServiceNotEnabledException,
168 ServiceNotAuthorizedException {
169 ReceivePack rp = receivePackFactory.create(dc, db);
170 InputStream in = dc.getInputStream();
171 OutputStream out = dc.getOutputStream();
172 rp.receive(in, out, null);
173 }
174 } };
175 }
176
177
178 public synchronized InetSocketAddress getAddress() {
179 return myAddress;
180 }
181
182
183
184
185
186
187
188
189
190
191 public synchronized DaemonService getService(String name) {
192 if (!name.startsWith("git-"))
193 name = "git-" + name;
194 for (final DaemonService s : services) {
195 if (s.getCommandName().equals(name))
196 return s;
197 }
198 return null;
199 }
200
201
202 public int getTimeout() {
203 return timeout;
204 }
205
206
207
208
209
210
211
212
213
214 public void setTimeout(final int seconds) {
215 timeout = seconds;
216 }
217
218
219 public PackConfig getPackConfig() {
220 return packConfig;
221 }
222
223
224
225
226
227
228
229
230 public void setPackConfig(PackConfig pc) {
231 this.packConfig = pc;
232 }
233
234
235
236
237
238
239
240 public void setRepositoryResolver(RepositoryResolver<DaemonClient> resolver) {
241 repositoryResolver = resolver;
242 }
243
244
245
246
247
248
249
250 @SuppressWarnings("unchecked")
251 public void setUploadPackFactory(UploadPackFactory<DaemonClient> factory) {
252 if (factory != null)
253 uploadPackFactory = factory;
254 else
255 uploadPackFactory = (UploadPackFactory<DaemonClient>) UploadPackFactory.DISABLED;
256 }
257
258
259
260
261
262
263
264 @SuppressWarnings("unchecked")
265 public void setReceivePackFactory(ReceivePackFactory<DaemonClient> factory) {
266 if (factory != null)
267 receivePackFactory = factory;
268 else
269 receivePackFactory = (ReceivePackFactory<DaemonClient>) ReceivePackFactory.DISABLED;
270 }
271
272
273
274
275
276
277
278
279
280 public synchronized void start() throws IOException {
281 if (acceptThread != null)
282 throw new IllegalStateException(JGitText.get().daemonAlreadyRunning);
283
284 final ServerSocket listenSock = new ServerSocket(
285 myAddress != null ? myAddress.getPort() : 0, BACKLOG,
286 myAddress != null ? myAddress.getAddress() : null);
287 myAddress = (InetSocketAddress) listenSock.getLocalSocketAddress();
288
289 run = true;
290 acceptThread = new Thread(processors, "Git-Daemon-Accept") {
291 public void run() {
292 while (isRunning()) {
293 try {
294 startClient(listenSock.accept());
295 } catch (InterruptedIOException e) {
296
297 } catch (IOException e) {
298 break;
299 }
300 }
301
302 try {
303 listenSock.close();
304 } catch (IOException err) {
305
306 } finally {
307 synchronized (Daemon.this) {
308 acceptThread = null;
309 }
310 }
311 }
312 };
313 acceptThread.start();
314 }
315
316
317 public synchronized boolean isRunning() {
318 return run;
319 }
320
321
322 public synchronized void stop() {
323 if (acceptThread != null) {
324 run = false;
325 acceptThread.interrupt();
326 }
327 }
328
329 private void startClient(final Socket s) {
330 final DaemonClient dc = new DaemonClient(this);
331
332 final SocketAddress peer = s.getRemoteSocketAddress();
333 if (peer instanceof InetSocketAddress)
334 dc.setRemoteAddress(((InetSocketAddress) peer).getAddress());
335
336 new Thread(processors, "Git-Daemon-Client " + peer.toString()) {
337 public void run() {
338 try {
339 dc.execute(s);
340 } catch (ServiceNotEnabledException e) {
341
342 } catch (ServiceNotAuthorizedException e) {
343
344 } catch (IOException e) {
345
346 } finally {
347 try {
348 s.getInputStream().close();
349 } catch (IOException e) {
350
351 }
352 try {
353 s.getOutputStream().close();
354 } catch (IOException e) {
355
356 }
357 }
358 }
359 }.start();
360 }
361
362 synchronized DaemonService matchService(final String cmd) {
363 for (final DaemonService d : services) {
364 if (d.handles(cmd))
365 return d;
366 }
367 return null;
368 }
369
370 Repository openRepository(DaemonClient client, String name)
371 throws ServiceMayNotContinueException {
372
373
374
375 name = name.replace('\\', '/');
376
377
378
379 if (!name.startsWith("/"))
380 return null;
381
382 try {
383 return repositoryResolver.open(client, name.substring(1));
384 } catch (RepositoryNotFoundException e) {
385
386
387 return null;
388 } catch (ServiceNotAuthorizedException e) {
389
390
391 return null;
392 } catch (ServiceNotEnabledException e) {
393
394
395 return null;
396 }
397 }
398 }