android_sdk.dart 16.7 KB
Newer Older
Ian Hickson's avatar
Ian Hickson committed
1
// Copyright 2014 The Flutter Authors. All rights reserved.
2 3 4
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

5 6
import 'package:meta/meta.dart';

7
import '../base/common.dart';
8
import '../base/file_system.dart';
9
import '../base/os.dart';
10
import '../base/platform.dart';
11
import '../base/process.dart';
12
import '../base/version.dart';
13
import '../convert.dart';
14
import '../globals.dart' as globals;
15
import 'android_studio.dart';
16

17 18
// ANDROID_HOME is deprecated.
// See https://developer.android.com/studio/command-line/variables.html#envar
19
const String kAndroidHome = 'ANDROID_HOME';
20
const String kAndroidSdkRoot = 'ANDROID_SDK_ROOT';
21

22 23
final RegExp _numberedAndroidPlatformRe = RegExp(r'^android-([0-9]+)$');
final RegExp _sdkVersionRe = RegExp(r'^ro.build.version.sdk=([0-9]+)$');
24

25
// Android SDK layout:
26

27
// $ANDROID_SDK_ROOT/platform-tools/adb
28

29 30 31 32 33
// $ANDROID_SDK_ROOT/build-tools/19.1.0/aapt, dx, zipalign
// $ANDROID_SDK_ROOT/build-tools/22.0.1/aapt
// $ANDROID_SDK_ROOT/build-tools/23.0.2/aapt
// $ANDROID_SDK_ROOT/build-tools/24.0.0-preview/aapt
// $ANDROID_SDK_ROOT/build-tools/25.0.2/apksigner
34

35 36 37
// $ANDROID_SDK_ROOT/platforms/android-22/android.jar
// $ANDROID_SDK_ROOT/platforms/android-23/android.jar
// $ANDROID_SDK_ROOT/platforms/android-N/android.jar
38
class AndroidSdk {
39
  AndroidSdk(this.directory) {
40
    reinitialize();
41 42
  }

43 44
  static const String _javaHomeEnvironmentVariable = 'JAVA_HOME';
  static const String _javaExecutable = 'java';
45

46
  /// The path to the Android SDK.
47 48 49 50 51
  final String directory;

  List<AndroidSdkVersion> _sdkVersions;
  AndroidSdkVersion _latestVersion;

52
  /// Whether the `platform-tools` or `cmdline-tools` directory exists in the Android SDK.
53 54 55 56 57
  ///
  /// It is possible to have an Android SDK folder that is missing this with
  /// the expectation that it will be downloaded later, e.g. by gradle or the
  /// sdkmanager. The [licensesAvailable] property should be used to determine
  /// whether the licenses are at least possibly accepted.
58 59 60
  bool get platformToolsAvailable =>
    globals.fs.directory(globals.fs.path.join(directory, 'cmdline-tools')).existsSync() ||
    globals.fs.directory(globals.fs.path.join(directory, 'platform-tools')).existsSync();
61 62 63 64 65 66 67 68

