JAXB или JAX-RS заключает числа в моих ответах JSON в кавычки, превращая их в строки.Почему это поведение по умолчанию и как это исправить? - PullRequest
5 голосов
/ 07 марта 2012

В настоящее время я работаю над RESTful API. У меня есть класс Employee и класс EmployeeResource. У меня также есть собственный DateAdapter, который меняет мои свойства Date на длинные метки времени. Однако в моих ответах JSON метки времени отображаются в виде строк (заключенных в двойные кавычки), а не в числа (без двойных кавычек). Вот сокращенная версия моего кода и записанный ответ JSON ...

Пользовательский DateAdapter

public class DateAdapter extends XmlAdapter<Long, Date> {
    @Override
    public Date unmarshal(Long v) throws Exception {
        return new Date(Long.valueOf(v));  
    }
    @Override
    public Long marshal(Date v) throws Exception {
        return v.getTime();  
    }
}

Класс сущности

@Entity
@javax.xml.bind.annotation.XmlRootElement
@XmlType(propOrder={"createdOn","empId"})
public class Employee implements Serializable {
    private Date createdOn;
    private Integer empId;

    @Column(nullable=false)
    @Temporal(TemporalType.TIMESTAMP)
    @XmlJavaTypeAdapter(DateAdapter.class)
    public Date getCreatedOn() {
        return createdOn;
    }
    public void setCreatedOn(Date createdOn) {
        this.createdOn = createdOn;
    }

    @Id
    @XmlID
    public Integer getEmpId() {
        return empId;
    }
    public void setEmpId(Integer empId) {
        this.empId = empId;
    }
}

EmployeeResource

@Path("/Employees")
@javax.xml.bind.annotation.XmlRootElement 
@XmlType(propOrder={"hateoas","employees"})
public class EmployeeResource {
    List<Employee> employees;

    public List<Employee> getEmployees() {
        return employees;
    }
    public void setEmployees(List<Employee> employees) {
        this.employees = employees;
    }
    @GET
    @Path("/{id}")
    @Produces("application/json")
    public Response getEmployee(@Context UriInfo ui, @PathParam("id") Integer id) {
        Session session = HibernateUtil.getSession();
        session.beginTransaction();
        Criteria criteria=session.createCriteria(Employee.class);
        criteria.add(Restrictions.eq("empId", new Integer(10150)));
        this.employees = criteria.list();
        return Response.ok(this).build();
    }
}

Текущий ответ JSON

{
  "employees":{
    "createdOn":"1330915130163",
    "empId":"10150"
  }
}

Ожидаемый ответ JSON

{
  "employees":{
    "createdOn":1330915130163,
    "empId":10150
  }
}

Я предполагаю, что есть какой-то способ запретить JAXB или JAX-RS заключать все числа в кавычки. Может ли кто-нибудь направить меня туда, где я могу это настроить?

Заранее спасибо!

РЕДАКТИРОВАТЬ # 1 2012.03.07 Итак, после некоторого дополнительного исследования, я думаю, что моя проблема связана с используемой по умолчанию JSONConfiguration.Notation, MAPPED. Похоже, что NATURAL JSONConfiguration.Notation даст мне то, что я хочу. Однако я не нашел четкого примера того, как применять это приложение в широком масштабе. Я предполагаю, что я бы указал это в своем классе ApplicationConfig, который расширяет javax.ws.rs.core.Application.

РЕДАКТИРОВАТЬ № 2 2012.03.10 Итак, после некоторого исследования я решил использовать библиотеку JSON-анализатора, Джексон. Кажется, это наиболее полное решение JSON, использующее только конфигурацию по умолчанию. По умолчанию даты переводятся в соответствующие им временные метки и сериализуются в виде чисел (без кавычек). Единственный недостаток, с которым я столкнулся, это то, что Джексон в настоящее время не поддерживает аннотации JAXB, "@XmlID" и "@XmlIDREF". Поскольку в моей модели данных есть прямые ссылки на себя (не показано выше), я создал еще один вопрос для обсуждения. Если кому-то интересно нажмите здесь, чтобы перейти к этой теме ...

