16 января 2018 г. 6:55

Как проверить доступность сервера

IP сервера с которым я работаю может выглядеть по разному в зависимости от того внутри сети я нахожусь или во вне. Соответственно когда я запускаю программу - мне нужно понять по какому адресу доступен сервер по внешнему или по внутреннему(или вообще не доступен).  Сейчас я вот так подключаюсь к серверу

url_sea_text = server_ip_loc+"search/sea_text/"+poisk_txt+"/"+user_id;

networkManager_sea_text = new QNetworkAccessManager();
 QNetworkReply *replay_sea_text =  networkManager_sea_text->get(QNetworkRequest(QUrl(url_sea_text)));

    connect(networkManager_sea_text, &QNetworkAccessManager::finished, this, &AppCore::onResult_sea_text);

    connect(replay_sea_text, SIGNAL(downloadProgress(qint64,qint64)),this, SLOT(updateDownloadProgress(qint64,qint64)));

NetworkAccessManager работает асинхронно. И чтобы сильно долго не ждать надо каким-то образом по таймеру проверить, что ответа нет и пробовать другой адрес. Насколько я понимаю мне в процедуру

void AppCore::updateDownloadProgress(qint64 bytesRead, qint64 totalBytes)
{

   QTextStream cout(stdout);
   cout << QString::number(bytesRead) +"/"+QString::number(totalBytes) << endl;
   qDebug() <<"count"<< count;

   if (totalBytes>0) {
    count=bytesRead/totalBytes;
   }
   if (count>0.2) {
    emit sendToQml(count);
   }
}
Добавить проверку по таймеру, который нужно запустить в момент создания коннекта. И если на заданный тайм-аут
bytesRead =0 
то делать replay_sea_text.abort()  ?
У Вас есть урок по ping - но как мне кажется в этом случае пинг не достаточен. И он не кросс - платформен.


  • #
  • 16 января 2018 г. 14:23

Добрый день!
Теоретически можно использовать QTcpSocket, у него есть метод connectToHost.

Возможно, что проверка доступности через этот класс будет осуществляться несколько быстрее, чем через QNetworkAccessManager.

Спасибо. Попробую. Проблема в том, что сервер может пинговаться, но служба не работать(не выдавать нужные ссылки - у меня же на серваке сервер приложений ORDS крутится). Но это уже другая проблема. Но ее тоже надо обработать. И в этом случае QNetworkAccessManager все-таки предпочтителен. Буду пробовать.

Для QTcpSocket - насколько я понял  - необходимы настройки прокси сервера(если он есть) - не хочется с этим возиться.

Как и писали на форуме ожидание ответа от  connect(networkManager_news, &QNetworkAccessManager::finished, this, &AppCore::onResult_net);  составляет 45 секунд. Очень долго.

Как проверить что спустя 10 секунд connect(networkManager_news, &QNetworkAccessManager::finished, this, &AppCore::onResult_net);     &QNetworkAccessManager::finished -  не финишировал?

  • alex_lip
  • #
  • Ответ был помечен как решение
  • 17 января 2018 г. 8:16

Сделайте в классе AppCore некоторую переменную, которая будет индикатором финиширования. Если QNetworkAccessManager::finished ещё не финишировал, то переменная должна иметь значение false, если финишировал, то в слоте onResult_net поставьте true.


Также можно просто прерывать запрос после 10 секунд, если QNetworkAccessManager не финишировал. Каждый запрос get, put, post возвращает QNetworkReply, выполнение которого можно прервать через его слот abort()

Точно!!!
Я сейчас так пробую

QNetworkReply *replay_news = networkManager_news->get(QNetworkRequest(QUrl(url_news)));
connect(networkManager_news, &QNetworkAccessManager::finished, this, &AppCore::onResult_net);
QTimer::singleShot(10000, this, &AppCore::singleShotFunc);

void AppCore::singleShotFunc()
{
     qDebug() << reply_news->readAll() ;
}


Как мне в
singleShotFunc
передать
*replay_news
Ну просто интересно. Что вернет
*replay_news - если запрос не выполнен
  • #
  • 17 января 2018 г. 9:45

Ну как вариант вместо singleShotFunc можно использовать лямбду и захватить QNetworkReply по указателю этой лямбдой

Ругается  ошибка: 'reply_news' was not declared in this scope
qDebug() << reply_news->readAll() ;
^

