diff --git a/.sdkmanrc b/.sdkmanrc index 4e7acfef..95502c0c 100644 --- a/.sdkmanrc +++ b/.sdkmanrc @@ -3,4 +3,4 @@ # See https://sdkman.io/usage#config # A summary is to add the following to ~/.sdkman/etc/config # sdkman_auto_env=true -java=17.0.11-tem +java=17.0.11-tem \ No newline at end of file diff --git a/langchain4j-qianfan-spring-boot-starter/pom.xml b/langchain4j-qianfan-spring-boot-starter/pom.xml index 4e58a540..b59d121c 100644 --- a/langchain4j-qianfan-spring-boot-starter/pom.xml +++ b/langchain4j-qianfan-spring-boot-starter/pom.xml @@ -6,7 +6,7 @@ dev.langchain4j langchain4j-spring - 0.30.0 + 0.32.0-SNAPSHOT ../pom.xml diff --git a/langchain4j-redis-spring-boot-starter/pom.xml b/langchain4j-redis-spring-boot-starter/pom.xml new file mode 100644 index 00000000..7227a0e7 --- /dev/null +++ b/langchain4j-redis-spring-boot-starter/pom.xml @@ -0,0 +1,91 @@ + + + 4.0.0 + + dev.langchain4j + langchain4j-spring + 0.32.0-SNAPSHOT + ../pom.xml + + + langchain4j-redis-spring-boot-starter + LangChain4j Spring Boot starter for Redis + jar + + + + + dev.langchain4j + langchain4j-redis + + + + org.springframework.boot + spring-boot-starter + + + + org.springframework.boot + spring-boot-autoconfigure-processor + true + + + + + org.projectlombok + lombok + provided + + + + + org.springframework.boot + spring-boot-configuration-processor + true + + + + org.springframework.boot + spring-boot-starter-test + test + + + + dev.langchain4j + langchain4j-embeddings-all-minilm-l6-v2-q + test + + + + dev.langchain4j + langchain4j-spring-boot-tests + ${project.version} + tests + test-jar + test + + + + com.redis.testcontainers + testcontainers-redis + 1.6.4 + test + + + + org.tinylog + tinylog-impl + test + + + + org.tinylog + slf4j-tinylog + test + + + + + \ No newline at end of file diff --git a/langchain4j-redis-spring-boot-starter/src/main/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreAutoConfiguration.java b/langchain4j-redis-spring-boot-starter/src/main/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreAutoConfiguration.java new file mode 100644 index 00000000..bff93acc --- /dev/null +++ b/langchain4j-redis-spring-boot-starter/src/main/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreAutoConfiguration.java @@ -0,0 +1,47 @@ +package dev.langchain4j.store.embedding.redis.spring; + +import dev.langchain4j.model.embedding.EmbeddingModel; +import dev.langchain4j.store.embedding.redis.RedisEmbeddingStore; +import org.springframework.boot.autoconfigure.AutoConfiguration; +import org.springframework.boot.autoconfigure.condition.ConditionalOnMissingBean; +import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty; +import org.springframework.boot.context.properties.EnableConfigurationProperties; +import org.springframework.context.annotation.Bean; +import org.springframework.lang.Nullable; + +import java.util.ArrayList; +import java.util.List; +import java.util.Optional; + +import static dev.langchain4j.store.embedding.redis.spring.RedisEmbeddingStoreProperties.PREFIX; + +@AutoConfiguration +@EnableConfigurationProperties(RedisEmbeddingStoreProperties.class) +@ConditionalOnProperty(prefix = PREFIX, name = "enabled", havingValue = "true", matchIfMissing = true) +public class RedisEmbeddingStoreAutoConfiguration { + + private static final String DEFAULT_HOST = "localhost"; + private static final int DEFAULT_PORT = 6379; + private static final String DEFAULT_INDEX_NAME = "langchain4j-index"; + + @Bean + @ConditionalOnMissingBean + public RedisEmbeddingStore redisEmbeddingStore(RedisEmbeddingStoreProperties properties, + @Nullable EmbeddingModel embeddingModel) { + String host = Optional.ofNullable(properties.getHost()).orElse(DEFAULT_HOST); + int port = Optional.ofNullable(properties.getPort()).orElse(DEFAULT_PORT); + String indexName = Optional.ofNullable(properties.getIndexName()).orElse(DEFAULT_INDEX_NAME); + Integer dimension = Optional.ofNullable(properties.getDimension()).orElseGet(() -> embeddingModel == null ? null : embeddingModel.dimension()); + List metadataKeys = Optional.ofNullable(properties.getMetadataKeys()).orElse(new ArrayList<>()); + + return RedisEmbeddingStore.builder() + .host(host) + .port(port) + .user(properties.getUser()) + .password(properties.getPassword()) + .indexName(indexName) + .dimension(dimension) + .metadataKeys(metadataKeys) + .build(); + } +} diff --git a/langchain4j-redis-spring-boot-starter/src/main/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreProperties.java b/langchain4j-redis-spring-boot-starter/src/main/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreProperties.java new file mode 100644 index 00000000..daf07961 --- /dev/null +++ b/langchain4j-redis-spring-boot-starter/src/main/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreProperties.java @@ -0,0 +1,24 @@ +package dev.langchain4j.store.embedding.redis.spring; + +import lombok.Getter; +import lombok.Setter; +import org.springframework.boot.context.properties.ConfigurationProperties; + +import java.util.ArrayList; +import java.util.List; + +@ConfigurationProperties(prefix = RedisEmbeddingStoreProperties.PREFIX) +@Getter +@Setter +public class RedisEmbeddingStoreProperties { + + static final String PREFIX = "langchain4j.redis"; + + private String host; + private Integer port; + private String user; + private String password; + private String indexName; + private Integer dimension; + private List metadataKeys; +} diff --git a/langchain4j-redis-spring-boot-starter/src/main/resources/META-INF/spring.factories b/langchain4j-redis-spring-boot-starter/src/main/resources/META-INF/spring.factories new file mode 100644 index 00000000..c03ec78c --- /dev/null +++ b/langchain4j-redis-spring-boot-starter/src/main/resources/META-INF/spring.factories @@ -0,0 +1 @@ +org.springframework.boot.autoconfigure.EnableAutoConfiguration=dev.langchain4j.store.embedding.redis.spring.RedisEmbeddingStoreAutoConfiguration \ No newline at end of file diff --git a/langchain4j-redis-spring-boot-starter/src/test/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreAutoConfigurationIT.java b/langchain4j-redis-spring-boot-starter/src/test/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreAutoConfigurationIT.java new file mode 100644 index 00000000..aa05e4a2 --- /dev/null +++ b/langchain4j-redis-spring-boot-starter/src/test/java/dev/langchain4j/store/embedding/redis/spring/RedisEmbeddingStoreAutoConfigurationIT.java @@ -0,0 +1,61 @@ +package dev.langchain4j.store.embedding.redis.spring; + +import com.redis.testcontainers.RedisContainer; +import dev.langchain4j.data.segment.TextSegment; +import dev.langchain4j.store.embedding.EmbeddingStore; +import dev.langchain4j.store.embedding.redis.RedisEmbeddingStore; +import dev.langchain4j.store.embedding.spring.EmbeddingStoreAutoConfigurationIT; +import org.junit.jupiter.api.AfterAll; +import org.junit.jupiter.api.BeforeAll; +import org.junit.jupiter.api.BeforeEach; +import org.testcontainers.containers.wait.strategy.Wait; +import redis.clients.jedis.JedisPooled; + +import static com.redis.testcontainers.RedisStackContainer.DEFAULT_IMAGE_NAME; +import static com.redis.testcontainers.RedisStackContainer.DEFAULT_TAG; + +class RedisEmbeddingStoreAutoConfigurationIT extends EmbeddingStoreAutoConfigurationIT { + + static RedisContainer redis = new RedisContainer(DEFAULT_IMAGE_NAME.withTag(DEFAULT_TAG)) + .waitingFor(Wait.defaultWaitStrategy()); + + @BeforeAll + static void beforeAll() { + redis.start(); + } + + @AfterAll + static void afterAll() { + redis.stop(); + } + + @BeforeEach + void beforeEach() { + try (JedisPooled jedis = new JedisPooled(redis.getHost(), redis.getFirstMappedPort())) { + jedis.flushDB(); // TODO fix: why redis returns embeddings from different indexes? + } + } + + @Override + protected Class autoConfigurationClass() { + return RedisEmbeddingStoreAutoConfiguration.class; + } + + @Override + protected Class> embeddingStoreClass() { + return RedisEmbeddingStore.class; + } + + @Override + protected String[] properties() { + return new String[]{ + "langchain4j.redis.host=" + redis.getHost(), + "langchain4j.redis.port=" + redis.getFirstMappedPort() + }; + } + + @Override + protected String dimensionPropertyKey() { + return "langchain4j.redis.dimension"; + } +} diff --git a/pom.xml b/pom.xml index 3fac0e75..b5df5ed1 100644 --- a/pom.xml +++ b/pom.xml @@ -24,6 +24,7 @@ langchain4j-azure-open-ai-spring-boot-starter langchain4j-vertex-ai-gemini-spring-boot-starter langchian4j-elasticsearch-spring-boot-starter + langchain4j-redis-spring-boot-starter langchain4j-qianfan-spring-boot-starter @@ -31,6 +32,7 @@ 17 17 UTF-8 + 2.6.2 3.2.6 1.19.8 2.6.2