android_sdk.dart 8.88 KB
Newer Older
1 2 3 4
// Copyright 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5
import '../base/common.dart';
6
import '../base/context.dart';
7
import '../base/file_system.dart';
8
import '../base/os.dart';
9
import '../base/platform.dart';
10
import '../base/process_manager.dart';
11
import '../base/version.dart';
12
import '../globals.dart';
13

14 15
AndroidSdk get androidSdk => context[AndroidSdk];

16 17
const String kAndroidHome = 'ANDROID_HOME';

18
// Android SDK layout:
19

20
// $ANDROID_HOME/platform-tools/adb
21

22 23 24
// $ANDROID_HOME/build-tools/19.1.0/aapt, dx, zipalign
// $ANDROID_HOME/build-tools/22.0.1/aapt
// $ANDROID_HOME/build-tools/23.0.2/aapt
25
// $ANDROID_HOME/build-tools/24.0.0-preview/aapt
26
// $ANDROID_HOME/build-tools/25.0.2/apksigner
27

28 29
// $ANDROID_HOME/platforms/android-22/android.jar
// $ANDROID_HOME/platforms/android-23/android.jar
30
// $ANDROID_HOME/platforms/android-N/android.jar
31

32 33
// Special case some version names in the sdk.
const Map<String, int> _namedVersionMap = const <String, int> {
34 35
  'android-N': 24,
  'android-stable': 24,
36
};
37

38 39 40
/// The minimum Android SDK version we support.
const int minimumAndroidSdkVersion = 25;

41
/// Locate ADB. Prefer to use one from an Android SDK, if we can locate that.
42 43 44
/// This should be used over accessing androidSdk.adbPath directly because it
/// will work for those users who have Android Platform Tools installed but
/// not the full SDK.
45 46 47 48
String getAdbPath([AndroidSdk existingSdk]) {
  if (existingSdk?.adbPath != null)
    return existingSdk.adbPath;

49
  final AndroidSdk sdk = AndroidSdk.locateAndroidSdk();
50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68

  if (sdk?.latestVersion == null) {
    return os.which('adb')?.path;
  } else {
    return sdk.adbPath;
  }
}

class AndroidSdk {
  AndroidSdk(this.directory) {
    _init();
  }

  final String directory;

  List<AndroidSdkVersion> _sdkVersions;
  AndroidSdkVersion _latestVersion;

  static AndroidSdk locateAndroidSdk() {
69
    String androidHomeDir;
70 71 72
    if (platform.environment.containsKey(kAndroidHome)) {
      androidHomeDir = platform.environment[kAndroidHome];
    } else if (platform.isLinux) {
73
      if (homeDirPath != null)
74
        androidHomeDir = fs.path.join(homeDirPath, 'Android', 'Sdk');
75
    } else if (platform.isMacOS) {
76
      if (homeDirPath != null)
77
        androidHomeDir = fs.path.join(homeDirPath, 'Library', 'Android', 'sdk');
78
    } else if (platform.isWindows) {
79
      if (homeDirPath != null)
80
        androidHomeDir = fs.path.join(homeDirPath, 'AppData', 'Local', 'Android', 'sdk');
81 82 83 84 85
    }

    if (androidHomeDir != null) {
      if (validSdkDirectory(androidHomeDir))
        return new AndroidSdk(androidHomeDir);
86 87
      if (validSdkDirectory(fs.path.join(androidHomeDir, 'sdk')))
        return new AndroidSdk(fs.path.join(androidHomeDir, 'sdk'));
88 89
    }

90 91 92
    // in build-tools/$version/aapt
    final List<File> aaptBins = os.whichAll('aapt');
    for (File aaptBin in aaptBins) {
93
      // Make sure we're using the aapt from the SDK.
94
      aaptBin = fs.file(aaptBin.resolveSymbolicLinksSync());
95
      final String dir = aaptBin.parent.parent.parent.path;
96 97 98 99
      if (validSdkDirectory(dir))
        return new AndroidSdk(dir);
    }

100 101 102
    // in platform-tools/adb
    final List<File> adbBins = os.whichAll('adb');
    for (File adbBin in adbBins) {
103
      // Make sure we're using the adb from the SDK.
104
      adbBin = fs.file(adbBin.resolveSymbolicLinksSync());
105
      final String dir = adbBin.parent.parent.path;
106 107 108 109 110 111 112 113 114 115
      if (validSdkDirectory(dir))
        return new AndroidSdk(dir);
    }

    // No dice.
    printTrace('Unable to locate an Android SDK.');
    return null;
  }

  static bool validSdkDirectory(String dir) {
116
    return fs.isDirectorySync(fs.path.join(dir, 'platform-tools'));
117 118 119 120 121 122 123 124
  }

  List<AndroidSdkVersion> get sdkVersions => _sdkVersions;

  AndroidSdkVersion get latestVersion => _latestVersion;

  String get adbPath => getPlatformToolsPath('adb');

125 126
  /// Validate the Android SDK. This returns an empty list if there are no
  /// issues; otherwise, it returns a list of issues found.
127
  List<String> validateSdkWellFormed({bool requireApkSigner = true}) {
128
    if (!processManager.canRun(adbPath))
129
      return <String>['Android SDK file not found: $adbPath.'];
130

131
    if (sdkVersions.isEmpty || latestVersion == null)
132
      return <String>['Android SDK is missing command line tools; download from https://goo.gl/XxQghQ'];
133

134
    return latestVersion.validateSdkWellFormed(requireApkSigner: requireApkSigner);
135 136 137
  }

