// Copyright 2016 The Chromium 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:async'; import 'dart:convert'; import 'dart:io' show IOSink; import 'package:args/command_runner.dart'; import 'package:flutter_tools/src/base/file_system.dart' hide IOSink; import 'package:flutter_tools/src/base/io.dart'; import 'package:flutter_tools/src/cache.dart'; import 'package:flutter_tools/src/commands/packages.dart'; import 'package:process/process.dart'; import 'package:test/test.dart'; import '../src/common.dart'; import '../src/context.dart'; void main() { Cache.disableLocking(); group('packages get/upgrade', () { Directory temp; setUp(() { temp = fs.systemTempDirectory.createTempSync('flutter_tools'); }); tearDown(() { temp.deleteSync(recursive: true); }); Future runCommand(String verb, { List args }) async { final String projectPath = await createProject(temp); final PackagesCommand command = new PackagesCommand(); final CommandRunner runner = createTestCommandRunner(command); final List commandArgs = ['packages', verb]; if (args != null) commandArgs.addAll(args); commandArgs.add(projectPath); await runner.run(commandArgs); return projectPath; } void expectExists(String projectPath, String relPath) { expect(fs.isFileSync(fs.path.join(projectPath, relPath)), true); } // Verify that we create a project that is well-formed. testUsingContext('get', () async { final String projectPath = await runCommand('get'); expectExists(projectPath, 'lib/main.dart'); expectExists(projectPath, '.packages'); }, timeout: allowForRemotePubInvocation); testUsingContext('get --offline', () async { final String projectPath = await runCommand('get', args: ['--offline']); expectExists(projectPath, 'lib/main.dart'); expectExists(projectPath, '.packages'); }); testUsingContext('upgrade', () async { final String projectPath = await runCommand('upgrade'); expectExists(projectPath, 'lib/main.dart'); expectExists(projectPath, '.packages'); }, timeout: allowForRemotePubInvocation); }); group('packages test/pub', () { MockProcessManager mockProcessManager; MockStdio mockStdio; setUp(() { mockProcessManager = new MockProcessManager(); mockStdio = new MockStdio(); }); testUsingContext('test', () async { await createTestCommandRunner(new PackagesCommand()).run(['packages', 'test']); final List commands = mockProcessManager.commands; expect(commands, hasLength(4)); expect(commands[0], matches(r'dart-sdk[\\/]bin[\\/]pub')); expect(commands[1], '--trace'); expect(commands[2], 'run'); expect(commands[3], 'test'); }, overrides: { ProcessManager: () => mockProcessManager, Stdio: () => mockStdio, }); testUsingContext('run', () async { await createTestCommandRunner(new PackagesCommand()).run(['packages', '--verbose', 'pub', 'run', '--foo', 'bar']); final List commands = mockProcessManager.commands; expect(commands, hasLength(4)); expect(commands[0], matches(r'dart-sdk[\\/]bin[\\/]pub')); expect(commands[1], 'run'); expect(commands[2], '--foo'); expect(commands[3], 'bar'); }, overrides: { ProcessManager: () => mockProcessManager, Stdio: () => mockStdio, }); testUsingContext('publish', () async { final PromptingProcess process = new PromptingProcess(); mockProcessManager.processFactory = (List commands) => process; final Future runPackages = createTestCommandRunner(new PackagesCommand()).run(['packages', 'pub', 'publish']); final Future runPrompt = process.showPrompt('Proceed (y/n)? ', ['hello', 'world']); final Future simulateUserInput = new Future(() { mockStdio.simulateStdin('y'); }); await Future.wait(>[runPackages, runPrompt, simulateUserInput]); final List commands = mockProcessManager.commands; expect(commands, hasLength(2)); expect(commands[0], matches(r'dart-sdk[\\/]bin[\\/]pub')); expect(commands[1], 'publish'); final List stdout = mockStdio.writtenToStdout; expect(stdout, hasLength(4)); expect(stdout.sublist(0, 2), contains('Proceed (y/n)? ')); expect(stdout.sublist(0, 2), contains('y\n')); expect(stdout[2], 'hello\n'); expect(stdout[3], 'world\n'); }, overrides: { ProcessManager: () => mockProcessManager, Stdio: () => mockStdio, }); }); } /// A strategy for creating Process objects from a list of commands. typedef Process ProcessFactory(List command); /// A ProcessManager that starts Processes by delegating to a ProcessFactory. class MockProcessManager implements ProcessManager { ProcessFactory processFactory = (List commands) => new MockProcess(); List commands; @override Future start( List command, { String workingDirectory, Map environment, bool includeParentEnvironment: true, bool runInShell: false, ProcessStartMode mode: ProcessStartMode.NORMAL, }) { commands = command; return new Future.value(processFactory(command)); } @override dynamic noSuchMethod(Invocation invocation) => null; } /// A process that prompts the user to proceed, then asynchronously writes /// some lines to stdout before it exits. class PromptingProcess implements Process { Future showPrompt(String prompt, List outputLines) async { _stdoutController.add(UTF8.encode(prompt)); final List bytesOnStdin = await _stdin.future; // Echo stdin to stdout. _stdoutController.add(bytesOnStdin); if (bytesOnStdin[0] == UTF8.encode('y')[0]) { for (final String line in outputLines) _stdoutController.add(UTF8.encode('$line\n')); } await _stdoutController.close(); } final StreamController> _stdoutController = new StreamController>(); final CompleterIOSink _stdin = new CompleterIOSink(); @override Stream> get stdout => _stdoutController.stream; @override Stream> get stderr => const Stream>.empty(); @override IOSink get stdin => _stdin; @override Future get exitCode async { await _stdoutController.done; return 0; } @override dynamic noSuchMethod(Invocation invocation) => null; } /// An inactive process that collects stdin and produces no output. class MockProcess implements Process { final IOSink _stdin = new MemoryIOSink(); @override Stream> get stdout => const Stream>.empty(); @override Stream> get stderr => const Stream>.empty(); @override IOSink get stdin => _stdin; @override Future get exitCode => new Future.value(0); @override dynamic noSuchMethod(Invocation invocation) => null; } /// An IOSink that completes a future with the first line written to it. class CompleterIOSink extends MemoryIOSink { final Completer> _completer = new Completer>(); Future> get future => _completer.future; @override void add(List data) { if (!_completer.isCompleted) _completer.complete(data); super.add(data); } } /// A Stdio that collects stdout and supports simulated stdin. class MockStdio extends Stdio { final MemoryIOSink _stdout = new MemoryIOSink(); final StreamController> _stdin = new StreamController>(); @override IOSink get stdout => _stdout; @override Stream> get stdin => _stdin.stream; void simulateStdin(String line) { _stdin.add(UTF8.encode('$line\n')); } List get writtenToStdout => _stdout.writes.map(_stdout.encoding.decode).toList(); } /// An IOSink that collects whatever is written to it. class MemoryIOSink implements IOSink { @override Encoding encoding = UTF8; final List> writes = >[]; @override void add(List data) { writes.add(data); } @override Future addStream(Stream> stream) { final Completer completer = new Completer(); stream.listen((List data) { add(data); }).onDone(() => completer.complete(null)); return completer.future; } @override void writeCharCode(int charCode) { add([charCode]); } @override void write(Object obj) { add(encoding.encode('$obj')); } @override void writeln([Object obj = '']) { add(encoding.encode('$obj\n')); } @override void writeAll(Iterable objects, [String separator = '']) { bool addSeparator = false; for (dynamic object in objects) { if (addSeparator) { write(separator); } write(object); addSeparator = true; } } @override void addError(dynamic error, [StackTrace stackTrace]) { throw new UnimplementedError(); } @override Future get done => close(); @override Future close() async => null; @override Future flush() async => null; }