OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 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 | 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. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 library command_lish; | 5 library command_lish; |
6 | 6 |
7 import 'dart:io'; | 7 import 'dart:io'; |
8 import 'dart:json'; | 8 import 'dart:json'; |
9 import 'dart:uri'; | 9 import 'dart:uri'; |
10 | 10 |
11 import '../../pkg/args/lib/args.dart'; | 11 import '../../pkg/args/lib/args.dart'; |
12 import '../../pkg/http/lib/http.dart' as http; | 12 import '../../pkg/http/lib/http.dart' as http; |
13 import 'pub.dart'; | 13 import 'pub.dart'; |
14 import 'io.dart'; | 14 import 'io.dart'; |
15 import 'git.dart' as git; | 15 import 'git.dart' as git; |
16 import 'oauth2.dart' as oauth2; | 16 import 'oauth2.dart' as oauth2; |
| 17 import 'validator.dart'; |
17 | 18 |
18 // TODO(nweiz): Make "publish" the primary name for this command. See issue | 19 // TODO(nweiz): Make "publish" the primary name for this command. See issue |
19 // 6949. | 20 // 6949. |
20 /// Handles the `lish` and `publish` pub commands. | 21 /// Handles the `lish` and `publish` pub commands. |
21 class LishCommand extends PubCommand { | 22 class LishCommand extends PubCommand { |
22 final description = "publish the current package to pub.dartlang.org"; | 23 final description = "publish the current package to pub.dartlang.org"; |
23 final usage = "pub publish [options]"; | 24 final usage = "pub publish [options]"; |
24 final aliases = const ["lish", "lush"]; | 25 final aliases = const ["lish", "lush"]; |
25 | 26 |
26 ArgParser get commandParser { | 27 ArgParser get commandParser { |
27 var parser = new ArgParser(); | 28 var parser = new ArgParser(); |
28 parser.addOption('server', defaultsTo: 'https://pub.dartlang.org', | 29 parser.addOption('server', defaultsTo: 'https://pub.dartlang.org', |
29 help: 'The package server to which to upload this package'); | 30 help: 'The package server to which to upload this package'); |
30 return parser; | 31 return parser; |
31 } | 32 } |
32 | 33 |
33 /// The URL of the server to which to upload the package. | 34 /// The URL of the server to which to upload the package. |
34 Uri get server => new Uri.fromString(commandOptions['server']); | 35 Uri get server => new Uri.fromString(commandOptions['server']); |
35 | 36 |
36 Future onRun() { | 37 Future onRun() { |
37 var cloudStorageUrl; | 38 var cloudStorageUrl; |
38 return oauth2.withClient(cache, (client) { | 39 return oauth2.withClient(cache, (client) { |
39 // TODO(nweiz): Cloud Storage can provide an XML-formatted error. We | 40 // TODO(nweiz): Cloud Storage can provide an XML-formatted error. We |
40 // should report that error and exit. | 41 // should report that error and exit. |
41 return Futures.wait([ | 42 return Futures.wait([ |
42 client.get(server.resolve("/packages/versions/new.json")), | 43 client.get(server.resolve("/packages/versions/new.json")), |
43 _filesToPublish.transform((files) { | 44 _filesToPublish.transform((files) { |
44 return createTarGz(files, baseDir: entrypoint.root.dir); | 45 return createTarGz(files, baseDir: entrypoint.root.dir); |
45 }).chain(consumeInputStream) | 46 }).chain(consumeInputStream), |
| 47 _validate() |
46 ]).chain((results) { | 48 ]).chain((results) { |
47 var response = results[0]; | 49 var response = results[0]; |
48 var packageBytes = results[1]; | 50 var packageBytes = results[1]; |
49 var parameters = _parseJson(response); | 51 var parameters = _parseJson(response); |
50 | 52 |
51 var url = _expectField(parameters, 'url', response); | 53 var url = _expectField(parameters, 'url', response); |
52 if (url is! String) _invalidServerResponse(response); | 54 if (url is! String) _invalidServerResponse(response); |
53 cloudStorageUrl = new Uri.fromString(url); | 55 cloudStorageUrl = new Uri.fromString(url); |
54 var request = new http.MultipartRequest('POST', cloudStorageUrl); | 56 var request = new http.MultipartRequest('POST', cloudStorageUrl); |
55 | 57 |
(...skipping 107 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
163 /// error if [map] doens't contain [key]. | 165 /// error if [map] doens't contain [key]. |
164 _expectField(Map map, String key, http.Response response) { | 166 _expectField(Map map, String key, http.Response response) { |
165 if (map.containsKey(key)) return map[key]; | 167 if (map.containsKey(key)) return map[key]; |
166 _invalidServerResponse(response); | 168 _invalidServerResponse(response); |
167 } | 169 } |
168 | 170 |
169 /// Throws an error describing an invalid response from the server. | 171 /// Throws an error describing an invalid response from the server. |
170 void _invalidServerResponse(http.Response response) { | 172 void _invalidServerResponse(http.Response response) { |
171 throw 'Invalid server response:\n${response.body}'; | 173 throw 'Invalid server response:\n${response.body}'; |
172 } | 174 } |
| 175 |
| 176 /// Validates the package. Throws an exception if it's invalid. |
| 177 Future _validate() { |
| 178 return Validator.runAll(entrypoint).chain((pair) { |
| 179 var errors = pair.first; |
| 180 var warnings = pair.last; |
| 181 |
| 182 if (errors.isEmpty && warnings.isEmpty) return new Future.immediate(null); |
| 183 if (!errors.isEmpty) throw "Package validation failed."; |
| 184 |
| 185 var s = warnings.length == 1 ? '' : 's'; |
| 186 stdout.writeString("Package has ${warnings.length} warning$s. Upload " |
| 187 "anyway (y/n)? "); |
| 188 return readLine().transform((line) { |
| 189 if (new RegExp(r"^[yY]").hasMatch(line)) return; |
| 190 throw "Package upload canceled."; |
| 191 }); |
| 192 }); |
| 193 } |
173 } | 194 } |
OLD | NEW |