Skip to content
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 Project Starter for Google Gemini API model - ChatLangauge, Streaming model and Embedding Model #74

Open
wants to merge 55 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 42 commits
Commits
Show all changes
55 commits
Select commit Hold shift + click to select a range
89fa967
Base Commit for Google Gemini AI Model
Suhas-Koheda Nov 13, 2024
cb8544f
Added Configuration files for Google Gemini AI - ChatModel
Suhas-Koheda Nov 13, 2024
3a8765c
Added Configuration files for Google Gemini AI - ChatModel
Suhas-Koheda Nov 13, 2024
0dfbc0d
Check
Suhas-Koheda Nov 13, 2024
8335e67
Adding Streaming Google AI Chat Model and Removing Lombok uses
Suhas-Koheda Nov 13, 2024
84bc5e3
Adding application.properties.local for easy usage
Suhas-Koheda Nov 13, 2024
6d7c4cf
Ignore all the .idea folders in subfolders
Suhas-Koheda Nov 13, 2024
300cfa2
Ignore all the .idea folders in subfolders
Suhas-Koheda Nov 13, 2024
9618410
Ignore all the .idea folders in subfolders
Suhas-Koheda Nov 13, 2024
7ee36c6
Changing pom.xml headers
Suhas-Koheda Nov 14, 2024
f696e82
Resolving API_KEY to apikey
Suhas-Koheda Nov 14, 2024
4b8f2ec
Resolving API_KEY to apikey
Suhas-Koheda Nov 14, 2024
9453ced
Resolving API_KEY to apikey
Suhas-Koheda Nov 14, 2024
4f267ff
Change Response Format to User Defined Props
Suhas-Koheda Nov 15, 2024
affefe3
Change Response Format to User Defined Props
Suhas-Koheda Nov 15, 2024
76053d9
Write test to check working of ChatLangaugeModel and StreamingCHatLan…
Suhas-Koheda Nov 15, 2024
6ee971e
Complete Test for Chat Language Model
Suhas-Koheda Nov 15, 2024
637cd0e
Complete Test for Chat Language Model
Suhas-Koheda Nov 15, 2024
3e5fca5
Implement test for Streaming Chat Language Model
Suhas-Koheda Nov 15, 2024
cfc5cae
Delete langchain4j-google-ai-gemini-spring-boot-starter/src/main/reso…
Suhas-Koheda Nov 15, 2024
66c828e
Changing Java Version in pom.xml from 21 to 17 to support maven (mvn)…
Suhas-Koheda Nov 16, 2024
11d7989
Merge remote-tracking branch 'refs/remotes/origin/main'
Suhas-Koheda Nov 16, 2024
ec1275b
Changing Java version back to 21 :) in pom.xml
Suhas-Koheda Nov 16, 2024
0fd6868
Resolving other issues except Embedding model
Suhas-Koheda Nov 18, 2024
070a71e
Adding env variable to main.yaml and release.yaml
Suhas-Koheda Nov 18, 2024
7aa79cb
Merge branch 'langchain4j:main' into main
Suhas-Koheda Nov 18, 2024
889aa2f
Update release.yaml to add Google AI Gemini API Key
Suhas-Koheda Nov 18, 2024
db889db
Update main.yaml to add Google AI Gemini API Key
Suhas-Koheda Nov 18, 2024
b626e12
Merge branch 'main' of https://github.com/Suhas-Koheda/langchain4j-sp…
Suhas-Koheda Nov 18, 2024
4ae025a
Changing the version of dependencies to project version
Suhas-Koheda Nov 18, 2024
9aad39f
adding GeminiSafetyProperties and GeminiSafetySettings to googleGemin…
Suhas-Koheda Nov 19, 2024
3c32174
Add tests for chatlanguagemodel with updated properties
Suhas-Koheda Nov 19, 2024
c4ba686
Adding Embedding model configuration and test for Embedding model
Suhas-Koheda Nov 20, 2024
ef71908
Merge branch 'langchain4j:main' into main
Suhas-Koheda Nov 20, 2024
b0a8f9a
Resolving issues in spring boot starter for google ai gemini
Suhas-Koheda Nov 26, 2024
89694f5
Update main.yaml for google ai gemini key
Suhas-Koheda Nov 26, 2024
907424b
Update release.yaml for google ai gemini key
Suhas-Koheda Nov 26, 2024
6159165
refactoring resources
Suhas-Koheda Nov 26, 2024
4a27ef1
Merge remote-tracking branch 'refs/remotes/origin/main'
Suhas-Koheda Nov 26, 2024
7554b25
Changing the safteySetting and toolConfig and writing test
Suhas-Koheda Nov 30, 2024
4afe606
Merge branch 'langchain4j:main' into main
Suhas-Koheda Dec 2, 2024
69660a4
Resolving silly mistake
Suhas-Koheda Dec 3, 2024
3107cb9
Resolving issues
Suhas-Koheda Dec 7, 2024
56d7546
Changing bean properties
Suhas-Koheda Dec 9, 2024
ed5afe0
cosmetics
Dec 16, 2024
9b52b6e
Merge branch 'langchain4j:main' into main
Suhas-Koheda Dec 21, 2024
9c167d1
Adding NPE check for Safety Setting and Tool Config
Suhas-Koheda Dec 21, 2024
816652c
Changing safetySettings to map of harm category and harm threshold
Suhas-Koheda Jan 3, 2025
692c243
Checking Safety Settings and Tool Config
Suhas-Koheda Jan 7, 2025
fd1a2f4
Remove unused imports
Suhas-Koheda Jan 7, 2025
3ee4b30
Add detailed tests
Suhas-Koheda Jan 8, 2025
ecd46fd
-
Suhas-Koheda Jan 8, 2025
5cc8465
-
Suhas-Koheda Jan 8, 2025
bf04308
Change in tests
Suhas-Koheda Jan 8, 2025
320d596
Merge branch 'langchain4j:main' into main
Suhas-Koheda Jan 11, 2025
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
3 changes: 2 additions & 1 deletion .github/workflows/main.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -34,6 +34,7 @@ jobs:
AZURE_SEARCH_ENDPOINT: ${{ secrets.AZURE_SEARCH_ENDPOINT }}
AZURE_SEARCH_KEY: ${{ secrets.AZURE_SEARCH_KEY }}
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
GOOGLE_AI_GEMINI_API_KEY: ${{secrets.GOOGLE_AI_GEMINI_API_KEY}}
OPENAI_API_KEY: ${{ secrets.OPENAI_API_KEY }}
VOYAGE_API_KEY: ${{ secrets.VOYAGE_API_KEY }}
OSSRH_USERNAME: ${{ secrets.OSSRH_USERNAME }}
Expand All @@ -50,4 +51,4 @@ jobs:
distribution: 'temurin'
cache: 'maven'
- name: License Compliance
run: mvn -U -P compliance org.honton.chas:license-maven-plugin:compliance
run: mvn -U -P compliance org.honton.chas:license-maven-plugin:compliance
1 change: 1 addition & 0 deletions .github/workflows/release.yaml
Original file line number Diff line number Diff line change
Expand Up @@ -33,6 +33,7 @@ jobs:
AZURE_SEARCH_ENDPOINT: ${{ secrets.AZURE_SEARCH_ENDPOINT }}
AZURE_SEARCH_KEY: ${{ secrets.AZURE_SEARCH_KEY }}
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
GOOGLE_AI_GEMINI_API_KEY: ${{secrets.GOOGLE_AI_GEMINI_API_KEY}}
OPENAI_API_KEY: ${{ secrets.OPENAI_API_KEY }}
VOYAGE_API_KEY: ${{ secrets.VOYAGE_API_KEY }}
GPG_PASSPHRASE: ${{ secrets.GPG_PASSPHRASE }}
Expand Down
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ target/

