在 HttpSession java.io.NotSerializableException 中保留状态机上下文:org.springframework.statemachine.support.DefaultStateMachineContext
Persiting a state machine context in HttpSession java.io.NotSerializableException: org.springframework.statemachine.support.DefaultStateMachineContext
我创建了一个 StateMachinePersist
实现来持久化 StateMachineContext 和 HttpSession:
@Suppress("UNCHECKED_CAST")
class SessionStateMachinePersist(private val session: HttpSession) : StateMachinePersist<States, String, String> {
private val logger = LoggerFactory.getLogger(javaClass)
override fun read(contextObj: String): StateMachineContext<States, String>? {
val stateMachineContext = session.getAttribute("${session.id}-$contextObj") as StateMachineContext<States, String>?
logger.debug("Session {} has state machine context {}?", session.id, contextObj, stateMachineContext != null)
return stateMachineContext
}
override fun write(context: StateMachineContext<States, String>, contextObj: String) {
session.setAttribute("${session.id}-$contextObj", context)
}
}
当它要保留 DefaultStateMachineContext
时,它会抛出以下异常,因为 DefaultStateMachineContext
不可序列化:Caused by: java.io.NotSerializableException: org.springframework.statemachine.support.DefaultStateMachineContext
如何使用可序列化的 StateMachineContext 实现?
这就是我设法使状态机上下文可序列化并将其持久保存在 Http 会话中的方法:
SerializableStateMachineContext
class SerializableStateMachineContext<S, E>(var id: String?, var childs: List<StateMachineContext<S, E>>?, var state: S?,
var historyStates: Map<S, S>?, var event: E?, var eventHeaders: Map<String, Any>?,
var extendedState: SerializableExtendedState?) : Serializable {
constructor() : this(null, null, null, null, null, null, null)
}
SerializableExtendedState
class SerializableExtendedState(var variables: Map<Any, Any>) : Serializable {
constructor() : this(mutableMapOf())
}
SessionStateMachinePersist
@Suppress("UNCHECKED_CAST")
class SessionStateMachinePersist(private val session: HttpSession) : StateMachinePersist<States, String, String> {
private val logger = LoggerFactory.getLogger(javaClass)
override fun read(contextObj: String): StateMachineContext<States, String>? {
val stateMachineContext = session.getAttribute("${session.id}-$contextObj") as SerializableStateMachineContext<States, String>?
logger.debug("Session {} has state machine context {}?", session.id, contextObj, stateMachineContext != null)
return DefaultStateMachineContext(stateMachineContext?.childs ?: mutableListOf(), stateMachineContext?.state, stateMachineContext?.event,
stateMachineContext?.eventHeaders, DefaultExtendedState(stateMachineContext?.extendedState?.variables ?: mutableMapOf()), stateMachineContext?.historyStates)
}
override fun write(context: StateMachineContext<States, String>, contextObj: String) {
val serializableExtendedState = SerializableExtendedState(context.extendedState.variables.toMap())
val serializableStateMachineContext = SerializableStateMachineContext(null, context.childs, context.state, context.historyStates, context.event, context.eventHeaders, serializableExtendedState)
session.setAttribute("${session.id}-$contextObj", serializableStateMachineContext)
}
}
然后,无论你在哪里需要操作状态机,你只需要恢复它,执行触发操作并持久化它,例如:
@PostMapping(value = ["/process-input"], consumes = [MediaType.APPLICATION_JSON_VALUE], produces = [MediaType.APPLICATION_JSON_VALUE])
@ResponseBody
fun processInput(session: HttpSession, @RequestBody projectCreationInput: ProjectCreationInput, model: ModelMap, status: SessionStatus) {
val stateMachinePersister = buildStateMachinePersister(session)
stateMachinePersister.restore(stateMachine, "myid")
...
triggerStateMachineExecution(projectCreationInput, event, status)
...
stateMachinePersister.persist(stateMachine, "myid")
}
private fun buildStateMachinePersister(session: HttpSession): DefaultStateMachinePersister<States, String, String> {
val stateMachinePersist = SessionStateMachinePersist(session)
return DefaultStateMachinePersister(stateMachinePersist)
}
我创建了一个 StateMachinePersist
实现来持久化 StateMachineContext 和 HttpSession:
@Suppress("UNCHECKED_CAST")
class SessionStateMachinePersist(private val session: HttpSession) : StateMachinePersist<States, String, String> {
private val logger = LoggerFactory.getLogger(javaClass)
override fun read(contextObj: String): StateMachineContext<States, String>? {
val stateMachineContext = session.getAttribute("${session.id}-$contextObj") as StateMachineContext<States, String>?
logger.debug("Session {} has state machine context {}?", session.id, contextObj, stateMachineContext != null)
return stateMachineContext
}
override fun write(context: StateMachineContext<States, String>, contextObj: String) {
session.setAttribute("${session.id}-$contextObj", context)
}
}
当它要保留 DefaultStateMachineContext
时,它会抛出以下异常,因为 DefaultStateMachineContext
不可序列化:Caused by: java.io.NotSerializableException: org.springframework.statemachine.support.DefaultStateMachineContext
如何使用可序列化的 StateMachineContext 实现?
这就是我设法使状态机上下文可序列化并将其持久保存在 Http 会话中的方法:
SerializableStateMachineContext
class SerializableStateMachineContext<S, E>(var id: String?, var childs: List<StateMachineContext<S, E>>?, var state: S?,
var historyStates: Map<S, S>?, var event: E?, var eventHeaders: Map<String, Any>?,
var extendedState: SerializableExtendedState?) : Serializable {
constructor() : this(null, null, null, null, null, null, null)
}
SerializableExtendedState
class SerializableExtendedState(var variables: Map<Any, Any>) : Serializable {
constructor() : this(mutableMapOf())
}
SessionStateMachinePersist
@Suppress("UNCHECKED_CAST")
class SessionStateMachinePersist(private val session: HttpSession) : StateMachinePersist<States, String, String> {
private val logger = LoggerFactory.getLogger(javaClass)
override fun read(contextObj: String): StateMachineContext<States, String>? {
val stateMachineContext = session.getAttribute("${session.id}-$contextObj") as SerializableStateMachineContext<States, String>?
logger.debug("Session {} has state machine context {}?", session.id, contextObj, stateMachineContext != null)
return DefaultStateMachineContext(stateMachineContext?.childs ?: mutableListOf(), stateMachineContext?.state, stateMachineContext?.event,
stateMachineContext?.eventHeaders, DefaultExtendedState(stateMachineContext?.extendedState?.variables ?: mutableMapOf()), stateMachineContext?.historyStates)
}
override fun write(context: StateMachineContext<States, String>, contextObj: String) {
val serializableExtendedState = SerializableExtendedState(context.extendedState.variables.toMap())
val serializableStateMachineContext = SerializableStateMachineContext(null, context.childs, context.state, context.historyStates, context.event, context.eventHeaders, serializableExtendedState)
session.setAttribute("${session.id}-$contextObj", serializableStateMachineContext)
}
}
然后,无论你在哪里需要操作状态机,你只需要恢复它,执行触发操作并持久化它,例如:
@PostMapping(value = ["/process-input"], consumes = [MediaType.APPLICATION_JSON_VALUE], produces = [MediaType.APPLICATION_JSON_VALUE])
@ResponseBody
fun processInput(session: HttpSession, @RequestBody projectCreationInput: ProjectCreationInput, model: ModelMap, status: SessionStatus) {
val stateMachinePersister = buildStateMachinePersister(session)
stateMachinePersister.restore(stateMachine, "myid")
...
triggerStateMachineExecution(projectCreationInput, event, status)
...
stateMachinePersister.persist(stateMachine, "myid")
}
private fun buildStateMachinePersister(session: HttpSession): DefaultStateMachinePersister<States, String, String> {
val stateMachinePersist = SessionStateMachinePersist(session)
return DefaultStateMachinePersister(stateMachinePersist)
}