blob: 640e0db0bc3edfeb5896e0d3a498ff38a3e901d1 [file] [log] [blame]
// Copyright 2014 The Flutter 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:typed_data';
import 'package:file/memory.dart';
import 'package:file_testing/file_testing.dart';
import 'package:flutter_tools/src/base/file_system.dart';
import 'package:flutter_tools/src/base/logger.dart';
import 'package:flutter_tools/src/build_system/file_store.dart';
import '../../src/common.dart';
void main() {
testWithoutContext('FileStore initializes file cache', () {
final FileSystem fileSystem = MemoryFileSystem.test();
final File cacheFile = fileSystem.file(FileStore.kFileCache);
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
);
fileCache.initialize();
fileCache.persist();
expect(cacheFile, exists);
final Uint8List buffer = cacheFile.readAsBytesSync();
final FileStorage fileStorage = FileStorage.fromBuffer(buffer);
expect(fileStorage.files, isEmpty);
expect(fileStorage.version, 2);
});
testWithoutContext('FileStore can use timestamp strategy', () async {
final FileSystem fileSystem = MemoryFileSystem.test();
final File cacheFile = fileSystem.file(FileStore.kFileCache);
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
strategy: FileStoreStrategy.timestamp,
);
fileCache.initialize();
final File file = fileSystem.file('test')..createSync();
// Initial run does not contain any timestamps for file.
expect(fileCache.diffFileList(<File>[file]), hasLength(1));
// Swap current timestamps to previous timestamps.
fileCache.persistIncremental();
// timestamp matches previous timestamp.
expect(fileCache.diffFileList(<File>[file]), isEmpty);
// clear current timestamp list.
fileCache.persistIncremental();
// modify the time stamp.
file.setLastModifiedSync(DateTime(1991));
// verify the file is marked as dirty again.
expect(fileCache.diffFileList(<File>[file]), hasLength(1));
});
testWithoutContext('FileStore saves and restores to file cache', () async {
final FileSystem fileSystem = MemoryFileSystem.test();
final File cacheFile = fileSystem.file(FileStore.kFileCache);
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
);
final File file = fileSystem.file('foo.dart')
..createSync()
..writeAsStringSync('hello');
fileCache.initialize();
fileCache.diffFileList(<File>[file]);
fileCache.persist();
final String? currentHash = fileCache.currentAssetKeys[file.path];
final Uint8List buffer = cacheFile
.readAsBytesSync();
FileStorage fileStorage = FileStorage.fromBuffer(buffer);
expect(fileStorage.files.single.hash, currentHash);
expect(fileStorage.files.single.path, file.path);
final FileStore newfileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
);
newfileCache.initialize();
expect(newfileCache.currentAssetKeys, isEmpty);
expect(newfileCache.previousAssetKeys['foo.dart'], currentHash);
newfileCache.persist();
// Still persisted correctly.
fileStorage = FileStorage.fromBuffer(buffer);
expect(fileStorage.files.single.hash, currentHash);
expect(fileStorage.files.single.path, file.path);
});
testWithoutContext('FileStore handles changed format', () async {
final FileSystem fileSystem = MemoryFileSystem.test();
final File cacheFile = fileSystem.file(FileStore.kFileCache)..writeAsStringSync(
'{"version":1,"files":[{"path_old":"foo.dart","hash_old":"f95b70fdc3088560732a5ac135644506"}]}');
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
);
fileCache.initialize();
expect(cacheFile, isNot(exists));
});
testWithoutContext('FileStore handles persisting with a missing build directory', () async {
final FileSystem fileSystem = MemoryFileSystem.test();
final File cacheFile = fileSystem
.directory('example')
.childFile(FileStore.kFileCache)
..createSync(recursive: true);
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
);
final File file = fileSystem.file('foo.dart')
..createSync()
..writeAsStringSync('hello');
fileCache.initialize();
cacheFile.parent.deleteSync(recursive: true);
fileCache.diffFileList(<File>[file]);
expect(fileCache.persist, returnsNormally);
});
testWithoutContext('FileStore handles hashing missing files', () async {
final FileSystem fileSystem = MemoryFileSystem.test();
final File cacheFile = fileSystem.file(FileStore.kFileCache);
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
);
fileCache.initialize();
final List<File> results = fileCache.diffFileList(<File>[fileSystem.file('hello.dart')]);
expect(results, hasLength(1));
expect(results.single.path, 'hello.dart');
expect(fileCache.currentAssetKeys, isNot(contains(fileSystem.path.absolute('hello.dart'))));
});
testWithoutContext('FileStore handles failure to persist file cache', () async {
final FileExceptionHandler handler = FileExceptionHandler();
final FileSystem fileSystem = MemoryFileSystem.test(opHandle: handler.opHandle);
final BufferLogger logger = BufferLogger.test();
final File cacheFile = fileSystem.file('foo')
..createSync();
handler.addError(cacheFile, FileSystemOp.write, const FileSystemException('Out of space!'));
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: logger,
);
fileCache.initialize();
fileCache.persist();
expect(logger.errorText, contains('Out of space!'));
});
testWithoutContext('FileStore handles failure to restore file cache', () async {
final FileExceptionHandler handler = FileExceptionHandler();
final FileSystem fileSystem = MemoryFileSystem.test(opHandle: handler.opHandle);
final BufferLogger logger = BufferLogger.test();
final File cacheFile = fileSystem.file('foo')
..createSync();
handler.addError(cacheFile, FileSystemOp.read, const FileSystemException('Out of space!'));
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: logger,
);
fileCache.initialize();
expect(logger.errorText, contains('Out of space!'));
});
testWithoutContext('FileStore handles chunked conversion of a file', () async {
final FileSystem fileSystem = MemoryFileSystem.test();
final File cacheFile = fileSystem
.directory('example')
.childFile(FileStore.kFileCache)
..createSync(recursive: true);
final FileStore fileCache = FileStore(
cacheFile: cacheFile,
logger: BufferLogger.test(),
);
final File file = fileSystem.file('foo.dart')
..createSync()
..writeAsStringSync('hello');
fileCache.initialize();
cacheFile.parent.deleteSync(recursive: true);
fileCache.diffFileList(<File>[file]);
expect(fileCache.currentAssetKeys['foo.dart'], '5d41402abc4b2a76b9719d911017c592');
});
}