262 lines
6.9 KiB
C++
262 lines
6.9 KiB
C++
#include "PrintHost.hpp"
|
|
|
|
#include <vector>
|
|
#include <thread>
|
|
#include <exception>
|
|
#include <boost/optional.hpp>
|
|
#include <boost/log/trivial.hpp>
|
|
#include <boost/filesystem.hpp>
|
|
|
|
#include <wx/string.h>
|
|
#include <wx/app.h>
|
|
|
|
#include "libslic3r/PrintConfig.hpp"
|
|
#include "libslic3r/Channel.hpp"
|
|
#include "OctoPrint.hpp"
|
|
#include "Duet.hpp"
|
|
#include "../GUI/PrintHostDialogs.hpp"
|
|
|
|
namespace fs = boost::filesystem;
|
|
using boost::optional;
|
|
using Slic3r::GUI::PrintHostQueueDialog;
|
|
|
|
namespace Slic3r {
|
|
|
|
|
|
PrintHost::~PrintHost() {}
|
|
|
|
PrintHost* PrintHost::get_print_host(DynamicPrintConfig *config)
|
|
{
|
|
const auto opt = config->option<ConfigOptionEnum<PrintHostType>>("host_type");
|
|
if (opt == nullptr) { return nullptr; }
|
|
|
|
switch (opt->value) {
|
|
case htOctoPrint: return new OctoPrint(config);
|
|
case htDuet: return new Duet(config);
|
|
case htSL1: return new SLAHost(config);
|
|
default: return nullptr;
|
|
}
|
|
}
|
|
|
|
wxString PrintHost::format_error(const std::string &body, const std::string &error, unsigned status) const
|
|
{
|
|
if (status != 0) {
|
|
auto wxbody = wxString::FromUTF8(body.data());
|
|
return wxString::Format("HTTP %u: %s", status, wxbody);
|
|
} else {
|
|
return wxString::FromUTF8(error.data());
|
|
}
|
|
}
|
|
|
|
|
|
struct PrintHostJobQueue::priv
|
|
{
|
|
// XXX: comment on how bg thread works
|
|
|
|
PrintHostJobQueue *q;
|
|
|
|
Channel<PrintHostJob> channel_jobs;
|
|
Channel<size_t> channel_cancels;
|
|
size_t job_id = 0;
|
|
int prev_progress = -1;
|
|
fs::path source_to_remove;
|
|
|
|
std::thread bg_thread;
|
|
bool bg_exit = false;
|
|
|
|
PrintHostQueueDialog *queue_dialog;
|
|
|
|
priv(PrintHostJobQueue *q) : q(q) {}
|
|
|
|
void emit_progress(int progress);
|
|
void emit_error(wxString error);
|
|
void emit_cancel(size_t id);
|
|
void start_bg_thread();
|
|
void stop_bg_thread();
|
|
void bg_thread_main();
|
|
void progress_fn(Http::Progress progress, bool &cancel);
|
|
void remove_source(const fs::path &path);
|
|
void remove_source();
|
|
void perform_job(PrintHostJob the_job);
|
|
};
|
|
|
|
PrintHostJobQueue::PrintHostJobQueue(PrintHostQueueDialog *queue_dialog)
|
|
: p(new priv(this))
|
|
{
|
|
p->queue_dialog = queue_dialog;
|
|
}
|
|
|
|
PrintHostJobQueue::~PrintHostJobQueue()
|
|
{
|
|
if (p) { p->stop_bg_thread(); }
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::emit_progress(int progress)
|
|
{
|
|
auto evt = new PrintHostQueueDialog::Event(GUI::EVT_PRINTHOST_PROGRESS, queue_dialog->GetId(), job_id, progress);
|
|
wxQueueEvent(queue_dialog, evt);
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::emit_error(wxString error)
|
|
{
|
|
auto evt = new PrintHostQueueDialog::Event(GUI::EVT_PRINTHOST_ERROR, queue_dialog->GetId(), job_id, std::move(error));
|
|
wxQueueEvent(queue_dialog, evt);
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::emit_cancel(size_t id)
|
|
{
|
|
auto evt = new PrintHostQueueDialog::Event(GUI::EVT_PRINTHOST_CANCEL, queue_dialog->GetId(), id);
|
|
wxQueueEvent(queue_dialog, evt);
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::start_bg_thread()
|
|
{
|
|
if (bg_thread.joinable()) { return; }
|
|
|
|
std::shared_ptr<priv> p2 = q->p;
|
|
bg_thread = std::thread([p2]() {
|
|
p2->bg_thread_main();
|
|
});
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::stop_bg_thread()
|
|
{
|
|
if (bg_thread.joinable()) {
|
|
bg_exit = true;
|
|
channel_jobs.push(PrintHostJob()); // Push an empty job to wake up bg_thread in case it's sleeping
|
|
bg_thread.detach(); // Let the background thread go, it should exit on its own
|
|
}
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::bg_thread_main()
|
|
{
|
|
// bg thread entry point
|
|
|
|
try {
|
|
// Pick up jobs from the job channel:
|
|
while (! bg_exit) {
|
|
auto job = channel_jobs.pop(); // Sleeps in a cond var if there are no jobs
|
|
if (job.empty()) {
|
|
// This happens when the thread is being stopped
|
|
break;
|
|
}
|
|
|
|
source_to_remove = job.upload_data.source_path;
|
|
|
|
BOOST_LOG_TRIVIAL(debug) << boost::format("PrintHostJobQueue/bg_thread: Received job: [%1%]: `%2%` -> `%3%`, cancelled: %4%")
|
|
% job_id
|
|
% job.upload_data.upload_path
|
|
% job.printhost->get_host()
|
|
% job.cancelled;
|
|
|
|
if (! job.cancelled) {
|
|
perform_job(std::move(job));
|
|
}
|
|
|
|
remove_source();
|
|
job_id++;
|
|
}
|
|
} catch (const std::exception &e) {
|
|
emit_error(e.what());
|
|
} catch (...) {
|
|
emit_error("Unknown exception");
|
|
}
|
|
|
|
// Cleanup leftover files, if any
|
|
remove_source();
|
|
auto jobs = channel_jobs.lock_rw();
|
|
for (const PrintHostJob &job : *jobs) {
|
|
remove_source(job.upload_data.source_path);
|
|
}
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::progress_fn(Http::Progress progress, bool &cancel)
|
|
{
|
|
if (cancel) {
|
|
// When cancel is true from the start, Http indicates request has been cancelled
|
|
emit_cancel(job_id);
|
|
return;
|
|
}
|
|
|
|
if (bg_exit) {
|
|
cancel = true;
|
|
return;
|
|
}
|
|
|
|
if (channel_cancels.size_hint() > 0) {
|
|
// Lock both queues
|
|
auto cancels = channel_cancels.lock_rw();
|
|
auto jobs = channel_jobs.lock_rw();
|
|
|
|
for (size_t cancel_id : *cancels) {
|
|
if (cancel_id == job_id) {
|
|
cancel = true;
|
|
} else if (cancel_id > job_id) {
|
|
const size_t idx = cancel_id - job_id - 1;
|
|
if (idx < jobs->size()) {
|
|
jobs->at(idx).cancelled = true;
|
|
BOOST_LOG_TRIVIAL(debug) << boost::format("PrintHostJobQueue: Job id %1% cancelled") % cancel_id;
|
|
emit_cancel(cancel_id);
|
|
}
|
|
}
|
|
}
|
|
|
|
cancels->clear();
|
|
}
|
|
|
|
if (! cancel) {
|
|
int gui_progress = progress.ultotal > 0 ? 100*progress.ulnow / progress.ultotal : 0;
|
|
if (gui_progress != prev_progress) {
|
|
emit_progress(gui_progress);
|
|
prev_progress = gui_progress;
|
|
}
|
|
}
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::remove_source(const fs::path &path)
|
|
{
|
|
if (! path.empty()) {
|
|
boost::system::error_code ec;
|
|
fs::remove(path, ec);
|
|
if (ec) {
|
|
BOOST_LOG_TRIVIAL(error) << boost::format("PrintHostJobQueue: Error removing file `%1%`: %2%") % path % ec;
|
|
}
|
|
}
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::remove_source()
|
|
{
|
|
remove_source(source_to_remove);
|
|
source_to_remove.clear();
|
|
}
|
|
|
|
void PrintHostJobQueue::priv::perform_job(PrintHostJob the_job)
|
|
{
|
|
emit_progress(0); // Indicate the upload is starting
|
|
|
|
bool success = the_job.printhost->upload(std::move(the_job.upload_data),
|
|
[this](Http::Progress progress, bool &cancel) { this->progress_fn(std::move(progress), cancel); },
|
|
[this](wxString error) {
|
|
emit_error(std::move(error));
|
|
}
|
|
);
|
|
|
|
if (success) {
|
|
emit_progress(100);
|
|
}
|
|
}
|
|
|
|
void PrintHostJobQueue::enqueue(PrintHostJob job)
|
|
{
|
|
p->start_bg_thread();
|
|
p->queue_dialog->append_job(job);
|
|
p->channel_jobs.push(std::move(job));
|
|
}
|
|
|
|
void PrintHostJobQueue::cancel(size_t id)
|
|
{
|
|
p->channel_cancels.push(id);
|
|
}
|
|
|
|
|
|
}
|