在 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)
    }