  /// Whether the `licenses` directory exists in the Android SDK.
  ///
  /// The existence of this folder normally indicates that the SDK licenses have
  /// been accepted, e.g. via the sdkmanager, Android Studio, or by copying them
  /// from another workstation such as in CI scenarios. If these files are valid
  /// gradle or the sdkmanager will be able to download and use other parts of
  /// the SDK on demand.
69
  bool get licensesAvailable => globals.fs.directory(globals.fs.path.join(directory, 'licenses')).existsSync();
70

71
  static AndroidSdk locateAndroidSdk() {
72 73
    String findAndroidHomeDir() {
      String androidHomeDir;
74 75 76 77 78 79 80
      if (globals.config.containsKey('android-sdk')) {
        androidHomeDir = globals.config.getValue('android-sdk') as String;
      } else if (globals.platform.environment.containsKey(kAndroidHome)) {
        androidHomeDir = globals.platform.environment[kAndroidHome];
      } else if (globals.platform.environment.containsKey(kAndroidSdkRoot)) {
        androidHomeDir = globals.platform.environment[kAndroidSdkRoot];
      } else if (globals.platform.isLinux) {
81 82 83 84 85 86
        if (globals.fsUtils.homeDirPath != null) {
          androidHomeDir = globals.fs.path.join(
            globals.fsUtils.homeDirPath,
            'Android',
            'Sdk',
          );
87
        }
88
      } else if (globals.platform.isMacOS) {
89 90 91 92 93 94 95
        if (globals.fsUtils.homeDirPath != null) {
          androidHomeDir = globals.fs.path.join(
            globals.fsUtils.homeDirPath,
            'Library',
            'Android',
            'sdk',
          );
96
        }
97
      } else if (globals.platform.isWindows) {
98 99 100 101 102 103 104 105
        if (globals.fsUtils.homeDirPath != null) {
          androidHomeDir = globals.fs.path.join(
            globals.fsUtils.homeDirPath,
            'AppData',
            'Local',
            'Android',
            'sdk',
          );
106
        }
107 108 109
      }

      if (androidHomeDir != null) {
110
        if (validSdkDirectory(androidHomeDir)) {
111
          return androidHomeDir;
112
        }
113 114
        if (validSdkDirectory(globals.fs.path.join(androidHomeDir, 'sdk'))) {
          return globals.fs.path.join(androidHomeDir, 'sdk');
115
        }
116 117 118
      }

      // in build-tools/$version/aapt
119
      final List<File> aaptBins = globals.os.whichAll('aapt');
120 121
      for (File aaptBin in aaptBins) {
        // Make sure we're using the aapt from the SDK.
122
        aaptBin = globals.fs.file(aaptBin.resolveSymbolicLinksSync());
123
        final String dir = aaptBin.parent.parent.parent.path;
124
        if (validSdkDirectory(dir)) {
125
          return dir;
126
        }
127 128 129
      }

      // in platform-tools/adb
130
      final List<File> adbBins = globals.os.whichAll('adb');
131 132
      for (File adbBin in adbBins) {
        // Make sure we're using the adb from the SDK.
133
        adbBin = globals.fs.file(adbBin.resolveSymbolicLinksSync());
134
        final String dir = adbBin.parent.parent.path;
135
        if (validSdkDirectory(dir)) {
136
          return dir;
137
        }
138 139 140 141 142 143 144 145
      }

      return null;
    }

    final String androidHomeDir = findAndroidHomeDir();
    if (androidHomeDir == null) {
      // No dice.
146
      globals.printTrace('Unable to locate an Android SDK.');
147
      return null;
148 149
    }

150
    return AndroidSdk(androidHomeDir);
151 152 153
  }

  static bool validSdkDirectory(String dir) {
Lau Ching Jun's avatar
Lau Ching Jun committed
154
    return sdkDirectoryHasLicenses(dir) || sdkDirectoryHasPlatformTools(dir);
155 156 157
  }

  static bool sdkDirectoryHasPlatformTools(String dir) {
158
    return globals.fs.isDirectorySync(globals.fs.path.join(dir, 'platform-tools'));
159 160
  }

Lau Ching Jun's avatar
Lau Ching Jun committed
161
  static bool sdkDirectoryHasLicenses(String dir) {
162
    return globals.fs.isDirectorySync(globals.fs.path.join(dir, 'licenses'));
163 164 165 166 167 168
  }

  List<AndroidSdkVersion> get sdkVersions => _sdkVersions;

