SpringBoot Web如何針對不同的客戶端定制errorPage

簡介###

客戶端兩個,針對兩類用戶辙浑,渲染的H5頁面各有不同激涤,報錯處理頁面也存在區(qū)別;將所有服務放置同一個Web項目例衍,需要在報錯的時候指向不同的錯誤頁面昔期;
最開始的想法是通過區(qū)分服務路徑,最終證明此方法可行佛玄,如下是實現(xiàn)步驟硼一。

啟動類定義EmbeddedServletContainerCustomizer對象###

這里定義的error page 路徑與MainsiteErrorController相關聯(lián);

@Bean
    public EmbeddedServletContainerCustomizer containerCustomizer(){
        return new EmbeddedServletContainerCustomizer(){
            @Override
            public void customize(ConfigurableEmbeddedServletContainer container) {
                container.addErrorPages(new ErrorPage(HttpStatus.NOT_FOUND, "/error/404"));
                container.addErrorPages(new ErrorPage(HttpStatus.INTERNAL_SERVER_ERROR, "/error/500"));
                container.addErrorPages(new ErrorPage(java.lang.Throwable.class,"/error/500"));
            }
        };
    }

實現(xiàn)ErrorController接口###


@Controller
@RequestMapping(value = "error")
@EnableConfigurationProperties({ ServerProperties.class })
public class MainsiteErrorController implements ErrorController {

    private static final String REQUEST_PATH_KEY = "path";

    private static final Logger logger = LoggerFactory.getLogger(MainsiteErrorController.class);

    private ErrorAttributes errorAttributes;

    @Autowired
    private ServerProperties serverProperties;

    /**
     * 初始化ExceptionController
     * 
     * @param errorAttributes
     */
    @Autowired
    public MainsiteErrorController(ErrorAttributes errorAttributes) {
        Assert.notNull(errorAttributes, "ErrorAttributes must not be null");
        this.errorAttributes = errorAttributes;
    }

    /**
     * 定義404的ModelAndView
     * 
     * @param request
     * @param response
     * @return
     */
    @RequestMapping(produces = "text/html", value = "404")
    public ModelAndView errorHtml404(HttpServletRequest request, HttpServletResponse response) {
        response.setStatus(getStatus(request).value());
        Map<String, Object> model = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML));
        return new ModelAndView(this.getTerminalErrorPageByRequestURL(model, "404"), model);
    }

    /**
     * 定義404的JSON數(shù)據(jù)
     * 
     * @param request
     * @return
     */
    @RequestMapping(value = "404")
    @ResponseBody
    public ResponseEntity<Map<String, Object>> error404(HttpServletRequest request) {
        Map<String, Object> body = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML));
        HttpStatus status = getStatus(request);
        return new ResponseEntity<Map<String, Object>>(body, status);
    }

    /**
     * 定義500的ModelAndView
     * 
     * @param request
     * @param response
     * @return
     */
    @RequestMapping(produces = "text/html", value = "500")
    public ModelAndView errorHtml500(HttpServletRequest request, HttpServletResponse response) {
        response.setStatus(getStatus(request).value());
        Map<String, Object> model = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML));
        return new ModelAndView(this.getTerminalErrorPageByRequestURL(model, "500"), model);
    }

    /**
     * 定義500的錯誤JSON信息
     * 
     * @param request
     * @return
     */
    @RequestMapping(value = "500")
    @ResponseBody
    public ResponseEntity<Map<String, Object>> error500(HttpServletRequest request) {
        Map<String, Object> body = getErrorAttributes(request, isIncludeStackTrace(request, MediaType.TEXT_HTML));
        HttpStatus status = getStatus(request);
        return new ResponseEntity<Map<String, Object>>(body, status);
    }

    /**
     * Determine if the stacktrace attribute should be included.
     * 
     * @param request
     *            the source request
     * @param produces
     *            the media type produced (or {@code MediaType.ALL})
     * @return if the stacktrace attribute should be included
     */
    protected boolean isIncludeStackTrace(HttpServletRequest request, MediaType produces) {
        ErrorProperties.IncludeStacktrace include = this.serverProperties.getError().getIncludeStacktrace();
        if (include == ErrorProperties.IncludeStacktrace.ALWAYS) {
            return true;
        }
        if (include == ErrorProperties.IncludeStacktrace.ON_TRACE_PARAM) {
            return getTraceParameter(request);
        }
        return false;
    }

    /**
     * 獲取錯誤的信息
     * 
     * @param request
     * @param includeStackTrace
     * @return
     */
    private Map<String, Object> getErrorAttributes(HttpServletRequest request, boolean includeStackTrace) {
        RequestAttributes requestAttributes = new ServletRequestAttributes(request);
        return this.errorAttributes.getErrorAttributes(requestAttributes, includeStackTrace);
    }

    /**
     * 是否包含trace
     * 
     * @param request
     * @return
     */
    private boolean getTraceParameter(HttpServletRequest request) {
        String parameter = request.getParameter("trace");
        if (parameter == null) {
            return false;
        }
        return !"false".equals(parameter.toLowerCase());
    }

    /**
     * 獲取錯誤編碼
     * 
     * @param request
     * @return
     */
    private HttpStatus getStatus(HttpServletRequest request) {
        Integer statusCode = (Integer) request.getAttribute("javax.servlet.error.status_code");
        if (statusCode == null) {
            return HttpStatus.INTERNAL_SERVER_ERROR;
        }
        try {
            return HttpStatus.valueOf(statusCode);
        } catch (Exception ex) {
            return HttpStatus.INTERNAL_SERVER_ERROR;
        }
    }

    private String getTerminalErrorPageByRequestURL(Map<String, Object> model, String errorCode) {
        String path = ((String) model.get(REQUEST_PATH_KEY));
        String terminalErrorPage = path.startsWith(GenericConstants.STUDENT_TERMINAL_TYPE)
                ? GenericConstants.STUDENT_TERMINAL_TYPE.concat("/").concat(errorCode) : errorCode;
        terminalErrorPage = "errorpage/" + terminalErrorPage;
        logger.debug("errorPage -> {}", terminalErrorPage);
        return terminalErrorPage;
    }

    /**
     * 實現(xiàn)錯誤路徑,暫時無用
     * 
     * @see ExceptionMvcAutoConfiguration#containerCustomizer()
     * @return
     */
    @Override
    public String getErrorPath() {
        return "";
    }
}

