2018-03-29 SpringCloud Feign Decoder

如題:這篇文章主要講的是Spring Cloud Feign的Decoder

前提:使用FeignClient
并且返回值 使用泛型篡石,例如 PhpResponse<Model>。
SpringCloud版本:Brixton (Spring Core 4.2.3)

在HTTP協(xié)議不是很規(guī)范的情況下,需要配置Decoder

例如PHP寫的服務(wù)饰抒,就不跟你區(qū)分 ContentType 是不是JSON了坑鱼。即便是大廠騰訊咽白,相信他們的接口例如微信支付等等备埃,是不區(qū)分的节槐。

具體來說:就是返回數(shù)據(jù)是JSON,而ContentType 為 text/html;charset=UTF-8
這不影響你讀取他的文本內(nèi)容锯仪。

對于SpringCloud Feign來說:

在沒有寫Fallback的情況下:

Could not extract response: no suitable HttpMessageConverter found 

如果寫了Fallback泵督,則即便返回了正常的JSON內(nèi)容,因為Decoder無法decode這種Response類型庶喜,則進(jìn)入了Fallback降級類了小腊。你看不到相關(guān)的錯誤日志,警告都沒有久窟。

問題現(xiàn)狀

如果是自己的公司秩冈,要求PHP傳 正確的ContentType ,不難斥扛,就是要說服他們改改入问。
也就一句話

header("ContentType", "application/json;charset=UTF-8");

而如果對方很調(diào)皮,表示怎么就你JAVA的要求這么高? Customer端也沒有要求這個稀颁?

而如果對方是騰訊的微信支付接口芬失,你讓人家改?

所以匾灶,還是要自己用一些方法處理的棱烂。
如果因此放棄Feign ,改用HTTPClient阶女,這也不甘心啊颊糜。

解決方案

貢獻(xiàn)出代碼:

@Configuration
public class FeignConfig {

    @Bean
    Logger.Level feignLoggerLevel() {
        return Logger.Level.FULL;
    }


    @Bean
    public Decoder feignDecoder() {
        return new ResponseEntityDecoder(new SpringDecoder(feignHttpMessageConverter()));
    }

    public ObjectFactory<HttpMessageConverters> feignHttpMessageConverter() {
        final HttpMessageConverters httpMessageConverters = new HttpMessageConverters(new PhpMappingJackson2HttpMessageConverter());
        return new ObjectFactory<HttpMessageConverters>() {
            @Override
            public HttpMessageConverters getObject() throws BeansException {
                return httpMessageConverters;
            }
        };
    }

    public class PhpMappingJackson2HttpMessageConverter extends MappingJackson2HttpMessageConverter {
        PhpMappingJackson2HttpMessageConverter(){
            List<MediaType> mediaTypes = new ArrayList<>();
            mediaTypes.add(MediaType.valueOf(MediaType.TEXT_HTML_VALUE + ";charset=UTF-8")); //關(guān)鍵
            setSupportedMediaTypes(mediaTypes);
        }
    }

}

這里關(guān)鍵位置,就添加了對這種Response的支持张肾。

RestTemplate的配置芭析,我還不會,這里就不寫了吞瞪。

思考

好像配置過FastJSON的MessageConverter的馁启?
下面的內(nèi)容將說明 我所用的FastJsonHttpMessageConverter沒有起作用的原因。

原理

下面看看源碼:

//SpringDecoder.java

@Override
    public Object decode(final Response response, Type type) throws IOException,
            FeignException {
        if (type instanceof Class || type instanceof ParameterizedType) {
            @SuppressWarnings({ "unchecked", "rawtypes" })
            HttpMessageConverterExtractor<?> extractor = new HttpMessageConverterExtractor(
                    type, this.messageConverters.getObject().getConverters());

            return extractor.extractData(new FeignResponseAdapter(response));
        }
        throw new DecodeException(
                "type is not an instance of Class or ParameterizedType: " + type);
    }

這里構(gòu)造了一個HttpMessageConverterExtractor芍秆,跟進(jìn)構(gòu)造函數(shù):

HttpMessageConverterExtractor(Type responseType, 
        List<HttpMessageConverter<?>> messageConverters, Log logger) {
        Assert.notNull(responseType, "'responseType' must not be null");
        Assert.notEmpty(messageConverters, "'messageConverters' must not be empty");
        this.responseType = responseType;
        this.responseClass = (responseType instanceof Class) ? (Class<T>) responseType : null;
        this.messageConverters = messageConverters;
        this.logger = logger;
    }

前文說的泛型的返回值類型惯疙,這里responseTypeParameterizedTypeImpl類型
Debug發(fā)現(xiàn)
這里Instaceof Classfalse,則 this.responseClass = null

