mirror of
				https://github.com/SoftFever/OrcaSlicer.git
				synced 2025-10-24 09:11:23 -06:00 
			
		
		
		
	Support upload g-code to third-party printers (#2)
* wip * fix cmake warning * add UI and config options for Moonraker connection * wip: copy whole print host UI from PS * add more needed options * wip 2 * fix string issue on Mac * wip3
This commit is contained in:
		
							parent
							
								
									159ff850e4
								
							
						
					
					
						commit
						dc5e0eed34
					
				
					 44 changed files with 5543 additions and 19 deletions
				
			
		
							
								
								
									
										286
									
								
								src/slic3r/Utils/Duet.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										286
									
								
								src/slic3r/Utils/Duet.cpp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,286 @@ | |||
| #include "Duet.hpp" | ||||
| 
 | ||||
| #include <algorithm> | ||||
| #include <ctime> | ||||
| #include <boost/filesystem/path.hpp> | ||||
| #include <boost/format.hpp> | ||||
| #include <boost/log/trivial.hpp> | ||||
| #include <boost/property_tree/ptree.hpp> | ||||
| #include <boost/property_tree/json_parser.hpp> | ||||
| 
 | ||||
| #include <wx/frame.h> | ||||
| #include <wx/event.h> | ||||
| #include <wx/progdlg.h> | ||||
| #include <wx/sizer.h> | ||||
| #include <wx/stattext.h> | ||||
| #include <wx/textctrl.h> | ||||
| #include <wx/checkbox.h> | ||||
| 
 | ||||
| #include "libslic3r/PrintConfig.hpp" | ||||
| #include "slic3r/GUI/GUI.hpp" | ||||
| #include "slic3r/GUI/I18N.hpp" | ||||
| #include "slic3r/GUI/MsgDialog.hpp" | ||||
| #include "Http.hpp" | ||||
| 
 | ||||
| namespace fs = boost::filesystem; | ||||
| namespace pt = boost::property_tree; | ||||
| 
 | ||||
| namespace Slic3r { | ||||
| 
 | ||||
| Duet::Duet(DynamicPrintConfig *config) : | ||||
| 	host(config->opt_string("print_host")), | ||||
| 	password(config->opt_string("printhost_apikey")) | ||||
| {} | ||||
| 
 | ||||
| const char* Duet::get_name() const { return "Duet"; } | ||||
| 
 | ||||
| bool Duet::test(wxString &msg) const | ||||
| { | ||||
| 	auto connectionType = connect(msg); | ||||
| 	disconnect(connectionType); | ||||
| 
 | ||||
| 	return connectionType != ConnectionType::error; | ||||
| } | ||||
| 
 | ||||
| wxString Duet::get_test_ok_msg () const | ||||
| { | ||||
| 	return _(L("Connection to Duet works correctly.")); | ||||
| } | ||||
| 
 | ||||
| wxString Duet::get_test_failed_msg (wxString &msg) const | ||||
| { | ||||
|     return GUI::from_u8((boost::format("%s: %s") | ||||
|                     % _utf8(L("Could not connect to Duet")) | ||||
|                     % std::string(msg.ToUTF8())).str()); | ||||
| } | ||||
| 
 | ||||
| bool Duet::upload(PrintHostUpload upload_data, ProgressFn prorgess_fn, ErrorFn error_fn) const | ||||
| { | ||||
| 	wxString connect_msg; | ||||
| 	auto connectionType = connect(connect_msg); | ||||
| 	if (connectionType == ConnectionType::error) { | ||||
| 		error_fn(std::move(connect_msg)); | ||||
| 		return false; | ||||
| 	} | ||||
| 
 | ||||
| 	bool res = true; | ||||
| 	bool dsf = (connectionType == ConnectionType::dsf); | ||||
| 
 | ||||
| 	auto upload_cmd = get_upload_url(upload_data.upload_path.string(), connectionType); | ||||
| 	BOOST_LOG_TRIVIAL(info) << boost::format("Duet: Uploading file %1%, filepath: %2%, post_action: %3%, command: %4%") | ||||
| 		% upload_data.source_path | ||||
| 		% upload_data.upload_path | ||||
| 		% int(upload_data.post_action) | ||||
| 		% upload_cmd; | ||||
| 
 | ||||
| 	auto http = (dsf ? Http::put(std::move(upload_cmd)) : Http::post(std::move(upload_cmd))); | ||||
| 	if (dsf) { | ||||
| 		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; | ||||
| 
 | ||||
| 			int err_code = dsf ? (status == 201 ? 0 : 1) : get_err_code_from_body(body); | ||||
| 			if (err_code != 0) { | ||||
| 				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)); | ||||
| 				res = false; | ||||
| 			} else if (upload_data.post_action == PrintHostPostUploadAction::StartPrint) { | ||||
| 				wxString errormsg; | ||||
| 				res = start_print(errormsg, upload_data.upload_path.string(), connectionType, false); | ||||
| 				if (! res) { | ||||
| 					error_fn(std::move(errormsg)); | ||||
| 				} | ||||
| 			} else if (upload_data.post_action == PrintHostPostUploadAction::StartSimulation) { | ||||
| 				wxString errormsg; | ||||
| 				res = start_print(errormsg, upload_data.upload_path.string(), connectionType, true); | ||||
| 				if (! res) { | ||||
| 					error_fn(std::move(errormsg)); | ||||
| 				} | ||||
| 			} | ||||
| 		}) | ||||
| 		.on_error([&](std::string body, std::string error, unsigned status) { | ||||
| 			BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Error uploading file: %1%, HTTP %2%, body: `%3%`") % error % status % body; | ||||
| 			error_fn(format_error(body, error, status)); | ||||
| 			res = false; | ||||
| 		}) | ||||
| 		.on_progress([&](Http::Progress progress, bool &cancel) { | ||||
| 			prorgess_fn(std::move(progress), cancel); | ||||
| 			if (cancel) { | ||||
| 				// Upload was canceled
 | ||||
| 				BOOST_LOG_TRIVIAL(info) << "Duet: Upload canceled"; | ||||
| 				res = false; | ||||
| 			} | ||||
| 		}) | ||||
| 		.perform_sync(); | ||||
| 
 | ||||
| 	disconnect(connectionType); | ||||
| 
 | ||||
| 	return res; | ||||
| } | ||||
| 
 | ||||
| Duet::ConnectionType Duet::connect(wxString &msg) const | ||||
| { | ||||
| 	auto res = ConnectionType::error; | ||||
| 	auto url = get_connect_url(false); | ||||
| 
 | ||||
| 	auto http = Http::get(std::move(url)); | ||||
| 	http.on_error([&](std::string body, std::string error, unsigned status) { | ||||
| 			auto dsfUrl = get_connect_url(true); | ||||
| 			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) { | ||||
| 			BOOST_LOG_TRIVIAL(debug) << boost::format("Duet: Got: %1%") % body; | ||||
| 
 | ||||
| 			int err_code = get_err_code_from_body(body); | ||||
| 			switch (err_code) { | ||||
| 				case 0: | ||||
| 					res = ConnectionType::rrf; | ||||
| 					break; | ||||
| 				case 1: | ||||
| 					msg = format_error(body, L("Wrong password"), 0); | ||||
| 					break; | ||||
| 				case 2: | ||||
| 					msg = format_error(body, L("Could not get resources to create a new connection"), 0); | ||||
| 					break; | ||||
| 				default: | ||||
| 					msg = format_error(body, L("Unknown error occured"), 0); | ||||
| 					break; | ||||
| 			} | ||||
| 
 | ||||
| 		}) | ||||
| 		.perform_sync(); | ||||
| 
 | ||||
| 	return res; | ||||
| } | ||||
| 
 | ||||
| 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") | ||||
| 			% get_base_url()).str(); | ||||
| 
 | ||||
| 	auto http = Http::get(std::move(url)); | ||||
| 	http.on_error([&](std::string body, std::string error, unsigned status) { | ||||
| 		// we don't care about it, if disconnect is not working Duet will disconnect automatically after some time
 | ||||
| 		BOOST_LOG_TRIVIAL(error) << boost::format("Duet: Error disconnecting: %1%, HTTP %2%, body: `%3%`") % error % status % body; | ||||
| 	}) | ||||
| 	.perform_sync(); | ||||
| } | ||||
| 
 | ||||
