package org.jboss.resteasy.spi.metadata; import org.jboss.resteasy.annotations.Body; import org.jboss.resteasy.annotations.Form; import org.jboss.resteasy.annotations.Query; import org.jboss.resteasy.annotations.Suspend; import org.jboss.resteasy.resteasy_jaxrs.i18n.LogMessages; import org.jboss.resteasy.resteasy_jaxrs.i18n.Messages; import org.jboss.resteasy.specimpl.ResteasyUriBuilder; import org.jboss.resteasy.spi.ResteasyDeployment; import org.jboss.resteasy.spi.ResteasyProviderFactory; import org.jboss.resteasy.util.IsHttpMethod; import org.jboss.resteasy.util.MediaTypeHelper; import org.jboss.resteasy.util.MethodHashing; import org.jboss.resteasy.util.PickConstructor; import org.jboss.resteasy.util.Types; import javax.ws.rs.BeanParam; import javax.ws.rs.Consumes; import javax.ws.rs.CookieParam; import javax.ws.rs.DefaultValue; import javax.ws.rs.Encoded; import javax.ws.rs.FormParam; import javax.ws.rs.HeaderParam; import javax.ws.rs.HttpMethod; import javax.ws.rs.MatrixParam; import javax.ws.rs.Path; import javax.ws.rs.PathParam; import javax.ws.rs.Produces; import javax.ws.rs.QueryParam; import javax.ws.rs.container.Suspended; import javax.ws.rs.core.Context; import javax.ws.rs.core.GenericType; import javax.ws.rs.core.MediaType; import java.lang.annotation.Annotation; import java.lang.reflect.AccessibleObject; import java.lang.reflect.Constructor; import java.lang.reflect.Field; import java.lang.reflect.Method; import java.lang.reflect.Modifier; import java.lang.reflect.Type; import java.util.ArrayList; import java.util.HashSet; import java.util.List; import java.util.Set; import static org.jboss.resteasy.util.FindAnnotation.findAnnotation; /** * @author <a href="mailto:bill@burkecentral.com">Bill Burke</a> * @version $Revision: 1 $ */ @SuppressWarnings(value = "unchecked") public class ResourceBuilder { public static class ResourceClassBuilder { final ResourceClass resourceClass; List<FieldParameter> fields = new ArrayList<FieldParameter>(); List<SetterParameter> setters = new ArrayList<SetterParameter>(); List<ResourceMethod> resourceMethods = new ArrayList<ResourceMethod>(); List<ResourceLocator> resourceLocators = new ArrayList<ResourceLocator>(); public ResourceClassBuilder(Class<?> root, String path) { this.resourceClass = new ResourceClass(root, path); } public ResourceMethodBuilder method(Method method) { return new ResourceMethodBuilder(this, method, method); } public ResourceMethodBuilder method(Method method, Method annotatedMethod) { return new ResourceMethodBuilder(this, method, annotatedMethod); } public ResourceLocatorBuilder locator(Method method) { return new ResourceLocatorBuilder(this, method, method); } public ResourceLocatorBuilder locator(Method method, Method annotatedMethod) { return new ResourceLocatorBuilder(this, method, annotatedMethod); } public FieldParameterBuilder field(Field field) { FieldParameter param = new FieldParameter(resourceClass, field); return new FieldParameterBuilder(this, param); } public SetterParameterBuilder setter(Method method) { SetterParameter param = new SetterParameter(resourceClass, method, method); return new SetterParameterBuilder(this, param); } public ResourceConstructorBuilder constructor(Constructor constructor) { return new ResourceConstructorBuilder(this, constructor); } public ResourceClass buildClass() { resourceClass.fields = fields.toArray(new FieldParameter[fields.size()]); resourceClass.setters = setters.toArray(new SetterParameter[setters.size()]); resourceClass.resourceMethods = resourceMethods.toArray(new ResourceMethod[resourceMethods.size()]); resourceClass.resourceLocators = resourceLocators.toArray(new ResourceLocator[resourceLocators.size()]); return resourceClass; } } public static class ParameterBuilder<T extends ParameterBuilder<T>> { final Parameter parameter; public ParameterBuilder(Parameter parameter) { this.parameter = parameter; } public T type(Class<?> type) { parameter.type = type; return (T)this; } public T genericType(Type type) { parameter.genericType = type; return (T)this; } public T type(GenericType type) { parameter.type = type.getRawType(); parameter.genericType = type.getType(); return (T)this; } public T beanParam() { parameter.paramType = Parameter.ParamType.BEAN_PARAM; return (T)this; } public T context() { parameter.paramType = Parameter.ParamType.CONTEXT; return (T)this; } public T messageBody() { parameter.paramType = Parameter.ParamType.MESSAGE_BODY; return (T)this; } public T encoded() { parameter.encoded = true; return (T)this; } public T defaultValue(String defaultValue) { parameter.defaultValue = defaultValue; return (T)this; } public T cookieParam(String name) { parameter.paramType = Parameter.ParamType.COOKIE_PARAM; parameter.paramName = name; return (T)this; } public T formParam(String name) { parameter.paramType = Parameter.ParamType.FORM_PARAM; parameter.paramName = name; return (T)this; } /** * Resteasy @Form specific injection parameter * * @param prefix * @return */ public T form(String prefix) { parameter.paramType = Parameter.ParamType.FORM; parameter.paramName = prefix; return (T)this; } /** * Resteasy @Form specific injection parameter * * @return */ public T form() { parameter.paramType = Parameter.ParamType.FORM; parameter.paramName = ""; return (T)this; } public T headerParam(String name) { parameter.paramType = Parameter.ParamType.HEADER_PARAM; parameter.paramName = name; return (T)this; } public T matrixParam(String name) { parameter.paramType = Parameter.ParamType.MATRIX_PARAM; parameter.paramName = name; return (T)this; } public T pathParam(String name) { parameter.paramType = Parameter.ParamType.PATH_PARAM; parameter.paramName = name; return (T)this; } public T queryParam(String name) { parameter.paramType = Parameter.ParamType.QUERY_PARAM; parameter.paramName = name; return (T)this; } public T fromAnnotations() { Annotation[] annotations = parameter.getAnnotations(); AccessibleObject injectTarget = parameter.getAccessibleObject(); Class<?> type = parameter.getResourceClass().getClazz(); parameter.encoded = findAnnotation(annotations, Encoded.class) != null || injectTarget.isAnnotationPresent(Encoded.class) || type.isAnnotationPresent(Encoded.class); DefaultValue defaultValue = findAnnotation(annotations, DefaultValue.class); if (defaultValue != null) parameter.defaultValue = defaultValue.value(); QueryParam queryParam; Query query; HeaderParam header; MatrixParam matrix; PathParam uriParam; CookieParam cookie; FormParam formParam; Form form; Suspend suspend; Suspended suspended; if ((queryParam = findAnnotation(annotations, QueryParam.class)) != null) { parameter.paramType = Parameter.ParamType.QUERY_PARAM; parameter.paramName = queryParam.value(); } else if(( query = findAnnotation(annotations, Query.class))!= null) { parameter.paramType = Parameter.ParamType.QUERY; parameter.paramName = ""; // TODO query.prefix(); } else if ((header = findAnnotation(annotations, HeaderParam.class)) != null) { parameter.paramType = Parameter.ParamType.HEADER_PARAM; parameter.paramName = header.value(); } else if ((formParam = findAnnotation(annotations, FormParam.class)) != null) { parameter.paramType = Parameter.ParamType.FORM_PARAM; parameter.paramName = formParam.value(); } else if ((cookie = findAnnotation(annotations, CookieParam.class)) != null) { parameter.paramType = Parameter.ParamType.COOKIE_PARAM; parameter.paramName = cookie.value(); } else if ((uriParam = findAnnotation(annotations, PathParam.class)) != null) { parameter.paramType = Parameter.ParamType.PATH_PARAM; parameter.paramName = uriParam.value(); } else if ((form = findAnnotation(annotations, Form.class)) != null) { parameter.paramType = Parameter.ParamType.FORM; parameter.paramName = form.prefix(); } else if (findAnnotation(annotations, BeanParam.class) != null) { parameter.paramType = Parameter.ParamType.BEAN_PARAM; } else if ((matrix = findAnnotation(annotations, MatrixParam.class)) != null) { parameter.paramType = Parameter.ParamType.MATRIX_PARAM; parameter.paramName = matrix.value(); } else if ((suspend = findAnnotation(annotations, Suspend.class)) != null) { parameter.paramType = Parameter.ParamType.SUSPEND; parameter.suspendTimeout = suspend.value(); } else if (findAnnotation(annotations, Context.class) != null) { parameter.paramType = Parameter.ParamType.CONTEXT; } else if ((suspended = findAnnotation(annotations, Suspended.class)) != null) { parameter.paramType = Parameter.ParamType.SUSPENDED; } else if (javax.ws.rs.container.AsyncResponse.class.isAssignableFrom(type)) { parameter.paramType = Parameter.ParamType.SUSPENDED; } else if (findAnnotation(annotations, Body.class) != null) { parameter.paramType = Parameter.ParamType.MESSAGE_BODY; } else { parameter.paramType = Parameter.ParamType.UNKNOWN; } return (T)this; } } public static class ConstructorParameterBuilder extends ParameterBuilder<ConstructorParameterBuilder> { final ResourceConstructorBuilder constructor; final ConstructorParameter param; public ConstructorParameterBuilder(ResourceConstructorBuilder builder, ConstructorParameter param) { super(param); this.constructor = builder; this.param = param; } public ConstructorParameterBuilder param(int i) { return constructor.param(i); } public ResourceClassBuilder buildConstructor() { return constructor.buildConstructor(); } } public static class LocatorMethodParameterBuilder<T extends LocatorMethodParameterBuilder<T>> extends ParameterBuilder<T> { final ResourceLocatorBuilder locator; final MethodParameter param; public LocatorMethodParameterBuilder(ResourceLocatorBuilder method, MethodParameter param) { super(param); this.locator = method; this.param = param; } public T param(int i) { return (T)locator.param(i); } public ResourceClassBuilder buildMethod() { return locator.buildMethod(); } } public static class ResourceMethodParameterBuilder extends LocatorMethodParameterBuilder<ResourceMethodParameterBuilder> { final ResourceMethodBuilder method; public ResourceMethodParameterBuilder(ResourceMethodBuilder method, MethodParameter param) { super(method, param); this.method = method; } public ResourceMethodParameterBuilder suspended() { method.method.asynchronous = true; parameter.paramType = Parameter.ParamType.SUSPENDED; return this; } public ResourceMethodParameterBuilder suspend(long timeout) { method.method.asynchronous = true; parameter.paramType = Parameter.ParamType.SUSPEND; parameter.suspendTimeout = timeout; return this; } @Override public ResourceMethodParameterBuilder fromAnnotations() { super.fromAnnotations(); if (param.paramType == Parameter.ParamType.SUSPEND || param.paramType == Parameter.ParamType.SUSPENDED) { method.method.asynchronous = true; } else if (param.paramType == Parameter.ParamType.UNKNOWN) { param.paramType = Parameter.ParamType.MESSAGE_BODY; } return this; } } public static class ResourceConstructorBuilder { ResourceConstructor constructor; ResourceClassBuilder resourceClassBuilder; public ResourceConstructorBuilder(ResourceClassBuilder resourceClassBuilder, Constructor constructor) { this.resourceClassBuilder = resourceClassBuilder; this.constructor = new ResourceConstructor(resourceClassBuilder.resourceClass, constructor); } public ConstructorParameterBuilder param(int i) { return new ConstructorParameterBuilder(this, constructor.getParams()[i]); } public ResourceClassBuilder buildConstructor() { resourceClassBuilder.resourceClass.constructor = constructor; return resourceClassBuilder; } } public static class ResourceLocatorBuilder<T extends ResourceLocatorBuilder<T>> { ResourceLocator locator; ResourceClassBuilder resourceClassBuilder; ResourceLocatorBuilder() { } public ResourceLocatorBuilder(ResourceClassBuilder resourceClassBuilder, Method method, Method annotatedMethod) { this.resourceClassBuilder = resourceClassBuilder; this.locator = new ResourceLocator(resourceClassBuilder.resourceClass, method, annotatedMethod); } public T returnType(Class<?> type) { locator.returnType = type; return (T)this; } public T genericReturnType(Type type) { locator.genericReturnType = type; return (T)this; } public T returnType(GenericType type) { locator.returnType = type.getRawType(); locator.genericReturnType = type.getType(); return (T)this; } public LocatorMethodParameterBuilder param(int i) { return new LocatorMethodParameterBuilder(this, locator.getParams()[i]); } public ResourceClassBuilder buildMethod() { ResteasyUriBuilder builder = new ResteasyUriBuilder(); if (locator.resourceClass.path != null) builder.path(locator.resourceClass.path); if (locator.path != null) builder.path(locator.path); String pathExpression = builder.getPath(); if (pathExpression == null) pathExpression = ""; locator.fullpath = pathExpression; if (locator.resourceClass.getClazz().isAnonymousClass()) { locator.getMethod().setAccessible(true); } resourceClassBuilder.resourceLocators.add(locator); return resourceClassBuilder; } public T path(String path) { locator.path = path; return (T)this; } } public static class ResourceMethodBuilder extends ResourceLocatorBuilder<ResourceMethodBuilder> { ResourceMethod method; ResourceMethodBuilder(ResourceClassBuilder resourceClassBuilder, Method method, Method annotatedMethod) { this.method = new ResourceMethod(resourceClassBuilder.resourceClass, method, annotatedMethod); this.locator = this.method; this.resourceClassBuilder = resourceClassBuilder; } public ResourceMethodBuilder httpMethod(String httpMethod) { method.httpMethods.add(httpMethod.toUpperCase()); return this; } public ResourceMethodBuilder get() { method.httpMethods.add(HttpMethod.GET); return this; } public ResourceMethodBuilder put() { method.httpMethods.add(HttpMethod.PUT); return this; } public ResourceMethodBuilder post() { method.httpMethods.add(HttpMethod.POST); return this; } public ResourceMethodBuilder delete() { method.httpMethods.add(HttpMethod.DELETE); return this; } public ResourceMethodBuilder options() { method.httpMethods.add(HttpMethod.OPTIONS); return this; } public ResourceMethodBuilder head() { method.httpMethods.add(HttpMethod.HEAD); return this; } public ResourceMethodBuilder produces(MediaType... produces) { method.produces = produces; return this; } public ResourceMethodBuilder produces(String... produces) { MediaType[] types = parseMediaTypes(produces); method.produces = types; for (MediaType mt : types) { if (!mt.getParameters().containsKey(MediaType.CHARSET_PARAMETER)) { if (MediaTypeHelper.isTextLike(mt)) { ResteasyDeployment deployment = ResteasyProviderFactory.getContextData(ResteasyDeployment.class); if (deployment != null && !deployment.isAddCharset()) { LogMessages.LOGGER.mediaTypeLacksCharset(mt, method.getMethod().getName()); } } } } return this; } protected MediaType[] parseMediaTypes(String[] produces) { List<MediaType> mediaTypes = new ArrayList<MediaType>(); for (String produce : produces) { String[] split = produce.split(","); for (String s : split) mediaTypes.add(MediaType.valueOf(s)); } MediaType[] types = new MediaType[mediaTypes.size()]; types = mediaTypes.toArray(types); return types; } public ResourceMethodBuilder consumes(MediaType... consumes) { method.consumes = consumes; return this; } public ResourceMethodBuilder consumes(String... consumes) { MediaType[] types = parseMediaTypes(consumes); method.consumes = types; return this; } public ResourceMethodParameterBuilder param(int i) { return new ResourceMethodParameterBuilder(this, locator.getParams()[i]); } public ResourceClassBuilder buildMethod() { ResteasyUriBuilder builder = new ResteasyUriBuilder(); if (method.resourceClass.path != null) builder.path(method.resourceClass.path); if (method.path != null) builder.path(method.path); String pathExpression = builder.getPath(); if (pathExpression == null) pathExpression = ""; method.fullpath = pathExpression; if (method.resourceClass.getClazz().isAnonymousClass()) { method.getMethod().setAccessible(true); } resourceClassBuilder.resourceMethods.add(method); return resourceClassBuilder; } } public static class FieldParameterBuilder extends ParameterBuilder<FieldParameterBuilder> { FieldParameter field; ResourceClassBuilder resourceClassBuilder; FieldParameterBuilder(ResourceClassBuilder resourceClassBuilder, FieldParameter parameter) { super(parameter); this.field = parameter; this.resourceClassBuilder = resourceClassBuilder; } public ResourceClassBuilder buildField() { field.field.setAccessible(true); resourceClassBuilder.fields.add(field); return resourceClassBuilder; } } public static class SetterParameterBuilder extends ParameterBuilder<SetterParameterBuilder> { SetterParameter setter; ResourceClassBuilder resourceClassBuilder; SetterParameterBuilder(ResourceClassBuilder resourceClassBuilder, SetterParameter parameter) { super(parameter); this.setter = parameter; this.resourceClassBuilder = resourceClassBuilder; } public ResourceClassBuilder buildSetter() { setter.setter.setAccessible(true); resourceClassBuilder.setters.add(setter); return resourceClassBuilder; } } public static ResourceClassBuilder rootResource(Class<?> root) { return new ResourceClassBuilder(root, "/"); } public static ResourceClassBuilder rootResource(Class<?> root, String path) { return new ResourceClassBuilder(root, path); } public static ResourceClassBuilder locator(Class<?> root) { return new ResourceClassBuilder(root, null); } /** * Picks a constructor from an annotated resource class based on spec rules * * @param annotatedResourceClass * @return */ public static ResourceConstructor constructor(Class<?> annotatedResourceClass) { Constructor constructor = PickConstructor.pickPerRequestConstructor(annotatedResourceClass); if (constructor == null) { throw new RuntimeException(Messages.MESSAGES.couldNotFindConstructor(annotatedResourceClass.getName())); } ResourceConstructorBuilder builder = rootResource(annotatedResourceClass).constructor(constructor); if (constructor.getParameterTypes() != null) { for (int i = 0; i < constructor.getParameterTypes().length; i++) builder.param(i).fromAnnotations(); } return builder.buildConstructor().buildClass().getConstructor(); } /** * Build metadata from annotations on classes and methods * * @return */ public static ResourceClass rootResourceFromAnnotations(Class<?> clazz) { return fromAnnotations(false, clazz); } public static ResourceClass locatorFromAnnotations(Class<?> clazz) { return fromAnnotations(true, clazz); } private static final String WELD_PROXY_INTERFACE_NAME = "org.jboss.weld.bean.proxy.ProxyObject"; /** * Whether the given class is a proxy created by Weld or not. This is * the case if the given class implements the interface * {@code org.jboss.weld.bean.proxy.ProxyObject}. * * @param clazz the class of interest * * @return {@code true} if the given class is a Weld proxy, * {@code false} otherwise */ private static boolean isWeldProxy(Class<?> clazz) { for ( Class<?> implementedInterface : clazz.getInterfaces() ) { if ( implementedInterface.getName().equals( WELD_PROXY_INTERFACE_NAME ) ) { return true; } } return false; } private static ResourceClass fromAnnotations(boolean isLocator, Class<?> clazz) { // stupid hack for Weld as it loses generic type information, but retains annotations. if (!clazz.isInterface() && clazz.getSuperclass() != null && !clazz.getSuperclass().equals(Object.class) && isWeldProxy(clazz)) { clazz = clazz.getSuperclass(); } ResourceClassBuilder builder = null; if (isLocator) builder = locator(clazz); else { Path path = clazz.getAnnotation(Path.class); if (path == null) builder = rootResource(clazz, null); else builder = rootResource(clazz, path.value()); } for (Method method : clazz.getMethods()) { if(!method.isSynthetic() && !method.getDeclaringClass().equals(Object.class)) processMethod(isLocator, builder, clazz, method); } if (!clazz.isInterface()) { processFields(builder, clazz); } processSetters(builder, clazz); return builder.buildClass(); } private static Method findAnnotatedInterfaceMethod(Class<?> root, Class<?> iface, Method implementation) { for (Method method : iface.getMethods()) { if (method.isSynthetic()) continue; if (!method.getName().equals(implementation.getName())) continue; if (method.getParameterTypes().length != implementation.getParameterTypes().length) continue; Method actual = Types.getImplementingMethod(root, method); if (!actual.equals(implementation)) continue; if (method.isAnnotationPresent(Path.class) || IsHttpMethod.getHttpMethods(method) != null) return method; } for (Class<?> extended : iface.getInterfaces()) { Method m = findAnnotatedInterfaceMethod(root, extended, implementation); if(m != null) return m; } return null; } private static Method findAnnotatedMethod(Class<?> root, Method implementation) { // check the method itself if (implementation.isAnnotationPresent(Path.class) || IsHttpMethod.getHttpMethods(implementation) != null) return implementation; if (implementation.isAnnotationPresent(Produces.class) || implementation.isAnnotationPresent(Consumes.class)) { // completely abort this method return null; } // Per http://download.oracle.com/auth/otn-pub/jcp/jaxrs-1.0-fr-oth-JSpec/jaxrs-1.0-final-spec.pdf // Section 3.2 Annotation Inheritance // Check possible superclass declarations for (Class<?> clazz = implementation.getDeclaringClass().getSuperclass(); clazz != null; clazz = clazz.getSuperclass()) { try { Method method = clazz.getDeclaredMethod(implementation.getName(), implementation.getParameterTypes()); if (method.isAnnotationPresent(Path.class) || IsHttpMethod.getHttpMethods(method) != null) return method; if (method.isAnnotationPresent(Produces.class) || method.isAnnotationPresent(Consumes.class)) { // completely abort this method return null; } } catch (NoSuchMethodException e) { // ignore } } // Not found yet, so next check ALL interfaces from the root, // but ensure no redefinition by peer interfaces (ambiguous) to preserve logic found in // original implementation for (Class<?> clazz = root; clazz != null; clazz = clazz.getSuperclass()) { Method method = null; for (Class<?> iface : clazz.getInterfaces()) { Method m = findAnnotatedInterfaceMethod(root, iface, implementation); if (m != null) { if(method != null && !m.equals(method)) throw new RuntimeException(Messages.MESSAGES.ambiguousInheritedAnnotations(implementation)); method = m; } } if (method != null) return method; } return null; } protected static void processFields(ResourceClassBuilder resourceClassBuilder, Class<?> root) { do { processDeclaredFields(resourceClassBuilder, root); root = root.getSuperclass(); // } while (root.getSuperclass() != null && !root.getSuperclass().equals(Object.class)); } while (root != null && !root.equals(Object.class)); } protected static void processSetters(ResourceClassBuilder resourceClassBuilder, Class<?> root) { HashSet<Long> hashes = new HashSet<Long>(); do { processDeclaredSetters(resourceClassBuilder, root, hashes); root = root.getSuperclass(); } while (root != null && !root.equals(Object.class)); } protected static void processDeclaredFields(ResourceClassBuilder resourceClassBuilder, Class<?> root) { for (Field field : root.getDeclaredFields()) { FieldParameterBuilder builder = resourceClassBuilder.field(field).fromAnnotations(); if (builder.field.paramType == Parameter.ParamType.MESSAGE_BODY && !field.isAnnotationPresent(Body.class)) continue; if (builder.field.paramType == Parameter.ParamType.UNKNOWN) continue; builder.buildField(); } } protected static void processDeclaredSetters(ResourceClassBuilder resourceClassBuilder, Class<?> root, Set<Long> visitedHashes) { for (Method method : root.getDeclaredMethods()) { if (!method.getName().startsWith("set")) continue; if (method.getParameterTypes().length != 1) continue; long hash = 0; try { hash = MethodHashing.methodHash(method); } catch (Exception e) { throw new RuntimeException(e); } if (!Modifier.isPrivate(method.getModifiers()) && visitedHashes.contains(hash)) continue; visitedHashes.add(hash); SetterParameterBuilder builder = resourceClassBuilder.setter(method).fromAnnotations(); if (builder.setter.paramType == Parameter.ParamType.MESSAGE_BODY && !method.isAnnotationPresent(Body.class)) continue; if (builder.setter.paramType == Parameter.ParamType.UNKNOWN) continue; builder.buildSetter(); } } protected static void processMethod(boolean isLocator, ResourceClassBuilder resourceClassBuilder, Class<?> root, Method implementation) { Method method = findAnnotatedMethod(root, implementation); if (method != null) { Set<String> httpMethods = IsHttpMethod.getHttpMethods(method); ResourceLocatorBuilder resourceLocatorBuilder; if (httpMethods == null) { resourceLocatorBuilder = resourceClassBuilder.locator(implementation, method); } else { ResourceMethodBuilder resourceMethodBuilder = resourceClassBuilder.method(implementation, method); resourceLocatorBuilder = resourceMethodBuilder; for (String httpMethod : httpMethods) { if (httpMethod.equalsIgnoreCase(HttpMethod.GET)) resourceMethodBuilder.get(); else if (httpMethod.equalsIgnoreCase(HttpMethod.PUT)) resourceMethodBuilder.put(); else if (httpMethod.equalsIgnoreCase(HttpMethod.POST)) resourceMethodBuilder.post(); else if (httpMethod.equalsIgnoreCase(HttpMethod.DELETE)) resourceMethodBuilder.delete(); else if (httpMethod.equalsIgnoreCase(HttpMethod.OPTIONS)) resourceMethodBuilder.options(); else if (httpMethod.equalsIgnoreCase(HttpMethod.HEAD)) resourceMethodBuilder.head(); else resourceMethodBuilder.httpMethod(httpMethod); } Produces produces = method.getAnnotation(Produces.class); if (produces == null) produces = resourceClassBuilder.resourceClass.getClazz().getAnnotation(Produces.class); if (produces == null) produces = method.getDeclaringClass().getAnnotation(Produces.class); if (produces != null) resourceMethodBuilder.produces(produces.value()); Consumes consumes = method.getAnnotation(Consumes.class); if (consumes == null) consumes = resourceClassBuilder.resourceClass.getClazz().getAnnotation(Consumes.class); if (consumes == null) consumes = method.getDeclaringClass().getAnnotation(Consumes.class); if (consumes != null) resourceMethodBuilder.consumes(consumes.value()); } Path methodPath = method.getAnnotation(Path.class); if (methodPath != null) resourceLocatorBuilder.path(methodPath.value()); for (int i = 0; i < resourceLocatorBuilder.locator.params.length; i++) { resourceLocatorBuilder.param(i).fromAnnotations(); } resourceLocatorBuilder.buildMethod(); } } }