[email protected] | ddfb164 | 2012-10-09 00:56:57 +0000 | [diff] [blame] | 1 | // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
| 2 | // for details. All rights reserved. Use of this source code is governed by a |
| 3 | // BSD-style license that can be found in the LICENSE file. |
| 4 | |
[email protected] | cefd1f2 | 2012-12-20 02:04:28 +0000 | [diff] [blame^] | 5 | /// Helper functionality for invoking Git. |
[email protected] | 05c9cc0 | 2012-10-09 01:25:43 +0000 | [diff] [blame] | 6 | library git; |
[email protected] | ddfb164 | 2012-10-09 00:56:57 +0000 | [diff] [blame] | 7 | |
[email protected] | 05c9cc0 | 2012-10-09 01:25:43 +0000 | [diff] [blame] | 8 | import 'io.dart'; |
[email protected] | 71976ef | 2012-12-08 02:57:00 +0000 | [diff] [blame] | 9 | import 'log.dart' as log; |
[email protected] | 05c9cc0 | 2012-10-09 01:25:43 +0000 | [diff] [blame] | 10 | import 'utils.dart'; |
[email protected] | ddfb164 | 2012-10-09 00:56:57 +0000 | [diff] [blame] | 11 | |
| 12 | /// Tests whether or not the git command-line app is available for use. |
| 13 | Future<bool> get isInstalled { |
| 14 | if (_isGitInstalledCache != null) { |
| 15 | // TODO(rnystrom): The sleep is to pump the message queue. Can use |
| 16 | // Future.immediate() when #3356 is fixed. |
| 17 | return sleep(0).transform((_) => _isGitInstalledCache); |
| 18 | } |
| 19 | |
| 20 | return _gitCommand.transform((git) => git != null); |
| 21 | } |
| 22 | |
| 23 | /// Run a git process with [args] from [workingDir]. Returns the stdout as a |
| 24 | /// list of strings if it succeeded. Completes to an exception if it failed. |
[email protected] | bc7ccb8 | 2012-10-17 20:58:19 +0000 | [diff] [blame] | 25 | Future<List<String>> run(List<String> args, {String workingDir}) { |
[email protected] | ddfb164 | 2012-10-09 00:56:57 +0000 | [diff] [blame] | 26 | return _gitCommand.chain((git) { |
[email protected] | b119eeb | 2012-10-16 16:03:47 +0000 | [diff] [blame] | 27 | return runProcess(git, args, workingDir: workingDir); |
[email protected] | ddfb164 | 2012-10-09 00:56:57 +0000 | [diff] [blame] | 28 | }).transform((result) { |
| 29 | if (!result.success) throw new Exception( |
[email protected] | 30a360f | 2012-10-09 20:54:05 +0000 | [diff] [blame] | 30 | 'Git error. Command: git ${Strings.join(args, " ")}\n' |
| 31 | '${Strings.join(result.stderr, "\n")}'); |
[email protected] | ddfb164 | 2012-10-09 00:56:57 +0000 | [diff] [blame] | 32 | |
| 33 | return result.stdout; |
| 34 | }); |
| 35 | } |
| 36 | |
| 37 | bool _isGitInstalledCache; |
| 38 | |
| 39 | /// The cached Git command. |
| 40 | String _gitCommandCache; |
| 41 | |
| 42 | /// Returns the name of the git command-line app, or null if Git could not be |
| 43 | /// found on the user's PATH. |
| 44 | Future<String> get _gitCommand { |
| 45 | // TODO(nweiz): Just use Future.immediate once issue 3356 is fixed. |
| 46 | if (_gitCommandCache != null) { |
| 47 | return sleep(0).transform((_) => _gitCommandCache); |
| 48 | } |
| 49 | |
| 50 | return _tryGitCommand("git").chain((success) { |
| 51 | if (success) return new Future.immediate("git"); |
| 52 | |
| 53 | // Git is sometimes installed on Windows as `git.cmd` |
| 54 | return _tryGitCommand("git.cmd").transform((success) { |
| 55 | if (success) return "git.cmd"; |
| 56 | return null; |
| 57 | }); |
| 58 | }).transform((command) { |
[email protected] | 71976ef | 2012-12-08 02:57:00 +0000 | [diff] [blame] | 59 | log.fine('Determined git command $command.'); |
[email protected] | ddfb164 | 2012-10-09 00:56:57 +0000 | [diff] [blame] | 60 | _gitCommandCache = command; |
| 61 | return command; |
| 62 | }); |
| 63 | } |
| 64 | |
| 65 | /// Checks whether [command] is the Git command for this computer. |
| 66 | Future<bool> _tryGitCommand(String command) { |
| 67 | var completer = new Completer<bool>(); |
| 68 | |
| 69 | // If "git --version" prints something familiar, git is working. |
| 70 | var future = runProcess(command, ["--version"]); |
| 71 | |
| 72 | future.then((results) { |
| 73 | var regex = new RegExp("^git version"); |
| 74 | completer.complete(results.stdout.length == 1 && |
| 75 | regex.hasMatch(results.stdout[0])); |
| 76 | }); |
| 77 | |
| 78 | future.handleException((err) { |
| 79 | // If the process failed, they probably don't have it. |
| 80 | completer.complete(false); |
| 81 | return true; |
| 82 | }); |
| 83 | |
| 84 | return completer.future; |
[email protected] | cefd1f2 | 2012-12-20 02:04:28 +0000 | [diff] [blame^] | 85 | } |