Почему JOOQ — идеальный инструмент для работы с БД при интеграционном тестировании API
Привет! Меня зовут Евгений, я SDET-специалист в SimbirSoft. Хочу поделиться примером того, как я автоматизировал тестирование API, заменив встроенные JDBC-средства на JOOQ. И расскажу, почему считаю это лучшим решением.
Все началось с того, что передо мной поставили задачу автоматизировать тестирование API с проверкой данных в БД. Так как проект только начинался, а я один отвечал за эту часть работы, то надо было сделать всё с нуля. Мне хотелось сделать все идеально (удобно, понятно, масштабируемо, с удобной поддержкой кода). Получилось все, кроме одного — масштабирование сверки данных из БД. Об этом и пойдет речь. А в конце вы найдете ссылку на исходный код.
Помним, что проект только начинал разрабатываться, поэтому база данных была небольшой и не сильно разветвленной. Поэтому мне показалось уместным использовать встроенные JDBC-средства для этой задачи.
На просторах интернета есть много информации на эту тему, в своей же статье я хотел бы рассказать о своем пути использования и улучшения данного метода с помощью дополнительного фреймворка JOOQ.
Используем JDBC
Для того чтобы подключиться к базе, создается подключение:
connection = DriverManager.getConnection(url, username, password);
После делается запрос в базу данных:
Statement statement = connection.createStatement();
ResultSet resultSet = statement.executeQuery(script);
И получение данных:
while (resultSet.next()) {
Integer id = resultSet.getInt("id”);
String name = resultSet.getString("name”);
}
В моем случае база данных имела много таблиц, которые продолжали изменяться и дополняться, что приводило к изменению уже созданных запросов. Я подумал, как можно улучшить эту систему, и в результате сделал генерацию строки запроса.
public interface CompanyFields {
String COMPANY_TABLE = "company c";
String COMPANY_ID_FIELD = "c.id";
String COMPANY_NAME_FIELD = "c.name";
}
public class QueryBuilder implements AirplaneFields, CompanyFields, ModelFields, FlightFields, PassengerFields
{
private final StringBuffer query = new StringBuffer();
public String build() {
return query.toString();
}
public QueryBuilder select() {
query.append("SELECT ");
return this;
}
public QueryBuilder all() {
query.append("*");
return this;
}
public QueryBuilder from() {
query.append(" FROM ");
return this;
}
public QueryBuilder where() {
query.append(" WHERE ");
return this;
}
public QueryBuilder equals() {
query.append(" = ");
return this;
}
public QueryBuilder addElement(long integer) {
query.append(integer);
return this;
}
public QueryBuilder companyTable() {
query.append(COMPANY_TABLE);
return this;
}
public QueryBuilder companyId() {
query.append(COMPANY_ID_FIELD);
return this;
}
}
В итоге создание запроса выглядело так:
new QueryBuilder().select().all().from().companyTable()
.where().companyId().equals().addElement(id).build()
А также получение итогового файла стало проще:
new CompanyDto(resultSet.getInt(COMPANY_ID_FIELD),
resultSet.getString(COMPANY_NAME_FIELD));
Система мне показалась довольно удобной, пока не начали появляться новые таблицы в очень большом количестве. И тогда я вернулся к тому, с чего начинал и о чем продолжал думать всё это время.
А что, так можно было?
Я задавался вопросом: масштабировал ли кто-нибудь ранее такую систему и, если и делал, то при помощи чего? Практически все статьи были про удобство Spring Data JPA для бэкенда. Я не хотел повторять уже сделанную работу бэкенда, поэтому такое решение было не самым лучшим. Но также во время поисков я наткнулся на статью о другом фреймворке для работы с базой данных. Он работает по схожему с JDBC принципом (формирование запросов) — и это JOOQ.
Если посмотреть на то, как составляется запрос в JOOQ и тот запрос, который я создал выше, можно увидеть большое сходство. Однако классы для описания таблиц в БД были намного сложнее того, что я и так уже сделал. Поэтому я этот вариант отбросил.
И насколько сильным было моё удивление, когда я узнал, что эти классы для JOOQ можно не прописывать, а подтягивать напрямую из самой базы данных!
Для этого нужно добавить несколько зависимостей:
org.postgresql
postgresql
${postgres.driver.version}
org.testng
testng
7.3.0
test
org.jooq
jooq
${jooq.version}
org.jooq
jooq-meta
${jooq.version}
org.jooq
jooq-codegen
${jooq.version}
org.projectlombok
lombok
1.18.24
Здесь можно взять jooq-config.xml файл для подгрузки таблиц из базы данных (для других баз и конфигураций):
org.postgresql.Driver
jdbc:postgresql://localhost:5432/jooq_DB
xuser
password
org.jooq.codegen.JavaGenerator
org.jooq.meta.postgres.PostgresDatabase
public
.*
org.example.jooq.db.autocreated
.\src\test\java
И методы для подключения к базе данных:
public final class DatabaseConnector {
private DatabaseConnector() {
}
private static Connection connection = getConnection();
private static DSLContext context;
public static synchronized Connection getConnection() {
if (connection == null) {
try {
// Получение данных, для подключения к БД
String dbDriverClass = ParametersProvider
.getProperty("jdbc.driver");
String dbUrl = ParametersProvider
.getProperty("jdbc.url");
String dbUsername = ParametersProvider
.getProperty("jdbc.username");
String dbPassword = ParametersProvider
.getProperty("jdbc.password");
Class.forName(dbDriverClass)
connection = DriverManager.getConnection(dbUrl,
dbUsername,
dbPassword);
} catch (SQLException | ClassNotFoundException e) {
throw new RuntimeException("Connection error", e);
}
}
return connection;
}
public static DSLContext getContext() {
if (context == null) {
context = DSL.using(connection, SQLDialect.POSTGRES);
try {
GenerationTool.generate(
Files.readString(
Path.of("src\\test\\resources\\jooq-config.xml")
)
);
} catch (Exception ignored) {
}
}
return context;
}
}
И вуаля, получил я все таблицы! Ну, а дальше их можно просто использовать в удобном виде:
Dtoшка:
@Data
@NoArgsConstructor
@AllArgsConstructor
public class CompanyDto {
private Integer id;
private String name;
}
Методы для получения данных:
public CompanyDto getCompanyById(Integer id) {
return context.select()
.from(Company.COMPANY)
.where(Company.COMPANY.ID.eq(id))
.fetch()
.map(this::getCompanyDtoByRecord)
.get(0);
}
public CompanyDto getCompanyDtoByRecord(Record record) {
return new dto.CompanyDto(
record.getValue(Company.COMPANY.ID),
record.getValue(Company.COMPANY.NAME)
);
}
Сам JOOQ позволяет строить абсолютно любые запросы, в том числе на создание, изменение, удаление как самих баз, так и данных внутри них.
Вывод
В интернете есть много статей, в которых сравнивают разные способы работы с базой данных. Но я так и не увидел ни одной, где бы сказали, что это идеальное решение.
По моему мнению, для интеграционного тестирования API JOOQ будет как раз таким. Простой способ взаимодействия с БД (через запросы), автоматическая подгрузка таблиц (в условиях, когда не наш код отвечает за формирование базы данных), простая поддержка (при изменении базы данных, все необходимые места для изменения в коде будут подсвечены). То есть удобно, понятно, масштабируемо и с удобной поддержкой кода. Как и требовалось
Исходный код находится здесь.
Напишите в комментариях, какое бы вы использовали решение данной задачи?
Спасибо за внимание!
Больше авторских материалов для SDET-специалистов от моих коллег читайте в соцсетях SimbirSoft — ВКонтакте и Telegram.