  AndroidSdkVersion get latestVersion => _latestVersion;

169
  String get adbPath => getPlatformToolsPath(globals.platform.isWindows ? 'adb.exe' : 'adb');
170

Danny Tuppeny's avatar
Danny Tuppeny committed
171
  String get emulatorPath => getEmulatorPath();
172

173 174
  String get avdManagerPath => getAvdManagerPath();

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
  /// Locate the path for storing AVD emulator images. Returns null if none found.
  String getAvdPath() {
    final List<String> searchPaths = <String>[
      globals.platform.environment['ANDROID_AVD_HOME'],
      if (globals.platform.environment['HOME'] != null)
        globals.fs.path.join(globals.platform.environment['HOME'], '.android', 'avd'),
    ];

    if (globals.platform.isWindows) {
      final String homeDrive = globals.platform.environment['HOMEDRIVE'];
      final String homePath = globals.platform.environment['HOMEPATH'];

      if (homeDrive != null && homePath != null) {
        // Can't use path.join for HOMEDRIVE/HOMEPATH
        // https://github.com/dart-lang/path/issues/37
        final String home = homeDrive + homePath;
        searchPaths.add(globals.fs.path.join(home, '.android', 'avd'));
      }
    }

    return searchPaths.where((String p) => p != null).firstWhere(
      (String p) => globals.fs.directory(p).existsSync(),
      orElse: () => null,
    );
  }

201
  Directory get _platformsDir => globals.fs.directory(globals.fs.path.join(directory, 'platforms'));
202 203 204 205 206 207 208 209 210 211 212

  Iterable<Directory> get _platforms {
    Iterable<Directory> platforms = <Directory>[];
    if (_platformsDir.existsSync()) {
      platforms = _platformsDir
        .listSync()
        .whereType<Directory>();
    }
    return platforms;
  }

213 214
  /// Validate the Android SDK. This returns an empty list if there are no
  /// issues; otherwise, it returns a list of issues found.
215
  List<String> validateSdkWellFormed() {
216
    if (adbPath == null || !globals.processManager.canRun(adbPath)) {
217
      return <String>['Android SDK file not found: ${adbPath ?? 'adb'}.'];
218
    }
219

220 221 222 223 224 225 226 227 228 229 230 231
    if (sdkVersions.isEmpty || latestVersion == null) {
      final StringBuffer msg = StringBuffer('No valid Android SDK platforms found in ${_platformsDir.path}.');
      if (_platforms.isEmpty) {
        msg.write(' Directory was empty.');
      } else {
        msg.write(' Candidates were:\n');
        msg.write(_platforms
          .map((Directory dir) => '  - ${dir.basename}')
          .join('\n'));
      }
      return <String>[msg.toString()];
    }
232

233
    return latestVersion.validateSdkWellFormed();
234 235 236
  }