### IntelliJ IDEA ###
.idea/*
**/.idea/
.idea/modules.xml
.idea/jarRepositories.xml
.idea/compiler.xml
Expand Down
63 changes: 63 additions & 0 deletions langchain4j-google-ai-gemini-spring-boot-starter/pom.xml
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns="http://maven.apache.org/POM/4.0.0"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>

<parent>
<groupId>dev.langchain4j</groupId>
<artifactId>langchain4j-spring</artifactId>
<version>0.37.0-SNAPSHOT</version>
<relativePath>../pom.xml</relativePath>
</parent>

<artifactId>langchain4j-google-ai-gemini-spring-boot-starter</artifactId>
Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved
<name>LangChain4j Spring Boot starter for Google AI Gemini</name>
<packaging>jar</packaging>

<dependencies>

<dependency>
<groupId>dev.langchain4j</groupId>
<artifactId>langchain4j-google-ai-gemini</artifactId>
<version>${project.version}</version>
</dependency>

<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter</artifactId>
</dependency>

Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-autoconfigure-processor</artifactId>
<optional>true</optional>
</dependency>

<!-- needed to generate automatic metadata about available config properties -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-configuration-processor</artifactId>
<optional>true</optional>
</dependency>

<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
</dependency>



</dependencies>

<licenses>
<license>
<name>Apache-2.0</name>
<url>https://www.apache.org/licenses/LICENSE-2.0.txt</url>
<distribution>repo</distribution>
<comments>A business-friendly OSS license</comments>
</license>
</licenses>

