blob: 9482b9ccf505eee5aee5f9abac77eba498cfde75 [file] [log] [blame]
// Copyright (c) 2020, the Dart project authors. Please see the AUTHORS file
// for details. 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:io';
import 'package:ffigen/src/code_generator.dart';
import 'package:ffigen/src/config_provider/config.dart';
import 'package:ffigen/src/strings.dart' as strings;
import 'package:logging/logging.dart';
import 'package:package_config/package_config_types.dart';
import 'package:path/path.dart' as path;
import 'package:test/test.dart';
import 'package:yaml/yaml.dart' as yaml;
extension LibraryTestExt on Library {
/// Get a [Binding]'s generated string with a given name.
String getBindingAsString(String name) {
try {
final b = bindings.firstWhere((element) => element.name == name);
return b.toBindingString(writer).string;
} catch (e) {
throw NotFoundException("Binding '$name' not found.");
}
}
/// Get a [Binding] with a given name.
Binding getBinding(String name) {
try {
final b = bindings.firstWhere((element) => element.name == name);
return b;
} catch (e) {
throw NotFoundException("Binding '$name' not found.");
}
}
}
/// Check whether a file generated by test/setup.dart exists and throw a helpful
/// exception if it does not.
void verifySetupFile(File file) {
if (!file.existsSync()) {
throw NotFoundException("The file ${file.path} does not exist.\n\n"
"You may need to run: dart run test/setup.dart\n");
}
}
// Remove '\r' for Windows compatibility, then apply user's normalizer.
String _normalizeGeneratedCode(
String generated, String Function(String)? codeNormalizer) {
final noCR = generated.replaceAll('\r', '');
if (codeNormalizer == null) return noCR;
return codeNormalizer(noCR);
}
/// Generates actual file using library and tests using [expect] with expected.
///
/// This will not delete the actual debug file incase [expect] throws an error.
void matchLibraryWithExpected(
Library library, String pathForActual, List<String> pathToExpected,
{String Function(String)? codeNormalizer, bool format = true}) {
_matchFileWithExpected(
library: library,
pathForActual: pathForActual,
pathToExpected: pathToExpected,
fileWriter: ({required Library library, required File file}) =>
library.generateFile(file, format: format),
codeNormalizer: codeNormalizer,
);
}
/// Generates actual file using library and tests using [expect] with expected.
///
/// This will not delete the actual debug file incase [expect] throws an error.
void matchLibrarySymbolFileWithExpected(Library library, String pathForActual,
List<String> pathToExpected, String importPath) {
_matchFileWithExpected(
library: library,
pathForActual: pathForActual,
pathToExpected: pathToExpected,
fileWriter: ({required Library library, required File file}) {
if (!library.writer.canGenerateSymbolOutput) library.generate();
library.generateSymbolOutputFile(file, importPath);
});
}
/// Generates actual file using library and tests using [expect] with expected.
///
/// This will not delete the actual debug file incase [expect] throws an error.
void _matchFileWithExpected({
required Library library,
required String pathForActual,
required List<String> pathToExpected,
required void Function({required Library library, required File file})
fileWriter,
String Function(String)? codeNormalizer,
}) {
final file = File(
path.join(strings.tmpDir, pathForActual),
);
fileWriter(library: library, file: file);
try {
final actual =
_normalizeGeneratedCode(file.readAsStringSync(), codeNormalizer);
final expected = _normalizeGeneratedCode(
File(path.joinAll(pathToExpected)).readAsStringSync(), codeNormalizer);
expect(actual.split('\n'), expected.split('\n'));
if (file.existsSync()) {
file.delete();
}
} catch (e) {
print('Failed test: Debug generated file: ${file.absolute.path}');
rethrow;
}
}
class NotFoundException implements Exception {
final String message;
NotFoundException(this.message);
@override
String toString() {
return message;
}
}
void logWarnings([Level level = Level.WARNING]) {
Logger.root.level = level;
Logger.root.onRecord.listen((record) {
print('${record.level.name.padRight(8)}: ${record.message}');
});
}
void logWarningsToArray(List<String> logArr, [Level level = Level.WARNING]) {
Logger.root.level = level;
Logger.root.onRecord.listen((record) {
logArr.add('${record.level.name.padRight(8)}: ${record.message}');
});
}
Config testConfig(String yamlBody, {String? filename}) {
return Config.fromYaml(
yaml.loadYaml(yamlBody) as yaml.YamlMap,
filename: filename,
packageConfig: PackageConfig([
Package(
'shared_bindings',
Uri.file(path.join(path.current, 'example', 'shared_bindings', 'lib/')),
),
]),
);
}
Config testConfigFromPath(String path) {
final file = File(path);
final yamlBody = file.readAsStringSync();
return testConfig(yamlBody, filename: path);
}
T withChDir<T>(String path, T Function() inner) {
final oldDir = Directory.current;
Directory.current = File(path).parent;
late T result;
try {
result = inner();
} finally {
Directory.current = oldDir;
}
return result;
}