Cucumber-JVM 3 - io.cucumber.datatable.UndefinedDataTableTypeException
Я обновил с Cucumber-JVM 2.4.0 до 3.0.2 в моем pom.xml
и DataTables начали выдавать это исключение:
io.cucumber.datatable.UndefinedDataTableTypeException: невозможно преобразовать DataTable в список
. Пожалуйста, зарегистрируйте DataTableType с помощью TableEntryTransformer или TableRowTransformer для класса jcucumberng.steps.pojos.Income.
Я изменил весь свой импорт на
import io.cucumber.datatable.DataTable;
Я сделал mvn clean install
и компиляция прошла успешно, но шаги, включающие DataTables, больше не работают после обновления.
Текущий код:
// Feature
When I Enter My Regular Income Sources
| name | amount | frequency |
| Salary | 25000 | every 2 weeks |
// Stepdef
@When("^I Enter My Regular Income Sources$")
public void I_Enter_My_Regular_Income_Sources(DataTable dataTable) throws Throwable {
List<Income> incomes = dataTable.asList(Income.class);
// More code
}
// Custom type
public class Income {
private String name = null;
private String amount = null;
private String frequency = null;
public Income(String name, String amount, String frequency) {
this.name = name;
this.amount = amount;
this.frequency = frequency;
}
// Getters and setters
}
Есть ли новый способ использования DataTables в Cucumber-JVM v3.xx?
2 ответа
Он был полностью обновлен. XStream был удален, поэтому более ранний код не будет работать.
Вам нужно будет добавить логику для преобразования данных и преобразования параметров. Обратитесь к ним - https://github.com/cucumber/cucumber/tree/master/datatable и https://github.com/cucumber/cucumber/tree/master/cucumber-expressions. Поместите код ниже класса в пакет, определенный в опции клея.
public class Configurer implements TypeRegistryConfigurer {
@Override
public void configureTypeRegistry(TypeRegistry registry) {
registry.defineDataTableType(new DataTableType(Income.class, new TableEntryTransformer<Income>() {
@Override
public Income transform(Map<String, String> entry) {
return new Income(entry.get("name"),entry.get("amount"),entry.get("frequency"));
}
}));
}
@Override
public Locale locale() {
return Locale.ENGLISH;
}
}
ОБНОВЛЕНО Импорт... Не все требуется, сохраните то, что актуально
import cucumber.api.TypeRegistry;
import cucumber.api.TypeRegistryConfigurer;
import io.cucumber.cucumberexpressions.ParameterType;
import io.cucumber.datatable.DataTable;
import io.cucumber.datatable.DataTableType;
import io.cucumber.datatable.TableCellTransformer;
import io.cucumber.datatable.TableEntryTransformer;
import io.cucumber.datatable.TableRowTransformer;
import io.cucumber.datatable.TableTransformer;
Используйте аннотацию DataTableType
@DataTableType
public Income incomeEntry(Map<String, String> entry) {
return new Income(entry.get("name"), entry.get("amount"), entry.get("frequency"));
}
Затем вы можете напрямую использовать список настраиваемого класса в определении шага
@When("^I Enter My Regular Income Sources$")
public void I_Enter_My_Regular_Income_Sources(List<Income> incomes) throws Throwable {
// More code
}
Миграция с v2.xx на v3.xx для DataTable
Публикация моего ответа послужит справкой для тех, кто может столкнуться с тем же. Для их объявления о выпуске, нажмите здесь.
Я решил поставить DataTableConfigurer.java
в своем собственном пакете, чтобы он не смешивался с моими отчётами:
Второе место:
@CucumberOptions(features = { "src/test/resources/features" }, tags = { "not @ignore" }, glue = {
"jcucumberng/steps/defs", "jcucumberng/steps/config", "jcucumberng/steps/hooks" }, ...
DataTableConfigurer:
import java.util.Locale;
import java.util.Map;
import cucumber.api.TypeRegistry;
import cucumber.api.TypeRegistryConfigurer;
import io.cucumber.datatable.DataTableType;
import io.cucumber.datatable.TableEntryTransformer;
import jcucumberng.steps.domain.Expense;
import jcucumberng.steps.domain.Income;
/*
* Maps datatables in feature files to custom domain objects.
*/
public class DataTableConfigurer implements TypeRegistryConfigurer {
@Override
public Locale locale() {
return Locale.ENGLISH;
}
@Override
public void configureTypeRegistry(TypeRegistry registry) {
registry.defineDataTableType(new DataTableType(Income.class, new TableEntryTransformer<Income>() {
@Override
public Income transform(Map<String, String> entry) {
return new Income(entry.get("name"), entry.get("amount"), entry.get("frequency"));
}
}));
registry.defineDataTableType(new DataTableType(Expense.class, new TableEntryTransformer<Expense>() {
@Override
public Expense transform(Map<String, String> entry) {
return new Expense(entry.get("name"), entry.get("amount"), entry.get("frequency"));
}
}));
}
}
У меня был другой нестандартный тип домена Expense
(в котором оказались те же поля), поэтому я просто зарегистрировал его снова на основе примера.
I have created a code which won't use **DataTable** concept. You can update this below implementation so that you won't get any failures in your scripts in future.
CucumberUtil.java:
-----------------
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class CucumberUtil {
// public static synchronized Map<String, String> TableDictionaryConverter(DataTable table) { -- removed this concept because of version issues in DataTable
public static synchronized Map<String, String> TableDictionaryConverter(List<List<String>> data) {
Map<String, String> mapTable = new HashMap<String, String>();
for(List<String> rows: data) {
mapTable.put(rows.get(0), rows.get(1));
}
return mapTable;
}
// Feature
When I Enter My Regular Income Sources
| name | Salary |
| amount | 25000 |
| frequency| every 2 weeks |
// Stepdef
@When("^I Enter My Regular Income Sources$")
public void I_Enter_My_Regular_Income_Sources(List<List<String>> table) throws Throwable {
Map<String, String> mapTable = CucumberUtil.TableDictionaryConverter(table);
String nameValue = mapTable.get("name"); // Salary
String amountValue = mapTable.get("name"); // 25000
String frequencyValue = mapTable.get("name"); // every 2 weeks
// More code
}