s20/server.cpp

192 lines
6.7 KiB
C++
Raw Normal View History

2015-02-01 01:42:45 +00:00
/*************************************************************************
* Copyright (C) 2015 by Andrius Štikonas <andrius@stikonas.eu> *
* *
* This program is free software; you can redistribute it and/or modify *
* it under the terms of the GNU General Public License as published by *
* the Free Software Foundation; either version 3 of the License, or *
* (at your option) any later version. *
* *
* This program is distributed in the hope that it will be useful, *
* but WITHOUT ANY WARRANTY; without even the implied warranty of *
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the *
* GNU General Public License for more details. *
* *
* You should have received a copy of the GNU General Public License *
* along with this program. If not, see <http://www.gnu.org/licenses/>.*
*************************************************************************/
2015-02-28 23:30:00 +00:00
#include <QNetworkConfiguration>
#include <QNetworkConfigurationManager>
#include <QNetworkSession>
#include <QTimer>
#include <QUdpSocket>
2015-02-21 21:14:22 +00:00
#include <iostream>
2015-02-01 01:42:45 +00:00
#include "consolereader.h"
#include "server.h"
2015-02-28 23:30:00 +00:00
Server::Server ( std::vector<Socket*> *sockets_vector )
2015-02-01 01:42:45 +00:00
{
sockets = sockets_vector;
udpSocketGet = new QUdpSocket();
2015-02-28 23:30:00 +00:00
udpSocketGet->bind ( QHostAddress::Any, 10000 );
connect ( udpSocketGet, &QUdpSocket::readyRead, this, &Server::readPendingDatagrams);
2015-02-28 23:30:00 +00:00
discoverSockets();
QTimer *discoverTimer = new QTimer(this);
discoverTimer->setInterval(1*60*1000); // 1 min
discoverTimer->setSingleShot(false);
connect(discoverTimer, &QTimer::timeout, this, &Server::discoverSockets);
discoverTimer->start();
2015-02-01 01:42:45 +00:00
start();
}
2015-02-01 01:42:45 +00:00
2015-02-28 23:30:00 +00:00
Server::Server(uint16_t port, QByteArray password)
{
2015-02-28 23:30:00 +00:00
QNetworkConfiguration *cfg = new QNetworkConfiguration;
QNetworkConfigurationManager *ncm = new QNetworkConfigurationManager;
QEventLoop *loop = new QEventLoop;
loop->connect(ncm, &QNetworkConfigurationManager::updateCompleted, loop, &QEventLoop::quit);
ncm->updateConfigurations();
loop->exec();
delete loop;
*cfg = ncm->defaultConfiguration();
QByteArray ssid = cfg->name().toLocal8Bit();
auto nc = ncm->allConfigurations();
for (auto &x : nc)
{
if (x.bearerType() == QNetworkConfiguration::BearerWLAN)
{
if (x.name() == "WiWo-S20")
{
std::cout << "Connecting to WiWo-S20 wireless" << std::endl;
cfg = &x;
}
}
}
auto session = new QNetworkSession(*cfg, this);
session->open();
std::cout << "Wait for connected!" << std::endl;
if (session->waitForOpened())
std::cout << "Connected!" << std::endl;
QUdpSocket *udpSocketSend = new QUdpSocket();
udpSocketGet = new QUdpSocket();
udpSocketGet->bind ( QHostAddress::Any, port);
2015-02-07 17:29:12 +00:00
QByteArray reply;
2015-02-01 01:42:45 +00:00
udpSocketGet->writeDatagram ( QByteArray::fromStdString("HF-A11ASSISTHREAD"), QHostAddress::Broadcast, port );
reply = listen(QByteArray::fromStdString("HF-A11ASSISTHREAD"));
QList<QByteArray> list = reply.split(',');
QHostAddress ip(QString::fromLatin1(list[0]));
std::cout << "IP: " << ip.toString().toStdString() << std::endl;
udpSocketGet->writeDatagram ( QByteArray::fromStdString("+ok"), ip, port );
udpSocketGet->writeDatagram ( QByteArray::fromStdString("AT+WSSSID=") + ssid + QByteArray::fromStdString("\r"), ip, port );
listen();
udpSocketGet->writeDatagram ( QByteArray::fromStdString("AT+WSKEY=WPA2PSK,AES,") + password + QByteArray::fromStdString("\r"), ip, port ); // FIXME: support different security settings
2015-02-28 23:30:00 +00:00
// OPEN, SHARED, WPAPSK......NONE, WEP, TKIP, AES
listen();
udpSocketGet->writeDatagram ( QByteArray::fromStdString("AT+WMODE=STA\r"), ip, port );
listen();
udpSocketGet->writeDatagram ( QByteArray::fromStdString("AT+Z\r"), ip, port ); // reboot
2015-02-28 23:30:00 +00:00
session->close();
// FIXME: discover the new socket
2015-02-28 23:30:00 +00:00
std::cout << "Finished" << std::endl;
2015-02-01 01:42:45 +00:00
}
Server::~Server()
{
delete udpSocketGet;
}
QByteArray Server::listen(QByteArray message)
{
QByteArray reply;
QHostAddress sender;
quint16 senderPort;
bool stop = false;
while ( !stop )
{
QThread::msleep(50);
while ( udpSocketGet->hasPendingDatagrams() )
{
reply.resize ( udpSocketGet->pendingDatagramSize() );
udpSocketGet->readDatagram ( reply.data(), reply.size(), &sender, &senderPort );
if (reply != message)
{
stop = true;
std::cout << reply.toStdString() << std::endl;
}
}
}
return reply;
}
void Server::run()
{
readPendingDatagrams();
}
2015-02-01 01:42:45 +00:00
void Server::readPendingDatagrams()
{
while ( udpSocketGet->hasPendingDatagrams() )
{
QByteArray reply;
reply.resize ( udpSocketGet->pendingDatagramSize() );
QHostAddress sender;
quint16 senderPort;
udpSocketGet->readDatagram ( reply.data(), reply.size(), &sender, &senderPort );
if ( reply != discover && reply.left ( 2 ) == magicKey ) // check for Magic Key
{
if ( reply.mid ( 4, 2 ) == QByteArray::fromHex ( "71 61" ) ) // Reply to discover packet
{
bool duplicate = false;
for ( std::vector<Socket*>::const_iterator i = sockets->begin() ; i != sockets->end(); ++i )
{
if ( (*i)->ip == sender )
{
duplicate = true;
break;
}
}
if ( !duplicate )
{
Socket *socket = new Socket ( sender, reply );
sockets->push_back ( socket );
2015-02-28 23:30:00 +00:00
Q_EMIT discovered();
2015-02-01 01:42:45 +00:00
}
}
else
{
QByteArray mac = reply.mid(6,6);
for ( std::vector<Socket*>::iterator i = sockets->begin() ; i != sockets->end(); ++i )
{
if ( (*i)->mac == mac )
{
(*i)->parseReply(reply);
break;
}
}
}
}
}
}
2015-02-28 23:30:00 +00:00
void Server::discoverSockets()
{
QUdpSocket *udpSocketSend = new QUdpSocket();
2015-02-28 23:30:00 +00:00
udpSocketSend->connectToHost ( QHostAddress::Broadcast, 10000 );
udpSocketSend->write ( discover );
udpSocketSend->write ( discover );
udpSocketSend->disconnectFromHost();
delete udpSocketSend;
}