Unverified Commit 6fe45fb1 authored by Zachary Anderson's avatar Zachary Anderson Committed by GitHub

[flutter_tool] Improve Windows flutter clean error message (#36784)

parent aa6384cb
......@@ -6,6 +6,7 @@ import 'dart:async';
import '../base/common.dart';
import '../base/file_system.dart';
import '../base/platform.dart';
import '../build_info.dart';
import '../globals.dart';
import '../project.dart';
......@@ -34,7 +35,10 @@ class CleanCommand extends FlutterCommand {
if (buildDir.existsSync()) {
try {
buildDir.deleteSync(recursive: true);
} catch (error) {
} on FileSystemException catch (error) {
if (platform.isWindows) {
_windowsDeleteFailure(buildDir.path);
}
throwToolExit(error.toString());
}
}
......@@ -43,11 +47,22 @@ class CleanCommand extends FlutterCommand {
if (flutterProject.dartTool.existsSync()) {
try {
flutterProject.dartTool.deleteSync(recursive: true);
} catch (error) {
} on FileSystemException catch (error) {
if (platform.isWindows) {
_windowsDeleteFailure(flutterProject.dartTool.path);
}
throwToolExit(error.toString());
}
}
return const FlutterCommandResult(ExitStatus.success);
}
void _windowsDeleteFailure(String path) {
printError(
'Failed to remove $path. '
'A program may still be using a file in the directory or the directory itself. '
'To find and stop such a program, see: '
'https://superuser.com/questions/1333118/cant-delete-empty-folder-because-it-is-used');
}
}
......@@ -2,25 +2,37 @@
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
import 'package:flutter_tools/src/base/common.dart';
import 'package:flutter_tools/src/base/config.dart';
import 'package:flutter_tools/src/base/context.dart';
import 'package:flutter_tools/src/base/file_system.dart';
import 'package:flutter_tools/src/base/logger.dart';
import 'package:flutter_tools/src/base/platform.dart';
import 'package:flutter_tools/src/commands/clean.dart';
import 'package:mockito/mockito.dart';
import '../../src/common.dart';
import '../../src/context.dart';
void main() {
final MockFileSystem mockFileSystem = MockFileSystem();
final MockDirectory currentDirectory = MockDirectory();
final MockDirectory exampleDirectory = MockDirectory();
final MockDirectory buildDirectory = MockDirectory();
final MockDirectory dartToolDirectory = MockDirectory();
final MockFile pubspec = MockFile();
final MockFile examplePubspec = MockFile();
MockFileSystem mockFileSystem;
MockDirectory currentDirectory;
MockDirectory exampleDirectory;
MockDirectory buildDirectory;
MockDirectory dartToolDirectory;
MockFile pubspec;
MockFile examplePubspec;
MockPlatform windowsPlatform;
setUp(() {
mockFileSystem = MockFileSystem();
currentDirectory = MockDirectory();
exampleDirectory = MockDirectory();
buildDirectory = MockDirectory();
dartToolDirectory = MockDirectory();
pubspec = MockFile();
examplePubspec = MockFile();
windowsPlatform = MockPlatform();
when(mockFileSystem.currentDirectory).thenReturn(currentDirectory);
when(currentDirectory.childDirectory('example')).thenReturn(exampleDirectory);
when(currentDirectory.childFile('pubspec.yaml')).thenReturn(pubspec);
......@@ -34,14 +46,30 @@ void main() {
when(mockFileSystem.path).thenReturn(fs.path);
when(buildDirectory.existsSync()).thenReturn(true);
when(dartToolDirectory.existsSync()).thenReturn(true);
when(windowsPlatform.isWindows).thenReturn(true);
});
group(CleanCommand, () {
testUsingContext('removes build and .dart_tool directories', () async {
await CleanCommand().runCommand();
verify(buildDirectory.deleteSync(recursive: true)).called(1);
verify(dartToolDirectory.deleteSync(recursive: true)).called(1);
}, overrides: <Type, Generator>{
Config: () => null,
FileSystem: () => mockFileSystem,
});
testUsingContext('prints a helpful error message on Windows', () async {
final BufferLogger logger = context.get<Logger>();
when(buildDirectory.deleteSync(recursive: true)).thenThrow(
const FileSystemException('Deletion failed'));
expect(() async => await CleanCommand().runCommand(), throwsA(isInstanceOf<ToolExit>()));
expect(logger.errorText, contains('A program may still be using a file'));
}, overrides: <Type, Generator>{
Config: () => null,
FileSystem: () => mockFileSystem,
Platform: () => windowsPlatform,
Logger: () => BufferLogger(),
});
});
}
......@@ -49,3 +77,4 @@ void main() {
class MockFileSystem extends Mock implements FileSystem {}
class MockFile extends Mock implements File {}
class MockDirectory extends Mock implements Directory {}
class MockPlatform extends Mock implements Platform {}
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment