#ifndef slic3r_AuthSession_hpp_ #define slic3r_AuthSession_hpp_ #include "Event.hpp" #include "libslic3r/AppConfig.hpp" #include #include #include #include #include namespace Slic3r { namespace GUI { using OpenPrusaAuthEvent = Event; using PrusaAuthSuccessEvent = Event; using PrusaAuthFailEvent = Event; wxDECLARE_EVENT(EVT_OPEN_PRUSAAUTH, OpenPrusaAuthEvent); wxDECLARE_EVENT(EVT_LOGGEDOUT_PRUSAAUTH, PrusaAuthSuccessEvent); wxDECLARE_EVENT(EVT_PA_ID_USER_SUCCESS, PrusaAuthSuccessEvent); wxDECLARE_EVENT(EVT_PA_ID_USER_FAIL, PrusaAuthFailEvent); wxDECLARE_EVENT(EVT_PRUSAAUTH_SUCCESS, PrusaAuthSuccessEvent); wxDECLARE_EVENT(EVT_PRUSACONNECT_PRINTERS_SUCCESS, PrusaAuthSuccessEvent); wxDECLARE_EVENT(EVT_PRUSAAUTH_FAIL, PrusaAuthFailEvent); wxDECLARE_EVENT(EVT_PRUSAAUTH_RESET, PrusaAuthFailEvent); typedef std::function UserActionSuccessFn; typedef std::function UserActionFailFn; // UserActions implements different operations via trigger() method. Stored in m_actions. enum class UserActionID { DUMMY_ACTION, REFRESH_TOKEN, CODE_FOR_TOKEN, TEST_CONNECTION, LOGIN_USER_ID, CONNECT_DUMMY, CONNECT_PRINTERS, }; class UserAction { public: UserAction(const std::string name, const std::string url) : m_action_name(name), m_url(url){} ~UserAction() {} virtual void perform(const std::string& access_token, UserActionSuccessFn success_callback, UserActionFailFn fail_callback, const std::string& input) = 0; protected: std::string m_action_name; std::string m_url; }; class UserActionGetWithEvent : public UserAction { public: UserActionGetWithEvent(const std::string name, const std::string url, wxEvtHandler* evt_handler, wxEventType succ_event_type, wxEventType fail_event_type) : m_succ_evt_type(succ_event_type) , m_fail_evt_type(fail_event_type) , m_evt_handler(evt_handler) , UserAction(name, url) {} ~UserActionGetWithEvent() {} void perform(const std::string& access_token, UserActionSuccessFn success_callback, UserActionFailFn fail_callback, const std::string& input) override; private: wxEventType m_succ_evt_type; wxEventType m_fail_evt_type; wxEvtHandler* m_evt_handler; }; class UserActionPost : public UserAction { public: UserActionPost(const std::string name, const std::string url) : UserAction(name, url) {} ~UserActionPost() {} void perform(const std::string& access_token, UserActionSuccessFn success_callback, UserActionFailFn fail_callback, const std::string& input) override; }; class DummyUserAction : public UserAction { public: DummyUserAction() : UserAction("Dummy", {}) {} ~DummyUserAction() {} void perform(const std::string& access_token, UserActionSuccessFn success_callback, UserActionFailFn fail_callback, const std::string& input) override { } }; struct ActionQueueData { UserActionID action_id; UserActionSuccessFn success_callback; UserActionFailFn fail_callback; std::string input; }; class AuthSession { public: AuthSession(wxEvtHandler* evt_handler, const std::string& access_token, const std::string& refresh_token, const std::string& shared_session_key, bool polling_enabled) : m_access_token(access_token) , m_refresh_token(refresh_token) , m_shared_session_key(shared_session_key) , m_polling_enabled(polling_enabled) { // do not forget to add delete to destructor m_actions[UserActionID::DUMMY_ACTION] = std::make_unique(); m_actions[UserActionID::REFRESH_TOKEN] = std::make_unique("EXCHANGE_TOKENS", "https://test-account.prusa3d.com/o/token/"); m_actions[UserActionID::CODE_FOR_TOKEN] = std::make_unique("EXCHANGE_TOKENS", "https://test-account.prusa3d.com/o/token/"); m_actions[UserActionID::TEST_CONNECTION] = std::make_unique("TEST_CONNECTION", "https://test-account.prusa3d.com/api/v1/me/", evt_handler, wxEVT_NULL, EVT_PRUSAAUTH_FAIL); m_actions[UserActionID::LOGIN_USER_ID] = std::make_unique("USER_ID", "https://test-account.prusa3d.com/api/v1/me/", evt_handler, EVT_PA_ID_USER_SUCCESS, EVT_PRUSAAUTH_FAIL); m_actions[UserActionID::CONNECT_DUMMY] = std::make_unique("CONNECT_DUMMY", "https://dev.connect.prusa3d.com/slicer/dummy"/*"dev.connect.prusa:8000/slicer/dummy"*/, evt_handler, EVT_PRUSAAUTH_SUCCESS, EVT_PRUSAAUTH_FAIL); m_actions[UserActionID::CONNECT_PRINTERS] = std::make_unique("CONNECT_PRINTERS", "https://dev.connect.prusa3d.com/slicer/printers"/*"dev.connect.prusa:8000/slicer/printers"*/, evt_handler, EVT_PRUSACONNECT_PRINTERS_SUCCESS, EVT_PRUSAAUTH_FAIL); } ~AuthSession() { m_actions[UserActionID::DUMMY_ACTION].reset(nullptr); m_actions[UserActionID::REFRESH_TOKEN].reset(nullptr); m_actions[UserActionID::CODE_FOR_TOKEN].reset(nullptr); m_actions[UserActionID::TEST_CONNECTION].reset(nullptr); m_actions[UserActionID::LOGIN_USER_ID].reset(nullptr); m_actions[UserActionID::CONNECT_DUMMY].reset(nullptr); m_actions[UserActionID::CONNECT_PRINTERS].reset(nullptr); //assert(m_actions.empty()); } void clear() { m_access_token.clear(); m_refresh_token.clear(); m_shared_session_key.clear(); m_proccessing_enabled = false; } // Functions that automatically enable action queu processing void init_with_code(const std::string& code, const std::string& code_verifier); void enqueue_action(UserActionID id, UserActionSuccessFn success_callback, UserActionFailFn fail_callback, const std::string& input); void process_action_queue(); bool is_initialized() { return !m_access_token.empty() || !m_refresh_token.empty(); } std::string get_access_token() const { return m_access_token; } std::string get_refresh_token() const { return m_refresh_token; } std::string get_shared_session_key() const { return m_shared_session_key; } void set_polling_enabled(bool enabled) {m_polling_enabled = enabled; } private: void enqueue_test_with_refresh(); void enqueue_refresh(const std::string& body); void refresh_failed_callback(const std::string& body); void cancel_queue(); std::string client_id() const { return "UfTRUm5QjWwaQEGpWQBHGHO3reAyuzgOdBaiqO52"; } // false prevents action queu to be processed - no communication is done // sets to true by init_with_code or enqueue_action call bool m_proccessing_enabled {false}; // triggers CONNECT_PRINTERS action when woken up on idle bool m_polling_enabled; std::string m_access_token; std::string m_refresh_token; std::string m_shared_session_key; std::queue m_action_queue; std::queue m_priority_action_queue; std::map> m_actions; }; } } #endif