Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
S
spring-cloud-netflix
Project
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
openSource
spring-cloud-netflix
Commits
a43d9d86
Commit
a43d9d86
authored
Dec 03, 2015
by
Dave Syer
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Add null check in annotation processing for Feign
Allows POST with a @RequestBody Fixes gh-689
parent
57206c63
Expand all
Show whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
128 additions
and
63 deletions
+128
-63
SpringMvcContract.java
...mework/cloud/netflix/feign/support/SpringMvcContract.java
+48
-32
SpringMvcContractTest.java
...rk/cloud/netflix/feign/support/SpringMvcContractTest.java
+80
-31
No files found.
spring-cloud-netflix-core/src/main/java/org/springframework/cloud/netflix/feign/support/SpringMvcContract.java
View file @
a43d9d86
...
@@ -33,12 +33,12 @@ import org.springframework.cloud.netflix.feign.annotation.RequestParamParameterP
...
@@ -33,12 +33,12 @@ import org.springframework.cloud.netflix.feign.annotation.RequestParamParameterP
import
org.springframework.util.Assert
;
import
org.springframework.util.Assert
;
import
org.springframework.web.bind.annotation.RequestMapping
;
import
org.springframework.web.bind.annotation.RequestMapping
;
import
feign.Contract
;
import
feign.MethodMetadata
;
import
static
feign
.
Util
.
checkState
;
import
static
feign
.
Util
.
checkState
;
import
static
feign
.
Util
.
emptyToNull
;
import
static
feign
.
Util
.
emptyToNull
;
import
feign.Contract
;
import
feign.MethodMetadata
;
/**
/**
* @author Spencer Gibb
* @author Spencer Gibb
*/
*/
...
@@ -51,16 +51,19 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -51,16 +51,19 @@ public class SpringMvcContract extends Contract.BaseContract {
private
final
Map
<
Class
<?
extends
Annotation
>,
AnnotatedParameterProcessor
>
annotatedArgumentProcessors
;
private
final
Map
<
Class
<?
extends
Annotation
>,
AnnotatedParameterProcessor
>
annotatedArgumentProcessors
;
public
SpringMvcContract
()
{
public
SpringMvcContract
()
{
this
(
Collections
.<
AnnotatedParameterProcessor
>
emptyList
());
this
(
Collections
.<
AnnotatedParameterProcessor
>
emptyList
());
}
}
public
SpringMvcContract
(
List
<
AnnotatedParameterProcessor
>
annotatedParameterProcessors
)
{
public
SpringMvcContract
(
Assert
.
notNull
(
annotatedParameterProcessors
,
"Parameter processors can not be null."
);
List
<
AnnotatedParameterProcessor
>
annotatedParameterProcessors
)
{
Assert
.
notNull
(
annotatedParameterProcessors
,
"Parameter processors can not be null."
);
List
<
AnnotatedParameterProcessor
>
processors
;
List
<
AnnotatedParameterProcessor
>
processors
;
if
(!
annotatedParameterProcessors
.
isEmpty
())
{
if
(!
annotatedParameterProcessors
.
isEmpty
())
{
processors
=
new
ArrayList
<>(
annotatedParameterProcessors
);
processors
=
new
ArrayList
<>(
annotatedParameterProcessors
);
}
else
{
}
else
{
processors
=
getDefaultAnnotatedArgumentsProcessors
();
processors
=
getDefaultAnnotatedArgumentsProcessors
();
}
}
this
.
annotatedArgumentProcessors
=
toAnnotatedArgumentProcessorMap
(
processors
);
this
.
annotatedArgumentProcessors
=
toAnnotatedArgumentProcessorMap
(
processors
);
...
@@ -75,7 +78,8 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -75,7 +78,8 @@ public class SpringMvcContract extends Contract.BaseContract {
// Prepend path from class annotation if specified
// Prepend path from class annotation if specified
if
(
classAnnotation
.
value
().
length
>
0
)
{
if
(
classAnnotation
.
value
().
length
>
0
)
{
String
pathValue
=
emptyToNull
(
classAnnotation
.
value
()[
0
]);
String
pathValue
=
emptyToNull
(
classAnnotation
.
value
()[
0
]);
checkState
(
pathValue
!=
null
,
"RequestMapping.value() was empty on type %s"
,
checkState
(
pathValue
!=
null
,
"RequestMapping.value() was empty on type %s"
,
method
.
getDeclaringClass
().
getName
());
method
.
getDeclaringClass
().
getName
());
if
(!
pathValue
.
startsWith
(
"/"
))
{
if
(!
pathValue
.
startsWith
(
"/"
))
{
pathValue
=
"/"
+
pathValue
;
pathValue
=
"/"
+
pathValue
;
...
@@ -84,16 +88,17 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -84,16 +88,17 @@ public class SpringMvcContract extends Contract.BaseContract {
}
}
// produces - use from class annotation only if method has not specified this
// produces - use from class annotation only if method has not specified this
if
(!
md
.
template
().
headers
().
containsKey
(
ACCEPT
))
{
if
(!
md
.
template
().
headers
().
containsKey
(
ACCEPT
))
{
parseProduces
(
md
,
method
,
classAnnotation
);
parseProduces
(
md
,
method
,
classAnnotation
);
}
}
// consumes -- use from class annotation only if method has not specified this
// consumes -- use from class annotation only if method has not specified this
if
(!
md
.
template
().
headers
().
containsKey
(
CONTENT_TYPE
))
{
if
(!
md
.
template
().
headers
().
containsKey
(
CONTENT_TYPE
))
{
parseConsumes
(
md
,
method
,
classAnnotation
);
parseConsumes
(
md
,
method
,
classAnnotation
);
}
}
// headers -- class annotation is inherited to methods, always write these if present
// headers -- class annotation is inherited to methods, always write these if
// present
parseHeaders
(
md
,
method
,
classAnnotation
);
parseHeaders
(
md
,
method
,
classAnnotation
);
}
}
return
md
;
return
md
;
...
@@ -113,11 +118,12 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -113,11 +118,12 @@ public class SpringMvcContract extends Contract.BaseContract {
// path
// path
checkAtMostOne
(
method
,
methodMapping
.
value
(),
"value"
);
checkAtMostOne
(
method
,
methodMapping
.
value
(),
"value"
);
if
(
methodMapping
.
value
().
length
>
0
)
{
if
(
methodMapping
.
value
().
length
>
0
)
{
String
pathValue
=
emptyToNull
(
methodMapping
.
value
()[
0
]);
String
pathValue
=
emptyToNull
(
methodMapping
.
value
()[
0
]);
if
(
pathValue
!=
null
)
{
if
(
pathValue
!=
null
)
{
// Append path from @RequestMapping if value is present on method
// Append path from @RequestMapping if value is present on method
if
(!
pathValue
.
startsWith
(
"/"
)
&&
!
data
.
template
().
toString
().
endsWith
(
"/"
))
{
if
(!
pathValue
.
startsWith
(
"/"
)
&&
!
data
.
template
().
toString
().
endsWith
(
"/"
))
{
pathValue
=
"/"
+
pathValue
;
pathValue
=
"/"
+
pathValue
;
}
}
data
.
template
().
append
(
pathValue
);
data
.
template
().
append
(
pathValue
);
...
@@ -134,7 +140,6 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -134,7 +140,6 @@ public class SpringMvcContract extends Contract.BaseContract {
parseHeaders
(
data
,
method
,
methodMapping
);
parseHeaders
(
data
,
method
,
methodMapping
);
}
}
private
void
checkAtMostOne
(
Method
method
,
Object
[]
values
,
String
fieldName
)
{
private
void
checkAtMostOne
(
Method
method
,
Object
[]
values
,
String
fieldName
)
{
checkState
(
values
!=
null
&&
(
values
.
length
==
0
||
values
.
length
==
1
),
checkState
(
values
!=
null
&&
(
values
.
length
==
0
||
values
.
length
==
1
),
"Method %s can only contain at most 1 %s field. Found: %s"
,
"Method %s can only contain at most 1 %s field. Found: %s"
,
...
@@ -149,20 +154,25 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -149,20 +154,25 @@ public class SpringMvcContract extends Contract.BaseContract {
}
}
@Override
@Override
protected
boolean
processAnnotationsOnParameter
(
MethodMetadata
data
,
Annotation
[]
annotations
,
int
paramIndex
)
{
protected
boolean
processAnnotationsOnParameter
(
MethodMetadata
data
,
Annotation
[]
annotations
,
int
paramIndex
)
{
boolean
isHttpAnnotation
=
false
;
boolean
isHttpAnnotation
=
false
;
AnnotatedParameterProcessor
.
AnnotatedParameterContext
context
=
AnnotatedParameterProcessor
.
AnnotatedParameterContext
context
=
new
SimpleAnnotatedParameterContext
(
new
SimpleAnnotatedParameterContext
(
data
,
paramIndex
);
data
,
paramIndex
);
for
(
Annotation
parameterAnnotation
:
annotations
)
{
for
(
Annotation
parameterAnnotation
:
annotations
)
{
AnnotatedParameterProcessor
processor
=
AnnotatedParameterProcessor
processor
=
this
.
annotatedArgumentProcessors
annotatedArgumentProcessors
.
get
(
parameterAnnotation
.
annotationType
());
.
get
(
parameterAnnotation
.
annotationType
());
isHttpAnnotation
|=
processor
.
processArgument
(
context
,
parameterAnnotation
);
if
(
processor
!=
null
)
{
isHttpAnnotation
|=
processor
.
processArgument
(
context
,
parameterAnnotation
);
}
}
}
return
isHttpAnnotation
;
return
isHttpAnnotation
;
}
}
private
void
parseProduces
(
MethodMetadata
md
,
Method
method
,
RequestMapping
annotation
)
{
private
void
parseProduces
(
MethodMetadata
md
,
Method
method
,
RequestMapping
annotation
)
{
checkAtMostOne
(
method
,
annotation
.
produces
(),
"produces"
);
checkAtMostOne
(
method
,
annotation
.
produces
(),
"produces"
);
String
[]
serverProduces
=
annotation
.
produces
();
String
[]
serverProduces
=
annotation
.
produces
();
String
clientAccepts
=
serverProduces
.
length
==
0
?
null
String
clientAccepts
=
serverProduces
.
length
==
0
?
null
...
@@ -172,7 +182,8 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -172,7 +182,8 @@ public class SpringMvcContract extends Contract.BaseContract {
}
}
}
}
private
void
parseConsumes
(
MethodMetadata
md
,
Method
method
,
RequestMapping
annotation
)
{
private
void
parseConsumes
(
MethodMetadata
md
,
Method
method
,
RequestMapping
annotation
)
{
checkAtMostOne
(
method
,
annotation
.
consumes
(),
"consumes"
);
checkAtMostOne
(
method
,
annotation
.
consumes
(),
"consumes"
);
String
[]
serverConsumes
=
annotation
.
consumes
();
String
[]
serverConsumes
=
annotation
.
consumes
();
String
clientProduces
=
serverConsumes
.
length
==
0
?
null
String
clientProduces
=
serverConsumes
.
length
==
0
?
null
...
@@ -182,7 +193,8 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -182,7 +193,8 @@ public class SpringMvcContract extends Contract.BaseContract {
}
}
}
}
private
void
parseHeaders
(
MethodMetadata
md
,
Method
method
,
RequestMapping
annotation
)
{
private
void
parseHeaders
(
MethodMetadata
md
,
Method
method
,
RequestMapping
annotation
)
{
// TODO: only supports one header value per key
// TODO: only supports one header value per key
if
(
annotation
.
headers
()
!=
null
&&
annotation
.
headers
().
length
>
0
)
{
if
(
annotation
.
headers
()
!=
null
&&
annotation
.
headers
().
length
>
0
)
{
for
(
String
header
:
annotation
.
headers
())
{
for
(
String
header
:
annotation
.
headers
())
{
...
@@ -193,9 +205,10 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -193,9 +205,10 @@ public class SpringMvcContract extends Contract.BaseContract {
}
}
}
}
private
Map
<
Class
<?
extends
Annotation
>,
AnnotatedParameterProcessor
>
toAnnotatedArgumentProcessorMap
(
List
<
AnnotatedParameterProcessor
>
processors
)
{
private
Map
<
Class
<?
extends
Annotation
>,
AnnotatedParameterProcessor
>
toAnnotatedArgumentProcessorMap
(
List
<
AnnotatedParameterProcessor
>
processors
)
{
Map
<
Class
<?
extends
Annotation
>,
AnnotatedParameterProcessor
>
result
=
new
HashMap
<>();
Map
<
Class
<?
extends
Annotation
>,
AnnotatedParameterProcessor
>
result
=
new
HashMap
<>();
for
(
AnnotatedParameterProcessor
processor
:
processors
)
{
for
(
AnnotatedParameterProcessor
processor
:
processors
)
{
result
.
put
(
processor
.
getAnnotationType
(),
processor
);
result
.
put
(
processor
.
getAnnotationType
(),
processor
);
}
}
return
result
;
return
result
;
...
@@ -212,34 +225,37 @@ public class SpringMvcContract extends Contract.BaseContract {
...
@@ -212,34 +225,37 @@ public class SpringMvcContract extends Contract.BaseContract {
return
annotatedArgumentResolvers
;
return
annotatedArgumentResolvers
;
}
}
private
class
SimpleAnnotatedParameterContext
implements
AnnotatedParameterProcessor
.
AnnotatedParameterContext
{
private
class
SimpleAnnotatedParameterContext
implements
AnnotatedParameterProcessor
.
AnnotatedParameterContext
{
private
final
MethodMetadata
methodMetadata
;
private
final
MethodMetadata
methodMetadata
;
private
final
int
parameterIndex
;
private
final
int
parameterIndex
;
public
SimpleAnnotatedParameterContext
(
MethodMetadata
methodMetadata
,
int
parameterIndex
)
{
public
SimpleAnnotatedParameterContext
(
MethodMetadata
methodMetadata
,
int
parameterIndex
)
{
this
.
methodMetadata
=
methodMetadata
;
this
.
methodMetadata
=
methodMetadata
;
this
.
parameterIndex
=
parameterIndex
;
this
.
parameterIndex
=
parameterIndex
;
}
}
@Override
@Override
public
MethodMetadata
getMethodMetadata
()
{
public
MethodMetadata
getMethodMetadata
()
{
return
methodMetadata
;
return
this
.
methodMetadata
;
}
}
@Override
@Override
public
int
getParameterIndex
()
{
public
int
getParameterIndex
()
{
return
parameterIndex
;
return
this
.
parameterIndex
;
}
}
@Override
@Override
public
void
setParameterName
(
String
name
)
{
public
void
setParameterName
(
String
name
)
{
nameParam
(
methodMetadata
,
name
,
parameterIndex
);
nameParam
(
this
.
methodMetadata
,
name
,
this
.
parameterIndex
);
}
}
@Override
@Override
public
Collection
<
String
>
setTemplateParameter
(
String
name
,
Collection
<
String
>
rest
)
{
public
Collection
<
String
>
setTemplateParameter
(
String
name
,
Collection
<
String
>
rest
)
{
return
addTemplatedParam
(
rest
,
name
);
return
addTemplatedParam
(
rest
,
name
);
}
}
}
}
...
...
spring-cloud-netflix-core/src/test/java/org/springframework/cloud/netflix/feign/support/SpringMvcContractTest.java
View file @
a43d9d86
This diff is collapsed.
Click to expand it.
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment