#include "NuulsUploader.hpp" #include "common/Env.hpp" #include "common/NetworkRequest.hpp" #include "providers/twitch/TwitchMessageBuilder.hpp" #include #include #include #include #define UPLOAD_DELAY 2000 // Delay between uploads in milliseconds namespace { boost::optional convertToPng(QImage image) { QByteArray imageData; QBuffer buf(&imageData); buf.open(QIODevice::WriteOnly); bool success = image.save(&buf, "png"); if (success) { return boost::optional(imageData); } else { return boost::optional(boost::none); } } } // namespace namespace chatterino { // These variables are only used from the main thread. auto uploadMutex = QMutex(); std::queue uploadQueue; void uploadImageToNuuls(RawImageData imageData, ChannelPtr channel, ResizingTextEdit &textEdit) { const static char *boundary = "thisistheboudaryasd"; const static QString contentType = QString("multipart/form-data; boundary=%1").arg(boundary); static QUrl url(Env::get().imageUploaderUrl); QHttpMultiPart *payload = new QHttpMultiPart(QHttpMultiPart::FormDataType); QHttpPart part = QHttpPart(); part.setBody(imageData.data); part.setHeader(QNetworkRequest::ContentTypeHeader, QString("image/%1").arg(imageData.format)); part.setHeader(QNetworkRequest::ContentLengthHeader, QVariant(imageData.data.length())); part.setHeader( QNetworkRequest::ContentDispositionHeader, QString("form-data; name=\"attachment\"; filename=\"control_v.%1\"") .arg(imageData.format)); payload->setBoundary(boundary); payload->append(part); NetworkRequest(url, NetworkRequestType::Post) .header("Content-Type", contentType) .multiPart(payload) .onSuccess([&textEdit, channel](NetworkResult result) -> Outcome { textEdit.insertPlainText(result.getData() + QString(" ")); if (uploadQueue.empty()) { channel->addMessage(makeSystemMessage( QString("Your image has been uploaded."))); uploadMutex.unlock(); } else { channel->addMessage(makeSystemMessage( QString("Your image has been uploaded. %1 left. Please " "wait until all of them are uploaded. About %2 " "seconds left.") .arg(uploadQueue.size()) .arg(uploadQueue.size() * (UPLOAD_DELAY / 1000 + 1)))); // 2 seconds for the timer that's there not to spam the remote server // and 1 second of actual uploading. QTimer::singleShot(UPLOAD_DELAY, [channel, &textEdit]() { uploadImageToNuuls(uploadQueue.front(), channel, textEdit); uploadQueue.pop(); }); } return Success; }) .onError([channel](NetworkResult result) -> bool { channel->addMessage(makeSystemMessage( QString("An error happened while uploading your image: %1") .arg(result.status()))); uploadMutex.unlock(); return true; }) .execute(); } void upload(const QMimeData *source, ChannelPtr channel, ResizingTextEdit &outputTextEdit) { if (!uploadMutex.tryLock()) { channel->addMessage(makeSystemMessage( QString("Please wait until the upload finishes."))); return; } channel->addMessage(makeSystemMessage(QString("Started upload..."))); if (source->hasFormat("image/png")) { uploadImageToNuuls({source->data("image/png"), "png"}, channel, outputTextEdit); } else if (source->hasFormat("image/jpeg")) { uploadImageToNuuls({source->data("image/jpeg"), "jpeg"}, channel, outputTextEdit); } else if (source->hasFormat("image/gif")) { uploadImageToNuuls({source->data("image/gif"), "gif"}, channel, outputTextEdit); } else if (source->hasUrls()) { auto mimeDb = QMimeDatabase(); // This path gets chosen when files are copied from a file manager, like explorer.exe, caja. // Each entry in source->urls() is a QUrl pointing to a file that was copied. for (const QUrl &path : source->urls()) { QString localPath = path.toLocalFile(); QMimeType mime = mimeDb.mimeTypeForUrl(path); qDebug() << mime.name(); if (mime.name().startsWith("image") && !mime.inherits("image/gif")) { channel->addMessage(makeSystemMessage( QString("Uploading image: %1").arg(localPath))); QImage img = QImage(localPath); if (img.isNull()) { channel->addMessage( makeSystemMessage(QString("Couldn't load image :("))); uploadMutex.unlock(); return; } boost::optional imageData = convertToPng(img); if (imageData) { RawImageData data = {imageData.get(), "png"}; uploadQueue.push(data); } else { channel->addMessage(makeSystemMessage( QString("Cannot upload file: %1, Couldn't convert " "image to png.") .arg(localPath))); uploadMutex.unlock(); return; } } else if (mime.inherits("image/gif")) { channel->addMessage(makeSystemMessage( QString("Uploading GIF: %1").arg(localPath))); QFile file(localPath); bool isOkay = file.open(QIODevice::ReadOnly); if (!isOkay) { channel->addMessage( makeSystemMessage(QString("Failed to open file. :("))); uploadMutex.unlock(); return; } RawImageData data = {file.readAll(), "gif"}; uploadQueue.push(data); file.close(); // file.readAll() => might be a bit big but it /should/ work } else { channel->addMessage(makeSystemMessage( QString("Cannot upload file: %1, not an image") .arg(localPath))); uploadMutex.unlock(); return; } } if (!uploadQueue.empty()) { uploadImageToNuuls(uploadQueue.front(), channel, outputTextEdit); uploadQueue.pop(); } } else { // not PNG, try loading it into QImage and save it to a PNG. QImage image = qvariant_cast(source->imageData()); boost::optional imageData = convertToPng(image); if (imageData) { uploadImageToNuuls({imageData.get(), "png"}, channel, outputTextEdit); } else { channel->addMessage(makeSystemMessage( QString("Cannot upload file, failed to convert to png."))); uploadMutex.unlock(); } } } } // namespace chatterino