再看extractData

//HttpMessageConverterExtractor.java

public T extractData(ClientHttpResponse response) throws IOException {
        MessageBodyClientHttpResponseWrapper responseWrapper = new MessageBodyClientHttpResponseWrapper(response);
        if (!responseWrapper.hasMessageBody() || responseWrapper.hasEmptyMessageBody()) {
            return null;
        }
        MediaType contentType = getContentType(responseWrapper);

        for (HttpMessageConverter<?> messageConverter : this.messageConverters) {
            if (messageConverter instanceof GenericHttpMessageConverter) {
                GenericHttpMessageConverter<?> genericMessageConverter = (GenericHttpMessageConverter<?>) messageConverter;
                if (genericMessageConverter.canRead(this.responseType, null, contentType)) {
                    ......
                    return (T) genericMessageConverter.read(this.responseType, null, responseWrapper);
                }
            }
            if (this.responseClass != null) {
                if (messageConverter.canRead(this.responseClass, contentType)) {
                    ......
                    return (T) messageConverter.read((Class) this.responseClass, responseWrapper);
                }
            }
        }

        throw new RestClientException("Could not extract response: no suitable HttpMessageConverter found " +
                "for response type [" + this.responseType + "] and content type [" + contentType + "]");
    }

如果添加的是FastJsonHttpMessageConverter妖啥,這里他非 GenericHttpMessageConverter接口的實現(xiàn)類(注:可能是當(dāng)前FastJson版本問題霉颠。),不是Spring自帶的荆虱,不是親兒子蒿偎。
直接因為 this.responseClass = null 進(jìn)入下面的內(nèi)容朽们,拋異常。

因為沒有一個HttpMessageConverter 是 CanRead這種類型的诉位。

所以還是找他:MappingJackson2HttpMessageConverter

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末骑脱,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子苍糠,更是在濱河造成了極大的恐慌叁丧,老刑警劉巖,帶你破解...
    沈念sama閱讀 217,907評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件岳瞭,死亡現(xiàn)場離奇詭異拥娄,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)瞳筏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,987評論 3 395
  • 文/潘曉璐 我一進(jìn)店門稚瘾,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人乏矾,你說我怎么就攤上這事孟抗。” “怎么了钻心?”我有些...
    開封第一講書人閱讀 164,298評論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長铅协。 經(jīng)常有香客問我捷沸,道長,這世上最難降的妖魔是什么狐史? 我笑而不...
    開封第一講書人閱讀 58,586評論 1 293
  • 正文 為了忘掉前任痒给,我火速辦了婚禮,結(jié)果婚禮上骏全,老公的妹妹穿的比我還像新娘苍柏。我一直安慰自己,他們只是感情好姜贡,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,633評論 6 392
  • 文/花漫 我一把揭開白布试吁。 她就那樣靜靜地躺著,像睡著了一般楼咳。 火紅的嫁衣襯著肌膚如雪熄捍。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,488評論 1 302
  • 那天母怜,我揣著相機(jī)與錄音余耽,去河邊找鬼。 笑死苹熏,一個胖子當(dāng)著我的面吹牛碟贾,可吹牛的內(nèi)容都是我干的币喧。 我是一名探鬼主播,決...
    沈念sama閱讀 40,275評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼袱耽,長吁一口氣:“原來是場噩夢啊……” “哼粱锐!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起扛邑,我...
    開封第一講書人閱讀 39,176評論 0 276
  • 序言:老撾萬榮一對情侶失蹤怜浅,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后蔬崩,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體恶座,經(jīng)...
    沈念sama閱讀 45,619評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,819評論 3 336
  • 正文 我和宋清朗相戀三年沥阳,在試婚紗的時候發(fā)現(xiàn)自己被綠了跨琳。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,932評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡桐罕,死狀恐怖脉让,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情功炮,我是刑警寧澤溅潜,帶...
    沈念sama閱讀 35,655評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站薪伏,受9級特大地震影響滚澜,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜嫁怀,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,265評論 3 329
  • 文/蒙蒙 一设捐、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧塘淑,春花似錦萝招、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,871評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至召噩,卻和暖如春母赵,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背具滴。 一陣腳步聲響...
    開封第一講書人閱讀 32,994評論 1 269
  • 我被黑心中介騙來泰國打工凹嘲, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人构韵。 一個月前我還...
    沈念sama閱讀 48,095評論 3 370
  • 正文 我出身青樓周蹭,卻偏偏與公主長得像趋艘,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子凶朗,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,884評論 2 354

推薦閱讀更多精彩內(nèi)容