1 Ответ

2 голосов
/ 07 марта 2012

Примечание: Я EclipseLink JAXB (MOXy) и являюсь членом JAXB 2 (JSR-222 ) экспертная группа.

Используемая вами реализация JAX-RS может использовать JAXB (JSR-222) с чем-то вроде Jettison производить JSON. Jettison предоставляет API-интерфейс StAX для взаимодействия с JSON, поскольку эти API-интерфейсы StAX не имеют никакого вида ввода текста WRT, все простые значения обрабатываются как строки:

Чтобы получить поведение, которое вы ищете, вы можете использовать другое решение для связывания. Мы добавляем эту поддержку в компонент MOXy для EclipseLink 2.4:

Чтобы настроить MOXy в качестве поставщика JSON-привязки в среде JAX-RS, вы можете создать MessageBodyReader / MessageBodyWriter, который будет выглядеть следующим образом:

import java.io.*;
import java.lang.annotation.Annotation;
import java.lang.reflect.Type;
import java.util.Collection;
import javax.xml.transform.stream.StreamSource;

import javax.ws.rs.*;
import javax.ws.rs.core.*;
import javax.ws.rs.ext.*;
import javax.xml.bind.*;

@Provider
@Produces(MediaType.APPLICATION_JSON)
@Consumes(MediaType.APPLICATION_JSON)
public class MOXyJSONProvider implements 
    MessageBodyReader<Object>, MessageBodyWriter<Object>{

    @Context
    protected Providers providers;

    public boolean isReadable(Class<?> type, Type genericType,
        Annotation[] annotations, MediaType mediaType) {
        return true;
    }

    public Object readFrom(Class<Object> type, Type genericType,
        Annotation[] annotations, MediaType mediaType,
        MultivaluedMap<String, String> httpHeaders, InputStream entityStream)
        throws IOException, WebApplicationException {
        try {
            Unmarshaller u = getJAXBContext(type, mediaType).createUnmarshaller();
            u.setProperty("eclipselink.media-type", mediaType.toString());
            u.setProperty("eclipselink.json.include-root", false);//tiny fix
            return u.unmarshal(new StreamSource(entityStream), (Class) genericType);
        } catch(JAXBException jaxbException) {
            throw new WebApplicationException(jaxbException);
        }
    }

    public boolean isWriteable(Class<?> type, Type genericType,
        Annotation[] annotations, MediaType mediaType) {
        return true;
    }

    public void writeTo(Object object, Class<?> type, Type genericType,
        Annotation[] annotations, MediaType mediaType,
        MultivaluedMap<String, Object> httpHeaders,
        OutputStream entityStream) throws IOException,
        WebApplicationException {
        try {
            Marshaller m = getJAXBContext(Customer.class, mediaType).createMarshaller();
            m.setProperty("eclipselink.media-type", mediaType.toString());
            m.setProperty("eclipselink.json.include-root", false);
            m.marshal(object, entityStream);
        } catch(JAXBException jaxbException) {
            throw new WebApplicationException(jaxbException);
        }
    }

    public long getSize(Object t, Class<?> type, Type genericType,
        Annotation[] annotations, MediaType mediaType) {
        return -1;
    }

    private JAXBContext getJAXBContext(Class<?> type, MediaType mediaType) 
        throws JAXBException {
        ContextResolver<JAXBContext> resolver 
            = providers.getContextResolver(JAXBContext.class, mediaType);
        JAXBContext jaxbContext;
        if(null == resolver || null == (jaxbContext = resolver.getContext(type))) {
            return JAXBContext.newInstance(type);
        } else {
            return jaxbContext;
        }
    }

}

Для получения дополнительной информации

...