tchttpservice.cpp 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341
  1. #include "tchttpservice.h"
  2. QScopedPointer<TCHttpService> TCHttpService::m_instance;
  3. TCHttpService *TCHttpService::getInstance()
  4. {
  5. if (m_instance.isNull()) {
  6. m_instance.reset(new TCHttpService);
  7. }
  8. return m_instance.data();
  9. }
  10. void TCHttpService::apiLogin()
  11. {
  12. QByteArray pwdMd5 = QCryptographicHash::hash(m_firstPwd.toUtf8(), QCryptographicHash::Md5);
  13. QString md5Hex = pwdMd5.toHex();
  14. QString urlStr;
  15. if (m_enableSsl)
  16. urlStr = "https://" + m_domain + "/api/login";
  17. else
  18. urlStr = "http://" + m_domain + "/api/login";
  19. QUrl url(urlStr);
  20. QJsonObject jsonObj;
  21. jsonObj["user"]=m_userName;
  22. jsonObj["pwd"]=md5Hex;
  23. QJsonDocument jsonDoc(jsonObj);
  24. QByteArray jsonData = jsonDoc.toJson();
  25. QMap<QString, QString> headers;
  26. headers.insert("Content-Type", "application/json");
  27. QNetworkReply* reply = nullptr;
  28. QNetworkRequest request(url);
  29. for (auto ite = headers.constBegin(); ite != headers.constEnd(); ite++){
  30. request.setRawHeader(ite.key().toUtf8(), ite.value().toUtf8());
  31. }
  32. reply = m_manager.post(request, jsonData);
  33. QObject::connect(reply, &QNetworkReply::finished, [this, reply]() {
  34. QJsonDocument jsonDoc = QJsonDocument::fromJson(reply->readAll());
  35. if (jsonDoc.isEmpty())
  36. return;
  37. QJsonObject jsonObj = jsonDoc.object();
  38. int code = jsonObj["code"].toInt();
  39. if (code == 0) {
  40. this->m_token = jsonObj["token"].toString();
  41. qDebug() << this->m_token;
  42. m_isOnline = true;
  43. ImageManager::instance()->addDomainUser(m_domain, m_userName);
  44. QString stdPath = QStandardPaths::writableLocation(QStandardPaths::AppConfigLocation);
  45. ImageManager::instance()->initialize(stdPath + "/picpanel.db");
  46. emit signal_loginSuc();
  47. }
  48. });
  49. }
  50. void TCHttpService::apiUpload(const QString &filePath)
  51. {
  52. QString urlStr;
  53. if (m_enableSsl)
  54. urlStr = "https://" + m_domain + "/api/upload";
  55. else
  56. urlStr = "http://" + m_domain + "/api/upload";
  57. QFile *file = new QFile(filePath);
  58. if (!file->open(QIODevice::ReadOnly)) {
  59. qDebug() << "无法打开文件:" << filePath;
  60. delete file;
  61. return;
  62. }
  63. // 计算文件 MD5
  64. QCryptographicHash hash(QCryptographicHash::Md5);
  65. hash.addData(file);
  66. QString md5 = hash.result().toHex();
  67. // 重置文件指针
  68. file->seek(0);
  69. // 获取文件大小
  70. qint64 fileSize = file->size();
  71. // 创建 multipart 请求
  72. #if (QT_VERSION >= QT_VERSION_CHECK(6, 0, 0))
  73. QHttpMultiPart *multiPart = new QHttpMultiPart(QHttpMultiPart::FormDataType);
  74. #else
  75. QHttpMultiPart *multiPart = new QHttpMultiPart(QHttpMultiPart::FormData);
  76. #endif
  77. // 添加文件部分
  78. QHttpPart filePart;
  79. filePart.setHeader(QNetworkRequest::ContentTypeHeader, QVariant("application/octet-stream"));
  80. filePart.setHeader(QNetworkRequest::ContentDispositionHeader,
  81. QVariant("form-data; name=\"file\"; filename=\"" + QFileInfo(filePath).fileName() + "\""));
  82. filePart.setBodyDevice(file);
  83. multiPart->append(filePart);
  84. // 添加用户信息部分
  85. QHttpPart userPart;
  86. userPart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"user\""));
  87. userPart.setBody(m_userName.toUtf8());
  88. multiPart->append(userPart);
  89. // 添加 MD5 部分
  90. QHttpPart md5Part;
  91. md5Part.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"md5\""));
  92. md5Part.setBody(md5.toUtf8());
  93. multiPart->append(md5Part);
  94. // 添加文件大小部分
  95. QHttpPart sizePart;
  96. sizePart.setHeader(QNetworkRequest::ContentDispositionHeader, QVariant("form-data; name=\"size\""));
  97. sizePart.setBody(QString::number(fileSize).toUtf8());
  98. multiPart->append(sizePart);
  99. // 创建网络请求
  100. QNetworkRequest request(urlStr); // 替换为实际的服务器地址
  101. request.setHeader(QNetworkRequest::ContentTypeHeader, "multipart/form-data; boundary=" + multiPart->boundary());
  102. // 发送请求
  103. // QNetworkAccessManager *manager = new QNetworkAccessManager(this);
  104. QNetworkReply *reply = m_manager.post(request, multiPart);
  105. // 连接信号槽
  106. connect(reply, &QNetworkReply::finished, this, [=]() {
  107. if (reply->error() == QNetworkReply::NoError) {
  108. qDebug() << "上传成功:";
  109. QJsonDocument jsonDoc = QJsonDocument::fromJson(reply->readAll());
  110. QJsonObject jsonObj = jsonDoc.object();
  111. QString fileUrl = jsonObj["url"].toString();
  112. ImageManager::instance()->addImageUrl(m_domain, m_userName, fileUrl);
  113. emit signal_uploadFileSec(fileUrl);
  114. } else {
  115. qDebug() << "上传失败:" << reply->errorString();
  116. }
  117. reply->deleteLater();
  118. multiPart->deleteLater();
  119. });
  120. connect(reply, &QNetworkReply::uploadProgress, this, [this](qint64 bytesSent, qint64 bytesTotal) {
  121. if (bytesTotal > 0) {
  122. int progress = (bytesSent * 100) / bytesTotal;
  123. emit signal_progressUpdate(progress);
  124. qDebug() << "上传进度:" << progress << "%";
  125. }
  126. });
  127. }
  128. void TCHttpService::apiSharePicShare(const QString &fileName, const QString &md5)
  129. {
  130. QString urlStr;
  131. if (m_enableSsl)
  132. urlStr = "https://" + m_domain + "/api/sharepic";
  133. else
  134. urlStr = "http://" + m_domain + "/api/sharepic";
  135. QMap<QString, QString> headers;
  136. QMap<QString, QString> body;
  137. QMap<QString, QString> params;
  138. headers.insert("Content-Type", "application/json");
  139. QString bodyStr = QString("\"%1\":\"%2\",\"%3\":\"%4\",\"%5\":\"%6\",\"%7\":\"%8\"").arg("token").arg(m_token).
  140. arg("user").arg(m_userName).arg("md5").arg(md5).arg("filename").arg(fileName);
  141. params.insert("cmd", "share");
  142. }
  143. void TCHttpService::setConfiguration(QString userName, QString firstPwd, QString domain)
  144. {
  145. this->m_domain = domain;
  146. this->m_userName = userName;
  147. this->m_firstPwd = firstPwd;
  148. apiLogin();
  149. }
  150. void TCHttpService::setSsl(bool enable)
  151. {
  152. this->m_enableSsl = enable;
  153. }
  154. bool TCHttpService::getOnlineState()
  155. {
  156. return m_isOnline;
  157. }
  158. void TCHttpService::setUploadNum(int nb)
  159. {
  160. this->m_uploadNum = nb;
  161. }
  162. void TCHttpService::downloadImage(const QString &requestId, const QUrl &imageUrl)
  163. {
  164. #if 0
  165. QPixmap cachedPixmap;
  166. if (QPixmapCache::find(imageUrl.toString(), &cachedPixmap)) {
  167. emit signal_imageDownloaded(requestId, cachedPixmap);
  168. return;
  169. }
  170. if (getCachedImage(imageUrl, cachedPixmap)) {
  171. QPixmapCache::insert(imageUrl.toString(), cachedPixmap);
  172. emit signal_imageDownloaded(requestId, cachedPixmap);
  173. return;
  174. }
  175. #endif
  176. QNetworkRequest request(imageUrl);
  177. QNetworkReply* reply = m_manager.get(request);
  178. connect(reply, &QNetworkReply::finished, [this, reply](){
  179. QString requestId = m_pendingRequests.take(reply);
  180. if (reply->error() != QNetworkReply::NoError) {
  181. emit signal_downloadFailed(requestId, reply->errorString());
  182. }
  183. else {
  184. QPixmap pixmap;
  185. if (pixmap.loadFromData(reply->readAll())) {
  186. emit signal_imageDownloaded(requestId, pixmap);
  187. }
  188. else {
  189. emit signal_downloadFailed(requestId, "Failed to load image data");
  190. }
  191. }
  192. reply->deleteLater();
  193. });
  194. m_pendingRequests[reply] = requestId;
  195. }
  196. QUrl TCHttpService::encodeUrl(QString urlStr, QMap<QString, QString> params)
  197. {
  198. QUrlQuery query;
  199. for(auto ite = params.constBegin(); ite != params.constEnd(); ite++){
  200. query.addQueryItem(ite.key(), ite.value());
  201. }
  202. QUrl url(urlStr);
  203. url.setQuery(query);
  204. return url;
  205. }
  206. void TCHttpService::cleanOldCacheFiles()
  207. {
  208. QDir cacheDir(QStandardPaths::writableLocation(QStandardPaths::CacheLocation));
  209. cacheDir.setNameFilters(QStringList() << "*.png" << "*.jpg" << "*.jpeg" << "*.svg");
  210. cacheDir.setFilter(QDir::Files);
  211. foreach (QString dirFile, cacheDir.entryList()) {
  212. QFileInfo fileInfo(cacheDir.absoluteFilePath(dirFile));
  213. if (fileInfo.lastModified().daysTo(QDateTime::currentDateTime()) > m_max_cache_days) {
  214. cacheDir.remove(dirFile);
  215. }
  216. }
  217. qint64 totalSize = 0;
  218. QFileInfoList files = cacheDir.entryInfoList(QDir::Files, QDir::Time | QDir::Reversed);
  219. foreach(QFileInfo file, files) {
  220. totalSize += file.size();
  221. }
  222. if (totalSize > m_disk_cache_size_mb * 1024 * 1024) {
  223. foreach (QFileInfo file, files) {
  224. if (totalSize <= m_disk_cache_size_mb * 1024 * 1024 * 0.9)
  225. break;
  226. totalSize -= file.size();
  227. QFile::remove(file.absoluteFilePath());
  228. }
  229. }
  230. }
  231. bool TCHttpService::getCachedImage(const QUrl &imageUrl, QPixmap &pixmap)
  232. {
  233. QString filePath = getCacheFilePath(imageUrl);
  234. if (QFile::exists(filePath)) {
  235. QFileInfo info(filePath);
  236. if (info.lastModified().daysTo(QDateTime::currentDateTime()) > m_max_cache_days) {
  237. QFile::remove(filePath);
  238. return false;
  239. }
  240. if (pixmap.load(filePath)) {
  241. return true;
  242. }
  243. }
  244. return false;
  245. }
  246. QString TCHttpService::getCacheFilePath(const QUrl &imageUrl)
  247. {
  248. QCryptographicHash hash(QCryptographicHash::Md5);
  249. hash.addData(imageUrl.toString().toUtf8());
  250. QString fileName = hash.result().toHex() + ".png";
  251. return QStandardPaths::writableLocation(QStandardPaths::CacheLocation) + "/" + fileName;
  252. }
  253. TCHttpService::TCHttpService(QObject *parent) : QObject(parent)
  254. {
  255. }
  256. void TCHttpService::slot_downloadFinished(QNetworkReply *reply)
  257. {
  258. QString requestId = m_pendingRequests.take(reply);
  259. if (reply->error() != QNetworkReply::NoError) {
  260. emit signal_downloadFailed(requestId, reply->errorString());
  261. }
  262. else {
  263. QPixmap pixmap;
  264. if (pixmap.loadFromData(reply->readAll())) {
  265. emit signal_imageDownloaded(requestId, pixmap);
  266. }
  267. else {
  268. emit signal_downloadFailed(requestId, "Failed to load image data");
  269. }
  270. }
  271. reply->deleteLater();
  272. }