From cc84f1f4971501b70ad62c3f1c54019101aaf1b9 Mon Sep 17 00:00:00 2001 From: LangChain4j Date: Thu, 21 Nov 2024 12:41:17 +0100 Subject: [PATCH 01/12] added workflow to automatically add new PRs to the project --- .github/workflows/add_new_pr_to_project.yaml | 17 +++++++++++++++++ 1 file changed, 17 insertions(+) create mode 100644 .github/workflows/add_new_pr_to_project.yaml diff --git a/.github/workflows/add_new_pr_to_project.yaml b/.github/workflows/add_new_pr_to_project.yaml new file mode 100644 index 00000000..1d2889f0 --- /dev/null +++ b/.github/workflows/add_new_pr_to_project.yaml @@ -0,0 +1,17 @@ +name: Add new PR to Project + +on: + pull_request: + types: + - opened + - reopened + +jobs: + add-to-project: + name: Add PR to Project + runs-on: ubuntu-latest + steps: + - uses: actions/add-to-project@v0.5.0 + with: + project-url: https://github.com/users/langchain4j/projects/2 + github-token: ${{ secrets.GITHUB_TOKEN }} From 2fcbf612f63ed66a3a1f2f9bfe2f610a62c2d149 Mon Sep 17 00:00:00 2001 From: LangChain4j Date: Thu, 21 Nov 2024 12:47:29 +0100 Subject: [PATCH 02/12] added workflow to automatically add new PRs to the project --- .github/workflows/add_new_pr_to_project.yaml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.github/workflows/add_new_pr_to_project.yaml b/.github/workflows/add_new_pr_to_project.yaml index 1d2889f0..dc602533 100644 --- a/.github/workflows/add_new_pr_to_project.yaml +++ b/.github/workflows/add_new_pr_to_project.yaml @@ -14,4 +14,4 @@ jobs: - uses: actions/add-to-project@v0.5.0 with: project-url: https://github.com/users/langchain4j/projects/2 - github-token: ${{ secrets.GITHUB_TOKEN }} + github-token: ${{ secrets.GH_TOKEN_ADD_NEW_PRS_TO_PROJECT }} From 6a1d418820185dc43e8faa15cfaf2abed095d91f Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Thu, 21 Nov 2024 14:22:12 +0100 Subject: [PATCH 03/12] Update dependency org.assertj:assertj-core to v3.26.3 (#67) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR contains the following updates: | Package | Change | Age | Adoption | Passing | Confidence | |---|---|---|---|---|---| | [org.assertj:assertj-core](https://assertj.github.io/doc/#assertj-core) ([source](https://redirect.github.com/assertj/assertj)) | `3.24.2` -> `3.26.3` | [![age](https://developer.mend.io/api/mc/badges/age/maven/org.assertj:assertj-core/3.26.3?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![adoption](https://developer.mend.io/api/mc/badges/adoption/maven/org.assertj:assertj-core/3.26.3?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![passing](https://developer.mend.io/api/mc/badges/compatibility/maven/org.assertj:assertj-core/3.24.2/3.26.3?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![confidence](https://developer.mend.io/api/mc/badges/confidence/maven/org.assertj:assertj-core/3.24.2/3.26.3?slim=true)](https://docs.renovatebot.com/merge-confidence/) | --- ### Configuration 📅 **Schedule**: Branch creation - At any time (no schedule defined), Automerge - At any time (no schedule defined). 🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied. ♻ **Rebasing**: Whenever PR becomes conflicted, or you tick the rebase/retry checkbox. 🔕 **Ignore**: Close this PR and you won't be reminded about this update again. --- - [ ] If you want to rebase/retry this PR, check this box --- This PR was generated by [Mend Renovate](https://mend.io/renovate/). View the [repository job log](https://developer.mend.io/github/langchain4j/langchain4j-spring). Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> --- langchain4j-reactor/pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/langchain4j-reactor/pom.xml b/langchain4j-reactor/pom.xml index ffc80663..718767aa 100644 --- a/langchain4j-reactor/pom.xml +++ b/langchain4j-reactor/pom.xml @@ -47,7 +47,7 @@ org.assertj assertj-core - 3.24.2 + 3.26.3 test From ff94cea41fc6029a8d3b5b28280acf25ee10d247 Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Thu, 21 Nov 2024 14:22:18 +0100 Subject: [PATCH 04/12] Update dependency org.apache.maven.plugins:maven-surefire-plugin to v3.5.2 (#66) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR contains the following updates: | Package | Change | Age | Adoption | Passing | Confidence | |---|---|---|---|---|---| | [org.apache.maven.plugins:maven-surefire-plugin](https://maven.apache.org/surefire/) | `3.2.3` -> `3.5.2` | [![age](https://developer.mend.io/api/mc/badges/age/maven/org.apache.maven.plugins:maven-surefire-plugin/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![adoption](https://developer.mend.io/api/mc/badges/adoption/maven/org.apache.maven.plugins:maven-surefire-plugin/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![passing](https://developer.mend.io/api/mc/badges/compatibility/maven/org.apache.maven.plugins:maven-surefire-plugin/3.2.3/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![confidence](https://developer.mend.io/api/mc/badges/confidence/maven/org.apache.maven.plugins:maven-surefire-plugin/3.2.3/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | --- ### Configuration 📅 **Schedule**: Branch creation - At any time (no schedule defined), Automerge - At any time (no schedule defined). 🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied. ♻ **Rebasing**: Whenever PR becomes conflicted, or you tick the rebase/retry checkbox. 🔕 **Ignore**: Close this PR and you won't be reminded about this update again. --- - [ ] If you want to rebase/retry this PR, check this box --- This PR was generated by [Mend Renovate](https://mend.io/renovate/). View the [repository job log](https://developer.mend.io/github/langchain4j/langchain4j-spring). Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> --- pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pom.xml b/pom.xml index b87d59d2..ae9703a3 100644 --- a/pom.xml +++ b/pom.xml @@ -121,7 +121,7 @@ org.apache.maven.plugins maven-surefire-plugin - 3.2.3 + 3.5.2 From 2902921bb37e7f19b10a3f3c82bc1a636a89e626 Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Thu, 21 Nov 2024 14:22:28 +0100 Subject: [PATCH 05/12] Update dependency org.apache.maven.plugins:maven-javadoc-plugin to v3.11.1 (#65) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR contains the following updates: | Package | Change | Age | Adoption | Passing | Confidence | |---|---|---|---|---|---| | [org.apache.maven.plugins:maven-javadoc-plugin](https://maven.apache.org/plugins/) | `3.6.3` -> `3.11.1` | [![age](https://developer.mend.io/api/mc/badges/age/maven/org.apache.maven.plugins:maven-javadoc-plugin/3.11.1?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![adoption](https://developer.mend.io/api/mc/badges/adoption/maven/org.apache.maven.plugins:maven-javadoc-plugin/3.11.1?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![passing](https://developer.mend.io/api/mc/badges/compatibility/maven/org.apache.maven.plugins:maven-javadoc-plugin/3.6.3/3.11.1?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![confidence](https://developer.mend.io/api/mc/badges/confidence/maven/org.apache.maven.plugins:maven-javadoc-plugin/3.6.3/3.11.1?slim=true)](https://docs.renovatebot.com/merge-confidence/) | --- ### Configuration 📅 **Schedule**: Branch creation - At any time (no schedule defined), Automerge - At any time (no schedule defined). 🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied. ♻ **Rebasing**: Whenever PR becomes conflicted, or you tick the rebase/retry checkbox. 🔕 **Ignore**: Close this PR and you won't be reminded about this update again. --- - [ ] If you want to rebase/retry this PR, check this box --- This PR was generated by [Mend Renovate](https://mend.io/renovate/). View the [repository job log](https://developer.mend.io/github/langchain4j/langchain4j-spring). Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> --- pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pom.xml b/pom.xml index ae9703a3..2e40bb0a 100644 --- a/pom.xml +++ b/pom.xml @@ -153,7 +153,7 @@ org.apache.maven.plugins maven-javadoc-plugin - 3.6.3 + 3.11.1 attach-javadocs From d8d1a857d41355521ad3a1af741b9b49375664f1 Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Thu, 21 Nov 2024 14:23:05 +0100 Subject: [PATCH 06/12] Update dependency org.apache.maven.plugins:maven-gpg-plugin to v3.2.7 (#63) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR contains the following updates: | Package | Change | Age | Adoption | Passing | Confidence | |---|---|---|---|---|---| | [org.apache.maven.plugins:maven-gpg-plugin](https://maven.apache.org/plugins/) | `3.1.0` -> `3.2.7` | [![age](https://developer.mend.io/api/mc/badges/age/maven/org.apache.maven.plugins:maven-gpg-plugin/3.2.7?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![adoption](https://developer.mend.io/api/mc/badges/adoption/maven/org.apache.maven.plugins:maven-gpg-plugin/3.2.7?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![passing](https://developer.mend.io/api/mc/badges/compatibility/maven/org.apache.maven.plugins:maven-gpg-plugin/3.1.0/3.2.7?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![confidence](https://developer.mend.io/api/mc/badges/confidence/maven/org.apache.maven.plugins:maven-gpg-plugin/3.1.0/3.2.7?slim=true)](https://docs.renovatebot.com/merge-confidence/) | --- ### Configuration 📅 **Schedule**: Branch creation - At any time (no schedule defined), Automerge - At any time (no schedule defined). 🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied. ♻ **Rebasing**: Whenever PR becomes conflicted, or you tick the rebase/retry checkbox. 🔕 **Ignore**: Close this PR and you won't be reminded about this update again. --- - [ ] If you want to rebase/retry this PR, check this box --- This PR was generated by [Mend Renovate](https://mend.io/renovate/). View the [repository job log](https://developer.mend.io/github/langchain4j/langchain4j-spring). Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> --- pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pom.xml b/pom.xml index 2e40bb0a..37fee9ed 100644 --- a/pom.xml +++ b/pom.xml @@ -240,7 +240,7 @@ org.apache.maven.plugins maven-gpg-plugin - 3.1.0 + 3.2.7 sign-artifacts From 17252f9315014be67371ab8f93d56c01ebc5fc0d Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Thu, 21 Nov 2024 14:23:20 +0100 Subject: [PATCH 07/12] Update dependency org.apache.maven.plugins:maven-failsafe-plugin to v3.5.2 (#62) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR contains the following updates: | Package | Change | Age | Adoption | Passing | Confidence | |---|---|---|---|---|---| | [org.apache.maven.plugins:maven-failsafe-plugin](https://maven.apache.org/surefire/) | `3.2.3` -> `3.5.2` | [![age](https://developer.mend.io/api/mc/badges/age/maven/org.apache.maven.plugins:maven-failsafe-plugin/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![adoption](https://developer.mend.io/api/mc/badges/adoption/maven/org.apache.maven.plugins:maven-failsafe-plugin/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![passing](https://developer.mend.io/api/mc/badges/compatibility/maven/org.apache.maven.plugins:maven-failsafe-plugin/3.2.3/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![confidence](https://developer.mend.io/api/mc/badges/confidence/maven/org.apache.maven.plugins:maven-failsafe-plugin/3.2.3/3.5.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | --- ### Configuration 📅 **Schedule**: Branch creation - At any time (no schedule defined), Automerge - At any time (no schedule defined). 🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied. ♻ **Rebasing**: Whenever PR becomes conflicted, or you tick the rebase/retry checkbox. 🔕 **Ignore**: Close this PR and you won't be reminded about this update again. --- - [ ] If you want to rebase/retry this PR, check this box --- This PR was generated by [Mend Renovate](https://mend.io/renovate/). View the [repository job log](https://developer.mend.io/github/langchain4j/langchain4j-spring). Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> --- pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pom.xml b/pom.xml index 37fee9ed..9a894458 100644 --- a/pom.xml +++ b/pom.xml @@ -167,7 +167,7 @@ org.apache.maven.plugins maven-failsafe-plugin - 3.2.3 + 3.5.2 From b1451290b0fbd9c00f443e7121cc1d85ec78870f Mon Sep 17 00:00:00 2001 From: "renovate[bot]" <29139614+renovate[bot]@users.noreply.github.com> Date: Thu, 21 Nov 2024 14:29:43 +0100 Subject: [PATCH 08/12] Update dependency org.apache.maven.plugins:maven-jar-plugin to v3.4.2 (#64) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit This PR contains the following updates: | Package | Change | Age | Adoption | Passing | Confidence | |---|---|---|---|---|---| | [org.apache.maven.plugins:maven-jar-plugin](https://maven.apache.org/plugins/) | `3.3.0` -> `3.4.2` | [![age](https://developer.mend.io/api/mc/badges/age/maven/org.apache.maven.plugins:maven-jar-plugin/3.4.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![adoption](https://developer.mend.io/api/mc/badges/adoption/maven/org.apache.maven.plugins:maven-jar-plugin/3.4.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![passing](https://developer.mend.io/api/mc/badges/compatibility/maven/org.apache.maven.plugins:maven-jar-plugin/3.3.0/3.4.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | [![confidence](https://developer.mend.io/api/mc/badges/confidence/maven/org.apache.maven.plugins:maven-jar-plugin/3.3.0/3.4.2?slim=true)](https://docs.renovatebot.com/merge-confidence/) | --- ### Configuration 📅 **Schedule**: Branch creation - At any time (no schedule defined), Automerge - At any time (no schedule defined). 🚦 **Automerge**: Disabled by config. Please merge this manually once you are satisfied. ♻ **Rebasing**: Whenever PR becomes conflicted, or you tick the rebase/retry checkbox. 🔕 **Ignore**: Close this PR and you won't be reminded about this update again. --- - [ ] If you want to rebase/retry this PR, check this box --- This PR was generated by [Mend Renovate](https://mend.io/renovate/). View the [repository job log](https://developer.mend.io/github/langchain4j/langchain4j-spring). Co-authored-by: renovate[bot] <29139614+renovate[bot]@users.noreply.github.com> --- langchain4j-spring-boot-tests/pom.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/langchain4j-spring-boot-tests/pom.xml b/langchain4j-spring-boot-tests/pom.xml index 8da75976..37a75db2 100644 --- a/langchain4j-spring-boot-tests/pom.xml +++ b/langchain4j-spring-boot-tests/pom.xml @@ -40,7 +40,7 @@ org.apache.maven.plugins maven-jar-plugin - 3.3.0 + 3.4.2 From 15f4c958d1917cf1c9d5b25e38420cdb4cadf5ed Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E9=9D=92=2E=EA=A7=94?= <93132738+qing-wq@users.noreply.github.com> Date: Thu, 21 Nov 2024 23:31:04 +0800 Subject: [PATCH 09/12] Fix #2133 Spring boot starter breaks configuration classes (#79) Fixes https://github.com/langchain4j/langchain4j/issues/2133 The problem is `beanfactory.getBeanswithanNotation()`. It creates an instance of the Bean in advance. https://github.com/langchain4j/langchain4j-spring/blob/405fddafbcf8c3463c7211dc82b7c5887f9531c6/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceScannerProcessor.java#L49 --- .../spring/AiServiceScannerProcessor.java | 36 ++++++++++--------- .../TestAutowireAiServiceApplication.java | 24 +++++++++++++ .../TestAutowireClassAiServiceIT.java | 28 +++++++++++++++ .../Issue2133/TestAutowireConfiguration.java | 11 ++++++ 4 files changed, 82 insertions(+), 17 deletions(-) create mode 100644 langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireAiServiceApplication.java create mode 100644 langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireClassAiServiceIT.java create mode 100644 langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireConfiguration.java diff --git a/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceScannerProcessor.java b/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceScannerProcessor.java index 02e16ae0..2649b862 100644 --- a/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceScannerProcessor.java +++ b/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceScannerProcessor.java @@ -46,25 +46,27 @@ private Set getBasePackages(ConfigurableListableBeanFactory beanFactory) } private void addComponentScanPackages(ConfigurableListableBeanFactory beanFactory, Set collectedBasePackages) { - beanFactory.getBeansWithAnnotation(ComponentScan.class).forEach((beanName, instance) -> { - Set componentScans = AnnotatedElementUtils.getMergedRepeatableAnnotations(instance.getClass(), ComponentScan.class); - for (ComponentScan componentScan : componentScans) { - Set basePackages = new LinkedHashSet<>(); - String[] basePackagesArray = componentScan.basePackages(); - for (String pkg : basePackagesArray) { - String[] tokenized = StringUtils.tokenizeToStringArray(this.environment.resolvePlaceholders(pkg), - ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS); - Collections.addAll(basePackages, tokenized); - } - for (Class clazz : componentScan.basePackageClasses()) { - basePackages.add(ClassUtils.getPackageName(clazz)); - } - if (basePackages.isEmpty()) { - basePackages.add(ClassUtils.getPackageName(instance.getClass())); + for (String beanName : beanFactory.getBeanNamesForAnnotation(ComponentScan.class)) { + Class beanClass = beanFactory.getType(beanName); + if (beanClass != null) { + Set componentScans = AnnotatedElementUtils.getMergedRepeatableAnnotations(beanClass, ComponentScan.class); + for (ComponentScan componentScan : componentScans) { + Set basePackages = new LinkedHashSet<>(); + for (String pkg : componentScan.basePackages()) { + String[] tokenized = StringUtils.tokenizeToStringArray(this.environment.resolvePlaceholders(pkg), + ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS); + Collections.addAll(basePackages, tokenized); + } + for (Class clazz : componentScan.basePackageClasses()) { + basePackages.add(ClassUtils.getPackageName(clazz)); + } + if (basePackages.isEmpty()) { + basePackages.add(ClassUtils.getPackageName(beanClass)); + } + collectedBasePackages.addAll(basePackages); } - collectedBasePackages.addAll(basePackages); } - }); + } } private void removeAiServicesWithInactiveProfiles(BeanDefinitionRegistry registry) { diff --git a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireAiServiceApplication.java b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireAiServiceApplication.java new file mode 100644 index 00000000..1c67ff9c --- /dev/null +++ b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireAiServiceApplication.java @@ -0,0 +1,24 @@ +package dev.langchain4j.service.spring.mode.automatic.Issue2133; + +import org.springframework.beans.factory.annotation.Autowired; +import org.springframework.boot.SpringApplication; +import org.springframework.boot.autoconfigure.SpringBootApplication; + +/** + * @author: qing + * @Date: 2024/11/20 + */ +@SpringBootApplication +public class TestAutowireAiServiceApplication { + + @Autowired + TestAutowireConfiguration testAutowireConfiguration; + + public static void main(String[] args) { + SpringApplication.run(TestAutowireAiServiceApplication.class, args); + } + + TestAutowireConfiguration getConfiguration() { + return testAutowireConfiguration; + } +} diff --git a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireClassAiServiceIT.java b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireClassAiServiceIT.java new file mode 100644 index 00000000..a640848b --- /dev/null +++ b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireClassAiServiceIT.java @@ -0,0 +1,28 @@ +package dev.langchain4j.service.spring.mode.automatic.Issue2133; + +import dev.langchain4j.service.spring.AiServicesAutoConfig; +import org.junit.jupiter.api.Test; +import org.springframework.boot.autoconfigure.AutoConfigurations; +import org.springframework.boot.test.context.runner.ApplicationContextRunner; + +import static org.junit.jupiter.api.Assertions.assertNotNull; + +class TestAutowireClassAiServiceIT { + + ApplicationContextRunner contextRunner = new ApplicationContextRunner() + .withConfiguration(AutoConfigurations.of(AiServicesAutoConfig.class)); + + @Test + void should_get_configuration_class() { + contextRunner + .withUserConfiguration(TestAutowireAiServiceApplication.class) + .withBean(TestAutowireConfiguration.class) + .run(context -> { + // given + TestAutowireAiServiceApplication application = context.getBean(TestAutowireAiServiceApplication.class); + + // should get the configuration class + assertNotNull(application.getConfiguration(), "TestConfiguration class should be not null"); + }); + } +} \ No newline at end of file diff --git a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireConfiguration.java b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireConfiguration.java new file mode 100644 index 00000000..15d7a88a --- /dev/null +++ b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/Issue2133/TestAutowireConfiguration.java @@ -0,0 +1,11 @@ +package dev.langchain4j.service.spring.mode.automatic.Issue2133; + +import org.springframework.context.annotation.Configuration; + +/** + * @author: qing + * @Date: 2024/11/20 + */ +@Configuration +class TestAutowireConfiguration { +} From 10c6cdc6328bf9b55f1f61f96ae8924186cba812 Mon Sep 17 00:00:00 2001 From: George Gastaldi Date: Fri, 22 Nov 2024 05:41:01 -0300 Subject: [PATCH 10/12] Auto-generate BOM (#88) This is the generated pom.xml: ```xml 4.0.0 dev.langchain4j langchain4j-spring-bom 0.37.0-SNAPSHOT LangChain4j Spring :: BOM pom Bill of Materials POM for getting full, complete set of compatible versions of LangChain4j Spring modules https://github.com/langchain4j/langchain4j-spring Apache-2.0 https://www.apache.org/licenses/LICENSE-2.0.txt repo scm:git:git://github.com/langchain4j/langchain4j-spring.git scm:git:git@github.com:langchain4j/langchain4j-spring.git https://github.com/langchain4j/langchain4j-spring HEAD ossrh Sonatype Central Snapshots https://s01.oss.sonatype.org/content/repositories/snapshots default deep-learning-dynamo deep-learning-dynamo deeplearningdynamo@gmail.com https://github.com/deep-learning-dynamo kuraleta kuraleta digital.kuraleta@gmail.com https://github.com/kuraleta dev.langchain4j langchain4j-open-ai-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-anthropic-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-ollama-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-azure-ai-search-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-azure-open-ai-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-voyage-ai-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-github-models-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-vertex-ai-gemini-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-elasticsearch-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-redis-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-qianfan-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-milvus-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-dashscope-spring-boot-starter 0.37.0-SNAPSHOT dev.langchain4j langchain4j-reactor 0.37.0-SNAPSHOT ``` --- pom.xml | 36 ++++++++++++++++++++++++++++++++++++ 1 file changed, 36 insertions(+) diff --git a/pom.xml b/pom.xml index 9a894458..b79d5c0f 100644 --- a/pom.xml +++ b/pom.xml @@ -178,6 +178,38 @@ + + io.sundr + sundr-maven-plugin + 0.200.0 + + + false + + generate-bom + + + + + langchain4j-spring-bom + LangChain4j Spring :: BOM + Bill of Materials POM for getting full, complete set of compatible versions of LangChain4j Spring modules + + true + true + + + + *:langchain4j-spring-boot-tests + + + + + + + + + @@ -193,11 +225,13 @@ deep-learning-dynamo + deep-learning-dynamo deeplearningdynamo@gmail.com https://github.com/deep-learning-dynamo kuraleta + kuraleta digital.kuraleta@gmail.com https://github.com/kuraleta @@ -212,6 +246,7 @@ ossrh-snapshots + Sonatype Central Snapshots https://s01.oss.sonatype.org/content/repositories/snapshots/ true @@ -222,6 +257,7 @@ ossrh + Sonatype Central Snapshots https://s01.oss.sonatype.org/content/repositories/snapshots From f934f16463df7fcc3053f1848a4179bf63dbeeaf Mon Sep 17 00:00:00 2001 From: Enigma Date: Fri, 22 Nov 2024 16:58:14 +0800 Subject: [PATCH 11/12] [FEATURE] support tools enhanced by AOP (#80) Issues link: https://github.com/langchain4j/langchain4j/issues/2113 --- langchain4j-spring-boot-starter/pom.xml | 7 +++ .../service/spring/AiServiceFactory.java | 36 +++++++++++++- .../service/spring/AiServicesAutoConfig.java | 6 ++- .../withTools/AiServicesAutoConfigIT.java | 48 +++++++++++++++++++ .../automatic/withTools/AopEnhancedTools.java | 28 +++++++++++ .../automatic/withTools/aop/ToolObserver.java | 18 +++++++ .../withTools/aop/ToolObserverAspect.java | 47 ++++++++++++++++++ 7 files changed, 188 insertions(+), 2 deletions(-) create mode 100644 langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AopEnhancedTools.java create mode 100644 langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserver.java create mode 100644 langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserverAspect.java diff --git a/langchain4j-spring-boot-starter/pom.xml b/langchain4j-spring-boot-starter/pom.xml index 30440842..7c721efa 100644 --- a/langchain4j-spring-boot-starter/pom.xml +++ b/langchain4j-spring-boot-starter/pom.xml @@ -53,6 +53,13 @@ test + + org.springframework.boot + spring-boot-starter-aop + ${spring.boot.version} + test + + dev.langchain4j langchain4j-core diff --git a/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceFactory.java b/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceFactory.java index e2a2bdfc..d5063192 100644 --- a/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceFactory.java +++ b/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServiceFactory.java @@ -1,5 +1,7 @@ package dev.langchain4j.service.spring; +import dev.langchain4j.agent.tool.Tool; +import dev.langchain4j.agent.tool.ToolSpecification; import dev.langchain4j.memory.ChatMemory; import dev.langchain4j.memory.chat.ChatMemoryProvider; import dev.langchain4j.model.chat.ChatLanguageModel; @@ -8,11 +10,20 @@ import dev.langchain4j.rag.RetrievalAugmentor; import dev.langchain4j.rag.content.retriever.ContentRetriever; import dev.langchain4j.service.AiServices; +import dev.langchain4j.service.tool.DefaultToolExecutor; +import dev.langchain4j.service.tool.ToolExecutor; import org.springframework.beans.factory.FactoryBean; +import java.lang.reflect.Method; +import java.util.Arrays; +import java.util.HashMap; import java.util.List; +import java.util.Map; +import static dev.langchain4j.agent.tool.ToolSpecifications.toolSpecificationFrom; import static dev.langchain4j.internal.Utils.isNullOrEmpty; +import static org.springframework.aop.framework.AopProxyUtils.ultimateTargetClass; +import static org.springframework.aop.support.AopUtils.isAopProxy; class AiServiceFactory implements FactoryBean { @@ -94,7 +105,13 @@ public Object getObject() { } if (!isNullOrEmpty(tools)) { - builder = builder.tools(tools); + for (Object tool : tools) { + if (isAopProxy(tool)) { + builder = builder.tools(aopEnhancedTools(tool)); + } else { + builder = builder.tools(tool); + } + } } return builder.build(); @@ -120,4 +137,21 @@ public boolean isSingleton() { * (such as java.io.Closeable.close()) will not be called automatically. * Instead, a FactoryBean should implement DisposableBean and delegate any such close call to the underlying object. */ + + private Map aopEnhancedTools(Object enhancedTool) { + Map toolExecutors = new HashMap<>(); + Class originalToolClass = ultimateTargetClass(enhancedTool); + for (Method originalToolMethod : originalToolClass.getDeclaredMethods()) { + if (originalToolMethod.isAnnotationPresent(Tool.class)) { + Arrays.stream(enhancedTool.getClass().getDeclaredMethods()) + .filter(m -> m.getName().equals(originalToolMethod.getName())) + .findFirst() + .ifPresent(enhancedMethod -> { + ToolSpecification toolSpecification = toolSpecificationFrom(originalToolMethod); + toolExecutors.put(toolSpecification, new DefaultToolExecutor(enhancedTool, enhancedMethod)); + }); + } + } + return toolExecutors; + } } diff --git a/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServicesAutoConfig.java b/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServicesAutoConfig.java index 90567db3..69f6fddf 100644 --- a/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServicesAutoConfig.java +++ b/langchain4j-spring-boot-starter/src/main/java/dev/langchain4j/service/spring/AiServicesAutoConfig.java @@ -49,7 +49,11 @@ BeanFactoryPostProcessor aiServicesRegisteringBeanFactoryPostProcessor() { Set tools = new HashSet<>(); for (String beanName : beanFactory.getBeanDefinitionNames()) { try { - Class beanClass = Class.forName(beanFactory.getBeanDefinition(beanName).getBeanClassName()); + String beanClassName = beanFactory.getBeanDefinition(beanName).getBeanClassName(); + if (beanClassName == null) { + continue; + } + Class beanClass = Class.forName(beanClassName); for (Method beanMethod : beanClass.getDeclaredMethods()) { if (beanMethod.isAnnotationPresent(Tool.class)) { tools.add(beanName); diff --git a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AiServicesAutoConfigIT.java b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AiServicesAutoConfigIT.java index 3cd8e1db..49bf50b5 100644 --- a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AiServicesAutoConfigIT.java +++ b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AiServicesAutoConfigIT.java @@ -1,14 +1,22 @@ package dev.langchain4j.service.spring.mode.automatic.withTools; import dev.langchain4j.service.spring.AiServicesAutoConfig; +import dev.langchain4j.service.spring.mode.automatic.withTools.aop.ToolObserverAspect; import org.junit.jupiter.api.Test; import org.springframework.boot.autoconfigure.AutoConfigurations; import org.springframework.boot.test.context.runner.ApplicationContextRunner; import static dev.langchain4j.service.spring.mode.ApiKeys.OPENAI_API_KEY; +import static dev.langchain4j.service.spring.mode.automatic.withTools.AopEnhancedTools.TOOL_OBSERVER_KEY; +import static dev.langchain4j.service.spring.mode.automatic.withTools.AopEnhancedTools.TOOL_OBSERVER_KEY_NAME_DESCRIPTION; +import static dev.langchain4j.service.spring.mode.automatic.withTools.AopEnhancedTools.TOOL_OBSERVER_PACKAGE_NAME; +import static dev.langchain4j.service.spring.mode.automatic.withTools.AopEnhancedTools.TOOL_OBSERVER_PACKAGE_NAME_DESCRIPTION; import static dev.langchain4j.service.spring.mode.automatic.withTools.PackagePrivateTools.CURRENT_TIME; import static dev.langchain4j.service.spring.mode.automatic.withTools.PublicTools.CURRENT_DATE; import static org.assertj.core.api.Assertions.assertThat; +import static org.junit.jupiter.api.Assertions.assertEquals; +import static org.junit.jupiter.api.Assertions.assertFalse; +import static org.junit.jupiter.api.Assertions.assertTrue; class AiServicesAutoConfigIT { @@ -61,6 +69,46 @@ void should_create_AI_service_with_tool_that_is_package_private_method_in_packag }); } + @Test + void should_create_AI_service_with_tool_which_is_enhanced_by_spring_aop() { + contextRunner + .withPropertyValues( + "langchain4j.open-ai.chat-model.api-key=" + OPENAI_API_KEY, + "langchain4j.open-ai.chat-model.temperature=0.0", + "langchain4j.open-ai.chat-model.log-requests=true", + "langchain4j.open-ai.chat-model.log-responses=true" + ) + .withUserConfiguration(AiServiceWithToolsApplication.class) + .run(context -> { + + // given + AiServiceWithTools aiService = context.getBean(AiServiceWithTools.class); + + // when + String answer = aiService.chat("Which package is the @ToolObserver annotation located in? " + + "And what is the key of the @ToolObserver annotation?" + + "And What is the current time?"); + + System.out.println("Answer: " + answer); + + // then should use AopEnhancedTools.getAspectPackage() + // & AopEnhancedTools.getToolObserverKey() + // & PackagePrivateTools.getCurrentTime() + assertThat(answer).contains(TOOL_OBSERVER_PACKAGE_NAME); + assertThat(answer).contains(TOOL_OBSERVER_KEY); + assertThat(answer).contains(String.valueOf(CURRENT_TIME.getMinute())); + + // and AOP aspect should be called + // & only for getToolObserverKey() which is annotated with @ToolObserver + ToolObserverAspect aspect = context.getBean(ToolObserverAspect.class); + assertTrue(aspect.aspectHasBeenCalled()); + + assertEquals(1, aspect.getObservedTools().size()); + assertTrue(aspect.getObservedTools().contains(TOOL_OBSERVER_KEY_NAME_DESCRIPTION)); + assertFalse(aspect.getObservedTools().contains(TOOL_OBSERVER_PACKAGE_NAME_DESCRIPTION)); + }); + } + // TODO tools which are not @Beans? // TODO negative cases // TODO no @AiServices in app, just models diff --git a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AopEnhancedTools.java b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AopEnhancedTools.java new file mode 100644 index 00000000..4db600d7 --- /dev/null +++ b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/AopEnhancedTools.java @@ -0,0 +1,28 @@ +package dev.langchain4j.service.spring.mode.automatic.withTools; + +import dev.langchain4j.agent.tool.Tool; +import dev.langchain4j.service.spring.mode.automatic.withTools.aop.ToolObserver; +import org.springframework.stereotype.Component; + +@Component +public class AopEnhancedTools { + + public static final String TOOL_OBSERVER_PACKAGE_NAME_DESCRIPTION = + "Find the package directory where @ToolObserver is located."; + public static final String TOOL_OBSERVER_PACKAGE_NAME = ToolObserver.class.getPackageName(); + + public static final String TOOL_OBSERVER_KEY_NAME_DESCRIPTION = + "Find the key name of @ToolObserver"; + public static final String TOOL_OBSERVER_KEY = "AOP_ENHANCED_TOOLS_SUPPORT_@_1122"; + + @Tool(TOOL_OBSERVER_PACKAGE_NAME_DESCRIPTION) + public String getToolObserverPackageName() { + return TOOL_OBSERVER_PACKAGE_NAME; + } + + @ToolObserver(key = TOOL_OBSERVER_KEY) + @Tool(TOOL_OBSERVER_KEY_NAME_DESCRIPTION) + public String getToolObserverKey() { + return TOOL_OBSERVER_KEY; + } +} diff --git a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserver.java b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserver.java new file mode 100644 index 00000000..0c95365f --- /dev/null +++ b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserver.java @@ -0,0 +1,18 @@ +package dev.langchain4j.service.spring.mode.automatic.withTools.aop; + +import java.lang.annotation.ElementType; +import java.lang.annotation.Retention; +import java.lang.annotation.RetentionPolicy; +import java.lang.annotation.Target; + +@Target({ElementType.METHOD}) +@Retention(RetentionPolicy.RUNTIME) +public @interface ToolObserver { + + /** + * key just for example + * + * @return the key + */ + String key(); +} diff --git a/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserverAspect.java b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserverAspect.java new file mode 100644 index 00000000..68eb013d --- /dev/null +++ b/langchain4j-spring-boot-starter/src/test/java/dev/langchain4j/service/spring/mode/automatic/withTools/aop/ToolObserverAspect.java @@ -0,0 +1,47 @@ +package dev.langchain4j.service.spring.mode.automatic.withTools.aop; + +import dev.langchain4j.agent.tool.Tool; +import org.aspectj.lang.ProceedingJoinPoint; +import org.aspectj.lang.annotation.Around; +import org.aspectj.lang.annotation.Aspect; +import org.aspectj.lang.reflect.MethodSignature; +import org.springframework.stereotype.Component; + +import java.util.ArrayList; +import java.util.Arrays; +import java.util.List; + +@Aspect +@Component +public class ToolObserverAspect { + + private final List observedTools = new ArrayList<>(); + + @Around("@annotation(toolObserver)") + public Object around(ProceedingJoinPoint joinPoint, ToolObserver toolObserver) throws Throwable { + var signature = (MethodSignature) joinPoint.getSignature(); + var method = signature.getMethod(); + String methodName = method.getName(); + if (method.isAnnotationPresent(Tool.class)) { + Tool toolAnnotation = method.getAnnotation(Tool.class); + observedTools.addAll(Arrays.asList(toolAnnotation.value())); + System.out.printf("Found @Tool %s for method: %s%n%n", Arrays.toString(toolAnnotation.value()), methodName); + } + Object result = joinPoint.proceed(); + System.out.printf(" | key: %s%n | Method name: %s%n | Method arguments: %s%n | Return type: %s%n | Method return value: %s%n%n", + toolObserver.key(), + methodName, + Arrays.toString(joinPoint.getArgs()), + method.getReturnType().getName(), + result); + return result; + } + + public boolean aspectHasBeenCalled() { + return !observedTools.isEmpty(); + } + + public List getObservedTools() { + return observedTools; + } +} From 457bca4cd5e9c33df1dd323ce027818866a48139 Mon Sep 17 00:00:00 2001 From: LangChain4j Date: Fri, 29 Nov 2024 18:01:38 +0100 Subject: [PATCH 12/12] Create pull_request_template.md --- .github/pull_request_template.md | 35 ++++++++++++++++++++++++++++++++ 1 file changed, 35 insertions(+) create mode 100644 .github/pull_request_template.md diff --git a/.github/pull_request_template.md b/.github/pull_request_template.md new file mode 100644 index 00000000..0a3af6ca --- /dev/null +++ b/.github/pull_request_template.md @@ -0,0 +1,35 @@ + + +## Issue + +Closes # + +## Change + + + +## General checklist + +- [ ] There are no breaking changes +- [ ] I have added unit and/or integration tests for my change +- [ ] The tests cover both positive and negative cases +- [ ] I have manually run all the unit and integration tests in the module I have added/changed, and they are all green + +- [ ] I have added/updated the [documentation](https://github.com/langchain4j/langchain4j/tree/main/docs/docs) +- [ ] I have added an example in the [examples repo](https://github.com/langchain4j/langchain4j-examples) (only for "big" features) + + +## Checklist for adding new Spring Boot starter + +- [ ] I have added my new starter in the root `pom.xml` +- [ ] I have added a `org.springframework.boot.autoconfigure.AutoConfiguration.imports` file in the `langchain4j-{integration}-spring-boot-starter/src/main/resources/META-INF/spring/` directory