You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1132 lines
30 KiB
1132 lines
30 KiB
#include <string.h>
|
|
|
|
#include <fstream>
|
|
#include <iostream>
|
|
#include <sstream>
|
|
#include <iomanip>
|
|
#include <vector>
|
|
#include <map>
|
|
|
|
#include <time.h>
|
|
|
|
#include <netdb.h>
|
|
#include <sys/socket.h>
|
|
//#include <arpa/inet.h>
|
|
#include <unistd.h>
|
|
|
|
//#include "md5.h"
|
|
#include <openssl/md5.h>
|
|
|
|
#include "connect.h"
|
|
//#include "base64.h"
|
|
|
|
#define BASENAME "FCM"
|
|
|
|
#include <curl/curl.h>
|
|
|
|
|
|
class CurledClass
|
|
{
|
|
public:
|
|
|
|
static int writer(char *data, size_t size, size_t nmemb, string *buffer_in)
|
|
{
|
|
|
|
// Is there anything in the buffer?
|
|
if (buffer_in != NULL)
|
|
{
|
|
// Append the data to the buffer
|
|
buffer_in->append(data, size * nmemb);
|
|
|
|
// How much did we write?
|
|
return size * nmemb;
|
|
}
|
|
return 0;
|
|
}
|
|
};
|
|
|
|
CConnect* CConnect::getInstance()
|
|
{
|
|
static CConnect* instance = NULL;
|
|
if(!instance)
|
|
instance = new CConnect();
|
|
return instance;
|
|
}
|
|
|
|
CConnect::CConnect()
|
|
{
|
|
//sockfb=0;
|
|
}
|
|
|
|
CConnect::~CConnect()
|
|
{
|
|
//
|
|
}
|
|
|
|
#if 0
|
|
/******************************************************************************
|
|
* functions
|
|
******************************************************************************/
|
|
#endif
|
|
int CConnect::connect2Host(const char *adr, int port)
|
|
{
|
|
//IPv4 or IPv6
|
|
int sockfd, rv;
|
|
struct addrinfo hints, *servinfo, *p;
|
|
|
|
ostringstream s_port;
|
|
s_port << port;
|
|
|
|
memset(&hints, 0, sizeof hints);
|
|
hints.ai_family = AF_UNSPEC; // use AF_INET6 to force IPv6
|
|
hints.ai_socktype = SOCK_STREAM;
|
|
|
|
if ((rv = getaddrinfo(adr, s_port.str().c_str(), &hints, &servinfo)) != 0) {
|
|
fprintf(stderr, "[%s] - getaddrinfo: %s\n", BASENAME, gai_strerror(rv));
|
|
return(-1);
|
|
}
|
|
|
|
//loop through all the results and connect to the first we can
|
|
for(p = servinfo; p != NULL; p = p->ai_next)
|
|
{
|
|
if (debug)
|
|
printf("[%s] - Try to connect to %s on port %i\n", BASENAME, adr, port);
|
|
|
|
if ((sockfd = socket(p->ai_family, p->ai_socktype,p->ai_protocol)) == -1) {
|
|
perror("socket");
|
|
continue;
|
|
}
|
|
|
|
if (connect(sockfd, p->ai_addr, p->ai_addrlen) == -1) {
|
|
close(sockfd);
|
|
perror("connect");
|
|
continue;
|
|
}
|
|
|
|
break; //if we get here, we must have connected successfully
|
|
}
|
|
|
|
if (p == NULL) {
|
|
//looped off the end of the list with no connection
|
|
fprintf(stderr, "[%s] - failed to connect\n", BASENAME);
|
|
return(-1);
|
|
}
|
|
|
|
freeaddrinfo(servinfo); //all done with this structure
|
|
return(sockfd);
|
|
}
|
|
|
|
string CConnect::post2fritz(const char* url, int port, const string data, const string curlOutFile)
|
|
{
|
|
CURL *curl;
|
|
CURLcode result;
|
|
|
|
// Create our curl handle
|
|
curl = curl_easy_init();
|
|
|
|
//errors
|
|
char errorBuffer[CURL_ERROR_SIZE];
|
|
|
|
// Write all expected data in here
|
|
string buffer;
|
|
|
|
if(!curl){
|
|
cerr << "Error init Curl!" << endl;
|
|
return string();
|
|
}
|
|
|
|
curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorBuffer);
|
|
curl_easy_setopt(curl, CURLOPT_URL, url);
|
|
curl_easy_setopt(curl, CURLOPT_HEADER, 0);
|
|
curl_easy_setopt(curl, CURLOPT_USERAGENT, "Mozilla/5.0");
|
|
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION, 1);
|
|
curl_easy_setopt(curl, CURLOPT_VERBOSE, debug?1:0);
|
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, 0);
|
|
curl_easy_setopt(curl, CURLOPT_PORT, port);
|
|
|
|
if(!data.empty())
|
|
{
|
|
curl_easy_setopt(curl, CURLOPT_POST, 1);
|
|
curl_easy_setopt(curl, CURLOPT_POSTFIELDS, data.c_str());
|
|
// print CURLOPT_POSTFIELDS
|
|
if(debug > 1 && !data.empty()) {cout << '[' << BASENAME << "] - CURLOPT_POSTFIELDS: " << data << endl;}
|
|
}
|
|
|
|
FILE *tmpFile = fopen(curlOutFile.c_str(), "w");
|
|
if (tmpFile) {
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, NULL);
|
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, tmpFile);
|
|
} else {
|
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, CurledClass::writer);
|
|
curl_easy_setopt(curl, CURLOPT_WRITEDATA, &buffer);
|
|
}
|
|
|
|
// Attempt to retrieve the remote page
|
|
result = curl_easy_perform(curl);
|
|
|
|
// Always cleanup
|
|
curl_easy_cleanup(curl);
|
|
if (tmpFile)
|
|
fclose(tmpFile);
|
|
|
|
if(debug > 1){
|
|
if(curlOutFile.empty())
|
|
cout << "DEBUG-OUT" << ">>" << buffer << "<<" << "DEBUG-OUT END" << endl;
|
|
else
|
|
cout << "DEBUG-OUT" << ">>" << curlOutFile << "<<" << "DEBUG-OUT END" << endl;
|
|
}
|
|
|
|
if(result == CURLE_OK)
|
|
return(buffer);
|
|
|
|
return string();
|
|
}
|
|
|
|
int CConnect::get2box(const char* host, int port, const char* msg, const char* upwd, const char* msgtype, int msgtimeout)
|
|
{
|
|
ostringstream url;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
url << host << "/control/message?" << msgtype << "=" << msg << "&timeout=" << msgtimeout;
|
|
|
|
string s = post2fritz(url.str().c_str(), port);
|
|
|
|
return(1);
|
|
}
|
|
|
|
int CConnect::get_challenge()
|
|
{
|
|
ostringstream url;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
//if(mysockfb <= 0){
|
|
// printf("[%s] - CConnect::get_challenge(): ERR connect2Host\n", BASENAME);
|
|
// return 0;
|
|
//}
|
|
|
|
//url << "/cgi-bin/webcm?getpage=../html/login_sid.xml"; // Firmwareversion xx.04.74
|
|
url << FritzAdr << "/login_sid.lua"; // OS 5.50
|
|
|
|
string s = post2fritz(url.str().c_str());
|
|
|
|
//search
|
|
string s1 = "<Challenge>";
|
|
string s2 = "</Challenge>";
|
|
string token;
|
|
size_t pos = 0;
|
|
|
|
if((pos = s.find(s1)) != string::npos)
|
|
{
|
|
s.erase(0, pos + s1.length());
|
|
|
|
pos = 0;
|
|
if((pos = s.find(s2)) != string::npos)
|
|
{
|
|
s.erase(pos, s.length());
|
|
if(!s.empty())
|
|
{
|
|
strncpy(challenge,s.c_str(),sizeof(challenge));
|
|
if(debug) {printf("[%s] - CHALLENGE %s \n", BASENAME, challenge);}
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
printf("[%s] - failed to get CHALLENGE\n", BASENAME);
|
|
return 0;
|
|
}
|
|
|
|
int CConnect::get_md5(const char *challenge, char *fritzPW)
|
|
{
|
|
//convert to utf16
|
|
//http://www.avm.de/de/Extern/Technical_Note_Session_ID.pdf
|
|
unsigned int i;
|
|
int y = 0;
|
|
|
|
string utf8_str = (string) challenge + "-" + (string) UTF8toISO(fritzPW);
|
|
string utf16_str = ("");
|
|
|
|
if(debug > 1)
|
|
printf("[%s] - Challenge-Password = \"%s-%s\"\n",BASENAME, challenge, UTF8toISO(fritzPW));
|
|
|
|
if(debug)
|
|
printf("[%s] - Binary for hash = ", BASENAME);
|
|
|
|
for (i=0; i < utf8_str.length(); i++)
|
|
{
|
|
y = i*2;
|
|
// change UNICODE > 255 in 46 (".")
|
|
if (utf8_str[i] > 0xAD)
|
|
{
|
|
utf16_str += '\x2E';
|
|
utf16_str += '\x00';
|
|
}
|
|
else
|
|
{
|
|
utf16_str += utf8_str[i];
|
|
utf16_str += '\x00';
|
|
}
|
|
if (debug)
|
|
printf("%02x%02x",utf16_str[y],utf16_str[y+1]);
|
|
}
|
|
if (debug)
|
|
printf("\n");
|
|
|
|
//get md5sum from utf16 binary hash
|
|
MD5_CTX ctx;
|
|
//struct MD5Context ctx; /*md5.h"*/
|
|
|
|
MD5_Init(&ctx);
|
|
MD5_Update(&ctx, utf16_str.c_str(), strlen(utf8_str.c_str())*2)/*size*/;
|
|
MD5_Final(digest, &ctx);
|
|
|
|
strcpy((char *)md5sum,"");
|
|
|
|
for (i = 0; i < 16; i++) {
|
|
char fdigest[5];
|
|
sprintf(fdigest,"%02x", digest[i]);
|
|
strcat((char *)md5sum, fdigest);
|
|
}
|
|
|
|
if(debug)
|
|
cout << '[' << BASENAME << "] - MD5 hash = " << md5sum << " (" << strlen((char *)md5sum) << ')' << endl;
|
|
|
|
return(strlen((char *)md5sum));
|
|
}
|
|
|
|
int CConnect::get_sid(const char *challenge, const unsigned char *md5)
|
|
{
|
|
ostringstream command;
|
|
|
|
command << "login:command/response="
|
|
<< challenge << '-' << md5
|
|
<< "&getpage=../html/login_sid.xml";
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
string s = post2fritz(FritzAdr,80, command.str());
|
|
if(debug) {cout << s << endl;}
|
|
|
|
//search
|
|
string s1 = "<SID>";
|
|
string s2 = "</SID>";
|
|
string token;
|
|
size_t pos = 0;
|
|
|
|
if((pos = s.find(s1)) != string::npos)
|
|
{
|
|
s.erase(0, pos + s1.length());
|
|
|
|
pos = 0;
|
|
if((pos = s.find(s2)) != string::npos)
|
|
{
|
|
s.erase(pos, s.length());
|
|
if(s != "0000000000000000" && !s.empty())
|
|
{
|
|
strncpy(sid,s.c_str(),sizeof(sid));
|
|
if(debug) {printf("[%s] - SID %s \n", BASENAME, sid);}
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
printf("[%s] - failed to get SID\n", BASENAME);
|
|
return(0);
|
|
}
|
|
|
|
int CConnect::get_sid_LUA(const char *challenge, const unsigned char *md5)
|
|
{
|
|
ostringstream url, command;
|
|
|
|
url << FritzAdr << "/login_sid.lua";
|
|
|
|
command << "response="
|
|
<< challenge << '-' << md5;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
string s = post2fritz(url.str().c_str(),80, command.str());
|
|
|
|
//search
|
|
string s1 = "<SID>";
|
|
string s2 = "</SID>";
|
|
string token;
|
|
size_t pos = 0;
|
|
|
|
if((pos = s.find(s1)) != string::npos)
|
|
{
|
|
s.erase(0, pos + s1.length());
|
|
|
|
pos = 0;
|
|
if((pos = s.find(s2)) != string::npos)
|
|
{
|
|
s.erase(pos, s.length());
|
|
if(s != "0000000000000000" && !s.empty())
|
|
{
|
|
strncpy(sid,s.c_str(),sizeof(sid));
|
|
if(debug) {printf("[%s] - SID %s \n", BASENAME, sid);}
|
|
return 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
/* //split into line
|
|
string delimiter = "\n";
|
|
string token;
|
|
size_t pos = 0;
|
|
|
|
while ((pos = s.find(delimiter)) != string::npos) {
|
|
i++;
|
|
token = s.substr(0, pos);
|
|
if(debug) {cout << "rec[" << i << "]" << token << endl;}
|
|
s.erase(0, pos + delimiter.length());
|
|
}
|
|
*/
|
|
|
|
printf("[%s] - failed to get SID\n", BASENAME);
|
|
return(0);
|
|
}
|
|
#if 0
|
|
/******************************************************************************
|
|
* tools
|
|
******************************************************************************/
|
|
#endif
|
|
char *CConnect::trim(char *txt)
|
|
{
|
|
register int l;
|
|
register char *p1, *p2;
|
|
|
|
if (*txt==' ')
|
|
{
|
|
for (p1=p2=txt;
|
|
(*p1==' ') || (*p1=='\t') || (*p1=='\n') || (*p1=='\r');
|
|
p1++);
|
|
while (*p1)
|
|
*p2++=*p1++;
|
|
*p2='\0';
|
|
}
|
|
if ((l=strlen(txt))>0)
|
|
for (p1=txt+l-1;
|
|
(*p1==' ') || (*p1=='\t') || (*p1=='\n') || (*p1=='\r');
|
|
*p1--='\0');
|
|
return(txt);
|
|
}
|
|
|
|
char *CConnect::UTF8toISO(char *txt)
|
|
{
|
|
//'ä','ö','ü','Ä','Ö','Ü','ß'
|
|
const char iso[7]={'\xe4','\xf6','\xfc','\xc4','\xd6','\xdc','\xdf'}, //ISO-8859-1 ö = 0xf6
|
|
utf[7]={'\xa4','\xb6','\xbc','\x84','\x96','\x9c','\x9f'}; //UTF-8 ö = 0xc3 0xb6
|
|
|
|
int i,found,quota=0;
|
|
char *rptr=txt,*tptr=txt;
|
|
|
|
while(*rptr != '\0')
|
|
{
|
|
if(*rptr=='\'')
|
|
{
|
|
quota^=1;
|
|
}
|
|
if (!quota && *rptr=='\xc3' && *(rptr+1))
|
|
{
|
|
found=0;
|
|
for(i=0; i<(int)sizeof(utf) && !found; i++)
|
|
{
|
|
if(*(rptr+1)==utf[i])
|
|
{
|
|
found=1;
|
|
*tptr=iso[i];
|
|
++rptr;
|
|
}
|
|
}
|
|
if(!found)
|
|
{
|
|
*tptr=*rptr;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
*tptr=*rptr;
|
|
}
|
|
tptr++;
|
|
rptr++;
|
|
}
|
|
*tptr=0;
|
|
return(txt);
|
|
}
|
|
|
|
int CConnect::ExistFile(const char *fname)
|
|
{
|
|
FILE *efh;
|
|
|
|
if((efh=fopen(fname,"r"))==NULL)
|
|
{
|
|
return(0);
|
|
}
|
|
fclose(efh);
|
|
return(1);
|
|
}
|
|
|
|
int CConnect::TouchFile(const char *fname)
|
|
{
|
|
FILE *tfh;
|
|
|
|
if((tfh=fopen(fname,"w"))==NULL)
|
|
{
|
|
return(0);
|
|
}
|
|
fclose(tfh);
|
|
return(1);
|
|
}
|
|
|
|
string CConnect::timestamp()
|
|
{
|
|
ostringstream txt;
|
|
|
|
time_t timestamp;
|
|
tm *now;
|
|
|
|
timestamp = time(0);
|
|
now = localtime(×tamp);
|
|
|
|
txt << '['
|
|
<< setfill('0') << setw (2) << static_cast<int>(now->tm_hour)
|
|
<< ':'
|
|
<< setfill('0') << setw (2) << static_cast<int>(now->tm_min)
|
|
<< ':'
|
|
<< setfill('0') << setw (2) << static_cast<int>(now->tm_sec)
|
|
<< ']';
|
|
|
|
return txt.str();
|
|
}
|
|
|
|
void CConnect::get_time(int *wday,int *hour,int *min)
|
|
{
|
|
time_t timestamp;
|
|
tm *now;
|
|
|
|
timestamp = time(0);
|
|
now = localtime(×tamp);
|
|
|
|
*wday=now->tm_wday;
|
|
*hour=now->tm_hour;
|
|
*min=now->tm_min;
|
|
}
|
|
|
|
int CConnect::my_atoi(const string text)
|
|
{
|
|
//atoi c++ way
|
|
int val;
|
|
stringstream s(text);
|
|
s >> val;
|
|
return(val);
|
|
}
|
|
|
|
string CConnect::itoString (int& i)
|
|
{
|
|
stringstream temp;
|
|
temp << i;
|
|
return temp.str();
|
|
}
|
|
|
|
void CConnect::StringReplace(string &str, const string search, const string rstr)
|
|
{
|
|
stringstream f(search); // stringstream f("string1;string2;stringX");
|
|
string s;
|
|
while (getline(f, s, ';'))
|
|
{
|
|
string::size_type ptr = 0;
|
|
string::size_type pos = 0;
|
|
|
|
while((ptr = str.find(s,pos)) != string::npos)
|
|
{
|
|
str.replace(ptr,s.length(),rstr);
|
|
pos = ptr + rstr.length();
|
|
}
|
|
}
|
|
}
|
|
|
|
string CConnect::parseString(const char* var, string& string_to_serarch)
|
|
{
|
|
string res="";
|
|
size_t pos1, pos2;
|
|
|
|
if((pos1 = string_to_serarch.find(var)) != string::npos)
|
|
{
|
|
// "ver":"84.05.50"
|
|
pos1 += strlen(var)+3;
|
|
string tmp = string_to_serarch.substr(pos1);
|
|
|
|
if((pos2 = tmp.find('"')) != string::npos)
|
|
{
|
|
res = tmp.substr(0,pos2);
|
|
//cout << " result: " << var << " = " << res << endl;
|
|
}
|
|
}
|
|
else
|
|
cout << " no result for " << '"' << var << '"' << res << endl;
|
|
|
|
return(res);
|
|
}
|
|
|
|
string CConnect::parseString(string search1, string search2, string str)
|
|
{
|
|
string ret, search;
|
|
size_t pos_wildcard, pos_firstline, pos_search1, pos_search2;
|
|
pos_wildcard = pos_firstline = pos_search1 = pos_search2 = string::npos;
|
|
|
|
if((pos_wildcard = search1.find('*')) != string::npos)
|
|
{
|
|
search = search1.substr(0, pos_wildcard);
|
|
//cout << "wildcard detected" << '\t' << "= " << search << "[*]" << search1.substr(pos_wildcard+1) << endl;
|
|
}
|
|
else
|
|
search = search1;
|
|
|
|
//cout << "search1" << "\t\t\t" << "= " << '"' << search << '"' << endl;
|
|
if((pos_search1 = str.find(search)) != string::npos)
|
|
{
|
|
//cout << "search1 found" << "\t\t" << "= " << '"' << search << '"' << " at pos "<< (int)(pos_search1) << " => " << str << endl;
|
|
|
|
pos_search1 += search.length();
|
|
|
|
if(pos_wildcard != string::npos)
|
|
{
|
|
size_t pos_wildcard_ext;
|
|
string wildcard_ext = search1.substr(pos_wildcard+1);
|
|
|
|
//cout << "wildcard_ext" << "\t\t" << "= " << '"' << wildcard_ext << '"' << endl;
|
|
if((pos_wildcard_ext = str.find(wildcard_ext,pos_wildcard+1)) != string::npos)
|
|
{
|
|
//cout << "wildcard_ext found" << "\t" << "= " << '"' << wildcard_ext << '"' << " at pos "<< (int)(pos_wildcard_ext) << " => " << str << endl;
|
|
pos_search1 = pos_wildcard_ext + wildcard_ext.length();
|
|
}
|
|
else
|
|
{
|
|
//cout << "wildcard_ext not found in line " << acc << " - exit" << endl;
|
|
return("");
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//cout << "search1 not found in line " << acc << " - exit" << endl;
|
|
return("");
|
|
}
|
|
|
|
if(pos_search1 != string::npos)
|
|
{
|
|
//cout << "search2 " << "\t\t" << "= " << '"' << search2 << '"' << endl;
|
|
|
|
if(search2 == "\n")
|
|
{
|
|
ret = str.substr(pos_search1, str.length() - pos_search1);
|
|
return(ret);
|
|
}
|
|
|
|
if((pos_search2 = str.find(search2, pos_search1)) != string::npos)
|
|
{
|
|
if(search2.empty())
|
|
pos_search2 = str.length();
|
|
|
|
//cout << "search2" << "\t\t\t" << "= " << '"' << search2 << '"' << " found at "<< (int)(pos_search2) << " => " << str << endl;
|
|
ret = str.substr(pos_search1, pos_search2 - pos_search1);
|
|
}
|
|
//else
|
|
//cout << "search2 not found in line " << acc << " - exit" << endl;
|
|
|
|
}
|
|
|
|
return(ret);
|
|
}
|
|
#if 0
|
|
/******************************************************************************
|
|
* get login to Fritz!Box
|
|
******************************************************************************/
|
|
#endif
|
|
int CConnect::get_login(const char* fritzPW)
|
|
{
|
|
if (!get_challenge()) {
|
|
printf("[%s] - ERROR get Challenge\n", BASENAME);
|
|
return(0);
|
|
}
|
|
|
|
get_md5(challenge,(char*)fritzPW);
|
|
|
|
if(!get_sid_LUA(challenge,md5sum)) //after 5.5x
|
|
{
|
|
printf("[%s] - login_sid.lua not found\n", BASENAME);
|
|
if(!get_sid(challenge,md5sum)) //before 5.5x
|
|
{
|
|
printf("[%s] - ERROR get SID\n", BASENAME);
|
|
return 0;
|
|
}
|
|
}
|
|
return(1);
|
|
}
|
|
#if 0
|
|
/******************************************************************************
|
|
* send logout to Fritz!Box
|
|
******************************************************************************/
|
|
#endif
|
|
int CConnect::send_logout(const char *sid)
|
|
{
|
|
ostringstream url, command;
|
|
|
|
url << FritzAdr << "/cgi-bin/webcm";
|
|
command << "sid=" << sid
|
|
<< "&security:command/logout=&getpage=../html/confirm_logout.htlm";
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
//(w_fritz, "GET /home/home.lua?sid=%s&logout=1 HTTP/1.1\r\nHost: 192.168.99.254\r\n\r\n", sid);
|
|
|
|
string s = post2fritz(url.str().c_str(),80, command.str().c_str());
|
|
|
|
return(1);
|
|
}
|
|
#if 0
|
|
/******************************************************************************
|
|
* send query's to Fritz!Box
|
|
******************************************************************************/
|
|
#endif
|
|
int CConnect::send_refresh(const char *sid)
|
|
{
|
|
ostringstream url, command;
|
|
|
|
url << FritzAdr << "/cgi-bin/webcm";
|
|
command << "getpage=../html/query.txt"
|
|
<< "&var:n[0]=telcfg:settings/RefreshJournal"
|
|
<< "&var:n[1]=telcfg:settings/Journal/count"
|
|
<< "&sid=" << sid;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
string s = post2fritz(url.str().c_str(),80, command.str().c_str());;
|
|
|
|
return(0);
|
|
}
|
|
|
|
|
|
void CConnect::parseXML(const string text)
|
|
{
|
|
device d;
|
|
int tag=0, val=0;
|
|
string tagname, value;
|
|
char* t= (char*) text.c_str();
|
|
|
|
d.tist = -1;
|
|
d.tsoll = -1;
|
|
d.absenk = -1;
|
|
d.komfort = -1;
|
|
|
|
while (*t!=0)
|
|
{
|
|
if(*t=='<')
|
|
{
|
|
tag=1;
|
|
val=0;
|
|
if(!value.empty())
|
|
{
|
|
if(tagname=="name") d.name = value;
|
|
if(tagname=="mode") d.mode = value;
|
|
if(tagname=="present") d.present = my_atoi(value);
|
|
if(tagname=="state") d.state = my_atoi(value);
|
|
if(tagname=="lock") d.lock = my_atoi(value);
|
|
if(tagname=="power") d.power = my_atoi(value);
|
|
if(tagname=="energy") d.energy = my_atoi(value);
|
|
if(tagname=="celsius") d.celsius = my_atoi(value);
|
|
if(tagname=="offset") d.offset = my_atoi(value);
|
|
if(tagname=="tist") d.tist = my_atoi(value);
|
|
if(tagname=="tsoll") d.tsoll = my_atoi(value);
|
|
if(tagname=="absenk") d.absenk = my_atoi(value);
|
|
if(tagname=="komfort") d.komfort = my_atoi(value);
|
|
}
|
|
value="";
|
|
tagname="";
|
|
}
|
|
|
|
if(*t!='>' && *t!='<')
|
|
{
|
|
if(tag==1)
|
|
tagname += *t;
|
|
|
|
if(val==1)
|
|
value += *t;
|
|
}
|
|
|
|
if(*t=='>')
|
|
{
|
|
tag=0;
|
|
val=1;
|
|
if(strstr(tagname.c_str(),"identifier="))
|
|
{
|
|
string s = parseString("identifier=\"","\"",tagname);
|
|
StringReplace(s," ","");
|
|
d.identifier = s;
|
|
}
|
|
if(strstr(tagname.c_str(),"productname="))
|
|
{
|
|
string s = parseString("productname=\"","\"",tagname);
|
|
StringReplace(s," ","");
|
|
d.productname = s;
|
|
}
|
|
|
|
if(!strcmp(tagname.c_str(),"/device") || !strcmp(tagname.c_str(),"/group"))
|
|
devices.push_back(d);
|
|
}
|
|
|
|
t++;
|
|
}
|
|
}
|
|
|
|
void CConnect::send2actor(const unsigned int& inx, int& t_soll)
|
|
{
|
|
int state_soll = -1;
|
|
int temperature = 0;
|
|
|
|
if(devices[inx].productname=="FRITZ!DECT200")
|
|
{
|
|
cout << '[' << BASENAME << "]\t\tTemperature: Plan(" << t_soll << ")/Now(" << devices[inx].celsius << "); state: " << (devices[inx].state == 1 ? "ON":"OFF") << endl;
|
|
|
|
// convert special temperature
|
|
switch(t_soll) {
|
|
case 1: state_soll = 1; // switch on
|
|
break;
|
|
case 0: state_soll = 0; // switch off
|
|
break;
|
|
}
|
|
|
|
if(state_soll == -1) {
|
|
if(t_soll <= devices[inx].celsius)
|
|
{ //temperature okay
|
|
cout << '[' << BASENAME << "]\t\tTemperature okay"<< endl;
|
|
state_soll = 0;
|
|
}
|
|
else if(t_soll > devices[inx].celsius)
|
|
{ //temperature to low
|
|
cout << '[' << BASENAME << "]\t\tTemperature to low"<< endl;
|
|
state_soll = 1;
|
|
}
|
|
}
|
|
|
|
if(state_soll >= 0 && state_soll != devices[inx].state)
|
|
{
|
|
cout << '[' << BASENAME << "]\t\t" << "\033[0;32m" << "sending Deviceswitch "<<(state_soll == 0? "OFF":"ON") << "\033[0m" << endl;
|
|
|
|
if(state_soll == 0)
|
|
smartHome(sid, "setswitchoff", devices[inx].identifier);
|
|
else
|
|
smartHome(sid, "setswitchon", devices[inx].identifier);
|
|
}
|
|
else
|
|
cout << '[' << BASENAME << "]\t\t" << "\033[0;31m" << "send nothing to actor, state is equal" << "\033[0m" << endl;
|
|
|
|
// set flagfile
|
|
string flag="/var/etc/.device_" + devices[inx].identifier;
|
|
setFlag(flag.c_str(),state_soll);
|
|
}
|
|
else if(devices[inx].productname=="CometDECT" || devices[inx].productname=="FRITZ!DECT300")
|
|
{
|
|
cout << '[' << BASENAME << "]\t\tTemperature: Plan " << t_soll << " (" << (t_soll > 1 ? t_soll * 0.2 : t_soll) << "), tist(" << devices[inx].tist << "); tsoll(" << devices[inx].tsoll << ')' << endl;
|
|
|
|
// convert temperature
|
|
switch(t_soll) {
|
|
case -1: temperature = -1; // do nothing
|
|
break;
|
|
case 0: temperature = 253; // switch off
|
|
break;
|
|
default: temperature = t_soll * 0.2; // temperature step up to 0.5°, 43 = 21,5°
|
|
break;
|
|
}
|
|
|
|
if(devices[inx].tsoll != temperature) // check if plan temperature stored in actor
|
|
{
|
|
if(temperature == -1)
|
|
cout << '[' << BASENAME << "]\t\t" << "\033[0;31m" << "send nothing to actor while Plan value is (-1)" << "\033[0m" << endl;
|
|
else {
|
|
cout << '[' << BASENAME << "]\t\t" << "\033[0;32m" << "send new tsoll temperature (" << temperature << ") to actor" << "\033[0m" << endl;
|
|
|
|
string command = "sethkrtsoll¶m=" + itoString(temperature);
|
|
smartHome(sid, command.c_str(), devices[inx].identifier);
|
|
}
|
|
}
|
|
else
|
|
cout << '[' << BASENAME << "]\t\t" << "\033[0;31m" << "send nothing to actor, storred tsoll temperature is equal" << "\033[0m" << endl;
|
|
}
|
|
else
|
|
cout << '[' << BASENAME << "]\t\t" << "\033[0;32m" << "productname \"" << devices[inx].productname << "\" not found" << "\033[0m" << endl;
|
|
}
|
|
|
|
int CConnect::checkdevice(map<string, vector<string> >& w, map<string, vector<string> >& t)
|
|
{
|
|
int wday,hour,min,ret=-1;
|
|
|
|
get_time(&wday,&hour,&min);
|
|
|
|
// for building device index
|
|
unsigned int dev_inx = 0;
|
|
|
|
for(vector<device>::iterator dev = devices.begin(); dev != devices.end(); ++dev, ++dev_inx)
|
|
{
|
|
// find cdw map key by device id
|
|
map<string, vector<string> >::iterator dev_search = w.find((*dev).identifier);
|
|
if(dev_search != w.end())
|
|
{
|
|
// check config table syntax
|
|
if(dev_search->second.size() != 7) {
|
|
cout << "[ERROR] - " << __FUNCTION__ << "(): WP table syntax error, we must have 7 values per line\n";
|
|
return(ret);
|
|
}
|
|
|
|
// get map value for weekday
|
|
string w_value = dev_search->second[wday];
|
|
cout << '[' << BASENAME << "] - " << __FUNCTION__ << "(): device[" << (*dev).identifier << "] weekday = " << wday << ", value = " << w_value << ", name = " << (*dev).name << endl;
|
|
|
|
// find cdt map key by cdw value
|
|
map<string, vector<string> >::iterator search = t.find(w_value);
|
|
if(search != t.end())
|
|
{
|
|
// check config table syntax
|
|
if( search->second.size() % 3 != 0 ) { // 3 because HH MM are 2 values
|
|
cout << '[' << BASENAME << "] - " << __FUNCTION__ << "(): DP table syntax error, we must have time and temperature pair (HH:MM,TTT)\n";
|
|
return(ret);
|
|
}
|
|
|
|
// read every time and temperature pair for cdt key
|
|
for (unsigned int i=0; i <search->second.size(); ++i)
|
|
{
|
|
// check time and temperature after reading 3 values (1=HH, 2=MM, 3=temperature)
|
|
unsigned int delim = 3;
|
|
if( (i+1) % delim == 0)
|
|
{
|
|
int now = hour *60 + min;
|
|
int ev = my_atoi(search->second[i-2]) * 60 + my_atoi(search->second[i-1]);
|
|
int next_ev = ev;
|
|
int temperature = my_atoi(search->second[i]);
|
|
|
|
// check for existing next time event
|
|
if( i+1 != search->second.size() )
|
|
next_ev = my_atoi(search->second[i+1]) * 60 + my_atoi(search->second[i+2]);
|
|
|
|
if((now >= ev && now <= next_ev) ||
|
|
(i+1 == search->second.size())) // last event was earlier than now
|
|
{
|
|
// time event
|
|
cout << '[' << BASENAME << "]\t\tTime event = " << ev/60.0 << " temperature = " << temperature << endl;
|
|
|
|
send2actor(dev_inx, temperature);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
cout << '[' << BASENAME << "]\t\tvalue not found in DP map" << endl;
|
|
}
|
|
else
|
|
cout << '[' << BASENAME << "] - " << __FUNCTION__ << ": device[" << (*dev).identifier << "] not found in table WP" << endl;
|
|
}
|
|
|
|
return(ret);
|
|
}
|
|
|
|
int CConnect::inTime(int index, vector<vector<string> >& week, vector<vector<string> >& day)
|
|
{
|
|
int wday,hour,min,ret=-1;
|
|
|
|
get_time(&wday,&hour,&min);
|
|
int pos = my_atoi(week[index][wday]);
|
|
|
|
cout<<'['<<BASENAME<<"] - "<< __FUNCTION__<< "() "<<"device index="<<index<<" wday="<<wday<<" hour="<<hour<<" min="<<min<<"; Weekplan ("<<pos<<") = ";
|
|
|
|
if(pos != 0)
|
|
{
|
|
pos--;
|
|
/*
|
|
for (unsigned int i=0; i<day[pos].size(); ++i)
|
|
{
|
|
cout << day[pos][i] << ' ';
|
|
}
|
|
cout<<endl;
|
|
*/
|
|
for (unsigned int i=0; i<day[pos].size(); ++i)
|
|
{
|
|
if(i % 4 == 0)
|
|
{
|
|
int now = hour *60 + min;
|
|
int from = my_atoi(day[pos][i]) *60 + my_atoi(day[pos][i+1]);
|
|
int to = my_atoi(day[pos][i+2]) *60 + my_atoi(day[pos][i+3]);
|
|
|
|
if(now >= from && now <= to)
|
|
{
|
|
cout<< day[pos][i]<<":"<<day[pos][i+1]<<"-"<<day[pos][i+2]<<":"<< day[pos][i+3]<<endl;
|
|
return(1);
|
|
}
|
|
cout<< day[pos][i]<<":"<<day[pos][i+1]<<"-"<<day[pos][i+2]<<":"<< day[pos][i+3]<<' ';
|
|
}
|
|
}
|
|
cout<<endl;
|
|
ret=0;
|
|
}
|
|
return(ret);
|
|
}
|
|
|
|
int CConnect::smartHome(const char *sid, const char *command, const string ain)
|
|
{
|
|
ostringstream url;
|
|
|
|
int ret=-1;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
url << FritzAdr << "/webservices/homeautoswitch.lua?switchcmd=" << command << (ain.empty()?"":"&ain=") << ain << "&sid=" << sid;
|
|
|
|
string s = post2fritz(url.str().c_str());
|
|
|
|
if(!strcmp(command,"getdevicelistinfos"))
|
|
{
|
|
//s = "<devicelist version=\"1\"><device identifier=\"08761 0831319\" id=\"16\" functionbitmask=\"1280\" fwversion=\"03.36\" manufacturer=\"AVM\" productname=\"FRITZ!DECT Repeater 100\"><present>1</present><name>Fritz!Dect Rep 100</name><temperature><celsius>200</celsius><offset>-24</offset></temperature></device><device identifier=\"08761 0021731\" id=\"17\" functionbitmask=\"896\" fwversion=\"03.36\" manufacturer=\"AVM\" productname=\"FRITZ!DECT 200\"><present>1</present><name>WZ Stehlampe</name><switch><state>0</state><mode>manuell</mode><lock>0</lock></switch><powermeter><power>0</power><energy>4282</energy></powermeter><temperature><celsius>190</celsius><offset>-27</offset></temperature></device><device identifier=\"08761 0068302\" id=\"18\" functionbitmask=\"896\" fwversion=\"03.36\" manufacturer=\"AVM\" productname=\"FRITZ!DECT 200\"><present>1</present><name>WZ Freiheitsstatue</name><switch><state>0</state><mode>auto</mode><lock>0</lock></switch><powermeter><power>0</power><energy>2167</energy></powermeter><temperature><celsius>190</celsius><offset>-17</offset></temperature></device><device identifier=\"08761 0069718\" id=\"19\" functionbitmask=\"896\" fwversion=\"03.36\" manufacturer=\"AVM\" productname=\"FRITZ!DECT 200\"><present>0</present><name>Bananen-Haus</name><switch><state></state><mode></mode><lock></lock></switch><powermeter><power></power><energy></energy></powermeter><temperature><celsius></celsius><offset></offset></temperature></device><device identifier=\"08761 0021736\" id=\"20\" functionbitmask=\"896\" fwversion=\"03.36\" manufacturer=\"AVM\" productname=\"FRITZ!DECT 200\"><present>1</present><name>RaspiServer</name><switch><state>1</state><mode>manuell</mode><lock>0</lock></switch><powermeter><power>4360</power><energy>53683</energy></powermeter><temperature><celsius>210</celsius><offset>-35</offset></temperature></device><device identifier=\"08761 0068647\" id=\"21\" functionbitmask=\"896\" fwversion=\"03.36\" manufacturer=\"AVM\" productname=\"FRITZ!DECT 200\"><present>1</present><name>Wohnmobil</name><switch><state>0</state><mode>manuell</mode><lock>0</lock></switch><powermeter><power>0</power><energy>12620</energy></powermeter><temperature><celsius>136</celsius><offset>-25</offset></temperature></device><group identifier=\"86:6D:75-900\" id=\"900\" functionbitmask=\"512\" fwversion=\"1.0\" manufacturer=\"AVM\" productname=\"\"><present>1</present><name>Testgruppe</name><switch><state>0</state><mode>manuell</mode><lock></lock></switch><groupinfo><masterdeviceid>0</masterdeviceid><members>17,21</members></groupinfo></group></devicelist>";
|
|
parseXML(s);
|
|
|
|
//cout <<"identifier;present;name;state;mode;lock;power;energy;celsius;offset;" <<endl;
|
|
|
|
unsigned int x=0;
|
|
for(vector<device>::iterator it = devices.begin(); it != devices.end(); ++it, ++x)
|
|
{
|
|
if(!ain.empty())
|
|
{
|
|
if((*it).identifier==ain)
|
|
{
|
|
cout<<'['<<BASENAME<<"] - " <<(*it).productname << ": "
|
|
<<(*it).identifier << ';'<<(*it).present << ';'<<(*it).name << ';'<<(*it).state << ';'<<(*it).mode << ';'
|
|
<<(*it).lock << ';'<<(*it).power << ';'<<(*it).energy << ';'<<(*it).celsius << ';'<<(*it).offset << ';'
|
|
<<(*it).tist << ';'<<(*it).tsoll << ';'<<(*it).absenk << ';'<<(*it).komfort << ';'<<endl;
|
|
return (x);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
cout<<'['<<BASENAME<<"] - " <<(*it).productname << ": "
|
|
<<(*it).identifier << ';'<<(*it).present << ';'<<(*it).name << ';'<<(*it).state << ';'<<(*it).mode << ';'
|
|
<<(*it).lock << ';'<<(*it).power << ';'<<(*it).energy << ';'<<(*it).celsius << ';'<<(*it).offset << ';'
|
|
<<(*it).tist << ';'<<(*it).tsoll << ';'<<(*it).absenk << ';'<<(*it).komfort << ';'<<endl;
|
|
}
|
|
}
|
|
}
|
|
|
|
return (ret);
|
|
}
|
|
|
|
#if 0
|
|
/******************************************************************************
|
|
* FritzCallMonitor
|
|
******************************************************************************/
|
|
#endif
|
|
//new query
|
|
int CConnect::send_TAMquery(const char *flag, const char *sid, const char *searchstr)
|
|
{
|
|
ostringstream url;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
url << FritzAdr << "/query.lua?sid=" << sid << searchstr;
|
|
|
|
string s = post2fritz(url.str().c_str());
|
|
StringReplace(s," ","");
|
|
string res = parseString("var", s);
|
|
|
|
if(!res.empty())
|
|
{
|
|
setFlag(flag,my_atoi(res));
|
|
}
|
|
else {
|
|
printf("[%s] - %s ERROR\n", BASENAME, timestamp().c_str());
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
void CConnect::setFlag(const char *flag, const int& i)
|
|
{
|
|
if(i==0)
|
|
{
|
|
cout << '[' << BASENAME << ']' << " - " << timestamp() << " disable flag "<<flag<<endl;
|
|
remove(flag);
|
|
}
|
|
else {
|
|
cout << '[' << BASENAME << ']' << " - " << timestamp() << " enable flag "<<flag<<endl;
|
|
if(!TouchFile(flag))
|
|
cout << "ERROR writing flag " << flag << endl;
|
|
}
|
|
}
|
|
|
|
#if 0
|
|
//old query
|
|
|
|
int CConnect::send_query(const char *flag, const char *sid, const char *searchstr)
|
|
{
|
|
ostringstream command;
|
|
int i=0;
|
|
char *line;
|
|
ssize_t read;
|
|
size_t len;
|
|
|
|
command << "getpage=../html/query.txt"
|
|
<< searchstr
|
|
<< "&sid=" << sid;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
post2fritz(command.str().c_str());
|
|
|
|
line=NULL;
|
|
while ((read = getline(&line, &len, r_fritz)) != -1)
|
|
{
|
|
i++;
|
|
if(debug>1)
|
|
printf("line[%d]: %s", i, line);
|
|
|
|
if(i==7)
|
|
{
|
|
if(strlen(line)>10) {
|
|
printf("ERROR\n");
|
|
break;
|
|
}
|
|
else if(atoi(trim(line))==0) {
|
|
printf("res=%s =>disable flag\n",trim(line));
|
|
remove(flag);
|
|
break;
|
|
}
|
|
else {
|
|
printf("res=%s =>enable flag\n",trim(line));
|
|
if(!TouchFile(flag))
|
|
printf("ERROR writing flag %s\n",flag);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if(line)
|
|
free(line);
|
|
|
|
quitfritz();
|
|
return(0);
|
|
}
|
|
#endif
|
|
|
|
int CConnect::get_callerlist(const char *sid, const char *file)
|
|
{
|
|
ostringstream url, command;
|
|
|
|
if(debug) {cout << '[' << BASENAME << "] - " << __FUNCTION__ << "()" << endl;}
|
|
|
|
url << FritzAdr << "/fon_num/foncalls_list.lua?csv=";
|
|
command << "refresh=&sid=" << sid;
|
|
|
|
string s = post2fritz(url.str().c_str(),80, command.str().c_str(), file);
|
|
|
|
return(0);
|
|
}
|
|
|