Skip to content

Commit 4bdc627

Browse files
committed
Add test coverage for clean cache and create project tasks.
1 parent e4ab9fb commit 4bdc627

File tree

4 files changed

+264
-11
lines changed

4 files changed

+264
-11
lines changed

lib/src/cli/task_clean_cache.dart

+38-11
Original file line numberDiff line numberDiff line change
@@ -18,38 +18,65 @@
1818
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
1919
// SOFTWARE.
2020

21-
import "dart:ffi";
2221
import "dart:io";
2322

24-
import "package:meta/meta.dart";
25-
2623
import "../common/common.dart";
27-
import "../producer/kradle.dart";
2824
import "cli.dart";
2925
import "context.dart";
3026

3127
/// Clean the kradle cache by deleting contents recursively.
3228
class CleanCache extends Task {
3329
/// Create new Task.
34-
CleanCache()
30+
CleanCache([CacheProvider? cacheProvider])
3531
: super(TaskName.clean, {
3632
TaskOption.root: RootDirectoryInput(),
37-
});
33+
}) {
34+
_cacheProvider = cacheProvider ?? CacheProvider();
35+
}
36+
37+
late final CacheProvider _cacheProvider;
3838

3939
@override
40-
Future<void> toBeExecuted(
40+
Future<CleanCachResult> toBeExecuted(
4141
Context context, Map<TaskOption, dynamic> options) async {
42+
final deleted = <FileSystemEntity>[];
43+
final notDeletedByError = <FileSystemEntity, String>{};
4244
void deleteIfExists(FileSystemEntity entity) {
4345
if (entity.existsSync()) {
4446
try {
4547
entity.deleteSync();
46-
} on Exception {
47-
// ignore
48+
deleted.add(entity);
49+
} on Exception catch (e) {
50+
notDeletedByError[entity] = e.toString();
4851
}
4952
}
5053
}
5154

52-
final cache = Directory(findPathToRoot(context, options)).kradleCache;
53-
cache.listSync(recursive: true).forEach(deleteIfExists);
55+
_cacheProvider.getCacheContent(context, options).forEach(deleteIfExists);
56+
return CleanCachResult(deleted, notDeletedByError);
5457
}
5558
}
59+
60+
/// Result of task [CleanCache].
61+
class CleanCachResult {
62+
/// Create a new instance of [CleanCacheResult].
63+
const CleanCachResult(this.deleted, this.notDeletedByError);
64+
65+
/// List of deleted entities.
66+
final List<FileSystemEntity> deleted;
67+
68+
/// Map of not deleted entities and the corresponding error message.
69+
final Map<FileSystemEntity, String> notDeletedByError;
70+
}
71+
72+
/// Wrapper to provide the kradle cache directory
73+
/// based on [Context] and [TaskOption] input.
74+
class CacheProvider {
75+
76+
/// Return the files and directory in the kradle cache directory.
77+
List<FileSystemEntity> getCacheContent(Context context, Map<TaskOption, dynamic> options) {
78+
final cache = Directory(findPathToRoot(context, options)).kradleCache;
79+
return cache.listSync(recursive: true);
80+
}
81+
82+
}

lib/src/common/utilities.dart

+8
Original file line numberDiff line numberDiff line change
@@ -118,6 +118,14 @@ extension FileUtil on FileSystemEntity {
118118
}
119119
}
120120

