From 88c9996ce6fa9e9a79c3f63814023559b73687d2 Mon Sep 17 00:00:00 2001
From: Hassan DRAGA
Date: Sun, 30 Jun 2024 23:30:38 -0400
Subject: [PATCH] Multi-Client Design (Heavy Core Change)
* Re-design the core to deal with multi-client requests
* Enhanced Security - Cookie Authorization
* Enhanced Security - One-time Token
* Enhanced Security - Isolated Window Binding IDs
---
.gitignore | 1 +
src/webui.c | 3583 +++++++++++++++++++++++++++------------------------
2 files changed, 1931 insertions(+), 1653 deletions(-)
diff --git a/.gitignore b/.gitignore
index 6b1edd0f..bc64f977 100644
--- a/.gitignore
+++ b/.gitignore
@@ -48,6 +48,7 @@ build/
*.recipe
*.idb
*.iobj
+*.pdb
# Visual Studio for Mac
.idea/
diff --git a/src/webui.c b/src/webui.c
index 0917e594..9946533b 100644
--- a/src/webui.c
+++ b/src/webui.c
@@ -67,6 +67,7 @@
#define WEBUI_CMD_SEND_RAW 0xF8 // Command: Send raw binary data to the UI
#define WEBUI_CMD_ADD_ID 0xF7 // Command: Add new bind ID
#define WEBUI_CMD_MULTI 0xF6 // Command: Multi packet data
+#define WEBUI_CMD_CHECK_TK 0xF5 // Command: Check validity of a client's token
#define WEBUI_PROTOCOL_SIZE (8) // Protocol header size in bytes
#define WEBUI_PROTOCOL_SIGN (0) // Protocol byte position: Signature (1 Byte)
#define WEBUI_PROTOCOL_TOKEN (1) // Protocol byte position: Token (4 Bytes)
@@ -97,6 +98,8 @@
#define WEBUI_MAX_X (3000) // Maximal window X (4K Monitor)
#define WEBUI_MAX_Y (1800) // Maximal window Y (4K Monitor)
#define WEBUI_PROFILE_NAME "WebUI" // Default browser profile name (Used only for Firefox)
+#define WEBUI_AUTH_COOKIE (32) // Authentification Cookie Len
+#define WEBUI_BROWSERS_COUNT (16) // Total supported browser (enum webui_browser)
#ifdef WEBUI_TLS
#define WEBUI_SECURE "TLS-Encryption"
@@ -151,6 +154,11 @@ typedef pthread_cond_t webui_condition_t;
#define WEBUI_COMPILER "Unknown"
#endif
+// Verbose Log
+#ifdef WEBUI_LOG
+ // #define WEBUI_LOG_VERBOSE
+#endif
+
// Timer
typedef struct _webui_timer_t {
struct timespec start;
@@ -159,10 +167,12 @@ typedef struct _webui_timer_t {
// Event data
typedef struct webui_event_inf_t {
- char* event_data[WEBUI_MAX_ARG + 1]; // Event data (string | num | bool | raw)
+ char* event_data[WEBUI_MAX_ARG + 1]; // Event data (string | num | bool | raw)
size_t event_size[WEBUI_MAX_ARG + 1]; // Event data size (in bytes)
- char* response; // Event response (string)
+ char* response; // Event response (string)
size_t count;
+ struct mg_connection* client;
+ size_t client_id;
} webui_event_inf_t;
// WebView
@@ -181,14 +191,14 @@ typedef struct webui_event_inf_t {
void* createWebViewControllerHandler_lpVtbl;
// WebUI Window
wchar_t* url;
- volatile bool navigate;
- volatile bool size;
- volatile bool position;
- volatile int width;
- volatile int height;
- volatile int x;
- volatile int y;
- volatile bool stop;
+ bool navigate;
+ bool size;
+ bool position;
+ int width;
+ int height;
+ int x;
+ int y;
+ bool stop;
} _webui_wv_win32_t;
#elif __linux__
void* libgtk;
@@ -220,11 +230,11 @@ typedef struct webui_event_inf_t {
gtk_window_move_func gtk_window_move = NULL;
gtk_window_close_func gtk_window_close = NULL;
gtk_window_resize_func gtk_window_resize = NULL;
- gtk_window_set_position_func gtk_window_set_position = NULL;
+ gtk_window_set_position_func gtk_window_set_position = NULL;
g_signal_connect_data_func g_signal_connect_data = NULL;
g_idle_add_func g_idle_add = NULL;
// WebKit Symbol Addresses
- typedef void *(*webkit_web_view_new_func)(void);
+ typedef void *(*webkit_web_view_new_func)(void);
typedef void (*webkit_web_view_load_uri_func)(void *, const char *);
typedef const char *(*webkit_web_view_get_title_func)(void *);
webkit_web_view_new_func webkit_web_view_new = NULL;
@@ -235,17 +245,17 @@ typedef struct webui_event_inf_t {
// Linux WebView
void* gtk_win;
void* gtk_wv;
- volatile bool open;
+ bool open;
// WebUI Window
char* url;
- volatile bool navigate;
- volatile bool size;
- volatile bool position;
- volatile unsigned int width;
- volatile unsigned int height;
- volatile unsigned int x;
- volatile unsigned int y;
- volatile bool stop;
+ bool navigate;
+ bool size;
+ bool position;
+ unsigned int width;
+ unsigned int height;
+ unsigned int x;
+ unsigned int y;
+ bool stop;
} _webui_wv_linux_t;
#else
extern bool _webui_macos_wv_new(int index);
@@ -264,33 +274,47 @@ typedef struct webui_event_inf_t {
int index;
// WebUI Window
char* url;
- volatile bool navigate;
- volatile bool size;
- volatile bool position;
- volatile unsigned int width;
- volatile unsigned int height;
- volatile unsigned int x;
- volatile unsigned int y;
- volatile bool stop;
+ bool navigate;
+ bool size;
+ bool position;
+ unsigned int width;
+ unsigned int height;
+ unsigned int x;
+ unsigned int y;
+ bool stop;
} _webui_wv_macos_t;
#endif
// Window
typedef struct _webui_window_t {
- size_t window_number;
- volatile bool server_running;
- volatile bool connected;
- volatile bool file_handled;
- bool ws_block;
- bool html_handled;
- bool bridge_handled;
- bool server_handled;
- bool is_embedded_html;
- bool is_closed;
- size_t custom_server_port;
+ // Client
+ struct mg_connection* clients[WEBUI_MAX_IDS];
+ size_t clients_count;
+ bool clients_token_check[WEBUI_MAX_IDS];
+ // Server
+ bool wait; // Let server thread wait more time for websocket
+ bool server_running; // Slow check
+ bool connected; // Fast check
size_t server_port;
char* url;
const char* html;
+ char* server_root_path;
+ #ifdef _WIN32
+ HANDLE server_thread;
+ #else
+ pthread_t server_thread;
+ #endif
+ // Window
+ uint32_t token;
+ size_t window_number;
+ char* html_elements[WEBUI_MAX_IDS];
+ bool has_all_events;
+ void(*cb[WEBUI_MAX_IDS])(webui_event_t* e);
+ void(*cb_interface[WEBUI_MAX_IDS])(size_t, size_t, char* , size_t, size_t);
+ bool ws_block;
+ bool is_embedded_html;
+ bool is_closed;
+ size_t custom_server_port;
const char* icon;
const char* icon_type;
size_t current_browser;
@@ -300,8 +324,6 @@ typedef struct _webui_window_t {
char* profile_path;
char* profile_name;
size_t runtime;
- bool has_events;
- char* server_root_path;
bool kiosk_mode;
bool disable_browser_high_contrast;
bool hide;
@@ -312,14 +334,7 @@ typedef struct _webui_window_t {
int y;
bool position_set;
size_t process_id;
- #ifdef _WIN32
- HANDLE server_thread;
- #else
- pthread_t server_thread;
- #endif
- const void * ( * files_handler)(const char* filename, int * length);
- uint32_t token;
- struct mg_connection * mg_connection;
+ const void*(*files_handler)(const char* filename, int* length);
webui_event_inf_t* events[WEBUI_MAX_IDS];
size_t events_count;
bool is_public;
@@ -327,7 +342,7 @@ typedef struct _webui_window_t {
char *proxy_server;
// WebView
bool allow_webview;
- volatile bool update_webview;
+ bool update_webview;
#ifdef _WIN32
_webui_wv_win32_t* webView;
#elif __linux__
@@ -345,29 +360,29 @@ typedef struct _webui_core_t {
bool show_auto_js_inject;
bool ws_block;
bool folder_monitor;
+ bool multi_client;
} config;
- volatile size_t servers;
- char* html_elements[WEBUI_MAX_IDS];
+ bool auth_cookie_set[WEBUI_BROWSERS_COUNT];
+ char auth_cookie[WEBUI_BROWSERS_COUNT][WEBUI_AUTH_COOKIE * 2];
+ size_t servers;
size_t used_ports[WEBUI_MAX_IDS];
size_t startup_timeout;
- volatile bool exit_now;
- volatile bool run_done[WEBUI_MAX_IDS]; // 2 Bytes ID
+ size_t cb_count;
+ bool exit_now;
+ bool run_done[WEBUI_MAX_IDS]; // 2 Bytes ID
char* run_userBuffer[WEBUI_MAX_IDS];
size_t run_userBufferLen[WEBUI_MAX_IDS];
bool run_error[WEBUI_MAX_IDS];
uint16_t run_last_id;
bool initialized;
- void( * cb[WEBUI_MAX_IDS])(webui_event_t* e);
- void( * cb_interface[WEBUI_MAX_IDS])(size_t, size_t, char* , size_t, size_t);
char* executable_path;
void * ptr_list[WEBUI_MAX_IDS * 2];
size_t ptr_position;
size_t ptr_size[WEBUI_MAX_IDS * 2];
size_t current_browser;
- _webui_window_t * wins[WEBUI_MAX_IDS];
+ _webui_window_t* wins[WEBUI_MAX_IDS];
bool wins_reserved[WEBUI_MAX_IDS];
size_t last_win_number;
- bool server_handled;
webui_mutex_t mutex_server_start;
webui_mutex_t mutex_send;
webui_mutex_t mutex_receive;
@@ -377,15 +392,16 @@ typedef struct _webui_core_t {
webui_mutex_t mutex_win_connect;
webui_mutex_t mutex_exit_now;
webui_mutex_t mutex_webview_stop;
+ webui_mutex_t mutex_http_handler;
+ webui_mutex_t mutex_client;
webui_condition_t condition_wait;
char* default_server_root_path;
bool ui;
- // bool little_endian;
#ifdef WEBUI_TLS
- uint8_t * root_cert;
- uint8_t * root_key;
- uint8_t * ssl_cert;
- uint8_t * ssl_key;
+ uint8_t* root_cert;
+ uint8_t* root_key;
+ uint8_t* ssl_cert;
+ uint8_t* ssl_key;
#endif
// WebView
bool is_browser_main_run;
@@ -403,22 +419,22 @@ _webui_core_t;
typedef struct _webui_cb_arg_t {
// Event
- _webui_window_t * window;
+ _webui_window_t* window;
size_t event_type;
char* element;
char* data;
size_t event_number;
- // Extras
- char* webui_internal_id;
}
_webui_cb_arg_t;
typedef struct _webui_recv_arg_t {
- _webui_window_t * win;
+ _webui_window_t* win;
void * ptr;
size_t len;
size_t recvNum;
int event_type;
+ struct mg_connection* client;
+ size_t client_id;
}
_webui_recv_arg_t;
@@ -429,7 +445,7 @@ typedef struct _webui_monitor_arg_t {
_webui_monitor_arg_t;
typedef struct _webui_cmd_async_t {
- _webui_window_t * win;
+ _webui_window_t* win;
char* cmd;
}
_webui_cmd_async_t;
@@ -438,13 +454,13 @@ _webui_cmd_async_t;
#ifdef _WIN32
static const char* webui_sep = "\\";
static DWORD WINAPI _webui_run_browser_task(LPVOID _arg);
-static int _webui_system_win32(_webui_window_t * win, char* cmd, bool show);
+static int _webui_system_win32(_webui_window_t* win, char* cmd, bool show);
static int _webui_system_win32_out(const char* cmd, char ** output, bool show);
static bool _webui_socket_test_listen_win32(size_t port_num);
static bool _webui_get_windows_reg_value(HKEY key, LPCWSTR reg, LPCWSTR value_name, char value[WEBUI_MAX_PATH]);
static bool _webui_str_to_wide(const char *s, wchar_t **w);
#define WEBUI_THREAD_SERVER_START DWORD WINAPI _webui_server_thread(LPVOID arg)
-#define WEBUI_THREAD_RECEIVE DWORD WINAPI _webui_process_thread(LPVOID _arg)
+#define WEBUI_THREAD_RECEIVE DWORD WINAPI _webui_ws_process_thread(LPVOID _arg)
#define WEBUI_THREAD_WEBVIEW DWORD WINAPI _webui_webview_thread(LPVOID arg)
#define WEBUI_THREAD_MONITOR DWORD WINAPI _webui_folder_monitor_thread(LPVOID arg)
#define WEBUI_THREAD_RETURN return 0;
@@ -452,87 +468,98 @@ static bool _webui_str_to_wide(const char *s, wchar_t **w);
static const char* webui_sep = "/";
static void * _webui_run_browser_task(void * _arg);
#define WEBUI_THREAD_SERVER_START void * _webui_server_thread(void * arg)
-#define WEBUI_THREAD_RECEIVE void * _webui_process_thread(void * _arg)
+#define WEBUI_THREAD_RECEIVE void * _webui_ws_process_thread(void * _arg)
#define WEBUI_THREAD_WEBVIEW void * _webui_webview_thread(void * arg)
#define WEBUI_THREAD_MONITOR void * _webui_folder_monitor_thread(void * arg)
#define WEBUI_THREAD_RETURN pthread_exit(NULL);
#endif
static void _webui_init(void);
-static bool _webui_show(_webui_window_t * win, const char* content, size_t browser);
-static size_t _webui_get_cb_index(char* webui_internal_id);
-static size_t _webui_set_cb_index(char* webui_internal_id);
+static bool _webui_show(_webui_window_t* win, struct mg_connection* client, const char* content, size_t browser);
+static bool _webui_get_cb_index(_webui_window_t* win, char* element, size_t* id);
static size_t _webui_get_free_port(void);
static void _webui_free_port(size_t port);
static char* _webui_get_current_path(void);
-static void _webui_ws_send(_webui_window_t * win, char* packet, size_t packets_size);
+static void _webui_send_client_ws(_webui_window_t* win, struct mg_connection* client,
+ size_t client_id, char* packet, size_t packets_size);
static void _webui_window_event(
- _webui_window_t * win, int event_type, char* element, size_t event_number, char* webui_internal_id
+ _webui_window_t* win, size_t client_id, int event_type, char* element, size_t event_number
);
-static int _webui_cmd_sync(_webui_window_t * win, char* cmd, bool show);
-static int _webui_cmd_async(_webui_window_t * win, char* cmd, bool show);
-static int _webui_run_browser(_webui_window_t * win, char* cmd);
+static int _webui_cmd_sync(_webui_window_t* win, char* cmd, bool show);
+static int _webui_cmd_async(_webui_window_t* win, char* cmd, bool show);
+static int _webui_run_browser(_webui_window_t* win, char* cmd);
static void _webui_clean(void);
-static bool _webui_browser_exist(_webui_window_t * win, size_t browser);
+static bool _webui_browser_exist(_webui_window_t* win, size_t browser);
static const char* _webui_get_temp_path();
static bool _webui_folder_exist(char* folder);
static void _webui_delete_folder(char* folder);
-static bool _webui_browser_create_new_profile(_webui_window_t * win, size_t browser);
-static bool _webui_browser_start_chrome(_webui_window_t * win, const char* address);
-static bool _webui_browser_start_edge(_webui_window_t * win, const char* address);
-static bool _webui_browser_start_epic(_webui_window_t * win, const char* address);
-static bool _webui_browser_start_vivaldi(_webui_window_t * win, const char* address);
-static bool _webui_browser_start_brave(_webui_window_t * win, const char* address);
-static bool _webui_browser_start_firefox(_webui_window_t * win, const char* address);
-static bool _webui_browser_start_yandex(_webui_window_t * win, const char* address);
-static bool _webui_browser_start_chromium(_webui_window_t * win, const char* address);
-static bool _webui_browser_start(_webui_window_t * win, const char* address, size_t _browser);
+static bool _webui_browser_create_new_profile(_webui_window_t* win, size_t browser);
+static bool _webui_browser_start_chrome(_webui_window_t* win, const char* address);
+static bool _webui_browser_start_edge(_webui_window_t* win, const char* address);
+static bool _webui_browser_start_epic(_webui_window_t* win, const char* address);
+static bool _webui_browser_start_vivaldi(_webui_window_t* win, const char* address);
+static bool _webui_browser_start_brave(_webui_window_t* win, const char* address);
+static bool _webui_browser_start_firefox(_webui_window_t* win, const char* address);
+static bool _webui_browser_start_yandex(_webui_window_t* win, const char* address);
+static bool _webui_browser_start_chromium(_webui_window_t* win, const char* address);
+static bool _webui_browser_start(_webui_window_t* win, const char* address, size_t _browser);
static long _webui_timer_diff(struct timespec * start, struct timespec* end);
-static void _webui_timer_start(_webui_timer_t * t);
-static bool _webui_timer_is_end(_webui_timer_t * t, size_t ms);
+static void _webui_timer_start(_webui_timer_t* t);
+static bool _webui_timer_is_end(_webui_timer_t* t, size_t ms);
static void _webui_timer_clock_gettime(struct timespec * spec);
-static bool _webui_set_root_folder(_webui_window_t * win, const char* path);
-static const char* _webui_generate_js_bridge(_webui_window_t * win);
+static bool _webui_set_root_folder(_webui_window_t* win, const char* path);
+static const char* _webui_generate_js_bridge(_webui_window_t* win, struct mg_connection* client);
static void _webui_free_mem(void * ptr);
static size_t _webui_mb(size_t size);
-static bool _webui_file_exist_mg(_webui_window_t * win, struct mg_connection * conn);
+static bool _webui_file_exist_mg(_webui_window_t* win, struct mg_connection* client);
static bool _webui_file_exist(char* path);
static void _webui_free_all_mem(void);
-static bool _webui_show_window(_webui_window_t * win, const char* content, int type, size_t browser);
-static char* _webui_generate_internal_id(_webui_window_t * win, const char* element);
+static bool _webui_show_window(_webui_window_t* win, struct mg_connection* client,
+ const char* content, int type, size_t browser);
static bool _webui_is_empty(const char* s);
static size_t _webui_strlen(const char* s);
static uint16_t _webui_get_run_id(void);
static void * _webui_malloc(size_t size);
static void _webui_sleep(long unsigned int ms);
-static size_t _webui_find_the_best_browser(_webui_window_t * win);
+static size_t _webui_find_the_best_browser(_webui_window_t* win);
static bool _webui_is_process_running(const char* process_name);
static void _webui_panic(char* msg);
static void _webui_kill_pid(size_t pid);
-static _webui_window_t * _webui_dereference_win_ptr(void * ptr);
-static int _webui_get_browser_args(_webui_window_t * win, size_t browser, char* buffer, size_t len);
-static void _webui_mutex_init(webui_mutex_t * mutex);
-static void _webui_mutex_lock(webui_mutex_t * mutex);
-static void _webui_mutex_unlock(webui_mutex_t * mutex);
-static void _webui_mutex_destroy(webui_mutex_t * mutex);
-static bool _webui_mutex_is_connected(_webui_window_t * win, int update);
+static _webui_window_t* _webui_dereference_win_ptr(void * ptr);
+static int _webui_get_browser_args(_webui_window_t* win, size_t browser, char* buffer, size_t len);
+static void _webui_mutex_init(webui_mutex_t* mutex);
+static void _webui_mutex_lock(webui_mutex_t* mutex);
+static void _webui_mutex_unlock(webui_mutex_t* mutex);
+static void _webui_mutex_destroy(webui_mutex_t* mutex);
+static bool _webui_mutex_is_connected(_webui_window_t* win, int update);
static bool _webui_mutex_is_exit_now(int update);
static bool _webui_mutex_is_webview_update(_webui_window_t* win, int update);
-static void _webui_condition_init(webui_condition_t * cond);
-static void _webui_condition_wait(webui_condition_t * cond, webui_mutex_t * mutex);
-static void _webui_condition_signal(webui_condition_t * cond);
-static void _webui_condition_destroy(webui_condition_t * cond);
-static void _webui_http_send(struct mg_connection * conn, const char* mime_type, const char* body);
-static void _webui_http_send_error_page(struct mg_connection * conn, const char* body, int status);
-static int _webui_http_handler(struct mg_connection * conn, void * _win);
-static int _webui_ws_connect_handler(const struct mg_connection * conn, void * _win);
-static void _webui_ws_ready_handler(struct mg_connection * conn, void * _win);
-static int _webui_ws_data_handler(struct mg_connection * conn, int opcode, char* data, size_t datasize, void * _win);
-static void _webui_ws_close_handler(const struct mg_connection * conn, void * _win);
-static void _webui_receive(_webui_window_t * win, int event_type, void * data, size_t len);
-static void _webui_process(_webui_window_t * win, void * ptr, size_t len, size_t recvNum, int event_type);
+static void _webui_condition_init(webui_condition_t* cond);
+static void _webui_condition_wait(webui_condition_t* cond, webui_mutex_t* mutex);
+static void _webui_condition_signal(webui_condition_t* cond);
+static void _webui_condition_destroy(webui_condition_t* cond);
+static void _webui_http_send(_webui_window_t* win, struct mg_connection* client,
+ const char* mime_type, const char* body, size_t body_len, bool cache);
+static void _webui_http_send_file(_webui_window_t* win, struct mg_connection* client,
+ const char* mime_type, const char* path, bool cache);
+static void _webui_http_send_header(_webui_window_t* win, struct mg_connection* client,
+ const char* mime_type, size_t body_len, bool cache);
+static void _webui_http_send_error(struct mg_connection* client, const char* body, int status);
+static int _webui_http_handler(struct mg_connection* client, void * _win);
+static int _webui_ws_connect_handler(const struct mg_connection* client, void * _win);
+static void _webui_ws_ready_handler(struct mg_connection* client, void * _win);
+static int _webui_ws_data_handler(struct mg_connection* client, int opcode, char* data, size_t datasize, void * _win);
+static void _webui_ws_close_handler(const struct mg_connection* client, void * _win);
+static void _webui_receive(_webui_window_t* win, struct mg_connection* client, int event_type, void * data, size_t len);
+static void _webui_ws_process(_webui_window_t* win, struct mg_connection* client, size_t client_id,
+ void* ptr, size_t len, size_t recvNum, int event_type);
+static bool _webui_client_save(_webui_window_t* win, struct mg_connection* client, size_t* client_id);
+static bool _webui_client_get_id(_webui_window_t* win, struct mg_connection* client, size_t* client_id);
+static void _webui_client_remove(_webui_window_t* win, struct mg_connection* client);
static void _webui_remove_firefox_profile_ini(const char* path, const char* profile_name);
static bool _webui_is_firefox_ini_profile_exist(const char* path, const char* profile_name);
-static void _webui_send(_webui_window_t * win, uint32_t token, uint16_t id, uint8_t cmd, const char* data, size_t len);
+static void _webui_send_client(_webui_window_t* win, struct mg_connection *client,
+ uint16_t id, uint8_t cmd, const char* data, size_t len);
+static void _webui_send_all(_webui_window_t* win, uint16_t id, uint8_t cmd, const char* data, size_t len);
static uint16_t _webui_get_id(const char* data);
static uint32_t _webui_get_token(const char* data);
static uint32_t _webui_generate_random_uint32();
@@ -542,6 +569,8 @@ static bool _webui_is_valid_url(const char* url);
static bool _webui_port_is_used(size_t port_num);
static char* _webui_str_dup(const char* src);
static void _webui_bridge_api_handler(webui_event_t* e);
+static void _webui_generate_cookie(char* cookie, size_t length);
+static bool _webui_check_auth_cookie(_webui_window_t *win, char* full_cookies);
// WebView
#ifdef _WIN32
// Microsoft Windows
@@ -580,7 +609,7 @@ static bool _webui_check_certificate(const char* certificate_pem, const char* pr
#ifdef WEBUI_LOG
static void _webui_print_hex(const char* data, size_t len);
static void _webui_print_ascii(const char* data, size_t len);
-static int _webui_http_log(const struct mg_connection * conn, const char* message);
+static int _webui_http_log(const struct mg_connection* client, const char* message);
#endif
static WEBUI_THREAD_SERVER_START;
static WEBUI_THREAD_RECEIVE;
@@ -589,52 +618,41 @@ static WEBUI_THREAD_MONITOR;
// Safe C STD
#ifdef _WIN32
-#define WEBUI_TOK(str, delim, context) strtok_s(str, delim, context)
-#define WEBUI_FOPEN(file, filename, mode) fopen_s(&file, filename, mode)
-#define WEBUI_SPF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__)
-#define WEBUI_SPF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__)
-#define WEBUI_SCOPY_DYN(dest, dest_size, src) strcpy_s(dest, _webui_mb(dest_size), src)
-#define WEBUI_SCOPY_STATIC(dest, dest_size, src) strcpy_s(dest, dest_size, src)
-#define WEBUI_SCAT_DYN(dest, dest_size, src) strcat_s(dest, _webui_mb(dest_size), src)
-#define WEBUI_SCAT_STATIC(dest, dest_size, src) strcat_s(dest, dest_size, src)
+#define WEBUI_STR_TOK(str, delim, context) strtok_s(str, delim, context)
+#define WEBUI_FILE_OPEN(file, filename, mode) fopen_s(&file, filename, mode)
+#define WEBUI_SN_PRINTF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__)
+#define WEBUI_SN_PRINTF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__)
+#define WEBUI_STR_COPY_DYN(dest, dest_size, src) strcpy_s(dest, _webui_mb(dest_size), src)
+#define WEBUI_STR_COPY_STATIC(dest, dest_size, src) strcpy_s(dest, dest_size, src)
+#define WEBUI_STR_CAT_DYN(dest, dest_size, src) strcat_s(dest, _webui_mb(dest_size), src)
+#define WEBUI_STR_CAT_STATIC(dest, dest_size, src) strcat_s(dest, dest_size, src)
#else
-#define WEBUI_TOK(str, delim, context) strtok_r(str, delim, context)
-#define WEBUI_FOPEN(file, filename, mode) ((file) = fopen(filename, mode))
-#define WEBUI_SPF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__)
-#define WEBUI_SPF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__)
-#define WEBUI_SCOPY_DYN(dest, dest_size, src) strncpy(dest, src, _webui_mb(dest_size))
-#define WEBUI_SCOPY_STATIC(dest, dest_size, src) strncpy(dest, src, dest_size)
-#define WEBUI_SCAT_DYN(dest, dest_size, src) strncat(dest, src, _webui_mb(dest_size))
-#define WEBUI_SCAT_STATIC(dest, dest_size, src) strncat(dest, src, dest_size)
+#define WEBUI_STR_TOK(str, delim, context) strtok_r(str, delim, context)
+#define WEBUI_FILE_OPEN(file, filename, mode) ((file) = fopen(filename, mode))
+#define WEBUI_SN_PRINTF_DYN(buffer, buffer_size, format, ...) snprintf(buffer, _webui_mb(buffer_size), format, ##__VA_ARGS__)
+#define WEBUI_SN_PRINTF_STATIC(buffer, buffer_size, format, ...) snprintf(buffer, buffer_size, format, ##__VA_ARGS__)
+#define WEBUI_STR_COPY_DYN(dest, dest_size, src) strncpy(dest, src, _webui_mb(dest_size))
+#define WEBUI_STR_COPY_STATIC(dest, dest_size, src) strncpy(dest, src, dest_size)
+#define WEBUI_STR_CAT_DYN(dest, dest_size, src) strncat(dest, src, _webui_mb(dest_size))
+#define WEBUI_STR_CAT_STATIC(dest, dest_size, src) strncat(dest, src, dest_size)
#endif
// Assert
-#define WEBUI_ASSERT(s) _webui_panic(s); assert(0 && s);
+#define WEBUI_ASSERT(s) _webui_panic(s);assert(0 && s);
// -- Heap ----------------------------
-static _webui_core_t _webui_core;
+static _webui_core_t _webui;
static const char* webui_html_served = "Access Denied"
"⚠ Access Denied
You can't access this content
because it's already in use in
another "
-"window.
WebUI v" WEBUI_VERSION "";
+"window.
Note: Multi-client mode is disabled.
"
+"WebUI v" WEBUI_VERSION "