-
Notifications
You must be signed in to change notification settings - Fork 2
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
add spring boot integration #83
base: master
Are you sure you want to change the base?
Changes from all commits
9c53ed8
20a56db
75ab59d
bc95096
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -5,14 +5,14 @@ | |
import lombok.Getter; | ||
import lombok.NoArgsConstructor; | ||
import lombok.Setter; | ||
import lombok.SneakyThrows; | ||
import org.apache.hc.client5.http.auth.AuthScope; | ||
import org.apache.hc.client5.http.auth.UsernamePasswordCredentials; | ||
import org.apache.hc.client5.http.impl.auth.BasicCredentialsProvider; | ||
import org.apache.hc.client5.http.impl.nio.PoolingAsyncClientConnectionManager; | ||
import org.apache.hc.client5.http.impl.nio.PoolingAsyncClientConnectionManagerBuilder; | ||
import org.apache.hc.client5.http.ssl.ClientTlsStrategyBuilder; | ||
import org.apache.hc.client5.http.ssl.NoopHostnameVerifier; | ||
import org.apache.hc.client5.http.ssl.TrustAllStrategy; | ||
import org.apache.hc.core5.http.HttpHost; | ||
import org.apache.hc.core5.http.nio.ssl.TlsStrategy; | ||
import org.apache.hc.core5.reactor.ssl.TlsDetails; | ||
|
@@ -144,7 +144,7 @@ private void connect() { | |
try { | ||
sslcontext = SSLContextBuilder | ||
.create() | ||
.loadTrustMaterial(null, (chains, authType) -> true) | ||
.loadTrustMaterial(null, new TrustAllStrategy()) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think this is fine for tests but not production, the `TrustAllStrategy is unsafe There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. this is unrelated to this PR (sorry, i just snuck it in here, but at least as an atomic commit) - this commit doesn't change the behaviour at all, it's just making it more obvious what's happening here. there's #29 to improve this situation (but no clear idea on how to solve it yet). |
||
.build(); | ||
} catch (final NoSuchAlgorithmException | KeyManagementException | KeyStoreException e) { | ||
throw new RuntimeException(e); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
package liquibase.ext.opensearch.integration.spring; | ||
|
||
import org.opensearch.client.opensearch.OpenSearchClient; | ||
import org.springframework.boot.autoconfigure.AutoConfiguration; | ||
import org.springframework.boot.autoconfigure.condition.AllNestedConditions; | ||
import org.springframework.boot.autoconfigure.condition.ConditionalOnClass; | ||
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; | ||
import org.springframework.boot.context.properties.EnableConfigurationProperties; | ||
import org.springframework.context.annotation.Bean; | ||
import org.springframework.context.annotation.Conditional; | ||
|
||
@AutoConfiguration | ||
@EnableConfigurationProperties(SpringLiquibaseOpenSearchProperties.class) | ||
@Conditional(LiquibaseOpenSearchAutoConfiguration.LiquibaseOpenSearchCondition.class) | ||
@ConditionalOnClass(OpenSearchClient.class) | ||
public class LiquibaseOpenSearchAutoConfiguration { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think we need to account for OpenSearch related beans (clients, etc) to be present (or not) in the context, I am happy to help you there (if it makes sense) There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
|
||
static final class LiquibaseOpenSearchCondition extends AllNestedConditions { | ||
LiquibaseOpenSearchCondition() { | ||
super(ConfigurationPhase.REGISTER_BEAN); | ||
} | ||
|
||
@ConditionalOnProperty(prefix = "opensearch", name = "liquibase.enabled", matchIfMissing = true) | ||
private static final class LiquibaseOpenSearchEnabledCondition { | ||
} | ||
|
||
@ConditionalOnProperty(prefix = "opensearch", name = "uris") | ||
private static final class LiquibaseOpenSearchUrlCondition { | ||
|
||
} | ||
} | ||
|
||
@Bean | ||
public SpringLiquibaseOpenSearch getLiquibaseOpenSearch() { | ||
return new SpringLiquibaseOpenSearch(); | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,54 @@ | ||
package liquibase.ext.opensearch.integration.spring; | ||
|
||
import liquibase.Scope; | ||
import liquibase.UpdateSummaryOutputEnum; | ||
import liquibase.command.CommandScope; | ||
import liquibase.command.core.UpdateCommandStep; | ||
import liquibase.command.core.helpers.DbUrlConnectionArgumentsCommandStep; | ||
import liquibase.command.core.helpers.ShowSummaryArgument; | ||
import liquibase.database.DatabaseFactory; | ||
import liquibase.ext.opensearch.database.OpenSearchConnection; | ||
import liquibase.ext.opensearch.database.OpenSearchLiquibaseDatabase; | ||
import liquibase.integration.spring.SpringResourceAccessor; | ||
import liquibase.ui.UIServiceEnum; | ||
import lombok.Getter; | ||
import org.springframework.beans.factory.InitializingBean; | ||
import org.springframework.beans.factory.annotation.Autowired; | ||
import org.springframework.core.io.ResourceLoader; | ||
|
||
import static liquibase.ext.opensearch.database.OpenSearchLiquibaseDatabase.OPENSEARCH_PREFIX; | ||
import static liquibase.ext.opensearch.database.OpenSearchLiquibaseDatabase.OPENSEARCH_URI_SEPARATOR; | ||
|
||
public class SpringLiquibaseOpenSearch implements InitializingBean { | ||
|
||
@Autowired | ||
private ResourceLoader resourceLoader; | ||
|
||
@Autowired | ||
private SpringLiquibaseOpenSearchProperties properties; | ||
|
||
@Getter | ||
protected UIServiceEnum uiService = UIServiceEnum.LOGGER; | ||
|
||
@Override | ||
public void afterPropertiesSet() throws Exception { | ||
// liquibase requires the prefix to identify this as an OpenSearch database. | ||
final var url = OPENSEARCH_PREFIX + String.join(OPENSEARCH_URI_SEPARATOR, properties.uris()); | ||
|
||
Scope.child(Scope.Attr.ui.name(), this.uiService.getUiServiceClass().getDeclaredConstructor().newInstance(), | ||
() -> { | ||
final var database = (OpenSearchLiquibaseDatabase) DatabaseFactory.getInstance().openDatabase(url, properties.username(), properties.password(), null, new SpringResourceAccessor(this.resourceLoader)); | ||
final var connection = (OpenSearchConnection) database.getConnection(); | ||
new CommandScope(UpdateCommandStep.COMMAND_NAME) | ||
.addArgumentValue(ShowSummaryArgument.SHOW_SUMMARY_OUTPUT, UpdateSummaryOutputEnum.LOG) | ||
.addArgumentValue(DbUrlConnectionArgumentsCommandStep.DATABASE_ARG, database) | ||
.addArgumentValue(UpdateCommandStep.CHANGELOG_FILE_ARG, properties.liquibase().changelogFile()) | ||
.addArgumentValue(UpdateCommandStep.CONTEXTS_ARG, properties.liquibase().contexts()) | ||
.addArgumentValue(UpdateCommandStep.LABEL_FILTER_ARG, properties.liquibase().labelFilterArgs()) | ||
.execute(); | ||
connection.close(); | ||
database.close(); | ||
}); | ||
} | ||
|
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,40 @@ | ||
package liquibase.ext.opensearch.integration.spring; | ||
|
||
import org.springframework.boot.context.properties.ConfigurationProperties; | ||
import org.springframework.boot.context.properties.bind.DefaultValue; | ||
|
||
import java.util.List; | ||
|
||
/** | ||
* @param uris URL of the OpenSearch instances to use. If multiple URLs are listed they must all belong to the same cluster. | ||
* @param username Username for authentication with OpenSearch. | ||
* @param password Password for authentication with OpenSearch. | ||
* @param liquibase Liquibase-specific properties. | ||
*/ | ||
@ConfigurationProperties("opensearch") | ||
public record SpringLiquibaseOpenSearchProperties( | ||
@DefaultValue("http://localhost:9200") | ||
List<String> uris, | ||
String username, | ||
String password, | ||
|
||
@DefaultValue() | ||
SpringLiquibaseProperties liquibase | ||
) { | ||
/** | ||
* @param changelogFile Path to the liquibase changelog file (must be on the classpath). | ||
* @param contexts Context filter to be used. | ||
* @param labelFilterArgs Label filter to be used. | ||
*/ | ||
public record SpringLiquibaseProperties( | ||
@DefaultValue("true") | ||
Boolean enabled, | ||
|
||
@DefaultValue("db.changelog/db.changelog-master.yaml") | ||
String changelogFile, | ||
|
||
String contexts, | ||
|
||
String labelFilterArgs | ||
) {} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1 @@ | ||
liquibase.ext.opensearch.integration.spring.LiquibaseOpenSearchAutoConfiguration |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,39 @@ | ||
package liquibase.ext.opensearch.integration.spring; | ||
|
||
import org.junit.jupiter.api.Test; | ||
import org.opensearch.testcontainers.OpensearchContainer; | ||
import org.springframework.boot.test.context.SpringBootTest; | ||
import org.springframework.test.context.DynamicPropertyRegistry; | ||
import org.springframework.test.context.DynamicPropertySource; | ||
import org.testcontainers.junit.jupiter.Container; | ||
import org.testcontainers.junit.jupiter.Testcontainers; | ||
import org.testcontainers.utility.DockerImageName; | ||
|
||
import static liquibase.ext.opensearch.AbstractOpenSearchLiquibaseIT.OPENSEARCH_DOCKER_IMAGE_NAME; | ||
|
||
@Testcontainers | ||
@SpringBootTest(properties = { | ||
"spring.datasource.url=jdbc:h2:mem:testdb;MODE=MySQL;DB_CLOSE_DELAY=-1;DB_CLOSE_ON_EXIT=FALSE", | ||
"spring.datasource.platform=h2", | ||
"spring.liquibase.change-log=classpath:/liquibase/db/liquibase-changeLog.xml" | ||
}) | ||
class SpringLiquibaseOpenSearchDataSourceIT { | ||
|
||
@Container | ||
protected static OpensearchContainer<?> container = new OpensearchContainer<>(DockerImageName.parse(OPENSEARCH_DOCKER_IMAGE_NAME)); | ||
|
||
@DynamicPropertySource | ||
static void setProperties(DynamicPropertyRegistry registry) { | ||
registry.add("opensearch.uris", container::getHttpHostAddress); | ||
registry.add("opensearch.username", container::getUsername); | ||
registry.add("opensearch.password", container::getPassword); | ||
} | ||
|
||
/** | ||
* On context load liquibase is automatically being triggered (hard to test as we'd have to construct our own OpenSearch client). | ||
*/ | ||
@Test | ||
void contextLoads() { | ||
} | ||
|
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
package liquibase.ext.opensearch.integration.spring; | ||
|
||
import org.junit.jupiter.api.Test; | ||
import org.springframework.beans.factory.annotation.Autowired; | ||
import org.springframework.boot.test.context.SpringBootTest; | ||
import org.springframework.context.ApplicationContext; | ||
import org.testcontainers.junit.jupiter.Testcontainers; | ||
|
||
import static org.hamcrest.collection.IsEmptyCollection.empty; | ||
import static org.hamcrest.MatcherAssert.assertThat; | ||
|
||
@Testcontainers | ||
@SpringBootTest(properties = { | ||
"spring.liquibase.enabled=false", | ||
"opensearch.liquibase.enabled=false" | ||
}) | ||
class SpringLiquibaseOpenSearchDisabledIT { | ||
@Autowired private ApplicationContext context; | ||
|
||
@Test | ||
void contextLoads() { | ||
assertThat(context.getBeansOfType(SpringLiquibaseOpenSearch.class).keySet(), empty()); | ||
} | ||
|
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,37 @@ | ||
package liquibase.ext.opensearch.integration.spring; | ||
|
||
import org.junit.jupiter.api.Test; | ||
import org.opensearch.testcontainers.OpensearchContainer; | ||
import org.springframework.boot.test.context.SpringBootTest; | ||
import org.springframework.test.context.DynamicPropertyRegistry; | ||
import org.springframework.test.context.DynamicPropertySource; | ||
import org.testcontainers.junit.jupiter.Container; | ||
import org.testcontainers.junit.jupiter.Testcontainers; | ||
import org.testcontainers.utility.DockerImageName; | ||
|
||
import static liquibase.ext.opensearch.AbstractOpenSearchLiquibaseIT.OPENSEARCH_DOCKER_IMAGE_NAME; | ||
|
||
@Testcontainers | ||
@SpringBootTest(properties = { | ||
"spring.liquibase.enabled=false" | ||
}) | ||
class SpringLiquibaseOpenSearchIT { | ||
|
||
@Container | ||
protected static OpensearchContainer<?> container = new OpensearchContainer<>(DockerImageName.parse(OPENSEARCH_DOCKER_IMAGE_NAME)); | ||
|
||
@DynamicPropertySource | ||
static void setProperties(DynamicPropertyRegistry registry) { | ||
registry.add("opensearch.uris", container::getHttpHostAddress); | ||
registry.add("opensearch.username", container::getUsername); | ||
registry.add("opensearch.password", container::getPassword); | ||
} | ||
|
||
/** | ||
* On context load liquibase is automatically being triggered (hard to test as we'd have to construct our own OpenSearch client). | ||
*/ | ||
@Test | ||
void contextLoads() { | ||
} | ||
|
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
package liquibase.ext.opensearch.integration.spring; | ||
|
||
import org.springframework.boot.SpringApplication; | ||
import org.springframework.boot.autoconfigure.SpringBootApplication; | ||
|
||
@SpringBootApplication | ||
public class SpringTestApplication { | ||
|
||
public static void main(String[] args) { | ||
SpringApplication.run(SpringTestApplication.class, args); | ||
} | ||
|
||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,3 @@ | ||
spring.application.name=liquibase-opensearch-test | ||
|
||
opensearch.liquibase.changelog-file=liquibase/ext/changelog.httprequest.yaml |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
<databaseChangeLog xmlns="http://www.liquibase.org/xml/ns/dbchangelog" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" | ||
xsi:schemaLocation="http://www.liquibase.org/xml/ns/dbchangelog http://www.liquibase.org/xml/ns/dbchangelog/dbchangelog-latest.xsd"> | ||
<changeSet id="1" author="user"> | ||
<createTable tableName="car"> | ||
<column name="id" type="bigint" autoIncrement="${autoIncrement}"> | ||
<constraints primaryKey="true" nullable="false" /> | ||
</column> | ||
<column name="make" type="varchar(255)"> | ||
<constraints nullable="true" /> | ||
</column> | ||
<column name="brand" type="varchar(255)"> | ||
<constraints nullable="true" /> | ||
</column> | ||
<column name="price" type="double"> | ||
<constraints nullable="true" /> | ||
</column> | ||
</createTable> | ||
</changeSet> | ||
</databaseChangeLog> |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It seems like liquibase does not follow the convention, but it would be great to have
spring-boot-liquibase-opensearch-starter
/liquibase-opensearch-starter
as a separate module