// 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 '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/io.dart'; import 'package:flutter_tools/src/base/logger.dart'; import 'package:flutter_tools/src/base/os.dart'; import 'package:flutter_tools/src/base/platform.dart'; import 'package:flutter_tools/src/base/terminal.dart'; import 'package:flutter_tools/src/cache.dart'; import 'package:test/fake.dart'; import '../src/common.dart'; import '../src/fake_http_client.dart'; import '../src/fakes.dart'; final Platform testPlatform = FakePlatform(); void main() { testWithoutContext('ArtifactUpdater can download a zip archive', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), exists); }); testWithoutContext('ArtifactUpdater can download a zip archive and delete stale files', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); // Unrelated file from another cache. fileSystem.file('out/bar').createSync(recursive: true); // Stale file from current cache. fileSystem.file('out/test/foo.txt').createSync(recursive: true); await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), exists); expect(fileSystem.file('out/bar'), exists); expect(fileSystem.file('out/test/foo.txt'), isNot(exists)); }); testWithoutContext('ArtifactUpdater will delete any denylisted files from the outputDirectory', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final Directory tempStorage = fileSystem.currentDirectory.childDirectory('temp'); final String localZipPath = tempStorage.childFile('test.zip').path; File? desiredArtifact; File? entitlementsFile; File? nestedWithoutEntitlementsFile; operatingSystemUtils.unzipCallbacks[localZipPath] = (Directory outputDirectory) { desiredArtifact = outputDirectory.childFile('artifact.bin')..createSync(); entitlementsFile = outputDirectory.childFile('entitlements.txt')..createSync(); nestedWithoutEntitlementsFile = outputDirectory .childDirectory('dir') .childFile('without_entitlements.txt') ..createSync(recursive: true); }; final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: tempStorage..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); // entitlements file cached from before the tool had a denylist final File staleEntitlementsFile = fileSystem.file('out/path/to/entitlements.txt')..createSync(recursive: true); expect(staleEntitlementsFile, exists); await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(logger.statusText, contains('test message')); expect(desiredArtifact, exists); expect(entitlementsFile, isNot(exists)); expect(nestedWithoutEntitlementsFile, isNot(exists)); expect(staleEntitlementsFile, isNot(exists)); }); testWithoutContext('ArtifactUpdater will not validate the md5 hash if the ' 'x-goog-hash header is present but missing an md5 entry', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.list(<FakeRequest>[ FakeRequest(Uri.parse('http://test.zip'), response: const FakeResponse( headers: <String, List<String>>{ 'x-goog-hash': <String>[], } )), ]), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), exists); }); testWithoutContext('ArtifactUpdater will validate the md5 hash if the ' 'x-goog-hash header is present', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.list(<FakeRequest>[ FakeRequest(Uri.parse('http://test.zip'), response: const FakeResponse( body: <int>[0], headers: <String, List<String>>{ 'x-goog-hash': <String>[ 'foo-bar-baz', 'md5=k7iFrf4NoInN9jSQT9WfcQ==', ], } )), ]), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), exists); }); testWithoutContext('ArtifactUpdater will validate the md5 hash if the ' 'x-goog-hash header is present and throw if it does not match', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.list(<FakeRequest>[ FakeRequest(Uri.parse('http://test.zip'), response: const FakeResponse( body: <int>[0], headers: <String, List<String>>{ 'x-goog-hash': <String>[ 'foo-bar-baz', 'md5=k7iFrf4SQT9WfcQ==', ], } )), FakeRequest(Uri.parse('http://test.zip'), response: const FakeResponse( headers: <String, List<String>>{ 'x-goog-hash': <String>[ 'foo-bar-baz', 'md5=k7iFrf4SQT9WfcQ==', ], } )), ]), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await expectLater(() async => artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ), throwsToolExit(message: 'k7iFrf4SQT9WfcQ==')); // validate that the hash mismatch message is included. }); testWithoutContext('ArtifactUpdater will restart the status ticker if it needs to retry the download', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final Logger logger = StdoutLogger( terminal: Terminal.test(supportsColor: true), stdio: FakeStdio(), outputPreferences: OutputPreferences.test(), ); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.list(<FakeRequest>[ FakeRequest(Uri.parse('http://test.zip'), responseError: const HttpException('')), FakeRequest(Uri.parse('http://test.zip')), ]), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(fileSystem.file('out/test'), exists); }); testWithoutContext('ArtifactUpdater will re-attempt on a non-200 response', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.list(<FakeRequest>[ FakeRequest(Uri.parse('http://test.zip'), response: const FakeResponse(statusCode: HttpStatus.preconditionFailed)), FakeRequest(Uri.parse('http://test.zip'), response: const FakeResponse(statusCode: HttpStatus.preconditionFailed)), ]), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await expectLater(() async => artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ), throwsToolExit()); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), isNot(exists)); }); testWithoutContext('ArtifactUpdater will tool exit on an ArgumentError from http client with base url override', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: FakePlatform( environment: <String, String>{ 'FLUTTER_STORAGE_BASE_URL': 'foo-bar', }, ), httpClient: FakeHttpClient.list(<FakeRequest>[ FakeRequest(Uri.parse('http://foo-bar/test.zip'), responseError: ArgumentError()), ]), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://foo-bar/test.zip'], ); await expectLater(() async => artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://foo-bar/test.zip'), fileSystem.currentDirectory.childDirectory('out'), ), throwsToolExit()); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), isNot(exists)); }); testWithoutContext('ArtifactUpdater will rethrow on an ArgumentError from http client without base url override', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.list(<FakeRequest>[ FakeRequest(Uri.parse('http://test.zip'), responseError: ArgumentError()), ]), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await expectLater(() async => artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ), throwsArgumentError); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), isNot(exists)); }); testWithoutContext('ArtifactUpdater will re-download a file if unzipping fails', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); operatingSystemUtils.failures = 1; await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), exists); }); testWithoutContext('ArtifactUpdater will de-download a file if unzipping fails on windows', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(windows: true); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); operatingSystemUtils.failures = 1; await artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(logger.statusText, contains('test message')); expect(fileSystem.file('out/test'), exists); }); testWithoutContext('ArtifactUpdater will bail with a tool exit if unzipping fails more than twice', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); operatingSystemUtils.failures = 2; expect(artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ), throwsToolExit()); expect(fileSystem.file('te,[/test'), isNot(exists)); expect(fileSystem.file('out/test'), isNot(exists)); }); testWithoutContext('ArtifactUpdater will bail if unzipping fails more than twice on Windows', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(windows: true); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); operatingSystemUtils.failures = 2; expect(artifactUpdater.downloadZipArchive( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ), throwsToolExit()); expect(fileSystem.file('te,[/test'), isNot(exists)); expect(fileSystem.file('out/test'), isNot(exists)); }); testWithoutContext('ArtifactUpdater can download a tar archive', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); await artifactUpdater.downloadZippedTarball( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ); expect(fileSystem.file('out/test'), exists); }); testWithoutContext('ArtifactUpdater will delete downloaded files if they exist.', () async { final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: testPlatform, httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); artifactUpdater.downloadedFiles.addAll(<File>[ fileSystem.file('a/b/c/d')..createSync(recursive: true), fileSystem.file('d/e/f'), ]); artifactUpdater.removeDownloadedFiles(); expect(fileSystem.file('a/b/c/d'), isNot(exists)); expect(logger.errorText, isEmpty); }); testWithoutContext('ArtifactUpdater will tool exit if deleting the existing artifacts fails with 32 on windows', () async { const int kSharingViolation = 32; final FileExceptionHandler handler = FileExceptionHandler(); final FakeOperatingSystemUtils operatingSystemUtils = FakeOperatingSystemUtils(); final MemoryFileSystem fileSystem = MemoryFileSystem.test(opHandle: handler.opHandle); final BufferLogger logger = BufferLogger.test(); final ArtifactUpdater artifactUpdater = ArtifactUpdater( fileSystem: fileSystem, logger: logger, operatingSystemUtils: operatingSystemUtils, platform: FakePlatform(operatingSystem: 'windows'), httpClient: FakeHttpClient.any(), tempStorage: fileSystem.currentDirectory.childDirectory('temp') ..createSync(), allowedBaseUrls: <String>['http://test.zip'], ); final Directory errorDirectory = fileSystem.currentDirectory .childDirectory('out') .childDirectory('test') ..createSync(recursive: true); handler.addError(errorDirectory, FileSystemOp.delete, const FileSystemException('', '', OSError('', kSharingViolation))); await expectLater(() async => artifactUpdater.downloadZippedTarball( 'test message', Uri.parse('http://test.zip'), fileSystem.currentDirectory.childDirectory('out'), ), throwsToolExit( message: 'Failed to delete /out/test because the local file/directory is in use by another process' )); expect(fileSystem.file('out/test'), isNot(exists)); }); } class FakeOperatingSystemUtils extends Fake implements OperatingSystemUtils { FakeOperatingSystemUtils({this.windows = false}); int failures = 0; final bool windows; /// A mapping of zip [file] paths to callbacks that receive the [targetDirectory]. /// /// Use this to have [unzip] generate an arbitrary set of [FileSystemEntity]s /// under [targetDirectory]. final Map<String, void Function(Directory)> unzipCallbacks = <String, void Function(Directory)>{}; @override void unzip(File file, Directory targetDirectory) { if (failures > 0) { failures -= 1; throw Exception(); } if (unzipCallbacks.containsKey(file.path)) { unzipCallbacks[file.path]!(targetDirectory); } else { targetDirectory.childFile(file.fileSystem.path.basenameWithoutExtension(file.path)) .createSync(); } } @override void unpack(File gzippedTarFile, Directory targetDirectory) { if (failures > 0) { failures -= 1; throw Exception(); } targetDirectory.childFile(gzippedTarFile.fileSystem.path.basenameWithoutExtension(gzippedTarFile.path)) .createSync(); } }