• Stars
    star
    168
  • Rank 217,622 (Top 5 %)
  • Language
    Java
  • License
    Apache License 2.0
  • Created about 10 years ago
  • Updated 5 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Tools for using Spring in Guice and Guice in Spring

This project provides bridges between Spring and Guice so that you can use one from the other (and vice versa). It works with Spring 6 (or at least the tests are green), but Guice does not support the Jakarta annotations so it may break in unexpected ways.

Build Status

Using a Spring ApplicationContext as a Module in Guice

The main bridge in this case is a Guice Module that wraps an existing Spring ApplicationContext. Example:

AnnotationConfigApplicationContext context = 
    new AnnotationConfigApplicationContext(ApplicationConfiguration.class);
Injector injector = Guice.createInjector(new SpringModule(context), new MyModule());
Service service = injector.getInstance(Service.class);

SpringModule (org.springframework.guice.module.SpringModule) will wrap the existing spring configurations for you.

Note that the ApplicationContext in this example might contain the Service definition or it might be in the Guice Module (MyModule), or if Service is a concrete class it could be neither, but Guice creates an instance and wires it for us.

If the ApplicationConfiguration is annotated @GuiceModule then it can filter the types of bean that are registered with the Guice binder. Example:

@Configuration
@GuiceModule(includeFilters=@Filter(pattern=.*\\.Service))
public class ApplicationConfiguration {
    @Bean
    public MyService service() {
        ...
    }
}

In this case, only bean types (or interfaces) called "Service" will match the include filter, and only those beans will be bound.

If there are multiple @Beans of the same type in the ApplicationContext then the SpringModule will register them all, and there will be a runtime exception if an Injector needs one. As with normal Spring dependency resolution, you can add the @Primary marker to a single bean to differentiate and hint to the Injector which instance to use.

Registering Spring Configuration Classes as a Guice Module

If your Spring @Configuration has dependencies that can only come from a Guice Module and you prefer to use the Guice APIs to build up the configuration (so you can't use @EnableGuiceModules below), then you can create a SpringModule from a Provider<ConfigurableListableBeanFactory> instead of from an existing ApplicationContext. There are some additional features that may also apply:

  • If the bean factory created by the provider is a DefaultListableBeanFactory (mostly it would be if it came from an ApplicationContext), then it will pick up a special Guice-aware AutowireCandidateResolver, meaning that it will be able to inject dependencies from Guice modules that are not registered as beans.

  • If the bean factory contains any beans of type ProvisionListener (a Guice lifecysle listener), then those will be instantiated and registered with Guice.

To take advantage of the autowiring the bean factory must come from an ApplicationContext that is not fully refreshed (refreshing would resolve all the dependencies and fail because the Guice resolver is not yet registered). To help you build bean factories that have this quality there is a convenience class called BeanFactoryProvider with static methods which you can use to create a provider to inject into a SpringModule. Example:

Injector injector = Guice.createInjector(new SimpleGuiceModule(), 
    new SpringModule(BeanFactoryProvider.from(SpringConfiguration.class)));

The SimpleGuiceModule contains a component that the SpringConfiguration depends on.

Using existing Guice Modules in a Spring ApplicationContext

The main feature here is a Spring @Configuration annotation: @EnableGuiceModules. If you have Guice Modules that you want to re-use (e.g. if they come from a third party) you can declare them in a Spring ApplicationContext as @Beans, and expose all their bindings. Example:

@EnableGuiceModules
@Configuration
public static class TestConfig {

    @Bean
    public static MyModule myModule() {
        return new MyModule();
    }

    @Bean
    public Spam spam(Service service) {
        return new Spam(service);
    }

}

The Service was defined in the Guice module MyModule, and then it was be bound to the autowired spam() method when Spring started.

Filtering out modules from startup of ApplicationContext

In certain cases you might need to ensure that some modules are not configured at all, even though they will be absent from the final ApplicationContext. This might be due to external code that may be hard to change that causes side effects at binding time, for example. To ensure that these modules are not called at all you can define one or moreModuleFilter beans that will be applied to filter out modules from the context before they are touched by the Spring-Guice bridge.

import org.springframework.context.annotation.Bean;
import org.springframework.guice.annotation.ModuleFilter;

public static class TestConfig {


  @Bean
  public static MyModule myModule() {
    return new MyModule();
  }

  @Bean
  public ModuleFilter myFilter() {
    return module -> !(module instanceof MyModule);
  }
}

This will ensure that no configure() methods are called on the filtered modules.

Configuration Class Enhancements

Note that the Module bean definition in the example above is declared in a static method. This is intentional and can be used to avoid accidentally preventing Spring from being able to enhance the parent @Configuration class. The default behaviour for Spring is to create a proxy for @Configuration classes so the @Bean methods can call each other and Spring will preserve the singleton nature of the bean factory - one bean of each type per bean id.

If you see logs like this when the context starts:

Mar 21, 2022 8:30:35 AM org.springframework.context.annotation.ConfigurationClassPostProcessor enhanceConfigurationClasses
INFO: Cannot enhance @Configuration bean definition 'TestConfig' since its singleton instance has been created too early. The typical cause is a non-static @Bean method with a BeanDefinitionRegistryPostProcessor return type: Consider declaring such methods as 'static'.

that is a sign that you might want to use static methods to define any beans of type Module. It is logged at INFO because Spring doesn't know if it was intentional or not. Most likely it was unintentional, and it is better to avoid nasty surprises later if we can.

Another way to avoid the warning is to declare the parent class as @Configuration(proxyBeanMethods = false) so that Spring knows that you don't even want to enhance the class. It's not a bad idea to use that flag wherever you can because it saves some time on start up if the proxy doesn't need to be created.

Using Guice as an API for accessing a Spring ApplicationContext

In this case the main feature is an Injector implementation that wraps a Spring ApplicationContext. Example:

AnnotationConfigApplicationContext context = 
    new AnnotationConfigApplicationContext(ApplicationConfiguration.class);
Injector injector = new SpringInjector(context);
Service service = injector.getInstance(Service.class);

If there is a @Bean of type Service it will be returned from the Injector. But there may actually not be a @Bean definition of type Service, and if it is a concrete type then the Injector will create it and autowire its dependencies for you. A side effect of this is that a BeanDefinition will be created.

In the example above, if ApplicationConfiguration was annotated @EnableGuiceModules then there is an Injector bean already waiting to be used, including wiring it into the application itself if you need it. So this works as well:

@Configuration
@EnableGuiceModules
public class ApplicationConfiguration {

    @Autowired
    private Injector injector;
    
    @Bean
    public Foo foo() {
        // Guice creates and does the wiring of Foo instead of Spring
        return injector.getInstance(Foo.class);
    }
}

In this example if the Injector has a binding for a Provider of Foo.class then the foo() method is redundant - it is already resolvable as a Spring dependency. But if Guice is being used as a factory for new objects that it doesn't have bindings for, then it makes sense.

Note: if you also have @GuiceModule in your context, then using the injector to create a @Bean directly is a bad idea (there's a dependency cycle). You can do it, and break the cycle, if you exclude the @Bean type from the Injector bindings using the @GuiceModule exclude filters.

Configurable Options

For a full list of configuration options, see the configuration metadata file.

Binding Deduplication - When using @EnableGuiceModules, if a Spring Bean and a Guice Binding both exist for the same type and Qualifier, creation of the Injector will fail. You may instead prefer to keep Spring's instance of the type instead of receiving this error. To accomplish this, you may set the property spring.guice.dedup=true.

Disable Guice just-in-time bindings - When enabled (default enabled), beans without explicit definitions will be created using Guice just-in-time bindings. Otherwise, it will fail with UnsatisfiedDependencyException. To disable, set the property spring.guice.autowireJIT=false.

Limitations

  • So far there is no support for the Guice SPI methods in SpringInjector so tooling may not work. It wouldn't be hard to do.

  • SpringInjector only knows about raw types and bean names, so it ignores additional meta-information in factory requests (like annotations other than @Named). Should be easy enough to fix, but some compromises might have to be made.

  • SpringInjector has no support for creating child or parent Injectors. Probably not difficult.

  • SpringModule treats all beans as singletons.

  • SpringModule binds all interfaces and all names of a bean it can find. This should work out OK, as long as those interfaces are not needed for injection (and if there is no @Primary bean).

More Repositories

1

spring-boot

Spring Boot
Java
71,736
star
2

spring-framework

Spring Framework
Java
54,504
star
3

spring-security

Spring Security
Java
8,263
star
4

spring-petclinic

A sample Spring-based application
CSS
7,090
star
5

spring-data-examples

Spring Data Example Projects
Java
5,026
star
6

spring-authorization-server

Spring Authorization Server
Java
4,660
star
7

spring-data-jpa

Simplifies the development of creating a JPA-based data access layer.
Java
2,853
star
8

spring-data-elasticsearch

Provide support to increase developer productivity in Java when using Elasticsearch. Uses familiar Spring concepts such as a template classes for core API usage and lightweight repository style data access.
Java
2,850
star
9

spring-loaded

Java agent that enables class reloading in a running JVM
Java
2,711
star
10

spring-batch

Spring Batch is a framework for writing batch applications using Java and Spring
Java
2,549
star
11

spring-integration-samples

You are looking for examples, code snippets, sample applications for Spring Integration? This is the place.
Java
2,267
star
12

spring-retry

Java
2,071
star
13

spring-kafka

Provides Familiar Spring Abstractions for Apache Kafka
Java
2,067
star
14

spring-session

Spring Session
Java
1,823
star
15

spring-data-redis

Provides support to increase developer productivity in Java when using Redis, a key-value store. Uses familiar Spring concepts such as a template classes for core API usage and lightweight repository style data access.
Java
1,667
star
16

spring-data-mongodb

Provides support to increase developer productivity in Java when using MongoDB. Uses familiar Spring concepts such as a template classes for core API usage and lightweight repository style data access.
Java
1,566
star
17

spring-integration

Spring Integration provides an extension of the Spring programming model to support the well-known Enterprise Integration Patterns (EIP)
Java
1,486
star
18

spring-graphql

Spring Integration for GraphQL
Java
1,473
star
19

spring-statemachine

Spring Statemachine is a framework for application developers to use state machine concepts with Spring.
Java
1,463
star
20

spring-ai

An Application Framework for AI Engineering
Java
1,288
star
21

spring-security-samples

Java
1,134
star
22

spring-restdocs

Test-driven documentation for RESTful services
Java
1,126
star
23

spring-hateoas

Spring HATEOAS - Library to support implementing representations for hyper-text driven REST web services.
Java
1,021
star
24

spring-data-rest

Simplifies building hypermedia-driven REST web services on top of Spring Data repositories
Java
896
star
25

sts4

The next generation of tooling for Spring Boot, including support for Cloud Foundry manifest files, Concourse CI pipeline definitions, BOSH deployment manifests, and more... - Available for Eclipse, Visual Studio Code, and Theia
Java
829
star
26

spring-net

Spring Framework for .NET
C#
827
star
27

spring-cloud

Umbrella project for Spring Cloud
821
star
28

spring-data-neo4j

Provide support to increase developer productivity in Java when using Neo4j. Uses familiar Spring concepts such as a template classes for core API usage and lightweight repository style data access.
Java
806
star
29

spring-amqp

Spring AMQP - support for Spring programming model with AMQP, especially but not limited to RabbitMQ
Java
793
star
30

spring-data-commons

Spring Data Commons. Interfaces and code shared between the various datastore specific implementations.
Java
747
star
31

spring-data-relational

Spring Data Relational. Home of Spring Data JDBC and Spring Data R2DBC.
Java
724
star
32

spring-data-r2dbc

Provide support to increase developer productivity in Java when using Reactive Relational Database Connectivity. Uses familiar Spring concepts such as a DatabaseClient for core API usage and lightweight repository style data access.
708
star
33

spring-shell

Spring based shell
Java
693
star
34

spring-modulith

Modular applications with Spring Boot
Java
647
star
35

spring-amqp-samples

Samples for Spring AMQP
Java
587
star
36

spring-plugin

Java
428
star
37

spring-webflow-samples

CSS
407
star
38

spring-data-cassandra

Provides support to increase developer productivity in Java when using Apache Cassandra. Uses familiar Spring concepts such as a template classes for core API usage and lightweight repository style data access.
Java
370
star
39

spring-hateoas-examples

Collection of examples on how (and why) to build hypermedia-driven apps with Spring HATEOAS
Java
365
star
40

spring-ldap

Spring LDAP
Java
339
star
41

spring-webflow

Spring Web Flow
Java
319
star
42

spring-ws

Spring Web Services
Java
304
star
43

spring-flo

JavaScript angular based embeddable graphical component for pipeline/graph building and editing
TypeScript
296
star
44

spring-vault

Provides familiar Spring abstractions for HashiCorp Vault
Java
277
star
45

spring-integration-extensions

The Spring Integration Extensions project provides extension components for Spring Integration
Java
276
star
46

spring-data-couchbase

Provides support to increase developer productivity in Java when using Couchbase. Uses familiar Spring concepts such as a template classes for core API usage and lightweight repository style data access.
Java
272
star
47

spring-data-envers

Envers extension of the Spring Data JPA module
263
star
48

spring-batch-extensions

Spring Batch Extensions
Java
228
star
49

spring-integration-aws

Java
201
star
50

spring-security-kerberos

Spring Security Kerberos
Java
173
star
51

spring-pulsar

Spring Friendly Abstractions for Apache Pulsar
Java
168
star
52

gradle-plugins

Gradle plugins for use in building and publishing open-source Spring projects, e.g. Spring Framework, Spring Integration
Groovy
139
star
53

spring-ws-samples

Java
138
star
54

spring-data-keyvalue

Project to provide infrastructure to implement Spring Data repositories on top of key-value-based, in-memory data stores.
Java
127
star
55

spring-data-gemfire

Spring Data integration for Pivotal GemFire
Java
99
star
56

spring-data-build

Modules to centralize common resources and configuration for Spring Data Maven builds.
Shell
94
star
57

spring-data

Spring Data
85
star
58

spring-cli

A CLI focused on developer productivity
Java
75
star
59

spring-data-ldap

Repository abstraction for Spring LDAP
Java
65
star
60

spring-data-geode

Spring Data support for Apache Geode
Java
49
star
61

spring-aot-smoke-tests

Smoke tests for Spring's AOT and native support
Java
47
star
62

spring-session-data-mongodb

Spring Session for MongoDB
Java
47
star
63

spring-boot-data-geode

Spring Boot support for Apache Geode and VMware GemFire
Java
45
star
64

spring-data-dev-tools

A collection of tools to support Spring Data development.
Java
39
star
65

spring-session-bom

Spring Session Bill of Materials
Java
31
star
66

spring-restdocs-samples

JavaScript
27
star
67

spring-credhub

Spring abstractions for Cloud Foundry CredHub
Java
25
star
68

spring-test-data-geode

A Spring-based Test Framework supporting Unit and Integration testing for Spring Boot applications using Spring Data with either Apache Geode or VMware Tanzu GemFire
Java
22
star
69

spring-integration-splunk

Java
21
star
70

spring-session-data-geode

Spring Session support for Apache Geode and VMware Tanzu GemFire
Java
21
star
71

spring-session-data-mongodb-examples

Java
16
star
72

spring-data-bom

Spring Data Bill of Materials
16
star
73

eclipse-integration-tcserver

Eclipse Integration for vFabric tc Server
Java
14
star
74

security-advisories

Report a Spring CVE
12
star
75

spring-lifecycle-smoke-tests

Java
11
star
76

spring-integration-flow

Java
10
star
77

gh-pages

Shared gh-pages content for Spring projects
JavaScript
8
star
78

.github

7
star
79

spring-data-release

Command-line application to ship Spring Data releases
Java
5
star
80

spring-rewrite-commons

Java
3
star
81

spring-graphql-examples

3
star