简体   繁体   English

Retrofit2:错误处理

[英]Retrofit2: Error Handling

I am using an XML API which returns lists of users, customers, articles and so on: 我正在使用XML API,该API返回用户,客户,文章等的列表:

<userlist access="ro">
<multipage>
<!-- Info about nums of pages, elements per page etc. -->
</multipage>
<user access="ro">
<surname access="rw" type="string">Jon</surname>
<lastname access="rw" type="string">Doe</lastname>
<scannerpin access="ro" type="int">1234</scannerpin>
</user>
<user>
<!-- ... -->
</user>
</userlist>

Currently, my POJO looks something like this: 目前,我的POJO看起来像这样:

@Root(name="userlist")
public class User extends XmlObject {
    @Element
    private XmlElement surname;

    @Element
    private XmlElement lastname;

    @Element
    private XmlElement scannerpin;
}

@Root(strict=false)
/* XXX no support for generics in SimpleXML. TODO Find better solution */
public class XmlUserList extends XmlList<User> {

    /**
     * Constructor for XmlUserList
     *
     * @param multipage
     */
    public XmlUserList(@Element(name = "multipage") Multipage multipage) {
        super(multipage);
    }
}

public class XmlElement extends XmlNode {
    protected static final String rcsid = "$Id: XmlElement.java 29660 2016-08-17 15:08:39Z jb $";

    /**
     * The element's value
     */
    @Text(required=false)
    String value;

    /**
     * Default constructor for XmlElement
     */
    public XmlElement() {
    }

    /**
     * Getter for value
     * @return this.value
     */
    public String getValue() {
        return this.value != null ? this.value : "";
    }

    /**
     * Setter for value
     * @param value The new value
     * @throws UnsupportedOperationException Iff this element is readOnly
     */
    public void setValue(String value) throws UnsupportedOperationException {
            this.value = value;
    }

    public void setValue(Integer value) {
        this.value = String.valueOf(value);
    }
}


public abstract class XmlList<T> extends XmlNode {
    protected static final String rcsid = "$Id: XmlList.java 29660 2016-08-17 15:08:39Z jb $";

    /**
     * List entries
     */
    @ElementListUnion({
            @ElementList(inline = true, name = "user", type = User.class,required=false),
            @ElementList(inline = true, name = "customer", type = Customer.class,required=false),
            @ElementList(inline = true, name = "article", type = Article.class,required=false)
    })
    private List<T> list;

    /**
     * Multipage object
     */
    private Multipage multipage;

    /**
     * Constructor for XmlList
     */
    public XmlList(@Element(name="multipage") Multipage multipage) {
        this.multipage = multipage;
    }

    /**
     * getter for list
     * @return this.list
     */
    public List<T> getList() {
        return (this.list);
    }

    public void setList(List<T> list) {
        this.list = list;
    }

    /**
     * Getter for Multipage
     * @return this.multipage
     */
    @Element(name="multipage")
    public Multipage getMultipage() {
        return (this.multipage);
    }
}

On error (eg wrong login, downtime), the backend does not return a list, but an error message and sends HTTP 200, anway: 发生错误(例如错误的登录,停机)时,后端不会返回列表,而是返回错误消息并发送HTTP 200,但仍会:

<error>
<message>Not logged in</message>
<softver>2.4.0_231445</softver>
</error>

I am wondering how to catch errors the right way here. 我想知道如何在这里正确地捕获错误。 I could add an optional field message to XmlObject, but this would mean that I had to annotate each and every field with required=false to prevent exceptions on error. 我可以在XmlObject中添加可选的字段message ,但这意味着我必须为每个字段加上required=false注释,以防止发生错误。 I tried this approach, but rolled back due to dependency chaos (every list MUST have a multipage member, for example). 我尝试了这种方法,但是由于依赖关系混乱而回滚(例如,每个列表必须具有多页成员)。 I could check for this in a @Commit method, but there are several constraints that I would like to enforce, preferably using SimpleXML annotation. 我可以使用@Commit方法进行检查,但是我想强制使用一些约束,最好使用SimpleXML批注。

How do you handle backend errors in Retrofit2 with SimpleXML converter? 如何使用SimpleXML转换器处理Retrofit2中的后端错误?

I found a solution for this. 我找到了解决方案。 The problem was that the backend returns HTTP/1.1 200 OK on error. 问题是后端在错误时返回HTTP/1.1 200 OK I wrote an interceptor which checks if the response can be parsed into an ApiError object and, if so, overrides the response's HTTP status code to HTTP/1.1 400 Bad Request . 我编写了一个拦截器,用于检查响应是否可以解析为ApiError对象,如果可以,则将响应的HTTP状态代码覆盖为HTTP/1.1 400 Bad Request

final class ErrorInterceptor implements Interceptor {

    @Override
    public Response intercept(Chain chain) throws IOException {
        Response originalResponse = chain.proceed(chain.request());
        Response.Builder responseBuilder = originalResponse.newBuilder();

        /* 
         * get body from response. caution! .string() cannot be called more 
         * than once, so a new response is built with the old response's body
         */
        String bodyString = originalResponse.body().string();

        // if response is an error: override response code to 400
        if (isError(bodyString)) {
            responseBuilder.code(400);
        }

        // clone response into new one
        responseBuilder.body(
            ResponseBody.create(originalResponse.body().contentType(),          
                                bodyString));

        return responseBuilder.build();
    }

    public boolean isError(String body) {
        Serializer ser = new Persister();

        try {
            return ser.validate(ErrorUtils.ApiError.class, body);
            // ser.validate() will throw if body can't be read into ApiError.class
        } catch (Exception e) {
            return false;
        }
    }
}

@Root(name="error")
public class ApiError {

    @Element
    private String message;

    @Element
    private String softver;

    public ApiError(@Element(name="message") String message, @Element(name="softver") String softver) {
        this.message = message;
        this.softver = softver;
    }

    public String getMessage() {
        return message;
    }

    public String getSoftver() {
        return softver;
    }
}

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM