Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Retrofit2 + SimpleXML in Kotlin: MethodException: Annotation must mark a set or get method

I want to fetch XML data from API and map it to Kotlin model object by using Retrofit2 + SimpleXML in Kotlin.

However, I got such as the following error message from SimpleXML.

org.simpleframework.xml.core.MethodException: Annotation @org.simpleframework.xml.Element(data=false, name=, required=true, type=void) must mark a set or get method

This is fetched XML data

<?xml version="1.0" encoding="UTF-8" standalone="yes"?>
<response>
    <result code="0">Success</result>
    <token>XXXXXXXXXXXXXXXXXXXX</token>
    <uid>4294967295</uid>
</response>

Kotlin model object is below

@Root(name = "response")
public class User() {
    @Element public var result: String? = null
    @Element public var token: String? = null
    @Element public var uid: String? = null
}

and APIClient is as follows.

interface  MyService {
    @GET("/testLogin.xml")
    fun getUser(): Call<User>
}

val retrofit = Retrofit.Builder()
        .baseUrl(baseURL)
        .addConverterFactory(SimpleXmlConverterFactory.create())
        .build()
val call = retrofit.create(MyService::class.java).getUser()
call.enqueue(object: Callback<User> {
        override fun onResponse(p0: Call<User>?, response: Response<User>?) {
            val response = response?.body()
        }
        override fun onFailure(p0: Call<User>?, t: Throwable?) {
            Log.e("APIClient", t?.message)
        }

I got HTTP status code 200 and correct XML data. So I think my declaration of model object is problem.

like image 840
c8112002 Avatar asked Mar 10 '16 10:03

c8112002


3 Answers

This is the same problem as: kotlin data class + bean validation jsr 303

You need to use Annotation use-site targets since the default for an annotation on a property is prioritized as:

  • parameter (if declared in constructor)
  • property (if the target site allows, but only Kotlin created annotations can do this)
  • field (likely what happened here, which isn't what you wanted).

Use get or set target to place the annotation on the getter or setter. Here it is for the getter:

@Root(name = "response")
public class User() {
    @get:Element public var result: String? = null
    @get:Element public var token: String? = null
    @get:Element public var uid: String? = null
}

See the linked answer for the details.

like image 121
3 revs, 2 users 94% Avatar answered Nov 06 '22 22:11

3 revs, 2 users 94%


To avoid the error in parse do one should place annotation tags @set e @get

@Root(name = "response", strict = false)
public class User() {

    @set:Element(name = "result")
    @get:Element(name = "result")
    public var result: String? = null

    @set:Element(name = "token")
    @get:Element(name = "token")
    @Element public var token: String? = null

    @set:Element(name = "uid")
    @get:Element(name = "uid")
    public var uid: String? = null
}
like image 4
Tairo Roberto Avatar answered Nov 06 '22 22:11

Tairo Roberto


Not familiar with the SimpleXml library but if it's annotation processor is looking for specific get and set methods you may want to try setting up your class in the following way:

@Root(name="response") class User() {
var result:String?=null
    @Element
    get
    @Element
    set
var token:String?=null
    @Element
    get
    @Element
    set
var uid:String?=null
    @Element
    get
    @Element
    set
}

As well, if the @Element annotation supports field types you could use this approach:

@Root(name="response") class User() {
    @Element @JvmField var result:String?=null
    @Element @JvmField var token:String?=null
    @Element @JvmField var uid:String?=null
}
like image 2
Matt Sams Avatar answered Nov 06 '22 22:11

Matt Sams