| std::string Duet::get_upload_url(const std::string &filename, ConnectionType connectionType) const | ||||
| { | ||||
|     assert(connectionType != ConnectionType::error); | ||||
| 
 | ||||
| 	if (connectionType == ConnectionType::dsf) { | ||||
| 		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 bool dsfUrl) const | ||||
| { | ||||
| 	if (dsfUrl)	{ | ||||
| 		return (boost::format("%1%machine/status") | ||||
| 				% get_base_url()).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 | ||||
| { | ||||
| 	if (host.find("http://") == 0 || host.find("https://") == 0) { | ||||
| 		if (host.back() == '/') { | ||||
| 			return host; | ||||
| 		} else { | ||||
| 			return (boost::format("%1%/") % host).str(); | ||||
| 		} | ||||
| 	} else { | ||||
| 		return (boost::format("http://%1%/") % host).str(); | ||||
| 	} | ||||
| } | ||||
| 
 | ||||
| std::string Duet::timestamp_str() const | ||||
| { | ||||
| 	enum { BUFFER_SIZE = 32 }; | ||||
| 
 | ||||
| 	auto t = std::time(nullptr); | ||||
| 	auto tm = *std::localtime(&t); | ||||
| 
 | ||||
| 	char buffer[BUFFER_SIZE]; | ||||
| 	std::strftime(buffer, BUFFER_SIZE, "time=%Y-%m-%dT%H:%M:%S", &tm); | ||||
| 
 | ||||
| 	return std::string(buffer); | ||||
| } | ||||
| 
 | ||||
| bool Duet::start_print(wxString &msg, const std::string &filename, ConnectionType connectionType, bool simulationMode) const | ||||
| { | ||||
|     assert(connectionType != ConnectionType::error); | ||||
| 
 | ||||
| 	bool res = false; | ||||
| 	bool dsf = (connectionType == ConnectionType::dsf); | ||||
| 
 | ||||
| 	auto url = dsf | ||||
| 		? (boost::format("%1%machine/code") | ||||
| 			% get_base_url()).str() | ||||
| 		: (boost::format(simulationMode | ||||
| 				? "%1%rr_gcode?gcode=M37%%20P\"0:/gcodes/%2%\"" | ||||
| 				: "%1%rr_gcode?gcode=M32%%20\"0:/gcodes/%2%\"") | ||||
| 			% get_base_url() | ||||
| 			% Http::url_encode(filename)).str(); | ||||
| 
 | ||||
| 	auto http = (dsf ? Http::post(std::move(url)) : Http::get(std::move(url))); | ||||
| 	if (dsf) { | ||||
| 		http.set_post_body( | ||||
| 				(boost::format(simulationMode | ||||
| 						? "M37 P\"0:/gcodes/%1%\"" | ||||
| 						: "M32 \"0:/gcodes/%1%\"") | ||||
| 					% filename).str() | ||||
| 				); | ||||
| 	} | ||||
| 	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; | ||||
| 			msg = format_error(body, error, status); | ||||
| 		}) | ||||
| 		.on_complete([&](std::string body, unsigned) { | ||||
| 			BOOST_LOG_TRIVIAL(debug) << boost::format("Duet: Got: %1%") % body; | ||||
| 			res = true; | ||||
| 		}) | ||||
| 		.perform_sync(); | ||||
| 
 | ||||
| 	return res; | ||||
| } | ||||
| 
 | ||||
| int Duet::get_err_code_from_body(const std::string &body) const | ||||
| { | ||||
| 	pt::ptree root; | ||||
| 	std::istringstream iss (body); // wrap returned json to istringstream
 | ||||
| 	pt::read_json(iss, root); | ||||
| 
 | ||||
| 	return root.get<int>("err", 0); | ||||
| } | ||||
| 
 | ||||
| } | ||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 SoftFever
						SoftFever