LazyInitializationException с Spring Messaging и GsonMessageConverter
Я использую Spring Boot
с Spring Messaging
а также Gson
для подключения через веб-сокет и отправки сообщений клиенту. Теперь я сталкиваюсь LazyInitializationException
ошибка после добавления нового поля в мою модель и отправки его клиенту.
Вот мой модельный класс:
@Entity
public class Vehicle {
@Expose
@Id
@GeneratedValue(strategy = GenerationType.AUTO)
private long id;
@Expose
private String name;
//...
//some other simple fields
//...
@Expose
@ElementCollection(fetch = FetchType.LAZY)
@CollectionTable(name = "vehicle_property_values", joinColumns = @JoinColumn(name = "vehicle_id"))
@MapKeyColumn(name = "key")
@Column(name="value")
private Map<String, String> properties = new HashMap<>();
}
я использую GsonMessageConverter
преобразовать объект в json и отправить клиенту с https://github.com/Coding/WebIDE-Backend/blob/master/src/main/java/net/coding/ide/web/message/GsonMessageConverter.java
Конвертер работает нормально, но недавно я добавил Map<String, String> properties
к моему Vehicle
модель и есть проблема. Я получаю исключение:
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: com.test.vehicles.model.Vehicle.properties, could not initialize proxy - no Session
в строке 105 на GsonMessageConverter
this.gson.toJson(payload, writer);
Вот как я отправляю сообщения в своем классе обслуживания:
@Service
@Transactional
class VehicleServiceImpl implements VehicleService {
@Autowired
VehicleRepository vehicleRepository; //JpaRepository
@Autowired
SimpMessagingTemplate simpMessagingTemplate;
private final ConcurrentLinkedDeque<Vehicle> pendingVehicles = new ConcurrentLinkedDeque<>();
//..some other fields and methods irrelevant in this case
//method called from controller
@Override
void addPendingVehicle(Vehicle vehicle) {
//set some variables in vehicle
vehicle = vehicleRepository.save(vehicle);
pendingVehicles.add(vehicle);
}
@Scheduled(initialDelay = 60000, fixedRate = 60000)
@Transactional
void sendPendingVehicles() {
if(pendingVehicles.size() > 0) {
simpMessagingTemplate.convertAndSend("/topic/vehicles", pendingVehicles);
pendingVehicles.clear();
}
}
}
Вот моя конфигурация WebSocket (только не пустые методы):
@Configuration
@EnableWebSocketMessageBroker
public class WebSocketConfig implements WebSocketMessageBrokerConfigurer {
@Override
public void configureMessageBroker(MessageBrokerRegistry config) {
config.enableSimpleBroker("/queue", "/topic");
}
@Override
public void registerStompEndpoints(StompEndpointRegistry registry) {
registry.addEndpoint("/testapp").setAllowedOrigins("*").withSockJS();
}
@Override
public boolean configureMessageConverters(List<MessageConverter> converters) {
Gson gson = new GsonBuilder().excludeFieldsWithoutExposeAnnotation().create();
converters.add(new GsonMessageConverter().setGson(gson));
return true;
}
}
Я понятия не имею, как это исправить. Как сделать это транзакционным?
1 ответ
Мне не нужно было properties
отправлять в websocket сообщения, поэтому я создал аннотацию
@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.FIELD)
public @interface ExcludeWebSocketSerialization { }
И добавил ExclusionStrategy
к моему Gson
.addSerializationExclusionStrategy(new ExclusionStrategy() {
@Override
public boolean shouldSkipField(FieldAttributes fieldAttributes) {
return fieldAttributes.getAnnotation(ExcludeWebSocketSerialization.class) != null;
}
@Override
public boolean shouldSkipClass(Class<?> aClass) {
return false;
}
})