-
-
Notifications
You must be signed in to change notification settings - Fork 197
/
Copy pathandroid-tools-info.ts
228 lines (194 loc) · 9.19 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
///<reference path=".d.ts"/>
"use strict";
import * as path from "path";
import * as semver from "semver";
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 = 21;
private static REQUIRED_BUILD_TOOLS_RANGE_PREFIX = ">=22";
private static VERSION_REGEX = /^(\d+\.){2}\d+$/;
private showWarningsAsErrors: boolean;
private toolsInfo: IAndroidToolsInfoData;
private selectedCompileSdk: number;
private installedTargetsCache: string[] = null;
private androidHome = process.env["ANDROID_HOME"];
constructor(private $childProcess: IChildProcess,
private $errors: IErrors,
private $fs: IFileSystem,
private $logger: ILogger,
private $options: IOptions) {}
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();
if(!toolsInfoData.androidHomeEnvVar || !this.$fs.exists(toolsInfoData.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.");
}
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]}.`;
}
this.printMessage("You need to have the Android SDK Build-tools installed on your system. " + message,
'Run "android" from your command-line to install required Android Build Tools.');
detectedErrors = true;
}
if(!toolsInfoData.supportRepositoryVersion) {
this.printMessage(`You need to have the latest Android Support Repository installed on your system.`,
'Run `$ android` to manage the Android Support Repository.');
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;
}).future<boolean>()();
}
/**
* 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.info(additionalMsg);
}
}
private getCompileSdk(): IFuture<number> {
return ((): number => {
if(!this.selectedCompileSdk) {
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()
.filter(buildTools => !!buildTools.match(AndroidToolsInfo.VERSION_REGEX));
this.$logger.trace(`Versions found in ${pathToDir} are ${subDirs.join(", ")}`);
selectedVersion = semver.maxSatisfying(subDirs, versionRange);
}
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) {
this.installedTargetsCache = [];
let output = this.$childProcess.exec('android list targets').wait();
output.replace(/id: \d+ or "(.+)"/g, (m:string, p1:string) => (this.installedTargetsCache.push(p1), m));
}
return this.installedTargetsCache;
}).future<string[]>()();
}
private getMaxSupportedVersion(): number {
return this.parseAndroidSdkString(_.last(AndroidToolsInfo.SUPPORTED_TARGETS.sort()));
}
}
$injector.register("androidToolsInfo", AndroidToolsInfo);