  String getPlatformToolsPath(String binaryName) {
138
    return fs.path.join(directory, 'platform-tools', binaryName);
139 140 141 142 143
  }

  void _init() {
    List<String> platforms = <String>[]; // android-22, ...

144
    final Directory platformsDir = fs.directory(fs.path.join(directory, 'platforms'));
145 146 147
    if (platformsDir.existsSync()) {
      platforms = platformsDir
        .listSync()
148
        .map<String>((FileSystemEntity entity) => fs.path.basename(entity.path))
149 150 151 152
        .where((String name) => name.startsWith('android-'))
        .toList();
    }

153
    List<Version> buildTools = <Version>[]; // 19.1.0, 22.0.1, ...
154

155
    final Directory buildToolsDir = fs.directory(fs.path.join(directory, 'build-tools'));
156
    if (buildToolsDir.existsSync()) {
157
      buildTools = buildToolsDir
158 159 160
        .listSync()
        .map((FileSystemEntity entity) {
          try {
161
            return new Version.parse(fs.path.basename(entity.path));
162 163 164 165 166 167 168 169
          } catch (error) {
            return null;
          }
        })
        .where((Version version) => version != null)
        .toList();
    }

170
    // Match up platforms with the best corresponding build-tools.
171 172
    _sdkVersions = platforms.map((String platformName) {
      int platformVersion;
173 174

      try {
175 176 177 178
        if (_namedVersionMap.containsKey(platformName))
          platformVersion = _namedVersionMap[platformName];
        else
          platformVersion = int.parse(platformName.substring('android-'.length));
179 180 181 182
      } catch (error) {
        return null;
      }

183 184
      Version buildToolsVersion = Version.primary(buildTools.where((Version version) {
        return version.major == platformVersion;
185 186
      }).toList());

187 188
      buildToolsVersion ??= Version.primary(buildTools);

189 190 191
      if (buildToolsVersion == null)
        return null;

192 193 194
      return new AndroidSdkVersion(
        this,
        platformVersionName: platformName,
195
        buildToolsVersion: buildToolsVersion
196
      );
197 198 199 200 201 202 203
    }).where((AndroidSdkVersion version) => version != null).toList();

    _sdkVersions.sort();

    _latestVersion = _sdkVersions.isEmpty ? null : _sdkVersions.last;
  }

204
  @override
205 206 207 208
  String toString() => 'AndroidSdk: $directory';
}

class AndroidSdkVersion implements Comparable<AndroidSdkVersion> {
209 210
  AndroidSdkVersion(this.sdk, {
    this.platformVersionName,
211
    this.buildToolsVersion,
212
  });
213 214

  final AndroidSdk sdk;
215
  final String platformVersionName;
216 217
  final Version buildToolsVersion;
  String get buildToolsVersionName => buildToolsVersion.toString();
218 219 220 221 222 223 224

  int get sdkLevel {
    if (_namedVersionMap.containsKey(platformVersionName))
      return _namedVersionMap[platformVersionName];
    else
      return int.parse(platformVersionName.substring('android-'.length));
  }
225 226 227 228 229

  String get androidJarPath => getPlatformsPath('android.jar');

  String get aaptPath => getBuildToolsPath('aapt');

230
  String get dxPath => getBuildToolsPath('dx');
231 232 233

  String get zipalignPath => getBuildToolsPath('zipalign');

234 235
  String get apksignerPath => getBuildToolsPath('apksigner');

236
  List<String> validateSdkWellFormed({bool requireApkSigner = true}) {
237 238 239 240
    if (buildToolsVersion.major < minimumAndroidSdkVersion) {
      return <String>['Minimum supported Android SDK version is $minimumAndroidSdkVersion '
                      'but this system has ${buildToolsVersion.major}. Please upgrade.'];
    }
241 242 243
    if (_exists(androidJarPath) != null)
      return <String>[_exists(androidJarPath)];

244 245
    if (_canRun(aaptPath) != null)
      return <String>[_canRun(aaptPath)];
246

247 248
    if (_canRun(dxPath) != null)
      return <String>[_canRun(dxPath)];
249

250 251
    if (_canRun(zipalignPath) != null)
      return <String>[_canRun(zipalignPath)];
252

253
    if (requireApkSigner && _canRun(apksignerPath) != null)
254 255
      return <String>[_canRun(apksignerPath) + '\napksigner requires Android SDK Build Tools 24.0.3 or newer.'];

256
    return <String>[];
257 258 259
  }

  String getPlatformsPath(String itemName) {
260
    return fs.path.join(sdk.directory, 'platforms', platformVersionName, itemName);
261 262
  }

263
  String getBuildToolsPath(String binaryName) {
264
    return fs.path.join(sdk.directory, 'build-tools', buildToolsVersionName, binaryName);
265 266
  }

267
  @override
268
  int compareTo(AndroidSdkVersion other) => sdkLevel - other.sdkLevel;
269

270
  @override
271
  String toString() => '[${sdk.directory}, SDK version $sdkLevel, build-tools $buildToolsVersionName]';
272

273
  String _exists(String path) {
274
    if (!fs.isFileSync(path))
275 276
      return 'Android SDK file not found: $path.';
    return null;
277
  }
278 279 280 281 282 283

  String _canRun(String path) {
    if (!processManager.canRun(path))
      return 'Android SDK file not found: $path.';
    return null;
  }
284
}