其中較為關鍵的方法是getTerminalErrorPageByRequestURL方法梦抢,該方法會根據(jù)請求的path找到各客戶端請求前綴(通過前綴區(qū)分提供給不同客戶端的服務般贼,以此報錯時指向不同的錯誤頁面)
這是基于Thymeleaf+SpringBoot實現(xiàn)的網(wǎng)站http://www.hqast.com/

最后編輯于
?著作權歸作者所有,轉載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市奥吩,隨后出現(xiàn)的幾起案子哼蛆,更是在濱河造成了極大的恐慌,老刑警劉巖霞赫,帶你破解...
    沈念sama閱讀 211,376評論 6 491
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件腮介,死亡現(xiàn)場離奇詭異,居然都是意外死亡端衰,警方通過查閱死者的電腦和手機叠洗,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,126評論 2 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來旅东,“玉大人灭抑,你說我怎么就攤上這事〉执” “怎么了腾节?”我有些...
    開封第一講書人閱讀 156,966評論 0 347
  • 文/不壞的土叔 我叫張陵,是天一觀的道長荤牍。 經(jīng)常有香客問我案腺,道長,這世上最難降的妖魔是什么康吵? 我笑而不...
    開封第一講書人閱讀 56,432評論 1 283
  • 正文 為了忘掉前任救湖,我火速辦了婚禮,結果婚禮上涎才,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好耍铜,可當我...
    茶點故事閱讀 65,519評論 6 385
  • 文/花漫 我一把揭開白布邑闺。 她就那樣靜靜地躺著,像睡著了一般棕兼。 火紅的嫁衣襯著肌膚如雪陡舅。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,792評論 1 290
  • 那天伴挚,我揣著相機與錄音靶衍,去河邊找鬼。 笑死茎芋,一個胖子當著我的面吹牛颅眶,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播田弥,決...
    沈念sama閱讀 38,933評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼涛酗,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了偷厦?” 一聲冷哼從身側響起商叹,我...
    開封第一講書人閱讀 37,701評論 0 266
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎只泼,沒想到半個月后剖笙,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,143評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡请唱,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,488評論 2 327
  • 正文 我和宋清朗相戀三年弥咪,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片籍滴。...
    茶點故事閱讀 38,626評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡酪夷,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出孽惰,到底是詐尸還是另有隱情晚岭,我是刑警寧澤,帶...
    沈念sama閱讀 34,292評論 4 329
  • 正文 年R本政府宣布勋功,位于F島的核電站坦报,受9級特大地震影響,放射性物質發(fā)生泄漏狂鞋。R本人自食惡果不足惜片择,卻給世界環(huán)境...
    茶點故事閱讀 39,896評論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望骚揍。 院中可真熱鬧字管,春花似錦啰挪、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,742評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至硫戈,卻和暖如春锰什,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背丁逝。 一陣腳步聲響...
    開封第一講書人閱讀 31,977評論 1 265
  • 我被黑心中介騙來泰國打工汁胆, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人霜幼。 一個月前我還...
    沈念sama閱讀 46,324評論 2 360
  • 正文 我出身青樓嫩码,卻偏偏與公主長得像,于是被迫代替她去往敵國和親辛掠。 傳聞我的和親對象是個殘疾皇子谢谦,可洞房花燭夜當晚...
    茶點故事閱讀 43,494評論 2 348

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

  • Spring Cloud為開發(fā)人員提供了快速構建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務發(fā)現(xiàn)萝衩,斷路器回挽,智...
    卡卡羅2017閱讀 134,629評論 18 139
  • 22年12月更新:個人網(wǎng)站關停,如果仍舊對舊教程有興趣參考 Github 的markdown內(nèi)容[https://...
    tangyefei閱讀 35,165評論 22 257
  • __block和__weak修飾符的區(qū)別其實是挺明顯的:1.__block不管是ARC還是MRC模式下都可以使用猩谊,...
    LZM輪回閱讀 3,291評論 0 6
  • 沒有人天生是工作狂千劈,只因為不想落人后,只因為不想輸?shù)秒y看
    93650345d0d1閱讀 107評論 0 1
  • 貌似今天的主題是高跟鞋牌捷,很遺憾不能像你們一樣如數(shù)家珍似的來談一下和它的親密感受墙牌。不過還是愿意從欣賞者的角度和立場純...
    疲憊的快樂閱讀 216評論 0 0