121+
/// Utils for easier Directory handling.
122+
extension DirectoryUtil on Directory {
123+
124+
/// Check if directory contains items.
125+
bool get isEmpty => listSync().isEmpty;
126+
127+
}
128+
121129
/// Utils for easier String manipulation.
122130
extension StringUtil on String {
123131
/// Create an absolute path to the given file or folder.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,194 @@
1+
// Copyright (c) 2021 - 2024 Buijs Software
2+
//
3+
// Permission is hereby granted, free of charge, to any person obtaining a copy
4+
// of this software and associated documentation files (the "Software"), to deal
5+
// in the Software without restriction, including without limitation the rights
6+
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
7+
// copies of the Software, and to permit persons to whom the Software is
8+
// furnished to do so, subject to the following conditions:
9+
//
10+
// The above copyright notice and this permission notice shall be included in all
11+
// copies or substantial portions of the Software.
12+
//
13+
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
14+
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15+
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
16+
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
17+
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18+
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
19+
// SOFTWARE.
20+
21+
import "dart:io";
22+
23+
import "package:klutter/klutter.dart";
24+
import "package:klutter/src/cli/context.dart";
25+
import "package:meta/meta.dart";
26+
import "package:test/test.dart";
27+
28+
void main() {
29+
30+
Future<TaskResult> runTest(Directory workingDirectory,
31+
Directory cachingDirectory,
32+
[CleanCache? cc]) {
33+
final task = cc ?? CleanCache();
34+
35+
final context = Context(
36+
workingDirectory: workingDirectory,
37+
taskName: TaskName.clean,
38+
taskOptions: {});
39+
40+
workingDirectory
41+
.resolveFile("kradle.env")
42+
.writeAsStringSync("cache=${cachingDirectory.absolutePath}");
43+
44+
return task.execute(context);
45+
}
46+
47+
test("Verify clean cache is successful when cache contains nothing", () async {
48+
final workingDirectory = Directory.systemTemp.resolveFolder("foo1")..maybeCreate;
49+
final cachingDirectory = Directory.systemTemp.resolveFolder("foo1/.kradle/cache")..maybeCreate;
50+
expect(cachingDirectory.isEmpty, true);
51+
final result = await runTest(workingDirectory, cachingDirectory);
52+
expect(result.isOk, true);
53+
expect(cachingDirectory.isEmpty, true);
54+
});
55+
56+
test("Verify clean cache deletes all cache entries", () async {
57+
final workingDirectory = Directory.systemTemp.resolveFolder("foo1")..maybeCreate;
58+
final cachingDirectory = Directory.systemTemp.resolveFolder("foo1/.kradle/cache")..maybeCreate;
59+
cachingDirectory.resolveFolder("3.10.6.linux.arm").createSync();
60+
expect(cachingDirectory.isEmpty, false);
61+
final result = await runTest(workingDirectory, cachingDirectory);
62+
expect(result.isOk, true);
63+
expect(cachingDirectory.isEmpty, true);
64+
final CleanCachResult ccr = result.output;
65+
expect(ccr.deleted.length, 1);
66+
expect(ccr.notDeletedByError.isEmpty, true);
67+
});
68+
69+
test("Verify exceptions are caught when deleting entities", () async {
70+
final workingDirectory = Directory.systemTemp.resolveFolder("foo2")..maybeCreate;
71+
final cachingDirectory = DirectoryStub();
72+
expect(cachingDirectory.isEmpty, false);
73+
final cacheProvider = CacheProviderStub();
74+
final cleanCache = CleanCache(cacheProvider);
75+
final result = await runTest(workingDirectory, cachingDirectory, cleanCache);
76+
expect(result.isOk, true);
77+
expect(cachingDirectory.isEmpty, false);
78+
final CleanCachResult ccr = result.output;
79+
expect(ccr.deleted.length, 0);
80+
expect(ccr.notDeletedByError.isNotEmpty, true);
81+
expect(ccr.notDeletedByError[DirectoryStub()], "KlutterException with cause: 'BOOM!'");
82+
});
83+
}
84+
85+
class CacheProviderStub extends CacheProvider {
86+
@override
87+
List<FileSystemEntity> getCacheContent(
88+
Context context,
89+
Map<TaskOption, dynamic> options,
90+
) => [DirectoryStub()];
91+
}
92+
93+
@immutable
94+
class DirectoryStub implements Directory {
95+
@override
96+
bool operator ==(Object other) => true;
97+
98+
@override
99+
int get hashCode => 1;
100+
101+
@override
102+
Directory get absolute => DirectoryStub();
103+
104+
@override
105+
Future<Directory> create({bool recursive = false}) {
106+
throw UnimplementedError();
107+
}
108+
109+
@override
110+
void createSync({bool recursive = false}) {
111+
}
112+
113+
@override
114+
Future<Directory> createTemp([String? prefix]) {
115+
throw UnimplementedError();
116+
}
117+
118+
@override
119+
Directory createTempSync([String? prefix]) {
120+
throw UnimplementedError();
121+
}
122+
123+
@override
124+
Future<FileSystemEntity> delete({bool recursive = false}) {
125+
throw KlutterException("BOOM!");
126+
}
127+
128+
@override
129+
void deleteSync({bool recursive = false}) {
130+
throw KlutterException("BOOM!");
131+
}
132+
133+
@override
134+
Future<bool> exists() => Future.value(true);
135+
136+
@override
137+
bool existsSync() => true;
138+
139+
@override
140+
bool get isAbsolute => throw UnimplementedError();
141+
142+
@override
143+
Stream<FileSystemEntity> list({bool recursive = false, bool followLinks = true}) =>
144+
Stream.value(DirectoryStub());
145+
146+
@override
147+
List<FileSystemEntity> listSync({bool recursive = false, bool followLinks = true}) =>
148+
[DirectoryStub()];
149+
150+
@override
151+
Directory get parent => throw UnimplementedError();
152+
153+
@override
154+
String get path => Directory.systemTemp.resolveFolder("stub").absolutePath;
155+
156+
@override
157+
Future<Directory> rename(String newPath) {
158+
throw UnimplementedError();
159+
}
160+
161+
@override
162+
Directory renameSync(String newPath) {
163+
throw UnimplementedError();
164+
}
165+
166+
@override
167+
Future<String> resolveSymbolicLinks() {
168+
throw UnimplementedError();
169+
}
170+
171+
@override
172+
String resolveSymbolicLinksSync() {
173+
throw UnimplementedError();
174+
}
175+
176+
@override
177+
Future<FileStat> stat() {
178+
throw UnimplementedError();
179+
}
180+
181+
@override
182+
FileStat statSync() {
183+
throw UnimplementedError();
184+
}
185+
186+
@override
187+
Uri get uri => throw UnimplementedError();
188+
189+
@override
190+
Stream<FileSystemEvent> watch({int events = FileSystemEvent.all, bool recursive = false}) {
191+
throw UnimplementedError();
192+
}
193+
194+
}

test/src/cli/task_project_create_test.dart

+24
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,10 @@
1818
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
1919
// SOFTWARE.
2020

21+
import "dart:io";
22+
2123
import "package:klutter/klutter.dart";
24+
import "package:klutter/src/cli/context.dart";
2225
import "package:test/test.dart";
2326

2427
void main() {
@@ -50,4 +53,25 @@ void main() {
5053
e is KlutterException &&
5154
e.cause == "invalid dependency notations: 1.2.beta")));
5255
});
56+
57+
test("Verify an exception is thrown if flutter sdk is not found", () async {
58+
final getFlutterTask = NoFlutterSDK();
59+
final task = CreateProject(getFlutterSDK: getFlutterTask);
60+
final result = await task.execute(Context(
61+
workingDirectory: Directory.systemTemp,
62+
taskName: TaskName.create,
63+
taskOptions: {},
64+
));
65+
66+
expect(result.isOk, false);
67+
expect(result.message, "BOOM!");
68+
});
69+
5370
}
71+
72+
class NoFlutterSDK extends GetFlutterSDK {
73+
@override
74+
Future<TaskResult<Directory>> execute(Context context) async {
75+
return const TaskResult(isOk: false, message: "BOOM!");
76+
}
77+
}

0 commit comments

Comments
 (0)