  String getPlatformToolsPath(String binaryName) {
237 238
    final String path = globals.fs.path.join(directory, 'platform-tools', binaryName);
    if (globals.fs.file(path).existsSync()) {
239
      return path;
240
    }
241
    return null;
242 243
  }

Danny Tuppeny's avatar
Danny Tuppeny committed
244
  String getEmulatorPath() {
245
    final String binaryName = globals.platform.isWindows ? 'emulator.exe' : 'emulator';
Danny Tuppeny's avatar
Danny Tuppeny committed
246 247 248 249
    // Emulator now lives inside "emulator" but used to live inside "tools" so
    // try both.
    final List<String> searchFolders = <String>['emulator', 'tools'];
    for (final String folder in searchFolders) {
250 251
      final String path = globals.fs.path.join(directory, folder, binaryName);
      if (globals.fs.file(path).existsSync()) {
Danny Tuppeny's avatar
Danny Tuppeny committed
252
        return path;
253
      }
Danny Tuppeny's avatar
Danny Tuppeny committed
254 255
    }
    return null;
256 257
  }

258
  String getAvdManagerPath() {
259 260 261
    final String binaryName = globals.platform.isWindows ? 'avdmanager.bat' : 'avdmanager';
    final String path = globals.fs.path.join(directory, 'tools', 'bin', binaryName);
    if (globals.fs.file(path).existsSync()) {
262
      return path;
263
    }
264 265 266
    return null;
  }

267 268 269 270 271
  /// Sets up various paths used internally.
  ///
  /// This method should be called in a case where the tooling may have updated
  /// SDK artifacts, such as after running a gradle build.
  void reinitialize() {
272
    List<Version> buildTools = <Version>[]; // 19.1.0, 22.0.1, ...
273

274
    final Directory buildToolsDir = globals.fs.directory(globals.fs.path.join(directory, 'build-tools'));
275
    if (buildToolsDir.existsSync()) {
276
      buildTools = buildToolsDir
277 278 279
        .listSync()
        .map((FileSystemEntity entity) {
          try {
280
            return Version.parse(entity.basename);
281
          } on Exception {
282 283 284 285 286 287 288
            return null;
          }
        })
        .where((Version version) => version != null)
        .toList();
    }

289
    // Match up platforms with the best corresponding build-tools.
290
    _sdkVersions = _platforms.map<AndroidSdkVersion>((Directory platformDir) {
291
      final String platformName = platformDir.basename;
292
      int platformVersion;
293 294

      try {
295 296 297 298 299 300 301
        final Match numberedVersion = _numberedAndroidPlatformRe.firstMatch(platformName);
        if (numberedVersion != null) {
          platformVersion = int.parse(numberedVersion.group(1));
        } else {
          final String buildProps = platformDir.childFile('build.prop').readAsStringSync();
          final String versionString = const LineSplitter()
              .convert(buildProps)
302
              .map<Match>(_sdkVersionRe.firstMatch)
303 304 305 306
              .firstWhere((Match match) => match != null)
              .group(1);
          platformVersion = int.parse(versionString);
        }
307
      } on Exception {
308 309 310
        return null;
      }

311 312
      Version buildToolsVersion = Version.primary(buildTools.where((Version version) {
        return version.major == platformVersion;
313 314
      }).toList());

315 316
      buildToolsVersion ??= Version.primary(buildTools);

317
      if (buildToolsVersion == null) {
318
        return null;
319
      }
320

321
      return AndroidSdkVersion._(
322
        this,
323 324 325
        sdkLevel: platformVersion,
        platformName: platformName,
        buildToolsVersion: buildToolsVersion,
326
      );
327 328 329 330 331 332 333
    }).where((AndroidSdkVersion version) => version != null).toList();

    _sdkVersions.sort();

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

334 335 336 337
  /// Returns the filesystem path of the Android SDK manager tool.
  ///
  /// The sdkmanager was previously in the tools directory but this component
  /// was marked as obsolete in 3.6.
338
  String get sdkManagerPath {
339
    final File cmdlineTool = globals.fs.file(
340 341 342 343 344
      globals.fs.path.join(directory, 'cmdline-tools', 'latest', 'bin',
        globals.platform.isWindows
          ? 'sdkmanager.bat'
          : 'sdkmanager'
      ),
345 346 347 348
    );
    if (cmdlineTool.existsSync()) {
      return cmdlineTool.path;
    }
349
    return globals.fs.path.join(directory, 'tools', 'bin', 'sdkmanager');
350 351
  }

352
  /// First try Java bundled with Android Studio, then sniff JAVA_HOME, then fallback to PATH.
353 354 355 356 357 358 359 360
  static String findJavaBinary({
    @required AndroidStudio androidStudio,
    @required FileSystem fileSystem,
    @required OperatingSystemUtils operatingSystemUtils,
    @required Platform platform,
  }) {
    if (androidStudio?.javaPath != null) {
      return fileSystem.path.join(androidStudio.javaPath, 'bin', 'java');
361
    }
362

363
    final String javaHomeEnv = platform.environment[_javaHomeEnvironmentVariable];
364 365
    if (javaHomeEnv != null) {
      // Trust JAVA_HOME.
366
      return fileSystem.path.join(javaHomeEnv, 'bin', 'java');
367 368 369 370
    }

    // MacOS specific logic to avoid popping up a dialog window.
    // See: http://stackoverflow.com/questions/14292698/how-do-i-check-if-the-java-jdk-is-installed-on-mac.
371
    if (platform.isMacOS) {
372
      try {
373
        final String javaHomeOutput = globals.processUtils.runSync(
374
          <String>['/usr/libexec/java_home', '-v', '1.8'],
375 376 377
          throwOnError: true,
          hideStdout: true,
        ).stdout.trim();
378
        if (javaHomeOutput != null) {
379 380
          if ((javaHomeOutput != null) && (javaHomeOutput.isNotEmpty)) {
            final String javaHome = javaHomeOutput.split('\n').last.trim();
381
            return fileSystem.path.join(javaHome, 'bin', 'java');
382 383
          }
        }
384
      } on Exception catch (_) { /* ignore */ }
385 386 387
    }

    // Fallback to PATH based lookup.
388
    return operatingSystemUtils.which(_javaExecutable)?.path;
389 390 391
  }

  Map<String, String> _sdkManagerEnv;
392 393
  /// Returns an environment with the Java folder added to PATH for use in calling
  /// Java-based Android SDK commands such as sdkmanager and avdmanager.
394 395 396 397
  Map<String, String> get sdkManagerEnv {
    if (_sdkManagerEnv == null) {
      // If we can locate Java, then add it to the path used to run the Android SDK manager.
      _sdkManagerEnv = <String, String>{};
398 399 400 401 402 403
      final String javaBinary = findJavaBinary(
        androidStudio: globals.androidStudio,
        fileSystem: globals.fs,
        operatingSystemUtils: globals.os,
        platform: globals.platform,
      );
404
      if (javaBinary != null) {
405 406 407
        _sdkManagerEnv['PATH'] = globals.fs.path.dirname(javaBinary) +
                                 globals.os.pathVarSeparator +
                                 globals.platform.environment['PATH'];
408 409 410 411 412
      }
    }
    return _sdkManagerEnv;
  }

413 414
  /// Returns the version of the Android SDK manager tool or null if not found.
  String get sdkManagerVersion {
415
    if (!globals.processManager.canRun(sdkManagerPath)) {
416
      throwToolExit('Android sdkmanager not found. Update to the latest Android SDK to resolve this.');
417
    }
418
    final RunResult result = globals.processUtils.runSync(
419 420 421
      <String>[sdkManagerPath, '--version'],
      environment: sdkManagerEnv,
    );
422
    if (result.exitCode != 0) {
423
      globals.printTrace('sdkmanager --version failed: exitCode: ${result.exitCode} stdout: ${result.stdout} stderr: ${result.stderr}');
424
      return null;
425 426 427 428
    }
    return result.stdout.trim();
  }

429
  @override
430 431 432 433
  String toString() => 'AndroidSdk: $directory';
}

class AndroidSdkVersion implements Comparable<AndroidSdkVersion> {
434 435
  AndroidSdkVersion._(
    this.sdk, {
436 437 438 439 440 441
    @required this.sdkLevel,
    @required this.platformName,
    @required this.buildToolsVersion,
  }) : assert(sdkLevel != null),
       assert(platformName != null),
       assert(buildToolsVersion != null);
442 443

