|  | // 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. | 
|  |  | 
|  | import 'dart:async'; | 
|  |  | 
|  | import 'package:meta/meta.dart'; | 
|  | import 'package:usage/usage_io.dart'; | 
|  |  | 
|  | import 'base/context.dart'; | 
|  | import 'base/file_system.dart'; | 
|  | import 'base/os.dart'; | 
|  | import 'base/platform.dart'; | 
|  | import 'base/utils.dart'; | 
|  | import 'globals.dart'; | 
|  | import 'version.dart'; | 
|  |  | 
|  | const String _kFlutterUA = 'UA-67589403-6'; | 
|  | const String kEventReloadReasonParameterName = 'cd5'; | 
|  | const String kEventReloadFinalLibraryCount = 'cd6'; | 
|  | const String kEventReloadSyncedLibraryCount = 'cd7'; | 
|  | const String kEventReloadSyncedClassesCount = 'cd8'; | 
|  | const String kEventReloadSyncedProceduresCount = 'cd9'; | 
|  | const String kEventReloadSyncedBytes = 'cd10'; | 
|  | const String kEventReloadInvalidatedSourcesCount = 'cd11'; | 
|  | const String kEventReloadTransferTimeInMs = 'cd12'; | 
|  | const String kEventReloadOverallTimeInMs = 'cd13'; | 
|  |  | 
|  | Usage get flutterUsage => Usage.instance; | 
|  |  | 
|  | class Usage { | 
|  | /// Create a new Usage instance; [versionOverride] and [configDirOverride] are | 
|  | /// used for testing. | 
|  | Usage({ String settingsName = 'flutter', String versionOverride, String configDirOverride}) { | 
|  | final FlutterVersion flutterVersion = FlutterVersion.instance; | 
|  | final String version = versionOverride ?? flutterVersion.getVersionString(redactUnknownBranches: true); | 
|  | _analytics = AnalyticsIO(_kFlutterUA, settingsName, version, | 
|  | documentDirectory: configDirOverride != null ? fs.directory(configDirOverride) : null); | 
|  |  | 
|  | // Report a more detailed OS version string than package:usage does by default. | 
|  | _analytics.setSessionValue('cd1', os.name); | 
|  | // Send the branch name as the "channel". | 
|  | _analytics.setSessionValue('cd2', flutterVersion.getBranchName(redactUnknownBranches: true)); | 
|  | // Record the host as the application installer ID - the context that flutter_tools is running in. | 
|  | if (platform.environment.containsKey('FLUTTER_HOST')) { | 
|  | _analytics.setSessionValue('aiid', platform.environment['FLUTTER_HOST']); | 
|  | } | 
|  | _analytics.analyticsOpt = AnalyticsOpt.optOut; | 
|  |  | 
|  | // Many CI systems don't do a full git checkout. | 
|  | if (version.endsWith('/unknown') || isRunningOnBot) { | 
|  | // If we think we're running on a CI system, suppress sending analytics. | 
|  | suppressAnalytics = true; | 
|  | } | 
|  | } | 
|  |  | 
|  | /// Returns [Usage] active in the current app context. | 
|  | static Usage get instance => context.get<Usage>(); | 
|  |  | 
|  | Analytics _analytics; | 
|  |  | 
|  | bool _printedWelcome = false; | 
|  | bool _suppressAnalytics = false; | 
|  |  | 
|  | bool get isFirstRun => _analytics.firstRun; | 
|  |  | 
|  | bool get enabled => _analytics.enabled; | 
|  |  | 
|  | bool get suppressAnalytics => _suppressAnalytics || _analytics.firstRun; | 
|  |  | 
|  | /// Suppress analytics for this session. | 
|  | set suppressAnalytics(bool value) { | 
|  | _suppressAnalytics = value; | 
|  | } | 
|  |  | 
|  | /// Enable or disable reporting analytics. | 
|  | set enabled(bool value) { | 
|  | _analytics.enabled = value; | 
|  | } | 
|  |  | 
|  | /// A stable randomly generated UUID used to deduplicate multiple identical | 
|  | /// reports coming from the same computer. | 
|  | String get clientId => _analytics.clientId; | 
|  |  | 
|  | void sendCommand(String command, { Map<String, String> parameters }) { | 
|  | if (suppressAnalytics) | 
|  | return; | 
|  |  | 
|  | parameters ??= const <String, String>{}; | 
|  |  | 
|  | _analytics.sendScreenView(command, parameters: parameters); | 
|  | } | 
|  |  | 
|  | void sendEvent( | 
|  | String category, | 
|  | String parameter, { | 
|  | Map<String, String> parameters, | 
|  | }) { | 
|  | if (suppressAnalytics) | 
|  | return; | 
|  |  | 
|  | parameters ??= const <String, String>{}; | 
|  |  | 
|  | _analytics.sendEvent(category, parameter, parameters: parameters); | 
|  | } | 
|  |  | 
|  | void sendTiming( | 
|  | String category, | 
|  | String variableName, | 
|  | Duration duration, { | 
|  | String label, | 
|  | }) { | 
|  | if (!suppressAnalytics) { | 
|  | _analytics.sendTiming( | 
|  | variableName, | 
|  | duration.inMilliseconds, | 
|  | category: category, | 
|  | label: label, | 
|  | ); | 
|  | } | 
|  | } | 
|  |  | 
|  | void sendException(dynamic exception, StackTrace trace) { | 
|  | if (!suppressAnalytics) | 
|  | _analytics.sendException('${exception.runtimeType}\n${sanitizeStacktrace(trace)}'); | 
|  | } | 
|  |  | 
|  | /// Fires whenever analytics data is sent over the network. | 
|  | @visibleForTesting | 
|  | Stream<Map<String, dynamic>> get onSend => _analytics.onSend; | 
|  |  | 
|  | /// Returns when the last analytics event has been sent, or after a fixed | 
|  | /// (short) delay, whichever is less. | 
|  | Future<void> ensureAnalyticsSent() async { | 
|  | // TODO(devoncarew): This may delay tool exit and could cause some analytics | 
|  | // events to not be reported. Perhaps we could send the analytics pings | 
|  | // out-of-process from flutter_tools? | 
|  | await _analytics.waitForLastPing(timeout: const Duration(milliseconds: 250)); | 
|  | } | 
|  |  | 
|  | void printWelcome() { | 
|  | // This gets called if it's the first run by the selected command, if any, | 
|  | // and on exit, in case there was no command. | 
|  | if (_printedWelcome) | 
|  | return; | 
|  | _printedWelcome = true; | 
|  |  | 
|  | printStatus(''); | 
|  | printStatus(''' | 
|  | ╔════════════════════════════════════════════════════════════════════════════╗ | 
|  | ║                 Welcome to Flutter! - https://flutter.dev                  ║ | 
|  | ║                                                                            ║ | 
|  | ║ The Flutter tool anonymously reports feature usage statistics and crash    ║ | 
|  | ║ reports to Google in order to help Google contribute improvements to       ║ | 
|  | ║ Flutter over time.                                                         ║ | 
|  | ║                                                                            ║ | 
|  | ║ Read about data we send with crash reports:                                ║ | 
|  | ║ https://github.com/flutter/flutter/wiki/Flutter-CLI-crash-reporting        ║ | 
|  | ║                                                                            ║ | 
|  | ║ See Google's privacy policy:                                               ║ | 
|  | ║ https://www.google.com/intl/en/policies/privacy/                           ║ | 
|  | ║                                                                            ║ | 
|  | ║ Use "flutter config --no-analytics" to disable analytics and crash         ║ | 
|  | ║ reporting.                                                                 ║ | 
|  | ╚════════════════════════════════════════════════════════════════════════════╝ | 
|  | ''', emphasis: true); | 
|  | } | 
|  | } |