QTimer::singleShot(5000, [=] {

       qDebug() << "Спустя 5 сек." << QTime::currentTime().toString("hh:mm:ss");
      qDebug() << reply_news->readAll() ;
    });
  • #
  • отредактировано 17 января 2018 г. 10:18
  • 17 января 2018 г. 10:11

и кстати

void AppCore::singleShotFunc()
{
         if (hatico_net==0){
        server_ip_loc = server_ip_ext;
        qDebug() << "Меняем IP сервера на другой";
        qDebug() << server_ip_loc;
        replay_news->abort();
    }
}

replay_news->abort();   -  программа вываливается



Сдаётся мне, что когда вы делаете обращение

replay_news->abort()
У вас указатель протухший оказывается. Проверяйте на nullptr его как минимум прежде, чем обращаться к нему.
Я сейчас решаю сходную задачу, но у меня такой проблемы нет при вызове метода abort().

ок. Проверю. В принципе обошелся вашим советом и искусственными задержками.
А все-таки -  сколько на приложение нужно
networkManager = new QNetworkAccessManager();  ?  Пишут что вроде бы один. С разными  QNetworkReply* replay_cont.
Но у меня в этом случае чушь получается.  Поэтому мне приходиться их штамповать на каждую форму (или уникальную ссылку для получения данных с сервера)  Как правильно?

Тут скорее всего ситуативно, но по факту да, в большинстве задач можно обойтись и одним network менеджером

Ответы

Только авторизованные пользователи могут отвечать на форуме.
Пожалуйста, Авторизуйтесь или Зарегистрируйтесь
22 февраля 2018 г. 18:58
Oleg_kgd

C++ - Тест 001. Первая программа и типы данных

  • Результат 66 баллов
  • Очки рейтинга -1
21 февраля 2018 г. 19:18
sentinel

Qt - Тест 001. Сигналы и слоты

  • Результат 78 баллов
  • Очки рейтинга 2
21 февраля 2018 г. 11:32
barilla

C++ - Тест 006. Перечисления

  • Результат 0 баллов
  • Очки рейтинга -10
Последние комментарии
22 февраля 2018 г. 16:42
soz7557

Qt/C++ - Урок 029. Изображение в базе данных в Qt – Сохранение и Восстановление

Hi, could you please show how to delete file from image Blob?  also if the same image exist in Blob then don't over write..

21 февраля 2018 г. 8:37
EVILEG

Qt/C++ - Урок 027. Полиморфизм в Qt на примере геометрических фигур в QGraphicsScene

Добрый день! 1) Эллипс можно реализовать так void Ellipse::paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget){ painter->setPen(QPen(...

20 февраля 2018 г. 22:10
Log159

Qt/C++ - Урок 027. Полиморфизм в Qt на примере геометрических фигур в QGraphicsScene

Здравствуйте! В программировании новичок и есть пара вопросов. Буду очень благодарен за ответ. Не совсем понимаю как: 1) реализовать подобным образом рисование эллипса(конкре...

18 февраля 2018 г. 14:42
EVILEG

QML - Урок 019. Navigation Drawer в Qt Qml Android

Да, теперь представляю, как то работает. Согласен, ваша правка определённо к месту здесь.

Сейчас обсуждают на форуме
21 февраля 2018 г. 22:19
vitaliy_antipov

Проблема с ComboBox

Спасибо за ответы, есть над чем подумать

21 февраля 2018 г. 13:26
sol11

Qtableviev после сортировки

Спасибо, всё заработало :) Единственное вот тут row на id поменял и всё круто :)) if(id == -1){ model->insertRow(model->rowCount(QModelIndex())); map...

20 февраля 2018 г. 13:18
alex_lip

Разбить один qml файл на несколько составляющих

Да спасибо. Просто после необходимости специфичных названий для файла - стараюсь обращать внимание на любую мелочь.

20 февраля 2018 г. 8:13
EVILEG

Передача файлов в django минуя временные папки django и nginx

Тогда я даже и не знаю, прошерстил документацию, но там нет информармации о возможности отключения сохранения временных файлов. Как я понял временные файлы используются, когда тело запро...

18 февраля 2018 г. 12:34
EVILEG

QGraphicsView

Добрый день!QGraphicsView - это виджет, а значит, что в качестве парента для него выступает QWidget, а не QObject.То есть из ошибок, которые сразу бросаются в глаза в этом коде, здесь прису...