474 lines
9.3 KiB
C
474 lines
9.3 KiB
C
/****************************************************************************
|
|
* This file is part of the project AqHome.
|
|
* AqHome (c) by 2025 Martin Preuss, all rights reserved.
|
|
*
|
|
* The license for this file can be found in the file COPYING which you
|
|
* should have received along with this file.
|
|
****************************************************************************/
|
|
|
|
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include "./msgrequest_p.h"
|
|
#include <aqhome/events2/fdobject.h>
|
|
#include <aqhome/msg/ipc/m_ipc.h>
|
|
|
|
#include <gwenhywfar/debug.h>
|
|
|
|
|
|
GWEN_INHERIT_FUNCTIONS(AQH_MSG_REQUEST)
|
|
GWEN_TREE2_FUNCTIONS(AQH_MSG_REQUEST, AQH_MsgRequest)
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* forward declarations
|
|
* ------------------------------------------------------------------------------------------------
|
|
*/
|
|
|
|
static void _freeFinishedRequests(AQH_MSG_REQUEST *rq);
|
|
|
|
|
|
|
|
/* ------------------------------------------------------------------------------------------------
|
|
* code
|
|
* ------------------------------------------------------------------------------------------------
|
|
*/
|
|
|
|
AQH_MSG_REQUEST *AQH_MsgRequest_new()
|
|
{
|
|
AQH_MSG_REQUEST *rq;
|
|
|
|
GWEN_NEW_OBJECT(AQH_MSG_REQUEST, rq);
|
|
GWEN_INHERIT_INIT(AQH_MSG_REQUEST, rq);
|
|
GWEN_TREE2_INIT(AQH_MSG_REQUEST, rq, AQH_MsgRequest);
|
|
|
|
return rq;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_free(AQH_MSG_REQUEST *rq)
|
|
{
|
|
if (rq) {
|
|
GWEN_TREE2_FINI(AQH_MSG_REQUEST, rq, AQH_MsgRequest);
|
|
GWEN_INHERIT_FINI(AQH_MSG_REQUEST, rq);
|
|
|
|
GWEN_Timestamp_free(rq->expiresAt);
|
|
GWEN_Timestamp_free(rq->createdAt);
|
|
AQH_Message_free(rq->requestMsg);
|
|
AQH_Message_List_free(rq->msgList);
|
|
AQH_Object_free(rq->endpoint);
|
|
|
|
GWEN_FREE_OBJECT(rq);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
int AQH_MsgRequest_GetRequestType(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->requestType:0;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetRequestType(AQH_MSG_REQUEST *rq, int t)
|
|
{
|
|
if (rq)
|
|
rq->requestType=t;
|
|
}
|
|
|
|
|
|
|
|
AQH_MESSAGE *AQH_MsgRequest_GetRequestMsg(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->requestMsg:NULL;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetRequestMsg(AQH_MSG_REQUEST *rq, AQH_MESSAGE *msg)
|
|
{
|
|
if (rq) {
|
|
AQH_Message_free(rq->requestMsg);
|
|
rq->requestMsg=msg;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
AQH_OBJECT *AQH_MsgRequest_GetEndpoint(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->endpoint:NULL;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetEndpoint(AQH_MSG_REQUEST *rq, AQH_OBJECT *ep)
|
|
{
|
|
if (rq) {
|
|
if (ep)
|
|
AQH_Object_IncRefCount(ep);
|
|
if (rq->endpoint)
|
|
AQH_Object_free(ep);
|
|
rq->endpoint=ep;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
uint32_t AQH_MsgRequest_GetRequestMsgId(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->requestMsgId:0;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetRequestMsgId(AQH_MSG_REQUEST *rq, uint32_t id)
|
|
{
|
|
if (rq)
|
|
rq->requestMsgId=id;
|
|
}
|
|
|
|
|
|
|
|
AQH_MESSAGE_LIST *AQH_MsgRequest_GetMsgList(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->msgList:NULL;
|
|
}
|
|
|
|
|
|
|
|
AQH_MESSAGE *AQH_MsgRequest_GetFirstMsgFromList(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return (rq && rq->msgList)?AQH_Message_List_First(rq->msgList):NULL;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_AddMsgToList(AQH_MSG_REQUEST *rq, AQH_MESSAGE *msg)
|
|
{
|
|
if (rq && msg) {
|
|
if (rq->msgList==NULL)
|
|
rq->msgList=AQH_Message_List_new();
|
|
AQH_Message_List_Add(msg, rq->msgList);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
const GWEN_TIMESTAMP *AQH_MsgRequest_GetCreatedAt(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->createdAt:NULL;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetCreatedAt(AQH_MSG_REQUEST *rq, GWEN_TIMESTAMP *ts)
|
|
{
|
|
if (rq) {
|
|
GWEN_Timestamp_free(rq->createdAt);
|
|
rq->createdAt=ts;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
const GWEN_TIMESTAMP *AQH_MsgRequest_GetExpiresAt(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->expiresAt:NULL;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetExpiresAt(AQH_MSG_REQUEST *rq, GWEN_TIMESTAMP *ts)
|
|
{
|
|
if (rq) {
|
|
GWEN_Timestamp_free(rq->expiresAt);
|
|
rq->expiresAt=ts;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetTimestamps(AQH_MSG_REQUEST *rq, int expiresInSecs)
|
|
{
|
|
if (rq) {
|
|
GWEN_TIMESTAMP *ts;
|
|
|
|
ts=GWEN_Timestamp_NowInLocalTime();
|
|
GWEN_Timestamp_free(rq->createdAt);
|
|
rq->createdAt=GWEN_Timestamp_dup(ts);
|
|
GWEN_Timestamp_AddSeconds(ts, expiresInSecs);
|
|
GWEN_Timestamp_free(rq->expiresAt);
|
|
rq->expiresAt=ts;
|
|
}
|
|
}
|
|
|
|
|
|
|
|
int AQH_MsgRequest_HandleResponse(AQH_MSG_REQUEST *rq, const AQH_MESSAGE *msg)
|
|
{
|
|
if (rq && rq->handleResponseFn)
|
|
return (rq->handleResponseFn)(rq, msg);
|
|
return AQH_MSG_REQUEST_RESULT_NOT_HANDLED;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SubRequestFinished(AQH_MSG_REQUEST *rq, AQH_MSG_REQUEST *subRq, int reason)
|
|
{
|
|
if (rq && rq->subRequestFinishedFn)
|
|
rq->subRequestFinishedFn(rq, subRq, reason);
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_Abort(AQH_MSG_REQUEST *rq, int reason)
|
|
{
|
|
if (rq && rq->abortFn) {
|
|
rq->abortFn(rq, reason);
|
|
AQH_MsgRequest_SetState(rq, AQH_MSG_REQUEST_STATE_DONE);
|
|
}
|
|
else {
|
|
AQH_MSG_REQUEST *rqParent;
|
|
|
|
AQH_MsgRequest_SetState(rq, AQH_MSG_REQUEST_STATE_DONE);
|
|
rqParent=AQH_MsgRequest_Tree2_GetParent(rq);
|
|
if (rqParent)
|
|
AQH_MsgRequest_SubRequestFinished(rqParent, rq, AQH_MSG_REQUEST_REASON_ABORTED);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
AQH_MSG_REQUEST_HANDLERESPONSE_FN AQH_MsgRequest_SetHandleResponseFn(AQH_MSG_REQUEST *rq,
|
|
AQH_MSG_REQUEST_HANDLERESPONSE_FN f)
|
|
{
|
|
if (rq) {
|
|
AQH_MSG_REQUEST_HANDLERESPONSE_FN oldFn;
|
|
|
|
oldFn=rq->handleResponseFn;
|
|
rq->handleResponseFn=f;
|
|
return oldFn;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
AQH_MSG_REQUEST_SUBREQUESTFINISHED_FN AQH_MsgRequest_SetSubRequestFinishedFn(AQH_MSG_REQUEST *rq,
|
|
AQH_MSG_REQUEST_SUBREQUESTFINISHED_FN f)
|
|
{
|
|
if (rq) {
|
|
AQH_MSG_REQUEST_SUBREQUESTFINISHED_FN oldFn;
|
|
|
|
oldFn=rq->subRequestFinishedFn;
|
|
rq->subRequestFinishedFn=f;
|
|
return oldFn;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
AQH_MSG_REQUEST_ABORT_FN AQH_MsgRequest_SetAbortFn(AQH_MSG_REQUEST *rq, AQH_MSG_REQUEST_ABORT_FN f)
|
|
{
|
|
if (rq) {
|
|
AQH_MSG_REQUEST_ABORT_FN oldFn;
|
|
|
|
oldFn=rq->abortFn;
|
|
rq->abortFn=f;
|
|
return oldFn;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
void *AQH_MsgRequest_GetPrivateData(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->privateData:NULL;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetPrivateData(AQH_MSG_REQUEST *rq, void *p)
|
|
{
|
|
if (rq)
|
|
rq->privateData=p;
|
|
}
|
|
|
|
|
|
|
|
int AQH_MsgRequest_GetResult(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->result:0;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetResult(AQH_MSG_REQUEST *rq, int result)
|
|
{
|
|
if (rq)
|
|
rq->result=result;
|
|
}
|
|
|
|
|
|
|
|
int AQH_MsgRequest_GetState(const AQH_MSG_REQUEST *rq)
|
|
{
|
|
return rq?rq->state:0;
|
|
}
|
|
|
|
|
|
|
|
void AQH_MsgRequest_SetState(AQH_MSG_REQUEST *rq, int i)
|
|
{
|
|
if (rq)
|
|
rq->state=i;
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
AQH_MSG_REQUEST *AQH_MsgRequest_Tree2_FindByEndpointAndMsgId(AQH_MSG_REQUEST *rootRq, AQH_OBJECT *ep, uint32_t refMsgId)
|
|
{
|
|
if (rootRq) {
|
|
AQH_MSG_REQUEST *rq;
|
|
|
|
rq=AQH_MsgRequest_Tree2_GetFirstChild(rootRq);
|
|
while(rq) {
|
|
if (rq->endpoint==ep && rq->requestMsgId==refMsgId)
|
|
return rq;
|
|
rq=AQH_MsgRequest_Tree2_GetBelow(rq);
|
|
} /* while */
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|
|
void AQH_Request_Tree2_CheckTimeouts(AQH_MSG_REQUEST *rootRq)
|
|
{
|
|
if (rootRq) {
|
|
AQH_MSG_REQUEST *rq;
|
|
GWEN_TIMESTAMP *now;
|
|
|
|
now=GWEN_Timestamp_NowInLocalTime();
|
|
rq=AQH_MsgRequest_Tree2_GetFirstChild(rootRq);
|
|
while(rq) {
|
|
const GWEN_TIMESTAMP *ts;
|
|
|
|
ts=AQH_MsgRequest_GetExpiresAt(rq);
|
|
if (GWEN_Timestamp_Compare(now, ts)>=0) {
|
|
/* timeout */
|
|
DBG_INFO(AQH_LOGDOMAIN, "Request timed out, aborting");
|
|
AQH_MsgRequest_Abort(rq, AQH_MSG_REQUEST_REASON_TIMEOUT);
|
|
}
|
|
rq=AQH_MsgRequest_Tree2_GetBelow(rq);
|
|
}
|
|
GWEN_Timestamp_free(now);
|
|
}
|
|
}
|
|
|
|
|
|
|
|
void AQH_Request_Tree2_Cleanup(AQH_MSG_REQUEST *rootRq)
|
|
{
|
|
if (rootRq) {
|
|
AQH_MSG_REQUEST *rq;
|
|
|
|
rq=AQH_MsgRequest_Tree2_GetFirstChild(rootRq);
|
|
while(rq) {
|
|
AQH_MSG_REQUEST *nextSubRq;
|
|
|
|
nextSubRq=AQH_MsgRequest_Tree2_GetNext(rq);
|
|
_freeFinishedRequests(rq);
|
|
rq=nextSubRq;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
|
|
int AQH_Request_Tree2_HandleIpcMsg(AQH_MSG_REQUEST *rootRq, AQH_OBJECT *srcEp, AQH_MESSAGE *recvdMsg)
|
|
{
|
|
if (rootRq) {
|
|
uint32_t refMsgId;
|
|
|
|
refMsgId=AQH_IpcMessage_GetRefMsgId(recvdMsg);
|
|
if (refMsgId) {
|
|
AQH_MSG_REQUEST *rq;
|
|
|
|
rq=AQH_MsgRequest_Tree2_GetFirstChild(rootRq);
|
|
while(rq) {
|
|
if (srcEp==AQH_MsgRequest_GetEndpoint(rq) && refMsgId==AQH_MsgRequest_GetRequestMsgId(rq)) {
|
|
if (AQH_MsgRequest_HandleResponse(rq, recvdMsg)==AQH_MSG_REQUEST_RESULT_HANDLED)
|
|
return AQH_MSG_REQUEST_RESULT_HANDLED;
|
|
}
|
|
|
|
rq=AQH_MsgRequest_Tree2_GetBelow(rq);
|
|
}
|
|
}
|
|
else {
|
|
DBG_INFO(AQH_LOGDOMAIN, "Message has no reference msg id, not a response");
|
|
}
|
|
}
|
|
return AQH_MSG_REQUEST_RESULT_NOT_HANDLED;
|
|
}
|
|
|
|
|
|
|
|
int AQH_Request_Tree2_HandleTtyMsg(AQH_MSG_REQUEST *rootRq, AQH_OBJECT *srcEp, AQH_MESSAGE *recvdMsg)
|
|
{
|
|
if (rootRq) {
|
|
AQH_MSG_REQUEST *rq;
|
|
|
|
rq=AQH_MsgRequest_Tree2_GetFirstChild(rootRq);
|
|
while(rq) {
|
|
if (srcEp==AQH_MsgRequest_GetEndpoint(rq)) {
|
|
if (AQH_MsgRequest_HandleResponse(rq, recvdMsg)==AQH_MSG_REQUEST_RESULT_HANDLED)
|
|
return AQH_MSG_REQUEST_RESULT_HANDLED;
|
|
}
|
|
|
|
rq=AQH_MsgRequest_Tree2_GetBelow(rq);
|
|
}
|
|
}
|
|
return AQH_MSG_REQUEST_RESULT_NOT_HANDLED;
|
|
}
|
|
|
|
|
|
|
|
void _freeFinishedRequests(AQH_MSG_REQUEST *rq)
|
|
{
|
|
AQH_MSG_REQUEST *subRq;
|
|
|
|
subRq=AQH_MsgRequest_Tree2_GetFirstChild(rq);
|
|
while(subRq) {
|
|
AQH_MSG_REQUEST *nextSubRq;
|
|
|
|
nextSubRq=AQH_MsgRequest_Tree2_GetNext(subRq);
|
|
_freeFinishedRequests(subRq);
|
|
subRq=nextSubRq;
|
|
}
|
|
|
|
if (AQH_MsgRequest_GetState(rq)==AQH_MSG_REQUEST_STATE_DONE) {
|
|
DBG_INFO(AQH_LOGDOMAIN, "Deleting request");
|
|
AQH_MsgRequest_free(rq);
|
|
}
|
|
}
|
|
|
|
|