mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-25 17:51:10 -06:00 
			
		
		
		
	Merged PR #4345 - support for DuetSoftwareSupport print host (thanks @wilriker)
This commit is contained in:
		
						commit
						81ce368e5d
					
				
					 4 changed files with 148 additions and 53 deletions
				
			
		|  | @ -36,12 +36,10 @@ const char* Duet::get_name() const { return "Duet"; } | ||||||
| 
 | 
 | ||||||
| bool Duet::test(wxString &msg) const | bool Duet::test(wxString &msg) const | ||||||
| { | { | ||||||
| 	bool connected = connect(msg); | 	auto connectionType = connect(msg); | ||||||
| 	if (connected) { | 	disconnect(connectionType); | ||||||
| 		disconnect(); |  | ||||||
| 	} |  | ||||||
| 
 | 
 | ||||||
| 	return connected; | 	return connectionType != ConnectionType::error; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| wxString Duet::get_test_ok_msg () const | wxString Duet::get_test_ok_msg () const | ||||||
|  | @ -59,33 +57,39 @@ wxString Duet::get_test_failed_msg (wxString &msg) const | ||||||
| bool Duet::upload(PrintHostUpload upload_data, ProgressFn prorgess_fn, ErrorFn error_fn) const | bool Duet::upload(PrintHostUpload upload_data, ProgressFn prorgess_fn, ErrorFn error_fn) const | ||||||
| { | { | ||||||
| 	wxString connect_msg; | 	wxString connect_msg; | ||||||
| 	if (!connect(connect_msg)) { | 	auto connectionType = connect(connect_msg); | ||||||
|  | 	if (connectionType == ConnectionType::error) { | ||||||
| 		error_fn(std::move(connect_msg)); | 		error_fn(std::move(connect_msg)); | ||||||
| 		return false; | 		return false; | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	bool res = true; | 	bool res = true; | ||||||
|  | 	bool dsf = (connectionType == ConnectionType::dsf); | ||||||
| 
 | 
 | ||||||
| 	auto upload_cmd = get_upload_url(upload_data.upload_path.string()); | 	auto upload_cmd = get_upload_url(upload_data.upload_path.string(), connectionType); | ||||||
| 	BOOST_LOG_TRIVIAL(info) << boost::format("Duet: Uploading file %1%, filepath: %2%, print: %3%, command: %4%") | 	BOOST_LOG_TRIVIAL(info) << boost::format("Duet: Uploading file %1%, filepath: %2%, print: %3%, command: %4%") | ||||||
| 		% upload_data.source_path | 		% upload_data.source_path | ||||||
| 		% upload_data.upload_path | 		% upload_data.upload_path | ||||||
| 		% upload_data.start_print | 		% upload_data.start_print | ||||||
| 		% upload_cmd; | 		% upload_cmd; | ||||||
| 
 | 
 | ||||||
| 	auto http = Http::post(std::move(upload_cmd)); | 	auto http = (dsf ? Http::put(std::move(upload_cmd)) : Http::post(std::move(upload_cmd))); | ||||||
| 	http.set_post_body(upload_data.source_path) | 	if (dsf) { | ||||||
| 		.on_complete([&](std::string body, unsigned status) { | 		http.set_put_body(upload_data.source_path); | ||||||
|  | 	} else { | ||||||
|  | 		http.set_post_body(upload_data.source_path); | ||||||
|  | 	} | ||||||
|  | 	http.on_complete([&](std::string body, unsigned status) { | ||||||
| 			BOOST_LOG_TRIVIAL(debug) << boost::format("Duet: File uploaded: HTTP %1%: %2%") % status % body; | 			BOOST_LOG_TRIVIAL(debug) << boost::format("Duet: File uploaded: HTTP %1%: %2%") % status % body; | ||||||
| 
 | 
 | ||||||
| 			int err_code = get_err_code_from_body(body); | 			int err_code = dsf ? (status == 201 ? 0 : 1) : get_err_code_from_body(body); | ||||||
| 			if (err_code != 0) { | 			if (err_code != 0) { | ||||||
| 				BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Request completed but error code was received: %1%") % err_code; | 				BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Request completed but error code was received: %1%") % err_code; | ||||||
| 				error_fn(format_error(body, L("Unknown error occured"), 0)); | 				error_fn(format_error(body, L("Unknown error occured"), 0)); | ||||||
| 				res = false; | 				res = false; | ||||||
| 			} else if (upload_data.start_print) { | 			} else if (upload_data.start_print) { | ||||||
| 				wxString errormsg; | 				wxString errormsg; | ||||||
| 				res = start_print(errormsg, upload_data.upload_path.string()); | 				res = start_print(errormsg, upload_data.upload_path.string(), connectionType); | ||||||
| 				if (! res) { | 				if (! res) { | ||||||
| 					error_fn(std::move(errormsg)); | 					error_fn(std::move(errormsg)); | ||||||
| 				} | 				} | ||||||
|  | @ -106,20 +110,28 @@ bool Duet::upload(PrintHostUpload upload_data, ProgressFn prorgess_fn, ErrorFn e | ||||||
| 		}) | 		}) | ||||||
| 		.perform_sync(); | 		.perform_sync(); | ||||||
| 
 | 
 | ||||||
| 	disconnect(); | 	disconnect(connectionType); | ||||||
| 
 | 
 | ||||||
| 	return res; | 	return res; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool Duet::connect(wxString &msg) const | Duet::ConnectionType Duet::connect(wxString &msg) const | ||||||
| { | { | ||||||
| 	bool res = false; | 	auto res = ConnectionType::error; | ||||||
| 	auto url = get_connect_url(); | 	auto url = get_connect_url(false); | ||||||
| 
 | 
 | ||||||
| 	auto http = Http::get(std::move(url)); | 	auto http = Http::get(std::move(url)); | ||||||
| 	http.on_error([&](std::string body, std::string error, unsigned status) { | 	http.on_error([&](std::string body, std::string error, unsigned status) { | ||||||
| 			BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Error connecting: %1%, HTTP %2%, body: `%3%`") % error % status % body; | 			auto dsfUrl = get_connect_url(true); | ||||||
| 			msg = format_error(body, error, status); | 			auto dsfHttp = Http::get(std::move(dsfUrl)); | ||||||
|  | 			dsfHttp.on_error([&](std::string body, std::string error, unsigned status) { | ||||||
|  | 					BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Error connecting: %1%, HTTP %2%, body: `%3%`") % error % status % body; | ||||||
|  | 					msg = format_error(body, error, status); | ||||||
|  | 				}) | ||||||
|  | 				.on_complete([&](std::string body, unsigned) { | ||||||
|  | 					res = ConnectionType::dsf; | ||||||
|  | 				}) | ||||||
|  | 				.perform_sync(); | ||||||
| 		}) | 		}) | ||||||
| 		.on_complete([&](std::string body, unsigned) { | 		.on_complete([&](std::string body, unsigned) { | ||||||
| 			BOOST_LOG_TRIVIAL(debug) << boost::format("Duet: Got: %1%") % body; | 			BOOST_LOG_TRIVIAL(debug) << boost::format("Duet: Got: %1%") % body; | ||||||
|  | @ -127,7 +139,7 @@ bool Duet::connect(wxString &msg) const | ||||||
| 			int err_code = get_err_code_from_body(body); | 			int err_code = get_err_code_from_body(body); | ||||||
| 			switch (err_code) { | 			switch (err_code) { | ||||||
| 				case 0: | 				case 0: | ||||||
| 					res = true; | 					res = ConnectionType::rrf; | ||||||
| 					break; | 					break; | ||||||
| 				case 1: | 				case 1: | ||||||
| 					msg = format_error(body, L("Wrong password"), 0); | 					msg = format_error(body, L("Wrong password"), 0); | ||||||
|  | @ -146,8 +158,12 @@ bool Duet::connect(wxString &msg) const | ||||||
| 	return res; | 	return res; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void Duet::disconnect() const | void Duet::disconnect(ConnectionType connectionType) const | ||||||
| { | { | ||||||
|  | 	// we don't need to disconnect from DSF or if it failed anyway
 | ||||||
|  | 	if (connectionType != ConnectionType::rrf) { | ||||||
|  | 		return; | ||||||
|  | 	} | ||||||
| 	auto url =  (boost::format("%1%rr_disconnect") | 	auto url =  (boost::format("%1%rr_disconnect") | ||||||
| 			% get_base_url()).str(); | 			% get_base_url()).str(); | ||||||
| 
 | 
 | ||||||
|  | @ -159,20 +175,33 @@ void Duet::disconnect() const | ||||||
| 	.perform_sync(); | 	.perform_sync(); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::string Duet::get_upload_url(const std::string &filename) const | std::string Duet::get_upload_url(const std::string &filename, ConnectionType connectionType) const | ||||||
| { | { | ||||||
| 	return (boost::format("%1%rr_upload?name=0:/gcodes/%2%&%3%") |     assert(connectionType != ConnectionType::error); | ||||||
| 			% get_base_url() | 
 | ||||||
| 			% Http::url_encode(filename)  | 	if (connectionType == ConnectionType::dsf) { | ||||||
| 			% timestamp_str()).str(); | 		return (boost::format("%1%machine/file/gcodes/%2%") | ||||||
|  | 				% get_base_url() | ||||||
|  | 				% Http::url_encode(filename)).str(); | ||||||
|  | 	} else { | ||||||
|  | 		return (boost::format("%1%rr_upload?name=0:/gcodes/%2%&%3%") | ||||||
|  | 				% get_base_url() | ||||||
|  | 				% Http::url_encode(filename) | ||||||
|  | 				% timestamp_str()).str(); | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::string Duet::get_connect_url() const | std::string Duet::get_connect_url(const bool dsfUrl) const | ||||||
| { | { | ||||||
| 	return (boost::format("%1%rr_connect?password=%2%&%3%") | 	if (dsfUrl)	{ | ||||||
| 			% get_base_url() | 		return (boost::format("%1%machine/status") | ||||||
| 			% (password.empty() ? "reprap" : password) | 				% get_base_url()).str(); | ||||||
| 			% timestamp_str()).str(); | 	} else { | ||||||
|  | 		return (boost::format("%1%rr_connect?password=%2%&%3%") | ||||||
|  | 				% get_base_url() | ||||||
|  | 				% (password.empty() ? "reprap" : password) | ||||||
|  | 				% timestamp_str()).str(); | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| std::string Duet::get_base_url() const | std::string Duet::get_base_url() const | ||||||
|  | @ -201,15 +230,27 @@ std::string Duet::timestamp_str() const | ||||||
| 	return std::string(buffer); | 	return std::string(buffer); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| bool Duet::start_print(wxString &msg, const std::string &filename) const  | bool Duet::start_print(wxString &msg, const std::string &filename, ConnectionType connectionType) const | ||||||
| { | { | ||||||
| 	bool res = false; |     assert(connectionType != ConnectionType::error); | ||||||
| 
 | 
 | ||||||
| 	auto url = (boost::format("%1%rr_gcode?gcode=M32%%20\"%2%\"") | 	bool res = false; | ||||||
|  | 	bool dsf = (connectionType == ConnectionType::dsf); | ||||||
|  | 
 | ||||||
|  | 	auto url = dsf | ||||||
|  | 		? (boost::format("%1%machine/code") | ||||||
|  | 			% get_base_url()).str() | ||||||
|  | 		: (boost::format("%1%rr_gcode?gcode=M32%%20\"0:/gcodes/%2%\"") | ||||||
| 			% get_base_url() | 			% get_base_url() | ||||||
| 			% Http::url_encode(filename)).str(); | 			% Http::url_encode(filename)).str(); | ||||||
| 
 | 
 | ||||||
| 	auto http = Http::get(std::move(url)); | 	auto http = (dsf ? Http::post(std::move(url)) : Http::get(std::move(url))); | ||||||
|  | 	if (dsf) { | ||||||
|  | 		http.set_post_body( | ||||||
|  | 				(boost::format("M32 \"0:/gcodes/%1%\"") | ||||||
|  | 					% filename).str() | ||||||
|  | 				); | ||||||
|  | 	} | ||||||
| 	http.on_error([&](std::string body, std::string error, unsigned status) { | 	http.on_error([&](std::string body, std::string error, unsigned status) { | ||||||
| 			BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Error starting print: %1%, HTTP %2%, body: `%3%`") % error % status % body; | 			BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Error starting print: %1%, HTTP %2%, body: `%3%`") % error % status % body; | ||||||
| 			msg = format_error(body, error, status); | 			msg = format_error(body, error, status); | ||||||
|  |  | ||||||
|  | @ -14,7 +14,7 @@ class Http; | ||||||
| class Duet : public PrintHost | class Duet : public PrintHost | ||||||
| { | { | ||||||
| public: | public: | ||||||
| 	Duet(DynamicPrintConfig *config); |     explicit Duet(DynamicPrintConfig *config); | ||||||
| 	~Duet() override = default; | 	~Duet() override = default; | ||||||
| 
 | 
 | ||||||
| 	const char* get_name() const override; | 	const char* get_name() const override; | ||||||
|  | @ -29,16 +29,17 @@ public: | ||||||
| 	std::string get_host() const override { return host; } | 	std::string get_host() const override { return host; } | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
|  | 	enum class ConnectionType { rrf, dsf, error }; | ||||||
| 	std::string host; | 	std::string host; | ||||||
| 	std::string password; | 	std::string password; | ||||||
| 
 | 
 | ||||||
| 	std::string get_upload_url(const std::string &filename) const; | 	std::string get_upload_url(const std::string &filename, ConnectionType connectionType) const; | ||||||
| 	std::string get_connect_url() const; | 	std::string get_connect_url(const bool dsfUrl) const; | ||||||
| 	std::string get_base_url() const; | 	std::string get_base_url() const; | ||||||
| 	std::string timestamp_str() const; | 	std::string timestamp_str() const; | ||||||
| 	bool connect(wxString &msg) const; | 	ConnectionType connect(wxString &msg) const; | ||||||
| 	void disconnect() const; | 	void disconnect(ConnectionType connectionType) const; | ||||||
| 	bool start_print(wxString &msg, const std::string &filename) const; | 	bool start_print(wxString &msg, const std::string &filename, ConnectionType connectionType) const; | ||||||
| 	int get_err_code_from_body(const std::string &body) const; | 	int get_err_code_from_body(const std::string &body) const; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -120,6 +120,7 @@ struct Http::priv | ||||||
| 	std::string error_buffer;    // Used for CURLOPT_ERRORBUFFER
 | 	std::string error_buffer;    // Used for CURLOPT_ERRORBUFFER
 | ||||||
| 	size_t limit; | 	size_t limit; | ||||||
| 	bool cancel; | 	bool cancel; | ||||||
|  |     std::unique_ptr<fs::ifstream> putFile; | ||||||
| 
 | 
 | ||||||
| 	std::thread io_thread; | 	std::thread io_thread; | ||||||
| 	Http::CompleteFn completefn; | 	Http::CompleteFn completefn; | ||||||
|  | @ -138,6 +139,8 @@ struct Http::priv | ||||||
| 	void set_timeout_connect(long timeout); | 	void set_timeout_connect(long timeout); | ||||||
| 	void form_add_file(const char *name, const fs::path &path, const char* filename); | 	void form_add_file(const char *name, const fs::path &path, const char* filename); | ||||||
| 	void set_post_body(const fs::path &path); | 	void set_post_body(const fs::path &path); | ||||||
|  | 	void set_post_body(const std::string body); | ||||||
|  | 	void set_put_body(const fs::path &path); | ||||||
| 
 | 
 | ||||||
| 	std::string curl_error(CURLcode curlcode); | 	std::string curl_error(CURLcode curlcode); | ||||||
| 	std::string body_size_error(); | 	std::string body_size_error(); | ||||||
|  | @ -152,6 +155,7 @@ Http::priv::priv(const std::string &url) | ||||||
| 	, error_buffer(CURL_ERROR_SIZE + 1, '\0') | 	, error_buffer(CURL_ERROR_SIZE + 1, '\0') | ||||||
| 	, limit(0) | 	, limit(0) | ||||||
| 	, cancel(false) | 	, cancel(false) | ||||||
|  | 	, putFile(nullptr) | ||||||
| { | { | ||||||
|     Http::tls_global_init(); |     Http::tls_global_init(); | ||||||
| 
 | 
 | ||||||
|  | @ -284,6 +288,22 @@ void Http::priv::set_post_body(const fs::path &path) | ||||||
| 	postfields = file_content; | 	postfields = file_content; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | void Http::priv::set_post_body(const std::string body) | ||||||
|  | { | ||||||
|  | 	postfields = body; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | void Http::priv::set_put_body(const fs::path &path) | ||||||
|  | { | ||||||
|  | 	boost::system::error_code ec; | ||||||
|  | 	boost::uintmax_t filesize = file_size(path, ec); | ||||||
|  | 	if (!ec) { | ||||||
|  |         putFile = std::make_unique<fs::ifstream>(path); | ||||||
|  |         ::curl_easy_setopt(curl, CURLOPT_READDATA, (void *) (putFile.get())); | ||||||
|  | 		::curl_easy_setopt(curl, CURLOPT_INFILESIZE, filesize); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| std::string Http::priv::curl_error(CURLcode curlcode) | std::string Http::priv::curl_error(CURLcode curlcode) | ||||||
| { | { | ||||||
| 	return (boost::format("%1%:\n%2%\n[Error %3%]") | 	return (boost::format("%1%:\n%2%\n[Error %3%]") | ||||||
|  | @ -335,6 +355,8 @@ void Http::priv::http_perform() | ||||||
| 
 | 
 | ||||||
| 	CURLcode res = ::curl_easy_perform(curl); | 	CURLcode res = ::curl_easy_perform(curl); | ||||||
| 
 | 
 | ||||||
|  |     putFile.reset(); | ||||||
|  | 
 | ||||||
| 	if (res != CURLE_OK) { | 	if (res != CURLE_OK) { | ||||||
| 		if (res == CURLE_ABORTED_BY_CALLBACK) { | 		if (res == CURLE_ABORTED_BY_CALLBACK) { | ||||||
| 			if (cancel) { | 			if (cancel) { | ||||||
|  | @ -373,6 +395,7 @@ Http::Http(Http &&other) : p(std::move(other.p)) {} | ||||||
| 
 | 
 | ||||||
| Http::~Http() | Http::~Http() | ||||||
| { | { | ||||||
|  |     assert(! putFile); | ||||||
| 	if (p && p->io_thread.joinable()) { | 	if (p && p->io_thread.joinable()) { | ||||||
| 		p->io_thread.detach(); | 		p->io_thread.detach(); | ||||||
| 	} | 	} | ||||||
|  | @ -465,6 +488,18 @@ Http& Http::set_post_body(const fs::path &path) | ||||||
| 	return *this; | 	return *this; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | Http& Http::set_post_body(const std::string body) | ||||||
|  | { | ||||||
|  | 	if (p) { p->set_post_body(body); } | ||||||
|  | 	return *this; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | Http& Http::set_put_body(const fs::path &path) | ||||||
|  | { | ||||||
|  | 	if (p) { p->set_put_body(path);} | ||||||
|  | 	return *this; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| Http& Http::on_complete(CompleteFn fn) | Http& Http::on_complete(CompleteFn fn) | ||||||
| { | { | ||||||
| 	if (p) { p->completefn = std::move(fn); } | 	if (p) { p->completefn = std::move(fn); } | ||||||
|  | @ -519,6 +554,13 @@ Http Http::post(std::string url) | ||||||
| 	return http; | 	return http; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | Http Http::put(std::string url) | ||||||
|  | { | ||||||
|  | 	Http http{std::move(url)}; | ||||||
|  | 	curl_easy_setopt(http.p->curl, CURLOPT_UPLOAD, 1L); | ||||||
|  | 	return http; | ||||||
|  | } | ||||||
|  | 
 | ||||||
| bool Http::ca_file_supported() | bool Http::ca_file_supported() | ||||||
| { | { | ||||||
| 	::CURL *curl = ::curl_easy_init(); | 	::CURL *curl = ::curl_easy_init(); | ||||||
|  |  | ||||||
|  | @ -49,6 +49,7 @@ public: | ||||||
| 	// for a GET and a POST request respectively.
 | 	// for a GET and a POST request respectively.
 | ||||||
| 	static Http get(std::string url); | 	static Http get(std::string url); | ||||||
| 	static Http post(std::string url); | 	static Http post(std::string url); | ||||||
|  | 	static Http put(std::string url); | ||||||
| 	~Http(); | 	~Http(); | ||||||
| 
 | 
 | ||||||
| 	Http(const Http &) = delete; | 	Http(const Http &) = delete; | ||||||
|  | @ -82,6 +83,16 @@ public: | ||||||
| 	// This can be used for hosts which do not support multipart requests.
 | 	// This can be used for hosts which do not support multipart requests.
 | ||||||
| 	Http& set_post_body(const boost::filesystem::path &path); | 	Http& set_post_body(const boost::filesystem::path &path); | ||||||
| 
 | 
 | ||||||
|  | 	// Set the POST request body.
 | ||||||
|  | 	// The data is used verbatim, it is not additionally encoded in any way.
 | ||||||
|  | 	// This can be used for hosts which do not support multipart requests.
 | ||||||
|  | 	Http& set_post_body(const std::string body); | ||||||
|  | 
 | ||||||
|  | 	// Set the file contents as a PUT request body.
 | ||||||
|  | 	// The data is used verbatim, it is not additionally encoded in any way.
 | ||||||
|  | 	// This can be used for hosts which do not support multipart requests.
 | ||||||
|  | 	Http& set_put_body(const boost::filesystem::path &path); | ||||||
|  | 
 | ||||||
| 	// Callback called on HTTP request complete
 | 	// Callback called on HTTP request complete
 | ||||||
| 	Http& on_complete(CompleteFn fn); | 	Http& on_complete(CompleteFn fn); | ||||||
| 	// Callback called on an error occuring at any stage of the requests: Url parsing, DNS lookup,
 | 	// Callback called on an error occuring at any stage of the requests: Url parsing, DNS lookup,
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Lukas Matena
						Lukas Matena