Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Sign in
Toggle navigation
F
Front-End
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
abdullh.alsoleman
Front-End
Commits
b9ecebf1
Unverified
Commit
b9ecebf1
authored
Feb 12, 2020
by
Zachary Anderson
Committed by
GitHub
Feb 12, 2020
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Reland: [flutter_tools] Disable analytics for more bots (#50641)
parent
3aa72cc2
Changes
12
Show whitespace changes
Inline
Side-by-side
Showing
12 changed files
with
205 additions
and
62 deletions
+205
-62
runner.dart
packages/flutter_tools/lib/runner.dart
+2
-3
bot_detector.dart
packages/flutter_tools/lib/src/base/bot_detector.dart
+82
-31
context_runner.dart
packages/flutter_tools/lib/src/context_runner.dart
+13
-2
pub.dart
packages/flutter_tools/lib/src/dart/pub.dart
+7
-8
globals.dart
packages/flutter_tools/lib/src/globals.dart
+11
-0
reporting.dart
packages/flutter_tools/lib/src/reporting/reporting.dart
+0
-1
usage.dart
packages/flutter_tools/lib/src/reporting/usage.dart
+5
-2
packages_test.dart
...er_tools/test/commands.shard/permeable/packages_test.dart
+2
-2
analytics_test.dart
...ages/flutter_tools/test/general.shard/analytics_test.dart
+15
-5
bot_detector_test.dart
...tter_tools/test/general.shard/base/bot_detector_test.dart
+63
-6
pub_get_test.dart
...s/flutter_tools/test/general.shard/dart/pub_get_test.dart
+1
-1
runner_test.dart
.../flutter_tools/test/general.shard/runner/runner_test.dart
+4
-1
No files found.
packages/flutter_tools/lib/runner.dart
View file @
b9ecebf1
...
@@ -9,7 +9,6 @@ import 'package:intl/intl.dart' as intl;
...
@@ -9,7 +9,6 @@ import 'package:intl/intl.dart' as intl;
import
'package:intl/intl_standalone.dart'
as
intl_standalone
;
import
'package:intl/intl_standalone.dart'
as
intl_standalone
;
import
'package:meta/meta.dart'
;
import
'package:meta/meta.dart'
;
import
'src/base/bot_detector.dart'
;
import
'src/base/common.dart'
;
import
'src/base/common.dart'
;
import
'src/base/context.dart'
;
import
'src/base/context.dart'
;
import
'src/base/file_system.dart'
;
import
'src/base/file_system.dart'
;
...
@@ -35,8 +34,8 @@ Future<int> run(
...
@@ -35,8 +34,8 @@ Future<int> run(
bool
reportCrashes
,
bool
reportCrashes
,
String
flutterVersion
,
String
flutterVersion
,
Map
<
Type
,
Generator
>
overrides
,
Map
<
Type
,
Generator
>
overrides
,
})
{
})
async
{
reportCrashes
??=
!
isRunningOnBot
(
globals
.
platform
)
;
reportCrashes
??=
!
await
globals
.
isRunningOnBot
;
if
(
muteCommandLogging
)
{
if
(
muteCommandLogging
)
{
// Remove the verbose option; for help and doctor, users don't need to see
// Remove the verbose option; for help and doctor, users don't need to see
...
...
packages/flutter_tools/lib/src/base/bot_detector.dart
View file @
b9ecebf1
...
@@ -5,16 +5,28 @@
...
@@ -5,16 +5,28 @@
import
'package:meta/meta.dart'
;
import
'package:meta/meta.dart'
;
import
'package:platform/platform.dart'
;
import
'package:platform/platform.dart'
;
import
'context.dart'
;
import
'io.dart'
;
import
'net.dart'
;
class
BotDetector
{
class
BotDetector
{
BotDetector
({
BotDetector
({
@required
HttpClientFactory
httpClientFactory
,
@required
Platform
platform
,
@required
Platform
platform
,
})
:
_platform
=
platform
;
})
:
_platform
=
platform
,
_azureDetector
=
AzureDetector
(
httpClientFactory:
httpClientFactory
,
);
final
Platform
_platform
;
final
Platform
_platform
;
final
AzureDetector
_azureDetector
;
bool
get
isRunningOnBot
{
bool
_isRunningOnBot
;
Future
<
bool
>
get
isRunningOnBot
async
{
if
(
_isRunningOnBot
!=
null
)
{
return
_isRunningOnBot
;
}
if
(
if
(
// Explicitly stated to not be a bot.
// Explicitly stated to not be a bot.
_platform
.
environment
[
'BOT'
]
==
'false'
_platform
.
environment
[
'BOT'
]
==
'false'
...
@@ -24,10 +36,10 @@ class BotDetector {
...
@@ -24,10 +36,10 @@ class BotDetector {
// When set, GA logs to a local file (normally for tests) so we don't need to filter.
// When set, GA logs to a local file (normally for tests) so we don't need to filter.
||
_platform
.
environment
.
containsKey
(
'FLUTTER_ANALYTICS_LOG_FILE'
)
||
_platform
.
environment
.
containsKey
(
'FLUTTER_ANALYTICS_LOG_FILE'
)
)
{
)
{
return
false
;
return
_isRunningOnBot
=
false
;
}
}
return
_platform
.
environment
[
'BOT'
]
==
'true'
return
_
isRunningOnBot
=
_
platform
.
environment
[
'BOT'
]
==
'true'
// https://docs.travis-ci.com/user/environment-variables/#Default-Environment-Variables
// https://docs.travis-ci.com/user/environment-variables/#Default-Environment-Variables
||
_platform
.
environment
[
'TRAVIS'
]
==
'true'
||
_platform
.
environment
[
'TRAVIS'
]
==
'true'
...
@@ -47,14 +59,53 @@ class BotDetector {
...
@@ -47,14 +59,53 @@ class BotDetector {
// https://wiki.jenkins.io/display/JENKINS/Building+a+software+project#Buildingasoftwareproject-belowJenkinsSetEnvironmentVariables
// https://wiki.jenkins.io/display/JENKINS/Building+a+software+project#Buildingasoftwareproject-belowJenkinsSetEnvironmentVariables
||
_platform
.
environment
.
containsKey
(
'JENKINS_URL'
)
||
_platform
.
environment
.
containsKey
(
'JENKINS_URL'
)
// https://help.github.com/en/actions/configuring-and-managing-workflows/using-environment-variables#default-environment-variables
||
_platform
.
environment
.
containsKey
(
'GITHUB_ACTIONS'
)
// Properties on Flutter's Chrome Infra bots.
// Properties on Flutter's Chrome Infra bots.
||
_platform
.
environment
[
'CHROME_HEADLESS'
]
==
'1'
||
_platform
.
environment
[
'CHROME_HEADLESS'
]
==
'1'
||
_platform
.
environment
.
containsKey
(
'BUILDBOT_BUILDERNAME'
)
||
_platform
.
environment
.
containsKey
(
'BUILDBOT_BUILDERNAME'
)
||
_platform
.
environment
.
containsKey
(
'SWARMING_TASK_ID'
);
||
_platform
.
environment
.
containsKey
(
'SWARMING_TASK_ID'
)
||
await
_azureDetector
.
isRunningOnAzure
;
}
}
}
}
bool
isRunningOnBot
(
Platform
platform
)
{
// Are we running on Azure?
final
BotDetector
botDetector
=
context
.
get
<
BotDetector
>()
??
BotDetector
(
platform:
platform
);
// https://docs.microsoft.com/en-us/azure/virtual-machines/linux/instance-metadata-service
return
botDetector
.
isRunningOnBot
;
@visibleForTesting
class
AzureDetector
{
AzureDetector
({
@required
HttpClientFactory
httpClientFactory
,
})
:
_httpClientFactory
=
httpClientFactory
;
static
const
String
_serviceUrl
=
'http://169.254.169.254/metadata/instance'
;
final
HttpClientFactory
_httpClientFactory
;
bool
_isRunningOnAzure
;
Future
<
bool
>
get
isRunningOnAzure
async
{
if
(
_isRunningOnAzure
!=
null
)
{
return
_isRunningOnAzure
;
}
final
HttpClient
client
=
_httpClientFactory
()
..
connectionTimeout
=
const
Duration
(
seconds:
1
);
try
{
final
HttpClientRequest
request
=
await
client
.
getUrl
(
Uri
.
parse
(
_serviceUrl
),
);
request
.
headers
.
add
(
'Metadata'
,
true
);
await
request
.
close
();
}
on
SocketException
{
// If there is an error on the socket, it probalby means that we are not
// running on Azure.
return
_isRunningOnAzure
=
false
;
}
on
HttpException
{
// If the connection gets set up, but encounters an error condition, it
// still means we're on Azure.
return
_isRunningOnAzure
=
true
;
}
// We got a response. We're running on Azure.
return
_isRunningOnAzure
=
true
;
}
}
}
packages/flutter_tools/lib/src/context_runner.dart
View file @
b9ecebf1
...
@@ -61,9 +61,18 @@ Future<T> runInContext<T>(
...
@@ -61,9 +61,18 @@ Future<T> runInContext<T>(
FutureOr
<
T
>
runner
(),
{
FutureOr
<
T
>
runner
(),
{
Map
<
Type
,
Generator
>
overrides
,
Map
<
Type
,
Generator
>
overrides
,
})
async
{
})
async
{
// Wrap runner with any asynchronous initialization that should run with the
// overrides and callbacks.
bool
runningOnBot
;
FutureOr
<
T
>
runnerWrapper
()
async
{
runningOnBot
=
await
globals
.
isRunningOnBot
;
return
runner
();
}
return
await
context
.
run
<
T
>(
return
await
context
.
run
<
T
>(
name:
'global fallbacks'
,
name:
'global fallbacks'
,
body:
runner
,
body:
runner
Wrapper
,
overrides:
overrides
,
overrides:
overrides
,
fallbacks:
<
Type
,
Generator
>{
fallbacks:
<
Type
,
Generator
>{
AndroidLicenseValidator:
()
=>
AndroidLicenseValidator
(),
AndroidLicenseValidator:
()
=>
AndroidLicenseValidator
(),
...
@@ -159,7 +168,9 @@ Future<T> runInContext<T>(
...
@@ -159,7 +168,9 @@ Future<T> runInContext<T>(
Stdio:
()
=>
const
Stdio
(),
Stdio:
()
=>
const
Stdio
(),
SystemClock:
()
=>
const
SystemClock
(),
SystemClock:
()
=>
const
SystemClock
(),
TimeoutConfiguration:
()
=>
const
TimeoutConfiguration
(),
TimeoutConfiguration:
()
=>
const
TimeoutConfiguration
(),
Usage:
()
=>
Usage
(),
Usage:
()
=>
Usage
(
runningOnBot:
runningOnBot
,
),
UserMessages:
()
=>
UserMessages
(),
UserMessages:
()
=>
UserMessages
(),
VisualStudio:
()
=>
VisualStudio
(),
VisualStudio:
()
=>
VisualStudio
(),
VisualStudioValidator:
()
=>
const
VisualStudioValidator
(),
VisualStudioValidator:
()
=>
const
VisualStudioValidator
(),
...
...
packages/flutter_tools/lib/src/dart/pub.dart
View file @
b9ecebf1
...
@@ -6,7 +6,6 @@ import 'dart:async';
...
@@ -6,7 +6,6 @@ import 'dart:async';
import
'package:meta/meta.dart'
;
import
'package:meta/meta.dart'
;
import
'../base/bot_detector.dart'
;
import
'../base/common.dart'
;
import
'../base/common.dart'
;
import
'../base/context.dart'
;
import
'../base/context.dart'
;
import
'../base/file_system.dart'
;
import
'../base/file_system.dart'
;
...
@@ -232,7 +231,7 @@ class _DefaultPub implements Pub {
...
@@ -232,7 +231,7 @@ class _DefaultPub implements Pub {
@required
bool
retry
,
@required
bool
retry
,
bool
showTraceForErrors
,
bool
showTraceForErrors
,
})
async
{
})
async
{
showTraceForErrors
??=
isRunningOnBot
(
globals
.
platform
)
;
showTraceForErrors
??=
await
globals
.
isRunningOnBot
;
String
lastPubMessage
=
'no message'
;
String
lastPubMessage
=
'no message'
;
bool
versionSolvingFailed
=
false
;
bool
versionSolvingFailed
=
false
;
...
@@ -259,7 +258,7 @@ class _DefaultPub implements Pub {
...
@@ -259,7 +258,7 @@ class _DefaultPub implements Pub {
_pubCommand
(
arguments
),
_pubCommand
(
arguments
),
workingDirectory:
directory
,
workingDirectory:
directory
,
mapFunction:
filterWrapper
,
// may set versionSolvingFailed, lastPubMessage
mapFunction:
filterWrapper
,
// may set versionSolvingFailed, lastPubMessage
environment:
_createPubEnvironment
(
context
),
environment:
await
_createPubEnvironment
(
context
),
);
);
String
message
;
String
message
;
switch
(
code
)
{
switch
(
code
)
{
...
@@ -304,7 +303,7 @@ class _DefaultPub implements Pub {
...
@@ -304,7 +303,7 @@ class _DefaultPub implements Pub {
final
io
.
Process
process
=
await
processUtils
.
start
(
final
io
.
Process
process
=
await
processUtils
.
start
(
_pubCommand
(
arguments
),
_pubCommand
(
arguments
),
workingDirectory:
directory
,
workingDirectory:
directory
,
environment:
_createPubEnvironment
(
PubContext
.
interactive
),
environment:
await
_createPubEnvironment
(
PubContext
.
interactive
),
);
);
// Pipe the Flutter tool stdin to the pub stdin.
// Pipe the Flutter tool stdin to the pub stdin.
...
@@ -348,10 +347,10 @@ typedef MessageFilter = String Function(String message);
...
@@ -348,10 +347,10 @@ typedef MessageFilter = String Function(String message);
///
///
/// [context] provides extra information to package server requests to
/// [context] provides extra information to package server requests to
/// understand usage.
/// understand usage.
Map
<
String
,
String
>
_createPubEnvironment
(
PubContext
context
)
{
Future
<
Map
<
String
,
String
>>
_createPubEnvironment
(
PubContext
context
)
async
{
final
Map
<
String
,
String
>
environment
=
<
String
,
String
>{
final
Map
<
String
,
String
>
environment
=
<
String
,
String
>{
'FLUTTER_ROOT'
:
Cache
.
flutterRoot
,
'FLUTTER_ROOT'
:
Cache
.
flutterRoot
,
_pubEnvironmentKey:
_getPubEnvironmentValue
(
context
),
_pubEnvironmentKey:
await
_getPubEnvironmentValue
(
context
),
};
};
final
String
pubCache
=
_getRootPubCacheIfAvailable
();
final
String
pubCache
=
_getRootPubCacheIfAvailable
();
if
(
pubCache
!=
null
)
{
if
(
pubCache
!=
null
)
{
...
@@ -374,13 +373,13 @@ const String _pubCacheEnvironmentKey = 'PUB_CACHE';
...
@@ -374,13 +373,13 @@ const String _pubCacheEnvironmentKey = 'PUB_CACHE';
///
///
/// [context] provides extra information to package server requests to
/// [context] provides extra information to package server requests to
/// understand usage.
/// understand usage.
String
_getPubEnvironmentValue
(
PubContext
pubContext
)
{
Future
<
String
>
_getPubEnvironmentValue
(
PubContext
pubContext
)
async
{
// DO NOT update this function without contacting kevmoo.
// DO NOT update this function without contacting kevmoo.
// We have server-side tooling that assumes the values are consistent.
// We have server-side tooling that assumes the values are consistent.
final
String
existing
=
globals
.
platform
.
environment
[
_pubEnvironmentKey
];
final
String
existing
=
globals
.
platform
.
environment
[
_pubEnvironmentKey
];
final
List
<
String
>
values
=
<
String
>[
final
List
<
String
>
values
=
<
String
>[
if
(
existing
!=
null
&&
existing
.
isNotEmpty
)
existing
,
if
(
existing
!=
null
&&
existing
.
isNotEmpty
)
existing
,
if
(
isRunningOnBot
(
globals
.
platform
)
)
'flutter_bot'
,
if
(
await
globals
.
isRunningOnBot
)
'flutter_bot'
,
'flutter_cli'
,
'flutter_cli'
,
...
pubContext
.
_values
,
...
pubContext
.
_values
,
];
];
...
...
packages/flutter_tools/lib/src/globals.dart
View file @
b9ecebf1
...
@@ -8,12 +8,14 @@ import 'package:process/process.dart';
...
@@ -8,12 +8,14 @@ import 'package:process/process.dart';
import
'android/android_sdk.dart'
;
import
'android/android_sdk.dart'
;
import
'android/android_studio.dart'
;
import
'android/android_studio.dart'
;
import
'artifacts.dart'
;
import
'artifacts.dart'
;
import
'base/bot_detector.dart'
;
import
'base/config.dart'
;
import
'base/config.dart'
;
import
'base/context.dart'
;
import
'base/context.dart'
;
import
'base/error_handling_file_system.dart'
;
import
'base/error_handling_file_system.dart'
;
import
'base/file_system.dart'
;
import
'base/file_system.dart'
;
import
'base/io.dart'
;
import
'base/io.dart'
;
import
'base/logger.dart'
;
import
'base/logger.dart'
;
import
'base/net.dart'
;
import
'base/os.dart'
;
import
'base/os.dart'
;
import
'base/terminal.dart'
;
import
'base/terminal.dart'
;
import
'base/user_messages.dart'
;
import
'base/user_messages.dart'
;
...
@@ -65,6 +67,15 @@ Xcode get xcode => context.get<Xcode>();
...
@@ -65,6 +67,15 @@ Xcode get xcode => context.get<Xcode>();
XCDevice
get
xcdevice
=>
context
.
get
<
XCDevice
>();
XCDevice
get
xcdevice
=>
context
.
get
<
XCDevice
>();
final
BotDetector
_defaultBotDetector
=
BotDetector
(
httpClientFactory:
context
.
get
<
HttpClientFactory
>()
??
()
=>
HttpClient
(),
platform:
platform
,
);
BotDetector
get
botDetector
=>
context
.
get
<
BotDetector
>()
??
_defaultBotDetector
;
Future
<
bool
>
get
isRunningOnBot
=>
botDetector
.
isRunningOnBot
;
/// Display an error level message to the user. Commands should use this if they
/// Display an error level message to the user. Commands should use this if they
/// fail in some way.
/// fail in some way.
///
///
...
...
packages/flutter_tools/lib/src/reporting/reporting.dart
View file @
b9ecebf1
...
@@ -10,7 +10,6 @@ import 'package:http/http.dart' as http;
...
@@ -10,7 +10,6 @@ import 'package:http/http.dart' as http;
import
'package:meta/meta.dart'
;
import
'package:meta/meta.dart'
;
import
'package:usage/usage_io.dart'
;
import
'package:usage/usage_io.dart'
;
import
'../base/bot_detector.dart'
;
import
'../base/context.dart'
;
import
'../base/context.dart'
;
import
'../base/file_system.dart'
;
import
'../base/file_system.dart'
;
import
'../base/io.dart'
;
import
'../base/io.dart'
;
...
...
packages/flutter_tools/lib/src/reporting/usage.dart
View file @
b9ecebf1
...
@@ -76,10 +76,12 @@ abstract class Usage {
...
@@ -76,10 +76,12 @@ abstract class Usage {
String
versionOverride
,
String
versionOverride
,
String
configDirOverride
,
String
configDirOverride
,
String
logFile
,
String
logFile
,
@required
bool
runningOnBot
,
})
=>
_DefaultUsage
(
settingsName:
settingsName
,
})
=>
_DefaultUsage
(
settingsName:
settingsName
,
versionOverride:
versionOverride
,
versionOverride:
versionOverride
,
configDirOverride:
configDirOverride
,
configDirOverride:
configDirOverride
,
logFile:
logFile
);
logFile:
logFile
,
runningOnBot:
runningOnBot
);
/// Returns [Usage] active in the current app context.
/// Returns [Usage] active in the current app context.
static
Usage
get
instance
=>
context
.
get
<
Usage
>();
static
Usage
get
instance
=>
context
.
get
<
Usage
>();
...
@@ -161,6 +163,7 @@ class _DefaultUsage implements Usage {
...
@@ -161,6 +163,7 @@ class _DefaultUsage implements Usage {
String
versionOverride
,
String
versionOverride
,
String
configDirOverride
,
String
configDirOverride
,
String
logFile
,
String
logFile
,
@required
bool
runningOnBot
,
})
{
})
{
final
FlutterVersion
flutterVersion
=
globals
.
flutterVersion
;
final
FlutterVersion
flutterVersion
=
globals
.
flutterVersion
;
final
String
version
=
versionOverride
??
flutterVersion
.
getVersionString
(
redactUnknownBranches:
true
);
final
String
version
=
versionOverride
??
flutterVersion
.
getVersionString
(
redactUnknownBranches:
true
);
...
@@ -176,7 +179,7 @@ class _DefaultUsage implements Usage {
...
@@ -176,7 +179,7 @@ class _DefaultUsage implements Usage {
// Many CI systems don't do a full git checkout.
// Many CI systems don't do a full git checkout.
version
.
endsWith
(
'/unknown'
)
||
version
.
endsWith
(
'/unknown'
)
||
// Ignore bots.
// Ignore bots.
isRunningOnBot
(
globals
.
platform
)
||
runningOnBot
||
// Ignore when suppressed by FLUTTER_SUPPRESS_ANALYTICS.
// Ignore when suppressed by FLUTTER_SUPPRESS_ANALYTICS.
suppressEnvFlag
suppressEnvFlag
))
{
))
{
...
...
packages/flutter_tools/test/commands.shard/permeable/packages_test.dart
View file @
b9ecebf1
...
@@ -25,7 +25,7 @@ class AlwaysTrueBotDetector implements BotDetector {
...
@@ -25,7 +25,7 @@ class AlwaysTrueBotDetector implements BotDetector {
const
AlwaysTrueBotDetector
();
const
AlwaysTrueBotDetector
();
@override
@override
bool
get
isRunningOnBot
=>
true
;
Future
<
bool
>
get
isRunningOnBot
async
=>
true
;
}
}
...
@@ -33,7 +33,7 @@ class AlwaysFalseBotDetector implements BotDetector {
...
@@ -33,7 +33,7 @@ class AlwaysFalseBotDetector implements BotDetector {
const
AlwaysFalseBotDetector
();
const
AlwaysFalseBotDetector
();
@override
@override
bool
get
isRunningOnBot
=>
false
;
Future
<
bool
>
get
isRunningOnBot
async
=>
false
;
}
}
...
...
packages/flutter_tools/test/general.shard/analytics_test.dart
View file @
b9ecebf1
...
@@ -69,6 +69,7 @@ void main() {
...
@@ -69,6 +69,7 @@ void main() {
Usage:
()
=>
Usage
(
Usage:
()
=>
Usage
(
configDirOverride:
tempDir
.
path
,
configDirOverride:
tempDir
.
path
,
logFile:
tempDir
.
childFile
(
'analytics.log'
).
path
,
logFile:
tempDir
.
childFile
(
'analytics.log'
).
path
,
runningOnBot:
true
,
),
),
});
});
...
@@ -91,13 +92,14 @@ void main() {
...
@@ -91,13 +92,14 @@ void main() {
Usage:
()
=>
Usage
(
Usage:
()
=>
Usage
(
configDirOverride:
tempDir
.
path
,
configDirOverride:
tempDir
.
path
,
logFile:
tempDir
.
childFile
(
'analytics.log'
).
path
,
logFile:
tempDir
.
childFile
(
'analytics.log'
).
path
,
runningOnBot:
true
,
),
),
});
});
testUsingContext
(
'Usage records one feature in experiment setting'
,
()
async
{
testUsingContext
(
'Usage records one feature in experiment setting'
,
()
async
{
when
<
bool
>(
mockFlutterConfig
.
getValue
(
flutterWebFeature
.
configSetting
)
as
bool
)
when
<
bool
>(
mockFlutterConfig
.
getValue
(
flutterWebFeature
.
configSetting
)
as
bool
)
.
thenReturn
(
true
);
.
thenReturn
(
true
);
final
Usage
usage
=
Usage
();
final
Usage
usage
=
Usage
(
runningOnBot:
true
);
usage
.
sendCommand
(
'test'
);
usage
.
sendCommand
(
'test'
);
final
String
featuresKey
=
cdKey
(
CustomDimensions
.
enabledFlutterFeatures
);
final
String
featuresKey
=
cdKey
(
CustomDimensions
.
enabledFlutterFeatures
);
...
@@ -119,7 +121,7 @@ void main() {
...
@@ -119,7 +121,7 @@ void main() {
.
thenReturn
(
true
);
.
thenReturn
(
true
);
when
<
bool
>(
mockFlutterConfig
.
getValue
(
flutterMacOSDesktopFeature
.
configSetting
)
as
bool
)
when
<
bool
>(
mockFlutterConfig
.
getValue
(
flutterMacOSDesktopFeature
.
configSetting
)
as
bool
)
.
thenReturn
(
true
);
.
thenReturn
(
true
);
final
Usage
usage
=
Usage
();
final
Usage
usage
=
Usage
(
runningOnBot:
true
);
usage
.
sendCommand
(
'test'
);
usage
.
sendCommand
(
'test'
);
final
String
featuresKey
=
cdKey
(
CustomDimensions
.
enabledFlutterFeatures
);
final
String
featuresKey
=
cdKey
(
CustomDimensions
.
enabledFlutterFeatures
);
...
@@ -213,7 +215,10 @@ void main() {
...
@@ -213,7 +215,10 @@ void main() {
mockTimes
=
<
int
>[
kMillis
];
mockTimes
=
<
int
>[
kMillis
];
// Since FLUTTER_ANALYTICS_LOG_FILE is set in the environment, analytics
// Since FLUTTER_ANALYTICS_LOG_FILE is set in the environment, analytics
// will be written to a file.
// will be written to a file.
final
Usage
usage
=
Usage
(
versionOverride:
'test'
);
final
Usage
usage
=
Usage
(
versionOverride:
'test'
,
runningOnBot:
true
,
);
usage
.
suppressAnalytics
=
false
;
usage
.
suppressAnalytics
=
false
;
usage
.
enabled
=
true
;
usage
.
enabled
=
true
;
...
@@ -239,7 +244,10 @@ void main() {
...
@@ -239,7 +244,10 @@ void main() {
mockTimes
=
<
int
>[
kMillis
];
mockTimes
=
<
int
>[
kMillis
];
// Since FLUTTER_ANALYTICS_LOG_FILE is set in the environment, analytics
// Since FLUTTER_ANALYTICS_LOG_FILE is set in the environment, analytics
// will be written to a file.
// will be written to a file.
final
Usage
usage
=
Usage
(
versionOverride:
'test'
);
final
Usage
usage
=
Usage
(
versionOverride:
'test'
,
runningOnBot:
true
,
);
usage
.
suppressAnalytics
=
false
;
usage
.
suppressAnalytics
=
false
;
usage
.
enabled
=
true
;
usage
.
enabled
=
true
;
...
@@ -272,7 +280,7 @@ void main() {
...
@@ -272,7 +280,7 @@ void main() {
tryToDelete
(
tempDir
);
tryToDelete
(
tempDir
);
});
});
testUsingContext
(
"don't send on bots"
,
()
async
{
testUsingContext
(
"don't send on bots
with unknown version
"
,
()
async
{
int
count
=
0
;
int
count
=
0
;
flutterUsage
.
onSend
.
listen
((
Map
<
String
,
dynamic
>
data
)
=>
count
++);
flutterUsage
.
onSend
.
listen
((
Map
<
String
,
dynamic
>
data
)
=>
count
++);
...
@@ -283,6 +291,7 @@ void main() {
...
@@ -283,6 +291,7 @@ void main() {
settingsName:
'flutter_bot_test'
,
settingsName:
'flutter_bot_test'
,
versionOverride:
'dev/unknown'
,
versionOverride:
'dev/unknown'
,
configDirOverride:
tempDir
.
path
,
configDirOverride:
tempDir
.
path
,
runningOnBot:
false
,
),
),
});
});
...
@@ -298,6 +307,7 @@ void main() {
...
@@ -298,6 +307,7 @@ void main() {
settingsName:
'flutter_bot_test'
,
settingsName:
'flutter_bot_test'
,
versionOverride:
'dev/unknown'
,
versionOverride:
'dev/unknown'
,
configDirOverride:
tempDir
.
path
,
configDirOverride:
tempDir
.
path
,
runningOnBot:
false
,
),
),
});
});
});
});
...
...
packages/flutter_tools/test/general.shard/base/bot_detector_test.dart
View file @
b9ecebf1
...
@@ -3,6 +3,8 @@
...
@@ -3,6 +3,8 @@
// found in the LICENSE file.
// found in the LICENSE file.
import
'package:flutter_tools/src/base/bot_detector.dart'
;
import
'package:flutter_tools/src/base/bot_detector.dart'
;
import
'package:flutter_tools/src/base/io.dart'
;
import
'package:mockito/mockito.dart'
;
import
'package:platform/platform.dart'
;
import
'package:platform/platform.dart'
;
import
'../../src/common.dart'
;
import
'../../src/common.dart'
;
...
@@ -12,39 +14,94 @@ void main() {
...
@@ -12,39 +14,94 @@ void main() {
group
(
'BotDetector'
,
()
{
group
(
'BotDetector'
,
()
{
FakePlatform
fakePlatform
;
FakePlatform
fakePlatform
;
MockStdio
mockStdio
;
MockStdio
mockStdio
;
MockHttpClient
mockHttpClient
;
MockHttpClientRequest
mockHttpClientRequest
;
MockHttpHeaders
mockHttpHeaders
;
BotDetector
botDetector
;
BotDetector
botDetector
;
setUp
(()
{
setUp
(()
{
fakePlatform
=
FakePlatform
()..
environment
=
<
String
,
String
>{};
fakePlatform
=
FakePlatform
()..
environment
=
<
String
,
String
>{};
mockStdio
=
MockStdio
();
mockStdio
=
MockStdio
();
botDetector
=
BotDetector
(
platform:
fakePlatform
);
mockHttpClient
=
MockHttpClient
();
mockHttpClientRequest
=
MockHttpClientRequest
();
mockHttpHeaders
=
MockHttpHeaders
();
botDetector
=
BotDetector
(
platform:
fakePlatform
,
httpClientFactory:
()
=>
mockHttpClient
,
);
});
});
group
(
'isRunningOnBot'
,
()
{
group
(
'isRunningOnBot'
,
()
{
testWithoutContext
(
'returns false unconditionally if BOT=false is set'
,
()
async
{
testWithoutContext
(
'returns false unconditionally if BOT=false is set'
,
()
async
{
fakePlatform
.
environment
[
'BOT'
]
=
'false'
;
fakePlatform
.
environment
[
'BOT'
]
=
'false'
;
fakePlatform
.
environment
[
'TRAVIS'
]
=
'true'
;
fakePlatform
.
environment
[
'TRAVIS'
]
=
'true'
;
expect
(
botDetector
.
isRunningOnBot
,
isFalse
);
expect
(
await
botDetector
.
isRunningOnBot
,
isFalse
);
});
});
testWithoutContext
(
'returns false unconditionally if FLUTTER_HOST is set'
,
()
async
{
testWithoutContext
(
'returns false unconditionally if FLUTTER_HOST is set'
,
()
async
{
fakePlatform
.
environment
[
'FLUTTER_HOST'
]
=
'foo'
;
fakePlatform
.
environment
[
'FLUTTER_HOST'
]
=
'foo'
;
fakePlatform
.
environment
[
'TRAVIS'
]
=
'true'
;
fakePlatform
.
environment
[
'TRAVIS'
]
=
'true'
;
expect
(
botDetector
.
isRunningOnBot
,
isFalse
);
expect
(
await
botDetector
.
isRunningOnBot
,
isFalse
);
});
});
testWithoutContext
(
'returns false with and without a terminal attached'
,
()
async
{
testWithoutContext
(
'returns false with and without a terminal attached'
,
()
async
{
when
(
mockHttpClient
.
getUrl
(
any
)).
thenAnswer
((
_
)
{
throw
const
SocketException
(
'HTTP connection timed out'
);
});
mockStdio
.
stdout
.
hasTerminal
=
true
;
mockStdio
.
stdout
.
hasTerminal
=
true
;
expect
(
botDetector
.
isRunningOnBot
,
isFalse
);
expect
(
await
botDetector
.
isRunningOnBot
,
isFalse
);
mockStdio
.
stdout
.
hasTerminal
=
false
;
mockStdio
.
stdout
.
hasTerminal
=
false
;
expect
(
botDetector
.
isRunningOnBot
,
isFalse
);
expect
(
await
botDetector
.
isRunningOnBot
,
isFalse
);
});
});
testWithoutContext
(
'can test analytics outputs on bots when outputting to a file'
,
()
async
{
testWithoutContext
(
'can test analytics outputs on bots when outputting to a file'
,
()
async
{
fakePlatform
.
environment
[
'TRAVIS'
]
=
'true'
;
fakePlatform
.
environment
[
'TRAVIS'
]
=
'true'
;
fakePlatform
.
environment
[
'FLUTTER_ANALYTICS_LOG_FILE'
]
=
'/some/file'
;
fakePlatform
.
environment
[
'FLUTTER_ANALYTICS_LOG_FILE'
]
=
'/some/file'
;
expect
(
botDetector
.
isRunningOnBot
,
isFalse
);
expect
(
await
botDetector
.
isRunningOnBot
,
isFalse
);
});
testWithoutContext
(
'returns true when azure metadata is reachable'
,
()
async
{
when
(
mockHttpClient
.
getUrl
(
any
)).
thenAnswer
((
_
)
{
return
Future
<
HttpClientRequest
>.
value
(
mockHttpClientRequest
);
});
when
(
mockHttpClientRequest
.
headers
).
thenReturn
(
mockHttpHeaders
);
expect
(
await
botDetector
.
isRunningOnBot
,
isTrue
);
});
});
});
group
(
'AzureDetector'
,
()
{
AzureDetector
azureDetector
;
MockHttpClient
mockHttpClient
;
MockHttpClientRequest
mockHttpClientRequest
;
MockHttpHeaders
mockHttpHeaders
;
setUp
(()
{
mockHttpClient
=
MockHttpClient
();
mockHttpClientRequest
=
MockHttpClientRequest
();
mockHttpHeaders
=
MockHttpHeaders
();
azureDetector
=
AzureDetector
(
httpClientFactory:
()
=>
mockHttpClient
,
);
});
testWithoutContext
(
'isRunningOnAzure returns false when connection times out'
,
()
async
{
when
(
mockHttpClient
.
getUrl
(
any
)).
thenAnswer
((
_
)
{
throw
const
SocketException
(
'HTTP connection timed out'
);
});
});
expect
(
await
azureDetector
.
isRunningOnAzure
,
isFalse
);
});
testWithoutContext
(
'isRunningOnAzure returns true when azure metadata is reachable'
,
()
async
{
when
(
mockHttpClient
.
getUrl
(
any
)).
thenAnswer
((
_
)
{
return
Future
<
HttpClientRequest
>.
value
(
mockHttpClientRequest
);
});
when
(
mockHttpClientRequest
.
headers
).
thenReturn
(
mockHttpHeaders
);
expect
(
await
azureDetector
.
isRunningOnAzure
,
isTrue
);
});
});
});
});
}
}
class
MockHttpClient
extends
Mock
implements
HttpClient
{}
class
MockHttpClientRequest
extends
Mock
implements
HttpClientRequest
{}
class
MockHttpHeaders
extends
Mock
implements
HttpHeaders
{}
packages/flutter_tools/test/general.shard/dart/pub_get_test.dart
View file @
b9ecebf1
...
@@ -365,7 +365,7 @@ void main() {
...
@@ -365,7 +365,7 @@ void main() {
class
BotDetectorAlwaysNo
implements
BotDetector
{
class
BotDetectorAlwaysNo
implements
BotDetector
{
const
BotDetectorAlwaysNo
();
const
BotDetectorAlwaysNo
();
@override
@override
bool
get
isRunningOnBot
=>
false
;
Future
<
bool
>
get
isRunningOnBot
async
=>
false
;
}
}
typedef
StartCallback
=
void
Function
(
List
<
dynamic
>
command
);
typedef
StartCallback
=
void
Function
(
List
<
dynamic
>
command
);
...
...
packages/flutter_tools/test/general.shard/runner/runner_test.dart
View file @
b9ecebf1
...
@@ -147,7 +147,10 @@ class CrashingFlutterCommand extends FlutterCommand {
...
@@ -147,7 +147,10 @@ class CrashingFlutterCommand extends FlutterCommand {
}
}
class
CrashingUsage
implements
Usage
{
class
CrashingUsage
implements
Usage
{
CrashingUsage
()
:
_impl
=
Usage
(
versionOverride:
'[user-branch]'
);
CrashingUsage
()
:
_impl
=
Usage
(
versionOverride:
'[user-branch]'
,
runningOnBot:
true
,
);
final
Usage
_impl
;
final
Usage
_impl
;
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment