'Resolving Pageable in Webflux
I spent a lot of time to find a solution about Pageable in Webflux, unfortunately, at the time of writing this, Webflux does not support Pageable so I came up with a solution and this is what I have implemented. It is a hacky solution to resolve Pageable and Sort type in Webflux controller/resource in Spring Boot.
Here is the solution: (I know it might be ugly but you can use it till Spring team fix the problem, there are on-going efforts for that. Spring Jira
Also, the gist is here: Github
Update 1: You can find the article here in medium as well
Versions:
Spring Boot: 2.0.2.RELEASE
Gradle Kotlin: 1.2.41
Feel free to give me hint to improve it. (Code is Kotlin)
@Configuration
class PageableSerializer {
@Bean
@ConditionalOnMissingBean
fun pageableCustomizer(properties: SpringDataWebProperties): PageableHandlerMethodArgumentResolverCustomizer {
return PageableHandlerMethodArgumentResolverCustomizer { resolver ->
val pageable = properties.pageable
resolver.setPageParameterName(pageable.pageParameter)
resolver.setSizeParameterName(pageable.sizeParameter)
resolver.setOneIndexedParameters(pageable.isOneIndexedParameters)
resolver.setPrefix(pageable.prefix)
resolver.setQualifierDelimiter(pageable.qualifierDelimiter)
resolver.setFallbackPageable(PageRequest.of(0, pageable.defaultPageSize))
resolver.setMaxPageSize(pageable.maxPageSize)
}
}
@Bean
@ConditionalOnMissingBean
fun sortCustomizer(properties: SpringDataWebProperties): SortHandlerMethodArgumentResolverCustomizer {
return SortHandlerMethodArgumentResolverCustomizer { resolver ->
resolver.setSortParameter(properties.sort.sortParameter)
}
}
@Bean
fun pageableHandler(pageableResolver: Optional<PageableHandlerMethodArgumentResolverCustomizer>, sortHandler: SortHandlerMethodArgumentResolver, reactiveAdapterRegistry: ReactiveAdapterRegistry): PageableHandlerMethodArgumentResolver {
val handler = PageableHandlerMethodArgumentResolver(sortHandler)
pageableResolver.ifPresent { c -> c.customize(handler) }
return handler
}
@Bean
fun sortHandler(sortResolver: Optional<SortHandlerMethodArgumentResolverCustomizer>): SortHandlerMethodArgumentResolver {
val handler = SortHandlerMethodArgumentResolver()
sortResolver.ifPresent { c -> c.customize(handler) }
return handler
}
}
And to register handler:
@Component
class PageableResolver(registry: ReactiveAdapterRegistry, private val resolver: PageableHandlerMethodArgumentResolver) : HandlerMethodArgumentResolverSupport(registry) {
override fun resolveArgument(parameter: MethodParameter, bindingContext: BindingContext, exchange: ServerWebExchange): Mono<Any> {
return Mono.just(resolver.resolveArgument(parameter, null, MockNative(exchange.request.queryParams), null))
}
override fun supportsParameter(parameter: MethodParameter): Boolean {
return this.resolver.supportsParameter(parameter)
}
private class MockNative(private val parameters: MultiValueMap<String, String>) : NativeWebRequest {
override fun getParameter(paramName: String): String? {
return this.parameters.getFirst(paramName)
}
override fun getParameterValues(paramName: String): Array<String>? {
return this.parameters[paramName]?.toTypedArray()
}
override fun isUserInRole(role: String): Boolean {
return false
}
override fun getRemoteUser(): String? {
return null
}
override fun getLocale(): Locale {
return Locale.getDefault()
}
override fun getParameterMap(): MutableMap<String, Array<String>> {
return mutableMapOf()
}
override fun getSessionId(): String {
return ""
}
override fun getAttributeNames(scope: Int): Array<String> {
return arrayOf()
}
override fun registerDestructionCallback(name: String, callback: Runnable, scope: Int) {
}
override fun resolveReference(key: String): Any? {
return null
}
override fun getHeaderValues(headerName: String): Array<String>? {
return null
}
override fun getUserPrincipal(): Principal? {
return null
}
override fun getDescription(includeClientInfo: Boolean): String {
return ""
}
override fun getSessionMutex(): Any {
return ""
}
override fun getNativeResponse(): Any? {
return null
}
override fun <T : Any?> getNativeResponse(requiredType: Class<T>?): T? {
return null
}
override fun getParameterNames(): MutableIterator<String> {
return mutableListOf<String>().iterator()
}
override fun getNativeRequest(): Any {
return ""
}
override fun <T : Any?> getNativeRequest(requiredType: Class<T>?): T? {
return null
}
override fun removeAttribute(name: String, scope: Int) {
}
override fun getHeader(headerName: String): String? {
return null
}
override fun getContextPath(): String {
return ""
}
override fun checkNotModified(lastModifiedTimestamp: Long): Boolean {
return false
}
override fun checkNotModified(etag: String): Boolean {
return false
}
override fun checkNotModified(etag: String?, lastModifiedTimestamp: Long): Boolean {
return false
}
override fun getHeaderNames(): MutableIterator<String> {
return mutableListOf<String>().iterator()
}
override fun getAttribute(name: String, scope: Int): Any? {
return null
}
override fun setAttribute(name: String, value: Any, scope: Int) {
}
override fun isSecure(): Boolean {
return false
}
}
}
Solution 1:[1]
See ReactivePageableHandlerMethodArgumentResolver class
Example with WebFluxConfigure
:
@Configuration
@ConditionalOnClass(EnableWebFlux.class)
@ConditionalOnWebApplication(type = ConditionalOnWebApplication.Type.REACTIVE)
public class WebfluxConfig implements WebFluxConfigurer {
@Override
public void configureArgumentResolvers(ArgumentResolverConfigurer configurer) {
configurer.addCustomResolver(new ReactivePageableHandlerMethodArgumentResolver());
}
}
Solution 2:[2]
Sample with standard config
@Configuration
class WebConfig {
@Bean
HandlerMethodArgumentResolver reactivePageableHandlerMethodArgumentResolver() {
return new ReactivePageableHandlerMethodArgumentResolver();
}
}
Sources
This article follows the attribution requirements of Stack Overflow and is licensed under CC BY-SA 3.0.
Source: Stack Overflow
Solution | Source |
---|---|
Solution 1 | |
Solution 2 | slawek |