</project>
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
package dev.langchain4j.googleaigemini.spring;

import dev.langchain4j.model.chat.ChatLanguageModel;
import dev.langchain4j.model.chat.StreamingChatLanguageModel;
import dev.langchain4j.model.embedding.EmbeddingModel;
import dev.langchain4j.model.googleai.GoogleAiEmbeddingModel;
import dev.langchain4j.model.googleai.GoogleAiGeminiChatModel;
import dev.langchain4j.model.googleai.GoogleAiGeminiStreamingChatModel;

import org.springframework.boot.autoconfigure.AutoConfiguration;
import org.springframework.boot.autoconfigure.condition.ConditionalOnProperty;
import org.springframework.boot.context.properties.EnableConfigurationProperties;
import org.springframework.context.annotation.Bean;

import java.util.Map;

import static dev.langchain4j.googleaigemini.spring.Properties.PREFIX;

@AutoConfiguration
@EnableConfigurationProperties(Properties.class)
public class AutoConfig {

@Bean
@ConditionalOnProperty(name = PREFIX + ".chatModel.enabled", havingValue = "true")
langchain4j marked this conversation as resolved.
Show resolved Hide resolved
ChatLanguageModel googleAiGeminiChatModel(Properties properties) {
Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved
ChatModelProperties chatModelProperties = properties.getChatModel();
return GoogleAiGeminiChatModel.builder()
.apiKey(properties.getApiKey())
.modelName(chatModelProperties.getModelName())
.temperature(chatModelProperties.getTemperature())
.topP(chatModelProperties.getTopP())
.topK(chatModelProperties.getTopK())
.maxOutputTokens(chatModelProperties.getMaxOutputTokens())
.responseFormat(chatModelProperties.getResponseFormat())
.logRequestsAndResponses(chatModelProperties.getLogRequestsAndResponses())
.safetySettings(
Map.of(chatModelProperties.getSafetySetting().getGeminiHarmCategory(),
chatModelProperties.getSafetySetting().getGeminiHarmBlockThreshold()))
.toolConfig(
chatModelProperties.getFunctionCallingConfig().getGeminiMode(),
chatModelProperties.getFunctionCallingConfig().getAllowedFunctionNames().toArray(new String[0]))
.build();
}

@Bean
@ConditionalOnProperty(name = PREFIX + ".streamingChatModel.enabled", havingValue = "true")
StreamingChatLanguageModel googleAiGeminiStreamingChatModel(Properties properties) {
Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved
ChatModelProperties chatModelProperties = properties.getStreamingChatModel();
return GoogleAiGeminiStreamingChatModel.builder()
.apiKey(properties.getApiKey())
.modelName(chatModelProperties.getModelName())
.temperature(chatModelProperties.getTemperature())
.topP(chatModelProperties.getTopP())
.topK(chatModelProperties.getTopK())
.responseFormat(chatModelProperties.getResponseFormat())
.logRequestsAndResponses(chatModelProperties.getLogRequestsAndResponses())
.safetySettings(
Map.of(chatModelProperties.getSafetySetting().getGeminiHarmCategory(),
chatModelProperties.getSafetySetting().getGeminiHarmBlockThreshold()))
.toolConfig(
chatModelProperties.getFunctionCallingConfig().getGeminiMode(),
chatModelProperties.getFunctionCallingConfig().getAllowedFunctionNames().toArray(new String[0]))
.build();
}

@Bean
@ConditionalOnProperty(name = PREFIX + ".embeddingModel.enabled", havingValue = "true")
EmbeddingModel googleAiGeminiEmbeddingModel(Properties properties) {
Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved
EmbeddingModelProperties embeddingModelProperties = properties.getEmbeddingModel();
return GoogleAiEmbeddingModel.builder()
.apiKey(properties.getApiKey())
.modelName(embeddingModelProperties.getModelName())
.logRequestsAndResponses(embeddingModelProperties.isLogRequestsAndResponses())
.maxRetries(embeddingModelProperties.getMaxRetries())
.outputDimensionality(embeddingModelProperties.getOutputDimensionality())
.taskType(embeddingModelProperties.getTaskType())
.timeout(embeddingModelProperties.getTimeout())
.titleMetadataKey(embeddingModelProperties.getTitleMetadataKey())
.build();
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,112 @@
package dev.langchain4j.googleaigemini.spring;

import dev.langchain4j.model.chat.request.ResponseFormat;
import org.springframework.boot.context.properties.NestedConfigurationProperty;

import java.time.Duration;

public class ChatModelProperties {
Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved

private String modelName;
private Double temperature;
private Double topP;
private Integer topK;
private Integer maxOutputTokens;
private ResponseFormat responseFormat;
private Boolean logRequestsAndResponses;
private Integer maxRetries;
private Duration timeout;

private GeminiSafetySetting safetySetting;

private GeminiFunctionCallingConfig functionCallingConfig;

public Integer getMaxRetries() {
return maxRetries;
}

public void setMaxRetries(Integer maxRetries) {
this.maxRetries = maxRetries;
}

public Duration getTimeout() {
return timeout;
}

public void setTimeout(Duration timeout) {
this.timeout = timeout;
}

public ResponseFormat getResponseFormat() {
return responseFormat;
}

public void setResponseFormat(ResponseFormat responseFormat) {
this.responseFormat = responseFormat;
}

public String getModelName() {
return modelName;
}

public void setModelName(String modelName) {
this.modelName = modelName;
}

public Double getTemperature() {
return temperature;
}

public void setTemperature(Double temperature) {
this.temperature = temperature;
}

public Double getTopP() {
return topP;
}

public void setTopP(Double topP) {
this.topP = topP;
}

public Integer getTopK() {
return topK;
}

public void setTopK(Integer topK) {
this.topK = topK;
}

public Boolean getLogRequestsAndResponses() {
return logRequestsAndResponses;
}

public void setLogRequestsAndResponses(Boolean logRequestsAndResponses) {
this.logRequestsAndResponses = logRequestsAndResponses;
}

public GeminiSafetySetting getSafetySetting() {
return safetySetting;
}

public void setSafetySetting(GeminiSafetySetting safetySetting) {
this.safetySetting = safetySetting;
}

public GeminiFunctionCallingConfig getFunctionCallingConfig() {
return functionCallingConfig;
}

public void setFunctionCallingConfig(GeminiFunctionCallingConfig functionCallingConfig) {
this.functionCallingConfig = functionCallingConfig;
}

public Integer getMaxOutputTokens() {
return maxOutputTokens;
}

public void setMaxOutputTokens(Integer maxOutputTokens) {
this.maxOutputTokens = maxOutputTokens;
}

}
Original file line number Diff line number Diff line change
@@ -0,0 +1,73 @@
package dev.langchain4j.googleaigemini.spring;

import dev.langchain4j.model.googleai.GoogleAiEmbeddingModel;
import dev.langchain4j.model.googleai.GoogleAiEmbeddingModel.TaskType;

import java.time.Duration;

public class EmbeddingModelProperties {
Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved

private String titleMetadataKey;
private String modelName;
private Boolean logRequestsAndResponses;
private Integer maxRetries;
private Integer outputDimensionality;
private TaskType taskType;
private Duration timeout;

public String getTitleMetadataKey() {
return titleMetadataKey;
}

public void setTitleMetadataKey(String titleMetadataKey) {
this.titleMetadataKey = titleMetadataKey;
}

public String getModelName() {
return modelName;
}

public void setModelName(String modelName) {
this.modelName = modelName;
}

public Boolean isLogRequestsAndResponses() {
return logRequestsAndResponses;
}

public void setLogRequestsAndResponses(boolean logRequestsAndResponses) {
this.logRequestsAndResponses = logRequestsAndResponses;
}

public Integer getMaxRetries() {
return maxRetries;
}

public void setMaxRetries(Integer maxRetries) {
this.maxRetries = maxRetries;
}

public Integer getOutputDimensionality() {
return outputDimensionality;
}

public void setOutputDimensionality(Integer outputDimensionality) {
this.outputDimensionality = outputDimensionality;
}

public GoogleAiEmbeddingModel.TaskType getTaskType() {
return taskType;
}

public void setTaskType(GoogleAiEmbeddingModel.TaskType taskType) {
this.taskType = taskType;
}

public Duration getTimeout() {
return timeout;
}

public void setTimeout(Duration timeout) {
this.timeout = timeout;
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,27 @@
package dev.langchain4j.googleaigemini.spring;

import dev.langchain4j.model.googleai.GeminiMode;

import java.util.List;

public class GeminiFunctionCallingConfig {
Suhas-Koheda marked this conversation as resolved.
Show resolved Hide resolved

private GeminiMode geminiMode;
private List<String> allowedFunctionNames;

public GeminiMode getGeminiMode() {
return geminiMode;
}

public void setGeminiMode(GeminiMode geminiMode) {
this.geminiMode = geminiMode;
}

public List<String> getAllowedFunctionNames() {
return allowedFunctionNames;
}

public void setAllowedFunctionNames(List<String> allowedFunctionNames) {
this.allowedFunctionNames = allowedFunctionNames;
}
}
Loading