Feign进行文件上传+表单调用

Feigin默认是不支持文件上传和表单提交的,需要做一些配置才能支持。

1、feign依赖

 图中红色为form支持必须的jar。

2、添加自定义Encoder类:

import java.lang.reflect.Type;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;

import org.springframework.web.multipart.MultipartFile;

import feign.RequestTemplate;
import feign.codec.EncodeException;
import feign.form.spring.SpringFormEncoder;
import lombok.val;
import sun.reflect.generics.reflectiveObjects.ParameterizedTypeImpl;

import static java.util.Collections.singletonMap;

/**
 * @author roger yang
 * @date 10/22/2019
 */
public class MultipartEncoder extends SpringFormEncoder {
    
    @SuppressWarnings("unchecked")
    @Override
    public void encode (Object object, Type bodyType, RequestTemplate template) throws EncodeException {
        if (bodyType.getClass().equals(ParameterizedTypeImpl.class) && ((ParameterizedTypeImpl) bodyType).getRawType().equals(Map.class)) {
            val data = (Map<String, Object>) object;
            Set<String> nullSet = new HashSet<>();
            for (Map.Entry<String, Object> entry : data.entrySet()) {
                if (entry.getValue() == null) {
                    nullSet.add(entry.getKey());
                }
            }
            for (String s : nullSet) {
                data.remove(s);
            }
            super.encode(data, MAP_STRING_WILDCARD, template);
            return;
        } else if (bodyType.equals(MultipartFile.class)) {
            val file = (MultipartFile) object;
            val data = singletonMap(file.getName(), object);
            super.encode(data, MAP_STRING_WILDCARD, template);
            return;
        } else if (bodyType.equals(MultipartFile[].class)) {
            val file = (MultipartFile[]) object;
            if (file != null) {
                val data = singletonMap(file.length == 0 ? "" : file[0].getName(), object);
                super.encode(data, MAP_STRING_WILDCARD, template);
                return;
            }
        }
        super.encode(object, bodyType, template);
    }
}

为什么要自定义呢?因为SpringFormEncoder这个类的源码里只对MultipartFile做了特殊处理,并未对MultipartFile[]数组进行处理,在传递多个文件时会报错。

  @Override
  public void encode (Object object, Type bodyType, RequestTemplate template) throws EncodeException {
    if (!bodyType.equals(MultipartFile.class)) {
      super.encode(object, bodyType, template);
      return;
    }

    val file = (MultipartFile) object;
    val data = singletonMap(file.getName(), object);
    super.encode(data, MAP_STRING_WILDCARD, template);
  }

另外,我们为了让文件和表单一起在http的body里一起传输,所有我们将他们都封装到map里统一处理。

3、feign接口类:

import java.util.Map;

import org.springframework.http.MediaType;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;

import com.longge.common.GlobalResponse;

/**
 * @author: yangzhilong
 * @description: Notification api
 * post with MULTIPART_FORM_DATA_VALUE
 * you client feign config:
 * -------------------------------------------------------------------
 *  import org.springframework.cloud.openfeign.FeignClient;
    import org.springframework.context.annotation.Bean;
    
    import com.longge.api.NotificationService;
    
    import feign.codec.Encoder;
    import com.longge.config.MultipartEncoder;
    
    @FeignClient(value = "notification-service", configuration = NotificationServiceFeign.FeignSimpleEncoderConfig.class)
    public interface NotificationServiceFeign extends NotificationMultipartService {
        class FeignSimpleEncoderConfig {
            
            @Bean
            public Encoder encoder(){
                return new MultipartEncoder();
            }
        }
    }
 * ---------------------------------------------------------------------
 * @date: 17:24 2019/7/3
 **/
@RequestMapping(value = "/v1/api")
public interface NotificationMultipartService {

    /**
     * common email channel ,use to send common email
     *
     * @param data
     *     map key have: 
     *          attachfile       -->          MultipartFile[]
     *          com.longge.dto.EmailDto.FieldName --- >   value
     *          
     * eg:
     *  Map<String, Object> data = new HashMap<>();
        data.put("attachfile", new MultipartFile[] {file});
        data.put("from", emailDto.getFrom());
        data.put("to", emailDto.getTo());
        data.put("bodyText", emailDto.getBodyText());
        data.put("subject", emailDto.getSubject());
     * @return GlobalResponse<Long>  email id
     */
    @RequestMapping(value = "/send_mail", method = RequestMethod.POST, produces = {MediaType.APPLICATION_JSON_UTF8_VALUE}, consumes = MediaType.MULTIPART_FORM_DATA_VALUE)
    GlobalResponse<String> sendMail(Map<String, Object> data);
}

其中为了支持 form请求,需要对此feign进行单独的配置:

import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.context.annotation.Bean;

import com.longge.api.NotificationMultipartService;

import feign.codec.Encoder;

@FeignClient(value = "notification-service", configuration = NotificationServiceFeign.FeignSimpleEncoderConfig.class)
public interface NotificationServiceFeign extends NotificationMultipartService {
    class FeignSimpleEncoderConfig {
        @Bean
        public Encoder encoder() {
            return new MyEncoder();
        }
    }
}

4、服务实现类

@RequestMapping(value = "/v1/api/send_mail", method = RequestMethod.POST, consumes = MediaType.MULTIPART_FORM_DATA_VALUE, produces = {MediaType.APPLICATION_JSON_VALUE})
public GlobalResponse<String> sendMail(@RequestParam(value = "attachfile", required = false) MultipartFile[] attachfile, EmailDto emailDto) {
    // TODO
}

5、EmailDto属性

private String from;
private String to;
private String subject;
private String bodyText;
private String bodyHtml;

感谢:https://blog.csdn.net/ytzzh0726/article/details/79731343 

原文地址:https://www.cnblogs.com/yangzhilong/p/11714620.html