View Javadoc
1   /*
2    * Copyright (C) 2006, Robin Rosenberg <robin.rosenberg@dewire.com>
3    * Copyright (C) 2008, Shawn O. Pearce <spearce@spearce.org>
4    * and other copyright owners as documented in the project's IP log.
5    *
6    * This program and the accompanying materials are made available
7    * under the terms of the Eclipse Distribution License v1.0 which
8    * accompanies this distribution, is reproduced below, and is
9    * available at http://www.eclipse.org/org/documents/edl-v10.php
10   *
11   * All rights reserved.
12   *
13   * Redistribution and use in source and binary forms, with or
14   * without modification, are permitted provided that the following
15   * conditions are met:
16   *
17   * - Redistributions of source code must retain the above copyright
18   *   notice, this list of conditions and the following disclaimer.
19   *
20   * - Redistributions in binary form must reproduce the above
21   *   copyright notice, this list of conditions and the following
22   *   disclaimer in the documentation and/or other materials provided
23   *   with the distribution.
24   *
25   * - Neither the name of the Eclipse Foundation, Inc. nor the
26   *   names of its contributors may be used to endorse or promote
27   *   products derived from this software without specific prior
28   *   written permission.
29   *
30   * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
31   * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
32   * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
33   * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
34   * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
35   * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
36   * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
37   * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
38   * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
39   * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
40   * STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
41   * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
42   * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
43   */
44  
45  package org.eclipse.jgit.pgm;
46  
47  import java.io.File;
48  import java.io.IOException;
49  import java.io.PrintWriter;
50  import java.lang.reflect.InvocationTargetException;
51  import java.net.MalformedURLException;
52  import java.net.URL;
53  import java.text.MessageFormat;
54  import java.util.ArrayList;
55  import java.util.List;
56  
57  import org.eclipse.jgit.awtui.AwtAuthenticator;
58  import org.eclipse.jgit.awtui.AwtCredentialsProvider;
59  import org.eclipse.jgit.errors.TransportException;
60  import org.eclipse.jgit.lib.Repository;
61  import org.eclipse.jgit.lib.RepositoryBuilder;
62  import org.eclipse.jgit.pgm.internal.CLIText;
63  import org.eclipse.jgit.pgm.opt.CmdLineParser;
64  import org.eclipse.jgit.pgm.opt.SubcommandHandler;
65  import org.eclipse.jgit.util.CachedAuthenticator;
66  import org.kohsuke.args4j.Argument;
67  import org.kohsuke.args4j.CmdLineException;
68  import org.kohsuke.args4j.ExampleMode;
69  import org.kohsuke.args4j.Option;
70  
71  /** Command line entry point. */
72  public class Main {
73  	@Option(name = "--help", usage = "usage_displayThisHelpText", aliases = { "-h" })
74  	private boolean help;
75  
76  	@Option(name = "--version", usage = "usage_displayVersion")
77  	private boolean version;
78  
79  	@Option(name = "--show-stack-trace", usage = "usage_displayThejavaStackTraceOnExceptions")
80  	private boolean showStackTrace;
81  
82  	@Option(name = "--git-dir", metaVar = "metaVar_gitDir", usage = "usage_setTheGitRepositoryToOperateOn")
83  	private String gitdir;
84  
85  	@Argument(index = 0, metaVar = "metaVar_command", required = true, handler = SubcommandHandler.class)
86  	private TextBuiltin subcommand;
87  
88  	@Argument(index = 1, metaVar = "metaVar_arg")
89  	private List<String> arguments = new ArrayList<String>();
90  
91  	/**
92  	 * Execute the command line.
93  	 *
94  	 * @param argv
95  	 *            arguments.
96  	 */
97  	public static void main(final String[] argv) {
98  		new Main().run(argv);
99  	}
100 
101 	/**
102 	 * Parse the command line and execute the requested action.
103 	 *
104 	 * Subclasses should allocate themselves and then invoke this method:
105 	 *
106 	 * <pre>
107 	 * class ExtMain {
108 	 * 	public static void main(String[] argv) {
109 	 * 		new ExtMain().run(argv);
110 	 * 	}
111 	 * }
112 	 * </pre>
113 	 *
114 	 * @param argv
115 	 *            arguments.
116 	 */
117 	protected void run(final String[] argv) {
118 		try {
119 			if (!installConsole()) {
120 				AwtAuthenticator.install();
121 				AwtCredentialsProvider.install();
122 			}
123 			configureHttpProxy();
124 			execute(argv);
125 		} catch (Die err) {
126 			if (err.isAborted())
127 				System.exit(1);
128 			System.err.println(MessageFormat.format(CLIText.get().fatalError, err.getMessage()));
129 			if (showStackTrace)
130 				err.printStackTrace();
131 			System.exit(128);
132 		} catch (Exception err) {
133 			// Try to detect errno == EPIPE and exit normally if that happens
134 			// There may be issues with operating system versions and locale,
135 			// but we can probably assume that these messages will not be thrown
136 			// under other circumstances.
137 			if (err.getClass() == IOException.class) {
138 				// Linux, OS X
139 				if (err.getMessage().equals("Broken pipe")) //$NON-NLS-1$
140 					System.exit(0);
141 				// Windows
142 				if (err.getMessage().equals("The pipe is being closed")) //$NON-NLS-1$
143 					System.exit(0);
144 			}
145 			if (!showStackTrace && err.getCause() != null
146 					&& err instanceof TransportException)
147 				System.err.println(MessageFormat.format(CLIText.get().fatalError, err.getCause().getMessage()));
148 
149 			if (err.getClass().getName().startsWith("org.eclipse.jgit.errors.")) { //$NON-NLS-1$
150 				System.err.println(MessageFormat.format(CLIText.get().fatalError, err.getMessage()));
151 				if (showStackTrace)
152 					err.printStackTrace();
153 				System.exit(128);
154 			}
155 			err.printStackTrace();
156 			System.exit(1);
157 		}
158 		if (System.out.checkError()) {
159 			System.err.println(CLIText.get().unknownIoErrorStdout);
160 			System.exit(1);
161 		}
162 		if (System.err.checkError()) {
163 			// No idea how to present an error here, most likely disk full or
164 			// broken pipe
165 			System.exit(1);
166 		}
167 	}
168 
169 	private void execute(final String[] argv) throws Exception {
170 		final CmdLineParser clp = new CmdLineParser(this);
171 		PrintWriter writer = new PrintWriter(System.err);
172 		try {
173 			clp.parseArgument(argv);
174 		} catch (CmdLineException err) {
175 			if (argv.length > 0 && !help && !version) {
176 				writer.println(MessageFormat.format(CLIText.get().fatalError, err.getMessage()));
177 				writer.flush();
178 				System.exit(1);
179 			}
180 		}
181 
182 		if (argv.length == 0 || help) {
183 			final String ex = clp.printExample(ExampleMode.ALL, CLIText.get().resourceBundle());
184 			writer.println("jgit" + ex + " command [ARG ...]"); //$NON-NLS-1$ //$NON-NLS-2$
185 			if (help) {
186 				writer.println();
187 				clp.printUsage(writer, CLIText.get().resourceBundle());
188 				writer.println();
189 			} else if (subcommand == null) {
190 				writer.println();
191 				writer.println(CLIText.get().mostCommonlyUsedCommandsAre);
192 				final CommandRef[] common = CommandCatalog.common();
193 				int width = 0;
194 				for (final CommandRef c : common)
195 					width = Math.max(width, c.getName().length());
196 				width += 2;
197 
198 				for (final CommandRef c : common) {
199 					writer.print(' ');
200 					writer.print(c.getName());
201 					for (int i = c.getName().length(); i < width; i++)
202 						writer.print(' ');
203 					writer.print(CLIText.get().resourceBundle().getString(c.getUsage()));
204 					writer.println();
205 				}
206 				writer.println();
207 			}
208 			writer.flush();
209 			System.exit(1);
210 		}
211 
212 		if (version) {
213 			String cmdId = Version.class.getSimpleName().toLowerCase();
214 			subcommand = CommandCatalog.get(cmdId).create();
215 		}
216 
217 		final TextBuiltin cmd = subcommand;
218 		if (cmd.requiresRepository())
219 			cmd.init(openGitDir(gitdir), null);
220 		else
221 			cmd.init(null, gitdir);
222 		try {
223 			cmd.execute(arguments.toArray(new String[arguments.size()]));
224 		} finally {
225 			if (cmd.outw != null)
226 				cmd.outw.flush();
227 			if (cmd.errw != null)
228 				cmd.errw.flush();
229 		}
230 	}
231 
232 	/**
233 	 * Evaluate the {@code --git-dir} option and open the repository.
234 	 *
235 	 * @param aGitdir
236 	 *            the {@code --git-dir} option given on the command line. May be
237 	 *            null if it was not supplied.
238 	 * @return the repository to operate on.
239 	 * @throws IOException
240 	 *             the repository cannot be opened.
241 	 */
242 	protected Repository openGitDir(String aGitdir) throws IOException {
243 		RepositoryBuilder rb = new RepositoryBuilder() //
244 				.setGitDir(aGitdir != null ? new File(aGitdir) : null) //
245 				.readEnvironment() //
246 				.findGitDir();
247 		if (rb.getGitDir() == null)
248 			throw new Die(CLIText.get().cantFindGitDirectory);
249 		return rb.build();
250 	}
251 
252 	private static boolean installConsole() {
253 		try {
254 			install("org.eclipse.jgit.console.ConsoleAuthenticator"); //$NON-NLS-1$
255 			install("org.eclipse.jgit.console.ConsoleCredentialsProvider"); //$NON-NLS-1$
256 			return true;
257 		} catch (ClassNotFoundException e) {
258 			return false;
259 		} catch (NoClassDefFoundError e) {
260 			return false;
261 		} catch (UnsupportedClassVersionError e) {
262 			return false;
263 
264 		} catch (IllegalArgumentException e) {
265 			throw new RuntimeException(CLIText.get().cannotSetupConsole, e);
266 		} catch (SecurityException e) {
267 			throw new RuntimeException(CLIText.get().cannotSetupConsole, e);
268 		} catch (IllegalAccessException e) {
269 			throw new RuntimeException(CLIText.get().cannotSetupConsole, e);
270 		} catch (InvocationTargetException e) {
271 			throw new RuntimeException(CLIText.get().cannotSetupConsole, e);
272 		} catch (NoSuchMethodException e) {
273 			throw new RuntimeException(CLIText.get().cannotSetupConsole, e);
274 		}
275 	}
276 
277 	private static void install(final String name)
278 			throws IllegalAccessException, InvocationTargetException,
279 			NoSuchMethodException, ClassNotFoundException {
280 		try {
281 		Class.forName(name).getMethod("install").invoke(null); //$NON-NLS-1$
282 		} catch (InvocationTargetException e) {
283 			if (e.getCause() instanceof RuntimeException)
284 				throw (RuntimeException) e.getCause();
285 			if (e.getCause() instanceof Error)
286 				throw (Error) e.getCause();
287 			throw e;
288 		}
289 	}
290 
291 	/**
292 	 * Configure the JRE's standard HTTP based on <code>http_proxy</code>.
293 	 * <p>
294 	 * The popular libcurl library honors the <code>http_proxy</code>,
295 	 * <code>https_proxy</code> environment variables as a means of specifying
296 	 * an HTTP/S proxy for requests made behind a firewall. This is not natively
297 	 * recognized by the JRE, so this method can be used by command line
298 	 * utilities to configure the JRE before the first request is sent.
299 	 *
300 	 * @throws MalformedURLException
301 	 *             the value in <code>http_proxy</code> or
302 	 *             <code>https_proxy</code> is unsupportable.
303 	 */
304 	private static void configureHttpProxy() throws MalformedURLException {
305 		for (String protocol : new String[] { "http", "https" }) { //$NON-NLS-1$ //$NON-NLS-2$
306 			final String s = System.getenv(protocol + "_proxy"); //$NON-NLS-1$
307 			if (s == null || s.equals("")) //$NON-NLS-1$
308 				return;
309 
310 			final URL u = new URL(
311 					(s.indexOf("://") == -1) ? protocol + "://" + s : s); //$NON-NLS-1$ //$NON-NLS-2$
312 			if (!u.getProtocol().startsWith("http")) //$NON-NLS-1$
313 				throw new MalformedURLException(MessageFormat.format(
314 						CLIText.get().invalidHttpProxyOnlyHttpSupported, s));
315 
316 			final String proxyHost = u.getHost();
317 			final int proxyPort = u.getPort();
318 
319 			System.setProperty(protocol + ".proxyHost", proxyHost); //$NON-NLS-1$
320 			if (proxyPort > 0)
321 				System.setProperty(protocol + ".proxyPort", //$NON-NLS-1$
322 						String.valueOf(proxyPort));
323 
324 			final String userpass = u.getUserInfo();
325 			if (userpass != null && userpass.contains(":")) { //$NON-NLS-1$
326 				final int c = userpass.indexOf(':');
327 				final String user = userpass.substring(0, c);
328 				final String pass = userpass.substring(c + 1);
329 				CachedAuthenticator.add(
330 						new CachedAuthenticator.CachedAuthentication(proxyHost,
331 								proxyPort, user, pass));
332 			}
333 		}
334 	}
335 }