  final AndroidSdk sdk;
444 445
  final int sdkLevel;
  final String platformName;
446
  final Version buildToolsVersion;
447

448
  String get buildToolsVersionName => buildToolsVersion.toString();
449 450 451 452 453

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

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

454
  List<String> validateSdkWellFormed() {
455
    if (_exists(androidJarPath) != null) {
456
      return <String>[_exists(androidJarPath)];
457
    }
458

459
    if (_canRun(aaptPath) != null) {
460
      return <String>[_canRun(aaptPath)];
461
    }
462 463

    return <String>[];
464 465 466
  }

  String getPlatformsPath(String itemName) {
467
    return globals.fs.path.join(sdk.directory, 'platforms', platformName, itemName);
468 469
  }

470
  String getBuildToolsPath(String binaryName) {
471
    return globals.fs.path.join(sdk.directory, 'build-tools', buildToolsVersionName, binaryName);
472 473
  }

474
  @override
475
  int compareTo(AndroidSdkVersion other) => sdkLevel - other.sdkLevel;
476

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

480
  String _exists(String path) {
481
    if (!globals.fs.isFileSync(path)) {
482
      return 'Android SDK file not found: $path.';
483
    }
484
    return null;
485
  }
486 487

  String _canRun(String path) {
488
    if (!globals.processManager.canRun(path)) {
489
      return 'Android SDK file not found: $path.';
490
    }
491 492
    return null;
  }
493
}