-
-
Notifications
You must be signed in to change notification settings - Fork 197
/
Copy pathandroid-tools-info.ts
385 lines (338 loc) · 16.3 KB
/
android-tools-info.ts
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
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
import * as path from "path";
import * as semver from "semver";
import {EOL} from "os";
export class AndroidToolsInfo implements IAndroidToolsInfo {
private static ANDROID_TARGET_PREFIX = "android";
private static SUPPORTED_TARGETS = ["android-17", "android-18", "android-19", "android-21", "android-22", "android-23"];
private static MIN_REQUIRED_COMPILE_TARGET = 22;
private static REQUIRED_BUILD_TOOLS_RANGE_PREFIX = ">=23";
private static VERSION_REGEX = /((\d+\.){2}\d+)/;
private static MIN_JAVA_VERSION = "1.8.0";
private showWarningsAsErrors: boolean;
private toolsInfo: IAndroidToolsInfoData;
private selectedCompileSdk: number;
private installedTargetsCache: string[] = null;
private androidHome = process.env["ANDROID_HOME"];
private pathToAndroidExecutable: string;
private _androidExecutableName: string;
private get androidExecutableName(): string {
if (!this._androidExecutableName) {
this._androidExecutableName = "android";
if (this.$hostInfo.isWindows) {
this._androidExecutableName += ".bat";
}
}
return this._androidExecutableName;
}
constructor(private $childProcess: IChildProcess,
private $errors: IErrors,
private $fs: IFileSystem,
private $hostInfo: IHostInfo,
private $logger: ILogger,
private $options: IOptions,
private $adb: Mobile.IAndroidDebugBridge) { }
public getPathToAndroidExecutable(options?: { showWarningsAsErrors: boolean }): IFuture<string> {
return ((): string => {
if (options) {
this.showWarningsAsErrors = options.showWarningsAsErrors;
}
if (!this.pathToAndroidExecutable) {
if (this.validateAndroidHomeEnvVariable(this.androidHome).wait()) {
let androidPath = path.join(this.androidHome, "tools", this.androidExecutableName);
if (!this.trySetAndroidPath(androidPath).wait() && !this.trySetAndroidPath(this.androidExecutableName).wait()) {
this.printMessage(`Unable to find "${this.androidExecutableName}" executable file. Make sure you have set ANDROID_HOME environment variable correctly.`);
}
} else {
this.$logger.trace("ANDROID_HOME environment variable is not set correctly.");
}
}
return this.pathToAndroidExecutable;
}).future<string>()();
}
private trySetAndroidPath(androidPath: string): IFuture<boolean> {
return ((): boolean => {
let isAndroidPathCorrect = true;
try {
let result = this.$adb.executeCommand(["--help"], { returnChildProcess: true }).wait();
if (result && result.stdout) {
this.$logger.trace(result.stdout);
this.pathToAndroidExecutable = androidPath;
} else {
this.$logger.trace(`Unable to find android executable from '${androidPath}'.`);
isAndroidPathCorrect = false;
}
} catch (err) {
this.$logger.trace(`Error occurred while checking androidExecutable from '${androidPath}'. ${err.message}`);
isAndroidPathCorrect = false;
}
return isAndroidPathCorrect;
}).future<boolean>()();
}
public getToolsInfo(): IFuture<IAndroidToolsInfoData> {
return ((): IAndroidToolsInfoData => {
if (!this.toolsInfo) {
let infoData: IAndroidToolsInfoData = Object.create(null);
infoData.androidHomeEnvVar = this.androidHome;
infoData.compileSdkVersion = this.getCompileSdk().wait();
infoData.buildToolsVersion = this.getBuildToolsVersion().wait();
infoData.targetSdkVersion = this.getTargetSdk().wait();
infoData.supportRepositoryVersion = this.getAndroidSupportRepositoryVersion().wait();
this.toolsInfo = infoData;
}
return this.toolsInfo;
}).future<IAndroidToolsInfoData>()();
}
public validateInfo(options?: { showWarningsAsErrors: boolean, validateTargetSdk: boolean }): IFuture<boolean> {
return ((): boolean => {
let detectedErrors = false;
this.showWarningsAsErrors = options && options.showWarningsAsErrors;
let toolsInfoData = this.getToolsInfo().wait();
let isAndroidHomeValid = this.validateAndroidHomeEnvVariable(toolsInfoData.androidHomeEnvVar).wait();
if (!toolsInfoData.compileSdkVersion) {
this.printMessage(`Cannot find a compatible Android SDK for compilation. To be able to build for Android, install Android SDK ${AndroidToolsInfo.MIN_REQUIRED_COMPILE_TARGET} or later.`,
"Run `$ android` to manage your Android SDK versions.");
detectedErrors = true;
}
if (!toolsInfoData.buildToolsVersion) {
let buildToolsRange = this.getBuildToolsRange();
let versionRangeMatches = buildToolsRange.match(/^.*?([\d\.]+)\s+.*?([\d\.]+)$/);
let message = `You can install any version in the following range: '${buildToolsRange}'.`;
// Improve message in case buildToolsRange is something like: ">=22.0.0 <=22.0.0" - same numbers on both sides
if (versionRangeMatches && versionRangeMatches[1] && versionRangeMatches[2] && versionRangeMatches[1] === versionRangeMatches[2]) {
message = `You have to install version ${versionRangeMatches[1]}.`;
}
let invalidBuildToolsAdditionalMsg = 'Run `android` from your command-line to install required `Android Build Tools`.';
if (!isAndroidHomeValid) {
invalidBuildToolsAdditionalMsg += ' In case you already have them installed, make sure `ANDROID_HOME` environment variable is set correctly.';
}
this.printMessage("You need to have the Android SDK Build-tools installed on your system. " + message, invalidBuildToolsAdditionalMsg);
detectedErrors = true;
}
if (!toolsInfoData.supportRepositoryVersion) {
let invalidSupportLibAdditionalMsg = 'Run `$ android` to manage the Local Maven repository for Support Libraries.';
if (!isAndroidHomeValid) {
invalidSupportLibAdditionalMsg += ' In case you already have it installed, make sure `ANDROID_HOME` environment variable is set correctly.';
}
this.printMessage(`You need to have Android SDK ${AndroidToolsInfo.MIN_REQUIRED_COMPILE_TARGET} or later and the latest Local Maven repository for Support Libraries installed on your system.`, invalidSupportLibAdditionalMsg);
detectedErrors = true;
}
if (options && options.validateTargetSdk) {
let targetSdk = toolsInfoData.targetSdkVersion;
let newTarget = `${AndroidToolsInfo.ANDROID_TARGET_PREFIX}-${targetSdk}`;
if (!_.contains(AndroidToolsInfo.SUPPORTED_TARGETS, newTarget)) {
let supportedVersions = AndroidToolsInfo.SUPPORTED_TARGETS.sort();
let minSupportedVersion = this.parseAndroidSdkString(_.first(supportedVersions));
if (targetSdk && (targetSdk < minSupportedVersion)) {
this.printMessage(`The selected Android target SDK ${newTarget} is not supported. You must target ${minSupportedVersion} or later.`);
detectedErrors = true;
} else if (!targetSdk || targetSdk > this.getMaxSupportedVersion()) {
this.$logger.warn(`Support for the selected Android target SDK ${newTarget} is not verified. Your Android app might not work as expected.`);
}
}
}
return detectedErrors || !isAndroidHomeValid;
}).future<boolean>()();
}
public validateJavacVersion(installedJavaVersion: string, options?: { showWarningsAsErrors: boolean }): IFuture<boolean> {
return ((): boolean => {
let hasProblemWithJavaVersion = false;
if (options) {
this.showWarningsAsErrors = options.showWarningsAsErrors;
}
let additionalMessage = "You will not be able to build your projects for Android." + EOL
+ "To be able to build for Android, verify that you have installed The Java Development Kit (JDK) and configured it according to system requirements as" + EOL +
" described in https://github.com/NativeScript/nativescript-cli#system-requirements.";
let matchingVersion = (installedJavaVersion || "").match(AndroidToolsInfo.VERSION_REGEX);
if (matchingVersion && matchingVersion[1]) {
if (semver.lt(matchingVersion[1], AndroidToolsInfo.MIN_JAVA_VERSION)) {
hasProblemWithJavaVersion = true;
this.printMessage(`Javac version ${installedJavaVersion} is not supported. You have to install at least ${AndroidToolsInfo.MIN_JAVA_VERSION}.`, additionalMessage);
}
} else {
hasProblemWithJavaVersion = true;
this.printMessage("Error executing command 'javac'. Make sure you have installed The Java Development Kit (JDK) and set JAVA_HOME environment variable.", additionalMessage);
}
return hasProblemWithJavaVersion;
}).future<boolean>()();
}
public getPathToAdbFromAndroidHome(): IFuture<string> {
return (() => {
if (this.androidHome) {
let pathToAdb = path.join(this.androidHome, "platform-tools", "adb");
try {
this.$childProcess.execFile(pathToAdb, ["help"]).wait();
return pathToAdb;
} catch (err) {
// adb does not exist, so ANDROID_HOME is not set correctly
// try getting default adb path (included in CLI package)
this.$logger.trace(`Error while executing '${pathToAdb} help'. Error is: ${err.message}`);
}
}
return null;
}).future<string>()();
}
/**
* Prints messages on the screen. In case the showWarningsAsErrors flag is set to true, warnings are shown, else - errors.
* Uses logger.warn for warnings and errors.failWithoutHelp when erros must be shown.
* In case additional details must be shown as info message, use the second parameter.
* NOTE: The additional information will not be printed when showWarningsAsErrors flag is set.
* @param {string} msg The message that will be shown as warning or error.
* @param {string} additionalMsg The additional message that will be shown as info message.
* @return {void}
*/
private printMessage(msg: string, additionalMsg?: string): void {
if (this.showWarningsAsErrors) {
this.$errors.failWithoutHelp(msg);
} else {
this.$logger.warn(msg);
}
if (additionalMsg) {
this.$logger.printMarkdown(additionalMsg);
}
}
private getCompileSdk(): IFuture<number> {
return ((): number => {
if (!this.selectedCompileSdk) {
let userSpecifiedCompileSdk = this.$options.compileSdk;
if (userSpecifiedCompileSdk) {
let installedTargets = this.getInstalledTargets().wait();
let androidCompileSdk = `${AndroidToolsInfo.ANDROID_TARGET_PREFIX}-${userSpecifiedCompileSdk}`;
if (!_.contains(installedTargets, androidCompileSdk)) {
this.$errors.failWithoutHelp(`You have specified '${userSpecifiedCompileSdk}' for compile sdk, but it is not installed on your system.`);
}
this.selectedCompileSdk = userSpecifiedCompileSdk;
} else {
let latestValidAndroidTarget = this.getLatestValidAndroidTarget().wait();
if (latestValidAndroidTarget) {
let integerVersion = this.parseAndroidSdkString(latestValidAndroidTarget);
if (integerVersion && integerVersion >= AndroidToolsInfo.MIN_REQUIRED_COMPILE_TARGET) {
this.selectedCompileSdk = integerVersion;
}
}
}
}
return this.selectedCompileSdk;
}).future<number>()();
}
private getTargetSdk(): IFuture<number> {
return ((): number => {
let targetSdk = this.$options.sdk ? parseInt(this.$options.sdk) : this.getCompileSdk().wait();
this.$logger.trace(`Selected targetSdk is: ${targetSdk}`);
return targetSdk;
}).future<number>()();
}
private getMatchingDir(pathToDir: string, versionRange: string): IFuture<string> {
return ((): string => {
let selectedVersion: string;
if (this.$fs.exists(pathToDir).wait()) {
let subDirs = this.$fs.readDirectory(pathToDir).wait();
this.$logger.trace(`Directories found in ${pathToDir} are ${subDirs.join(", ")}`);
let subDirsVersions = subDirs
.map(dirName => {
let dirNameGroups = dirName.match(AndroidToolsInfo.VERSION_REGEX);
if (dirNameGroups) {
return dirNameGroups[1];
}
return null;
})
.filter(dirName => !!dirName);
this.$logger.trace(`Versions found in ${pathToDir} are ${subDirsVersions.join(", ")}`);
let version = semver.maxSatisfying(subDirsVersions, versionRange);
if (version) {
selectedVersion = _.find(subDirs, dir => dir.indexOf(version) !== -1);
}
}
this.$logger.trace("Selected version is: ", selectedVersion);
return selectedVersion;
}).future<string>()();
}
private getBuildToolsRange(): string {
return `${AndroidToolsInfo.REQUIRED_BUILD_TOOLS_RANGE_PREFIX} <=${this.getMaxSupportedVersion()}`;
}
private getBuildToolsVersion(): IFuture<string> {
return ((): string => {
let buildToolsVersion: string;
if (this.androidHome) {
let pathToBuildTools = path.join(this.androidHome, "build-tools");
let buildToolsRange = this.getBuildToolsRange();
buildToolsVersion = this.getMatchingDir(pathToBuildTools, buildToolsRange).wait();
}
return buildToolsVersion;
}).future<string>()();
}
private getAppCompatRange(): IFuture<string> {
return ((): string => {
let compileSdkVersion = this.getCompileSdk().wait();
let requiredAppCompatRange: string;
if (compileSdkVersion) {
requiredAppCompatRange = `>=${compileSdkVersion} <${compileSdkVersion + 1}`;
}
return requiredAppCompatRange;
}).future<string>()();
}
private getAndroidSupportRepositoryVersion(): IFuture<string> {
return ((): string => {
let selectedAppCompatVersion: string;
let requiredAppCompatRange = this.getAppCompatRange().wait();
if (this.androidHome && requiredAppCompatRange) {
let pathToAppCompat = path.join(this.androidHome, "extras", "android", "m2repository", "com", "android", "support", "appcompat-v7");
selectedAppCompatVersion = this.getMatchingDir(pathToAppCompat, requiredAppCompatRange).wait();
}
this.$logger.trace(`Selected AppCompat version is: ${selectedAppCompatVersion}`);
return selectedAppCompatVersion;
}).future<string>()();
}
private getLatestValidAndroidTarget(): IFuture<string> {
return (() => {
let installedTargets = this.getInstalledTargets().wait();
return _.findLast(AndroidToolsInfo.SUPPORTED_TARGETS.sort(), supportedTarget => _.contains(installedTargets, supportedTarget));
}).future<string>()();
}
private parseAndroidSdkString(androidSdkString: string): number {
return parseInt(androidSdkString.replace(`${AndroidToolsInfo.ANDROID_TARGET_PREFIX}-`, ""));
}
private getInstalledTargets(): IFuture<string[]> {
return (() => {
if (!this.installedTargetsCache) {
try {
let pathToAndroidExecutable = this.getPathToAndroidExecutable().wait();
if (pathToAndroidExecutable) {
let result = this.$childProcess.spawnFromEvent(pathToAndroidExecutable, ["list", "targets"], "close", {}, { throwError: false }).wait();
if (result && result.stdout) {
this.$logger.trace(result.stdout);
this.installedTargetsCache = [];
result.stdout.replace(/id: \d+ or "(.+)"/g, (m: string, p1: string) => (this.installedTargetsCache.push(p1), m));
}
}
} catch (err) {
this.$logger.trace("Unable to get Android targets. Error is: " + err);
}
}
return this.installedTargetsCache;
}).future<string[]>()();
}
private getMaxSupportedVersion(): number {
return this.parseAndroidSdkString(_.last(AndroidToolsInfo.SUPPORTED_TARGETS.sort()));
}
private _cachedAndroidHomeValidationResult: boolean = null;
private validateAndroidHomeEnvVariable(androidHomeEnvVar: string): IFuture<boolean> {
return ((): boolean => {
if (this._cachedAndroidHomeValidationResult === null) {
this._cachedAndroidHomeValidationResult = true;
let expectedDirectoriesInAndroidHome = ["build-tools", "tools", "platform-tools", "extras"];
if (!androidHomeEnvVar || !this.$fs.exists(androidHomeEnvVar).wait()) {
this.printMessage("The ANDROID_HOME environment variable is not set or it points to a non-existent directory. You will not be able to perform any build-related operations for Android.",
"To be able to perform Android build-related operations, set the `ANDROID_HOME` variable to point to the root of your Android SDK installation directory.");
this._cachedAndroidHomeValidationResult = false;
} else if (!_.any(expectedDirectoriesInAndroidHome.map(dir => this.$fs.exists(path.join(androidHomeEnvVar, dir)).wait()))) {
this.printMessage("The ANDROID_HOME environment variable points to incorrect directory. You will not be able to perform any build-related operations for Android.",
"To be able to perform Android build-related operations, set the `ANDROID_HOME` variable to point to the root of your Android SDK installation directory, " +
"where you will find `tools` and `platform-tools` directories.");
this._cachedAndroidHomeValidationResult = false;
}
}
return this._cachedAndroidHomeValidationResult;
}).future<boolean>()();
}
}
$injector.register("androidToolsInfo", AndroidToolsInfo);