// basic JASON API to BPQ Node // Authentication is via Telnet USER records or bbs records #define _CRT_SECURE_NO_DEPRECATE #define WIN32_LEAN_AND_MEAN // Exclude rarely-used stuff from Windows headers //#include #include "CHeaders.h" #include #include "bpqmail.h" #include "httpconnectioninfo.h" struct MsgInfo * GetMsgFromNumber(int msgno); BOOL CheckUserMsg(struct MsgInfo * Msg, char * Call, BOOL SYSOP); char * doXMLTransparency(char * string); // Constants #define TOKEN_SIZE 32 // Length of the authentication token #define TOKEN_EXPIRATION 7200 // Token expiration time in seconds (2 hours) // Token data structure typedef struct MailToken { char token[TOKEN_SIZE + 1]; time_t expiration_time; struct UserInfo * User; char Call[10]; int Auth; // Security level of user struct MailToken* next; } MailToken; static MailToken * token_list = NULL; typedef struct MailAPI { char *URL; int URLLen; int (* APIRoutine)(struct HTTPConnectionInfo * Session, char * response, char * Rest, int Auth); int Auth; } MailAPI; // Auth defines #define AuthNone 0 #define AuthUser 1 #define AuthBBSUser 2 #define AuthSysop 4 static int verify_token(const char* token); static void remove_expired_tokens(); static int request_token(char * response); static void add_token_to_list(MailToken* token); static MailToken * find_token(const char* token); int sendMsgList(struct HTTPConnectionInfo * Session, char * response, char * Rest, int Auth); int sendFwdQueueLen(struct HTTPConnectionInfo * Session, char * response, char * Rest, int Auth); int sendFwdConfig(struct HTTPConnectionInfo * Session, char * response, char * Rest, int Auth); static struct MailAPI APIList[] = { "/mail/api/v1/msgs", 17, sendMsgList, 0, "/mail/api/v1/FwdQLen", 20, sendFwdQueueLen, AuthSysop, "/mail/api/v1/FwdConfig", 22, sendFwdConfig, AuthSysop, }; static int APICount = sizeof(APIList) / sizeof(struct MailAPI); #ifndef WIN32 iconv_t * icu = NULL; #endif void APIConvertTitletoUTF8(char * Title, char * UTF8Title, int Len) { if (WebIsUTF8(Title, (int)strlen(Title)) == FALSE) { // With Windows it is simple - convert using current codepage // I think the only reliable way is to convert to unicode and back int origlen = (int)strlen(Title) + 1; #ifdef WIN32 WCHAR BufferW[128]; int wlen; int len = origlen; wlen = MultiByteToWideChar(CP_ACP, 0, Title, len, BufferW, origlen * 2); len = WideCharToMultiByte(CP_UTF8, 0, BufferW, wlen, UTF8Title, origlen * 2, NULL, NULL); #else size_t left = Len - 1; size_t len = origlen; if (icu == NULL) icu = iconv_open("UTF-8//IGNORE", "CP1252"); if (icu == (iconv_t)-1) { strcpy(UTF8Title, Title); icu = NULL; return; } char * orig = UTF8Title; iconv(icu, NULL, NULL, NULL, NULL); // Reset State Machine iconv(icu, &Title, &len, (char ** __restrict__)&UTF8Title, &left); #endif } else strcpy(UTF8Title, Title); } static MailToken * generate_token() { // Generate a random authentication token int i; MailToken * token = malloc(sizeof(MailToken)); srand(time(NULL)); for (i = 0; i < TOKEN_SIZE; i++) { token->token[i] = 'A' + rand() % 26; // Random uppercase alphabet character } token->token[TOKEN_SIZE] = '\0'; // Null-terminate the token token->expiration_time = time(NULL) + TOKEN_EXPIRATION; // Set token expiration time add_token_to_list(token); return token; } // Function to add the token to the token_list static void add_token_to_list(MailToken * token) { if (token_list == NULL) { token_list = token; token->next = NULL; } else { MailToken * current = token_list; while (current->next != NULL) current = current->next; current->next = token; token->next = NULL; } } static void remove_expired_tokens() { time_t current_time = time(NULL); MailToken* current_token = token_list; MailToken* prev_token = NULL; MailToken* next_token; while (current_token != NULL) { if (current_time > current_token->expiration_time) { // Token has expired, remove it from the token list if (prev_token == NULL) { token_list = current_token->next; } else { prev_token->next = current_token->next; } next_token = current_token->next; free(current_token); current_token = next_token; } else { prev_token = current_token; current_token = current_token->next; } } } static MailToken * find_token(const char* token) { MailToken * current_token = token_list; while (current_token != NULL) { if (strcmp(current_token->token, token) == 0) { return current_token; } current_token = current_token->next; } return NULL; } static int send_http_response(char * response, const char* msg) { return sprintf(response, "HTTP/1.1 %s\r\nContent-Length: 0\r\nConnection: close\r\n\r\n", msg); } int MailAPIProcessHTTPMessage(struct HTTPConnectionInfo * Session, char * response, char * Method, char * URL, char * request, BOOL LOCAL, char *Params, char * TokenString) { char * pass = strlop(Params, '&'); int Flags = 0, n; MailToken * Token; char Msg[64]; struct UserInfo * User; int Auth = 0; if (LOCAL) Auth = AuthSysop; // Check if the request is for token generation if (strcmp(Method, "GET") != 0) return send_http_response(response, "403 (Bad Method)"); if (_stricmp(URL, "/mail/api/v1/login") == 0) { // Key is in Session->Key // Signon may have been validated in Node. If Session->Callsign is set if (Session->Callsign[0] == 0) { // Try BBS logon User = LookupCall(Params); if (User) { // Check Password if (pass[0] == 0 || strcmp(User->pass, pass) != 0 || User->flags & F_Excluded) return send_http_response(response, "403 (Login Failed)"); strcpy(Session->Callsign, User->Call); Auth = AuthBBSUser; if (User->flags & F_SYSOP) Auth |= AuthSysop; } } else { User = LookupCall(Session->Callsign); if (User) { Auth = AuthUser; if (User->flags & F_SYSOP) Auth |= AuthSysop; } } n = sprintf_s(Msg, sizeof(Msg), "API Connect from %s", _strupr(Params)); WriteLogLine(NULL, '|',Msg, n, LOG_BBS); Token = zalloc(sizeof(MailToken)); strcpy(Token->token, Session->Key); strcpy(Token->Call, Session->Callsign); Token->Auth = Auth; Token->expiration_time = time(NULL) + TOKEN_EXPIRATION; // Set token expiration time add_token_to_list(Token); // Return Token sprintf(response, "{\"access_token\":\"%s\", \"expires_at\":%d, \"scope\":\"create\"}\r\n", Token->token, Token->expiration_time); return strlen(response); } // Find Token if (TokenString[0]) // Token form Auth Header Token = find_token(TokenString); else Token = find_token(Params); // Token form URL if (Token != NULL) { // Check if the token has expired time_t current_time = time(NULL); if (current_time > Token->expiration_time) { // Token has expired, remove it from the token list remove_expired_tokens(); Token = NULL; } } if (Token) Auth |= Token->Auth; // Determine the requested API endpoint for (n = 0; n < APICount; n++) { struct MailAPI * APIEntry; char * rest; APIEntry = &APIList[n]; if (_memicmp(URL, APIEntry->URL, APIEntry->URLLen) == 0) { rest = &request[4 + APIEntry->URLLen]; // Anything following? if (rest[0] =='?') { //Key strlop(rest, ' '); strlop(rest, '&'); Token = find_token(&rest[1]); if (Token) { strcpy(Session->Callsign, Token->Call); strcpy(Session->Key, Token->token); } else return send_http_response(response, "403 (Invalid Security Token)"); } if (APIEntry->Auth) { // Check Level if ((Auth & APIEntry->Auth) == 0) return send_http_response(response, "403 (Not Authorized)"); } if (rest[0] == ' ' || rest[0] == '/' || rest[0] == '?') { return APIEntry->APIRoutine(Session, response, rest, Auth); } } } return send_http_response(response, "401 Invalid API Call"); return 0; } int WebMailAPIProcessHTTPMessage(char * response, char * Method, char * URL, char * request, BOOL LOCAL, char *Params) { char * pass = strlop(Params, '&'); int Flags = 0; MailToken * Token; // Check if the request is for token generation if (strcmp(Method, "GET") != 0) return send_http_response(response, "403 (Bad Method)"); if (_stricmp(URL, "/mail/api/login") == 0) { // user is in Params and Password in pass struct UserInfo * User; char Msg[256]; int n; User = LookupCall(Params); if (User) { // Check Password if (pass[0] == 0 || strcmp(User->pass, pass) != 0 || User->flags & F_Excluded) return send_http_response(response, "403 (Login Failed)"); n=sprintf_s(Msg, sizeof(Msg), "API Connect from %s", _strupr(Params)); WriteLogLine(NULL, '|',Msg, n, LOG_BBS); Token = generate_token(); add_token_to_list(Token); Token->User = User; strcpy(Token->Call, Params); // Return Token sprintf(response, "{\"access_token\":\"%s\", \"expires_in\":%d, \"scope\":\"create\"}\r\n", Token->token, Token->expiration_time); return strlen(response); } } return 0; } // Unauthenticated users can only get bulls. // Authenticated users may read only that users messages or all messages depending on sysop status int sendMsgList(struct HTTPConnectionInfo * Session, char * response, char * Rest, int Auth) { struct UserInfo * User = LookupCall(Session->Callsign); int m; struct MsgInfo * Msg; char * ptr = response; int n = NumberofMessages; //LineCount; char Via[64]; int Count = 0; struct UserInfo DummyUser = {""}; ptr[0] = 0; if (User == 0) User=&DummyUser; n = sprintf(ptr,"{\"msgs\":[\r\n"); ptr += n; for (m = LatestMsg; m >= 1; m--) { if (ptr > &response[244000]) break; // protect buffer Msg = GetMsgFromNumber(m); if (Msg == 0 || Msg->type == 0 || Msg->status == 0) continue; // Protect against corrupt messages if (Msg && CheckUserMsg(Msg, User->Call, Auth & AuthSysop)) { char UTF8Title[4096]; char * EncodedTitle; // List if it is the right type ptr += sprintf(ptr, "{\r\n"); strcpy(Via, Msg->via); strlop(Via, '.'); // make sure title is HTML safe (no < > etc) and UTF 8 encoded EncodedTitle = doXMLTransparency(Msg->title); memset(UTF8Title, 0, 4096); // In case convert fails part way through APIConvertTitletoUTF8(EncodedTitle, UTF8Title, 4095); ptr += sprintf(ptr, "\"id\": \"%d\",\r\n", Msg->number); ptr += sprintf(ptr, "\"mid\": \"%s\",\r\n", Msg->bid); ptr += sprintf(ptr, "\"rcvd\": \"%d\",\r\n", Msg->datecreated); ptr += sprintf(ptr, "\"type\": \"%c\",\r\n", Msg->type); ptr += sprintf(ptr, "\"status\": \"%c\",\r\n", Msg->status); ptr += sprintf(ptr, "\"to\": \"%s\",\r\n", Msg->to); ptr += sprintf(ptr, "\"from\": \"%s\",\r\n", Msg->from); ptr += sprintf(ptr, "\"size\": \"%d\",\r\n", Msg->length); ptr += sprintf(ptr, "\"subject\": \"%s\"\r\n", UTF8Title); free(EncodedTitle); ptr += sprintf(ptr, "},\r\n"); } } if (response[n] == 0) // No entries { response[strlen(response) - 2] = '\0'; // remove \r\n strcat(response, "]}\r\n"); } else { response[strlen(response)-3 ] = '\0'; // remove ,\r\n strcat(response, "\r\n]}\r\n"); } return strlen(response); } int sendFwdQueueLen(struct HTTPConnectionInfo * Session, char * response, char * Rest, int Auth) { struct UserInfo * USER; char * ptr = response; int n; int i = 0; int Len = 0; n = sprintf(ptr,"{\"forwardqueuelength\":[\r\n"); ptr += n; for (USER = BBSChain; USER; USER = USER->BBSNext) { int Count = CountMessagestoForward (USER); ptr += sprintf(ptr, "{"); ptr += sprintf(ptr, "\"call\": \"%s\",", USER->Call); ptr += sprintf(ptr, "\"queueLength\": \"%d\"", Count); ptr += sprintf(ptr, "},\r\n"); } if (response[n] == 0) // No entries { response[strlen(response) - 2] = '\0'; // remove \r\n strcat(response, "]}\r\n"); } else { response[strlen(response)-3 ] = '\0'; // remove ,\r\n strcat(response, "\r\n]}\r\n"); } return strlen(response); } VOID APIMultiStringValue(char ** values, char * Multi) { char ** Calls; char * ptr = &Multi[0]; *ptr = 0; if (values) { Calls = values; while(Calls[0]) { ptr += sprintf(ptr, "\"%s\",", Calls[0]); Calls++; } if (ptr != &Multi[0]) *(--ptr) = 0; } } char * APIConvTime(int ss) { int hh, mm; static char timebuf[64]; hh = ss / 3600; mm = (ss - (hh * 3600)) / 60; ss = ss % 60; sprintf(timebuf, "\"%02d:%02d:%02d\"", hh, mm, ss); return timebuf; } int sendFwdConfig(struct HTTPConnectionInfo * Session, char * response, char * Rest, int Auth) { struct UserInfo * USER; char * ptr = response; int n = 0; int i = 0; int Len = 0; response[n] = 0; n = sprintf(ptr, "{\r\n"); ptr += n; for (USER = BBSChain; USER; USER = USER->BBSNext) { struct BBSForwardingInfo * FWDInfo = USER->ForwardingInfo; int Count = CountMessagestoForward (USER); char TO[2048] = ""; char AT[2048] = ""; char TIMES[2048] = ""; char FWD[100000] = ""; char HRB[2048] = ""; char HRP[2048] = ""; APIMultiStringValue(FWDInfo->TOCalls, TO); APIMultiStringValue(FWDInfo->ATCalls, AT); APIMultiStringValue(FWDInfo->FWDTimes, TIMES); APIMultiStringValue(FWDInfo->ConnectScript, FWD); APIMultiStringValue(FWDInfo->Haddresses, HRB); APIMultiStringValue(FWDInfo->HaddressesP, HRP); ptr += sprintf(ptr, " \"%s\": {\r\n", USER->Call); ptr += sprintf(ptr, " \"queueLength\": %d,\r\n", Count); ptr += sprintf(ptr, " \"to\": [%s],\r\n", TO); ptr += sprintf(ptr, " \"at\": [%s],\r\n", AT); ptr += sprintf(ptr, " \"hrp\": [%s],\r\n",HRP); ptr += sprintf(ptr, " \"hrb\": [%s],\r\n",HRB); ptr += sprintf(ptr, " \"times\": [%s],\r\n",TIMES); ptr += sprintf(ptr, " \"connectScript\": [%s],\r\n",FWD); ptr += sprintf(ptr, " \"bbsHa\": \"%s\",\r\n", (FWDInfo->BBSHA)?FWDInfo->BBSHA:""); ptr += sprintf(ptr, " \"enableForwarding\": %s,\r\n", (FWDInfo->Enabled)?"true":"false"); ptr += sprintf(ptr, " \"forwardingInterval\": %s,\r\n", APIConvTime(FWDInfo->FwdInterval)); ptr += sprintf(ptr, " \"requestReverse\": %s,\r\n", (FWDInfo->ReverseFlag)?"true":"false"); ptr += sprintf(ptr, " \"reverseInterval\": %s,\r\n", APIConvTime(FWDInfo->RevFwdInterval)); ptr += sprintf(ptr, " \"sendNewMessagesWithoutWaiting\": %s,\r\n", (FWDInfo->SendNew)?"true":"false"); ptr += sprintf(ptr, " \"fbbBlocked\": %s,\r\n", (FWDInfo->AllowBlocked)?"true":"false"); ptr += sprintf(ptr, " \"maxBlock\": %d,\r\n", FWDInfo->MaxFBBBlockSize); ptr += sprintf(ptr, " \"sendPersonalMailOnly\": %s,\r\n", (FWDInfo->PersonalOnly)?"true":"false"); ptr += sprintf(ptr, " \"allowBinary\": %s,\r\n", (FWDInfo->AllowCompressed)?"true":"false"); ptr += sprintf(ptr, " \"useB1Protocol\": %s,\r\n", (FWDInfo->AllowB1)?"true":"false"); ptr += sprintf(ptr, " \"useB2Protocol\": %s,\r\n", (FWDInfo->AllowB2)?"true":"false"); ptr += sprintf(ptr, " \"incomingConnectTimeout\": %s\r\n", APIConvTime(FWDInfo->ConTimeout)); ptr += sprintf(ptr, " },\r\n"); } if (response[n] == 0) // No entries { strcpy(response, "{}\r\n"); } else { response[strlen(response)-3 ] = '\0'; // remove ,\r\n strcat(response, "\r\n}\r\n"); } return strlen(response); } /* { "GB7BEX": { "queueLength": 0, "forwardingConfig": { "to": [], "at": [ "OARC", "GBR", "WW" ], "times": [], "connectScript": [ "PAUSE 3", "INTERLOCK 3", "NC 3 !GB7BEX" ], "hierarchicalRoutes": [], "hr": [ "#38.GBR.EURO" ], "bbsHa": "GB7BEX.#38.GBR.EURO", "enableForwarding": true, "forwardingInterval": "00:56:40", "requestReverse": false, "reverseInterval": "00:56:40", "sendNewMessagesWithoutWaiting": true, "fbbBlocked": true, "maxBlock": 1000, "sendPersonalMailOnly": false, "allowBinary": true, "useB1Protocol": false, "useB2Protocol": true, "sendCtrlZInsteadOfEx": false, "incomingConnectTimeout": "00:02:00" } }, "GB7RDG": { "queueLength": 0, "forwardingConfig": { "to": [], ... "incomingConnectTimeout": "00:02:00" } } } # HELP packetmail_queue_length The number of messages in the packetmail queue # TYPE packetmail_queue_length gauge packetmail_queue_length{partner="DM4RW"} 0 1729090716916 packetmail_queue_length{partner="G8BPQ"} 3 1729090716916 packetmail_queue_length{partner="GB7BEX"} 0 1729090716916 packetmail_queue_length{partner="GB7BPQ"} 1 1729090716916 packetmail_queue_length{partner="GB7MNS"} 0 1729090716916 packetmail_queue_length{partner="GB7NOT"} 0 1729090716916 packetmail_queue_length{partner="GB7NWL"} 0 1729090716916 packetmail_queue_length{partner="GM8BPQ"} 0 1729090716916 */ // Stuff send to packetnodes.spots.radio/api/bbsdata/{bbsCall} //https://nodes.ukpacketradio.network/swagger/index.html /* BbsData{ callsign* [...] time* [...] hroute* [...] peers [...] software* [...] version* [...] mailQueues [...] messages [...] latitude [...] longitude [...] locator [...] location [...] unroutable [...] } [ { "callsign": "GE8PZT", "time": "2024-11-25T10:07:41+00:00", "hroute": ".#24.GBR.EU", "peers": [ "GB7BBS", "VE2PKT", "GB7NXT", "VA2OM" ], "software": "XrLin", "version": "504a", "mailQueues": [], "messages": [ { "to": "TECH@WW", "mid": "20539_GB7CIP", "rcvd": "2024-11-24T09:27:59+00:00", "routing": [ "R:241124/0927Z @:GE8PZT.#24.GBR.EU [Lamanva] #:2315 XrLin504a", { "to": "TNC@WW", "mid": "37_PA2SNK", "rcvd": "2024-11-18T21:56:55+00:00", "routing": [ "R:241118/2156Z @:GE8PZT.#24.GBR.EU [] #:2215 XrLin504a", "R:241118/2156Z 12456@VE2PKT.#TRV.QC.CAN.NOAM BPQ6.0.24", "R:241118/2130Z 51539@VE3KPG.#ECON.ON.CAN.NOAM BPQK6.0.23", "R:241118/2130Z 26087@VE3CGR.#SCON.ON.CAN.NOAM LinBPQ6.0.24", "R:241118/2130Z 37521@PA8F.#ZH1.NLD.EURO LinBPQ6.0.24", "R:241118/2129Z 48377@PI8LAP.#ZLD.NLD.EURO LinBPQ6.0.24", "R:241118/2129Z @:PD0LPM.FRL.EURO.NLD #:33044 [Joure] $:37_PA2SNK" ] } ], "latitude": 50.145832, "longitude": -5.125, "locator": "IO70KD", "location": "Lamanva", "unroutable": [ { "type": "P", "at": "WW" }, { "type": "P", "at": "G8PZT-2" }, { "type": "P", "at": "g8pzt._24.gbr.eu" }, { "type": "P", "at": "G8PZT.#24.GBR.EU" }, { "type": "P", "at": "GE8PZT.#24.GBR.EU" }, { "type": "P", "at": "G8PZT.#24.GBR.EURO" } ] }, */ // https://packetnodes.spots.radio/swagger/index.html // "unroutable": [{"type": "P","at": "WW"}, {"type": "P", "at": "G8PZT.#24.GBR.EURO"}] char * ViaList[100000]; // Pointers to the Message Header field char TypeList[100000]; int unroutableCount = 0; void CheckifRoutable(struct MsgInfo * Msg) { char NextBBS[64]; int n; if (Msg->status == 'K') return; if (Msg->via[0] == 0) // No routing return; strcpy(NextBBS, Msg->via); strlop(NextBBS, '.'); if (strcmp(NextBBS, BBSName) == 0) // via this BBS return; if ((memcmp(Msg->fbbs, zeros, NBMASK) != 0) || (memcmp(Msg->forw, zeros, NBMASK) != 0)) // Has Forwarding Info return; // See if we already have it for (n = 0; n < unroutableCount; n++) { if ((TypeList[n] == Msg->type) && strcmp(ViaList[n], Msg->via) == 0) return; } // Add to list TypeList[unroutableCount] = Msg->type; ViaList[unroutableCount] = Msg->via; unroutableCount++; } extern char LOC[7]; DllExport VOID WINAPI SendWebRequest(char * Host, char * Request, char * Params, char * Return); #ifdef LINBPQ extern double LatFromLOC; extern double LonFromLOC; #else typedef int (WINAPI FAR *FARPROCX)(); extern FARPROCX pSendWebRequest; extern FARPROCX pGetLatLon; double LatFromLOC = 0; double LonFromLOC = 0; #endif void SendBBSDataToPktMap() { char Request[64]; char * Params; char * ptr; int paramLen; struct MsgInfo * Msg; struct UserInfo * ourBBSRec = LookupCall(BBSName); struct UserInfo * USER; char Time[64]; struct tm * tm; time_t Date = time(NULL); char Peers[2048] = "[]"; char MsgQueues[16000] = "[]"; char * Messages = malloc(1000000); char * Unroutables; int m; char * MsgBytes; char * Rlineptr; char * Rlineend; char * RLines; char * ptr1, * ptr2; int n; #ifndef LINBPQ if (pSendWebRequest == 0) return; // Old Version of bpq32.dll pGetLatLon(&LatFromLOC, &LonFromLOC); #endif if (ourBBSRec == 0) return; // Wot!! // Get peers and Mail Queues ptr = &Peers[1]; ptr1 = &MsgQueues[1]; for (USER = BBSChain; USER; USER = USER->BBSNext) { if (strcmp(USER->Call, BBSName) != 0) { int Bytes; int Count = CountMessagestoForward(USER); ptr += sprintf(ptr, "\"%s\",", USER->Call); if (Count) { Bytes = CountBytestoForward(USER); ptr1 += sprintf(ptr1, "{\"peerCall\": \"%s\", \"numQueued\": %d, \"bytesQueued\": %d},", USER->Call, Count, Bytes); } } } if ((*ptr) != ']') // Have some entries { ptr--; // over trailing comms *(ptr++) = ']'; *(ptr) = 0; } if ((*ptr1) != ']') // Have some entries { ptr1--; // over trailing comms *(ptr1++) = ']'; *(ptr1) = 0; } // Get Messages strcpy(Messages, "[]"); ptr = &Messages[1]; for (m = LatestMsg; m >= 1; m--) { if (ptr > &Messages[999000]) break; // protect buffer Msg = GetMsgFromNumber(m); if (Msg == 0 || Msg->type == 0 || Msg->status == 0) continue; // Protect against corrupt messages // Paula suggests including H and K but limit it to the last 30 days or the last 100 messages, whichever is the smaller. // if (Msg->status == 'K' || Msg->status == 'H') // continue; if ((Date - Msg->datereceived) > 30 * 86400) // Too old continue; CheckifRoutable(Msg); tm = gmtime((time_t *)&Msg->datereceived); sprintf(Time, "%04d-%02d-%02dT%02d:%02d:%02d+00:00", tm->tm_year + 1900, tm->tm_mon+1, tm->tm_mday, tm->tm_hour, tm->tm_min, tm->tm_sec); // Get Routing MsgBytes = ReadMessageFile(Msg->number); RLines = malloc(Msg->length * 2); // Very unlikely to need so much but better safe.. strcpy(RLines, "[]"); ptr2 = &RLines[1]; // Need to skip B2 header if B2 Message Rlineptr = MsgBytes; // If it is a B2 Message, Must Skip B2 Header if (Msg->B2Flags & B2Msg) { Rlineptr = strstr(Rlineptr, "\r\n\r\n"); if (Rlineptr) Rlineptr += 4; else Rlineptr = MsgBytes; } // We have to process R: lines one at a time as we need to send each one as a separate string while (memcmp(Rlineptr, "R:", 2) == 0) { // Have R Lines Rlineend = strstr(Rlineptr, "\r\n"); Rlineend[0] = 0; ptr2 += sprintf(ptr2, "\"%s\",", Rlineptr); Rlineptr = Rlineend + 2; // over crlf } if ((*ptr2) == ']') // no entries continue; ptr2--; // over trailing comms *(ptr2++) = ']'; *(ptr2) = 0; ptr += sprintf(ptr, "{\"to\": \"%s\", \"mid\": \"%s\", \"rcvd\": \"%s\", \"routing\": %s},", Msg->to, Msg->bid, Time, RLines); free(MsgBytes); free(RLines); } if ((*ptr) != ']') // Have some entries? { ptr--; // over trailing comms *(ptr++) = ']'; *(ptr) = 0; } // Get unroutables Unroutables = malloc((unroutableCount + 1) * 100); strcpy(Unroutables, "[]"); ptr = &Unroutables[1]; for (n = 0; n < unroutableCount; n++) { ptr += sprintf(ptr, "{\"type\": \"%c\",\"at\": \"%s\"},", TypeList[n], ViaList[n]); } if ((*ptr) != ']') // Have some entries? { ptr--; // over trailing comms *(ptr++) = ']'; *(ptr) = 0; } /* char * ViaList[100000]; // Pointers to the Message Header field char TypeList[100000]; int unroutableCount = 0; "unroutable": [{"type": "P","at": "WW"}, {"type": "P", "at": "G8PZT.#24.GBR.EURO"}] */ tm = gmtime(&Date); sprintf(Time, "%04d-%02d-%02dT%02d:%02d:%02d+00:00", tm->tm_year + 1900, tm->tm_mon+1, tm->tm_mday, tm->tm_hour, tm->tm_min, tm->tm_sec); paramLen = strlen(Peers) + strlen(MsgQueues) + strlen(Messages) + strlen(Unroutables); Params = malloc(paramLen + 1000); if (Params == 0) { free(Messages); free(Unroutables); return; } ptr = Params; sprintf(Request, "/api/bbsdata/%s", BBSName); ptr += sprintf(ptr, "{\"callsign\": \"%s\",\r\n", BBSName); ptr += sprintf(ptr, "\"time\": \"%s\",\r\n", Time); ptr += sprintf(ptr, "\"hroute\": \"%s\",\r\n", HRoute); ptr += sprintf(ptr, "\"peers\": %s,\r\n", Peers); #ifdef LINBPQ ptr += sprintf(ptr, "\"software\": \"%s\",\r\n", "linbpq"); #else ptr += sprintf(ptr, "\"software\": \"%s\",\r\n", "BPQMail"); #endif ptr += sprintf(ptr, "\"version\": \"%s\",\r\n", VersionString); ptr += sprintf(ptr, "\"mailQueues\": %s,\r\n", MsgQueues); ptr += sprintf(ptr, "\"messages\": %s,\r\n", Messages); ptr += sprintf(ptr, "\"latitude\": %1.6f,\r\n", LatFromLOC); ptr += sprintf(ptr, "\"longitude\": %.6f,\r\n", LonFromLOC); ptr += sprintf(ptr, "\"locator\": \"%s\",\r\n", LOC); ptr += sprintf(ptr, "\"location\": \"%s\",\r\n", ourBBSRec->Address); ptr += sprintf(ptr, "\"unroutable\": %s\r\n}\r\n", Unroutables); #ifdef LINBPQ SendWebRequest("packetnodes.spots.radio", Request, Params, 0); #else pSendWebRequest("packetnodes.spots.radio", Request, Params, 0); #endif free(Messages); free(Unroutables); free(Params); }