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.
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:
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.
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
}
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
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With