Compare commits
8 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d98f013a51 | ||
|
|
2be48a13f7 | ||
|
|
70ba9d5831 | ||
|
|
f91f3ea68c | ||
|
|
5b78c0826d | ||
|
|
5c31f47bea | ||
|
|
51a8e785c0 | ||
|
|
8899e49f73 |
3
.gitignore
vendored
3
.gitignore
vendored
@@ -21,5 +21,4 @@ debian/uhub.postrm.debhelper
|
|||||||
debian/uhub.prerm.debhelper
|
debian/uhub.prerm.debhelper
|
||||||
debian/uhub.substvars
|
debian/uhub.substvars
|
||||||
uhub-passwd
|
uhub-passwd
|
||||||
src/version.h
|
src/version.h
|
||||||
src/system.h
|
|
||||||
11
.travis.yml
11
.travis.yml
@@ -1,11 +0,0 @@
|
|||||||
language: cpp
|
|
||||||
compiler:
|
|
||||||
- gcc
|
|
||||||
- clang
|
|
||||||
env:
|
|
||||||
- CONFIG=minimal
|
|
||||||
- CONFIG=full
|
|
||||||
install:
|
|
||||||
- autotest/travis/install-build-depends.sh
|
|
||||||
script:
|
|
||||||
- autotest/travis/build-and-test.sh
|
|
||||||
2
AUTHORS
2
AUTHORS
@@ -4,7 +4,7 @@ Authors of uhub
|
|||||||
Jan Vidar Krey, Design and implementation
|
Jan Vidar Krey, Design and implementation
|
||||||
E_zombie, Centos/RedHat customization scripts and heavy load testing
|
E_zombie, Centos/RedHat customization scripts and heavy load testing
|
||||||
FleetCommand, Hub topic plugin code
|
FleetCommand, Hub topic plugin code
|
||||||
MiMic, Implemented user commands, and plugins
|
MiMic, Implemented user commands
|
||||||
Boris Pek (tehnick), Debian/Ubuntu packaging
|
Boris Pek (tehnick), Debian/Ubuntu packaging
|
||||||
Tillmann Karras (Tilka), Misc. bug fixes
|
Tillmann Karras (Tilka), Misc. bug fixes
|
||||||
Yoran Heling (Yorhel), TLS/SSL handshake detection bugfixes
|
Yoran Heling (Yorhel), TLS/SSL handshake detection bugfixes
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
##
|
##
|
||||||
## Makefile for uhub
|
## Makefile for uhub
|
||||||
## Copyright (C) 2007-2013, Jan Vidar Krey <janvidar@extatic.org>
|
## Copyright (C) 2007-2012, Jan Vidar Krey <janvidar@extatic.org>
|
||||||
#
|
#
|
||||||
|
|
||||||
cmake_minimum_required (VERSION 2.8.2)
|
cmake_minimum_required (VERSION 2.8.2)
|
||||||
@@ -9,14 +9,14 @@ project (uhub NONE)
|
|||||||
enable_language(C)
|
enable_language(C)
|
||||||
|
|
||||||
set (UHUB_VERSION_MAJOR 0)
|
set (UHUB_VERSION_MAJOR 0)
|
||||||
set (UHUB_VERSION_MINOR 5)
|
set (UHUB_VERSION_MINOR 4)
|
||||||
set (UHUB_VERSION_PATCH 0)
|
set (UHUB_VERSION_PATCH 1)
|
||||||
|
|
||||||
set (PROJECT_SOURCE_DIR "${CMAKE_SOURCE_DIR}/src")
|
set (PROJECT_SOURCE_DIR "${CMAKE_SOURCE_DIR}/src")
|
||||||
set (CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} ${CMAKE_SOURCE_DIR}/cmake/Modules)
|
set (CMAKE_MODULE_PATH ${CMAKE_MODULE_PATH} ${CMAKE_SOURCE_DIR}/cmake/Modules)
|
||||||
|
|
||||||
option(RELEASE "Release build, debug build if disabled" ON)
|
option(RELEASE "Release build, debug build if disabled" ON)
|
||||||
option(LOWLEVEL_DEBUG, "Enable low level debug messages." OFF)
|
option(LINK_SUPPORT "Allow hub linking" OFF)
|
||||||
option(SSL_SUPPORT "Enable SSL support" ON)
|
option(SSL_SUPPORT "Enable SSL support" ON)
|
||||||
option(USE_OPENSSL "Use OpenSSL's SSL support" ON )
|
option(USE_OPENSSL "Use OpenSSL's SSL support" ON )
|
||||||
option(SYSTEMD_SUPPORT "Enable systemd notify and journal logging" OFF)
|
option(SYSTEMD_SUPPORT "Enable systemd notify and journal logging" OFF)
|
||||||
@@ -25,24 +25,6 @@ option(ADC_STRESS "Enable the stress tester client" OFF)
|
|||||||
find_package(Git)
|
find_package(Git)
|
||||||
find_package(Sqlite3)
|
find_package(Sqlite3)
|
||||||
|
|
||||||
include(TestBigEndian)
|
|
||||||
include(CheckSymbolExists)
|
|
||||||
include(CheckIncludeFile)
|
|
||||||
include(CheckTypeSize)
|
|
||||||
|
|
||||||
#Some functions need this to be found
|
|
||||||
add_definitions(-D_GNU_SOURCE)
|
|
||||||
set(CMAKE_REQUIRED_DEFINITIONS "${CMAKE_REQUIRED_DEFINITIONS} -D_GNU_SOURCE")
|
|
||||||
|
|
||||||
TEST_BIG_ENDIAN(BIGENDIAN)
|
|
||||||
if (BIGENDIAN)
|
|
||||||
add_definitions(-DARCH_BIGENDIAN)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (NOT RELEASE)
|
|
||||||
add_definitions(-DDEBUG)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (SSL_SUPPORT)
|
if (SSL_SUPPORT)
|
||||||
if (USE_OPENSSL)
|
if (USE_OPENSSL)
|
||||||
find_package(OpenSSL)
|
find_package(OpenSSL)
|
||||||
@@ -50,7 +32,7 @@ if (SSL_SUPPORT)
|
|||||||
find_package(GnuTLS)
|
find_package(GnuTLS)
|
||||||
endif()
|
endif()
|
||||||
if (NOT GNUTLS_FOUND AND NOT OPENSSL_FOUND)
|
if (NOT GNUTLS_FOUND AND NOT OPENSSL_FOUND)
|
||||||
message(FATAL_ERROR "Neither OpenSSL nor GnuTLS were found!")
|
message(FATAL_ERROR "Neither OpenSSL nor GnuTLS are not found!")
|
||||||
endif()
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
@@ -64,18 +46,8 @@ if (MSVC)
|
|||||||
add_definitions(-D_CRT_SECURE_NO_WARNINGS)
|
add_definitions(-D_CRT_SECURE_NO_WARNINGS)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
check_include_file(stdint.h HAVE_STDINT_H)
|
|
||||||
check_include_file(sys/types.h HAVE_SYS_TYPES_H)
|
|
||||||
if (HAVE_SYS_TYPES_H)
|
|
||||||
set (CMAKE_EXTRA_INCLUDE_FILES ${CMAKE_EXTRA_INCLUDE_FILES} "sys/types.h")
|
|
||||||
endif()
|
|
||||||
check_type_size( ssize_t SSIZE_T )
|
|
||||||
check_symbol_exists(memmem string.h HAVE_MEMMEM)
|
|
||||||
check_symbol_exists(strndup string.h HAVE_STRNDUP)
|
|
||||||
|
|
||||||
include_directories("${PROJECT_SOURCE_DIR}")
|
include_directories("${PROJECT_SOURCE_DIR}")
|
||||||
include_directories(${SQLITE3_INCLUDE_DIRS})
|
include_directories(${SQLITE3_INCLUDE_DIRS})
|
||||||
link_directories(${SQLITE3_LIBRARY_DIRS})
|
|
||||||
|
|
||||||
file (GLOB uhub_SOURCES ${PROJECT_SOURCE_DIR}/core/*.c)
|
file (GLOB uhub_SOURCES ${PROJECT_SOURCE_DIR}/core/*.c)
|
||||||
list (REMOVE_ITEM uhub_SOURCES
|
list (REMOVE_ITEM uhub_SOURCES
|
||||||
@@ -95,7 +67,7 @@ set (adcclient_SOURCES
|
|||||||
add_library(adc STATIC ${adc_SOURCES})
|
add_library(adc STATIC ${adc_SOURCES})
|
||||||
add_library(network STATIC ${network_SOURCES})
|
add_library(network STATIC ${network_SOURCES})
|
||||||
add_library(utils STATIC ${utils_SOURCES})
|
add_library(utils STATIC ${utils_SOURCES})
|
||||||
if ("${CMAKE_C_COMPILER_ID}" STREQUAL "GNU" OR "${CMAKE_C_COMPILER_ID}" STREQUAL "Clang")
|
if(CMAKE_COMPILER_IS_GNUCC)
|
||||||
set_target_properties(utils PROPERTIES COMPILE_FLAGS -fPIC)
|
set_target_properties(utils PROPERTIES COMPILE_FLAGS -fPIC)
|
||||||
set_target_properties(network PROPERTIES COMPILE_FLAGS -fPIC)
|
set_target_properties(network PROPERTIES COMPILE_FLAGS -fPIC)
|
||||||
endif()
|
endif()
|
||||||
@@ -112,7 +84,6 @@ add_library(mod_welcome MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_welcome.c)
|
|||||||
add_library(mod_logging MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_logging.c ${PROJECT_SOURCE_DIR}/adc/sid.c)
|
add_library(mod_logging MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_logging.c ${PROJECT_SOURCE_DIR}/adc/sid.c)
|
||||||
add_library(mod_auth_simple MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_auth_simple.c )
|
add_library(mod_auth_simple MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_auth_simple.c )
|
||||||
add_library(mod_chat_history MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_chat_history.c )
|
add_library(mod_chat_history MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_chat_history.c )
|
||||||
add_library(mod_chat_history_sqlite MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_chat_history_sqlite.c )
|
|
||||||
add_library(mod_chat_only MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_chat_only.c)
|
add_library(mod_chat_only MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_chat_only.c)
|
||||||
add_library(mod_topic MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_topic.c)
|
add_library(mod_topic MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_topic.c)
|
||||||
add_library(mod_no_guest_downloads MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_no_guest_downloads.c)
|
add_library(mod_no_guest_downloads MODULE ${PROJECT_SOURCE_DIR}/plugins/mod_no_guest_downloads.c)
|
||||||
@@ -132,7 +103,6 @@ set_target_properties(
|
|||||||
mod_auth_simple
|
mod_auth_simple
|
||||||
mod_auth_sqlite
|
mod_auth_sqlite
|
||||||
mod_chat_history
|
mod_chat_history
|
||||||
mod_chat_history_sqlite
|
|
||||||
mod_chat_only
|
mod_chat_only
|
||||||
mod_no_guest_downloads
|
mod_no_guest_downloads
|
||||||
mod_topic
|
mod_topic
|
||||||
@@ -146,7 +116,6 @@ target_link_libraries(mod_welcome utils)
|
|||||||
target_link_libraries(mod_auth_simple utils)
|
target_link_libraries(mod_auth_simple utils)
|
||||||
target_link_libraries(mod_auth_sqlite ${SQLITE3_LIBRARIES} utils)
|
target_link_libraries(mod_auth_sqlite ${SQLITE3_LIBRARIES} utils)
|
||||||
target_link_libraries(mod_chat_history utils)
|
target_link_libraries(mod_chat_history utils)
|
||||||
target_link_libraries(mod_chat_history_sqlite ${SQLITE3_LIBRARIES} utils)
|
|
||||||
target_link_libraries(mod_no_guest_downloads utils)
|
target_link_libraries(mod_no_guest_downloads utils)
|
||||||
target_link_libraries(mod_chat_only utils)
|
target_link_libraries(mod_chat_only utils)
|
||||||
target_link_libraries(mod_logging utils)
|
target_link_libraries(mod_logging utils)
|
||||||
@@ -224,24 +193,19 @@ if (SYSTEMD_SUPPORT)
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
configure_file ("${PROJECT_SOURCE_DIR}/version.h.in" "${PROJECT_SOURCE_DIR}/version.h")
|
configure_file ("${PROJECT_SOURCE_DIR}/version.h.in" "${PROJECT_SOURCE_DIR}/version.h")
|
||||||
configure_file ("${PROJECT_SOURCE_DIR}/system.h.in" "${PROJECT_SOURCE_DIR}/system.h")
|
|
||||||
|
|
||||||
# mark_as_advanced(FORCE CMAKE_BUILD_TYPE)
|
mark_as_advanced(FORCE CMAKE_BUILD_TYPE)
|
||||||
# if (RELEASE)
|
if (RELEASE)
|
||||||
# set(CMAKE_BUILD_TYPE Release)
|
set(CMAKE_BUILD_TYPE Release)
|
||||||
# add_definitions(-DNDEBUG)
|
add_definitions(-DNDEBUG)
|
||||||
#else()
|
else()
|
||||||
# set(CMAKE_BUILD_TYPE Debug)
|
set(CMAKE_BUILD_TYPE Debug)
|
||||||
# add_definitions(-DDEBUG)
|
# add_definitions(-DDEBUG)
|
||||||
#endif()
|
|
||||||
|
|
||||||
if (LOWLEVEL_DEBUG)
|
|
||||||
add_definitions(-DLOWLEVEL_DEBUG)
|
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (UNIX)
|
if (UNIX)
|
||||||
install( TARGETS uhub uhub-passwd RUNTIME DESTINATION bin )
|
install( TARGETS uhub uhub-passwd RUNTIME DESTINATION bin )
|
||||||
install( TARGETS mod_example mod_welcome mod_logging mod_auth_simple mod_auth_sqlite mod_chat_history mod_chat_history_sqlite mod_chat_only mod_topic mod_no_guest_downloads DESTINATION /usr/lib/uhub/ OPTIONAL )
|
install( TARGETS mod_example mod_welcome mod_logging mod_auth_simple mod_auth_sqlite mod_chat_history mod_chat_only mod_topic mod_no_guest_downloads DESTINATION /usr/lib/uhub/ OPTIONAL )
|
||||||
install( FILES ${CMAKE_SOURCE_DIR}/doc/uhub.conf ${CMAKE_SOURCE_DIR}/doc/plugins.conf ${CMAKE_SOURCE_DIR}/doc/rules.txt ${CMAKE_SOURCE_DIR}/doc/motd.txt DESTINATION /etc/uhub OPTIONAL )
|
install( FILES ${CMAKE_SOURCE_DIR}/doc/uhub.conf ${CMAKE_SOURCE_DIR}/doc/plugins.conf ${CMAKE_SOURCE_DIR}/doc/rules.txt ${CMAKE_SOURCE_DIR}/doc/motd.txt DESTINATION /etc/uhub OPTIONAL )
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
|||||||
20
ChangeLog
20
ChangeLog
@@ -1,23 +1,3 @@
|
|||||||
0.5.0:
|
|
||||||
- Use TLS 1.2 and strong ciphers by default, but made this configurable.
|
|
||||||
- Fix TLS event handling which caused some busy loops
|
|
||||||
- TLS: Support certificate chains
|
|
||||||
- Fix bug #211: Better Hublist pinger support by adding the AP flag of the INF message.
|
|
||||||
- Fix bug #198: Timers could cause infinite loops
|
|
||||||
- Sqlite3 is now mandatory
|
|
||||||
- Added mod_chat_history_sqlite and mod_chat_is_privileged.
|
|
||||||
- Support for systemd notify and journal logging
|
|
||||||
- Improved flood control counting to strictly not allow more than the given amount of messages in the configured interval.
|
|
||||||
- Optimize lookups by CID and nick.
|
|
||||||
- Added an NMDC and ADC hub redirectors written in Python.
|
|
||||||
- Fix all Clang compile warnings.
|
|
||||||
- Install uhub-passwd also.
|
|
||||||
- Add support for detecting HTTP connections to the hub. Enough to tell browsers to stop calling.
|
|
||||||
- Compile fixes for OpenBSD, including warnings about strcat.
|
|
||||||
- Fix crashing autotest due to wrong initialization of the usermanager.
|
|
||||||
- mod_topic: check argument for NULL
|
|
||||||
- rename !cleartopic to !resettopic
|
|
||||||
|
|
||||||
0.4.1:
|
0.4.1:
|
||||||
- Converted to CMake which replaces Visual Studio project files and GNU makefiles
|
- Converted to CMake which replaces Visual Studio project files and GNU makefiles
|
||||||
- Fix issues with SSL causing excessive CPU usage.
|
- Fix issues with SSL causing excessive CPU usage.
|
||||||
|
|||||||
@@ -516,15 +516,6 @@ int main(int argc, char** argv)
|
|||||||
exotic_add_test(&handle, &exotic_test_list_get_last_prev_2, "list_get_last_prev_2");
|
exotic_add_test(&handle, &exotic_test_list_get_last_prev_2, "list_get_last_prev_2");
|
||||||
exotic_add_test(&handle, &exotic_test_list_get_last_prev_next_1, "list_get_last_prev_next_1");
|
exotic_add_test(&handle, &exotic_test_list_get_last_prev_next_1, "list_get_last_prev_next_1");
|
||||||
exotic_add_test(&handle, &exotic_test_list_clear, "list_clear");
|
exotic_add_test(&handle, &exotic_test_list_clear, "list_clear");
|
||||||
exotic_add_test(&handle, &exotic_test_list_remove_first_1_1, "list_remove_first_1_1");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_remove_first_1_2, "list_remove_first_1_2");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_remove_first_1_3, "list_remove_first_1_3");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_remove_first_1_4, "list_remove_first_1_4");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_remove_first_1_5, "list_remove_first_1_5");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_append_list_1, "list_append_list_1");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_append_list_2, "list_append_list_2");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_append_list_3, "list_append_list_3");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_clear_list_last, "list_clear_list_last");
|
|
||||||
exotic_add_test(&handle, &exotic_test_list_destroy_1, "list_destroy_1");
|
exotic_add_test(&handle, &exotic_test_list_destroy_1, "list_destroy_1");
|
||||||
exotic_add_test(&handle, &exotic_test_list_destroy_2, "list_destroy_2");
|
exotic_add_test(&handle, &exotic_test_list_destroy_2, "list_destroy_2");
|
||||||
exotic_add_test(&handle, &exotic_test_test_message_refc_1, "test_message_refc_1");
|
exotic_add_test(&handle, &exotic_test_test_message_refc_1, "test_message_refc_1");
|
||||||
@@ -732,8 +723,7 @@ int main(int argc, char** argv)
|
|||||||
exotic_add_test(&handle, &exotic_test_rbtree_size_4, "rbtree_size_4");
|
exotic_add_test(&handle, &exotic_test_rbtree_size_4, "rbtree_size_4");
|
||||||
exotic_add_test(&handle, &exotic_test_rbtree_check_10000, "rbtree_check_10000");
|
exotic_add_test(&handle, &exotic_test_rbtree_check_10000, "rbtree_check_10000");
|
||||||
exotic_add_test(&handle, &exotic_test_rbtree_iterate_10000, "rbtree_iterate_10000");
|
exotic_add_test(&handle, &exotic_test_rbtree_iterate_10000, "rbtree_iterate_10000");
|
||||||
exotic_add_test(&handle, &exotic_test_rbtree_remove_10000, "rbtree_remove_10000");
|
exotic_add_test(&handle, &exotic_test_rbtree_remove_5000, "rbtree_remove_5000");
|
||||||
exotic_add_test(&handle, &exotic_test_rbtree_destroy_1, "rbtree_destroy_1");
|
|
||||||
exotic_add_test(&handle, &exotic_test_sid_create_pool, "sid_create_pool");
|
exotic_add_test(&handle, &exotic_test_sid_create_pool, "sid_create_pool");
|
||||||
exotic_add_test(&handle, &exotic_test_sid_check_0a, "sid_check_0a");
|
exotic_add_test(&handle, &exotic_test_sid_check_0a, "sid_check_0a");
|
||||||
exotic_add_test(&handle, &exotic_test_sid_check_0b, "sid_check_0b");
|
exotic_add_test(&handle, &exotic_test_sid_check_0b, "sid_check_0b");
|
||||||
@@ -771,7 +761,9 @@ int main(int argc, char** argv)
|
|||||||
exotic_add_test(&handle, &exotic_test_timer_add_5_events_1, "timer_add_5_events_1");
|
exotic_add_test(&handle, &exotic_test_timer_add_5_events_1, "timer_add_5_events_1");
|
||||||
exotic_add_test(&handle, &exotic_test_timer_check_5_events_1, "timer_check_5_events_1");
|
exotic_add_test(&handle, &exotic_test_timer_check_5_events_1, "timer_check_5_events_1");
|
||||||
exotic_add_test(&handle, &exotic_test_timer_process_5_events_1, "timer_process_5_events_1");
|
exotic_add_test(&handle, &exotic_test_timer_process_5_events_1, "timer_process_5_events_1");
|
||||||
exotic_add_test(&handle, &exotic_test_timer_shutdown, "timer_shutdown");
|
exotic_add_test(&handle, &exotic_test_timer_clear_1, "timer_clear_1");
|
||||||
|
exotic_add_test(&handle, &exotic_test_timer_bulk_1, "timer_bulk_1");
|
||||||
|
exotic_add_test(&handle, &exotic_test_timer_bulk_2, "timer_bulk_2");
|
||||||
exotic_add_test(&handle, &exotic_test_tokenizer_basic_0, "tokenizer_basic_0");
|
exotic_add_test(&handle, &exotic_test_tokenizer_basic_0, "tokenizer_basic_0");
|
||||||
exotic_add_test(&handle, &exotic_test_tokenizer_basic_1, "tokenizer_basic_1");
|
exotic_add_test(&handle, &exotic_test_tokenizer_basic_1, "tokenizer_basic_1");
|
||||||
exotic_add_test(&handle, &exotic_test_tokenizer_basic_1a, "tokenizer_basic_1a");
|
exotic_add_test(&handle, &exotic_test_tokenizer_basic_1a, "tokenizer_basic_1a");
|
||||||
|
|||||||
@@ -14,8 +14,10 @@ static void inf_create_hub()
|
|||||||
{
|
{
|
||||||
net_initialize();
|
net_initialize();
|
||||||
inf_hub = (struct hub_info*) hub_malloc_zero(sizeof(struct hub_info));
|
inf_hub = (struct hub_info*) hub_malloc_zero(sizeof(struct hub_info));
|
||||||
|
inf_hub->users = (struct hub_user_manager*) hub_malloc_zero(sizeof(struct hub_user_manager));
|
||||||
|
inf_hub->users->list = list_create();
|
||||||
|
inf_hub->users->sids = sid_pool_create(500);
|
||||||
|
|
||||||
inf_hub->users = uman_init();
|
|
||||||
inf_hub->acl = (struct acl_handle*) hub_malloc_zero(sizeof(struct acl_handle));
|
inf_hub->acl = (struct acl_handle*) hub_malloc_zero(sizeof(struct acl_handle));
|
||||||
inf_hub->config = (struct hub_config*) hub_malloc_zero(sizeof(struct hub_config));
|
inf_hub->config = (struct hub_config*) hub_malloc_zero(sizeof(struct hub_config));
|
||||||
|
|
||||||
@@ -25,9 +27,12 @@ static void inf_create_hub()
|
|||||||
|
|
||||||
static void inf_destroy_hub()
|
static void inf_destroy_hub()
|
||||||
{
|
{
|
||||||
uman_shutdown(inf_hub->users);
|
/* FIXME */
|
||||||
|
list_destroy(inf_hub->users->list);
|
||||||
|
sid_pool_destroy(inf_hub->users->sids);
|
||||||
acl_shutdown(inf_hub->acl);
|
acl_shutdown(inf_hub->acl);
|
||||||
free_config(inf_hub->config);
|
free_config(inf_hub->config);
|
||||||
|
hub_free(inf_hub->users);
|
||||||
hub_free(inf_hub->acl);
|
hub_free(inf_hub->acl);
|
||||||
hub_free(inf_hub->config);
|
hub_free(inf_hub->config);
|
||||||
hub_free(inf_hub);
|
hub_free(inf_hub);
|
||||||
|
|||||||
@@ -1,16 +1,10 @@
|
|||||||
#include <uhub.h>
|
#include <uhub.h>
|
||||||
|
|
||||||
static struct linked_list* list = NULL;
|
static struct linked_list* list = NULL;
|
||||||
static struct linked_list* list2 = NULL;
|
|
||||||
|
|
||||||
static char A[2] = { 'A', 0 };
|
static char A[2] = { 'A', 0 };
|
||||||
static char B[2] = { 'B', 0 };
|
static char B[2] = { 'B', 0 };
|
||||||
static char C[2] = { 'C', 0 };
|
static char C[2] = { 'C', 0 };
|
||||||
static char A2[2] = { 'a', 0 };
|
|
||||||
static char B2[2] = { 'b', 0 };
|
|
||||||
static char C2[2] = { 'c', 0 };
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static void null_free(void* ptr)
|
static void null_free(void* ptr)
|
||||||
{
|
{
|
||||||
@@ -130,83 +124,6 @@ EXO_TEST(list_clear, {
|
|||||||
return list->size == 0 && list->first == 0 && list->last == 0 && list->iterator == 0;
|
return list->size == 0 && list->first == 0 && list->last == 0 && list->iterator == 0;
|
||||||
});
|
});
|
||||||
|
|
||||||
static int g_remove_flag = 0;
|
|
||||||
static void null_free_inc_flag(void* ptr)
|
|
||||||
{
|
|
||||||
(void) ptr;
|
|
||||||
g_remove_flag++;
|
|
||||||
}
|
|
||||||
|
|
||||||
EXO_TEST(list_remove_first_1_1,
|
|
||||||
{
|
|
||||||
list_append(list, A);
|
|
||||||
list_append(list, B);
|
|
||||||
list_append(list, C);
|
|
||||||
return list->size == 3;
|
|
||||||
});
|
|
||||||
|
|
||||||
EXO_TEST(list_remove_first_1_2,
|
|
||||||
{
|
|
||||||
g_remove_flag = 0;
|
|
||||||
list_remove_first(list, null_free_inc_flag);
|
|
||||||
return list->size == 2 && g_remove_flag == 1;
|
|
||||||
});
|
|
||||||
|
|
||||||
EXO_TEST(list_remove_first_1_3,
|
|
||||||
{
|
|
||||||
list_remove_first(list, NULL);
|
|
||||||
return list->size == 1;
|
|
||||||
});
|
|
||||||
|
|
||||||
EXO_TEST(list_remove_first_1_4,
|
|
||||||
{
|
|
||||||
list_remove_first(list, NULL);
|
|
||||||
return list->size == 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
EXO_TEST(list_remove_first_1_5,
|
|
||||||
{
|
|
||||||
list_remove_first(list, NULL);
|
|
||||||
return list->size == 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
EXO_TEST(list_append_list_1,
|
|
||||||
{
|
|
||||||
list_append(list, A);
|
|
||||||
list_append(list, B);
|
|
||||||
list_append(list, C);
|
|
||||||
list2 = list_create();
|
|
||||||
list_append(list2, A2);
|
|
||||||
list_append(list2, B2);
|
|
||||||
list_append(list2, C2);
|
|
||||||
return list->size == 3 && list2->size == 3;
|
|
||||||
});
|
|
||||||
|
|
||||||
EXO_TEST(list_append_list_2,
|
|
||||||
{
|
|
||||||
list_append_list(list, list2);
|
|
||||||
return list->size == 6 && list2->size == 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
EXO_TEST(list_append_list_3,
|
|
||||||
{
|
|
||||||
list_destroy(list2);
|
|
||||||
return list_get_index(list, 0) == A &&
|
|
||||||
list_get_index(list, 1) == B &&
|
|
||||||
list_get_index(list, 2) == C &&
|
|
||||||
list_get_index(list, 3) == A2 &&
|
|
||||||
list_get_index(list, 4) == B2 &&
|
|
||||||
list_get_index(list, 5) == C2;
|
|
||||||
});
|
|
||||||
|
|
||||||
EXO_TEST(list_clear_list_last,
|
|
||||||
{
|
|
||||||
list_clear(list, &null_free);
|
|
||||||
return 1;
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
EXO_TEST(list_destroy_1, {
|
EXO_TEST(list_destroy_1, {
|
||||||
list_destroy(list);
|
list_destroy(list);
|
||||||
|
|||||||
@@ -86,7 +86,7 @@ EXO_TEST(rbtree_size_3, { return rb_tree_size(tree) == 0; });
|
|||||||
|
|
||||||
EXO_TEST(rbtree_insert_10000, {
|
EXO_TEST(rbtree_insert_10000, {
|
||||||
int i;
|
int i;
|
||||||
for (i = 0; i < MAX_NODES; i++)
|
for (i = 0; i < MAX_NODES ; i++)
|
||||||
{
|
{
|
||||||
const char* key = strdup(uhub_itoa(i));
|
const char* key = strdup(uhub_itoa(i));
|
||||||
const char* val = strdup(uhub_itoa(i + 16384));
|
const char* val = strdup(uhub_itoa(i + 16384));
|
||||||
@@ -96,11 +96,11 @@ EXO_TEST(rbtree_insert_10000, {
|
|||||||
return 1;
|
return 1;
|
||||||
});
|
});
|
||||||
|
|
||||||
EXO_TEST(rbtree_size_4, { return rb_tree_size(tree) == MAX_NODES; });
|
EXO_TEST(rbtree_size_4, { return rb_tree_size(tree) == MAX_NODES ; });
|
||||||
|
|
||||||
EXO_TEST(rbtree_check_10000, {
|
EXO_TEST(rbtree_check_10000, {
|
||||||
int i;
|
int i;
|
||||||
for (i = 0; i < MAX_NODES; i++)
|
for (i = 0; i < MAX_NODES ; i++)
|
||||||
{
|
{
|
||||||
char* key = strdup(uhub_itoa(i));
|
char* key = strdup(uhub_itoa(i));
|
||||||
const char* expect = uhub_itoa(i + 16384);
|
const char* expect = uhub_itoa(i + 16384);
|
||||||
@@ -116,35 +116,29 @@ EXO_TEST(rbtree_iterate_10000, {
|
|||||||
struct rb_node* n = (struct rb_node*) rb_tree_first(tree);
|
struct rb_node* n = (struct rb_node*) rb_tree_first(tree);
|
||||||
while (n)
|
while (n)
|
||||||
{
|
{
|
||||||
|
struct rb_node* p = n;
|
||||||
n = (struct rb_node*) rb_tree_next(tree);
|
n = (struct rb_node*) rb_tree_next(tree);
|
||||||
i++;
|
i++;
|
||||||
}
|
}
|
||||||
return i == MAX_NODES;
|
return i == MAX_NODES ;
|
||||||
});
|
});
|
||||||
|
|
||||||
static int freed_nodes = 0;
|
|
||||||
static void free_node(struct rb_node* n)
|
static void free_node(struct rb_node* n)
|
||||||
{
|
{
|
||||||
hub_free((void*) n->key);
|
hub_free((void*) n->key);
|
||||||
hub_free((void*) n->value);
|
hub_free((void*) n->value);
|
||||||
freed_nodes += 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
EXO_TEST(rbtree_remove_10000, {
|
EXO_TEST(rbtree_remove_5000, {
|
||||||
int i;
|
int i = 0;
|
||||||
int j;
|
struct rb_node* n = (struct rb_node*) rb_tree_first(tree);
|
||||||
for (j = 0; j < 2; j++)
|
for (i = 0; i < MAX_NODES ; i += 2)
|
||||||
{
|
{
|
||||||
for (i = j; i < MAX_NODES; i += 2)
|
const char* key = uhub_itoa(i);
|
||||||
{
|
rb_tree_remove_node(tree, key, &free_node);
|
||||||
const char* key = uhub_itoa(i);
|
|
||||||
rb_tree_remove_node(tree, key, &free_node);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
return freed_nodes == MAX_NODES;
|
|
||||||
});
|
|
||||||
|
|
||||||
EXO_TEST(rbtree_destroy_1, {
|
|
||||||
rb_tree_destroy(tree);
|
|
||||||
return 1;
|
return 1;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
#include <uhub.h>
|
#include <uhub.h>
|
||||||
|
|
||||||
#define MAX_EVENTS 15
|
#define MAX_EVENTS 100
|
||||||
static struct timeout_queue* g_queue;
|
static struct timeout_queue* g_queue;
|
||||||
static time_t g_now;
|
static time_t g_now;
|
||||||
static size_t g_max;
|
static size_t g_max;
|
||||||
@@ -118,8 +118,27 @@ EXO_TEST(timer_process_5_events_1,{
|
|||||||
return timeout_queue_process(g_queue, g_now) == g_triggered;
|
return timeout_queue_process(g_queue, g_now) == g_triggered;
|
||||||
});
|
});
|
||||||
|
|
||||||
EXO_TEST(timer_shutdown,{
|
EXO_TEST(timer_clear_1,{
|
||||||
timeout_queue_shutdown(g_queue);
|
size_t n;
|
||||||
hub_free(g_queue);
|
for (n = 0; n < MAX_EVENTS; n++)
|
||||||
return 1;
|
timeout_queue_remove(g_queue, &g_events[n]);
|
||||||
|
return timeout_queue_get_next_timeout(g_queue, g_now) == g_max;
|
||||||
});
|
});
|
||||||
|
|
||||||
|
EXO_TEST(timer_bulk_1,{
|
||||||
|
size_t n;
|
||||||
|
g_now = 10;
|
||||||
|
for (n = 0; n < MAX_EVENTS; n++)
|
||||||
|
{
|
||||||
|
timeout_queue_insert(g_queue, &g_events[0], 0);
|
||||||
|
}
|
||||||
|
return timeout_queue_process(g_queue, g_now) == 10;
|
||||||
|
});
|
||||||
|
|
||||||
|
EXO_TEST(timer_bulk_2,{
|
||||||
|
g_now = 110;
|
||||||
|
return timeout_queue_process(g_queue, g_now) == 90;
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,38 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
set -x
|
|
||||||
|
|
||||||
export CFLAGS="$(dpkg-buildflags --get CFLAGS) $(dpkg-buildflags --get CPPFLAGS)"
|
|
||||||
export LDFLAGS="$(dpkg-buildflags --get LDFLAGS) -Wl,--as-needed"
|
|
||||||
|
|
||||||
mkdir -p builddir
|
|
||||||
cd builddir
|
|
||||||
|
|
||||||
CMAKEOPTS="..
|
|
||||||
-DCMAKE_INSTALL_PREFIX=/usr"
|
|
||||||
|
|
||||||
if [ "${CONFIG}" = "full" ]; then
|
|
||||||
CMAKEOPTS="${CMAKEOPTS}
|
|
||||||
-DRELEASE=OFF
|
|
||||||
-DLOWLEVEL_DEBUG=ON
|
|
||||||
-DSSL_SUPPORT=ON
|
|
||||||
-DUSE_OPENSSL=ON
|
|
||||||
-DADC_STRESS=ON"
|
|
||||||
else
|
|
||||||
CMAKEOPTS="${CMAKEOPTS}
|
|
||||||
-DRELEASE=ON
|
|
||||||
-DLOWLEVEL_DEBUG=OFF
|
|
||||||
-DSSL_SUPPORT=OFF
|
|
||||||
-DADC_STRESS=OFF"
|
|
||||||
fi
|
|
||||||
|
|
||||||
|
|
||||||
cmake ${CMAKEOPTS} \
|
|
||||||
-DCMAKE_C_FLAGS="${CFLAGS}" \
|
|
||||||
-DCMAKE_EXE_LINKER_FLAGS="${LDFLAGS}"
|
|
||||||
make VERBOSE=1
|
|
||||||
|
|
||||||
|
|
||||||
sudo make install
|
|
||||||
du -shc /etc/uhub/ /usr/bin/uhub* /usr/lib/uhub/
|
|
||||||
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
#!/bin/sh
|
|
||||||
|
|
||||||
sudo apt-get update -qq
|
|
||||||
|
|
||||||
sudo apt-get install -qq cmake
|
|
||||||
|
|
||||||
if [ "${CONFIG}" = "full" ]; then
|
|
||||||
sudo apt-get install -qq libsqlite3-dev libssl-dev
|
|
||||||
fi
|
|
||||||
|
|
||||||
@@ -44,7 +44,6 @@ IF ( NOT SQLITE3_FOUND AND NOT PKG_CONFIG_FOUND )
|
|||||||
/opt/local/lib
|
/opt/local/lib
|
||||||
/sw/lib
|
/sw/lib
|
||||||
/usr/lib
|
/usr/lib
|
||||||
/usr/lib/${CMAKE_LIBRARY_ARCHITECTURE}
|
|
||||||
/usr/local/lib
|
/usr/local/lib
|
||||||
/usr/lib64
|
/usr/lib64
|
||||||
/usr/local/lib64
|
/usr/local/lib64
|
||||||
|
|||||||
@@ -13,15 +13,6 @@
|
|||||||
#
|
#
|
||||||
plugin /usr/lib/uhub/mod_auth_sqlite.so "file=/etc/uhub/users.db"
|
plugin /usr/lib/uhub/mod_auth_sqlite.so "file=/etc/uhub/users.db"
|
||||||
|
|
||||||
# Topic commands.
|
|
||||||
# Note: "topic" == "hub description" (as configured in uhub.conf)
|
|
||||||
#
|
|
||||||
# !topic - change the topic (op required)
|
|
||||||
# !showtopic - show the topic
|
|
||||||
# !resettopic - reset the topic to the default (op required)
|
|
||||||
#
|
|
||||||
# This plugins takes no parameters.
|
|
||||||
#plugin /usr/lib/uhub/mod_topic.so
|
|
||||||
|
|
||||||
# Log file writer
|
# Log file writer
|
||||||
#
|
#
|
||||||
@@ -31,8 +22,7 @@ plugin /usr/lib/uhub/mod_auth_sqlite.so "file=/etc/uhub/users.db"
|
|||||||
plugin /usr/lib/uhub/mod_logging.so "file=/var/log/uhub.log"
|
plugin /usr/lib/uhub/mod_logging.so "file=/var/log/uhub.log"
|
||||||
|
|
||||||
# A simple example plugin
|
# A simple example plugin
|
||||||
#plugin /usr/lib/uhub/mod_example.so
|
# plugin /usr/lib/uhub/mod_example.so
|
||||||
|
|
||||||
# A plugin sending a welcome message.
|
# A plugin sending a welcome message.
|
||||||
#
|
#
|
||||||
# This plugin provides the following commands:
|
# This plugin provides the following commands:
|
||||||
|
|||||||
@@ -104,11 +104,11 @@ msg_auth_invalid_password = Password is wrong
|
|||||||
msg_auth_user_not_found = User not found in password database
|
msg_auth_user_not_found = User not found in password database
|
||||||
msg_user_share_size_low = User is not sharing enough
|
msg_user_share_size_low = User is not sharing enough
|
||||||
msg_user_share_size_high = User is sharing too much
|
msg_user_share_size_high = User is sharing too much
|
||||||
msg_user_slots_low = User has too few upload slots
|
msg_user_slots_low = User have too few upload slots
|
||||||
msg_user_slots_high = User has too many upload slots
|
msg_user_slots_high = User have too many upload slots
|
||||||
msg_user_hub_limit_low = User is on too few hubs
|
msg_user_hub_limit_low = User is on too few hubs
|
||||||
msg_user_hub_limit_high = User is on too many hubs
|
msg_user_hub_limit_high = User is on too many hubs
|
||||||
msg_error_no_memory = Out of memory
|
msg_error_no_memory = No memory
|
||||||
msg_user_flood_chat = Chat flood detected, messages are dropped.
|
msg_user_flood_chat = Chat flood detected, messages are dropped.
|
||||||
msg_user_flood_connect = Connect flood detected, connection refused.
|
msg_user_flood_connect = Connect flood detected, connection refused.
|
||||||
msg_user_flood_search = Search flood detected, search is stopped.
|
msg_user_flood_search = Search flood detected, search is stopped.
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -111,10 +111,7 @@ typedef uint32_t fourcc_t;
|
|||||||
#define ADC_INF_FLAG_CLIENT_ID "ID" /* client id, aka CID */
|
#define ADC_INF_FLAG_CLIENT_ID "ID" /* client id, aka CID */
|
||||||
#define ADC_INF_FLAG_NICK "NI" /* nick name */
|
#define ADC_INF_FLAG_NICK "NI" /* nick name */
|
||||||
#define ADC_INF_FLAG_DESCRIPTION "DE" /* user description */
|
#define ADC_INF_FLAG_DESCRIPTION "DE" /* user description */
|
||||||
|
#define ADC_INF_FLAG_USER_AGENT "VE" /* software version */
|
||||||
#define ADC_INF_FLAG_USER_AGENT_PRODUCT "AP" /* software name */
|
|
||||||
#define ADC_INF_FLAG_USER_AGENT_VERSION "VE" /* software version */
|
|
||||||
|
|
||||||
#define ADC_INF_FLAG_SUPPORT "SU" /* support (extensions, feature cast) */
|
#define ADC_INF_FLAG_SUPPORT "SU" /* support (extensions, feature cast) */
|
||||||
#define ADC_INF_FLAG_SHARED_SIZE "SS" /* size of total files shared in bytes */
|
#define ADC_INF_FLAG_SHARED_SIZE "SS" /* size of total files shared in bytes */
|
||||||
#define ADC_INF_FLAG_SHARED_FILES "SF" /* number of files shared */
|
#define ADC_INF_FLAG_SHARED_FILES "SF" /* number of files shared */
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -168,24 +168,24 @@ int adc_msg_get_arg_offset(struct adc_message* msg)
|
|||||||
{
|
{
|
||||||
if (!msg || !msg->cache)
|
if (!msg || !msg->cache)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
switch (msg->cache[0])
|
switch (msg->cache[0])
|
||||||
{
|
{
|
||||||
/* These *SHOULD* never be seen on a hub */
|
/* These *SHOULD* never be seen on a hub */
|
||||||
case 'U':
|
case 'U':
|
||||||
case 'C':
|
case 'C':
|
||||||
return 4; /* Actually: 4 + strlen(cid). */
|
return 4; /* Actually: 4 + strlen(cid). */
|
||||||
|
|
||||||
case 'I':
|
case 'I':
|
||||||
case 'H':
|
case 'H':
|
||||||
return 4;
|
return 4;
|
||||||
|
|
||||||
case 'B':
|
case 'B':
|
||||||
return 9;
|
return 9;
|
||||||
|
|
||||||
case 'F':
|
case 'F':
|
||||||
return (10 + (list_size(msg->feature_cast_include)*5) + (list_size(msg->feature_cast_exclude)*5));
|
return (10 + (list_size(msg->feature_cast_include)*5) + (list_size(msg->feature_cast_exclude)*5));
|
||||||
|
|
||||||
case 'D':
|
case 'D':
|
||||||
case 'E':
|
case 'E':
|
||||||
return 14;
|
return 14;
|
||||||
@@ -200,10 +200,10 @@ int adc_msg_is_empty(struct adc_message* msg)
|
|||||||
|
|
||||||
if (offset == -1)
|
if (offset == -1)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
if ((msg->length - 1) == (size_t) offset)
|
if ((msg->length - 1) == (size_t) offset)
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -283,19 +283,23 @@ struct adc_message* adc_msg_copy(const struct adc_message* cmd)
|
|||||||
if (cmd->feature_cast_include)
|
if (cmd->feature_cast_include)
|
||||||
{
|
{
|
||||||
copy->feature_cast_include = list_create();
|
copy->feature_cast_include = list_create();
|
||||||
LIST_FOREACH(char*, tmp, cmd->feature_cast_include,
|
tmp = list_get_first(cmd->feature_cast_include);
|
||||||
|
while (tmp)
|
||||||
{
|
{
|
||||||
list_append(copy->feature_cast_include, hub_strdup(tmp));
|
list_append(copy->feature_cast_include, hub_strdup(tmp));
|
||||||
});
|
tmp = list_get_next(cmd->feature_cast_include);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cmd->feature_cast_exclude)
|
if (cmd->feature_cast_exclude)
|
||||||
{
|
{
|
||||||
copy->feature_cast_exclude = list_create();
|
copy->feature_cast_exclude = list_create();
|
||||||
LIST_FOREACH(char*, tmp, cmd->feature_cast_exclude,
|
tmp = list_get_first(cmd->feature_cast_exclude);
|
||||||
|
while (tmp)
|
||||||
{
|
{
|
||||||
list_append(copy->feature_cast_exclude, hub_strdup(tmp));
|
list_append(copy->feature_cast_exclude, hub_strdup(tmp));
|
||||||
});
|
tmp = list_get_next(cmd->feature_cast_exclude);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
ADC_MSG_ASSERT(copy);
|
ADC_MSG_ASSERT(copy);
|
||||||
@@ -307,17 +311,17 @@ struct adc_message* adc_msg_copy(const struct adc_message* cmd)
|
|||||||
struct adc_message* adc_msg_parse_verify(struct hub_user* u, const char* line, size_t length)
|
struct adc_message* adc_msg_parse_verify(struct hub_user* u, const char* line, size_t length)
|
||||||
{
|
{
|
||||||
struct adc_message* command = adc_msg_parse(line, length);
|
struct adc_message* command = adc_msg_parse(line, length);
|
||||||
|
|
||||||
if (!command)
|
if (!command)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
if (command->source && (!u || command->source != u->id.sid))
|
if (command->source && (!u || command->source != u->id.sid))
|
||||||
{
|
{
|
||||||
LOG_DEBUG("Command does not match user's SID (command->source=%d, user->id.sid=%d)", command->source, (u ? u->id.sid : 0));
|
LOG_DEBUG("Command does not match user's SID (command->source=%d, user->id.sid=%d)", command->source, (u ? u->id.sid : 0));
|
||||||
adc_msg_free(command);
|
adc_msg_free(command);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
return command;
|
return command;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -331,7 +335,7 @@ struct adc_message* adc_msg_parse(const char* line, size_t length)
|
|||||||
int ok = 1;
|
int ok = 1;
|
||||||
int need_terminate = 0;
|
int need_terminate = 0;
|
||||||
struct linked_list* feature_cast_list;
|
struct linked_list* feature_cast_list;
|
||||||
|
|
||||||
if (command == NULL)
|
if (command == NULL)
|
||||||
return NULL; /* OOM */
|
return NULL; /* OOM */
|
||||||
|
|
||||||
@@ -595,14 +599,14 @@ int adc_msg_remove_named_argument(struct adc_message* cmd, const char prefix_[2]
|
|||||||
int found = 0;
|
int found = 0;
|
||||||
int arg_offset = adc_msg_get_arg_offset(cmd);
|
int arg_offset = adc_msg_get_arg_offset(cmd);
|
||||||
size_t temp_len = 0;
|
size_t temp_len = 0;
|
||||||
|
|
||||||
adc_msg_unterminate(cmd);
|
adc_msg_unterminate(cmd);
|
||||||
|
|
||||||
start = memmem(&cmd->cache[arg_offset], (cmd->length - arg_offset), prefix, 3);
|
start = memmem(&cmd->cache[arg_offset], (cmd->length - arg_offset), prefix, 3);
|
||||||
while (start)
|
while (start)
|
||||||
{
|
{
|
||||||
endInfo = &cmd->cache[cmd->length];
|
endInfo = &cmd->cache[cmd->length];
|
||||||
|
|
||||||
if (&start[0] < &endInfo[0])
|
if (&start[0] < &endInfo[0])
|
||||||
{
|
{
|
||||||
end = memchr(&start[1], ' ', &endInfo[0]-&start[1]);
|
end = memchr(&start[1], ' ', &endInfo[0]-&start[1]);
|
||||||
@@ -611,13 +615,13 @@ int adc_msg_remove_named_argument(struct adc_message* cmd, const char prefix_[2]
|
|||||||
{
|
{
|
||||||
end = NULL;
|
end = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (end)
|
if (end)
|
||||||
{
|
{
|
||||||
|
|
||||||
temp_len = &end[0] - &start[0]; // strlen(start);
|
temp_len = &end[0] - &start[0]; // strlen(start);
|
||||||
endlen = strlen(end);
|
endlen = strlen(end);
|
||||||
|
|
||||||
memmove(start, end, endlen);
|
memmove(start, end, endlen);
|
||||||
start[endlen] = '\0';
|
start[endlen] = '\0';
|
||||||
found++;
|
found++;
|
||||||
@@ -632,9 +636,9 @@ int adc_msg_remove_named_argument(struct adc_message* cmd, const char prefix_[2]
|
|||||||
}
|
}
|
||||||
start = memmem(&cmd->cache[arg_offset], (cmd->length - arg_offset), prefix, 3);
|
start = memmem(&cmd->cache[arg_offset], (cmd->length - arg_offset), prefix, 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
adc_msg_terminate(cmd);
|
adc_msg_terminate(cmd);
|
||||||
|
|
||||||
return found;
|
return found;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -683,7 +687,7 @@ char* adc_msg_get_named_argument(struct adc_message* cmd, const char prefix_[2])
|
|||||||
length = &end[0] - &start[0];
|
length = &end[0] - &start[0];
|
||||||
|
|
||||||
argument = hub_strndup(start, length);
|
argument = hub_strndup(start, length);
|
||||||
|
|
||||||
if (length > 0 && argument[length-1] == '\n')
|
if (length > 0 && argument[length-1] == '\n')
|
||||||
{
|
{
|
||||||
argument[length-1] = 0;
|
argument[length-1] = 0;
|
||||||
@@ -783,6 +787,13 @@ int adc_msg_add_argument(struct adc_message* cmd, const char* string)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int adc_msg_add_argument_string(struct adc_message* cmd, const char* string)
|
||||||
|
{
|
||||||
|
char* escaped = adc_msg_escape(string);
|
||||||
|
int ret = adc_msg_add_argument(cmd, escaped);
|
||||||
|
hub_free(escaped);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
char* adc_msg_get_argument(struct adc_message* cmd, int offset)
|
char* adc_msg_get_argument(struct adc_message* cmd, int offset)
|
||||||
{
|
{
|
||||||
@@ -862,8 +873,6 @@ int adc_msg_get_argument_index(struct adc_message* cmd, const char prefix[2])
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
int adc_msg_escape_length(const char* str)
|
int adc_msg_escape_length(const char* str)
|
||||||
{
|
{
|
||||||
int add = 0;
|
int add = 0;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -171,6 +171,12 @@ extern int adc_msg_replace_named_argument(struct adc_message* cmd, const char pr
|
|||||||
*/
|
*/
|
||||||
extern int adc_msg_add_argument(struct adc_message* cmd, const char* string);
|
extern int adc_msg_add_argument(struct adc_message* cmd, const char* string);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add an argument string, the string will be automatcally escaped.
|
||||||
|
* @return 0 if successful, or -1 if an error occured (out of memory).
|
||||||
|
*/
|
||||||
|
extern int adc_msg_add_argument_string(struct adc_message* cmd, const char* string);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Append a named argument
|
* Append a named argument
|
||||||
*
|
*
|
||||||
@@ -209,6 +215,13 @@ extern char* adc_msg_unescape(const char* string);
|
|||||||
*/
|
*/
|
||||||
extern int adc_msg_unescape_to_target(const char* string, char* target, size_t target_size);
|
extern int adc_msg_unescape_to_target(const char* string, char* target, size_t target_size);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the length of the string once escaped with
|
||||||
|
* adc_msg_escape().
|
||||||
|
*
|
||||||
|
* The string must be NULL terminated.
|
||||||
|
*/
|
||||||
|
extern int adc_msg_escape_length(const char* str);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Convert a string to a ADC command escaped string.
|
* Convert a string to a ADC command escaped string.
|
||||||
@@ -229,9 +242,10 @@ void adc_msg_unterminate(struct adc_message* cmd);
|
|||||||
/**
|
/**
|
||||||
* @return the offset for the first command argument in msg->cache.
|
* @return the offset for the first command argument in msg->cache.
|
||||||
* or -1 if the command is not understood.
|
* or -1 if the command is not understood.
|
||||||
* NOTE: for 'U' and 'C' commands (normally not seen by hubs),
|
* NOTE: for 'U' and 'C' commands (normally not seen by hubs),
|
||||||
* this returns 4. Should be 4 + lengthOf(cid).
|
* this returns 4. Should be 4 + lengthOf(cid).
|
||||||
*/
|
*/
|
||||||
int adc_msg_get_arg_offset(struct adc_message* msg);
|
int adc_msg_get_arg_offset(struct adc_message* msg);
|
||||||
|
|
||||||
|
|
||||||
#endif /* HAVE_UHUB_COMMAND_H */
|
#endif /* HAVE_UHUB_COMMAND_H */
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -47,7 +47,7 @@ sid_t string_to_sid(const char* sid)
|
|||||||
sid_t nsid = 0;
|
sid_t nsid = 0;
|
||||||
sid_t n, x;
|
sid_t n, x;
|
||||||
sid_t factors[] = { 32768, 1024, 32, 1};
|
sid_t factors[] = { 32768, 1024, 32, 1};
|
||||||
|
|
||||||
if (!sid || strlen(sid) != 4) return 0;
|
if (!sid || strlen(sid) != 4) return 0;
|
||||||
|
|
||||||
for (n = 0; n < 4; n++) {
|
for (n = 0; n < 4; n++) {
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -26,20 +26,20 @@
|
|||||||
static int check_cmd_bool(const char* cmd, struct linked_list* list, char* line, int line_count)
|
static int check_cmd_bool(const char* cmd, struct linked_list* list, char* line, int line_count)
|
||||||
{
|
{
|
||||||
char* data;
|
char* data;
|
||||||
|
|
||||||
if (!strncmp(line, cmd, strlen(cmd)))
|
if (!strncmp(line, cmd, strlen(cmd)))
|
||||||
{
|
{
|
||||||
data = &line[strlen(cmd)];
|
data = &line[strlen(cmd)];
|
||||||
data[0] = '\0';
|
data[0] = '\0';
|
||||||
data++;
|
data++;
|
||||||
|
|
||||||
data = strip_white_space(data);
|
data = strip_white_space(data);
|
||||||
if (!*data)
|
if (!*data)
|
||||||
{
|
{
|
||||||
LOG_FATAL("ACL parse error on line %d", line_count);
|
LOG_FATAL("ACL parse error on line %d", line_count);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
list_append(list, hub_strdup(data));
|
list_append(list, hub_strdup(data));
|
||||||
LOG_DEBUG("ACL: Deny access for: '%s' (%s)", data, cmd);
|
LOG_DEBUG("ACL: Deny access for: '%s' (%s)", data, cmd);
|
||||||
return 1;
|
return 1;
|
||||||
@@ -190,18 +190,18 @@ int acl_initialize(struct hub_config* config, struct acl_handle* handle)
|
|||||||
{
|
{
|
||||||
int ret;
|
int ret;
|
||||||
memset(handle, 0, sizeof(struct acl_handle));
|
memset(handle, 0, sizeof(struct acl_handle));
|
||||||
|
|
||||||
handle->users = list_create();
|
handle->users = list_create();
|
||||||
handle->users_denied = list_create();
|
handle->users_denied = list_create();
|
||||||
handle->users_banned = list_create();
|
handle->users_banned = list_create();
|
||||||
handle->cids = list_create();
|
handle->cids = list_create();
|
||||||
handle->networks = list_create();
|
handle->networks = list_create();
|
||||||
handle->nat_override = list_create();
|
handle->nat_override = list_create();
|
||||||
|
|
||||||
if (!handle->users || !handle->cids || !handle->networks || !handle->users_denied || !handle->users_banned || !handle->nat_override)
|
if (!handle->users || !handle->cids || !handle->networks || !handle->users_denied || !handle->users_banned || !handle->nat_override)
|
||||||
{
|
{
|
||||||
LOG_FATAL("acl_initialize: Out of memory");
|
LOG_FATAL("acl_initialize: Out of memory");
|
||||||
|
|
||||||
list_destroy(handle->users);
|
list_destroy(handle->users);
|
||||||
list_destroy(handle->users_denied);
|
list_destroy(handle->users_denied);
|
||||||
list_destroy(handle->users_banned);
|
list_destroy(handle->users_banned);
|
||||||
@@ -210,11 +210,11 @@ int acl_initialize(struct hub_config* config, struct acl_handle* handle)
|
|||||||
list_destroy(handle->nat_override);
|
list_destroy(handle->nat_override);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (config)
|
if (config)
|
||||||
{
|
{
|
||||||
if (!*config->file_acl) return 0;
|
if (!*config->file_acl) return 0;
|
||||||
|
|
||||||
ret = file_read_lines(config->file_acl, handle, &acl_parse_line);
|
ret = file_read_lines(config->file_acl, handle, &acl_parse_line);
|
||||||
if (ret == -1)
|
if (ret == -1)
|
||||||
return -1;
|
return -1;
|
||||||
@@ -249,26 +249,26 @@ int acl_shutdown(struct acl_handle* handle)
|
|||||||
list_clear(handle->users, &acl_free_access_info);
|
list_clear(handle->users, &acl_free_access_info);
|
||||||
list_destroy(handle->users);
|
list_destroy(handle->users);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (handle->users_denied)
|
if (handle->users_denied)
|
||||||
{
|
{
|
||||||
list_clear(handle->users_denied, &hub_free);
|
list_clear(handle->users_denied, &hub_free);
|
||||||
list_destroy(handle->users_denied);
|
list_destroy(handle->users_denied);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (handle->users_banned)
|
if (handle->users_banned)
|
||||||
{
|
{
|
||||||
list_clear(handle->users_banned, &hub_free);
|
list_clear(handle->users_banned, &hub_free);
|
||||||
list_destroy(handle->users_banned);
|
list_destroy(handle->users_banned);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (handle->cids)
|
if (handle->cids)
|
||||||
{
|
{
|
||||||
list_clear(handle->cids, &hub_free);
|
list_clear(handle->cids, &hub_free);
|
||||||
list_destroy(handle->cids);
|
list_destroy(handle->cids);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (handle->networks)
|
if (handle->networks)
|
||||||
{
|
{
|
||||||
list_clear(handle->networks, &acl_free_ip_info);
|
list_clear(handle->networks, &acl_free_ip_info);
|
||||||
@@ -330,11 +330,13 @@ struct auth_info* acl_get_access_info(struct hub_info* hub, const char* name)
|
|||||||
}
|
}
|
||||||
|
|
||||||
#define STR_LIST_CONTAINS(LIST, STR) \
|
#define STR_LIST_CONTAINS(LIST, STR) \
|
||||||
LIST_FOREACH(char*, str, LIST, \
|
str = (char*) list_get_first(LIST); \
|
||||||
|
while (str) \
|
||||||
{ \
|
{ \
|
||||||
if (strcasecmp(str, STR) == 0) \
|
if (strcasecmp(str, STR) == 0) \
|
||||||
return 1; \
|
return 1; \
|
||||||
}); \
|
str = (char*) list_get_next(LIST); \
|
||||||
|
} \
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
int acl_is_cid_banned(struct acl_handle* handle, const char* data)
|
int acl_is_cid_banned(struct acl_handle* handle, const char* data)
|
||||||
@@ -398,28 +400,34 @@ int acl_user_unban_cid(struct acl_handle* handle, const char* cid)
|
|||||||
int acl_is_ip_banned(struct acl_handle* handle, const char* ip_address)
|
int acl_is_ip_banned(struct acl_handle* handle, const char* ip_address)
|
||||||
{
|
{
|
||||||
struct ip_addr_encap raw;
|
struct ip_addr_encap raw;
|
||||||
struct ip_range* info;
|
struct ip_range* info = (struct ip_range*) list_get_first(handle->networks);
|
||||||
|
|
||||||
ip_convert_to_binary(ip_address, &raw);
|
ip_convert_to_binary(ip_address, &raw);
|
||||||
LIST_FOREACH(struct ip_range*, info, handle->networks,
|
|
||||||
|
while (info)
|
||||||
{
|
{
|
||||||
if (ip_in_range(&raw, info))
|
if (ip_in_range(&raw, info))
|
||||||
|
{
|
||||||
return 1;
|
return 1;
|
||||||
});
|
}
|
||||||
|
info = (struct ip_range*) list_get_next(handle->networks);
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
int acl_is_ip_nat_override(struct acl_handle* handle, const char* ip_address)
|
int acl_is_ip_nat_override(struct acl_handle* handle, const char* ip_address)
|
||||||
{
|
{
|
||||||
struct ip_addr_encap raw;
|
struct ip_addr_encap raw;
|
||||||
struct ip_range* info;
|
struct ip_range* info = (struct ip_range*) list_get_first(handle->nat_override);
|
||||||
|
|
||||||
ip_convert_to_binary(ip_address, &raw);
|
ip_convert_to_binary(ip_address, &raw);
|
||||||
LIST_FOREACH(struct ip_range*, info, handle->nat_override,
|
|
||||||
|
while (info)
|
||||||
{
|
{
|
||||||
if (ip_in_range(&raw, info))
|
if (ip_in_range(&raw, info))
|
||||||
|
{
|
||||||
return 1;
|
return 1;
|
||||||
});
|
}
|
||||||
|
info = (struct ip_range*) list_get_next(handle->nat_override);
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -468,10 +476,10 @@ int acl_password_verify(struct hub_info* hub, struct hub_user* user, const char*
|
|||||||
base32_decode(challenge, (unsigned char*) raw_challenge, MAX_CID_LEN);
|
base32_decode(challenge, (unsigned char*) raw_challenge, MAX_CID_LEN);
|
||||||
|
|
||||||
password_len = strlen(access->password);
|
password_len = strlen(access->password);
|
||||||
|
|
||||||
memcpy(&buf[0], access->password, password_len);
|
memcpy(&buf[0], access->password, password_len);
|
||||||
memcpy(&buf[password_len], raw_challenge, TIGERSIZE);
|
memcpy(&buf[password_len], raw_challenge, TIGERSIZE);
|
||||||
|
|
||||||
tiger((uint64_t*) buf, TIGERSIZE+password_len, (uint64_t*) tiger_res);
|
tiger((uint64_t*) buf, TIGERSIZE+password_len, (uint64_t*) tiger_res);
|
||||||
base32_encode((unsigned char*) tiger_res, TIGERSIZE, password_calc);
|
base32_encode((unsigned char*) tiger_res, TIGERSIZE, password_calc);
|
||||||
password_calc[MAX_CID_LEN] = 0;
|
password_calc[MAX_CID_LEN] = 0;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -26,7 +26,7 @@ static void hub_command_args_free(struct hub_command* cmd)
|
|||||||
if (!cmd->args)
|
if (!cmd->args)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
LIST_FOREACH(struct hub_command_arg_data*, data, cmd->args,
|
for (data = (struct hub_command_arg_data*) list_get_first(cmd->args); data; data = (struct hub_command_arg_data*) list_get_next(cmd->args))
|
||||||
{
|
{
|
||||||
switch (data->type)
|
switch (data->type)
|
||||||
{
|
{
|
||||||
@@ -39,7 +39,7 @@ static void hub_command_args_free(struct hub_command* cmd)
|
|||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
|
||||||
list_clear(cmd->args, hub_free);
|
list_clear(cmd->args, hub_free);
|
||||||
list_destroy(cmd->args);
|
list_destroy(cmd->args);
|
||||||
@@ -64,7 +64,6 @@ static enum command_parse_status command_extract_arguments(struct hub_info* hub,
|
|||||||
char* token = NULL;
|
char* token = NULL;
|
||||||
char* tmp = NULL;
|
char* tmp = NULL;
|
||||||
size_t size = 0;
|
size_t size = 0;
|
||||||
size_t offset = 0;
|
|
||||||
struct hub_command_arg_data* data = NULL;
|
struct hub_command_arg_data* data = NULL;
|
||||||
enum command_parse_status status = cmd_status_ok;
|
enum command_parse_status status = cmd_status_ok;
|
||||||
|
|
||||||
@@ -78,15 +77,15 @@ static enum command_parse_status command_extract_arguments(struct hub_info* hub,
|
|||||||
if (greedy)
|
if (greedy)
|
||||||
{
|
{
|
||||||
size = 1;
|
size = 1;
|
||||||
LIST_FOREACH(char*, tmp, tokens, { size += (strlen(tmp) + 1); });
|
for (tmp = (char*) list_get_first(tokens); tmp; tmp = (char*) list_get_next(tokens))
|
||||||
|
size += (strlen(tmp) + 1);
|
||||||
token = hub_malloc_zero(size);
|
token = hub_malloc_zero(size);
|
||||||
|
|
||||||
while ((tmp = list_get_first(tokens)))
|
while ((tmp = list_get_first(tokens)))
|
||||||
{
|
{
|
||||||
if (offset > 0)
|
if (*token)
|
||||||
token[offset++] = ' ';
|
strcat(token, " ");
|
||||||
memcpy(token + offset, tmp, strlen(tmp));
|
strcat(token, tmp);
|
||||||
offset += strlen(tmp);
|
|
||||||
list_remove(tokens, tmp);
|
list_remove(tokens, tmp);
|
||||||
hub_free(tmp);
|
hub_free(tmp);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -31,6 +31,8 @@ static int send_command_not_found(struct command_base* cbase, struct hub_user* u
|
|||||||
static int send_command_syntax_error(struct command_base* cbase, struct hub_user* user);
|
static int send_command_syntax_error(struct command_base* cbase, struct hub_user* user);
|
||||||
static int send_command_missing_arguments(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd);
|
static int send_command_missing_arguments(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd);
|
||||||
|
|
||||||
|
static void null_free(void* ptr) { }
|
||||||
|
|
||||||
struct command_base
|
struct command_base
|
||||||
{
|
{
|
||||||
struct hub_info* hub;
|
struct hub_info* hub;
|
||||||
@@ -96,14 +98,14 @@ struct command_handle* command_handler_lookup(struct command_base* cbase, const
|
|||||||
struct command_handle* handler = NULL;
|
struct command_handle* handler = NULL;
|
||||||
size_t prefix_len = strlen(prefix);
|
size_t prefix_len = strlen(prefix);
|
||||||
|
|
||||||
LIST_FOREACH(struct command_handle*, handler, cbase->handlers,
|
for (handler = (struct command_handle*) list_get_first(cbase->handlers); handler; handler = (struct command_handle*) list_get_next(cbase->handlers))
|
||||||
{
|
{
|
||||||
if (prefix_len != handler->length)
|
if (prefix_len != handler->length)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (!memcmp(prefix, handler->prefix, handler->length))
|
if (!memcmp(prefix, handler->prefix, handler->length))
|
||||||
return handler;
|
return handler;
|
||||||
});
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -290,7 +292,7 @@ static int command_help(struct command_base* cbase, struct hub_user* user, struc
|
|||||||
{
|
{
|
||||||
cbuf_append(buf, "Available commands:\n");
|
cbuf_append(buf, "Available commands:\n");
|
||||||
|
|
||||||
LIST_FOREACH(struct command_handle*, command, cbase->handlers,
|
for (command = (struct command_handle*) list_get_first(cbase->handlers); command; command = (struct command_handle*) list_get_next(cbase->handlers))
|
||||||
{
|
{
|
||||||
if (command_is_available(command, user->credentials))
|
if (command_is_available(command, user->credentials))
|
||||||
{
|
{
|
||||||
@@ -299,7 +301,7 @@ static int command_help(struct command_base* cbase, struct hub_user* user, struc
|
|||||||
cbuf_append(buf, " ");
|
cbuf_append(buf, " ");
|
||||||
cbuf_append_format(buf, " - %s\n", command->description);
|
cbuf_append_format(buf, " - %s\n", command->description);
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
@@ -351,6 +353,7 @@ static int command_kick(struct command_base* cbase, struct hub_user* user, struc
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
hub_notify(cbase->hub, notify_info, "Kicking user \"%s\" (%s)", target->id.nick, user->id.nick);
|
||||||
cbuf_append_format(buf, "Kicking user \"%s\".", target->id.nick);
|
cbuf_append_format(buf, "Kicking user \"%s\".", target->id.nick);
|
||||||
hub_disconnect_user(cbase->hub, target, quit_kicked);
|
hub_disconnect_user(cbase->hub, target, quit_kicked);
|
||||||
}
|
}
|
||||||
@@ -360,6 +363,7 @@ static int command_kick(struct command_base* cbase, struct hub_user* user, struc
|
|||||||
static int command_reload(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd)
|
static int command_reload(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd)
|
||||||
{
|
{
|
||||||
cbase->hub->status = hub_status_restart;
|
cbase->hub->status = hub_status_restart;
|
||||||
|
hub_notify(cbase->hub, notify_info, "Reloading configuration (%s)", user->id.nick);
|
||||||
return command_status(cbase, user, cmd, cbuf_create_const("Reloading configuration..."));
|
return command_status(cbase, user, cmd, cbuf_create_const("Reloading configuration..."));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -382,6 +386,7 @@ static int command_unload(struct command_base* cbase, struct hub_user* user, str
|
|||||||
|
|
||||||
static int command_shutdown_hub(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd)
|
static int command_shutdown_hub(struct command_base* cbase, struct hub_user* user, struct hub_command* cmd)
|
||||||
{
|
{
|
||||||
|
hub_notify(cbase->hub, notify_warn, "Shutting down hub (%s)", user->id.nick);
|
||||||
cbase->hub->status = hub_status_shutdown;
|
cbase->hub->status = hub_status_shutdown;
|
||||||
return command_status(cbase, user, cmd, cbuf_create_const("Hub shutting down..."));
|
return command_status(cbase, user, cmd, cbuf_create_const("Hub shutting down..."));
|
||||||
}
|
}
|
||||||
@@ -422,7 +427,7 @@ static int command_whoip(struct command_base* cbase, struct hub_user* user, stru
|
|||||||
ret = uman_get_user_by_addr(cbase->hub->users, users, arg->data.range);
|
ret = uman_get_user_by_addr(cbase->hub->users, users, arg->data.range);
|
||||||
if (!ret)
|
if (!ret)
|
||||||
{
|
{
|
||||||
list_clear(users, NULL);
|
list_clear(users, &null_free);
|
||||||
list_destroy(users);
|
list_destroy(users);
|
||||||
return command_status(cbase, user, cmd, cbuf_create_const("No users found."));
|
return command_status(cbase, user, cmd, cbuf_create_const("No users found."));
|
||||||
}
|
}
|
||||||
@@ -430,14 +435,16 @@ static int command_whoip(struct command_base* cbase, struct hub_user* user, stru
|
|||||||
buf = cbuf_create(128 + ((MAX_NICK_LEN + INET6_ADDRSTRLEN + 5) * ret));
|
buf = cbuf_create(128 + ((MAX_NICK_LEN + INET6_ADDRSTRLEN + 5) * ret));
|
||||||
cbuf_append_format(buf, "*** %s: Found %d match%s:\n", cmd->prefix, ret, ((ret != 1) ? "es" : ""));
|
cbuf_append_format(buf, "*** %s: Found %d match%s:\n", cmd->prefix, ret, ((ret != 1) ? "es" : ""));
|
||||||
|
|
||||||
LIST_FOREACH(struct hub_user*, u, users,
|
u = (struct hub_user*) list_get_first(users);
|
||||||
|
while (u)
|
||||||
{
|
{
|
||||||
cbuf_append_format(buf, "%s (%s)\n", u->id.nick, user_get_address(u));
|
cbuf_append_format(buf, "%s (%s)\n", u->id.nick, user_get_address(u));
|
||||||
});
|
u = (struct hub_user*) list_get_next(users);
|
||||||
|
}
|
||||||
cbuf_append(buf, "\n");
|
cbuf_append(buf, "\n");
|
||||||
|
|
||||||
send_message(cbase, user, buf);
|
send_message(cbase, user, buf);
|
||||||
list_clear(users, NULL);
|
list_clear(users, &null_free);
|
||||||
list_destroy(users);
|
list_destroy(users);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -458,7 +465,8 @@ static int command_broadcast(struct command_base* cbase, struct hub_user* user,
|
|||||||
memcpy(from_sid, sid_to_string(user->id.sid), sizeof(from_sid));
|
memcpy(from_sid, sid_to_string(user->id.sid), sizeof(from_sid));
|
||||||
memcpy(pm_flag + 2, from_sid, sizeof(from_sid));
|
memcpy(pm_flag + 2, from_sid, sizeof(from_sid));
|
||||||
|
|
||||||
LIST_FOREACH(struct hub_user*, target, cbase->hub->users->list,
|
target = (struct hub_user*) list_get_first(cbase->hub->users->list);
|
||||||
|
while (target)
|
||||||
{
|
{
|
||||||
if (target != user)
|
if (target != user)
|
||||||
{
|
{
|
||||||
@@ -475,7 +483,8 @@ static int command_broadcast(struct command_base* cbase, struct hub_user* user,
|
|||||||
route_to_user(cbase->hub, target, command);
|
route_to_user(cbase->hub, target, command);
|
||||||
adc_msg_free(command);
|
adc_msg_free(command);
|
||||||
}
|
}
|
||||||
});
|
target = (struct hub_user*) list_get_next(cbase->hub->users->list);
|
||||||
|
}
|
||||||
|
|
||||||
cbuf_append_format(buf, "*** %s: Delivered to " PRINTF_SIZE_T " user%s", cmd->prefix, recipients, (recipients != 1 ? "s" : ""));
|
cbuf_append_format(buf, "*** %s: Delivered to " PRINTF_SIZE_T " user%s", cmd->prefix, recipients, (recipients != 1 ? "s" : ""));
|
||||||
send_message(cbase, user, buf);
|
send_message(cbase, user, buf);
|
||||||
@@ -507,7 +516,8 @@ static int command_log(struct command_base* cbase, struct hub_user* user, struct
|
|||||||
command_status(cbase, user, cmd, buf);
|
command_status(cbase, user, cmd, buf);
|
||||||
|
|
||||||
buf = cbuf_create(MAX_HELP_LINE);
|
buf = cbuf_create(MAX_HELP_LINE);
|
||||||
LIST_FOREACH(struct hub_logout_info*, log, messages,
|
log = (struct hub_logout_info*) list_get_first(messages);
|
||||||
|
while (log)
|
||||||
{
|
{
|
||||||
const char* address = ip_convert_to_string(&log->addr);
|
const char* address = ip_convert_to_string(&log->addr);
|
||||||
int show = 0;
|
int show = 0;
|
||||||
@@ -531,7 +541,8 @@ static int command_log(struct command_base* cbase, struct hub_user* user, struct
|
|||||||
send_message(cbase, user, buf);
|
send_message(cbase, user, buf);
|
||||||
buf = cbuf_create(MAX_HELP_LINE);
|
buf = cbuf_create(MAX_HELP_LINE);
|
||||||
}
|
}
|
||||||
});
|
log = (struct hub_logout_info*) list_get_next(messages);
|
||||||
|
}
|
||||||
|
|
||||||
if (search_len)
|
if (search_len)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -76,7 +76,7 @@ static int config_parse_line(char* line, int line_count, void* ptr_data)
|
|||||||
struct hub_config* config = (struct hub_config*) ptr_data;
|
struct hub_config* config = (struct hub_config*) ptr_data;
|
||||||
|
|
||||||
strip_off_ini_line_comments(line, line_count);
|
strip_off_ini_line_comments(line, line_count);
|
||||||
|
|
||||||
if (!*line) return 0;
|
if (!*line) return 0;
|
||||||
|
|
||||||
LOG_DUMP("config_parse_line(): '%s'", line);
|
LOG_DUMP("config_parse_line(): '%s'", line);
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
289
src/core/config.pl
Executable file
289
src/core/config.pl
Executable file
@@ -0,0 +1,289 @@
|
|||||||
|
#!/usr/bin/perl -w
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use XML::DOM;
|
||||||
|
|
||||||
|
sub write_c_header(@);
|
||||||
|
sub write_sql_dump(@);
|
||||||
|
sub get_data($);
|
||||||
|
|
||||||
|
my $dump_to_sql = 0;
|
||||||
|
|
||||||
|
# initialize parser and read the file
|
||||||
|
my $input = "./config.xml";
|
||||||
|
my $parser = new XML::DOM::Parser;
|
||||||
|
my $tree = $parser->parsefile($input) || die "Unable to parse XML file.";
|
||||||
|
|
||||||
|
# Get data
|
||||||
|
my $nodes = $tree->getElementsByTagName("option");
|
||||||
|
my @options = ();
|
||||||
|
for (my $i = 0; $i < $nodes->getLength; $i++)
|
||||||
|
{
|
||||||
|
my @data = get_data($nodes->item($i));
|
||||||
|
push @options, \@data;
|
||||||
|
}
|
||||||
|
|
||||||
|
write_c_header(@options);
|
||||||
|
write_sql_dump(@options) if ($dump_to_sql);
|
||||||
|
|
||||||
|
my $config_defaults = "void config_defaults(struct hub_config* config)\n{\n";
|
||||||
|
my $config_apply = "static int apply_config(struct hub_config* config, char* key, char* data, int line_count)\n{\n\tint max = 0;\n\tint min = 0;\n\n";
|
||||||
|
my $config_free = "void free_config(struct hub_config* config)\n{\n";
|
||||||
|
my $config_dump = "void dump_config(struct hub_config* config, int ignore_defaults)\n{\n";
|
||||||
|
|
||||||
|
foreach my $option (@options)
|
||||||
|
{
|
||||||
|
my ($type, $name, $default, $advanced, $short, $desc, $since, $example, $check, $ifdef) = @$option;
|
||||||
|
my $string = ($type =~ /(string|file|message)/);
|
||||||
|
my $min = undef;
|
||||||
|
my $max = undef;
|
||||||
|
my $regexp = undef;
|
||||||
|
|
||||||
|
if (defined $check)
|
||||||
|
{
|
||||||
|
$min = $check->getAttribute("min");
|
||||||
|
$max = $check->getAttribute("max");
|
||||||
|
$regexp = $check->getAttribute("regexp");
|
||||||
|
|
||||||
|
$max = undef if ($max eq "");
|
||||||
|
$min = undef if ($min eq "");
|
||||||
|
$regexp = undef if ($regexp eq "");
|
||||||
|
}
|
||||||
|
|
||||||
|
$config_defaults .= "#ifdef $ifdef\n" if ($ifdef ne "");
|
||||||
|
$config_defaults .= "\tconfig->$name = ";
|
||||||
|
$config_defaults .= "hub_strdup(\"" if ($string);
|
||||||
|
$config_defaults .= $default;
|
||||||
|
$config_defaults .= "\")" if ($string);
|
||||||
|
$config_defaults .= ";\n";
|
||||||
|
$config_defaults .= "#endif /* $ifdef */\n" if ($ifdef ne "");
|
||||||
|
|
||||||
|
$config_apply .= "#ifdef $ifdef\n" if ($ifdef ne "");
|
||||||
|
$config_apply .= "\tif (!strcmp(key, \"" . $name . "\"))\n\t{\n";
|
||||||
|
|
||||||
|
if ($type eq "int")
|
||||||
|
{
|
||||||
|
$config_apply .= "\t\tmin = $min;\n" if (defined $min);
|
||||||
|
$config_apply .= "\t\tmax = $max;\n" if (defined $max);
|
||||||
|
$config_apply .= "\t\tif (!apply_integer(key, data, &config->$name, ";
|
||||||
|
if (defined $min) { $config_apply .= "&min"; } else { $config_apply .= "0"; }
|
||||||
|
$config_apply .= ", ";
|
||||||
|
if (defined $max) { $config_apply .= "&max"; } else { $config_apply .= "0"; }
|
||||||
|
$config_apply .= "))\n";
|
||||||
|
}
|
||||||
|
elsif ($type eq "boolean")
|
||||||
|
{
|
||||||
|
$config_apply .= "\t\tif (!apply_boolean(key, data, &config->$name))\n";
|
||||||
|
}
|
||||||
|
elsif ($string)
|
||||||
|
{
|
||||||
|
$config_apply .="\t\tif (!apply_string(key, data, &config->$name, (char*) \"\"))\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$config_apply .= "\t\t{\n" .
|
||||||
|
"\t\t\tLOG_ERROR(\"Configuration parse error on line %d\", line_count);\n" .
|
||||||
|
"\t\t\treturn -1;\n" .
|
||||||
|
"\t\t}\n" .
|
||||||
|
"\t\treturn 0;\n" .
|
||||||
|
"\t}\n";
|
||||||
|
$config_apply .= "#endif /* $ifdef */\n" if ($ifdef ne "");
|
||||||
|
$config_apply .= "\n";
|
||||||
|
|
||||||
|
if ($string)
|
||||||
|
{
|
||||||
|
$config_free .= "#ifdef $ifdef\n" if ($ifdef ne "");
|
||||||
|
$config_free .= "\thub_free(config->" . $name . ");\n";
|
||||||
|
$config_free .= "#endif /* $ifdef */\n" if ($ifdef ne "");
|
||||||
|
$config_free .= "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
my $out = "%s";
|
||||||
|
my $val = "config->$name";
|
||||||
|
my $test = "config->$name != $default";
|
||||||
|
|
||||||
|
$out = "%d" if ($type eq "int");
|
||||||
|
$val = "config->$name ? \"yes\" : \"no\"" if ($type eq "boolean");
|
||||||
|
|
||||||
|
if ($string)
|
||||||
|
{
|
||||||
|
$out = "\\\"%s\\\"";
|
||||||
|
$test = "strcmp(config->$name, \"$default\") != 0";
|
||||||
|
}
|
||||||
|
|
||||||
|
$config_dump .= "#ifdef $ifdef\n" if ($ifdef ne "");
|
||||||
|
$config_dump .= "\tif (!ignore_defaults || $test)\n";
|
||||||
|
$config_dump .= "\t\tfprintf(stdout, \"$name = $out\\n\", $val);\n";
|
||||||
|
$config_dump .= "#endif /* $ifdef */\n" if ($ifdef ne "");
|
||||||
|
$config_dump .= "\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
$config_apply .= "\t/* Still here -- unknown directive */\n";
|
||||||
|
$config_apply .= "\tLOG_ERROR(\"Unknown configuration directive: '%s'\", key);\n";
|
||||||
|
$config_apply .= "\treturn -1;\n";
|
||||||
|
$config_apply .= "}\n\n";
|
||||||
|
$config_defaults .= "}\n\n";
|
||||||
|
$config_free .= "}\n\n";
|
||||||
|
$config_dump .= "}\n\n";
|
||||||
|
|
||||||
|
open GENIMPL, ">gen_config.c" || die "Unable to write source file";
|
||||||
|
print GENIMPL "/* THIS FILE IS AUTOGENERATED - DO NOT CHANGE IT! */\n\n";
|
||||||
|
print GENIMPL $config_defaults;
|
||||||
|
print GENIMPL $config_apply;
|
||||||
|
print GENIMPL $config_free;
|
||||||
|
print GENIMPL $config_dump;
|
||||||
|
|
||||||
|
|
||||||
|
sub get_data($)
|
||||||
|
{
|
||||||
|
my $p = shift;
|
||||||
|
|
||||||
|
my $short = "";
|
||||||
|
my $example = "";
|
||||||
|
my $description = "";
|
||||||
|
my $since = "";
|
||||||
|
my $ifdef = "";
|
||||||
|
|
||||||
|
$short = $p->getElementsByTagName("short")->item(0)->getFirstChild()->getData() if ($p->getElementsByTagName("short")->getLength());
|
||||||
|
$since = $p->getElementsByTagName("since")->item(0)->getFirstChild()->getData() if ($p->getElementsByTagName("since")->getLength());
|
||||||
|
$example = $p->getElementsByTagName("example")->item(0)->getFirstChild()->getData() if ($p->getElementsByTagName("example")->getLength());
|
||||||
|
$description = $p->getElementsByTagName("description")->item(0)->getFirstChild()->getData() if ($p->getElementsByTagName("description")->getLength());
|
||||||
|
my $check = $p->getElementsByTagName("check")->item(0);
|
||||||
|
$ifdef = $p->getElementsByTagName("ifdef")->item(0)->getFirstChild()->getData() if ($p->getElementsByTagName("ifdef")->getLength());
|
||||||
|
|
||||||
|
my @data = (
|
||||||
|
$p->getAttribute("type"),
|
||||||
|
$p->getAttribute("name"),
|
||||||
|
$p->getAttribute("default"),
|
||||||
|
$p->getAttribute("advanced"),
|
||||||
|
$short,
|
||||||
|
$description,
|
||||||
|
$since,
|
||||||
|
$example,
|
||||||
|
$check,
|
||||||
|
$ifdef
|
||||||
|
);
|
||||||
|
return @data;
|
||||||
|
}
|
||||||
|
|
||||||
|
# Write header file
|
||||||
|
sub write_c_header(@)
|
||||||
|
{
|
||||||
|
my @data = @_;
|
||||||
|
|
||||||
|
open GENHEAD, ">gen_config.h" || die "Unable to write header file";
|
||||||
|
print GENHEAD "/* THIS FILE IS AUTOGENERATED - DO NOT CHANGE IT! */\n\n";
|
||||||
|
print GENHEAD "struct hub_config\n{\n";
|
||||||
|
|
||||||
|
foreach my $option (@data)
|
||||||
|
{
|
||||||
|
my ($type, $name, $default, $advanced, $short, $desc, $since, $example, $check, $ifdef) = @$option;
|
||||||
|
|
||||||
|
my $string = ($type =~ /(string|file|message)/);
|
||||||
|
|
||||||
|
print GENHEAD "#ifdef $ifdef\n" if ($ifdef ne "");
|
||||||
|
|
||||||
|
print GENHEAD "\t";
|
||||||
|
print GENHEAD "int " if ($type eq "int");
|
||||||
|
print GENHEAD "int " if ($type eq "boolean");
|
||||||
|
print GENHEAD "char*" if ($string);
|
||||||
|
print GENHEAD " " . $name . ";";
|
||||||
|
|
||||||
|
my $comment = "";
|
||||||
|
if ($type eq "message")
|
||||||
|
{
|
||||||
|
$comment = "\"" . $default . "\"";
|
||||||
|
}
|
||||||
|
elsif (defined $short && length $short > 0)
|
||||||
|
{
|
||||||
|
$comment = $short;
|
||||||
|
if (defined $default)
|
||||||
|
{
|
||||||
|
$comment .= " (default: ";
|
||||||
|
$comment .= "\"" if ($string);
|
||||||
|
$comment .= $default;
|
||||||
|
$comment .= "\"" if ($string);
|
||||||
|
$comment .= ")";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (length $comment > 0)
|
||||||
|
{
|
||||||
|
my $pad = "";
|
||||||
|
for (my $i = length $name; $i < 32; $i++)
|
||||||
|
{
|
||||||
|
$pad .= " ";
|
||||||
|
}
|
||||||
|
$comment = $pad . "/*<<< " . $comment . " */";
|
||||||
|
}
|
||||||
|
print GENHEAD $comment . "\n";
|
||||||
|
print GENHEAD "#endif /* $ifdef */\n" if ($ifdef ne "");
|
||||||
|
}
|
||||||
|
|
||||||
|
print GENHEAD "};\n\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
sub write_sql_dump(@)
|
||||||
|
{
|
||||||
|
my @data = @_;
|
||||||
|
|
||||||
|
# Write SQL dump code
|
||||||
|
open GENSQL, ">gen_config.sql" || die "Unable to write SQL dump";
|
||||||
|
print GENSQL "START TRANSACTION;\n\n
|
||||||
|
DROP TABLE uhub_config IF EXISTS;\n\n
|
||||||
|
CREATE TABLE uhub_config (
|
||||||
|
name VARCHAR(32) UNIQUE NOT NULL,
|
||||||
|
defaultValue TINYTEXT NOT NULL,
|
||||||
|
description LONGTEXT NOT NULL,
|
||||||
|
type TINYTEXT NOT NULL,
|
||||||
|
advanced BOOLEAN,
|
||||||
|
example LONGTEXT,
|
||||||
|
since TINYTEXT
|
||||||
|
);\n\n";
|
||||||
|
|
||||||
|
foreach my $option (@data)
|
||||||
|
{
|
||||||
|
my ($type, $name, $default, $advanced, $short, $desc, $since, $example, $check, $ifdef) = @$option;
|
||||||
|
|
||||||
|
if ($type =~ /(string|file|message)/ )
|
||||||
|
{
|
||||||
|
$default = "\\\"$default\\\"";
|
||||||
|
}
|
||||||
|
|
||||||
|
$desc =~ s/\"/\\\"/g;
|
||||||
|
$type =~ s/^int$/integer/;
|
||||||
|
|
||||||
|
my $stmt = "INSERT INTO uhub_config VALUES(";
|
||||||
|
$stmt .= "\"$name\", ";
|
||||||
|
$stmt .= "\"$default\", ";
|
||||||
|
$stmt .= "\"$desc\", ";
|
||||||
|
$stmt .= "\"$type\", ";
|
||||||
|
|
||||||
|
if (defined $example)
|
||||||
|
{
|
||||||
|
my $example_str = $example;
|
||||||
|
$example_str =~ s/\\/\\\\/g;
|
||||||
|
$example_str =~ s/\"/\\\"/g;
|
||||||
|
$stmt .= "\"$example_str\", ";
|
||||||
|
} else {
|
||||||
|
$stmt .= "NULL, ";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined $since) {
|
||||||
|
$stmt .= "\"$since\", ";
|
||||||
|
} else {
|
||||||
|
$stmt .= "NULL, ";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (defined $advanced) {
|
||||||
|
$stmt .= "\"$advanced\"";
|
||||||
|
} else {
|
||||||
|
$stmt .= "NULL";
|
||||||
|
}
|
||||||
|
|
||||||
|
$stmt .= ");\n";
|
||||||
|
|
||||||
|
print GENSQL $stmt;
|
||||||
|
}
|
||||||
|
print GENSQL "\n\nCOMMIT;\n\n";
|
||||||
|
}
|
||||||
@@ -1,255 +0,0 @@
|
|||||||
#!/usr/bin/env python
|
|
||||||
"""
|
|
||||||
uhub - A tiny ADC p2p connection hub
|
|
||||||
Copyright (C) 2007-2013, Jan Vidar Krey
|
|
||||||
"""
|
|
||||||
|
|
||||||
from xml.dom import minidom, Node
|
|
||||||
from datetime import datetime
|
|
||||||
import argparse
|
|
||||||
|
|
||||||
class OptionParseError(Exception):
|
|
||||||
pass
|
|
||||||
|
|
||||||
class Option(object):
|
|
||||||
def _get(self, node, name):
|
|
||||||
self.__dict__[name] = None
|
|
||||||
if (node.getElementsByTagName(name)):
|
|
||||||
self.__dict__[name] = node.getElementsByTagName(name)[0].firstChild.nodeValue
|
|
||||||
|
|
||||||
def _attr(self, node, name, required = False):
|
|
||||||
try:
|
|
||||||
return node.attributes[name].value
|
|
||||||
except Exception:
|
|
||||||
pass
|
|
||||||
if (required):
|
|
||||||
raise OptionParseError("Option %s is required but not found!" % name)
|
|
||||||
return None
|
|
||||||
|
|
||||||
def __init__(self, node):
|
|
||||||
self.otype = self._attr(node, 'type', True)
|
|
||||||
|
|
||||||
# Verify that the type is known
|
|
||||||
if not self.otype in ["int", "boolean", "string", "message", "file"]:
|
|
||||||
raise OptionParseError("Option %s has unknown type" % self.name)
|
|
||||||
|
|
||||||
self.name = self._attr(node, 'name', True)
|
|
||||||
self.default = self._attr(node, 'default', True)
|
|
||||||
self.advanced = self._attr(node, 'advanced', False)
|
|
||||||
self.is_string = self.otype in ["string", "message", "file"]
|
|
||||||
|
|
||||||
self._get(node, "short");
|
|
||||||
self._get(node, "description");
|
|
||||||
self._get(node, "syntax");
|
|
||||||
self._get(node, "since");
|
|
||||||
self._get(node, "example");
|
|
||||||
|
|
||||||
check = node.getElementsByTagName("check")
|
|
||||||
if (check):
|
|
||||||
check = node.getElementsByTagName("check")[0]
|
|
||||||
self.check_min = self._attr(check, 'min', False)
|
|
||||||
self.check_max = self._attr(check, 'max', False)
|
|
||||||
self.check_regexp = self._attr(check, 'regexp', False)
|
|
||||||
else:
|
|
||||||
self.check_min = None
|
|
||||||
self.check_max = None
|
|
||||||
self.check_regexp = None
|
|
||||||
|
|
||||||
|
|
||||||
def c_type(self):
|
|
||||||
if self.otype == "boolean":
|
|
||||||
return "int"
|
|
||||||
elif self.is_string:
|
|
||||||
return "char*"
|
|
||||||
else:
|
|
||||||
return self.otype
|
|
||||||
|
|
||||||
def sql_type(self):
|
|
||||||
if self.otype == "int":
|
|
||||||
return "integer"
|
|
||||||
return self.otype
|
|
||||||
|
|
||||||
def c_comment(self):
|
|
||||||
comment = ""
|
|
||||||
if (self.otype == "message"):
|
|
||||||
comment = self.formatted_default()
|
|
||||||
elif len(self.short):
|
|
||||||
comment = "%s (default: %s)" % (self.short, self.formatted_default())
|
|
||||||
return comment
|
|
||||||
|
|
||||||
def formatted_default(self):
|
|
||||||
if self.is_string:
|
|
||||||
return "\"%s\"" % self.default
|
|
||||||
return self.default
|
|
||||||
|
|
||||||
class SourceGenerator(object):
|
|
||||||
def __init__(self, filename, cppStyle = True):
|
|
||||||
print "Generating %s..." % filename
|
|
||||||
self.f = open(filename, 'w');
|
|
||||||
|
|
||||||
def write_header(self, Comment = True):
|
|
||||||
if Comment:
|
|
||||||
s = "/*\n * uhub - A tiny ADC p2p connection hub\n"
|
|
||||||
s += " * Copyright (C) 2007-%s, Jan Vidar Krey\n *\n" % datetime.now().strftime("%Y")
|
|
||||||
s += " * THIS FILE IS AUTOGENERATED - DO NOT MODIFY\n"
|
|
||||||
s += " * Created %s, by config.py\n */\n\n" % datetime.now().strftime("%Y-%m-%d %H:%M")
|
|
||||||
self.f.write(s)
|
|
||||||
|
|
||||||
class CHeaderGenerator(SourceGenerator):
|
|
||||||
def __init__(self, filename):
|
|
||||||
super(CHeaderGenerator, self).__init__(filename)
|
|
||||||
|
|
||||||
def _write_declaration(self, option):
|
|
||||||
comment = ' ' * (32 - len(option.name)) + "/*<<< %s */" % option.c_comment()
|
|
||||||
ptype = option.c_type() + (5 - len(option.c_type())) * ' '
|
|
||||||
self.f.write("\t%(type)s %(name)s;%(comment)s\n" % {
|
|
||||||
"type": ptype,
|
|
||||||
"name": option.name,
|
|
||||||
"comment": comment})
|
|
||||||
|
|
||||||
def write(self, options):
|
|
||||||
self.write_header()
|
|
||||||
self.f.write("struct hub_config\n{\n")
|
|
||||||
for option in options:
|
|
||||||
self._write_declaration(option)
|
|
||||||
self.f.write("};\n\n")
|
|
||||||
|
|
||||||
class CSourceGenerator(SourceGenerator):
|
|
||||||
def __init__(self, filename):
|
|
||||||
super(CSourceGenerator, self).__init__(filename)
|
|
||||||
|
|
||||||
def _write_default_impl(self, option):
|
|
||||||
s = "\tconfig->%s = " % option.name
|
|
||||||
if option.is_string:
|
|
||||||
s += "hub_strdup(%s);\n" % option.formatted_default()
|
|
||||||
else:
|
|
||||||
s += option.formatted_default() + ";\n"
|
|
||||||
self.f.write(s)
|
|
||||||
|
|
||||||
def _write_apply_impl(self, option):
|
|
||||||
s = "\tif (!strcmp(key, \"%s\"))\n\t{\n" % option.name
|
|
||||||
if option.otype == "int":
|
|
||||||
s_min = "0"
|
|
||||||
s_max = "0"
|
|
||||||
if (option.check_min):
|
|
||||||
s += "\t\tmin = %s;\n" % option.check_min
|
|
||||||
s_min = "&min"
|
|
||||||
if (option.check_max):
|
|
||||||
s += "\t\tmax = %s;\n" % option.check_max
|
|
||||||
s_max = "&max"
|
|
||||||
s+= "\t\tif (!apply_integer(key, data, &config->%s, %s, %s))\n" % (option.name, s_min, s_max)
|
|
||||||
elif option.otype == "boolean":
|
|
||||||
s += "\t\tif (!apply_boolean(key, data, &config->%s))\n" % option.name
|
|
||||||
elif option.is_string:
|
|
||||||
s += "\t\tif (!apply_string(key, data, &config->%s, (char*) \"\"))\n" % option.name
|
|
||||||
s += "\t\t{\n\t\t\tLOG_ERROR(\"Configuration parse error on line %d\", line_count);\n\t\t\treturn -1;\n\t\t}\n\t\treturn 0;\n\t}\n\n"
|
|
||||||
self.f.write(s)
|
|
||||||
|
|
||||||
def _write_free_impl(self, option):
|
|
||||||
if option.is_string:
|
|
||||||
self.f.write("\thub_free(config->%s);\n\n" % option.name)
|
|
||||||
|
|
||||||
def _write_dump_impl(self, option):
|
|
||||||
s = ""
|
|
||||||
fmt = "%s"
|
|
||||||
val = "config->%s" % option.name
|
|
||||||
test = "config->%s != %s" % (option.name, option.default)
|
|
||||||
|
|
||||||
if (option.otype == "int"):
|
|
||||||
fmt = "%d"
|
|
||||||
elif (option.otype == "boolean"):
|
|
||||||
val = "config->%s ? \"yes\" : \"no\"" % option.name
|
|
||||||
elif (option.is_string):
|
|
||||||
fmt = "\\\"%s\\\"";
|
|
||||||
test = "strcmp(config->%s, %s) != 0" % (option.name, option.formatted_default())
|
|
||||||
|
|
||||||
s += "\tif (!ignore_defaults || %s)\n" % test;
|
|
||||||
s += "\t\tfprintf(stdout, \"%s = %s\\n\", %s);\n\n" % (option.name, fmt, val)
|
|
||||||
self.f.write(s)
|
|
||||||
|
|
||||||
def write(self, options):
|
|
||||||
self.write_header()
|
|
||||||
self.f.write("void config_defaults(struct hub_config* config)\n{\n")
|
|
||||||
for option in options:
|
|
||||||
self._write_default_impl(option)
|
|
||||||
self.f.write("}\n\n")
|
|
||||||
self.f.write("static int apply_config(struct hub_config* config, char* key, char* data, int line_count)\n{\n\tint max = 0;\n\tint min = 0;\n\n")
|
|
||||||
for option in options:
|
|
||||||
self._write_apply_impl(option)
|
|
||||||
self.f.write("\t/* Still here -- unknown directive */\n\tLOG_ERROR(\"Unknown configuration directive: '%s'\", key);\n\treturn -1;\n}\n\n")
|
|
||||||
self.f.write("void free_config(struct hub_config* config)\n{\n")
|
|
||||||
for option in options:
|
|
||||||
self._write_free_impl(option)
|
|
||||||
self.f.write("}\n\n")
|
|
||||||
self.f.write("void dump_config(struct hub_config* config, int ignore_defaults)\n{\n")
|
|
||||||
for option in options:
|
|
||||||
self._write_dump_impl(option)
|
|
||||||
self.f.write("}\n\n")
|
|
||||||
|
|
||||||
class SqlWebsiteDocsGenerator(SourceGenerator):
|
|
||||||
def __init__(self, filename, sqlite_support = False):
|
|
||||||
self.sqlite_support = sqlite_support
|
|
||||||
super(SqlWebsiteDocsGenerator, self).__init__(filename)
|
|
||||||
|
|
||||||
def _sql_escape(self, s):
|
|
||||||
if self.sqlite_support:
|
|
||||||
return s.replace("\"", "\"\"")
|
|
||||||
return s.replace("\"", "\\\"")
|
|
||||||
|
|
||||||
|
|
||||||
def _write_or_null(self, s):
|
|
||||||
if (not s or len(s) == 0):
|
|
||||||
return "NULL"
|
|
||||||
return "\"%s\"" % self._sql_escape(s)
|
|
||||||
|
|
||||||
def write(self, options):
|
|
||||||
self.write_header(False)
|
|
||||||
table = "uhub_config"
|
|
||||||
|
|
||||||
s = ""
|
|
||||||
if not self.sqlite_support:
|
|
||||||
s += "START TRANSACTION;\n\nDROP TABLE %(table)s IF EXISTS;" % { "table": table }
|
|
||||||
s += "\n\nCREATE TABLE %(table)s (\n\tname VARCHAR(32) UNIQUE NOT NULL,\n\tdefaultValue TINYTEXT NOT NULL,\n\tdescription LONGTEXT NOT NULL,\n\ttype TINYTEXT NOT NULL,\n\tadvanced BOOLEAN,\n\texample LONGTEXT,\n\tsince TINYTEXT\n);\n\n" % { "table": table }
|
|
||||||
self.f.write(s)
|
|
||||||
|
|
||||||
for option in options:
|
|
||||||
s = "INSERT INTO %(table)s VALUES(\"%(name)s\", \"%(default)s\", \"%(description)s\", \"%(type)s\", %(example)s, %(since)s, %(advanced)s);\n" % {
|
|
||||||
"table": table,
|
|
||||||
"name": self._sql_escape(option.name),
|
|
||||||
"default": self._sql_escape(option.formatted_default()),
|
|
||||||
"description": self._sql_escape(option.description),
|
|
||||||
"type": option.sql_type(),
|
|
||||||
"example": self._write_or_null(option.example),
|
|
||||||
"since": self._write_or_null(option.since),
|
|
||||||
"advanced": self._write_or_null(option.example),
|
|
||||||
}
|
|
||||||
self.f.write(s)
|
|
||||||
|
|
||||||
if not self.sqlite_support:
|
|
||||||
self.f.write("\n\nCOMMIT;\n\n")
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
|
||||||
# parser = argparse.ArgumentParser(description = "Configuration file parser and source generator")
|
|
||||||
# parser.add_argument("--in", nargs=1, type=argparse.FileType('r'), default="config.xml", help="Input file (config.xml)", required = True)
|
|
||||||
# parser.add_argument("--c-decl", nargs=1, type=argparse.FileType('w'), default="gen_config.h", help="Output file for C declarations (gen_config.h)")
|
|
||||||
# parser.add_argument("--c-impl", nargs=1, type=argparse.FileType('w'), default="gen_config.c", help="Output file for C implementation (gen_config.c)")
|
|
||||||
# parser.add_argument("--doc-sql", nargs=1, type=argparse.FileType('w'), help="Output file for SQL documentation")
|
|
||||||
# args = parser.parse_args()
|
|
||||||
|
|
||||||
xmldoc = minidom.parse("./config.xml")
|
|
||||||
opt_tags = xmldoc.getElementsByTagName('option')
|
|
||||||
options = []
|
|
||||||
for option in opt_tags:
|
|
||||||
opt = Option(option)
|
|
||||||
options.append(opt)
|
|
||||||
|
|
||||||
header = CHeaderGenerator("./gen_config.h");
|
|
||||||
header.write(options);
|
|
||||||
|
|
||||||
source = CSourceGenerator("./gen_config.c");
|
|
||||||
source.write(options);
|
|
||||||
|
|
||||||
#sql = SqlWebsiteDocsGenerator("./gen_config.sql", True);
|
|
||||||
#sql.write(options);
|
|
||||||
|
|
||||||
|
|
||||||
@@ -17,7 +17,7 @@
|
|||||||
</option>
|
</option>
|
||||||
|
|
||||||
<option name="server_bind_addr" type="string" default="any">
|
<option name="server_bind_addr" type="string" default="any">
|
||||||
<check regexp="[\x:.]+|any|loopback" />
|
<check regexp="(\d\.\d\.\d\.\d\)|(any)|(loopback)|(.*)" /><!-- FIXME: add better IPv6 regexp in the future! -->
|
||||||
<short>Server bind address</short>
|
<short>Server bind address</short>
|
||||||
<description><![CDATA[
|
<description><![CDATA[
|
||||||
Specify the IP address the local hub should bind to. This can be an IPv4 or IPv6 address, or one of the special addresses "any" or "loopback". <br />
|
Specify the IP address the local hub should bind to. This can be an IPv4 or IPv6 address, or one of the special addresses "any" or "loopback". <br />
|
||||||
@@ -58,7 +58,7 @@
|
|||||||
</option>
|
</option>
|
||||||
|
|
||||||
<option name="server_alt_ports" type="string" default="">
|
<option name="server_alt_ports" type="string" default="">
|
||||||
<check regexp="\d+(,\d+)*" />
|
<check regexp="((\d+)(,(\d+))*)?" />
|
||||||
<short>Comma separated list of alternative ports to listen to</short>
|
<short>Comma separated list of alternative ports to listen to</short>
|
||||||
<description><![CDATA[
|
<description><![CDATA[
|
||||||
In addition to the server_port the hub can listen to a list of alternative ports.
|
In addition to the server_port the hub can listen to a list of alternative ports.
|
||||||
@@ -157,7 +157,7 @@
|
|||||||
</option>
|
</option>
|
||||||
|
|
||||||
<option name="redirect_addr" type="string" default="">
|
<option name="redirect_addr" type="string" default="">
|
||||||
<check regexp="(adcs?|dchub)://.*" />
|
<check regexp="(adc|adcs|dchub)://.*" />
|
||||||
<short>A common hub redirect address.</short>
|
<short>A common hub redirect address.</short>
|
||||||
<description><![CDATA[
|
<description><![CDATA[
|
||||||
This is the redirect address used when the hub wants to redirect a client for not fulfilling some requirements.
|
This is the redirect address used when the hub wants to redirect a client for not fulfilling some requirements.
|
||||||
@@ -419,7 +419,7 @@
|
|||||||
</option>
|
</option>
|
||||||
|
|
||||||
<option name="tls_require_redirect_addr" type="string" default="">
|
<option name="tls_require_redirect_addr" type="string" default="">
|
||||||
<check regexp="(adcs?|dchub)://.*" />
|
<check regexp="(adc|adcs|dchub)://.*" />
|
||||||
<short>A redirect address in case a client connects using "adc://" when "adcs://" is required.</short>
|
<short>A redirect address in case a client connects using "adc://" when "adcs://" is required.</short>
|
||||||
<description><![CDATA[
|
<description><![CDATA[
|
||||||
This is the redirect address used when the hub wants to redirect a client for not using ADCS.
|
This is the redirect address used when the hub wants to redirect a client for not using ADCS.
|
||||||
@@ -428,10 +428,11 @@
|
|||||||
<since>0.3.3</since>
|
<since>0.3.3</since>
|
||||||
</option>
|
</option>
|
||||||
|
|
||||||
|
|
||||||
<option name="tls_certificate" type="file" default="">
|
<option name="tls_certificate" type="file" default="">
|
||||||
<short>Certificate file</short>
|
<short>Certificate file</short>
|
||||||
<description><![CDATA[
|
<description><![CDATA[
|
||||||
Path to a TLS/SSL certificate or certificate chain (PEM format).
|
Path to a TLS/SSL certificate (PEM format).
|
||||||
]]></description>
|
]]></description>
|
||||||
<since>0.3.0</since>
|
<since>0.3.0</since>
|
||||||
</option>
|
</option>
|
||||||
@@ -444,40 +445,6 @@
|
|||||||
<since>0.3.0</since>
|
<since>0.3.0</since>
|
||||||
</option>
|
</option>
|
||||||
|
|
||||||
<option name="tls_ciphersuite" type="string" default="ECDH+AESGCM:DH+AESGCM:ECDH+AES256:DH+AES256:ECDH+AES128:DH+AES:ECDH+3DES:DH+3DES:RSA+AESGCM:RSA+AES:RSA+3DES:!aNULL:!MD5:!DSS">
|
|
||||||
<short>List of TLS ciphers to use</short>
|
|
||||||
<description><![CDATA[
|
|
||||||
This is a colon separated list of preferred ciphers in the OpenSSL format.
|
|
||||||
]]></description>
|
|
||||||
<since>0.5.0</since>
|
|
||||||
<example><![CDATA[
|
|
||||||
<p>
|
|
||||||
High security with emphasis on forward secrecy:<br />
|
|
||||||
tls_ciphersuite = "ECDH+AESGCM:DH+AESGCM:ECDH+AES256:DH+AES256:ECDH+AES128:DH+AES:ECDH+3DES:DH+3DES:RSA+AESGCM:RSA+AES:RSA+3DES:!aNULL:!MD5:!DSS"
|
|
||||||
</p>
|
|
||||||
<p>
|
|
||||||
Allow ChaCha20/Poly1305 which are secure, yet generally faster:<br />
|
|
||||||
tls_ciphersuite = "ECDHE-ECDSA-CHACHA20-POLY1305:ECDHE-RSA-CHACHA20-POLY1305:DHE-RSA-CHACHA20-POLY1305:ECDHE-ECDSA-AES128-GCM-SHA256:ECDHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES256-GCM-SHA384:ECDHE-RSA-AES256-GCM-SHA384:DHE-RSA-AES128-GCM-SHA256:ECDHE-ECDSA-AES128-SHA256:ECDHE-RSA-AES128-SHA256:ECDHE-ECDSA-AES256-SHA384:ECDHE-RSA-AES256-SHA384:DHE-RSA-AES128-SHA256:ECDHE-ECDSA-AES128-SHA:ECDHE-RSA-AES128-SHA:ECDHE-ECDSA-AES256-SHA:ECDHE-RSA-AES256-SHA:DHE-RSA-AES128-SHA:AES128-GCM-SHA256:AES256-GCM-SHA384:AES128-SHA256:AES256-SHA256:AES128-SHA:AES256-SHA"
|
|
||||||
</p>
|
|
||||||
]]></example>
|
|
||||||
</option>
|
|
||||||
|
|
||||||
<option name="tls_version" type="string" default="1.2">
|
|
||||||
<short>Specify minimum TLS version supported.</short>
|
|
||||||
<description><![CDATA[
|
|
||||||
<p>
|
|
||||||
This allows you to specify the minimum TLS version the hub requires from connecting clients in order to
|
|
||||||
connect to the hub.
|
|
||||||
</p>
|
|
||||||
<p>
|
|
||||||
TLS version 1.2 is recommended and enabled by default.
|
|
||||||
TLS version 1.1 is acceptable without any known flaws, and allows for older clients to connect.
|
|
||||||
TLS version 1.0 should be avoided, even though it is the most compatible with older ADC clients.
|
|
||||||
</p>
|
|
||||||
]]></description>
|
|
||||||
<since>0.5.0</since>
|
|
||||||
</option>
|
|
||||||
|
|
||||||
<option name="file_acl" type="file" default="">
|
<option name="file_acl" type="file" default="">
|
||||||
<short>File containing access control lists</short>
|
<short>File containing access control lists</short>
|
||||||
<description><![CDATA[
|
<description><![CDATA[
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -32,20 +32,20 @@ int event_queue_initialize(struct event_queue** queue, event_queue_callback call
|
|||||||
*queue = (struct event_queue*) hub_malloc_zero(sizeof(struct event_queue));
|
*queue = (struct event_queue*) hub_malloc_zero(sizeof(struct event_queue));
|
||||||
if (!(*queue))
|
if (!(*queue))
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
(*queue)->q1 = list_create();
|
(*queue)->q1 = list_create();
|
||||||
(*queue)->q2 = list_create();
|
(*queue)->q2 = list_create();
|
||||||
|
|
||||||
if (!(*queue)->q1 || !(*queue)->q2)
|
if (!(*queue)->q1 || !(*queue)->q2)
|
||||||
{
|
{
|
||||||
list_destroy((*queue)->q1);
|
list_destroy((*queue)->q1);
|
||||||
list_destroy((*queue)->q2);
|
list_destroy((*queue)->q2);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
(*queue)->callback = callback;
|
(*queue)->callback = callback;
|
||||||
(*queue)->callback_data = ptr;
|
(*queue)->callback_data = ptr;
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -73,27 +73,35 @@ int event_queue_process(struct event_queue* queue)
|
|||||||
struct event_data* data;
|
struct event_data* data;
|
||||||
if (queue->locked)
|
if (queue->locked)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
/* lock primary queue, and handle the primary queue messages. */
|
/* lock primary queue, and handle the primary queue messages. */
|
||||||
queue->locked = 1;
|
queue->locked = 1;
|
||||||
|
|
||||||
LIST_FOREACH(struct event_data*, data, queue->q1,
|
data = (struct event_data*) list_get_first(queue->q1);
|
||||||
|
while (data)
|
||||||
{
|
{
|
||||||
#ifdef EQ_DEBUG
|
#ifdef EQ_DEBUG
|
||||||
eq_debug("EXEC", data);
|
eq_debug("EXEC", data);
|
||||||
#endif
|
#endif
|
||||||
queue->callback(queue->callback_data, data);
|
queue->callback(queue->callback_data, data);
|
||||||
});
|
data = (struct event_data*) list_get_next(queue->q1);
|
||||||
|
}
|
||||||
|
|
||||||
list_clear(queue->q1, event_queue_cleanup_callback);
|
list_clear(queue->q1, event_queue_cleanup_callback);
|
||||||
uhub_assert(list_size(queue->q1) == 0);
|
uhub_assert(list_size(queue->q1) == 0);
|
||||||
|
|
||||||
/* unlock queue */
|
/* unlock queue */
|
||||||
queue->locked = 0;
|
queue->locked = 0;
|
||||||
|
|
||||||
/* transfer from secondary queue to the primary queue. */
|
/* transfer from secondary queue to the primary queue. */
|
||||||
list_append_list(queue->q1, queue->q2);
|
data = (struct event_data*) list_get_first(queue->q2);
|
||||||
|
while (data)
|
||||||
|
{
|
||||||
|
list_remove(queue->q2, data);
|
||||||
|
list_append(queue->q1, data);
|
||||||
|
data = (struct event_data*) list_get_first(queue->q2);
|
||||||
|
}
|
||||||
|
|
||||||
/* if more events exist, schedule it */
|
/* if more events exist, schedule it */
|
||||||
if (list_size(queue->q1))
|
if (list_size(queue->q1))
|
||||||
{
|
{
|
||||||
@@ -106,18 +114,18 @@ void event_queue_post(struct event_queue* queue, struct event_data* message)
|
|||||||
{
|
{
|
||||||
struct linked_list* q = (!queue->locked) ? queue->q1 : queue->q2;
|
struct linked_list* q = (!queue->locked) ? queue->q1 : queue->q2;
|
||||||
struct event_data* data;
|
struct event_data* data;
|
||||||
|
|
||||||
data = (struct event_data*) hub_malloc(sizeof(struct event_data));
|
data = (struct event_data*) hub_malloc(sizeof(struct event_data));
|
||||||
if (data)
|
if (data)
|
||||||
{
|
{
|
||||||
data->id = message->id;
|
data->id = message->id;
|
||||||
data->ptr = message->ptr;
|
data->ptr = message->ptr;
|
||||||
data->flags = message->flags;
|
data->flags = message->flags;
|
||||||
|
|
||||||
#ifdef EQ_DEBUG
|
#ifdef EQ_DEBUG
|
||||||
eq_debug("POST", data);
|
eq_debug("POST", data);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
list_append(q, data);
|
list_append(q, data);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,10 +1,4 @@
|
|||||||
/*
|
/* THIS FILE IS AUTOGENERATED - DO NOT CHANGE IT! */
|
||||||
* uhub - A tiny ADC p2p connection hub
|
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
|
||||||
*
|
|
||||||
* THIS FILE IS AUTOGENERATED - DO NOT MODIFY
|
|
||||||
* Created 2014-07-29 12:22, by config.py
|
|
||||||
*/
|
|
||||||
|
|
||||||
void config_defaults(struct hub_config* config)
|
void config_defaults(struct hub_config* config)
|
||||||
{
|
{
|
||||||
@@ -51,8 +45,6 @@ void config_defaults(struct hub_config* config)
|
|||||||
config->tls_require_redirect_addr = hub_strdup("");
|
config->tls_require_redirect_addr = hub_strdup("");
|
||||||
config->tls_certificate = hub_strdup("");
|
config->tls_certificate = hub_strdup("");
|
||||||
config->tls_private_key = hub_strdup("");
|
config->tls_private_key = hub_strdup("");
|
||||||
config->tls_ciphersuite = hub_strdup("ECDH+AESGCM:DH+AESGCM:ECDH+AES256:DH+AES256:ECDH+AES128:DH+AES:ECDH+3DES:DH+3DES:RSA+AESGCM:RSA+AES:RSA+3DES:!aNULL:!MD5:!DSS");
|
|
||||||
config->tls_version = hub_strdup("1.2");
|
|
||||||
config->file_acl = hub_strdup("");
|
config->file_acl = hub_strdup("");
|
||||||
config->file_plugins = hub_strdup("");
|
config->file_plugins = hub_strdup("");
|
||||||
config->msg_hub_full = hub_strdup("Hub is full");
|
config->msg_hub_full = hub_strdup("Hub is full");
|
||||||
@@ -554,26 +546,6 @@ static int apply_config(struct hub_config* config, char* key, char* data, int li
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!strcmp(key, "tls_ciphersuite"))
|
|
||||||
{
|
|
||||||
if (!apply_string(key, data, &config->tls_ciphersuite, (char*) ""))
|
|
||||||
{
|
|
||||||
LOG_ERROR("Configuration parse error on line %d", line_count);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!strcmp(key, "tls_version"))
|
|
||||||
{
|
|
||||||
if (!apply_string(key, data, &config->tls_version, (char*) ""))
|
|
||||||
{
|
|
||||||
LOG_ERROR("Configuration parse error on line %d", line_count);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!strcmp(key, "file_acl"))
|
if (!strcmp(key, "file_acl"))
|
||||||
{
|
{
|
||||||
if (!apply_string(key, data, &config->file_acl, (char*) ""))
|
if (!apply_string(key, data, &config->file_acl, (char*) ""))
|
||||||
@@ -977,10 +949,6 @@ void free_config(struct hub_config* config)
|
|||||||
|
|
||||||
hub_free(config->tls_private_key);
|
hub_free(config->tls_private_key);
|
||||||
|
|
||||||
hub_free(config->tls_ciphersuite);
|
|
||||||
|
|
||||||
hub_free(config->tls_version);
|
|
||||||
|
|
||||||
hub_free(config->file_acl);
|
hub_free(config->file_acl);
|
||||||
|
|
||||||
hub_free(config->file_plugins);
|
hub_free(config->file_plugins);
|
||||||
@@ -1190,12 +1158,6 @@ void dump_config(struct hub_config* config, int ignore_defaults)
|
|||||||
if (!ignore_defaults || strcmp(config->tls_private_key, "") != 0)
|
if (!ignore_defaults || strcmp(config->tls_private_key, "") != 0)
|
||||||
fprintf(stdout, "tls_private_key = \"%s\"\n", config->tls_private_key);
|
fprintf(stdout, "tls_private_key = \"%s\"\n", config->tls_private_key);
|
||||||
|
|
||||||
if (!ignore_defaults || strcmp(config->tls_ciphersuite, "ECDH+AESGCM:DH+AESGCM:ECDH+AES256:DH+AES256:ECDH+AES128:DH+AES:ECDH+3DES:DH+3DES:RSA+AESGCM:RSA+AES:RSA+3DES:!aNULL:!MD5:!DSS") != 0)
|
|
||||||
fprintf(stdout, "tls_ciphersuite = \"%s\"\n", config->tls_ciphersuite);
|
|
||||||
|
|
||||||
if (!ignore_defaults || strcmp(config->tls_version, "1.2") != 0)
|
|
||||||
fprintf(stdout, "tls_version = \"%s\"\n", config->tls_version);
|
|
||||||
|
|
||||||
if (!ignore_defaults || strcmp(config->file_acl, "") != 0)
|
if (!ignore_defaults || strcmp(config->file_acl, "") != 0)
|
||||||
fprintf(stdout, "file_acl = \"%s\"\n", config->file_acl);
|
fprintf(stdout, "file_acl = \"%s\"\n", config->file_acl);
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,4 @@
|
|||||||
/*
|
/* THIS FILE IS AUTOGENERATED - DO NOT CHANGE IT! */
|
||||||
* uhub - A tiny ADC p2p connection hub
|
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
|
||||||
*
|
|
||||||
* THIS FILE IS AUTOGENERATED - DO NOT MODIFY
|
|
||||||
* Created 2014-07-29 12:22, by config.py
|
|
||||||
*/
|
|
||||||
|
|
||||||
struct hub_config
|
struct hub_config
|
||||||
{
|
{
|
||||||
@@ -51,8 +45,6 @@ struct hub_config
|
|||||||
char* tls_require_redirect_addr; /*<<< A redirect address in case a client connects using "adc://" when "adcs://" is required. (default: "") */
|
char* tls_require_redirect_addr; /*<<< A redirect address in case a client connects using "adc://" when "adcs://" is required. (default: "") */
|
||||||
char* tls_certificate; /*<<< Certificate file (default: "") */
|
char* tls_certificate; /*<<< Certificate file (default: "") */
|
||||||
char* tls_private_key; /*<<< Private key file (default: "") */
|
char* tls_private_key; /*<<< Private key file (default: "") */
|
||||||
char* tls_ciphersuite; /*<<< List of TLS ciphers to use (default: "ECDH+AESGCM:DH+AESGCM:ECDH+AES256:DH+AES256:ECDH+AES128:DH+AES:ECDH+3DES:DH+3DES:RSA+AESGCM:RSA+AES:RSA+3DES:!aNULL:!MD5:!DSS") */
|
|
||||||
char* tls_version; /*<<< Specify minimum TLS version supported. (default: "1.2") */
|
|
||||||
char* file_acl; /*<<< File containing access control lists (default: "") */
|
char* file_acl; /*<<< File containing access control lists (default: "") */
|
||||||
char* file_plugins; /*<<< Plugin configuration file (default: "") */
|
char* file_plugins; /*<<< Plugin configuration file (default: "") */
|
||||||
char* msg_hub_full; /*<<< "Hub is full" */
|
char* msg_hub_full; /*<<< "Hub is full" */
|
||||||
|
|||||||
120
src/core/hub.c
120
src/core/hub.c
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -653,6 +653,7 @@ static struct net_connection* start_listening_socket(const char* bind_addr, uint
|
|||||||
if (ret == -1)
|
if (ret == -1)
|
||||||
{
|
{
|
||||||
LOG_ERROR("hub_start_service(): Unable to bind to TCP local address. errno=%d, str=%s", net_error(), net_error_string(net_error()));
|
LOG_ERROR("hub_start_service(): Unable to bind to TCP local address. errno=%d, str=%s", net_error(), net_error_string(net_error()));
|
||||||
|
hub_notify(hub, notify_error, "Unable to bind to network address %s on port %d: %s (%d)", bind_addr, port, net_error_string(net_error()), net_error());
|
||||||
net_close(sd);
|
net_close(sd);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -671,6 +672,60 @@ static struct net_connection* start_listening_socket(const char* bind_addr, uint
|
|||||||
return server;
|
return server;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
int hub_is_running(struct hub_info* hub)
|
||||||
|
{
|
||||||
|
return hub->status == hub_status_running || hub->status == hub_status_restart;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void hub_notify(struct hub_info* hub, enum notify_verbosity verbosity, const char* fmt, ...)
|
||||||
|
{
|
||||||
|
struct cbuffer* buf;
|
||||||
|
struct adc_message* msg;
|
||||||
|
va_list args;
|
||||||
|
char temp[1024];
|
||||||
|
|
||||||
|
va_start(args, fmt);
|
||||||
|
vsnprintf(temp, sizeof(temp), fmt, args);
|
||||||
|
va_end(args);
|
||||||
|
|
||||||
|
buf = cbuf_create(strlen(temp) + 8);
|
||||||
|
|
||||||
|
switch (verbosity)
|
||||||
|
{
|
||||||
|
case notify_error:
|
||||||
|
cbuf_append(buf, "ERROR: ");
|
||||||
|
LOG_ERROR(temp);
|
||||||
|
break;
|
||||||
|
case notify_warn:
|
||||||
|
cbuf_append(buf, "WARN: ");
|
||||||
|
LOG_WARN(temp);
|
||||||
|
break;
|
||||||
|
case notify_info:
|
||||||
|
cbuf_append(buf, "INFO: ");
|
||||||
|
LOG_INFO(temp);
|
||||||
|
break;
|
||||||
|
case notify_debug:
|
||||||
|
cbuf_append(buf, "DEBUG: ");
|
||||||
|
LOG_DEBUG(temp);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
cbuf_append(buf, temp);
|
||||||
|
|
||||||
|
if (hub_is_running(hub))
|
||||||
|
{
|
||||||
|
msg = adc_msg_construct(ADC_CMD_IMSG, 5 + adc_msg_escape_length(cbuf_get(buf)) + 2);
|
||||||
|
adc_msg_add_argument_string(msg, cbuf_get(buf));
|
||||||
|
route_to_operators(hub, msg);
|
||||||
|
adc_msg_free(msg);
|
||||||
|
}
|
||||||
|
|
||||||
|
cbuf_destroy(buf);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
struct server_alt_port_data
|
struct server_alt_port_data
|
||||||
{
|
{
|
||||||
struct hub_info* hub;
|
struct hub_info* hub;
|
||||||
@@ -731,11 +786,7 @@ static int load_ssl_certificates(struct hub_info* hub, struct hub_config* config
|
|||||||
{
|
{
|
||||||
if (config->tls_enable)
|
if (config->tls_enable)
|
||||||
{
|
{
|
||||||
hub->ctx = net_ssl_context_create(config->tls_version, config->tls_ciphersuite);
|
hub->ctx = net_ssl_context_create();
|
||||||
|
|
||||||
if (!hub->ctx)
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
if (ssl_load_certificate(hub->ctx, config->tls_certificate) &&
|
if (ssl_load_certificate(hub->ctx, config->tls_certificate) &&
|
||||||
ssl_load_private_key(hub->ctx, config->tls_private_key) &&
|
ssl_load_private_key(hub->ctx, config->tls_private_key) &&
|
||||||
ssl_check_private_key(hub->ctx))
|
ssl_check_private_key(hub->ctx))
|
||||||
@@ -755,6 +806,54 @@ static void unload_ssl_certificates(struct hub_info* hub)
|
|||||||
}
|
}
|
||||||
#endif /* SSL_SUPPORT */
|
#endif /* SSL_SUPPORT */
|
||||||
|
|
||||||
|
// #ifdef BOT_SUPPORT
|
||||||
|
|
||||||
|
static void route_privmsg_to_operators(struct hub_user* bot, sid_t from, const char* escaped_msg, int action)
|
||||||
|
{
|
||||||
|
struct hub_info* hub = bot->hub;
|
||||||
|
struct hub_user* user = (struct hub_user*) list_get_first(hub->users->list);
|
||||||
|
while (user)
|
||||||
|
{
|
||||||
|
if (from != user->id.sid && user_flag_get(user, flag_opnotify))
|
||||||
|
{
|
||||||
|
struct adc_message* msg = adc_msg_construct_source_dest(ADC_CMD_EMSG, from, user->id.sid, strlen(escaped_msg) + (action * 4) + 7);
|
||||||
|
adc_msg_add_argument(msg, escaped_msg);
|
||||||
|
adc_msg_add_named_argument(msg, ADC_MSG_FLAG_PRIVATE, sid_to_string(bot->id.sid));
|
||||||
|
if (action) adc_msg_add_named_argument(msg, ADC_MSG_FLAG_ACTION, "1");
|
||||||
|
route_to_user(hub, user, msg);
|
||||||
|
adc_msg_free(msg);
|
||||||
|
}
|
||||||
|
user = (struct hub_user*) list_get_next(hub->users->list);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// This receives private messages and transmits them to the connected operators.
|
||||||
|
static void hub_bot_op_notify_handle(struct hub_user* bot, struct adc_message* msg)
|
||||||
|
{
|
||||||
|
char* chat;
|
||||||
|
LOG_TRACE("Invoked hub_bot_op_notify_handle()");
|
||||||
|
switch (msg->cmd)
|
||||||
|
{
|
||||||
|
case ADC_CMD_EMSG:
|
||||||
|
case ADC_CMD_DMSG:
|
||||||
|
chat = adc_msg_get_argument(msg, 0);
|
||||||
|
LOG_DEBUG("Hub chat: \"%s\"", chat);
|
||||||
|
route_privmsg_to_operators(bot, msg->source, chat, adc_msg_has_named_argument(msg, ADC_MSG_FLAG_ACTION) ? 1 : 0);
|
||||||
|
hub_free(chat);
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
/* ignore these messages! */
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void hub_bot_op_notify_create(struct hub_info* hub)
|
||||||
|
{
|
||||||
|
struct hub_user* opcom = user_create_bot(hub, "Operations", "Hub operators", hub_bot_op_notify_handle);
|
||||||
|
uman_add(hub->users, opcom);
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
|
||||||
struct hub_info* hub_start_service(struct hub_config* config)
|
struct hub_info* hub_start_service(struct hub_config* config)
|
||||||
{
|
{
|
||||||
struct hub_info* hub = 0;
|
struct hub_info* hub = 0;
|
||||||
@@ -838,6 +937,8 @@ struct hub_info* hub_start_service(struct hub_config* config)
|
|||||||
|
|
||||||
// Start the hub command sub-system
|
// Start the hub command sub-system
|
||||||
hub->commands = command_initialize(hub);
|
hub->commands = command_initialize(hub);
|
||||||
|
|
||||||
|
hub_bot_op_notify_create(hub);
|
||||||
return hub;
|
return hub;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -909,8 +1010,7 @@ void hub_set_variables(struct hub_info* hub, struct acl_handle* acl)
|
|||||||
if (hub->command_info)
|
if (hub->command_info)
|
||||||
{
|
{
|
||||||
adc_msg_add_named_argument(hub->command_info, ADC_INF_FLAG_CLIENT_TYPE, ADC_CLIENT_TYPE_HUB);
|
adc_msg_add_named_argument(hub->command_info, ADC_INF_FLAG_CLIENT_TYPE, ADC_CLIENT_TYPE_HUB);
|
||||||
adc_msg_add_named_argument(hub->command_info, ADC_INF_FLAG_USER_AGENT_PRODUCT, PRODUCT);
|
adc_msg_add_named_argument(hub->command_info, ADC_INF_FLAG_USER_AGENT, server);
|
||||||
adc_msg_add_named_argument(hub->command_info, ADC_INF_FLAG_USER_AGENT_VERSION, GIT_VERSION);
|
|
||||||
|
|
||||||
tmp = adc_msg_escape(hub->config->hub_name);
|
tmp = adc_msg_escape(hub->config->hub_name);
|
||||||
adc_msg_add_named_argument(hub->command_info, ADC_INF_FLAG_NICK, tmp);
|
adc_msg_add_named_argument(hub->command_info, ADC_INF_FLAG_NICK, tmp);
|
||||||
@@ -1325,7 +1425,9 @@ void hub_logout_log(struct hub_info* hub, struct hub_user* user)
|
|||||||
list_append(hub->logout_info, loginfo);
|
list_append(hub->logout_info, loginfo);
|
||||||
while (list_size(hub->logout_info) > (size_t) hub->config->max_logout_log)
|
while (list_size(hub->logout_info) > (size_t) hub->config->max_logout_log)
|
||||||
{
|
{
|
||||||
list_remove_first(hub->logout_info, hub_free);
|
struct hub_logout_info* entry = list_get_first(hub->logout_info);
|
||||||
|
list_remove(hub->logout_info, entry);
|
||||||
|
hub_free(entry);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -27,7 +27,7 @@ enum status_message
|
|||||||
status_msg_hub_registered_users_only = -3, /* hub is for registered users only */
|
status_msg_hub_registered_users_only = -3, /* hub is for registered users only */
|
||||||
status_msg_inf_error_nick_missing = -4, /* no nickname given */
|
status_msg_inf_error_nick_missing = -4, /* no nickname given */
|
||||||
status_msg_inf_error_nick_multiple = -5, /* multiple nicknames given */
|
status_msg_inf_error_nick_multiple = -5, /* multiple nicknames given */
|
||||||
status_msg_inf_error_nick_invalid = -6, /* generic/unknown */
|
status_msg_inf_error_nick_invalid = -6, /* generic/unkown */
|
||||||
status_msg_inf_error_nick_long = -7, /* nickname too long */
|
status_msg_inf_error_nick_long = -7, /* nickname too long */
|
||||||
status_msg_inf_error_nick_short = -8, /* nickname too short */
|
status_msg_inf_error_nick_short = -8, /* nickname too short */
|
||||||
status_msg_inf_error_nick_spaces = -9, /* nickname cannot start with spaces */
|
status_msg_inf_error_nick_spaces = -9, /* nickname cannot start with spaces */
|
||||||
@@ -50,12 +50,12 @@ enum status_message
|
|||||||
status_msg_user_share_size_high = -41, /* User is sharing too much. */
|
status_msg_user_share_size_high = -41, /* User is sharing too much. */
|
||||||
status_msg_user_slots_low = -42, /* User has too few slots open. */
|
status_msg_user_slots_low = -42, /* User has too few slots open. */
|
||||||
status_msg_user_slots_high = -43, /* User has too many slots open. */
|
status_msg_user_slots_high = -43, /* User has too many slots open. */
|
||||||
status_msg_user_hub_limit_low = -44, /* User is on too few hubs. */
|
status_msg_user_hub_limit_low = -44, /* Use is on too few hubs. */
|
||||||
status_msg_user_hub_limit_high = -45, /* User is on too many hubs. */
|
status_msg_user_hub_limit_high = -45, /* Use is on too many hubs. */
|
||||||
|
|
||||||
status_msg_proto_no_common_hash = -50, /* No common hash algorithms */
|
status_msg_proto_no_common_hash = -50, /* No common hash algorithms */
|
||||||
status_msg_proto_obsolete_adc0 = -51, /* Client is using an obsolete protocol version */
|
status_msg_proto_obsolete_adc0 = -51, /* Client is using an obsolete protocol version */
|
||||||
|
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -106,7 +106,7 @@ struct hub_info
|
|||||||
struct adc_message* command_support; /* The hub's SUP command */
|
struct adc_message* command_support; /* The hub's SUP command */
|
||||||
struct adc_message* command_banner; /* The default welcome message */
|
struct adc_message* command_banner; /* The default welcome message */
|
||||||
time_t tm_started;
|
time_t tm_started;
|
||||||
int status;
|
enum hub_state status;
|
||||||
char* recvbuf; /* Global receive buffer */
|
char* recvbuf; /* Global receive buffer */
|
||||||
char* sendbuf; /* Global send buffer */
|
char* sendbuf; /* Global send buffer */
|
||||||
|
|
||||||
@@ -233,6 +233,20 @@ extern void hub_set_variables(struct hub_info* hub, struct acl_handle* acl);
|
|||||||
*/
|
*/
|
||||||
extern void hub_free_variables(struct hub_info* hub);
|
extern void hub_free_variables(struct hub_info* hub);
|
||||||
|
|
||||||
|
|
||||||
|
enum notify_verbosity
|
||||||
|
{
|
||||||
|
notify_error = 0,
|
||||||
|
notify_warn = 1,
|
||||||
|
notify_info = 2,
|
||||||
|
notify_debug = 3,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a notification message to operators.
|
||||||
|
*/
|
||||||
|
extern void hub_notify(struct hub_info* hub, enum notify_verbosity verbosity, const char* fmt, ...);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a string for the given status_message (See enum status_message).
|
* Returns a string for the given status_message (See enum status_message).
|
||||||
*/
|
*/
|
||||||
@@ -350,6 +364,10 @@ extern void hub_disconnect_user(struct hub_info* hub, struct hub_user* user, int
|
|||||||
*/
|
*/
|
||||||
extern void hub_logout_log(struct hub_info* hub, struct hub_user* user);
|
extern void hub_logout_log(struct hub_info* hub, struct hub_user* user);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns 1 if the hub is running, and 0 otherwise.
|
||||||
|
*/
|
||||||
|
extern int hub_is_running(struct hub_info* hub);
|
||||||
|
|
||||||
#endif /* HAVE_UHUB_HUB_H */
|
#endif /* HAVE_UHUB_HUB_H */
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -31,6 +31,10 @@ void on_login_success(struct hub_info* hub, struct hub_user* u)
|
|||||||
user_set_state(u, state_normal);
|
user_set_state(u, state_normal);
|
||||||
uman_add(hub->users, u);
|
uman_add(hub->users, u);
|
||||||
|
|
||||||
|
// Make operators receive hub notifications by default.
|
||||||
|
if (user_is_protected(u))
|
||||||
|
user_flag_set(u, flag_opnotify);
|
||||||
|
|
||||||
/* Announce new user to all connected users */
|
/* Announce new user to all connected users */
|
||||||
if (user_is_logged_in(u))
|
if (user_is_logged_in(u))
|
||||||
route_info_message(hub, u);
|
route_info_message(hub, u);
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
113
src/core/inf.c
113
src/core/inf.c
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -109,14 +109,14 @@ static int check_cid(struct hub_info* hub, struct hub_user* user, struct adc_mes
|
|||||||
hub_free(pid);
|
hub_free(pid);
|
||||||
return status_msg_inf_error_cid_invalid;
|
return status_msg_inf_error_cid_invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (strlen(pid) != MAX_CID_LEN)
|
if (strlen(pid) != MAX_CID_LEN)
|
||||||
{
|
{
|
||||||
hub_free(cid);
|
hub_free(cid);
|
||||||
hub_free(pid);
|
hub_free(pid);
|
||||||
return status_msg_inf_error_pid_invalid;
|
return status_msg_inf_error_pid_invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
for (pos = 0; pos < MAX_CID_LEN; pos++)
|
for (pos = 0; pos < MAX_CID_LEN; pos++)
|
||||||
{
|
{
|
||||||
if (!is_valid_base32_char(cid[pos]))
|
if (!is_valid_base32_char(cid[pos]))
|
||||||
@@ -133,18 +133,18 @@ static int check_cid(struct hub_info* hub, struct hub_user* user, struct adc_mes
|
|||||||
return status_msg_inf_error_pid_invalid;
|
return status_msg_inf_error_pid_invalid;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!check_hash_tiger(cid, pid))
|
if (!check_hash_tiger(cid, pid))
|
||||||
{
|
{
|
||||||
hub_free(cid);
|
hub_free(cid);
|
||||||
hub_free(pid);
|
hub_free(pid);
|
||||||
return status_msg_inf_error_cid_invalid;
|
return status_msg_inf_error_cid_invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Set the cid in the user object */
|
/* Set the cid in the user object */
|
||||||
memcpy(user->id.cid, cid, MAX_CID_LEN);
|
memcpy(user->id.cid, cid, MAX_CID_LEN);
|
||||||
user->id.cid[MAX_CID_LEN] = 0;
|
user->id.cid[MAX_CID_LEN] = 0;
|
||||||
|
|
||||||
hub_free(cid);
|
hub_free(cid);
|
||||||
hub_free(pid);
|
hub_free(pid);
|
||||||
return 0;
|
return 0;
|
||||||
@@ -154,7 +154,7 @@ static int check_cid(struct hub_info* hub, struct hub_user* user, struct adc_mes
|
|||||||
static int check_required_login_flags(struct hub_info* hub, struct hub_user* user, struct adc_message* cmd)
|
static int check_required_login_flags(struct hub_info* hub, struct hub_user* user, struct adc_message* cmd)
|
||||||
{
|
{
|
||||||
int num = 0;
|
int num = 0;
|
||||||
|
|
||||||
num = adc_msg_has_named_argument(cmd, ADC_INF_FLAG_CLIENT_ID);
|
num = adc_msg_has_named_argument(cmd, ADC_INF_FLAG_CLIENT_ID);
|
||||||
if (num != 1)
|
if (num != 1)
|
||||||
{
|
{
|
||||||
@@ -162,7 +162,7 @@ static int check_required_login_flags(struct hub_info* hub, struct hub_user* use
|
|||||||
return status_msg_inf_error_cid_missing;
|
return status_msg_inf_error_cid_missing;
|
||||||
return status_msg_inf_error_cid_invalid;
|
return status_msg_inf_error_cid_invalid;
|
||||||
}
|
}
|
||||||
|
|
||||||
num = adc_msg_has_named_argument(cmd, ADC_INF_FLAG_PRIVATE_ID);
|
num = adc_msg_has_named_argument(cmd, ADC_INF_FLAG_PRIVATE_ID);
|
||||||
if (num != 1)
|
if (num != 1)
|
||||||
{
|
{
|
||||||
@@ -190,7 +190,7 @@ static int check_required_login_flags(struct hub_info* hub, struct hub_user* use
|
|||||||
static int check_network(struct hub_info* hub, struct hub_user* user, struct adc_message* cmd)
|
static int check_network(struct hub_info* hub, struct hub_user* user, struct adc_message* cmd)
|
||||||
{
|
{
|
||||||
const char* address = user_get_address(user);
|
const char* address = user_get_address(user);
|
||||||
|
|
||||||
/* Check for NAT override address */
|
/* Check for NAT override address */
|
||||||
if (acl_is_ip_nat_override(hub->acl, address))
|
if (acl_is_ip_nat_override(hub->acl, address))
|
||||||
{
|
{
|
||||||
@@ -236,12 +236,12 @@ static int nick_length_ok(const char* nick)
|
|||||||
{
|
{
|
||||||
return nick_invalid_short;
|
return nick_invalid_short;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (length > MAX_NICK_LEN)
|
if (length > MAX_NICK_LEN)
|
||||||
{
|
{
|
||||||
return nick_invalid_long;
|
return nick_invalid_long;
|
||||||
}
|
}
|
||||||
|
|
||||||
return nick_ok;
|
return nick_ok;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -249,16 +249,16 @@ static int nick_length_ok(const char* nick)
|
|||||||
static int nick_bad_characters(const char* nick)
|
static int nick_bad_characters(const char* nick)
|
||||||
{
|
{
|
||||||
const char* tmp;
|
const char* tmp;
|
||||||
|
|
||||||
/* Nick must not start with a space */
|
/* Nick must not start with a space */
|
||||||
if (nick[0] == ' ')
|
if (nick[0] == ' ')
|
||||||
return nick_invalid_spaces;
|
return nick_invalid_spaces;
|
||||||
|
|
||||||
/* Check for ASCII values below 32 */
|
/* Check for ASCII values below 32 */
|
||||||
for (tmp = nick; *tmp; tmp++)
|
for (tmp = nick; *tmp; tmp++)
|
||||||
if ((*tmp < 32) && (*tmp > 0))
|
if ((*tmp < 32) && (*tmp > 0))
|
||||||
return nick_invalid_bad_ascii;
|
return nick_invalid_bad_ascii;
|
||||||
|
|
||||||
return nick_ok;
|
return nick_ok;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -286,7 +286,7 @@ static int check_nick(struct hub_info* hub, struct hub_user* user, struct adc_me
|
|||||||
nick = adc_msg_unescape(tmp);
|
nick = adc_msg_unescape(tmp);
|
||||||
free(tmp); tmp = 0;
|
free(tmp); tmp = 0;
|
||||||
if (!nick) return 0;
|
if (!nick) return 0;
|
||||||
|
|
||||||
status = nick_length_ok(nick);
|
status = nick_length_ok(nick);
|
||||||
if (status != nick_ok)
|
if (status != nick_ok)
|
||||||
{
|
{
|
||||||
@@ -295,7 +295,7 @@ static int check_nick(struct hub_info* hub, struct hub_user* user, struct adc_me
|
|||||||
return status_msg_inf_error_nick_short;
|
return status_msg_inf_error_nick_short;
|
||||||
return status_msg_inf_error_nick_long;
|
return status_msg_inf_error_nick_long;
|
||||||
}
|
}
|
||||||
|
|
||||||
status = nick_bad_characters(nick);
|
status = nick_bad_characters(nick);
|
||||||
if (status != nick_ok)
|
if (status != nick_ok)
|
||||||
{
|
{
|
||||||
@@ -304,20 +304,20 @@ static int check_nick(struct hub_info* hub, struct hub_user* user, struct adc_me
|
|||||||
return status_msg_inf_error_nick_spaces;
|
return status_msg_inf_error_nick_spaces;
|
||||||
return status_msg_inf_error_nick_bad_chars;
|
return status_msg_inf_error_nick_bad_chars;
|
||||||
}
|
}
|
||||||
|
|
||||||
status = nick_is_utf8(nick);
|
status = nick_is_utf8(nick);
|
||||||
if (status != nick_ok)
|
if (status != nick_ok)
|
||||||
{
|
{
|
||||||
hub_free(nick);
|
hub_free(nick);
|
||||||
return status_msg_inf_error_nick_not_utf8;
|
return status_msg_inf_error_nick_not_utf8;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (user_is_connecting(user))
|
if (user_is_connecting(user))
|
||||||
{
|
{
|
||||||
memcpy(user->id.nick, nick, strlen(nick));
|
memcpy(user->id.nick, nick, strlen(nick));
|
||||||
user->id.nick[strlen(nick)] = 0;
|
user->id.nick[strlen(nick)] = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
hub_free(nick);
|
hub_free(nick);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
@@ -327,12 +327,12 @@ static int check_logged_in(struct hub_info* hub, struct hub_user* user, struct a
|
|||||||
{
|
{
|
||||||
struct hub_user* lookup1 = uman_get_user_by_nick(hub->users, user->id.nick);
|
struct hub_user* lookup1 = uman_get_user_by_nick(hub->users, user->id.nick);
|
||||||
struct hub_user* lookup2 = uman_get_user_by_cid(hub->users, user->id.cid);
|
struct hub_user* lookup2 = uman_get_user_by_cid(hub->users, user->id.cid);
|
||||||
|
|
||||||
if (lookup1 == user)
|
if (lookup1 == user)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (lookup1 || lookup2)
|
if (lookup1 || lookup2)
|
||||||
{
|
{
|
||||||
if (lookup1 == lookup2)
|
if (lookup1 == lookup2)
|
||||||
@@ -366,37 +366,21 @@ static int check_logged_in(struct hub_info* hub, struct hub_user* user, struct a
|
|||||||
*/
|
*/
|
||||||
static int check_user_agent(struct hub_info* hub, struct hub_user* user, struct adc_message* cmd)
|
static int check_user_agent(struct hub_info* hub, struct hub_user* user, struct adc_message* cmd)
|
||||||
{
|
{
|
||||||
char* ua_name_encoded = 0;
|
char* ua_encoded = 0;
|
||||||
char* ua_version_encoded = 0;
|
char* ua = 0;
|
||||||
char* str = 0;
|
|
||||||
size_t offset = 0;
|
|
||||||
|
|
||||||
/* Get client user agent version */
|
/* Get client user agent version */
|
||||||
ua_name_encoded = adc_msg_get_named_argument(cmd, ADC_INF_FLAG_USER_AGENT_PRODUCT);
|
ua_encoded = adc_msg_get_named_argument(cmd, ADC_INF_FLAG_USER_AGENT);
|
||||||
ua_version_encoded = adc_msg_get_named_argument(cmd, ADC_INF_FLAG_USER_AGENT_VERSION);
|
if (ua_encoded)
|
||||||
if (ua_name_encoded)
|
|
||||||
{
|
{
|
||||||
str = adc_msg_unescape(ua_name_encoded);
|
ua = adc_msg_unescape(ua_encoded);
|
||||||
if (str)
|
if (ua)
|
||||||
{
|
{
|
||||||
offset = strlen(str);
|
memcpy(user->id.user_agent, ua, MIN(strlen(ua), MAX_UA_LEN));
|
||||||
memcpy(user->id.user_agent, str, MIN(offset, MAX_UA_LEN));
|
hub_free(ua);
|
||||||
hub_free(str);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
hub_free(ua_encoded);
|
||||||
if (ua_version_encoded)
|
|
||||||
{
|
|
||||||
str = adc_msg_unescape(ua_version_encoded);
|
|
||||||
if (str)
|
|
||||||
{
|
|
||||||
memcpy(user->id.user_agent + offset, str, MIN(strlen(str), MAX_UA_LEN - offset));
|
|
||||||
hub_free(str);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
hub_free(ua_name_encoded);
|
|
||||||
hub_free(ua_version_encoded);
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -407,7 +391,7 @@ static int check_acl(struct hub_info* hub, struct hub_user* user, struct adc_mes
|
|||||||
{
|
{
|
||||||
return status_msg_ban_permanently;
|
return status_msg_ban_permanently;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (acl_is_user_banned(hub->acl, user->id.nick))
|
if (acl_is_user_banned(hub->acl, user->id.nick))
|
||||||
{
|
{
|
||||||
return status_msg_ban_permanently;
|
return status_msg_ban_permanently;
|
||||||
@@ -417,7 +401,7 @@ static int check_acl(struct hub_info* hub, struct hub_user* user, struct adc_mes
|
|||||||
{
|
{
|
||||||
return status_msg_inf_error_nick_restricted;
|
return status_msg_inf_error_nick_restricted;
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -486,7 +470,7 @@ static int check_limits(struct hub_info* hub, struct hub_user* user, struct adc_
|
|||||||
hub_free(arg);
|
hub_free(arg);
|
||||||
arg = 0;
|
arg = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
arg = adc_msg_get_named_argument(cmd, ADC_INF_FLAG_UPLOAD_SLOTS);
|
arg = adc_msg_get_named_argument(cmd, ADC_INF_FLAG_UPLOAD_SLOTS);
|
||||||
if (arg)
|
if (arg)
|
||||||
{
|
{
|
||||||
@@ -550,7 +534,7 @@ static int set_credentials(struct hub_info* hub, struct hub_user* user, struct a
|
|||||||
{
|
{
|
||||||
int ret = 0;
|
int ret = 0;
|
||||||
struct auth_info* info = acl_get_access_info(hub, user->id.nick);
|
struct auth_info* info = acl_get_access_info(hub, user->id.nick);
|
||||||
|
|
||||||
if (info)
|
if (info)
|
||||||
{
|
{
|
||||||
user->credentials = info->credentials;
|
user->credentials = info->credentials;
|
||||||
@@ -566,11 +550,11 @@ static int set_credentials(struct hub_info* hub, struct hub_user* user, struct a
|
|||||||
{
|
{
|
||||||
case auth_cred_none:
|
case auth_cred_none:
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case auth_cred_bot:
|
case auth_cred_bot:
|
||||||
adc_msg_add_argument(cmd, ADC_INF_FLAG_CLIENT_TYPE ADC_CLIENT_TYPE_BOT);
|
adc_msg_add_argument(cmd, ADC_INF_FLAG_CLIENT_TYPE ADC_CLIENT_TYPE_BOT);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case auth_cred_guest:
|
case auth_cred_guest:
|
||||||
/* Nothing to be added to the info message */
|
/* Nothing to be added to the info message */
|
||||||
break;
|
break;
|
||||||
@@ -582,7 +566,7 @@ static int set_credentials(struct hub_info* hub, struct hub_user* user, struct a
|
|||||||
case auth_cred_operator:
|
case auth_cred_operator:
|
||||||
adc_msg_add_argument(cmd, ADC_INF_FLAG_CLIENT_TYPE ADC_CLIENT_TYPE_OPERATOR);
|
adc_msg_add_argument(cmd, ADC_INF_FLAG_CLIENT_TYPE ADC_CLIENT_TYPE_OPERATOR);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case auth_cred_super:
|
case auth_cred_super:
|
||||||
adc_msg_add_argument(cmd, ADC_INF_FLAG_CLIENT_TYPE ADC_CLIENT_TYPE_SUPER_USER);
|
adc_msg_add_argument(cmd, ADC_INF_FLAG_CLIENT_TYPE ADC_CLIENT_TYPE_SUPER_USER);
|
||||||
break;
|
break;
|
||||||
@@ -594,7 +578,7 @@ static int set_credentials(struct hub_info* hub, struct hub_user* user, struct a
|
|||||||
case auth_cred_link:
|
case auth_cred_link:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -627,10 +611,10 @@ static int hub_handle_info_common(struct hub_user* user, struct adc_message* cmd
|
|||||||
{
|
{
|
||||||
/* Remove server restricted flags */
|
/* Remove server restricted flags */
|
||||||
remove_server_restricted_flags(cmd);
|
remove_server_restricted_flags(cmd);
|
||||||
|
|
||||||
/* Update/set the feature cast flags. */
|
/* Update/set the feature cast flags. */
|
||||||
set_feature_cast_supports(user, cmd);
|
set_feature_cast_supports(user, cmd);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -638,8 +622,7 @@ static int hub_handle_info_low_bandwidth(struct hub_info* hub, struct hub_user*
|
|||||||
{
|
{
|
||||||
if (hub->config->low_bandwidth_mode)
|
if (hub->config->low_bandwidth_mode)
|
||||||
{
|
{
|
||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_USER_AGENT_VERSION);
|
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_USER_AGENT);
|
||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_USER_AGENT_PRODUCT);
|
|
||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_SHARED_FILES);
|
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_SHARED_FILES);
|
||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_COUNT_HUB_NORMAL);
|
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_COUNT_HUB_NORMAL);
|
||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_COUNT_HUB_REGISTER);
|
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_COUNT_HUB_REGISTER);
|
||||||
@@ -652,7 +635,7 @@ static int hub_handle_info_low_bandwidth(struct hub_info* hub, struct hub_user*
|
|||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_DESCRIPTION);
|
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_DESCRIPTION);
|
||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_EMAIL);
|
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_EMAIL);
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -697,20 +680,20 @@ int hub_handle_info_login(struct hub_info* hub, struct hub_user* user, struct ad
|
|||||||
{
|
{
|
||||||
return status_msg_hub_full;
|
return status_msg_hub_full;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (check_registered_users_only(hub, user))
|
if (check_registered_users_only(hub, user))
|
||||||
{
|
{
|
||||||
return status_msg_hub_registered_users_only;
|
return status_msg_hub_registered_users_only;
|
||||||
}
|
}
|
||||||
|
|
||||||
INF_CHECK(check_limits, hub, user, cmd);
|
INF_CHECK(check_limits, hub, user, cmd);
|
||||||
|
|
||||||
/* strip off stuff if low_bandwidth_mode is enabled */
|
/* strip off stuff if low_bandwidth_mode is enabled */
|
||||||
hub_handle_info_low_bandwidth(hub, user, cmd);
|
hub_handle_info_low_bandwidth(hub, user, cmd);
|
||||||
|
|
||||||
/* Set initial user info */
|
/* Set initial user info */
|
||||||
user_set_info(user, cmd);
|
user_set_info(user, cmd);
|
||||||
|
|
||||||
return code;
|
return code;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -72,7 +72,7 @@ size_t ioq_recv_set(struct ioq_recv* q, void* buf, size_t bufsize)
|
|||||||
q->buf = 0;
|
q->buf = 0;
|
||||||
q->size = 0;
|
q->size = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!bufsize)
|
if (!bufsize)
|
||||||
{
|
{
|
||||||
return 0;
|
return 0;
|
||||||
@@ -130,7 +130,7 @@ void ioq_send_add(struct ioq_send* q, struct adc_message* msg_)
|
|||||||
q->size += msg->length;
|
q->size += msg->length;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ioq_send_remove(struct ioq_send* q, struct adc_message* msg)
|
void ioq_send_remove(struct ioq_send* q, struct adc_message* msg)
|
||||||
{
|
{
|
||||||
#ifdef DEBUG_SENDQ
|
#ifdef DEBUG_SENDQ
|
||||||
debug_msg("ioq_send_remove", msg);
|
debug_msg("ioq_send_remove", msg);
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -87,7 +87,7 @@ void setup_signal_handlers(struct hub_info* hub)
|
|||||||
act.sa_mask = sig_set;
|
act.sa_mask = sig_set;
|
||||||
act.sa_flags = SA_ONSTACK | SA_RESTART;
|
act.sa_flags = SA_ONSTACK | SA_RESTART;
|
||||||
act.sa_handler = hub_handle_signal;
|
act.sa_handler = hub_handle_signal;
|
||||||
|
|
||||||
for (i = 0; signals[i]; i++)
|
for (i = 0; signals[i]; i++)
|
||||||
{
|
{
|
||||||
if (sigaction(signals[i], &act, 0) != 0)
|
if (sigaction(signals[i], &act, 0) != 0)
|
||||||
@@ -124,7 +124,7 @@ int main_loop()
|
|||||||
hub_log_initialize(arg_log, arg_log_syslog);
|
hub_log_initialize(arg_log, arg_log_syslog);
|
||||||
hub_set_log_verbosity(arg_verbose);
|
hub_set_log_verbosity(arg_verbose);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (read_config(arg_config, &configuration, !arg_have_config) == -1)
|
if (read_config(arg_config, &configuration, !arg_have_config) == -1)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
@@ -150,7 +150,7 @@ int main_loop()
|
|||||||
#if !defined(WIN32)
|
#if !defined(WIN32)
|
||||||
setup_signal_handlers(hub);
|
setup_signal_handlers(hub);
|
||||||
#ifdef SYSTEMD
|
#ifdef SYSTEMD
|
||||||
/* Notify the service manager that this daemon has
|
/* Notify the service manager that this daemon has
|
||||||
* been successfully initalized and shall enter the
|
* been successfully initalized and shall enter the
|
||||||
* main loop.
|
* main loop.
|
||||||
*/
|
*/
|
||||||
@@ -174,7 +174,7 @@ int main_loop()
|
|||||||
#if !defined(WIN32)
|
#if !defined(WIN32)
|
||||||
shutdown_signal_handlers(hub);
|
shutdown_signal_handlers(hub);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (hub)
|
if (hub)
|
||||||
{
|
{
|
||||||
hub_shutdown_service(hub);
|
hub_shutdown_service(hub);
|
||||||
@@ -288,7 +288,7 @@ void parse_command_line(int argc, char** argv)
|
|||||||
arg_dump_config = 1;
|
arg_dump_config = 1;
|
||||||
arg_check_config = 1;
|
arg_check_config = 1;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'S':
|
case 'S':
|
||||||
arg_dump_config = 2;
|
arg_dump_config = 2;
|
||||||
arg_check_config = 1;
|
arg_check_config = 1;
|
||||||
@@ -297,7 +297,7 @@ void parse_command_line(int argc, char** argv)
|
|||||||
case 'l':
|
case 'l':
|
||||||
arg_log = optarg;
|
arg_log = optarg;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'L':
|
case 'L':
|
||||||
arg_log_syslog = 1;
|
arg_log_syslog = 1;
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -40,11 +40,14 @@ static int plugin_command_dispatch(struct command_base* cbase, struct hub_user*
|
|||||||
|
|
||||||
LOG_PLUGIN("plugin_command_dispatch: cmd=%s", cmd->prefix);
|
LOG_PLUGIN("plugin_command_dispatch: cmd=%s", cmd->prefix);
|
||||||
|
|
||||||
LIST_FOREACH(struct plugin_command_handle*, cmdh, data->commands,
|
cmdh = (struct plugin_command_handle*) list_get_first(data->commands);
|
||||||
|
while (cmdh)
|
||||||
{
|
{
|
||||||
if (strcmp(cmdh->prefix, cmd->prefix) == 0)
|
if (strcmp(cmdh->prefix, cmd->prefix) == 0)
|
||||||
return cmdh->handler(plugin, puser, pcommand);
|
return cmdh->handler(plugin, puser, pcommand);
|
||||||
});
|
|
||||||
|
cmdh = (struct plugin_command_handle*) list_get_next(data->commands);
|
||||||
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -65,17 +68,6 @@ static int cbfunc_send_message(struct plugin_handle* plugin, struct plugin_user*
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int cbfunc_send_broadcast(struct plugin_handle* plugin, const char* message)
|
|
||||||
{
|
|
||||||
char* buffer = adc_msg_escape(message);
|
|
||||||
struct adc_message* command = adc_msg_construct(ADC_CMD_IMSG, strlen(buffer) + 6);
|
|
||||||
adc_msg_add_argument(command, buffer);
|
|
||||||
route_to_all(plugin_get_hub(plugin), command);
|
|
||||||
adc_msg_free(command);
|
|
||||||
hub_free(buffer);
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int cbfunc_send_status(struct plugin_handle* plugin, struct plugin_user* user, int code, const char* message)
|
static int cbfunc_send_status(struct plugin_handle* plugin, struct plugin_user* user, int code, const char* message)
|
||||||
{
|
{
|
||||||
char code_str[4];
|
char code_str[4];
|
||||||
@@ -112,6 +104,7 @@ static int cbfunc_command_add(struct plugin_handle* plugin, struct plugin_comman
|
|||||||
cmdh->internal_handle = command;
|
cmdh->internal_handle = command;
|
||||||
list_append(data->commands, cmdh);
|
list_append(data->commands, cmdh);
|
||||||
command_add(plugin_get_hub(plugin)->commands, command, (void*) plugin);
|
command_add(plugin_get_hub(plugin)->commands, command, (void*) plugin);
|
||||||
|
LOG_DEBUG("*** Add plugin command: %s (%p, %p)\n", command->prefix, command, cmdh);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -120,6 +113,7 @@ static int cbfunc_command_del(struct plugin_handle* plugin, struct plugin_comman
|
|||||||
struct plugin_callback_data* data = get_callback_data(plugin);
|
struct plugin_callback_data* data = get_callback_data(plugin);
|
||||||
struct command_handle* command = (struct command_handle*) cmdh->internal_handle;
|
struct command_handle* command = (struct command_handle*) cmdh->internal_handle;
|
||||||
|
|
||||||
|
LOG_DEBUG("*** Del plugin command: %s (%p, %p)\n", command->prefix, command, cmdh);
|
||||||
list_remove(data->commands, cmdh);
|
list_remove(data->commands, cmdh);
|
||||||
command_del(plugin_get_hub(plugin)->commands, command);
|
command_del(plugin_get_hub(plugin)->commands, command);
|
||||||
hub_free(command);
|
hub_free(command);
|
||||||
@@ -139,12 +133,6 @@ struct plugin_command_arg_data* cbfunc_command_arg_next(struct plugin_handle* pl
|
|||||||
return (struct plugin_command_arg_data*) hub_command_arg_next((struct hub_command*) cmd, (enum hub_command_arg_type) t);
|
return (struct plugin_command_arg_data*) hub_command_arg_next((struct hub_command*) cmd, (enum hub_command_arg_type) t);
|
||||||
}
|
}
|
||||||
|
|
||||||
static size_t cbfunc_get_usercount(struct plugin_handle* plugin)
|
|
||||||
{
|
|
||||||
struct hub_info* hub = plugin_get_hub(plugin);
|
|
||||||
return hub->users->count;
|
|
||||||
}
|
|
||||||
|
|
||||||
static char* cbfunc_get_hub_name(struct plugin_handle* plugin)
|
static char* cbfunc_get_hub_name(struct plugin_handle* plugin)
|
||||||
{
|
{
|
||||||
struct hub_info* hub = plugin_get_hub(plugin);
|
struct hub_info* hub = plugin_get_hub(plugin);
|
||||||
@@ -170,7 +158,7 @@ static void cbfunc_set_hub_name(struct plugin_handle* plugin, const char* str)
|
|||||||
char* new_str = adc_msg_escape(str ? str : hub->config->hub_name);
|
char* new_str = adc_msg_escape(str ? str : hub->config->hub_name);
|
||||||
|
|
||||||
adc_msg_replace_named_argument(hub->command_info, ADC_INF_FLAG_NICK, new_str);
|
adc_msg_replace_named_argument(hub->command_info, ADC_INF_FLAG_NICK, new_str);
|
||||||
|
|
||||||
// Broadcast hub name
|
// Broadcast hub name
|
||||||
command = adc_msg_construct(ADC_CMD_IINF, (strlen(new_str) + 8));
|
command = adc_msg_construct(ADC_CMD_IINF, (strlen(new_str) + 8));
|
||||||
adc_msg_add_named_argument(command, ADC_INF_FLAG_NICK, new_str);
|
adc_msg_add_named_argument(command, ADC_INF_FLAG_NICK, new_str);
|
||||||
@@ -187,7 +175,7 @@ static void cbfunc_set_hub_description(struct plugin_handle* plugin, const char*
|
|||||||
char* new_str = adc_msg_escape(str ? str : hub->config->hub_description);
|
char* new_str = adc_msg_escape(str ? str : hub->config->hub_description);
|
||||||
|
|
||||||
adc_msg_replace_named_argument(hub->command_info, ADC_INF_FLAG_DESCRIPTION, new_str);
|
adc_msg_replace_named_argument(hub->command_info, ADC_INF_FLAG_DESCRIPTION, new_str);
|
||||||
|
|
||||||
// Broadcast hub description
|
// Broadcast hub description
|
||||||
command = adc_msg_construct(ADC_CMD_IINF, (strlen(new_str) + 8));
|
command = adc_msg_construct(ADC_CMD_IINF, (strlen(new_str) + 8));
|
||||||
adc_msg_add_named_argument(command, ADC_INF_FLAG_DESCRIPTION, new_str);
|
adc_msg_add_named_argument(command, ADC_INF_FLAG_DESCRIPTION, new_str);
|
||||||
@@ -200,14 +188,12 @@ static void cbfunc_set_hub_description(struct plugin_handle* plugin, const char*
|
|||||||
void plugin_register_callback_functions(struct plugin_handle* handle)
|
void plugin_register_callback_functions(struct plugin_handle* handle)
|
||||||
{
|
{
|
||||||
handle->hub.send_message = cbfunc_send_message;
|
handle->hub.send_message = cbfunc_send_message;
|
||||||
handle->hub.send_broadcast_message = cbfunc_send_broadcast;
|
|
||||||
handle->hub.send_status_message = cbfunc_send_status;
|
handle->hub.send_status_message = cbfunc_send_status;
|
||||||
handle->hub.user_disconnect = cbfunc_user_disconnect;
|
handle->hub.user_disconnect = cbfunc_user_disconnect;
|
||||||
handle->hub.command_add = cbfunc_command_add;
|
handle->hub.command_add = cbfunc_command_add;
|
||||||
handle->hub.command_del = cbfunc_command_del;
|
handle->hub.command_del = cbfunc_command_del;
|
||||||
handle->hub.command_arg_reset = cbfunc_command_arg_reset;
|
handle->hub.command_arg_reset = cbfunc_command_arg_reset;
|
||||||
handle->hub.command_arg_next = cbfunc_command_arg_next;
|
handle->hub.command_arg_next = cbfunc_command_arg_next;
|
||||||
handle->hub.get_usercount = cbfunc_get_usercount;
|
|
||||||
handle->hub.get_name = cbfunc_get_hub_name;
|
handle->hub.get_name = cbfunc_get_hub_name;
|
||||||
handle->hub.set_name = cbfunc_set_hub_name;
|
handle->hub.set_name = cbfunc_set_hub_name;
|
||||||
handle->hub.get_description = cbfunc_get_hub_description;
|
handle->hub.get_description = cbfunc_get_hub_description;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -27,12 +27,13 @@
|
|||||||
PLUGIN_DEBUG(HUB, # FUNCNAME) \
|
PLUGIN_DEBUG(HUB, # FUNCNAME) \
|
||||||
if (HUB->plugins && HUB->plugins->loaded) \
|
if (HUB->plugins && HUB->plugins->loaded) \
|
||||||
{ \
|
{ \
|
||||||
struct plugin_handle* plugin;\
|
struct plugin_handle* plugin = (struct plugin_handle*) list_get_first(HUB->plugins->loaded); \
|
||||||
LIST_FOREACH(struct plugin_handle*, plugin, HUB->plugins->loaded, \
|
while (plugin) \
|
||||||
{ \
|
{ \
|
||||||
if (plugin->funcs.FUNCNAME) \
|
if (plugin->funcs.FUNCNAME) \
|
||||||
CODE \
|
CODE \
|
||||||
}); \
|
plugin = (struct plugin_handle*) list_get_next(HUB->plugins->loaded); \
|
||||||
|
} \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define PLUGIN_INVOKE_STATUS_1(HUB, FUNCNAME, ARG1) \
|
#define PLUGIN_INVOKE_STATUS_1(HUB, FUNCNAME, ARG1) \
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -229,16 +229,16 @@ int plugin_initialize(struct hub_config* config, struct hub_info* hub)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void plugin_unload_ptr(void* ptr)
|
|
||||||
{
|
|
||||||
struct plugin_handle* plugin = (struct plugin_handle*) ptr;
|
|
||||||
plugin_unload(plugin);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void plugin_shutdown(struct uhub_plugins* handle)
|
void plugin_shutdown(struct uhub_plugins* handle)
|
||||||
{
|
{
|
||||||
list_clear(handle->loaded, plugin_unload_ptr);
|
struct plugin_handle* plugin = (struct plugin_handle*) list_get_first(handle->loaded);
|
||||||
|
while (plugin)
|
||||||
|
{
|
||||||
|
list_remove(handle->loaded, plugin);
|
||||||
|
plugin_unload(plugin);
|
||||||
|
plugin = (struct plugin_handle*) list_get_first(handle->loaded);
|
||||||
|
}
|
||||||
|
|
||||||
list_destroy(handle->loaded);
|
list_destroy(handle->loaded);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -71,18 +71,9 @@ static void probe_net_event(struct net_connection* con, int events, void *arg)
|
|||||||
probe_destroy(probe);
|
probe_destroy(probe);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
else if ((memcmp(probe_recvbuf, "GET ", 4) == 0) ||
|
|
||||||
(memcmp(probe_recvbuf, "POST", 4) == 0) ||
|
|
||||||
(memcmp(probe_recvbuf, "HEAD", 4) == 0))
|
|
||||||
{
|
|
||||||
/* Looks like HTTP - Not supported, but we log it. */
|
|
||||||
LOG_TRACE("Probed HTTP connection. Not supported closing connection (%s)", ip_convert_to_string(&probe->addr));
|
|
||||||
const char* buf = "501 Not implemented\r\n\r\n";
|
|
||||||
net_con_send(con, buf, strlen(buf));
|
|
||||||
}
|
|
||||||
#ifdef SSL_SUPPORT
|
#ifdef SSL_SUPPORT
|
||||||
else if (bytes >= 11 &&
|
else if (bytes >= 11 &&
|
||||||
probe_recvbuf[0] == 22 &&
|
probe_recvbuf[0] == 22 &&
|
||||||
probe_recvbuf[1] == 3 && /* protocol major version */
|
probe_recvbuf[1] == 3 && /* protocol major version */
|
||||||
probe_recvbuf[5] == 1 && /* message type */
|
probe_recvbuf[5] == 1 && /* message type */
|
||||||
probe_recvbuf[9] == probe_recvbuf[1])
|
probe_recvbuf[9] == probe_recvbuf[1])
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -28,7 +28,7 @@ int route_message(struct hub_info* hub, struct hub_user* u, struct adc_message*
|
|||||||
case 'B': /* Broadcast to all logged in clients */
|
case 'B': /* Broadcast to all logged in clients */
|
||||||
route_to_all(hub, msg);
|
route_to_all(hub, msg);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'D':
|
case 'D':
|
||||||
target = uman_get_user_by_sid(hub->users, msg->target);
|
target = uman_get_user_by_sid(hub->users, msg->target);
|
||||||
if (target)
|
if (target)
|
||||||
@@ -36,7 +36,7 @@ int route_message(struct hub_info* hub, struct hub_user* u, struct adc_message*
|
|||||||
route_to_user(hub, target, msg);
|
route_to_user(hub, target, msg);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'E':
|
case 'E':
|
||||||
target = uman_get_user_by_sid(hub->users, msg->target);
|
target = uman_get_user_by_sid(hub->users, msg->target);
|
||||||
if (target)
|
if (target)
|
||||||
@@ -45,11 +45,11 @@ int route_message(struct hub_info* hub, struct hub_user* u, struct adc_message*
|
|||||||
route_to_user(hub, u, msg);
|
route_to_user(hub, u, msg);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case 'F':
|
case 'F':
|
||||||
route_to_subscribers(hub, msg);
|
route_to_subscribers(hub, msg);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
/* Ignore the message */
|
/* Ignore the message */
|
||||||
break;
|
break;
|
||||||
@@ -104,7 +104,19 @@ int route_to_user(struct hub_info* hub, struct hub_user* user, struct adc_messag
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
if (!user->connection)
|
if (!user->connection)
|
||||||
return 0;
|
{
|
||||||
|
switch (user->type)
|
||||||
|
{
|
||||||
|
case user_type_client:
|
||||||
|
return 0; // No connection - we're about to drop this user.
|
||||||
|
case user_type_bot:
|
||||||
|
{
|
||||||
|
bot_recv_msg handler = (bot_recv_msg) user->ptr;
|
||||||
|
handler(user, msg);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
uhub_assert(msg->cache && *msg->cache);
|
uhub_assert(msg->cache && *msg->cache);
|
||||||
|
|
||||||
@@ -139,53 +151,76 @@ int route_flush_pipeline(struct hub_info* hub, struct hub_user* u)
|
|||||||
|
|
||||||
int route_to_all(struct hub_info* hub, struct adc_message* command) /* iterate users */
|
int route_to_all(struct hub_info* hub, struct adc_message* command) /* iterate users */
|
||||||
{
|
{
|
||||||
struct hub_user* user;
|
struct hub_user* user = (struct hub_user*) list_get_first(hub->users->list);
|
||||||
LIST_FOREACH(struct hub_user*, user, hub->users->list,
|
while (user)
|
||||||
{
|
{
|
||||||
route_to_user(hub, user, command);
|
route_to_user(hub, user, command);
|
||||||
});
|
user = (struct hub_user*) list_get_next(hub->users->list);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int route_to_operators(struct hub_info* hub, struct adc_message* command) /* iterate users */
|
||||||
|
{
|
||||||
|
struct hub_user* user = (struct hub_user*) list_get_first(hub->users->list);
|
||||||
|
while (user)
|
||||||
|
{
|
||||||
|
if (user_flag_get(user, flag_opnotify))
|
||||||
|
route_to_user(hub, user, command);
|
||||||
|
user = (struct hub_user*) list_get_next(hub->users->list);
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int route_to_subscribers(struct hub_info* hub, struct adc_message* command) /* iterate users */
|
int route_to_subscribers(struct hub_info* hub, struct adc_message* command) /* iterate users */
|
||||||
{
|
{
|
||||||
int do_send;
|
int do_send;
|
||||||
char* tmp;
|
char* tmp;
|
||||||
|
|
||||||
struct hub_user* user;
|
struct hub_user* user = (struct hub_user*) list_get_first(hub->users->list);
|
||||||
LIST_FOREACH(struct hub_user*, user, hub->users->list,
|
while (user)
|
||||||
{
|
{
|
||||||
if (user->feature_cast)
|
if (user->feature_cast)
|
||||||
{
|
{
|
||||||
do_send = 1;
|
do_send = 1;
|
||||||
|
|
||||||
LIST_FOREACH(char*, tmp, command->feature_cast_include,
|
tmp = list_get_first(command->feature_cast_include);
|
||||||
|
while (tmp)
|
||||||
{
|
{
|
||||||
if (!user_have_feature_cast_support(user, tmp))
|
if (!user_have_feature_cast_support(user, tmp))
|
||||||
{
|
{
|
||||||
do_send = 0;
|
do_send = 0;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
tmp = list_get_next(command->feature_cast_include);;
|
||||||
|
}
|
||||||
if (!do_send)
|
|
||||||
|
if (!do_send) {
|
||||||
|
user = (struct hub_user*) list_get_next(hub->users->list);
|
||||||
continue;
|
continue;
|
||||||
|
}
|
||||||
LIST_FOREACH(char*, tmp, command->feature_cast_exclude,
|
|
||||||
|
tmp = list_get_first(command->feature_cast_exclude);
|
||||||
|
while (tmp)
|
||||||
{
|
{
|
||||||
if (user_have_feature_cast_support(user, tmp))
|
if (user_have_feature_cast_support(user, tmp))
|
||||||
{
|
{
|
||||||
do_send = 0;
|
do_send = 0;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
tmp = list_get_next(command->feature_cast_exclude);
|
||||||
|
}
|
||||||
|
|
||||||
if (do_send)
|
if (do_send)
|
||||||
|
{
|
||||||
route_to_user(hub, user, command);
|
route_to_user(hub, user, command);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
user = (struct hub_user*) list_get_next(hub->users->list);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -200,17 +235,20 @@ int route_info_message(struct hub_info* hub, struct hub_user* u)
|
|||||||
struct adc_message* cmd = adc_msg_copy(u->info);
|
struct adc_message* cmd = adc_msg_copy(u->info);
|
||||||
const char* address = user_get_address(u);
|
const char* address = user_get_address(u);
|
||||||
struct hub_user* user = 0;
|
struct hub_user* user = 0;
|
||||||
|
|
||||||
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_IPV4_ADDR);
|
adc_msg_remove_named_argument(cmd, ADC_INF_FLAG_IPV4_ADDR);
|
||||||
adc_msg_add_named_argument(cmd, ADC_INF_FLAG_IPV4_ADDR, address);
|
adc_msg_add_named_argument(cmd, ADC_INF_FLAG_IPV4_ADDR, address);
|
||||||
|
|
||||||
LIST_FOREACH(struct hub_user*, user, hub->users->list,
|
user = (struct hub_user*) list_get_first(hub->users->list);
|
||||||
|
while (user)
|
||||||
{
|
{
|
||||||
if (user_is_nat_override(user))
|
if (user_is_nat_override(user))
|
||||||
route_to_user(hub, user, cmd);
|
route_to_user(hub, user, cmd);
|
||||||
else
|
else
|
||||||
route_to_user(hub, user, u->info);
|
route_to_user(hub, user, u->info);
|
||||||
});
|
|
||||||
|
user = (struct hub_user*) list_get_next(hub->users->list);
|
||||||
|
}
|
||||||
adc_msg_free(cmd);
|
adc_msg_free(cmd);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -40,6 +40,14 @@ extern int route_to_user(struct hub_info* hub, struct hub_user*, struct adc_mess
|
|||||||
*/
|
*/
|
||||||
extern int route_to_all(struct hub_info* hub, struct adc_message* command);
|
extern int route_to_all(struct hub_info* hub, struct adc_message* command);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Send a message to all operators.
|
||||||
|
* Technically it sends to all users with the flag_opnotify set
|
||||||
|
* which is intended to be set for only operators - allowing operators to opt-out
|
||||||
|
* of these messages.
|
||||||
|
*/
|
||||||
|
extern int route_to_operators(struct hub_info* hub, struct adc_message* command);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Broadcast message to all users subscribing to the type of message.
|
* Broadcast message to all users subscribing to the type of message.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -38,11 +38,10 @@ static const char* user_log_str(struct hub_user* user)
|
|||||||
struct hub_user* user_create(struct hub_info* hub, struct net_connection* con, struct ip_addr_encap* addr)
|
struct hub_user* user_create(struct hub_info* hub, struct net_connection* con, struct ip_addr_encap* addr)
|
||||||
{
|
{
|
||||||
struct hub_user* user = NULL;
|
struct hub_user* user = NULL;
|
||||||
|
|
||||||
LOG_TRACE("user_create(), hub=%p, con[sd=%d]", hub, net_con_get_sd(con));
|
LOG_TRACE("user_create(), hub=%p, con[sd=%d]", hub, net_con_get_sd(con));
|
||||||
|
|
||||||
user = (struct hub_user*) hub_malloc_zero(sizeof(struct hub_user));
|
user = (struct hub_user*) hub_malloc_zero(sizeof(struct hub_user));
|
||||||
|
|
||||||
if (user == NULL)
|
if (user == NULL)
|
||||||
return NULL; /* OOM */
|
return NULL; /* OOM */
|
||||||
|
|
||||||
@@ -54,6 +53,7 @@ struct hub_user* user_create(struct hub_info* hub, struct net_connection* con, s
|
|||||||
|
|
||||||
memcpy(&user->id.addr, addr, sizeof(struct ip_addr_encap));
|
memcpy(&user->id.addr, addr, sizeof(struct ip_addr_encap));
|
||||||
user_set_state(user, state_protocol);
|
user_set_state(user, state_protocol);
|
||||||
|
user->type = user_type_client;
|
||||||
|
|
||||||
flood_control_reset(&user->flood_chat);
|
flood_control_reset(&user->flood_chat);
|
||||||
flood_control_reset(&user->flood_connect);
|
flood_control_reset(&user->flood_connect);
|
||||||
@@ -65,6 +65,54 @@ struct hub_user* user_create(struct hub_info* hub, struct net_connection* con, s
|
|||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// generate a semi-stable CID: base32_encode(tiger({hub name} + {nick}))
|
||||||
|
static const char* generate_bot_cid(struct hub_info* hub, const char* nick)
|
||||||
|
{
|
||||||
|
static char result[MAX_CID_LEN+1];
|
||||||
|
char buf[(MAX_NICK_LEN*2)+1];
|
||||||
|
uint64_t tiger_res[3];
|
||||||
|
|
||||||
|
memset(buf, 0, sizeof(buf));
|
||||||
|
snprintf(buf, sizeof(buf), "%s%s", hub->config->hub_name, nick);
|
||||||
|
tiger((uint64_t*) buf, sizeof(buf), (uint64_t*) tiger_res);
|
||||||
|
base32_encode((unsigned char*) tiger_res, TIGERSIZE, result);
|
||||||
|
result[MAX_CID_LEN] = 0;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct hub_user* user_create_bot(struct hub_info* hub, const char* nick, const char* description, bot_recv_msg msg_handler)
|
||||||
|
{
|
||||||
|
struct hub_user* user = NULL;
|
||||||
|
LOG_TRACE("user_create_bot(), hub=%p, nick=\"%s\"", hub, nick);
|
||||||
|
|
||||||
|
user = (struct hub_user*) hub_malloc_zero(sizeof(struct hub_user));
|
||||||
|
if (user == NULL)
|
||||||
|
return NULL; /* OOM */
|
||||||
|
|
||||||
|
strcpy(user->id.nick, nick);
|
||||||
|
uman_get_free_sid(hub->users, user);
|
||||||
|
|
||||||
|
user_set_state(user, state_normal);
|
||||||
|
user->type = user_type_bot;
|
||||||
|
user->credentials = auth_cred_bot;
|
||||||
|
|
||||||
|
// The message handler
|
||||||
|
user->ptr = (void*) msg_handler;
|
||||||
|
|
||||||
|
user->info = adc_msg_construct(ADC_CMD_BINF, 15);
|
||||||
|
if (user->info)
|
||||||
|
{
|
||||||
|
adc_msg_add_argument(user->info, sid_to_string(user->id.sid));
|
||||||
|
adc_msg_add_named_argument(user->info, ADC_INF_FLAG_CLIENT_TYPE, ADC_CLIENT_TYPE_BOT);
|
||||||
|
adc_msg_add_named_argument_string(user->info, ADC_INF_FLAG_USER_AGENT, PRODUCT_STRING);
|
||||||
|
adc_msg_add_named_argument_string(user->info, ADC_INF_FLAG_NICK, nick);
|
||||||
|
adc_msg_add_named_argument_string(user->info, ADC_INF_FLAG_DESCRIPTION, description);
|
||||||
|
adc_msg_add_named_argument(user->info, ADC_INF_FLAG_CLIENT_ID, generate_bot_cid(hub, nick));
|
||||||
|
}
|
||||||
|
|
||||||
|
user->hub = hub;
|
||||||
|
return user;
|
||||||
|
}
|
||||||
|
|
||||||
void user_destroy(struct hub_user* user)
|
void user_destroy(struct hub_user* user)
|
||||||
{
|
{
|
||||||
@@ -90,7 +138,7 @@ void user_set_state(struct hub_user* user, enum user_state state)
|
|||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
user->state = state;
|
user->state = state;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -135,7 +183,7 @@ void user_update_info(struct hub_user* u, struct adc_message* cmd)
|
|||||||
prefix[1] = argument[1];
|
prefix[1] = argument[1];
|
||||||
adc_msg_replace_named_argument(cmd_new, prefix, argument+2);
|
adc_msg_replace_named_argument(cmd_new, prefix, argument+2);
|
||||||
}
|
}
|
||||||
|
|
||||||
hub_free(argument);
|
hub_free(argument);
|
||||||
argument = adc_msg_get_argument(cmd, n++);
|
argument = adc_msg_get_argument(cmd, n++);
|
||||||
}
|
}
|
||||||
@@ -232,12 +280,15 @@ void user_support_remove(struct hub_user* user, int fourcc)
|
|||||||
|
|
||||||
int user_have_feature_cast_support(struct hub_user* user, char feature[4])
|
int user_have_feature_cast_support(struct hub_user* user, char feature[4])
|
||||||
{
|
{
|
||||||
char* tmp;
|
char* tmp = list_get_first(user->feature_cast);
|
||||||
LIST_FOREACH(char*, tmp, user->feature_cast,
|
while (tmp)
|
||||||
{
|
{
|
||||||
if (strncmp(tmp, feature, 4) == 0)
|
if (strncmp(tmp, feature, 4) == 0)
|
||||||
return 1;
|
return 1;
|
||||||
});
|
|
||||||
|
tmp = list_get_next(user->feature_cast);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -47,11 +47,12 @@ enum user_flags
|
|||||||
feature_link = 0x00000100, /** LINK: Hub link (not supported) */
|
feature_link = 0x00000100, /** LINK: Hub link (not supported) */
|
||||||
feature_adcs = 0x00000200, /** ADCS: ADC over TLS/SSL */
|
feature_adcs = 0x00000200, /** ADCS: ADC over TLS/SSL */
|
||||||
feature_bas0 = 0x00000400, /** BAS0: Obsolete pre-ADC/1.0 protocol version */
|
feature_bas0 = 0x00000400, /** BAS0: Obsolete pre-ADC/1.0 protocol version */
|
||||||
|
flag_opnotify = 0x00200000, /** User should receive operation notifications. NOTE: Only operators should have this flag! */
|
||||||
flag_flood = 0x00400000, /** User has been notified about flooding. */
|
flag_flood = 0x00400000, /** User has been notified about flooding. */
|
||||||
flag_muted = 0x00800000, /** User is muted (cannot chat) */
|
flag_muted = 0x00800000, /** User is muted (cannot chat) */
|
||||||
flag_ignore = 0x01000000, /** Ignore further reads */
|
flag_ignore = 0x01000000, /** Ignore further reads */
|
||||||
flag_maxbuf = 0x02000000, /** Hit max buf read, ignore msg */
|
flag_maxbuf = 0x02000000, /** Hit max buf read, ignore msg */
|
||||||
flag_choke = 0x04000000, /** Choked: Cannot send, waiting for write event */
|
flag_choke = 0x04000000, /** Choked: Cannot send, waiting for write event */
|
||||||
flag_want_read = 0x08000000, /** Need to read (SSL) */
|
flag_want_read = 0x08000000, /** Need to read (SSL) */
|
||||||
flag_want_write = 0x10000000, /** Need to write (SSL) */
|
flag_want_write = 0x10000000, /** Need to write (SSL) */
|
||||||
flag_user_list = 0x20000000, /** Send queue bypass (when receiving the send queue) */
|
flag_user_list = 0x20000000, /** Send queue bypass (when receiving the send queue) */
|
||||||
@@ -104,21 +105,30 @@ struct hub_user_limits
|
|||||||
size_t hub_count_total; /** The number of hubs connected to in total */
|
size_t hub_count_total; /** The number of hubs connected to in total */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
enum user_type
|
||||||
|
{
|
||||||
|
user_type_client, /** A user connected normally as an ADC client */
|
||||||
|
user_type_bot, /** Not really a user, but a bot inside the hub */
|
||||||
|
};
|
||||||
|
|
||||||
|
typedef void (*bot_recv_msg)(struct hub_user*, struct adc_message* msg);
|
||||||
|
|
||||||
struct hub_user
|
struct hub_user
|
||||||
{
|
{
|
||||||
struct hub_user_info id; /** Contains nick name and CID */
|
struct hub_user_info id; /** Contains nick name and CID */
|
||||||
enum auth_credentials credentials; /** see enum user_credentials */
|
enum auth_credentials credentials; /** see enum user_credentials */
|
||||||
enum user_state state; /** see enum user_state */
|
enum user_state state; /** see enum user_state */
|
||||||
|
enum user_type type;
|
||||||
uint32_t flags; /** see enum user_flags */
|
uint32_t flags; /** see enum user_flags */
|
||||||
struct linked_list* feature_cast; /** Features supported by feature cast */
|
struct linked_list* feature_cast; /** Features supported by feature cast */
|
||||||
struct adc_message* info; /** ADC 'INF' message (broadcasted to everyone joining the hub) */
|
struct adc_message* info; /** ADC 'INF' message (broadcasted to everyone joining the hub) */
|
||||||
struct hub_info* hub; /** The hub instance this user belong to */
|
struct hub_info* hub; /** The hub instance this user belong to */
|
||||||
|
void* ptr;
|
||||||
struct ioq_recv* recv_queue;
|
struct ioq_recv* recv_queue;
|
||||||
struct ioq_send* send_queue;
|
struct ioq_send* send_queue;
|
||||||
struct net_connection* connection; /** Connection data */
|
struct net_connection* connection; /** Connection data */
|
||||||
struct hub_user_limits limits; /** Data used for limitation */
|
struct hub_user_limits limits; /** Data used for limitation */
|
||||||
enum user_quit_reason quit_reason; /** Quit reason (see user_quit_reason) */
|
enum user_quit_reason quit_reason; /** Quit reason (see user_quit_reason) */
|
||||||
|
|
||||||
struct flood_control flood_chat;
|
struct flood_control flood_chat;
|
||||||
struct flood_control flood_connect;
|
struct flood_control flood_connect;
|
||||||
struct flood_control flood_search;
|
struct flood_control flood_search;
|
||||||
@@ -138,6 +148,7 @@ struct hub_user
|
|||||||
* @return User object or NULL if not enough memory is available.
|
* @return User object or NULL if not enough memory is available.
|
||||||
*/
|
*/
|
||||||
extern struct hub_user* user_create(struct hub_info* hub, struct net_connection* con, struct ip_addr_encap* addr);
|
extern struct hub_user* user_create(struct hub_info* hub, struct net_connection* con, struct ip_addr_encap* addr);
|
||||||
|
extern struct hub_user* user_create_bot(struct hub_info* hub, const char* nick, const char* description, bot_recv_msg msg_handler);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete a user.
|
* Delete a user.
|
||||||
@@ -150,7 +161,7 @@ extern void user_destroy(struct hub_user* user);
|
|||||||
* This associates a INF message to the user.
|
* This associates a INF message to the user.
|
||||||
* If the user already has a INF message associated, then this is
|
* If the user already has a INF message associated, then this is
|
||||||
* released before setting the new one.
|
* released before setting the new one.
|
||||||
*
|
*
|
||||||
* @param info new inf message (can be NULL)
|
* @param info new inf message (can be NULL)
|
||||||
*/
|
*/
|
||||||
extern void user_set_info(struct hub_user* user, struct adc_message* info);
|
extern void user_set_info(struct hub_user* user, struct adc_message* info);
|
||||||
@@ -244,7 +255,7 @@ extern int user_flag_get(struct hub_user* user, enum user_flags flag);
|
|||||||
* Check if a user supports 'feature' for feature casting (basis for 'Fxxx' messages)
|
* Check if a user supports 'feature' for feature casting (basis for 'Fxxx' messages)
|
||||||
* The feature cast is specified as the 'SU' argument to the user's
|
* The feature cast is specified as the 'SU' argument to the user's
|
||||||
* INF-message.
|
* INF-message.
|
||||||
*
|
*
|
||||||
* @param feature a feature to lookup (example: 'TCP4' or 'UDP4')
|
* @param feature a feature to lookup (example: 'TCP4' or 'UDP4')
|
||||||
* @return 1 if 'feature' supported, or 0 otherwise
|
* @return 1 if 'feature' supported, or 0 otherwise
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -28,7 +28,7 @@ static void clear_user_list_callback(void* ptr)
|
|||||||
if (ptr)
|
if (ptr)
|
||||||
{
|
{
|
||||||
struct hub_user* u = (struct hub_user*) ptr;
|
struct hub_user* u = (struct hub_user*) ptr;
|
||||||
|
|
||||||
/* Mark the user as already being disconnected.
|
/* Mark the user as already being disconnected.
|
||||||
* This prevents the hub from trying to send
|
* This prevents the hub from trying to send
|
||||||
* quit messages to other users.
|
* quit messages to other users.
|
||||||
@@ -38,11 +38,6 @@ static void clear_user_list_callback(void* ptr)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static int uman_map_compare(const void* a, const void* b)
|
|
||||||
{
|
|
||||||
return strcmp((const char*) a, (const char*) b);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
struct hub_user_manager* uman_init()
|
struct hub_user_manager* uman_init()
|
||||||
{
|
{
|
||||||
@@ -51,10 +46,15 @@ struct hub_user_manager* uman_init()
|
|||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
users->list = list_create();
|
users->list = list_create();
|
||||||
users->nickmap = rb_tree_create(uman_map_compare, NULL, NULL);
|
|
||||||
users->cidmap = rb_tree_create(uman_map_compare, NULL, NULL);
|
|
||||||
users->sids = sid_pool_create(net_get_max_sockets());
|
users->sids = sid_pool_create(net_get_max_sockets());
|
||||||
|
|
||||||
|
if (!users->list)
|
||||||
|
{
|
||||||
|
list_destroy(users->list);
|
||||||
|
hub_free(users);
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
|
||||||
return users;
|
return users;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -64,18 +64,11 @@ int uman_shutdown(struct hub_user_manager* users)
|
|||||||
if (!users)
|
if (!users)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
if (users->nickmap)
|
|
||||||
rb_tree_destroy(users->nickmap);
|
|
||||||
|
|
||||||
if (users->cidmap)
|
|
||||||
rb_tree_destroy(users->cidmap);
|
|
||||||
|
|
||||||
if (users->list)
|
if (users->list)
|
||||||
{
|
{
|
||||||
list_clear(users->list, &clear_user_list_callback);
|
list_clear(users->list, &clear_user_list_callback);
|
||||||
list_destroy(users->list);
|
list_destroy(users->list);
|
||||||
}
|
}
|
||||||
|
|
||||||
sid_pool_destroy(users->sids);
|
sid_pool_destroy(users->sids);
|
||||||
|
|
||||||
hub_free(users);
|
hub_free(users);
|
||||||
@@ -88,9 +81,6 @@ int uman_add(struct hub_user_manager* users, struct hub_user* user)
|
|||||||
if (!users || !user)
|
if (!users || !user)
|
||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
rb_tree_insert(users->nickmap, user->id.nick, user);
|
|
||||||
rb_tree_insert(users->cidmap, user->id.cid, user);
|
|
||||||
|
|
||||||
list_append(users->list, user);
|
list_append(users->list, user);
|
||||||
users->count++;
|
users->count++;
|
||||||
users->count_peak = MAX(users->count, users->count_peak);
|
users->count_peak = MAX(users->count, users->count_peak);
|
||||||
@@ -106,8 +96,6 @@ int uman_remove(struct hub_user_manager* users, struct hub_user* user)
|
|||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
list_remove(users->list, user);
|
list_remove(users->list, user);
|
||||||
rb_tree_remove(users->nickmap, user->id.nick);
|
|
||||||
rb_tree_remove(users->cidmap, user->id.cid);
|
|
||||||
|
|
||||||
if (users->count > 0)
|
if (users->count > 0)
|
||||||
{
|
{
|
||||||
@@ -132,29 +120,42 @@ struct hub_user* uman_get_user_by_sid(struct hub_user_manager* users, sid_t sid)
|
|||||||
|
|
||||||
struct hub_user* uman_get_user_by_cid(struct hub_user_manager* users, const char* cid)
|
struct hub_user* uman_get_user_by_cid(struct hub_user_manager* users, const char* cid)
|
||||||
{
|
{
|
||||||
struct hub_user* user = (struct hub_user*) rb_tree_get(users->cidmap, (const void*) cid);
|
struct hub_user* user = (struct hub_user*) list_get_first(users->list); /* iterate users - only on incoming INF msg */
|
||||||
return user;
|
while (user)
|
||||||
|
{
|
||||||
|
if (strcmp(user->id.cid, cid) == 0)
|
||||||
|
return user;
|
||||||
|
user = (struct hub_user*) list_get_next(users->list);
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
struct hub_user* uman_get_user_by_nick(struct hub_user_manager* users, const char* nick)
|
struct hub_user* uman_get_user_by_nick(struct hub_user_manager* users, const char* nick)
|
||||||
{
|
{
|
||||||
struct hub_user* user = (struct hub_user*) rb_tree_get(users->nickmap, nick);
|
struct hub_user* user = (struct hub_user*) list_get_first(users->list); /* iterate users - only on incoming INF msg */
|
||||||
return user;
|
while (user)
|
||||||
|
{
|
||||||
|
if (strcmp(user->id.nick, nick) == 0)
|
||||||
|
return user;
|
||||||
|
user = (struct hub_user*) list_get_next(users->list);
|
||||||
|
}
|
||||||
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t uman_get_user_by_addr(struct hub_user_manager* users, struct linked_list* target, struct ip_range* range)
|
size_t uman_get_user_by_addr(struct hub_user_manager* users, struct linked_list* target, struct ip_range* range)
|
||||||
{
|
{
|
||||||
size_t num = 0;
|
size_t num = 0;
|
||||||
struct hub_user* user;
|
struct hub_user* user = (struct hub_user*) list_get_first(users->list); /* iterate users - only on incoming INF msg */
|
||||||
LIST_FOREACH(struct hub_user*, user, users->list,
|
while (user)
|
||||||
{
|
{
|
||||||
if (ip_in_range(&user->id.addr, range))
|
if (ip_in_range(&user->id.addr, range))
|
||||||
{
|
{
|
||||||
list_append(target, user);
|
list_append(target, user);
|
||||||
num++;
|
num++;
|
||||||
}
|
}
|
||||||
});
|
user = (struct hub_user*) list_get_next(users->list);
|
||||||
|
}
|
||||||
return num;
|
return num;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -163,8 +164,8 @@ int uman_send_user_list(struct hub_info* hub, struct hub_user_manager* users, st
|
|||||||
int ret = 1;
|
int ret = 1;
|
||||||
struct hub_user* user;
|
struct hub_user* user;
|
||||||
user_flag_set(target, flag_user_list);
|
user_flag_set(target, flag_user_list);
|
||||||
|
user = (struct hub_user*) list_get_first(users->list); /* iterate users - only on INF or PAS msg */
|
||||||
LIST_FOREACH(struct hub_user*, user, users->list,
|
while (user)
|
||||||
{
|
{
|
||||||
if (user_is_logged_in(user))
|
if (user_is_logged_in(user))
|
||||||
{
|
{
|
||||||
@@ -172,7 +173,8 @@ int uman_send_user_list(struct hub_info* hub, struct hub_user_manager* users, st
|
|||||||
if (!ret)
|
if (!ret)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
user = (struct hub_user*) list_get_next(users->list);
|
||||||
|
}
|
||||||
|
|
||||||
#if 0
|
#if 0
|
||||||
FIXME: FIXME FIXME handle send queue excess
|
FIXME: FIXME FIXME handle send queue excess
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -24,12 +24,10 @@ struct hub_user_manager
|
|||||||
{
|
{
|
||||||
size_t count; /**<< "Number of all fully connected and logged in users" */
|
size_t count; /**<< "Number of all fully connected and logged in users" */
|
||||||
size_t count_peak; /**<< "Peak number of users" */
|
size_t count_peak; /**<< "Peak number of users" */
|
||||||
|
struct sid_pool* sids;
|
||||||
uint64_t shared_size; /**<< "The total number of shared bytes among fully connected users." */
|
uint64_t shared_size; /**<< "The total number of shared bytes among fully connected users." */
|
||||||
uint64_t shared_files; /**<< "The total number of shared files among fully connected users." */
|
uint64_t shared_files; /**<< "The total number of shared files among fully connected users." */
|
||||||
struct sid_pool* sids; /**<< "Maps SIDs to users (constant time)" */
|
|
||||||
struct linked_list* list; /**<< "Contains all logged in users" */
|
struct linked_list* list; /**<< "Contains all logged in users" */
|
||||||
struct rb_tree* nickmap; /**<< "Maps nicknames to users (red black tree)" */
|
|
||||||
struct rb_tree* cidmap; /**<< "Maps CIDs to users (red black tree)" */
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -79,7 +77,7 @@ extern sid_t uman_get_free_sid(struct hub_user_manager* users, struct hub_user*
|
|||||||
*
|
*
|
||||||
* NOTE: This function will only search connected users, which means
|
* NOTE: This function will only search connected users, which means
|
||||||
* that SIDs assigned to users who are not yet completely logged in,
|
* that SIDs assigned to users who are not yet completely logged in,
|
||||||
* or are in the process of being disconnected will result in this
|
* or are in the process of being disconnected will result in this
|
||||||
* function returning NULL even though the sid is not freely available.
|
* function returning NULL even though the sid is not freely available.
|
||||||
*
|
*
|
||||||
* FIXME: Is that really safe / sensible ?
|
* FIXME: Is that really safe / sensible ?
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -102,7 +102,14 @@ void net_backend_shutdown()
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void net_backend_update(struct net_connection* con, int events)
|
void net_con_reinitialize(struct net_connection* con, net_connection_cb callback, const void* ptr, int events)
|
||||||
|
{
|
||||||
|
con->callback = callback;
|
||||||
|
con->ptr = (void*) ptr;
|
||||||
|
net_con_update(con, events);
|
||||||
|
}
|
||||||
|
|
||||||
|
void net_con_update(struct net_connection* con, int events)
|
||||||
{
|
{
|
||||||
g_backend->handler.con_mod(g_backend->data, con, events);
|
g_backend->handler.con_mod(g_backend->data, con, events);
|
||||||
}
|
}
|
||||||
@@ -141,7 +148,7 @@ int net_backend_process()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Process pending DNS results
|
// Process pending DNS results
|
||||||
// net_dns_process();
|
net_dns_process();
|
||||||
|
|
||||||
g_backend->handler.backend_process(g_backend->data, res);
|
g_backend->handler.backend_process(g_backend->data, res);
|
||||||
|
|
||||||
@@ -159,8 +166,8 @@ void net_con_initialize(struct net_connection* con, int sd, net_connection_cb ca
|
|||||||
{
|
{
|
||||||
g_backend->handler.con_init(g_backend->data, con, sd, callback, ptr);
|
g_backend->handler.con_init(g_backend->data, con, sd, callback, ptr);
|
||||||
|
|
||||||
net_set_nonblocking(net_con_get_sd(con), 1);
|
net_set_nonblocking(con->sd, 1);
|
||||||
net_set_nosigpipe(net_con_get_sd(con), 1);
|
net_set_nosigpipe(con->sd, 1);
|
||||||
|
|
||||||
g_backend->handler.con_add(g_backend->data, con, events);
|
g_backend->handler.con_add(g_backend->data, con, events);
|
||||||
g_backend->common.num++;
|
g_backend->common.num++;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -75,14 +75,6 @@ extern void net_backend_shutdown();
|
|||||||
*/
|
*/
|
||||||
extern int net_backend_process();
|
extern int net_backend_process();
|
||||||
|
|
||||||
/**
|
|
||||||
* Update the event mask.
|
|
||||||
*
|
|
||||||
* @param con Connection handle.
|
|
||||||
* @param events Event mask (NET_EVENT_*)
|
|
||||||
*/
|
|
||||||
extern void net_backend_update(struct net_connection* con, int events);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the current time.
|
* Get the current time.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2013, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -19,7 +19,6 @@
|
|||||||
|
|
||||||
#include "uhub.h"
|
#include "uhub.h"
|
||||||
#include "network/common.h"
|
#include "network/common.h"
|
||||||
#include "network/backend.h"
|
|
||||||
|
|
||||||
static int is_blocked_or_interrupted()
|
static int is_blocked_or_interrupted()
|
||||||
{
|
{
|
||||||
@@ -117,27 +116,10 @@ void* net_con_get_ptr(struct net_connection* con)
|
|||||||
return con->ptr;
|
return con->ptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
void net_con_update(struct net_connection* con, int events)
|
|
||||||
{
|
|
||||||
#ifdef SSL_SUPPORT
|
|
||||||
if (con->ssl)
|
|
||||||
net_ssl_update(con, events);
|
|
||||||
else
|
|
||||||
#endif
|
|
||||||
net_backend_update(con, events);
|
|
||||||
}
|
|
||||||
|
|
||||||
void net_con_reinitialize(struct net_connection* con, net_connection_cb callback, const void* ptr, int events)
|
|
||||||
{
|
|
||||||
con->callback = callback;
|
|
||||||
con->ptr = (void*) ptr;
|
|
||||||
net_con_update(con, events);
|
|
||||||
}
|
|
||||||
|
|
||||||
void net_con_destroy(struct net_connection* con)
|
void net_con_destroy(struct net_connection* con)
|
||||||
{
|
{
|
||||||
#ifdef SSL_SUPPORT
|
#ifdef SSL_SUPPORT
|
||||||
if (con && con->ssl)
|
if (con->ssl)
|
||||||
net_ssl_destroy(con);
|
net_ssl_destroy(con);
|
||||||
#endif
|
#endif
|
||||||
hub_free(con);
|
hub_free(con);
|
||||||
@@ -196,12 +178,11 @@ static int net_connect_job_check(struct net_connect_job* job)
|
|||||||
int af = job->addr.ss_family;
|
int af = job->addr.ss_family;
|
||||||
enum net_connect_status status;
|
enum net_connect_status status;
|
||||||
|
|
||||||
int ret = net_connect(net_con_get_sd(con), (struct sockaddr*) &job->addr, af == AF_INET ? sizeof(struct sockaddr_in) : sizeof(struct sockaddr_in6));
|
int ret = net_connect(net_con_get_sd(con), (struct sockaddr*) &job->addr, af == AF_INET ? sizeof(struct sockaddr_in6) : sizeof(struct sockaddr_in));
|
||||||
if (ret == 0 || (ret == -1 && net_error() == EISCONN))
|
if (ret == 0 || (ret == -1 && net_error() == EISCONN))
|
||||||
{
|
{
|
||||||
LOG_TRACE("net_connect_job_check(): Socket connected!");
|
LOG_TRACE("net_connect_job_check(): Socket connected!");
|
||||||
job->con = NULL;
|
job->con = NULL;
|
||||||
net_con_clear_timeout(con);
|
|
||||||
net_connect_callback(job->handle, net_connect_status_ok, con);
|
net_connect_callback(job->handle, net_connect_status_ok, con);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@@ -283,6 +264,7 @@ static int net_connect_job_process(struct net_connect_job* job)
|
|||||||
*/
|
*/
|
||||||
static void net_connect_job_internal_cb(struct net_connection* con, int event, void* ptr)
|
static void net_connect_job_internal_cb(struct net_connection* con, int event, void* ptr)
|
||||||
{
|
{
|
||||||
|
int ret;
|
||||||
struct net_connect_job* job = net_con_get_ptr(con);
|
struct net_connect_job* job = net_con_get_ptr(con);
|
||||||
struct net_connect_job* next_job = job->next;
|
struct net_connect_job* next_job = job->next;
|
||||||
struct net_connect_handle* handle = job->handle;
|
struct net_connect_handle* handle = job->handle;
|
||||||
@@ -325,7 +307,7 @@ static void net_connect_job_internal_cb(struct net_connection* con, int event, v
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void net_connect_cancel(struct net_connect_handle* handle)
|
static int net_connect_cancel(struct net_connect_handle* handle)
|
||||||
{
|
{
|
||||||
struct net_connect_job* job;
|
struct net_connect_job* job;
|
||||||
|
|
||||||
@@ -381,7 +363,6 @@ static int net_connect_process(struct net_connect_handle* handle)
|
|||||||
return 1; // Connected - cool!
|
return 1; // Connected - cool!
|
||||||
|
|
||||||
net_connect_process_queue(handle, handle->job4);
|
net_connect_process_queue(handle, handle->job4);
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@@ -440,6 +421,7 @@ static int net_con_connect_dns_callback(struct net_dns_job* job, const struct ne
|
|||||||
struct net_connect_handle* handle = (struct net_connect_handle*) net_dns_job_get_ptr(job);
|
struct net_connect_handle* handle = (struct net_connect_handle*) net_dns_job_get_ptr(job);
|
||||||
handle->dns = NULL;
|
handle->dns = NULL;
|
||||||
size_t usable = 0;
|
size_t usable = 0;
|
||||||
|
int ret;
|
||||||
|
|
||||||
LOG_TRACE("net_con_connect(): async - Got DNS results");
|
LOG_TRACE("net_con_connect(): async - Got DNS results");
|
||||||
if (!result)
|
if (!result)
|
||||||
@@ -524,34 +506,3 @@ static void net_connect_callback(struct net_connect_handle* handle, enum net_con
|
|||||||
// Cleanup
|
// Cleanup
|
||||||
net_connect_destroy(handle);
|
net_connect_destroy(handle);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void timeout_callback(struct timeout_evt* evt)
|
|
||||||
{
|
|
||||||
net_con_callback((struct net_connection*) evt->ptr, NET_EVENT_TIMEOUT);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void net_con_set_timeout(struct net_connection* con, int seconds)
|
|
||||||
{
|
|
||||||
if (!con->timeout)
|
|
||||||
{
|
|
||||||
con->timeout = hub_malloc_zero(sizeof(struct timeout_evt));
|
|
||||||
timeout_evt_initialize(con->timeout, timeout_callback, con);
|
|
||||||
timeout_queue_insert(net_backend_get_timeout_queue(), con->timeout, seconds);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
timeout_queue_reschedule(net_backend_get_timeout_queue(), con->timeout, seconds);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void net_con_clear_timeout(struct net_connection* con)
|
|
||||||
{
|
|
||||||
if (con->timeout && timeout_evt_is_scheduled(con->timeout))
|
|
||||||
{
|
|
||||||
timeout_queue_remove(net_backend_get_timeout_queue(), con->timeout);
|
|
||||||
hub_free(con->timeout);
|
|
||||||
con->timeout = 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2013, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -28,7 +28,6 @@
|
|||||||
#define NET_EVENT_TIMEOUT 0x0001
|
#define NET_EVENT_TIMEOUT 0x0001
|
||||||
#define NET_EVENT_READ 0x0002
|
#define NET_EVENT_READ 0x0002
|
||||||
#define NET_EVENT_WRITE 0x0004
|
#define NET_EVENT_WRITE 0x0004
|
||||||
#define NET_EVENT_ERROR 0x1000
|
|
||||||
|
|
||||||
struct net_connection
|
struct net_connection
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2013, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -68,11 +68,6 @@ static void shutdown_free_results(void* ptr)
|
|||||||
net_dns_result_free(result);
|
net_dns_result_free(result);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void notify_callback(struct uhub_notify_handle* handle, void* ptr)
|
|
||||||
{
|
|
||||||
net_dns_process();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// NOTE: Any job manipulating the members of this
|
// NOTE: Any job manipulating the members of this
|
||||||
// struct must lock the mutex!
|
// struct must lock the mutex!
|
||||||
@@ -81,8 +76,6 @@ struct net_dns_subsystem
|
|||||||
struct linked_list* jobs; // currently running jobs
|
struct linked_list* jobs; // currently running jobs
|
||||||
struct linked_list* results; // queue of results that are awaiting being delivered to callback.
|
struct linked_list* results; // queue of results that are awaiting being delivered to callback.
|
||||||
uhub_mutex_t mutex;
|
uhub_mutex_t mutex;
|
||||||
|
|
||||||
struct uhub_notify_handle* notify_handle; // used to signal back to the event loop that there is something to process.
|
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct net_dns_subsystem* g_dns = NULL;
|
static struct net_dns_subsystem* g_dns = NULL;
|
||||||
@@ -94,11 +87,13 @@ void net_dns_initialize()
|
|||||||
g_dns->jobs = list_create();
|
g_dns->jobs = list_create();
|
||||||
g_dns->results = list_create();
|
g_dns->results = list_create();
|
||||||
uhub_mutex_init(&g_dns->mutex);
|
uhub_mutex_init(&g_dns->mutex);
|
||||||
g_dns->notify_handle = net_notify_create(notify_callback, g_dns);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void net_dns_destroy()
|
void net_dns_destroy()
|
||||||
{
|
{
|
||||||
|
struct net_dns_job* job;
|
||||||
|
struct net_dns_result* result;
|
||||||
|
|
||||||
uhub_mutex_lock(&g_dns->mutex);
|
uhub_mutex_lock(&g_dns->mutex);
|
||||||
LOG_TRACE("net_dns_destroy(): jobs=%d", (int) list_size(g_dns->jobs));
|
LOG_TRACE("net_dns_destroy(): jobs=%d", (int) list_size(g_dns->jobs));
|
||||||
list_clear(g_dns->jobs, &shutdown_free_jobs);
|
list_clear(g_dns->jobs, &shutdown_free_jobs);
|
||||||
@@ -110,18 +105,22 @@ void net_dns_destroy()
|
|||||||
list_destroy(g_dns->jobs);
|
list_destroy(g_dns->jobs);
|
||||||
list_destroy(g_dns->results);
|
list_destroy(g_dns->results);
|
||||||
uhub_mutex_destroy(&g_dns->mutex);
|
uhub_mutex_destroy(&g_dns->mutex);
|
||||||
net_notify_destroy(g_dns->notify_handle);
|
|
||||||
hub_free(g_dns);
|
hub_free(g_dns);
|
||||||
g_dns = NULL;
|
g_dns = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void dummy_free(void* ptr)
|
||||||
|
{
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void net_dns_process()
|
void net_dns_process()
|
||||||
{
|
{
|
||||||
struct net_dns_result* result;
|
struct net_dns_result* result;
|
||||||
uhub_mutex_lock(&g_dns->mutex);
|
uhub_mutex_lock(&g_dns->mutex);
|
||||||
LOG_TRACE("net_dns_process(): jobs=%d, results=%d", (int) list_size(g_dns->jobs), (int) list_size(g_dns->results));
|
LOG_DUMP("net_dns_process(): jobs=%d, results=%d", (int) list_size(g_dns->jobs), (int) list_size(g_dns->results));
|
||||||
|
|
||||||
LIST_FOREACH(struct net_dns_result*, result, g_dns->results,
|
for (result = (struct net_dns_result*) list_get_first(g_dns->results); result; result = (struct net_dns_result*) list_get_next(g_dns->results))
|
||||||
{
|
{
|
||||||
struct net_dns_job* job = result->job;
|
struct net_dns_job* job = result->job;
|
||||||
#ifdef DEBUG_LOOKUP_TIME
|
#ifdef DEBUG_LOOKUP_TIME
|
||||||
@@ -147,9 +146,9 @@ void net_dns_process()
|
|||||||
result->job = NULL;
|
result->job = NULL;
|
||||||
free_job(job);
|
free_job(job);
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
|
||||||
list_clear(g_dns->results, NULL);
|
list_clear(g_dns->results, &dummy_free);
|
||||||
uhub_mutex_unlock(&g_dns->mutex);
|
uhub_mutex_unlock(&g_dns->mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -216,7 +215,6 @@ static void* job_thread_resolve_name(void* ptr)
|
|||||||
uhub_mutex_lock(&g_dns->mutex);
|
uhub_mutex_lock(&g_dns->mutex);
|
||||||
list_remove(g_dns->jobs, job);
|
list_remove(g_dns->jobs, job);
|
||||||
list_append(g_dns->results, dns_results);
|
list_append(g_dns->results, dns_results);
|
||||||
net_notify_signal(g_dns->notify_handle, 1);
|
|
||||||
uhub_mutex_unlock(&g_dns->mutex);
|
uhub_mutex_unlock(&g_dns->mutex);
|
||||||
|
|
||||||
return dns_results;
|
return dns_results;
|
||||||
@@ -275,14 +273,14 @@ extern struct net_dns_job* net_dns_gethostbyaddr(struct ip_addr_encap* ipaddr, n
|
|||||||
static struct net_dns_job* find_and_remove_job(struct net_dns_job* job)
|
static struct net_dns_job* find_and_remove_job(struct net_dns_job* job)
|
||||||
{
|
{
|
||||||
struct net_dns_job* it;
|
struct net_dns_job* it;
|
||||||
LIST_FOREACH(struct net_dns_job*, it, g_dns->jobs,
|
for (it = (struct net_dns_job*) list_get_first(g_dns->jobs); it; it = (struct net_dns_job*) list_get_next(g_dns->jobs))
|
||||||
{
|
{
|
||||||
if (it == job)
|
if (it == job)
|
||||||
{
|
{
|
||||||
list_remove(g_dns->jobs, it);
|
list_remove(g_dns->jobs, it);
|
||||||
return job;
|
return job;
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -290,14 +288,14 @@ static struct net_dns_job* find_and_remove_job(struct net_dns_job* job)
|
|||||||
static struct net_dns_result* find_and_remove_result(struct net_dns_job* job)
|
static struct net_dns_result* find_and_remove_result(struct net_dns_job* job)
|
||||||
{
|
{
|
||||||
struct net_dns_result* it;
|
struct net_dns_result* it;
|
||||||
LIST_FOREACH(struct net_dns_result*, it, g_dns->results,
|
for (it = (struct net_dns_result*) list_get_first(g_dns->results); it; it = (struct net_dns_result*) list_get_next(g_dns->results))
|
||||||
{
|
{
|
||||||
if (it->job == job)
|
if (it->job == job)
|
||||||
{
|
{
|
||||||
list_remove(g_dns->results, it);
|
list_remove(g_dns->results, it);
|
||||||
return it;
|
return it;
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2013, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -25,10 +25,10 @@ int ip_is_valid_ipv4(const char* address)
|
|||||||
int o = 0; /* octet number */
|
int o = 0; /* octet number */
|
||||||
int n = 0; /* numbers after each dot */
|
int n = 0; /* numbers after each dot */
|
||||||
int d = 0; /* dots */
|
int d = 0; /* dots */
|
||||||
|
|
||||||
if (!address || strlen(address) > 15 || strlen(address) < 7)
|
if (!address || strlen(address) > 15 || strlen(address) < 7)
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
for (; i < strlen(address); i++)
|
for (; i < strlen(address); i++)
|
||||||
{
|
{
|
||||||
if (is_num(address[i]))
|
if (is_num(address[i]))
|
||||||
@@ -49,9 +49,9 @@ int ip_is_valid_ipv4(const char* address)
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (n == 0 || n > 3 || o > 255 || d != 3) return 0;
|
if (n == 0 || n > 3 || o > 255 || d != 3) return 0;
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -107,9 +107,9 @@ int ip_convert_address(const char* text_address, int port, struct sockaddr* addr
|
|||||||
struct sockaddr_in addr4;
|
struct sockaddr_in addr4;
|
||||||
size_t sockaddr_size;
|
size_t sockaddr_size;
|
||||||
const char* taddr = 0;
|
const char* taddr = 0;
|
||||||
|
|
||||||
int ipv6sup = net_is_ipv6_supported();
|
int ipv6sup = net_is_ipv6_supported();
|
||||||
|
|
||||||
if (strcmp(text_address, "any") == 0)
|
if (strcmp(text_address, "any") == 0)
|
||||||
{
|
{
|
||||||
if (ipv6sup)
|
if (ipv6sup)
|
||||||
@@ -136,8 +136,8 @@ int ip_convert_address(const char* text_address, int port, struct sockaddr* addr
|
|||||||
{
|
{
|
||||||
taddr = text_address;
|
taddr = text_address;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (ip_is_valid_ipv6(taddr) && ipv6sup)
|
if (ip_is_valid_ipv6(taddr) && ipv6sup)
|
||||||
{
|
{
|
||||||
sockaddr_size = sizeof(struct sockaddr_in6);
|
sockaddr_size = sizeof(struct sockaddr_in6);
|
||||||
@@ -152,7 +152,7 @@ int ip_convert_address(const char* text_address, int port, struct sockaddr* addr
|
|||||||
|
|
||||||
memcpy(addr, &addr6, sockaddr_size);
|
memcpy(addr, &addr6, sockaddr_size);
|
||||||
*addr_len = sockaddr_size;
|
*addr_len = sockaddr_size;
|
||||||
|
|
||||||
}
|
}
|
||||||
else if (ip_is_valid_ipv4(taddr))
|
else if (ip_is_valid_ipv4(taddr))
|
||||||
{
|
{
|
||||||
@@ -185,28 +185,28 @@ int ip_mask_create_left(int af, int bits, struct ip_addr_encap* result)
|
|||||||
|
|
||||||
memset(result, 0, sizeof(struct ip_addr_encap));
|
memset(result, 0, sizeof(struct ip_addr_encap));
|
||||||
result->af = af;
|
result->af = af;
|
||||||
|
|
||||||
if (bits < 0) bits = 0;
|
if (bits < 0) bits = 0;
|
||||||
|
|
||||||
if (af == AF_INET)
|
if (af == AF_INET)
|
||||||
{
|
{
|
||||||
if (bits > 32) bits = 32;
|
if (bits > 32) bits = 32;
|
||||||
mask = (0xffffffff << (32 - bits));
|
mask = (0xffffffff << (32 - bits));
|
||||||
if (bits == 0) mask = 0;
|
if (bits == 0) mask = 0;
|
||||||
|
|
||||||
result->internal_ip_data.in.s_addr = (((uint8_t*) &mask)[0] << 24) | (((uint8_t*) &mask)[1] << 16) | (((uint8_t*) &mask)[2] << 8) | (((uint8_t*) &mask)[3] << 0);
|
result->internal_ip_data.in.s_addr = (((uint8_t*) &mask)[0] << 24) | (((uint8_t*) &mask)[1] << 16) | (((uint8_t*) &mask)[2] << 8) | (((uint8_t*) &mask)[3] << 0);
|
||||||
}
|
}
|
||||||
else if (af == AF_INET6)
|
else if (af == AF_INET6)
|
||||||
{
|
{
|
||||||
if (bits > 128) bits = 128;
|
if (bits > 128) bits = 128;
|
||||||
|
|
||||||
fill = (128-bits) / 8;
|
fill = (128-bits) / 8;
|
||||||
remain_bits = (128-bits) % 8;
|
remain_bits = (128-bits) % 8;
|
||||||
mask = (0xff << (8 - remain_bits));
|
mask = (0xff << (8 - remain_bits));
|
||||||
|
|
||||||
for (n = 0; n < fill; n++)
|
for (n = 0; n < fill; n++)
|
||||||
((uint8_t*) &result->internal_ip_data.in6)[n] = (uint8_t) 0xff;
|
((uint8_t*) &result->internal_ip_data.in6)[n] = (uint8_t) 0xff;
|
||||||
|
|
||||||
if (fill < 16)
|
if (fill < 16)
|
||||||
((uint8_t*) &result->internal_ip_data.in6)[fill] = (uint8_t) mask;
|
((uint8_t*) &result->internal_ip_data.in6)[fill] = (uint8_t) mask;
|
||||||
}
|
}
|
||||||
@@ -233,32 +233,32 @@ int ip_mask_create_right(int af, int bits, struct ip_addr_encap* result)
|
|||||||
|
|
||||||
memset(result, 0, sizeof(struct ip_addr_encap));
|
memset(result, 0, sizeof(struct ip_addr_encap));
|
||||||
result->af = af;
|
result->af = af;
|
||||||
|
|
||||||
if (bits < 0) bits = 0;
|
if (bits < 0) bits = 0;
|
||||||
|
|
||||||
if (af == AF_INET)
|
if (af == AF_INET)
|
||||||
{
|
{
|
||||||
if (bits > 32) bits = 32;
|
if (bits > 32) bits = 32;
|
||||||
mask = (0xffffffff >> (32-bits));
|
mask = (0xffffffff >> (32-bits));
|
||||||
if (bits == 0) mask = 0;
|
if (bits == 0) mask = 0;
|
||||||
result->internal_ip_data.in.s_addr = (((uint8_t*) &mask)[0] << 24) | (((uint8_t*) &mask)[1] << 16) | (((uint8_t*) &mask)[2] << 8) | (((uint8_t*) &mask)[3] << 0);
|
result->internal_ip_data.in.s_addr = (((uint8_t*) &mask)[0] << 24) | (((uint8_t*) &mask)[1] << 16) | (((uint8_t*) &mask)[2] << 8) | (((uint8_t*) &mask)[3] << 0);
|
||||||
|
|
||||||
}
|
}
|
||||||
else if (af == AF_INET6)
|
else if (af == AF_INET6)
|
||||||
{
|
{
|
||||||
if (bits > 128) bits = 128;
|
if (bits > 128) bits = 128;
|
||||||
|
|
||||||
fill = (128-bits) / 8;
|
fill = (128-bits) / 8;
|
||||||
remain_bits = (128-bits) % 8;
|
remain_bits = (128-bits) % 8;
|
||||||
mask8 = (0xff >> (8 - remain_bits));
|
mask8 = (0xff >> (8 - remain_bits));
|
||||||
start = 16-fill;
|
start = 16-fill;
|
||||||
|
|
||||||
for (n = 0; n < start; n++)
|
for (n = 0; n < start; n++)
|
||||||
((uint8_t*) &result->internal_ip_data.in6)[n] = (uint8_t) 0x00;
|
((uint8_t*) &result->internal_ip_data.in6)[n] = (uint8_t) 0x00;
|
||||||
|
|
||||||
for (n = start; n < 16; n++)
|
for (n = start; n < 16; n++)
|
||||||
((uint8_t*) &result->internal_ip_data.in6)[n] = (uint8_t) 0xff;
|
((uint8_t*) &result->internal_ip_data.in6)[n] = (uint8_t) 0xff;
|
||||||
|
|
||||||
if (start > 0)
|
if (start > 0)
|
||||||
((uint8_t*) &result->internal_ip_data.in6)[start-1] = (uint8_t) mask8;
|
((uint8_t*) &result->internal_ip_data.in6)[start-1] = (uint8_t) mask8;
|
||||||
}
|
}
|
||||||
@@ -266,13 +266,13 @@ int ip_mask_create_right(int af, int bits, struct ip_addr_encap* result)
|
|||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef IP_CALC_DEBUG
|
#ifdef IP_CALC_DEBUG
|
||||||
char* r_str = hub_strdup(ip_convert_to_string(result));
|
char* r_str = hub_strdup(ip_convert_to_string(result));
|
||||||
LOG_DUMP("Created right mask: %s", r_str);
|
LOG_DUMP("Created right mask: %s", r_str);
|
||||||
hub_free(r_str);
|
hub_free(r_str);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -281,7 +281,7 @@ void ip_mask_apply_AND(struct ip_addr_encap* addr, struct ip_addr_encap* mask, s
|
|||||||
{
|
{
|
||||||
memset(result, 0, sizeof(struct ip_addr_encap));
|
memset(result, 0, sizeof(struct ip_addr_encap));
|
||||||
result->af = addr->af;
|
result->af = addr->af;
|
||||||
|
|
||||||
if (addr->af == AF_INET)
|
if (addr->af == AF_INET)
|
||||||
{
|
{
|
||||||
result->internal_ip_data.in.s_addr = addr->internal_ip_data.in.s_addr & mask->internal_ip_data.in.s_addr;
|
result->internal_ip_data.in.s_addr = addr->internal_ip_data.in.s_addr & mask->internal_ip_data.in.s_addr;
|
||||||
@@ -294,19 +294,19 @@ void ip_mask_apply_AND(struct ip_addr_encap* addr, struct ip_addr_encap* mask, s
|
|||||||
for (n = 0; n < 4; n++)
|
for (n = 0; n < 4; n++)
|
||||||
{
|
{
|
||||||
offset = n * 4;
|
offset = n * 4;
|
||||||
|
|
||||||
A = (((uint8_t*) &addr->internal_ip_data.in6)[offset+0] << 24) |
|
A = (((uint8_t*) &addr->internal_ip_data.in6)[offset+0] << 24) |
|
||||||
(((uint8_t*) &addr->internal_ip_data.in6)[offset+1] << 16) |
|
(((uint8_t*) &addr->internal_ip_data.in6)[offset+1] << 16) |
|
||||||
(((uint8_t*) &addr->internal_ip_data.in6)[offset+2] << 8) |
|
(((uint8_t*) &addr->internal_ip_data.in6)[offset+2] << 8) |
|
||||||
(((uint8_t*) &addr->internal_ip_data.in6)[offset+3] << 0);
|
(((uint8_t*) &addr->internal_ip_data.in6)[offset+3] << 0);
|
||||||
|
|
||||||
B = (((uint8_t*) &mask->internal_ip_data.in6)[offset+0] << 24) |
|
B = (((uint8_t*) &mask->internal_ip_data.in6)[offset+0] << 24) |
|
||||||
(((uint8_t*) &mask->internal_ip_data.in6)[offset+1] << 16) |
|
(((uint8_t*) &mask->internal_ip_data.in6)[offset+1] << 16) |
|
||||||
(((uint8_t*) &mask->internal_ip_data.in6)[offset+2] << 8) |
|
(((uint8_t*) &mask->internal_ip_data.in6)[offset+2] << 8) |
|
||||||
(((uint8_t*) &mask->internal_ip_data.in6)[offset+3] << 0);
|
(((uint8_t*) &mask->internal_ip_data.in6)[offset+3] << 0);
|
||||||
|
|
||||||
C = A & B;
|
C = A & B;
|
||||||
|
|
||||||
D = (((uint8_t*) &C)[0] << 24) |
|
D = (((uint8_t*) &C)[0] << 24) |
|
||||||
(((uint8_t*) &C)[1] << 16) |
|
(((uint8_t*) &C)[1] << 16) |
|
||||||
(((uint8_t*) &C)[2] << 8) |
|
(((uint8_t*) &C)[2] << 8) |
|
||||||
@@ -321,7 +321,7 @@ void ip_mask_apply_OR(struct ip_addr_encap* addr, struct ip_addr_encap* mask, st
|
|||||||
{
|
{
|
||||||
memset(result, 0, sizeof(struct ip_addr_encap));
|
memset(result, 0, sizeof(struct ip_addr_encap));
|
||||||
result->af = addr->af;
|
result->af = addr->af;
|
||||||
|
|
||||||
if (addr->af == AF_INET)
|
if (addr->af == AF_INET)
|
||||||
{
|
{
|
||||||
result->internal_ip_data.in.s_addr = addr->internal_ip_data.in.s_addr | mask->internal_ip_data.in.s_addr;
|
result->internal_ip_data.in.s_addr = addr->internal_ip_data.in.s_addr | mask->internal_ip_data.in.s_addr;
|
||||||
@@ -334,19 +334,19 @@ void ip_mask_apply_OR(struct ip_addr_encap* addr, struct ip_addr_encap* mask, st
|
|||||||
for (n = 0; n < 4; n++)
|
for (n = 0; n < 4; n++)
|
||||||
{
|
{
|
||||||
offset = n * 4;
|
offset = n * 4;
|
||||||
|
|
||||||
A = (((uint8_t*) &addr->internal_ip_data.in6)[offset+0] << 24) |
|
A = (((uint8_t*) &addr->internal_ip_data.in6)[offset+0] << 24) |
|
||||||
(((uint8_t*) &addr->internal_ip_data.in6)[offset+1] << 16) |
|
(((uint8_t*) &addr->internal_ip_data.in6)[offset+1] << 16) |
|
||||||
(((uint8_t*) &addr->internal_ip_data.in6)[offset+2] << 8) |
|
(((uint8_t*) &addr->internal_ip_data.in6)[offset+2] << 8) |
|
||||||
(((uint8_t*) &addr->internal_ip_data.in6)[offset+3] << 0);
|
(((uint8_t*) &addr->internal_ip_data.in6)[offset+3] << 0);
|
||||||
|
|
||||||
B = (((uint8_t*) &mask->internal_ip_data.in6)[offset+0] << 24) |
|
B = (((uint8_t*) &mask->internal_ip_data.in6)[offset+0] << 24) |
|
||||||
(((uint8_t*) &mask->internal_ip_data.in6)[offset+1] << 16) |
|
(((uint8_t*) &mask->internal_ip_data.in6)[offset+1] << 16) |
|
||||||
(((uint8_t*) &mask->internal_ip_data.in6)[offset+2] << 8) |
|
(((uint8_t*) &mask->internal_ip_data.in6)[offset+2] << 8) |
|
||||||
(((uint8_t*) &mask->internal_ip_data.in6)[offset+3] << 0);
|
(((uint8_t*) &mask->internal_ip_data.in6)[offset+3] << 0);
|
||||||
|
|
||||||
C = A | B;
|
C = A | B;
|
||||||
|
|
||||||
D = (((uint8_t*) &C)[0] << 24) |
|
D = (((uint8_t*) &C)[0] << 24) |
|
||||||
(((uint8_t*) &C)[1] << 16) |
|
(((uint8_t*) &C)[1] << 16) |
|
||||||
(((uint8_t*) &C)[2] << 8) |
|
(((uint8_t*) &C)[2] << 8) |
|
||||||
@@ -368,7 +368,7 @@ int ip_compare(struct ip_addr_encap* a, struct ip_addr_encap* b)
|
|||||||
(((uint8_t*) &a->internal_ip_data.in.s_addr)[1] << 16) |
|
(((uint8_t*) &a->internal_ip_data.in.s_addr)[1] << 16) |
|
||||||
(((uint8_t*) &a->internal_ip_data.in.s_addr)[2] << 8) |
|
(((uint8_t*) &a->internal_ip_data.in.s_addr)[2] << 8) |
|
||||||
(((uint8_t*) &a->internal_ip_data.in.s_addr)[3] << 0);
|
(((uint8_t*) &a->internal_ip_data.in.s_addr)[3] << 0);
|
||||||
|
|
||||||
B = (((uint8_t*) &b->internal_ip_data.in.s_addr)[0] << 24) |
|
B = (((uint8_t*) &b->internal_ip_data.in.s_addr)[0] << 24) |
|
||||||
(((uint8_t*) &b->internal_ip_data.in.s_addr)[1] << 16) |
|
(((uint8_t*) &b->internal_ip_data.in.s_addr)[1] << 16) |
|
||||||
(((uint8_t*) &b->internal_ip_data.in.s_addr)[2] << 8) |
|
(((uint8_t*) &b->internal_ip_data.in.s_addr)[2] << 8) |
|
||||||
@@ -386,19 +386,19 @@ int ip_compare(struct ip_addr_encap* a, struct ip_addr_encap* b)
|
|||||||
(((uint8_t*) &a->internal_ip_data.in6)[offset+1] << 16) |
|
(((uint8_t*) &a->internal_ip_data.in6)[offset+1] << 16) |
|
||||||
(((uint8_t*) &a->internal_ip_data.in6)[offset+2] << 8) |
|
(((uint8_t*) &a->internal_ip_data.in6)[offset+2] << 8) |
|
||||||
(((uint8_t*) &a->internal_ip_data.in6)[offset+3] << 0);
|
(((uint8_t*) &a->internal_ip_data.in6)[offset+3] << 0);
|
||||||
|
|
||||||
B = (((uint8_t*) &b->internal_ip_data.in6)[offset+0] << 24) |
|
B = (((uint8_t*) &b->internal_ip_data.in6)[offset+0] << 24) |
|
||||||
(((uint8_t*) &b->internal_ip_data.in6)[offset+1] << 16) |
|
(((uint8_t*) &b->internal_ip_data.in6)[offset+1] << 16) |
|
||||||
(((uint8_t*) &b->internal_ip_data.in6)[offset+2] << 8) |
|
(((uint8_t*) &b->internal_ip_data.in6)[offset+2] << 8) |
|
||||||
(((uint8_t*) &b->internal_ip_data.in6)[offset+3] << 0);
|
(((uint8_t*) &b->internal_ip_data.in6)[offset+3] << 0);
|
||||||
|
|
||||||
if (A == B) continue;
|
if (A == B) continue;
|
||||||
|
|
||||||
return A - B;
|
return A - B;
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef IP_CALC_DEBUG
|
#ifdef IP_CALC_DEBUG
|
||||||
char* a_str = hub_strdup(ip_convert_to_string(a));
|
char* a_str = hub_strdup(ip_convert_to_string(a));
|
||||||
char* b_str = hub_strdup(ip_convert_to_string(b));
|
char* b_str = hub_strdup(ip_convert_to_string(b));
|
||||||
@@ -406,7 +406,7 @@ int ip_compare(struct ip_addr_encap* a, struct ip_addr_encap* b)
|
|||||||
hub_free(a_str);
|
hub_free(a_str);
|
||||||
hub_free(b_str);
|
hub_free(b_str);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2009, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -18,7 +18,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This file is used for fiddling with IP-addresses,
|
* This file is used for fiddling with IP-addresses,
|
||||||
* primarily used for IP-banning in uhub.
|
* primarily used for IP-banning in uhub.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -449,7 +449,7 @@ int net_is_ipv6_supported()
|
|||||||
is_ipv6_supported = 0;
|
is_ipv6_supported = 0;
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
net_error_out(ret, "net_is_ipv6_supported");
|
net_error_out(ret, "net_is_ipv6_supported");
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -508,7 +508,7 @@ const char* net_address_to_string(int af, const void* src, char* dst, socklen_t
|
|||||||
size_t size;
|
size_t size;
|
||||||
LPSOCKADDR addr;
|
LPSOCKADDR addr;
|
||||||
DWORD len = cnt;
|
DWORD len = cnt;
|
||||||
|
|
||||||
switch (af)
|
switch (af)
|
||||||
{
|
{
|
||||||
case AF_INET:
|
case AF_INET:
|
||||||
@@ -531,7 +531,7 @@ const char* net_address_to_string(int af, const void* src, char* dst, socklen_t
|
|||||||
default:
|
default:
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (WSAAddressToStringA(addr, size, NULL, dst, &len) == 0)
|
if (WSAAddressToStringA(addr, size, NULL, dst, &len) == 0)
|
||||||
{
|
{
|
||||||
return dst;
|
return dst;
|
||||||
@@ -602,15 +602,15 @@ const char* net_get_peer_address(int fd)
|
|||||||
struct sockaddr_in* name4;
|
struct sockaddr_in* name4;
|
||||||
struct sockaddr* name;
|
struct sockaddr* name;
|
||||||
socklen_t namelen;
|
socklen_t namelen;
|
||||||
|
|
||||||
memset(address, 0, INET6_ADDRSTRLEN);
|
memset(address, 0, INET6_ADDRSTRLEN);
|
||||||
namelen = sizeof(struct sockaddr_storage);
|
namelen = sizeof(struct sockaddr_storage);
|
||||||
memset(&storage, 0, namelen);
|
memset(&storage, 0, namelen);
|
||||||
|
|
||||||
name6 = (struct sockaddr_in6*) &storage;
|
name6 = (struct sockaddr_in6*) &storage;
|
||||||
name4 = (struct sockaddr_in*) &storage;
|
name4 = (struct sockaddr_in*) &storage;
|
||||||
name = (struct sockaddr*) &storage;
|
name = (struct sockaddr*) &storage;
|
||||||
|
|
||||||
if (getpeername(fd, (struct sockaddr*) name, &namelen) != -1)
|
if (getpeername(fd, (struct sockaddr*) name, &namelen) != -1)
|
||||||
{
|
{
|
||||||
int af = storage.ss_family;
|
int af = storage.ss_family;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,102 +0,0 @@
|
|||||||
/*
|
|
||||||
* uhub - A tiny ADC p2p connection hub
|
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation; either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "uhub.h"
|
|
||||||
|
|
||||||
struct uhub_notify_handle
|
|
||||||
{
|
|
||||||
net_notify_callback callback;
|
|
||||||
void* ptr;
|
|
||||||
#ifndef WIN32
|
|
||||||
int pipe_fd[2];
|
|
||||||
struct net_connection* con;
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This contains a mechanism to wake up the main thread
|
|
||||||
* in a thread safe manner while it would be blocking
|
|
||||||
* in select() or something equivalent typically invoked from
|
|
||||||
* net_backend_process().
|
|
||||||
*
|
|
||||||
* The main usage is for the DNS resolver to notify the
|
|
||||||
* main thread that there are DNS results to be
|
|
||||||
* processed.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a notification handle.
|
|
||||||
*/
|
|
||||||
#ifndef WIN32
|
|
||||||
static void notify_callback(struct net_connection* con, int event, void* ptr)
|
|
||||||
{
|
|
||||||
LOG_TRACE("notify_callback()");
|
|
||||||
struct uhub_notify_handle* handle = (struct uhub_notify_handle*) ptr;
|
|
||||||
char buf;
|
|
||||||
int ret = read(handle->pipe_fd[0], &buf, 1);
|
|
||||||
if (ret == 1)
|
|
||||||
{
|
|
||||||
if (handle->callback)
|
|
||||||
handle->callback(handle, handle->ptr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
struct uhub_notify_handle* net_notify_create(net_notify_callback cb, void* ptr)
|
|
||||||
{
|
|
||||||
LOG_TRACE("net_notify_create()");
|
|
||||||
struct uhub_notify_handle* handle = (struct uhub_notify_handle*) hub_malloc(sizeof(struct uhub_notify_handle));
|
|
||||||
handle->callback = cb;
|
|
||||||
handle->ptr = ptr;
|
|
||||||
#ifndef WIN32
|
|
||||||
int ret = pipe(handle->pipe_fd);
|
|
||||||
if (ret == -1)
|
|
||||||
{
|
|
||||||
LOG_ERROR("Unable to setup notification pipes.");
|
|
||||||
hub_free(handle);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
handle->con = net_con_create();
|
|
||||||
net_con_initialize(handle->con, handle->pipe_fd[0], notify_callback, handle, NET_EVENT_READ);
|
|
||||||
#endif
|
|
||||||
return handle;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void net_notify_destroy(struct uhub_notify_handle* handle)
|
|
||||||
{
|
|
||||||
LOG_TRACE("net_notify_destroy()");
|
|
||||||
#ifndef WIN32
|
|
||||||
net_con_destroy(handle->con);
|
|
||||||
close(handle->pipe_fd[0]);
|
|
||||||
close(handle->pipe_fd[1]);
|
|
||||||
handle->pipe_fd[0] = -1;
|
|
||||||
handle->pipe_fd[0] = -1;
|
|
||||||
#endif
|
|
||||||
hub_free(handle);
|
|
||||||
}
|
|
||||||
|
|
||||||
void net_notify_signal(struct uhub_notify_handle* handle, char data)
|
|
||||||
{
|
|
||||||
LOG_TRACE("net_notify_signal()");
|
|
||||||
#ifndef WIN32
|
|
||||||
write(handle->pipe_fd[1], &data, 1);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
@@ -1,56 +0,0 @@
|
|||||||
/*
|
|
||||||
* uhub - A tiny ADC p2p connection hub
|
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation; either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef HAVE_UHUB_NETWORK_NOTIFY_API_H
|
|
||||||
#define HAVE_UHUB_NETWORK_NOTIFY_API_H
|
|
||||||
|
|
||||||
struct uhub_notify_handle;
|
|
||||||
|
|
||||||
typedef void (*net_notify_callback)(struct uhub_notify_handle* handle, void* ptr);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* This contains a mechanism to wake up the main thread
|
|
||||||
* in a thread safe manner while it would be blocking
|
|
||||||
* in select() or something equivalent typically invoked from
|
|
||||||
* net_backend_process().
|
|
||||||
*
|
|
||||||
* The main usage is for the DNS resolver to notify the
|
|
||||||
* main thread that there are DNS results to be
|
|
||||||
* processed.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a notification handle.
|
|
||||||
*/
|
|
||||||
struct uhub_notify_handle* net_notify_create(net_notify_callback cb, void* ptr);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Destroy a notification handle.
|
|
||||||
*/
|
|
||||||
void net_notify_destroy(struct uhub_notify_handle*);
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Signal the notification handle, this will surely
|
|
||||||
* interrupt the net_backend_process(), and force it to
|
|
||||||
* process messages.
|
|
||||||
*/
|
|
||||||
void net_notify_signal(struct uhub_notify_handle*, char data);
|
|
||||||
|
|
||||||
|
|
||||||
#endif /* HAVE_UHUB_NETWORK_NOTIFY_API_H */
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -20,7 +20,6 @@
|
|||||||
#include "uhub.h"
|
#include "uhub.h"
|
||||||
#include "network/common.h"
|
#include "network/common.h"
|
||||||
#include "network/tls.h"
|
#include "network/tls.h"
|
||||||
#include "network/backend.h"
|
|
||||||
|
|
||||||
#ifdef SSL_SUPPORT
|
#ifdef SSL_SUPPORT
|
||||||
#ifdef SSL_USE_OPENSSL
|
#ifdef SSL_USE_OPENSSL
|
||||||
@@ -33,9 +32,6 @@ struct net_ssl_openssl
|
|||||||
SSL* ssl;
|
SSL* ssl;
|
||||||
BIO* bio;
|
BIO* bio;
|
||||||
enum ssl_state state;
|
enum ssl_state state;
|
||||||
int events;
|
|
||||||
int ssl_read_events;
|
|
||||||
int ssl_write_events;
|
|
||||||
uint32_t flags;
|
uint32_t flags;
|
||||||
size_t bytes_rx;
|
size_t bytes_rx;
|
||||||
size_t bytes_tx;
|
size_t bytes_tx;
|
||||||
@@ -43,7 +39,8 @@ struct net_ssl_openssl
|
|||||||
|
|
||||||
struct net_context_openssl
|
struct net_context_openssl
|
||||||
{
|
{
|
||||||
SSL_CTX* ssl;
|
SSL_METHOD* ssl_method;
|
||||||
|
SSL_CTX* ssl_ctx;
|
||||||
};
|
};
|
||||||
|
|
||||||
static struct net_ssl_openssl* get_handle(struct net_connection* con)
|
static struct net_ssl_openssl* get_handle(struct net_connection* con)
|
||||||
@@ -52,29 +49,6 @@ static struct net_ssl_openssl* get_handle(struct net_connection* con)
|
|||||||
return (struct net_ssl_openssl*) con->ssl;
|
return (struct net_ssl_openssl*) con->ssl;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef DEBUG
|
|
||||||
static const char* get_state_str(enum ssl_state state)
|
|
||||||
{
|
|
||||||
switch (state)
|
|
||||||
{
|
|
||||||
case tls_st_none: return "tls_st_none";
|
|
||||||
case tls_st_error: return "tls_st_error";
|
|
||||||
case tls_st_accepting: return "tls_st_accepting";
|
|
||||||
case tls_st_connecting: return "tls_st_connecting";
|
|
||||||
case tls_st_connected: return "tls_st_connected";
|
|
||||||
case tls_st_disconnecting: return "tls_st_disconnecting";
|
|
||||||
}
|
|
||||||
uhub_assert(!"This should not happen - invalid state!");
|
|
||||||
return "(UNKNOWN STATE)";
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
static void net_ssl_set_state(struct net_ssl_openssl* handle, enum ssl_state new_state)
|
|
||||||
{
|
|
||||||
LOG_DEBUG("net_ssl_set_state(): prev_state=%s, new_state=%s", get_state_str(handle->state), get_state_str(new_state));
|
|
||||||
handle->state = new_state;
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* net_ssl_get_provider()
|
const char* net_ssl_get_provider()
|
||||||
{
|
{
|
||||||
return OPENSSL_VERSION_TEXT;
|
return OPENSSL_VERSION_TEXT;
|
||||||
@@ -96,10 +70,10 @@ int net_ssl_library_shutdown()
|
|||||||
ENGINE_cleanup();
|
ENGINE_cleanup();
|
||||||
CONF_modules_unload(1);
|
CONF_modules_unload(1);
|
||||||
|
|
||||||
ERR_free_strings();
|
ERR_free_strings();
|
||||||
EVP_cleanup();
|
EVP_cleanup();
|
||||||
CRYPTO_cleanup_all_ex_data();
|
CRYPTO_cleanup_all_ex_data();
|
||||||
|
|
||||||
// sk_SSL_COMP_free(SSL_COMP_get_compression_methods());
|
// sk_SSL_COMP_free(SSL_COMP_get_compression_methods());
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
@@ -119,83 +93,43 @@ static void add_io_stats(struct net_ssl_openssl* handle)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static const SSL_METHOD* get_ssl_method(const char* tls_version)
|
|
||||||
{
|
|
||||||
if (!tls_version || !*tls_version)
|
|
||||||
{
|
|
||||||
LOG_ERROR("tls_version is not set.");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!strcmp(tls_version, "1.0"))
|
|
||||||
return TLSv1_method();
|
|
||||||
if (!strcmp(tls_version, "1.1"))
|
|
||||||
return TLSv1_1_method();
|
|
||||||
if (!strcmp(tls_version, "1.2"))
|
|
||||||
return TLSv1_2_method();
|
|
||||||
|
|
||||||
LOG_ERROR("Unable to recognize tls_version.");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new SSL context.
|
* Create a new SSL context.
|
||||||
*/
|
*/
|
||||||
struct ssl_context_handle* net_ssl_context_create(const char* tls_version, const char* tls_ciphersuite)
|
struct ssl_context_handle* net_ssl_context_create()
|
||||||
{
|
{
|
||||||
|
|
||||||
struct net_context_openssl* ctx = (struct net_context_openssl*) hub_malloc_zero(sizeof(struct net_context_openssl));
|
struct net_context_openssl* ctx = (struct net_context_openssl*) hub_malloc_zero(sizeof(struct net_context_openssl));
|
||||||
const SSL_METHOD* ssl_method = get_ssl_method(tls_version);
|
ctx->ssl_method = (SSL_METHOD*) SSLv23_method(); /* TLSv1_method() */
|
||||||
|
ctx->ssl_ctx = SSL_CTX_new(ctx->ssl_method);
|
||||||
if (!ssl_method)
|
|
||||||
{
|
|
||||||
hub_free(ctx);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx->ssl = SSL_CTX_new(ssl_method);
|
|
||||||
|
|
||||||
/* Disable SSLv2 */
|
/* Disable SSLv2 */
|
||||||
SSL_CTX_set_options(ctx->ssl, SSL_OP_NO_SSLv2);
|
SSL_CTX_set_options(ctx->ssl_ctx, SSL_OP_NO_SSLv2);
|
||||||
|
|
||||||
// #ifdef SSL_OP_NO_SSLv3
|
|
||||||
/* Disable SSLv3 */
|
|
||||||
SSL_CTX_set_options(ctx->ssl, SSL_OP_NO_SSLv3);
|
|
||||||
// #endif
|
|
||||||
|
|
||||||
// FIXME: Why did we need this again?
|
|
||||||
SSL_CTX_set_quiet_shutdown(ctx->ssl, 1);
|
|
||||||
|
|
||||||
#ifdef SSL_OP_NO_COMPRESSION
|
#ifdef SSL_OP_NO_COMPRESSION
|
||||||
/* Disable compression */
|
/* Disable compression? */
|
||||||
LOG_TRACE("Disabling SSL compression."); /* "CRIME" attack */
|
LOG_TRACE("Disabling SSL compression."); /* "CRIME" attack */
|
||||||
SSL_CTX_set_options(ctx->ssl, SSL_OP_NO_COMPRESSION);
|
SSL_CTX_set_options(ctx->ssl_ctx, SSL_OP_NO_COMPRESSION);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Set preferred cipher suite */
|
SSL_CTX_set_quiet_shutdown(ctx->ssl_ctx, 1);
|
||||||
if (SSL_CTX_set_cipher_list(ctx->ssl, tls_ciphersuite) != 1)
|
|
||||||
{
|
|
||||||
LOG_ERROR("Unable to set cipher suite.");
|
|
||||||
SSL_CTX_free(ctx->ssl);
|
|
||||||
hub_free(ctx);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (struct ssl_context_handle*) ctx;
|
return (struct ssl_context_handle*) ctx;
|
||||||
}
|
}
|
||||||
|
|
||||||
void net_ssl_context_destroy(struct ssl_context_handle* ctx_)
|
extern void net_ssl_context_destroy(struct ssl_context_handle* ctx_)
|
||||||
{
|
{
|
||||||
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
||||||
SSL_CTX_free(ctx->ssl);
|
SSL_CTX_free(ctx->ssl_ctx);
|
||||||
hub_free(ctx);
|
hub_free(ctx);
|
||||||
}
|
}
|
||||||
|
|
||||||
int ssl_load_certificate(struct ssl_context_handle* ctx_, const char* pem_file)
|
int ssl_load_certificate(struct ssl_context_handle* ctx_, const char* pem_file)
|
||||||
{
|
{
|
||||||
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
||||||
if (SSL_CTX_use_certificate_chain_file(ctx->ssl, pem_file) < 0)
|
if (SSL_CTX_use_certificate_file(ctx->ssl_ctx, pem_file, SSL_FILETYPE_PEM) < 0)
|
||||||
{
|
{
|
||||||
LOG_ERROR("SSL_CTX_use_certificate_chain_file: %s", ERR_error_string(ERR_get_error(), NULL));
|
LOG_ERROR("SSL_CTX_use_certificate_file: %s", ERR_error_string(ERR_get_error(), NULL));
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -205,7 +139,7 @@ int ssl_load_certificate(struct ssl_context_handle* ctx_, const char* pem_file)
|
|||||||
int ssl_load_private_key(struct ssl_context_handle* ctx_, const char* pem_file)
|
int ssl_load_private_key(struct ssl_context_handle* ctx_, const char* pem_file)
|
||||||
{
|
{
|
||||||
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
||||||
if (SSL_CTX_use_PrivateKey_file(ctx->ssl, pem_file, SSL_FILETYPE_PEM) < 0)
|
if (SSL_CTX_use_PrivateKey_file(ctx->ssl_ctx, pem_file, SSL_FILETYPE_PEM) < 0)
|
||||||
{
|
{
|
||||||
LOG_ERROR("SSL_CTX_use_PrivateKey_file: %s", ERR_error_string(ERR_get_error(), NULL));
|
LOG_ERROR("SSL_CTX_use_PrivateKey_file: %s", ERR_error_string(ERR_get_error(), NULL));
|
||||||
return 0;
|
return 0;
|
||||||
@@ -216,7 +150,7 @@ int ssl_load_private_key(struct ssl_context_handle* ctx_, const char* pem_file)
|
|||||||
int ssl_check_private_key(struct ssl_context_handle* ctx_)
|
int ssl_check_private_key(struct ssl_context_handle* ctx_)
|
||||||
{
|
{
|
||||||
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
struct net_context_openssl* ctx = (struct net_context_openssl*) ctx_;
|
||||||
if (SSL_CTX_check_private_key(ctx->ssl) != 1)
|
if (SSL_CTX_check_private_key(ctx->ssl_ctx) != 1)
|
||||||
{
|
{
|
||||||
LOG_FATAL("SSL_CTX_check_private_key: Private key does not match the certificate public key: %s", ERR_error_string(ERR_get_error(), NULL));
|
LOG_FATAL("SSL_CTX_check_private_key: Private key does not match the certificate public key: %s", ERR_error_string(ERR_get_error(), NULL));
|
||||||
return 0;
|
return 0;
|
||||||
@@ -224,7 +158,7 @@ int ssl_check_private_key(struct ssl_context_handle* ctx_)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int handle_openssl_error(struct net_connection* con, int ret, int read)
|
static int handle_openssl_error(struct net_connection* con, int ret, enum ssl_state forced_rwstate)
|
||||||
{
|
{
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
struct net_ssl_openssl* handle = get_handle(con);
|
||||||
int err = SSL_get_error(handle->ssl, ret);
|
int err = SSL_get_error(handle->ssl, ret);
|
||||||
@@ -235,43 +169,33 @@ static int handle_openssl_error(struct net_connection* con, int ret, int read)
|
|||||||
return -1;
|
return -1;
|
||||||
|
|
||||||
case SSL_ERROR_WANT_READ:
|
case SSL_ERROR_WANT_READ:
|
||||||
if (read)
|
handle->state = forced_rwstate;
|
||||||
handle->ssl_read_events = NET_EVENT_READ;
|
net_con_update(con, NET_EVENT_READ);
|
||||||
else
|
|
||||||
handle->ssl_write_events = NET_EVENT_READ;
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
case SSL_ERROR_WANT_WRITE:
|
case SSL_ERROR_WANT_WRITE:
|
||||||
if (read)
|
handle->state = forced_rwstate;
|
||||||
handle->ssl_read_events = NET_EVENT_WRITE;
|
net_con_update(con, NET_EVENT_WRITE);
|
||||||
else
|
|
||||||
handle->ssl_write_events = NET_EVENT_WRITE;
|
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
case SSL_ERROR_SSL:
|
|
||||||
net_ssl_set_state(handle, tls_st_error);
|
|
||||||
return -2;
|
|
||||||
|
|
||||||
case SSL_ERROR_SYSCALL:
|
case SSL_ERROR_SYSCALL:
|
||||||
net_ssl_set_state(handle, tls_st_error);
|
handle->state = tls_st_error;
|
||||||
return -2;
|
return -2;
|
||||||
}
|
}
|
||||||
|
|
||||||
return -2;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t net_con_ssl_accept(struct net_connection* con)
|
ssize_t net_con_ssl_accept(struct net_connection* con)
|
||||||
{
|
{
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
struct net_ssl_openssl* handle = get_handle(con);
|
||||||
|
handle->state = tls_st_accepting;
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
net_ssl_set_state(handle, tls_st_accepting);
|
|
||||||
|
|
||||||
ret = SSL_accept(handle->ssl);
|
ret = SSL_accept(handle->ssl);
|
||||||
LOG_PROTO("SSL_accept() ret=%d", ret);
|
LOG_PROTO("SSL_accept() ret=%d", ret);
|
||||||
if (ret > 0)
|
if (ret > 0)
|
||||||
{
|
{
|
||||||
net_con_update(con, NET_EVENT_READ);
|
net_con_update(con, NET_EVENT_READ);
|
||||||
net_ssl_set_state(handle, tls_st_connected);
|
handle->state = tls_st_connected;
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
return handle_openssl_error(con, ret, tls_st_accepting);
|
return handle_openssl_error(con, ret, tls_st_accepting);
|
||||||
@@ -281,34 +205,30 @@ ssize_t net_con_ssl_connect(struct net_connection* con)
|
|||||||
{
|
{
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
struct net_ssl_openssl* handle = get_handle(con);
|
||||||
ssize_t ret;
|
ssize_t ret;
|
||||||
net_ssl_set_state(handle, tls_st_connecting);
|
handle->state = tls_st_connecting;
|
||||||
|
|
||||||
ret = SSL_connect(handle->ssl);
|
ret = SSL_connect(handle->ssl);
|
||||||
LOG_PROTO("SSL_connect() ret=%d", ret);
|
LOG_PROTO("SSL_connect() ret=%d", ret);
|
||||||
|
|
||||||
if (ret > 0)
|
if (ret > 0)
|
||||||
{
|
{
|
||||||
|
handle->state = tls_st_connected;
|
||||||
net_con_update(con, NET_EVENT_READ);
|
net_con_update(con, NET_EVENT_READ);
|
||||||
net_ssl_set_state(handle, tls_st_connected);
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
return handle_openssl_error(con, ret, tls_st_connecting);
|
||||||
ret = handle_openssl_error(con, ret, tls_st_connecting);
|
|
||||||
LOG_ERROR("net_con_ssl_connect: ret=%d", ret);
|
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t net_con_ssl_handshake(struct net_connection* con, enum net_con_ssl_mode ssl_mode, struct ssl_context_handle* ssl_ctx)
|
ssize_t net_con_ssl_handshake(struct net_connection* con, enum net_con_ssl_mode ssl_mode, struct ssl_context_handle* ssl_ctx)
|
||||||
{
|
{
|
||||||
uhub_assert(con);
|
uhub_assert(con);
|
||||||
uhub_assert(ssl_ctx);
|
|
||||||
|
|
||||||
struct net_context_openssl* ctx = (struct net_context_openssl*) ssl_ctx;
|
struct net_context_openssl* ctx = (struct net_context_openssl*) ssl_ctx;
|
||||||
struct net_ssl_openssl* handle = (struct net_ssl_openssl*) hub_malloc_zero(sizeof(struct net_ssl_openssl));
|
struct net_ssl_openssl* handle = (struct net_ssl_openssl*) hub_malloc_zero(sizeof(struct net_ssl_openssl));
|
||||||
|
|
||||||
if (ssl_mode == net_con_ssl_mode_server)
|
if (ssl_mode == net_con_ssl_mode_server)
|
||||||
{
|
{
|
||||||
handle->ssl = SSL_new(ctx->ssl);
|
handle->ssl = SSL_new(ctx->ssl_ctx);
|
||||||
if (!handle->ssl)
|
if (!handle->ssl)
|
||||||
{
|
{
|
||||||
LOG_ERROR("Unable to create new SSL stream\n");
|
LOG_ERROR("Unable to create new SSL stream\n");
|
||||||
@@ -321,37 +241,31 @@ ssize_t net_con_ssl_handshake(struct net_connection* con, enum net_con_ssl_mode
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
handle->ssl = SSL_new(ctx->ssl);
|
handle->ssl = SSL_new(SSL_CTX_new(TLSv1_method()));
|
||||||
SSL_set_fd(handle->ssl, con->sd);
|
SSL_set_fd(handle->ssl, con->sd);
|
||||||
handle->bio = SSL_get_rbio(handle->ssl);
|
handle->bio = SSL_get_rbio(handle->ssl);
|
||||||
con->ssl = (struct ssl_handle*) handle;
|
con->ssl = (struct ssl_handle*) handle;
|
||||||
return net_con_ssl_connect(con);
|
return net_con_ssl_connect(con);
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t net_ssl_send(struct net_connection* con, const void* buf, size_t len)
|
ssize_t net_ssl_send(struct net_connection* con, const void* buf, size_t len)
|
||||||
{
|
{
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
struct net_ssl_openssl* handle = get_handle(con);
|
||||||
|
|
||||||
LOG_ERROR("net_ssl_send(), state=%d", (int) handle->state);
|
uhub_assert(handle->state == tls_st_connected || handle->state == tls_st_need_write);
|
||||||
|
|
||||||
if (handle->state == tls_st_error)
|
|
||||||
return -2;
|
|
||||||
|
|
||||||
uhub_assert(handle->state == tls_st_connected);
|
|
||||||
|
|
||||||
|
|
||||||
ERR_clear_error();
|
ERR_clear_error();
|
||||||
ssize_t ret = SSL_write(handle->ssl, buf, len);
|
ssize_t ret = SSL_write(handle->ssl, buf, len);
|
||||||
add_io_stats(handle);
|
add_io_stats(handle);
|
||||||
LOG_PROTO("SSL_write(con=%p, buf=%p, len=" PRINTF_SIZE_T ") => %d", con, buf, len, ret);
|
LOG_PROTO("SSL_write(con=%p, buf=%p, len=" PRINTF_SIZE_T ") => %d", con, buf, len, ret);
|
||||||
if (ret > 0)
|
if (ret > 0)
|
||||||
handle->ssl_write_events = 0;
|
{
|
||||||
else
|
handle->state = tls_st_connected;
|
||||||
ret = handle_openssl_error(con, ret, 0);
|
return ret;
|
||||||
|
}
|
||||||
net_ssl_update(con, handle->events); // Update backend only
|
return handle_openssl_error(con, ret, tls_st_need_write);
|
||||||
return ret;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ssize_t net_ssl_recv(struct net_connection* con, void* buf, size_t len)
|
ssize_t net_ssl_recv(struct net_connection* con, void* buf, size_t len)
|
||||||
@@ -362,10 +276,7 @@ ssize_t net_ssl_recv(struct net_connection* con, void* buf, size_t len)
|
|||||||
if (handle->state == tls_st_error)
|
if (handle->state == tls_st_error)
|
||||||
return -2;
|
return -2;
|
||||||
|
|
||||||
if (handle->state == tls_st_accepting || handle->state == tls_st_connecting)
|
uhub_assert(handle->state == tls_st_connected || handle->state == tls_st_need_read);
|
||||||
return -1;
|
|
||||||
|
|
||||||
uhub_assert(handle->state == tls_st_connected);
|
|
||||||
|
|
||||||
ERR_clear_error();
|
ERR_clear_error();
|
||||||
|
|
||||||
@@ -373,43 +284,29 @@ ssize_t net_ssl_recv(struct net_connection* con, void* buf, size_t len)
|
|||||||
add_io_stats(handle);
|
add_io_stats(handle);
|
||||||
LOG_PROTO("SSL_read(con=%p, buf=%p, len=" PRINTF_SIZE_T ") => %d", con, buf, len, ret);
|
LOG_PROTO("SSL_read(con=%p, buf=%p, len=" PRINTF_SIZE_T ") => %d", con, buf, len, ret);
|
||||||
if (ret > 0)
|
if (ret > 0)
|
||||||
handle->ssl_read_events = 0;
|
{
|
||||||
else
|
handle->state = tls_st_connected;
|
||||||
ret = handle_openssl_error(con, ret, 1);
|
return ret;
|
||||||
|
}
|
||||||
net_ssl_update(con, handle->events); // Update backend only
|
return handle_openssl_error(con, ret, tls_st_need_read);
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
void net_ssl_update(struct net_connection* con, int events)
|
|
||||||
{
|
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
|
||||||
handle->events = events;
|
|
||||||
net_backend_update(con, handle->events | handle->ssl_read_events | handle->ssl_write_events);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void net_ssl_shutdown(struct net_connection* con)
|
void net_ssl_shutdown(struct net_connection* con)
|
||||||
{
|
{
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
struct net_ssl_openssl* handle = get_handle(con);
|
||||||
if (handle)
|
SSL_shutdown(handle->ssl);
|
||||||
{
|
SSL_clear(handle->ssl);
|
||||||
SSL_shutdown(handle->ssl);
|
|
||||||
SSL_clear(handle->ssl);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void net_ssl_destroy(struct net_connection* con)
|
void net_ssl_destroy(struct net_connection* con)
|
||||||
{
|
{
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
struct net_ssl_openssl* handle = get_handle(con);
|
||||||
LOG_TRACE("net_ssl_destroy: %p", con);
|
|
||||||
SSL_free(handle->ssl);
|
SSL_free(handle->ssl);
|
||||||
hub_free(handle);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void net_ssl_callback(struct net_connection* con, int events)
|
void net_ssl_callback(struct net_connection* con, int events)
|
||||||
{
|
{
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
struct net_ssl_openssl* handle = get_handle(con);
|
||||||
int ret;
|
|
||||||
|
|
||||||
switch (handle->state)
|
switch (handle->state)
|
||||||
{
|
{
|
||||||
@@ -418,7 +315,7 @@ void net_ssl_callback(struct net_connection* con, int events)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case tls_st_error:
|
case tls_st_error:
|
||||||
con->callback(con, NET_EVENT_ERROR, con->ptr);
|
con->callback(con, NET_EVENT_READ, con->ptr);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case tls_st_accepting:
|
case tls_st_accepting:
|
||||||
@@ -427,27 +324,19 @@ void net_ssl_callback(struct net_connection* con, int events)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case tls_st_connecting:
|
case tls_st_connecting:
|
||||||
ret = net_con_ssl_connect(con);
|
if (net_con_ssl_connect(con) != 0)
|
||||||
if (ret == 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (ret > 0)
|
|
||||||
{
|
|
||||||
LOG_DEBUG("%p SSL connected!", con);
|
|
||||||
con->callback(con, NET_EVENT_READ, con->ptr);
|
con->callback(con, NET_EVENT_READ, con->ptr);
|
||||||
}
|
break;
|
||||||
else
|
|
||||||
{
|
case tls_st_need_read:
|
||||||
LOG_DEBUG("%p SSL handshake failed!", con);
|
con->callback(con, NET_EVENT_READ, con->ptr);
|
||||||
con->callback(con, NET_EVENT_ERROR, con->ptr);
|
break;
|
||||||
}
|
|
||||||
|
case tls_st_need_write:
|
||||||
|
con->callback(con, NET_EVENT_WRITE, con->ptr);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case tls_st_connected:
|
case tls_st_connected:
|
||||||
if (handle->ssl_read_events & events)
|
|
||||||
events |= NET_EVENT_READ;
|
|
||||||
if (handle->ssl_write_events & events)
|
|
||||||
events |= NET_EVENT_WRITE;
|
|
||||||
con->callback(con, events, con->ptr);
|
con->callback(con, events, con->ptr);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -456,18 +345,6 @@ void net_ssl_callback(struct net_connection* con, int events)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const char* net_ssl_get_tls_version(struct net_connection* con)
|
|
||||||
{
|
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
|
||||||
return SSL_get_version(handle->ssl);
|
|
||||||
}
|
|
||||||
|
|
||||||
const char* net_ssl_get_tls_cipher(struct net_connection* con)
|
|
||||||
{
|
|
||||||
struct net_ssl_openssl* handle = get_handle(con);
|
|
||||||
const SSL_CIPHER *cipher = SSL_get_current_cipher(handle->ssl);
|
|
||||||
return SSL_CIPHER_get_name(cipher);
|
|
||||||
}
|
|
||||||
|
|
||||||
#endif /* SSL_USE_OPENSSL */
|
#endif /* SSL_USE_OPENSSL */
|
||||||
#endif /* SSL_SUPPORT */
|
#endif /* SSL_SUPPORT */
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2013, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -42,7 +42,6 @@ void timeout_queue_initialize(struct timeout_queue* t, time_t now, size_t max)
|
|||||||
{
|
{
|
||||||
t->last = now;
|
t->last = now;
|
||||||
t->max = max;
|
t->max = max;
|
||||||
memset(&t->lock, 0, sizeof(t->lock));
|
|
||||||
t->events = hub_malloc_zero(max * sizeof(struct timeout_evt*));
|
t->events = hub_malloc_zero(max * sizeof(struct timeout_evt*));
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -53,66 +52,37 @@ void timeout_queue_shutdown(struct timeout_queue* t)
|
|||||||
t->max = 0;
|
t->max = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int timeout_queue_locked(struct timeout_queue* t)
|
|
||||||
{
|
|
||||||
return t->lock.ptr != NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void timeout_queue_lock(struct timeout_queue* t)
|
|
||||||
{
|
|
||||||
t->lock.ptr = t;
|
|
||||||
}
|
|
||||||
|
|
||||||
// unlock and flush the locked events to the main timeout queue.
|
|
||||||
static void timeout_queue_unlock(struct timeout_queue* t)
|
|
||||||
{
|
|
||||||
struct timeout_evt* evt, *tmp, *first;
|
|
||||||
size_t pos;
|
|
||||||
t->lock.ptr = NULL;
|
|
||||||
|
|
||||||
evt = t->lock.next;
|
|
||||||
while (evt)
|
|
||||||
{
|
|
||||||
tmp = evt->next;
|
|
||||||
pos = evt->timestamp % t->max;
|
|
||||||
first = t->events[pos];
|
|
||||||
if (first)
|
|
||||||
{
|
|
||||||
first->prev->next = evt;
|
|
||||||
evt->prev = first->prev;
|
|
||||||
first->prev = evt;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
t->events[pos] = evt;
|
|
||||||
evt->prev = evt;
|
|
||||||
}
|
|
||||||
evt->next = 0;
|
|
||||||
evt = tmp;
|
|
||||||
}
|
|
||||||
|
|
||||||
t->lock.next = 0;
|
|
||||||
t->lock.prev = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
size_t timeout_queue_process(struct timeout_queue* t, time_t now)
|
size_t timeout_queue_process(struct timeout_queue* t, time_t now)
|
||||||
{
|
{
|
||||||
size_t pos = (size_t) t->last;
|
size_t pos = (size_t) t->last;
|
||||||
size_t events = 0;
|
size_t events = 0;
|
||||||
struct timeout_evt* evt = 0;
|
struct timeout_evt* evt = 0;
|
||||||
|
|
||||||
|
uhub_assert(t->last <= now);
|
||||||
t->last = now;
|
t->last = now;
|
||||||
timeout_queue_lock(t);
|
|
||||||
|
// We can optimize in case we need to wrap around
|
||||||
|
// the buffer, so we only do it once
|
||||||
|
if (MAX(pos, now) - MIN(pos, now) > t->max)
|
||||||
|
{
|
||||||
|
// FIXME: Double check this calculation
|
||||||
|
pos = (now - t->max);
|
||||||
|
}
|
||||||
|
|
||||||
for (; pos <= now; pos++)
|
for (; pos <= now; pos++)
|
||||||
{
|
{
|
||||||
while ((evt = t->events[pos % t->max]))
|
evt = t->events[pos % t->max];
|
||||||
|
while (evt)
|
||||||
{
|
{
|
||||||
timeout_queue_remove(t, evt);
|
if (evt->timestamp < pos)
|
||||||
evt->callback(evt);
|
{
|
||||||
events++;
|
timeout_queue_remove(t, evt);
|
||||||
|
evt->callback(evt);
|
||||||
|
events++;
|
||||||
|
}
|
||||||
|
evt = evt->next;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
timeout_queue_unlock(t);
|
|
||||||
return events;
|
return events;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -128,88 +98,25 @@ size_t timeout_queue_get_next_timeout(struct timeout_queue* t, time_t now)
|
|||||||
return seconds;
|
return seconds;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void timeout_queue_insert_locked(struct timeout_queue* t, struct timeout_evt* evt)
|
|
||||||
{
|
|
||||||
/* All events point back to the sentinel.
|
|
||||||
* this means the event is considered schedule (see timeout_evt_is_scheduled),
|
|
||||||
* and it is easy to tell if the event is in the wait queue or not.
|
|
||||||
*/
|
|
||||||
evt->prev = &t->lock;
|
|
||||||
evt->next = NULL;
|
|
||||||
|
|
||||||
// The sentinel next points to the first event in the locked queue
|
|
||||||
// The sentinel prev points to the last evetnt in the locked queue.
|
|
||||||
// NOTE: if prev is != NULL then next also must be != NULL.
|
|
||||||
if (t->lock.prev)
|
|
||||||
{
|
|
||||||
t->lock.prev->next = evt;
|
|
||||||
t->lock.prev = evt;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
t->lock.next = evt;
|
|
||||||
t->lock.prev = evt;
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void timeout_queue_remove_locked(struct timeout_queue* t, struct timeout_evt* evt)
|
|
||||||
{
|
|
||||||
uhub_assert(evt->prev == &t->lock);
|
|
||||||
if (t->lock.next == evt)
|
|
||||||
{
|
|
||||||
t->lock.next = evt->next;
|
|
||||||
if (t->lock.prev == evt)
|
|
||||||
t->lock.prev = evt->next;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
struct timeout_evt *prev, *it;
|
|
||||||
prev = 0;
|
|
||||||
it = t->lock.next;
|
|
||||||
while (it)
|
|
||||||
{
|
|
||||||
prev = it;
|
|
||||||
it = it->next;
|
|
||||||
if (it == evt)
|
|
||||||
{
|
|
||||||
prev->next = it->next;
|
|
||||||
if (!prev->next)
|
|
||||||
t->lock.prev = prev;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
timeout_evt_reset(evt);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
void timeout_queue_insert(struct timeout_queue* t, struct timeout_evt* evt, size_t seconds)
|
void timeout_queue_insert(struct timeout_queue* t, struct timeout_evt* evt, size_t seconds)
|
||||||
{
|
{
|
||||||
struct timeout_evt* first;
|
struct timeout_evt* it, *first;
|
||||||
size_t pos = ((t->last + seconds) % t->max);
|
size_t pos = ((t->last + seconds) % t->max);
|
||||||
evt->timestamp = t->last + seconds;
|
evt->timestamp = t->last + seconds;
|
||||||
evt->next = 0;
|
evt->next = 0;
|
||||||
|
|
||||||
if (timeout_queue_locked(t))
|
|
||||||
{
|
|
||||||
timeout_queue_insert_locked(t, evt);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
first = t->events[pos];
|
first = t->events[pos];
|
||||||
|
|
||||||
if (first)
|
if (first)
|
||||||
{
|
{
|
||||||
uhub_assert(first->timestamp == evt->timestamp);
|
for (it = first; it->next; it = it->next) { }
|
||||||
first->prev->next = evt;
|
it->next = evt;
|
||||||
evt->prev = first->prev;
|
evt->prev = it;
|
||||||
first->prev = evt;
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
t->events[pos] = evt;
|
t->events[pos] = evt;
|
||||||
evt->prev = evt;
|
evt->prev = evt; // point to self.
|
||||||
}
|
}
|
||||||
evt->next = 0;
|
evt->next = 0;
|
||||||
}
|
}
|
||||||
@@ -219,13 +126,6 @@ void timeout_queue_remove(struct timeout_queue* t, struct timeout_evt* evt)
|
|||||||
size_t pos = (evt->timestamp % t->max);
|
size_t pos = (evt->timestamp % t->max);
|
||||||
struct timeout_evt* first = t->events[pos];
|
struct timeout_evt* first = t->events[pos];
|
||||||
|
|
||||||
// Removing a locked event
|
|
||||||
if (evt->prev == &t->lock)
|
|
||||||
{
|
|
||||||
timeout_queue_remove_locked(t, evt);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!first || !evt->prev)
|
if (!first || !evt->prev)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2013, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -43,7 +43,6 @@ struct timeout_queue
|
|||||||
{
|
{
|
||||||
time_t last;
|
time_t last;
|
||||||
size_t max;
|
size_t max;
|
||||||
struct timeout_evt lock;
|
|
||||||
struct timeout_evt** events;
|
struct timeout_evt** events;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -20,4 +20,31 @@
|
|||||||
#include "uhub.h"
|
#include "uhub.h"
|
||||||
#include "network/connection.h"
|
#include "network/connection.h"
|
||||||
|
|
||||||
|
static void timeout_callback(struct timeout_evt* evt)
|
||||||
|
{
|
||||||
|
net_con_callback((struct net_connection*) evt->ptr, NET_EVENT_TIMEOUT);
|
||||||
|
}
|
||||||
|
|
||||||
|
void net_con_set_timeout(struct net_connection* con, int seconds)
|
||||||
|
{
|
||||||
|
if (!con->timeout)
|
||||||
|
{
|
||||||
|
con->timeout = hub_malloc_zero(sizeof(struct timeout_evt));
|
||||||
|
timeout_evt_initialize(con->timeout, timeout_callback, con);
|
||||||
|
timeout_queue_insert(net_backend_get_timeout_queue(), con->timeout, seconds);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
timeout_queue_reschedule(net_backend_get_timeout_queue(), con->timeout, seconds);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void net_con_clear_timeout(struct net_connection* con)
|
||||||
|
{
|
||||||
|
if (con->timeout && timeout_evt_is_scheduled(con->timeout))
|
||||||
|
{
|
||||||
|
timeout_queue_remove(net_backend_get_timeout_queue(), con->timeout);
|
||||||
|
hub_free(con->timeout);
|
||||||
|
con->timeout = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -32,6 +32,8 @@ enum ssl_state
|
|||||||
tls_st_accepting,
|
tls_st_accepting,
|
||||||
tls_st_connecting,
|
tls_st_connecting,
|
||||||
tls_st_connected,
|
tls_st_connected,
|
||||||
|
tls_st_need_read, /* special case of connected */
|
||||||
|
tls_st_need_write, /* special case of connected */
|
||||||
tls_st_disconnecting,
|
tls_st_disconnecting,
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -56,9 +58,8 @@ extern int net_ssl_library_shutdown();
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new SSL context.
|
* Create a new SSL context.
|
||||||
* Specify a TLS version as a string: "1.2" for TLS 1.2.
|
|
||||||
*/
|
*/
|
||||||
extern struct ssl_context_handle* net_ssl_context_create(const char* tls_version, const char* tls_ciphersuite);
|
extern struct ssl_context_handle* net_ssl_context_create();
|
||||||
extern void net_ssl_context_destroy(struct ssl_context_handle* ctx);
|
extern void net_ssl_context_destroy(struct ssl_context_handle* ctx);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -89,26 +90,15 @@ extern ssize_t net_con_ssl_connect(struct net_connection*);
|
|||||||
extern ssize_t net_ssl_send(struct net_connection* con, const void* buf, size_t len);
|
extern ssize_t net_ssl_send(struct net_connection* con, const void* buf, size_t len);
|
||||||
extern ssize_t net_ssl_recv(struct net_connection* con, void* buf, size_t len);
|
extern ssize_t net_ssl_recv(struct net_connection* con, void* buf, size_t len);
|
||||||
|
|
||||||
/**
|
|
||||||
* Update the event mask. Additional events may be requested depending on the
|
|
||||||
* needs of the TLS layer.
|
|
||||||
*
|
|
||||||
* @param con Connection handle.
|
|
||||||
* @param events Event mask (NET_EVENT_*)
|
|
||||||
*/
|
|
||||||
extern void net_ssl_update(struct net_connection* con, int events);
|
|
||||||
|
|
||||||
extern void net_ssl_shutdown(struct net_connection* con);
|
extern void net_ssl_shutdown(struct net_connection* con);
|
||||||
extern void net_ssl_destroy(struct net_connection* con);
|
extern void net_ssl_destroy(struct net_connection* con);
|
||||||
extern void net_ssl_callback(struct net_connection* con, int events);
|
extern void net_ssl_callback(struct net_connection* con, int events);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
extern ssize_t net_con_ssl_handshake(struct net_connection* con, enum net_con_ssl_mode, struct ssl_context_handle* ssl_ctx);
|
extern ssize_t net_con_ssl_handshake(struct net_connection* con, enum net_con_ssl_mode, struct ssl_context_handle* ssl_ctx);
|
||||||
extern int net_con_is_ssl(struct net_connection* con);
|
extern int net_con_is_ssl(struct net_connection* con);
|
||||||
|
|
||||||
extern const char* net_ssl_get_tls_version(struct net_connection* con);
|
|
||||||
extern const char* net_ssl_get_tls_cipher(struct net_connection* con);
|
|
||||||
|
|
||||||
#endif /* SSL_SUPPORT */
|
#endif /* SSL_SUPPORT */
|
||||||
#endif /* HAVE_UHUB_NETWORK_TLS_H */
|
#endif /* HAVE_UHUB_NETWORK_TLS_H */
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -110,7 +110,6 @@ struct plugin_command;
|
|||||||
struct plugin_command_arg_data;
|
struct plugin_command_arg_data;
|
||||||
|
|
||||||
typedef int (*hfunc_send_message)(struct plugin_handle*, struct plugin_user* user, const char* message);
|
typedef int (*hfunc_send_message)(struct plugin_handle*, struct plugin_user* user, const char* message);
|
||||||
typedef int (*hfunc_send_broadcast_message)(struct plugin_handle*, const char* message);
|
|
||||||
typedef int (*hfunc_send_status)(struct plugin_handle*, struct plugin_user* to, int code, const char* message);
|
typedef int (*hfunc_send_status)(struct plugin_handle*, struct plugin_user* to, int code, const char* message);
|
||||||
typedef int (*hfunc_user_disconnect)(struct plugin_handle*, struct plugin_user* user);
|
typedef int (*hfunc_user_disconnect)(struct plugin_handle*, struct plugin_user* user);
|
||||||
typedef int (*hfunc_command_add)(struct plugin_handle*, struct plugin_command_handle*);
|
typedef int (*hfunc_command_add)(struct plugin_handle*, struct plugin_command_handle*);
|
||||||
@@ -119,8 +118,6 @@ typedef int (*hfunc_command_del)(struct plugin_handle*, struct plugin_command_ha
|
|||||||
typedef size_t (*hfunc_command_arg_reset)(struct plugin_handle*, struct plugin_command*);
|
typedef size_t (*hfunc_command_arg_reset)(struct plugin_handle*, struct plugin_command*);
|
||||||
typedef struct plugin_command_arg_data* (*hfunc_command_arg_next)(struct plugin_handle*, struct plugin_command*, enum plugin_command_arg_type);
|
typedef struct plugin_command_arg_data* (*hfunc_command_arg_next)(struct plugin_handle*, struct plugin_command*, enum plugin_command_arg_type);
|
||||||
|
|
||||||
typedef size_t (*hfunc_get_usercount)(struct plugin_handle*);
|
|
||||||
|
|
||||||
typedef char* (*hfunc_get_hub_name)(struct plugin_handle*);
|
typedef char* (*hfunc_get_hub_name)(struct plugin_handle*);
|
||||||
typedef void (*hfunc_set_hub_name)(struct plugin_handle*, const char*);
|
typedef void (*hfunc_set_hub_name)(struct plugin_handle*, const char*);
|
||||||
typedef char* (*hfunc_get_hub_description)(struct plugin_handle*);
|
typedef char* (*hfunc_get_hub_description)(struct plugin_handle*);
|
||||||
@@ -133,14 +130,12 @@ typedef void (*hfunc_set_hub_description)(struct plugin_handle*, const char*);
|
|||||||
struct plugin_hub_funcs
|
struct plugin_hub_funcs
|
||||||
{
|
{
|
||||||
hfunc_send_message send_message;
|
hfunc_send_message send_message;
|
||||||
hfunc_send_broadcast_message send_broadcast_message;
|
|
||||||
hfunc_send_status send_status_message;
|
hfunc_send_status send_status_message;
|
||||||
hfunc_user_disconnect user_disconnect;
|
hfunc_user_disconnect user_disconnect;
|
||||||
hfunc_command_add command_add;
|
hfunc_command_add command_add;
|
||||||
hfunc_command_del command_del;
|
hfunc_command_del command_del;
|
||||||
hfunc_command_arg_reset command_arg_reset;
|
hfunc_command_arg_reset command_arg_reset;
|
||||||
hfunc_command_arg_next command_arg_next;
|
hfunc_command_arg_next command_arg_next;
|
||||||
hfunc_get_usercount get_usercount;
|
|
||||||
hfunc_get_hub_name get_name;
|
hfunc_get_hub_name get_name;
|
||||||
hfunc_set_hub_name set_name;
|
hfunc_set_hub_name set_name;
|
||||||
hfunc_get_hub_description get_description;
|
hfunc_get_hub_description get_description;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2010, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -177,15 +177,16 @@ static void unload_acl(struct acl_data* data)
|
|||||||
static plugin_st get_user(struct plugin_handle* plugin, const char* nickname, struct auth_info* data)
|
static plugin_st get_user(struct plugin_handle* plugin, const char* nickname, struct auth_info* data)
|
||||||
{
|
{
|
||||||
struct acl_data* acl = (struct acl_data*) plugin->ptr;
|
struct acl_data* acl = (struct acl_data*) plugin->ptr;
|
||||||
struct auth_info* info;
|
struct auth_info* info = (struct auth_info*) list_get_first(acl->users);
|
||||||
LIST_FOREACH(struct auth_info*, info, acl->users,
|
while (info)
|
||||||
{
|
{
|
||||||
if (strcasecmp((char*)info->nickname, nickname) == 0)
|
if (strcasecmp((char*)info->nickname, nickname) == 0)
|
||||||
{
|
{
|
||||||
memcpy(data, info, sizeof(struct auth_info));
|
memcpy(data, info, sizeof(struct auth_info));
|
||||||
return st_allow;
|
return st_allow;
|
||||||
}
|
}
|
||||||
});
|
info = (struct auth_info*) list_get_next(acl->users);
|
||||||
|
}
|
||||||
if (acl->exclusive)
|
if (acl->exclusive)
|
||||||
return st_deny;
|
return st_deny;
|
||||||
return st_default;
|
return st_default;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -199,7 +199,7 @@ static plugin_st get_user(struct plugin_handle* plugin, const char* nickname, st
|
|||||||
sqlite3_free(errMsg);
|
sqlite3_free(errMsg);
|
||||||
return st_default;
|
return st_default;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (result.found)
|
if (result.found)
|
||||||
return st_allow;
|
return st_allow;
|
||||||
return st_default;
|
return st_default;
|
||||||
@@ -222,7 +222,7 @@ static plugin_st register_user(struct plugin_handle* plugin, struct auth_info* u
|
|||||||
return st_deny;
|
return st_deny;
|
||||||
}
|
}
|
||||||
return st_allow;
|
return st_allow;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static plugin_st update_user(struct plugin_handle* plugin, struct auth_info* user)
|
static plugin_st update_user(struct plugin_handle* plugin, struct auth_info* user)
|
||||||
@@ -243,7 +243,7 @@ static plugin_st update_user(struct plugin_handle* plugin, struct auth_info* use
|
|||||||
return st_deny;
|
return st_deny;
|
||||||
}
|
}
|
||||||
return st_allow;
|
return st_allow;
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static plugin_st delete_user(struct plugin_handle* plugin, struct auth_info* user)
|
static plugin_st delete_user(struct plugin_handle* plugin, struct auth_info* user)
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -50,7 +50,9 @@ static void history_add(struct plugin_handle* plugin, struct plugin_user* from,
|
|||||||
list_append(data->chat_history, log);
|
list_append(data->chat_history, log);
|
||||||
while (list_size(data->chat_history) > data->history_max)
|
while (list_size(data->chat_history) > data->history_max)
|
||||||
{
|
{
|
||||||
list_remove_first(data->chat_history, hub_free);
|
char* msg = list_get_first(data->chat_history);
|
||||||
|
list_remove(data->chat_history, msg);
|
||||||
|
hub_free(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -77,14 +79,16 @@ static size_t get_messages(struct chat_history_data* data, size_t num, struct cb
|
|||||||
skiplines = total - num;
|
skiplines = total - num;
|
||||||
|
|
||||||
cbuf_append(outbuf, "\n");
|
cbuf_append(outbuf, "\n");
|
||||||
LIST_FOREACH(char*, message, messages,
|
message = (char*) list_get_first(messages);
|
||||||
|
while (message)
|
||||||
{
|
{
|
||||||
if (--skiplines < 0)
|
if (--skiplines < 0)
|
||||||
{
|
{
|
||||||
cbuf_append(outbuf, message);
|
cbuf_append(outbuf, message);
|
||||||
lines++;
|
lines++;
|
||||||
}
|
}
|
||||||
});
|
message = (char*) list_get_next(messages);
|
||||||
|
}
|
||||||
cbuf_append(outbuf, "\n");
|
cbuf_append(outbuf, "\n");
|
||||||
return lines;
|
return lines;
|
||||||
}
|
}
|
||||||
@@ -135,7 +139,7 @@ static int command_history(struct plugin_handle* plugin, struct plugin_user* use
|
|||||||
maxlines = arg->data.integer;
|
maxlines = arg->data.integer;
|
||||||
else
|
else
|
||||||
maxlines = data->history_default;
|
maxlines = data->history_default;
|
||||||
|
|
||||||
buf = cbuf_create(MAX_HISTORY_SIZE);
|
buf = cbuf_create(MAX_HISTORY_SIZE);
|
||||||
cbuf_append_format(buf, "*** %s: Chat History:\n", cmd->prefix);
|
cbuf_append_format(buf, "*** %s: Chat History:\n", cmd->prefix);
|
||||||
get_messages(data, maxlines, buf);
|
get_messages(data, maxlines, buf);
|
||||||
|
|||||||
@@ -1,347 +0,0 @@
|
|||||||
/*
|
|
||||||
* uhub - A tiny ADC p2p connection hub
|
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation; either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "plugin_api/handle.h"
|
|
||||||
#include "plugin_api/command_api.h"
|
|
||||||
#include "util/config_token.h"
|
|
||||||
#include <sqlite3.h>
|
|
||||||
#include "util/memory.h"
|
|
||||||
#include "util/misc.h"
|
|
||||||
#include "util/list.h"
|
|
||||||
#include "util/cbuffer.h"
|
|
||||||
|
|
||||||
#define MAX_HISTORY_SIZE 16384
|
|
||||||
|
|
||||||
struct chat_history_data
|
|
||||||
{
|
|
||||||
size_t history_max; ///<<< "the maximum number of chat messages kept in history."
|
|
||||||
size_t history_default; ///<<< "the default number of chat messages returned if no limit was provided"
|
|
||||||
size_t history_connect; ///<<< "the number of chat messages provided when users connect to the hub."
|
|
||||||
sqlite3* db; ///<<< "The chat history storage database."
|
|
||||||
struct plugin_command_handle* command_history_handle; ///<<< "A handle to the !history command."
|
|
||||||
struct plugin_command_handle* command_historycleanup_handle; ///<<< "A handle to the !historycleanup command."
|
|
||||||
};
|
|
||||||
|
|
||||||
struct chat_history_line
|
|
||||||
{
|
|
||||||
char message[MAX_HISTORY_SIZE];
|
|
||||||
char from[MAX_NICK_LEN];
|
|
||||||
char time[20];
|
|
||||||
};
|
|
||||||
|
|
||||||
static int null_callback(void* ptr, int argc, char **argv, char **colName) { return 0; }
|
|
||||||
|
|
||||||
static const char* sql_escape_string(const char* str)
|
|
||||||
{
|
|
||||||
static char out[1024];
|
|
||||||
size_t i = 0;
|
|
||||||
size_t n = 0;
|
|
||||||
for (; n < strlen(str); n++)
|
|
||||||
{
|
|
||||||
if (str[n] == '\'')
|
|
||||||
out[i++] = '\'';
|
|
||||||
out[i++] = str[n];
|
|
||||||
}
|
|
||||||
out[i++] = '\0';
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int sql_execute(struct chat_history_data* sql, int (*callback)(void* ptr, int argc, char **argv, char **colName), void* ptr, const char* sql_fmt, ...)
|
|
||||||
{
|
|
||||||
va_list args;
|
|
||||||
char query[1024];
|
|
||||||
char* errMsg;
|
|
||||||
int rc;
|
|
||||||
|
|
||||||
va_start(args, sql_fmt);
|
|
||||||
vsnprintf(query, sizeof(query), sql_fmt, args);
|
|
||||||
|
|
||||||
rc = sqlite3_exec(sql->db, query, callback, ptr, &errMsg);
|
|
||||||
if (rc != SQLITE_OK)
|
|
||||||
{
|
|
||||||
sqlite3_free(errMsg);
|
|
||||||
return -rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
rc = sqlite3_changes(sql->db);
|
|
||||||
return rc;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void create_tables(struct plugin_handle* plugin)
|
|
||||||
{
|
|
||||||
const char* table_create = "CREATE TABLE IF NOT EXISTS chat_history"
|
|
||||||
"("
|
|
||||||
"from_nick CHAR,"
|
|
||||||
"message TEXT,"
|
|
||||||
"time TIMESTAMP DEFAULT (DATETIME('NOW'))"
|
|
||||||
");";
|
|
||||||
|
|
||||||
struct chat_history_data* data = (struct chat_history_data*) plugin->ptr;
|
|
||||||
sql_execute(data, null_callback, NULL, table_create);
|
|
||||||
}
|
|
||||||
/**
|
|
||||||
* Add a chat message to history.
|
|
||||||
*/
|
|
||||||
static void history_add(struct plugin_handle* plugin, struct plugin_user* from, const char* message, int flags)
|
|
||||||
{
|
|
||||||
struct chat_history_data* data = (struct chat_history_data*) plugin->ptr;
|
|
||||||
char* history_line = strdup(sql_escape_string(message));
|
|
||||||
char* history_nick = strdup(sql_escape_string(from->nick));
|
|
||||||
|
|
||||||
sql_execute(data, null_callback, NULL, "INSERT INTO chat_history (from_nick, message) VALUES('%s', '%s');DELETE FROM chat_history WHERE time <= (SELECT time FROM chat_history ORDER BY time DESC LIMIT %d,1);", history_nick, history_line, data->history_max);
|
|
||||||
|
|
||||||
hub_free(history_line);
|
|
||||||
hub_free(history_nick);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Obtain messages from the chat history as a linked list
|
|
||||||
*/
|
|
||||||
|
|
||||||
static int get_messages_callback(void* ptr, int argc, char **argv, char **colName)
|
|
||||||
{
|
|
||||||
struct linked_list* messages = (struct linked_list*) ptr;
|
|
||||||
struct chat_history_line* line = hub_malloc(sizeof(struct chat_history_line));
|
|
||||||
int i = 0;
|
|
||||||
|
|
||||||
memset(line, 0, sizeof(struct chat_history_line));
|
|
||||||
|
|
||||||
for (; i < argc; i++) {
|
|
||||||
if (strcmp(colName[i], "from_nick") == 0)
|
|
||||||
strncpy(line->from, argv[i], MAX_NICK_LEN);
|
|
||||||
else if (strcmp(colName[i], "message") == 0)
|
|
||||||
strncpy(line->message, argv[i], MAX_HISTORY_SIZE);
|
|
||||||
else if (strcmp(colName[i], "time") == 0)
|
|
||||||
strncpy(line->time, argv[i], 20);
|
|
||||||
}
|
|
||||||
|
|
||||||
list_append(messages, line);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
void user_login(struct plugin_handle* plugin, struct plugin_user* user)
|
|
||||||
{
|
|
||||||
struct chat_history_data* data = (struct chat_history_data*) plugin->ptr;
|
|
||||||
struct cbuffer* buf = NULL;
|
|
||||||
struct linked_list* found = (struct linked_list*) list_create();
|
|
||||||
|
|
||||||
sql_execute(data, get_messages_callback, found, "SELECT * FROM chat_history ORDER BY time DESC LIMIT 0,%d;", (int) data->history_connect);
|
|
||||||
|
|
||||||
if (data->history_connect > 0 && list_size(found) > 0)
|
|
||||||
{
|
|
||||||
buf = cbuf_create(MAX_HISTORY_SIZE);
|
|
||||||
cbuf_append(buf, "Chat history:\n\n");
|
|
||||||
struct chat_history_line* history_line;
|
|
||||||
history_line = (struct chat_history_line*) list_get_last(found);
|
|
||||||
while (history_line)
|
|
||||||
{
|
|
||||||
cbuf_append_format(buf, "[%s] <%s> %s\n", history_line->time, history_line->from, history_line->message);
|
|
||||||
list_remove(found, history_line);
|
|
||||||
hub_free(history_line);
|
|
||||||
history_line = (struct chat_history_line*) list_get_last(found);
|
|
||||||
}
|
|
||||||
plugin->hub.send_message(plugin, user, cbuf_get(buf));
|
|
||||||
cbuf_destroy(buf);
|
|
||||||
}
|
|
||||||
list_clear(found, &hub_free);
|
|
||||||
list_destroy(found);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The callback function for handling the !history command.
|
|
||||||
*/
|
|
||||||
static int command_history(struct plugin_handle* plugin, struct plugin_user* user, struct plugin_command* cmd)
|
|
||||||
{
|
|
||||||
struct chat_history_data* data = (struct chat_history_data*) plugin->ptr;
|
|
||||||
struct cbuffer* buf = cbuf_create(MAX_HISTORY_SIZE);
|
|
||||||
struct linked_list* found = (struct linked_list*) list_create();
|
|
||||||
struct plugin_command_arg_data* arg = plugin->hub.command_arg_next(plugin, cmd, plugin_cmd_arg_type_integer);
|
|
||||||
int maxlines;
|
|
||||||
|
|
||||||
if (arg)
|
|
||||||
maxlines = arg->data.integer;
|
|
||||||
else
|
|
||||||
maxlines = data->history_default;
|
|
||||||
|
|
||||||
sql_execute(data, get_messages_callback, found, "SELECT * FROM chat_history ORDER BY time DESC LIMIT 0,%d;", maxlines);
|
|
||||||
|
|
||||||
size_t linecount = list_size(found);
|
|
||||||
|
|
||||||
if (linecount > 0)
|
|
||||||
{
|
|
||||||
cbuf_append_format(buf, "*** %s: Chat History:\n\n", cmd->prefix);
|
|
||||||
struct chat_history_line* history_line;
|
|
||||||
history_line = (struct chat_history_line*) list_get_last(found);
|
|
||||||
while (history_line)
|
|
||||||
{
|
|
||||||
cbuf_append_format(buf, "[%s] <%s> %s\n", history_line->time, history_line->from, history_line->message);
|
|
||||||
list_remove(found, history_line);
|
|
||||||
hub_free(history_line);
|
|
||||||
history_line = (struct chat_history_line*) list_get_last(found);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
cbuf_append_format(buf, "*** %s: No messages found.", cmd->prefix);
|
|
||||||
}
|
|
||||||
|
|
||||||
plugin->hub.send_message(plugin, user, cbuf_get(buf));
|
|
||||||
cbuf_destroy(buf);
|
|
||||||
list_clear(found, &hub_free);
|
|
||||||
list_destroy(found);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static int command_historycleanup(struct plugin_handle* plugin, struct plugin_user* user, struct plugin_command* cmd)
|
|
||||||
{
|
|
||||||
struct chat_history_data* data = (struct chat_history_data*) plugin->ptr;
|
|
||||||
struct cbuffer* buf = cbuf_create(128);
|
|
||||||
int rc = 0;
|
|
||||||
|
|
||||||
rc = sql_execute(data, null_callback, NULL, "DELETE FROM chat_history;");
|
|
||||||
|
|
||||||
if (!rc)
|
|
||||||
cbuf_append_format(buf, "*** %s: Unable to clean chat history table.", cmd->prefix);
|
|
||||||
else
|
|
||||||
cbuf_append_format(buf, "*** %s: Cleaned chat history table.", cmd->prefix);
|
|
||||||
|
|
||||||
plugin->hub.send_message(plugin, user, cbuf_get(buf));
|
|
||||||
cbuf_destroy(buf);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void set_error_message(struct plugin_handle* plugin, const char* msg)
|
|
||||||
{
|
|
||||||
plugin->error_msg = msg;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct chat_history_data* parse_config(const char* line, struct plugin_handle* plugin)
|
|
||||||
{
|
|
||||||
struct chat_history_data* data = (struct chat_history_data*) hub_malloc_zero(sizeof(struct chat_history_data));
|
|
||||||
struct cfg_tokens* tokens = cfg_tokenize(line);
|
|
||||||
char* token = cfg_token_get_first(tokens);
|
|
||||||
|
|
||||||
uhub_assert(data != NULL);
|
|
||||||
|
|
||||||
data->history_max = 200;
|
|
||||||
data->history_default = 25;
|
|
||||||
data->history_connect = 5;
|
|
||||||
|
|
||||||
while (token)
|
|
||||||
{
|
|
||||||
struct cfg_settings* setting = cfg_settings_split(token);
|
|
||||||
|
|
||||||
if (!setting)
|
|
||||||
{
|
|
||||||
set_error_message(plugin, "Unable to parse startup parameters");
|
|
||||||
cfg_tokens_free(tokens);
|
|
||||||
hub_free(data);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (strcmp(cfg_settings_get_key(setting), "file") == 0)
|
|
||||||
{
|
|
||||||
if (!data->db)
|
|
||||||
{
|
|
||||||
if (sqlite3_open(cfg_settings_get_value(setting), &data->db))
|
|
||||||
{
|
|
||||||
cfg_tokens_free(tokens);
|
|
||||||
cfg_settings_free(setting);
|
|
||||||
hub_free(data);
|
|
||||||
set_error_message(plugin, "Unable to open database file");
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (strcmp(cfg_settings_get_key(setting), "history_max") == 0)
|
|
||||||
{
|
|
||||||
data->history_max = (size_t) uhub_atoi(cfg_settings_get_value(setting));
|
|
||||||
}
|
|
||||||
else if (strcmp(cfg_settings_get_key(setting), "history_default") == 0)
|
|
||||||
{
|
|
||||||
data->history_default = (size_t) uhub_atoi(cfg_settings_get_value(setting));
|
|
||||||
}
|
|
||||||
else if (strcmp(cfg_settings_get_key(setting), "history_connect") == 0)
|
|
||||||
{
|
|
||||||
data->history_connect = (size_t) uhub_atoi(cfg_settings_get_value(setting));
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
set_error_message(plugin, "Unknown startup parameters given");
|
|
||||||
cfg_tokens_free(tokens);
|
|
||||||
cfg_settings_free(setting);
|
|
||||||
hub_free(data);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
cfg_settings_free(setting);
|
|
||||||
token = cfg_token_get_next(tokens);
|
|
||||||
}
|
|
||||||
cfg_tokens_free(tokens);
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
int plugin_register(struct plugin_handle* plugin, const char* config)
|
|
||||||
{
|
|
||||||
struct chat_history_data* data;
|
|
||||||
PLUGIN_INITIALIZE(plugin, "SQLite chat history plugin", "1.0", "Provide a global chat history log.");
|
|
||||||
|
|
||||||
plugin->funcs.on_user_chat_message = history_add;
|
|
||||||
plugin->funcs.on_user_login = user_login;
|
|
||||||
data = parse_config(config, plugin);
|
|
||||||
|
|
||||||
if (!data)
|
|
||||||
return -1;
|
|
||||||
|
|
||||||
plugin->ptr = data;
|
|
||||||
|
|
||||||
create_tables(plugin);
|
|
||||||
|
|
||||||
data->command_history_handle = (struct plugin_command_handle*) hub_malloc(sizeof(struct plugin_command_handle));
|
|
||||||
PLUGIN_COMMAND_INITIALIZE(data->command_history_handle, plugin, "history", "?N", auth_cred_guest, &command_history, "Show chat message history.");
|
|
||||||
plugin->hub.command_add(plugin, data->command_history_handle);
|
|
||||||
|
|
||||||
data->command_historycleanup_handle = (struct plugin_command_handle*) hub_malloc(sizeof(struct plugin_command_handle));
|
|
||||||
PLUGIN_COMMAND_INITIALIZE(data->command_historycleanup_handle, plugin, "historycleanup", "", auth_cred_admin, &command_historycleanup, "Clean chat message history.");
|
|
||||||
plugin->hub.command_add(plugin, data->command_historycleanup_handle);
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int plugin_unregister(struct plugin_handle* plugin)
|
|
||||||
{
|
|
||||||
struct chat_history_data* data = (struct chat_history_data*) plugin->ptr;
|
|
||||||
|
|
||||||
if (data)
|
|
||||||
{
|
|
||||||
sqlite3_close(data->db);
|
|
||||||
|
|
||||||
plugin->hub.command_del(plugin, data->command_history_handle);
|
|
||||||
plugin->hub.command_del(plugin, data->command_historycleanup_handle);
|
|
||||||
hub_free(data->command_history_handle);
|
|
||||||
hub_free(data->command_historycleanup_handle);
|
|
||||||
hub_free(data);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,171 +0,0 @@
|
|||||||
/*
|
|
||||||
* uhub - A tiny ADC p2p connection hub
|
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
|
||||||
*
|
|
||||||
* This program is free software; you can redistribute it and/or modify
|
|
||||||
* it under the terms of the GNU General Public License as published by
|
|
||||||
* the Free Software Foundation; either version 3 of the License, or
|
|
||||||
* (at your option) any later version.
|
|
||||||
*
|
|
||||||
* This program is distributed in the hope that it will be useful,
|
|
||||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
||||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
||||||
* GNU General Public License for more details.
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License
|
|
||||||
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "plugin_api/handle.h"
|
|
||||||
#include "util/memory.h"
|
|
||||||
|
|
||||||
struct user_info
|
|
||||||
{
|
|
||||||
sid_t sid;
|
|
||||||
int warnings;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct chat_restrictions_data
|
|
||||||
{
|
|
||||||
size_t num_users; // number of users tracked.
|
|
||||||
size_t max_users; // max users (hard limit max 1M users due to limitations in the SID (20 bits)).
|
|
||||||
struct user_info* users; // array of max_users
|
|
||||||
|
|
||||||
enum auth_credentials allow_privchat; // minimum credentials to allow using private chat
|
|
||||||
enum auth_credentials allow_op_contact; // minimum credentials to allow private chat to operators (including super and admins).
|
|
||||||
enum auth_credentials allow_mainchat; // minimum credentials to allow using main chat
|
|
||||||
};
|
|
||||||
|
|
||||||
static struct chat_data* parse_config(struct plugin_handle* plugin, const char* line)
|
|
||||||
{
|
|
||||||
struct chat_data* data = (struct chat_data*) hub_malloc(sizeof(struct chat_data));
|
|
||||||
struct cfg_tokens* tokens = cfg_tokenize(line);
|
|
||||||
char* token = cfg_token_get_first(tokens);
|
|
||||||
|
|
||||||
// defaults
|
|
||||||
data->num_users = 0;
|
|
||||||
data->max_users = 512;
|
|
||||||
data->users = hub_malloc_zero(sizeof(struct user_info) * data->max_users);
|
|
||||||
data->allow_mainchat = auth_cred_guest;
|
|
||||||
data->allow_op_contact = auth_cred_guest;
|
|
||||||
data->allow_privchat = auth_cred_guest;
|
|
||||||
|
|
||||||
while (token)
|
|
||||||
{
|
|
||||||
struct cfg_settings* setting = cfg_settings_split(token);
|
|
||||||
|
|
||||||
if (!setting)
|
|
||||||
{
|
|
||||||
set_error_message(plugin, "Unable to parse startup parameters");
|
|
||||||
cfg_tokens_free(tokens);
|
|
||||||
hub_free(data);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (strcmp(cfg_settings_get_key(setting), "allow_privchat") == 0)
|
|
||||||
{
|
|
||||||
if (!string_to_boolean(cfg_settings_get_value(setting), &data->allow_privchat))
|
|
||||||
data->allow_privchat = 0;
|
|
||||||
}
|
|
||||||
else if (strcmp(cfg_settings_get_key(setting), "minimum_access") == 0)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
set_error_message(plugin, "Unknown startup parameters given");
|
|
||||||
cfg_tokens_free(tokens);
|
|
||||||
cfg_settings_free(setting);
|
|
||||||
hub_free(data);
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
cfg_settings_free(setting);
|
|
||||||
token = cfg_token_get_next(tokens);
|
|
||||||
}
|
|
||||||
cfg_tokens_free(tokens);
|
|
||||||
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
static struct user_info* get_user_info(struct chat_data* data, sid_t sid)
|
|
||||||
{
|
|
||||||
struct user_info* u;
|
|
||||||
|
|
||||||
// resize buffer if needed.
|
|
||||||
if (sid >= data->max_users)
|
|
||||||
{
|
|
||||||
u = hub_malloc_zero(sizeof(struct user_info) * (sid + 1));
|
|
||||||
memcpy(u, data->users, data->max_users);
|
|
||||||
hub_free(data->users);
|
|
||||||
data->users = u;
|
|
||||||
data->max_users = sid;
|
|
||||||
u = NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
u = &data->users[sid];
|
|
||||||
|
|
||||||
// reset counters if the user was not previously known.
|
|
||||||
if (!u->sid)
|
|
||||||
{
|
|
||||||
u->sid = sid;
|
|
||||||
u->warnings = 0;
|
|
||||||
data->num_users++;
|
|
||||||
}
|
|
||||||
return u;
|
|
||||||
}
|
|
||||||
|
|
||||||
static void on_user_login(struct plugin_handle* plugin, struct plugin_user* user)
|
|
||||||
{
|
|
||||||
struct chat_data* data = (struct chat_data*) plugin->ptr;
|
|
||||||
/*struct user_info* info = */
|
|
||||||
get_user_info(data, user->sid);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void on_user_logout(struct plugin_handle* plugin, struct plugin_user* user, const char* reason)
|
|
||||||
{
|
|
||||||
struct chat_data* data = (struct chat_data*) plugin->ptr;
|
|
||||||
struct user_info* info = get_user_info(data, user->sid);
|
|
||||||
if (info->sid)
|
|
||||||
data->num_users--;
|
|
||||||
info->warnings = 0;
|
|
||||||
info->sid = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
plugin_st on_chat_msg(struct plugin_handle* plugin, struct plugin_user* from, const char* message)
|
|
||||||
{
|
|
||||||
struct chat_data* data = (struct chat_data*) plugin->ptr;
|
|
||||||
if (from->credentials >=
|
|
||||||
return st_default;
|
|
||||||
}
|
|
||||||
|
|
||||||
plugin_st on_private_msg(struct plugin_handle* plugin, struct plugin_user* from, struct plugin_user* to, const char* message)
|
|
||||||
{
|
|
||||||
return st_default;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int plugin_register(struct plugin_handle* plugin, const char* config)
|
|
||||||
{
|
|
||||||
PLUGIN_INITIALIZE(plugin, "Privileged chat hub", "1.0", "Only registered users can send messages on the main chat.");
|
|
||||||
plugin->ptr = cip_initialize();
|
|
||||||
|
|
||||||
plugin->funcs.on_user_login = on_user_login;
|
|
||||||
plugin->funcs.on_user_logout = on_user_logout;
|
|
||||||
plugin->funcs.on_chat_msg = on_chat_msg;
|
|
||||||
plugin->funcs.on_private_msg = on_private_msg;
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
int plugin_unregister(struct plugin_handle* plugin)
|
|
||||||
{
|
|
||||||
struct chat_data* data = (struct chat_data*) plugin->ptr;
|
|
||||||
if (data)
|
|
||||||
{
|
|
||||||
hub_free(data->users);
|
|
||||||
hub_free(data);
|
|
||||||
}
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2011, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -25,7 +25,7 @@
|
|||||||
struct topic_plugin_data
|
struct topic_plugin_data
|
||||||
{
|
{
|
||||||
struct plugin_command_handle* topic;
|
struct plugin_command_handle* topic;
|
||||||
struct plugin_command_handle* resettopic;
|
struct plugin_command_handle* cleartopic;
|
||||||
struct plugin_command_handle* showtopic;
|
struct plugin_command_handle* showtopic;
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -33,20 +33,19 @@ static int command_topic_handler(struct plugin_handle* plugin, struct plugin_use
|
|||||||
{
|
{
|
||||||
struct cbuffer* buf = cbuf_create(128);
|
struct cbuffer* buf = cbuf_create(128);
|
||||||
struct plugin_command_arg_data* arg = plugin->hub.command_arg_next(plugin, cmd, plugin_cmd_arg_type_string);
|
struct plugin_command_arg_data* arg = plugin->hub.command_arg_next(plugin, cmd, plugin_cmd_arg_type_string);
|
||||||
char* topic = arg ? arg->data.string : "";
|
|
||||||
|
|
||||||
plugin->hub.set_description(plugin, topic);
|
plugin->hub.set_description(plugin, arg ? arg->data.string : NULL);
|
||||||
cbuf_append_format(buf, "*** %s: Topic set to \"%s\"", cmd->prefix, topic);
|
cbuf_append_format(buf, "*** %s: Topic set to \"%s\"", cmd->prefix, arg->data.string);
|
||||||
plugin->hub.send_message(plugin, user, cbuf_get(buf));
|
plugin->hub.send_message(plugin, user, cbuf_get(buf));
|
||||||
cbuf_destroy(buf);
|
cbuf_destroy(buf);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int command_resettopic_handler(struct plugin_handle* plugin, struct plugin_user* user, struct plugin_command* cmd)
|
static int command_cleartopic_handler(struct plugin_handle* plugin, struct plugin_user* user, struct plugin_command* cmd)
|
||||||
{
|
{
|
||||||
struct cbuffer* buf = cbuf_create(128);
|
struct cbuffer* buf = cbuf_create(128);
|
||||||
plugin->hub.set_description(plugin, NULL);
|
plugin->hub.set_description(plugin, NULL);
|
||||||
cbuf_append_format(buf, "*** %s: Topic reset.", cmd->prefix);
|
cbuf_append_format(buf, "*** %s: Topic cleared.", cmd->prefix);
|
||||||
plugin->hub.send_message(plugin, user, cbuf_get(buf));
|
plugin->hub.send_message(plugin, user, cbuf_get(buf));
|
||||||
cbuf_destroy(buf);
|
cbuf_destroy(buf);
|
||||||
return 0;
|
return 0;
|
||||||
@@ -68,17 +67,17 @@ int plugin_register(struct plugin_handle* plugin, const char* config)
|
|||||||
struct topic_plugin_data* data = (struct topic_plugin_data*) hub_malloc(sizeof(struct topic_plugin_data));
|
struct topic_plugin_data* data = (struct topic_plugin_data*) hub_malloc(sizeof(struct topic_plugin_data));
|
||||||
|
|
||||||
data->topic = (struct plugin_command_handle*) hub_malloc_zero(sizeof(struct plugin_command_handle));
|
data->topic = (struct plugin_command_handle*) hub_malloc_zero(sizeof(struct plugin_command_handle));
|
||||||
data->resettopic = (struct plugin_command_handle*) hub_malloc_zero(sizeof(struct plugin_command_handle));
|
data->cleartopic = (struct plugin_command_handle*) hub_malloc_zero(sizeof(struct plugin_command_handle));
|
||||||
data->showtopic = (struct plugin_command_handle*) hub_malloc_zero(sizeof(struct plugin_command_handle));
|
data->showtopic = (struct plugin_command_handle*) hub_malloc_zero(sizeof(struct plugin_command_handle));
|
||||||
|
|
||||||
PLUGIN_INITIALIZE(plugin, "Topic plugin", "1.0", "Add commands for changing the hub topic (description)");
|
PLUGIN_INITIALIZE(plugin, "Topic plugin", "1.0", "Add commands for changing the hub topic (description)");
|
||||||
|
|
||||||
PLUGIN_COMMAND_INITIALIZE(data->topic, (void*) data, "topic", "+m", auth_cred_operator, command_topic_handler, "Set new topic");
|
PLUGIN_COMMAND_INITIALIZE(data->topic, (void*) data, "topic", "+m", auth_cred_operator, command_topic_handler, "Set new topic");
|
||||||
PLUGIN_COMMAND_INITIALIZE(data->resettopic, (void*) data, "resettopic", "", auth_cred_operator, command_resettopic_handler, "Set topic to default");
|
PLUGIN_COMMAND_INITIALIZE(data->cleartopic, (void*) data, "cleartopic", "", auth_cred_operator, command_cleartopic_handler, "Clear the current topic");
|
||||||
PLUGIN_COMMAND_INITIALIZE(data->showtopic, (void*) data, "showtopic", "", auth_cred_guest, command_showtopic_handler, "Shows the current topic");
|
PLUGIN_COMMAND_INITIALIZE(data->showtopic, (void*) data, "showtopic", "", auth_cred_guest, command_showtopic_handler, "Shows the current topic");
|
||||||
|
|
||||||
plugin->hub.command_add(plugin, data->topic);
|
plugin->hub.command_add(plugin, data->topic);
|
||||||
plugin->hub.command_add(plugin, data->resettopic);
|
plugin->hub.command_add(plugin, data->cleartopic);
|
||||||
plugin->hub.command_add(plugin, data->showtopic);
|
plugin->hub.command_add(plugin, data->showtopic);
|
||||||
plugin->ptr = data;
|
plugin->ptr = data;
|
||||||
|
|
||||||
@@ -91,10 +90,10 @@ int plugin_unregister(struct plugin_handle* plugin)
|
|||||||
struct topic_plugin_data* data = (struct topic_plugin_data*) plugin->ptr;
|
struct topic_plugin_data* data = (struct topic_plugin_data*) plugin->ptr;
|
||||||
|
|
||||||
plugin->hub.command_del(plugin, data->topic);
|
plugin->hub.command_del(plugin, data->topic);
|
||||||
plugin->hub.command_del(plugin, data->resettopic);
|
plugin->hub.command_del(plugin, data->cleartopic);
|
||||||
plugin->hub.command_del(plugin, data->showtopic);
|
plugin->hub.command_del(plugin, data->showtopic);
|
||||||
hub_free(data->topic);
|
hub_free(data->topic);
|
||||||
hub_free(data->resettopic);
|
hub_free(data->cleartopic);
|
||||||
hub_free(data->showtopic);
|
hub_free(data->showtopic);
|
||||||
hub_free(data);
|
hub_free(data);
|
||||||
plugin->ptr = NULL;
|
plugin->ptr = NULL;
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -22,6 +22,10 @@
|
|||||||
|
|
||||||
#define _FILE_OFFSET_BITS 64
|
#define _FILE_OFFSET_BITS 64
|
||||||
|
|
||||||
|
#ifndef _GNU_SOURCE
|
||||||
|
#define _GNU_SOURCE
|
||||||
|
#endif
|
||||||
|
|
||||||
#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__bsdi__) || defined(__DragonFly__) || (defined(__APPLE__) && defined(__MACH__)) || defined(__FreeBSD_kernel__)
|
#if defined(__FreeBSD__) || defined(__NetBSD__) || defined(__OpenBSD__) || defined(__bsdi__) || defined(__DragonFly__) || (defined(__APPLE__) && defined(__MACH__)) || defined(__FreeBSD_kernel__)
|
||||||
#define BSD_LIKE
|
#define BSD_LIKE
|
||||||
#endif
|
#endif
|
||||||
@@ -34,14 +38,10 @@
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if defined(__CYGWIN__) || defined(__MINGW32__)
|
#if defined(__CYGWIN__) || defined(__MINGW32__)
|
||||||
|
#define HAVE_SSIZE_T
|
||||||
#define NEED_GETOPT
|
#define NEED_GETOPT
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#cmakedefine HAVE_SYS_TYPES_H
|
|
||||||
#ifdef HAVE_SYS_TYPES_H
|
|
||||||
#include <sys/types.h>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef WINSOCK
|
#ifdef WINSOCK
|
||||||
#ifndef FD_SETSIZE
|
#ifndef FD_SETSIZE
|
||||||
#define FD_SETSIZE 4096
|
#define FD_SETSIZE 4096
|
||||||
@@ -49,6 +49,7 @@
|
|||||||
#include <winsock2.h>
|
#include <winsock2.h>
|
||||||
#include <ws2tcpip.h>
|
#include <ws2tcpip.h>
|
||||||
#else
|
#else
|
||||||
|
#include <sys/types.h>
|
||||||
#include <sys/time.h>
|
#include <sys/time.h>
|
||||||
#include <arpa/inet.h>
|
#include <arpa/inet.h>
|
||||||
#include <netdb.h>
|
#include <netdb.h>
|
||||||
@@ -61,8 +62,7 @@
|
|||||||
#include <errno.h>
|
#include <errno.h>
|
||||||
#include <fcntl.h>
|
#include <fcntl.h>
|
||||||
|
|
||||||
#cmakedefine HAVE_STDINT_H
|
#ifndef __sun__
|
||||||
#ifdef HAVE_STDINT_H
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -79,14 +79,16 @@
|
|||||||
#include <grp.h>
|
#include <grp.h>
|
||||||
#include <pwd.h>
|
#include <pwd.h>
|
||||||
#include <sys/resource.h>
|
#include <sys/resource.h>
|
||||||
|
#define HAVE_STRNDUP
|
||||||
#define HAVE_DLOPEN
|
#define HAVE_DLOPEN
|
||||||
#define HAVE_GETOPT
|
#define HAVE_GETOPT
|
||||||
|
#define HAVE_SSIZE_T
|
||||||
#include <dlfcn.h>
|
#include <dlfcn.h>
|
||||||
|
#ifndef __HAIKU__
|
||||||
|
#define HAVE_MEMMEM
|
||||||
|
#endif
|
||||||
#define HAVE_GETRLIMIT
|
#define HAVE_GETRLIMIT
|
||||||
#endif
|
#endif
|
||||||
#cmakedefine HAVE_SSIZE_T
|
|
||||||
#cmakedefine HAVE_STRNDUP
|
|
||||||
#cmakedefine HAVE_MEMMEM
|
|
||||||
|
|
||||||
/* printf support for size_t and uint64_t */
|
/* printf support for size_t and uint64_t */
|
||||||
#if defined(WIN32)
|
#if defined(WIN32)
|
||||||
@@ -101,8 +103,6 @@
|
|||||||
#ifdef SSL_USE_OPENSSL
|
#ifdef SSL_USE_OPENSSL
|
||||||
#include <openssl/ssl.h>
|
#include <openssl/ssl.h>
|
||||||
#include <openssl/err.h>
|
#include <openssl/err.h>
|
||||||
#include <openssl/engine.h>
|
|
||||||
#include <openssl/conf.h>
|
|
||||||
#endif /* SSL_USE_OPENSSL */
|
#endif /* SSL_USE_OPENSSL */
|
||||||
#ifdef SSL_USE_GNUTLS
|
#ifdef SSL_USE_GNUTLS
|
||||||
#include <gnutls/gnutls.h>
|
#include <gnutls/gnutls.h>
|
||||||
@@ -136,6 +136,11 @@
|
|||||||
#include <sys/select.h>
|
#include <sys/select.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#if ( defined(BSD_LIKE) && !defined(__FreeBSD_kernel__) ) || defined(__sun__)
|
||||||
|
#undef HAVE_STRNDUP
|
||||||
|
#undef HAVE_MEMMEM
|
||||||
|
#endif
|
||||||
|
|
||||||
#ifdef HAVE_GETOPT
|
#ifdef HAVE_GETOPT
|
||||||
#include <getopt.h>
|
#include <getopt.h>
|
||||||
#endif
|
#endif
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -27,18 +27,6 @@
|
|||||||
|
|
||||||
// #define ADCC_DEBUG
|
// #define ADCC_DEBUG
|
||||||
// #define ADC_CLIENT_DEBUG_PROTO
|
// #define ADC_CLIENT_DEBUG_PROTO
|
||||||
|
|
||||||
struct ADC_client_global
|
|
||||||
{
|
|
||||||
size_t references;
|
|
||||||
#ifdef SSL_SUPPORT
|
|
||||||
struct ssl_context_handle* ctx;
|
|
||||||
#endif
|
|
||||||
};
|
|
||||||
|
|
||||||
static struct ADC_client_global* g_adc_client = NULL;
|
|
||||||
|
|
||||||
|
|
||||||
enum ADC_client_state
|
enum ADC_client_state
|
||||||
{
|
{
|
||||||
ps_none, /* Not connected */
|
ps_none, /* Not connected */
|
||||||
@@ -190,16 +178,7 @@ static void event_callback(struct net_connection* con, int events, void *arg)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (events == NET_EVENT_ERROR)
|
ADC_client_on_connected_ssl(client);
|
||||||
{
|
|
||||||
ADC_client_on_disconnected(client);
|
|
||||||
client->callback(client, ADC_CLIENT_DISCONNECTED, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
ADC_client_on_connected_ssl(client);
|
|
||||||
}
|
|
||||||
break;
|
break;
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -387,7 +366,7 @@ static int ADC_client_on_recv_line(struct ADC_client* client, const char* line,
|
|||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
break;
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -529,8 +508,7 @@ void ADC_client_send_info(struct ADC_client* client)
|
|||||||
adc_msg_add_named_argument_string(client->info, ADC_INF_FLAG_DESCRIPTION, client->desc);
|
adc_msg_add_named_argument_string(client->info, ADC_INF_FLAG_DESCRIPTION, client->desc);
|
||||||
}
|
}
|
||||||
|
|
||||||
adc_msg_add_named_argument_string(client->info, ADC_INF_FLAG_USER_AGENT_PRODUCT, PRODUCT);
|
adc_msg_add_named_argument_string(client->info, ADC_INF_FLAG_USER_AGENT, PRODUCT " " VERSION);
|
||||||
adc_msg_add_named_argument_string(client->info, ADC_INF_FLAG_USER_AGENT_VERSION, VERSION);
|
|
||||||
adc_msg_add_named_argument_int(client->info, ADC_INF_FLAG_UPLOAD_SLOTS, 0);
|
adc_msg_add_named_argument_int(client->info, ADC_INF_FLAG_UPLOAD_SLOTS, 0);
|
||||||
adc_msg_add_named_argument_int(client->info, ADC_INF_FLAG_SHARED_SIZE, 0);
|
adc_msg_add_named_argument_int(client->info, ADC_INF_FLAG_SHARED_SIZE, 0);
|
||||||
adc_msg_add_named_argument_int(client->info, ADC_INF_FLAG_SHARED_FILES, 0);
|
adc_msg_add_named_argument_int(client->info, ADC_INF_FLAG_SHARED_FILES, 0);
|
||||||
@@ -562,17 +540,6 @@ struct ADC_client* ADC_client_create(const char* nickname, const char* descripti
|
|||||||
client->recv_queue = ioq_recv_create();
|
client->recv_queue = ioq_recv_create();
|
||||||
|
|
||||||
client->ptr = ptr;
|
client->ptr = ptr;
|
||||||
|
|
||||||
if (!g_adc_client)
|
|
||||||
{
|
|
||||||
g_adc_client = (struct ADC_client_global*) hub_malloc_zero(sizeof(struct ADC_client_global));
|
|
||||||
#ifdef SSL_SUPPORT
|
|
||||||
g_adc_client->ctx = net_ssl_context_create("1.2", "HIGH");
|
|
||||||
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
g_adc_client->references++;
|
|
||||||
|
|
||||||
return client;
|
return client;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -588,20 +555,6 @@ void ADC_client_destroy(struct ADC_client* client)
|
|||||||
hub_free(client->desc);
|
hub_free(client->desc);
|
||||||
hub_free(client->address.hostname);
|
hub_free(client->address.hostname);
|
||||||
hub_free(client);
|
hub_free(client);
|
||||||
|
|
||||||
if (g_adc_client && g_adc_client->references > 0)
|
|
||||||
{
|
|
||||||
g_adc_client->references--;
|
|
||||||
if (!g_adc_client->references)
|
|
||||||
{
|
|
||||||
#ifdef SSL_SUPPORT
|
|
||||||
net_ssl_context_destroy(g_adc_client->ctx);
|
|
||||||
g_adc_client->ctx = NULL;
|
|
||||||
#endif
|
|
||||||
hub_free(g_adc_client);
|
|
||||||
g_adc_client = NULL;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void connect_callback(struct net_connect_handle* handle, enum net_connect_status status, struct net_connection* con, void* ptr)
|
static void connect_callback(struct net_connect_handle* handle, enum net_connect_status status, struct net_connection* con, void* ptr)
|
||||||
@@ -647,17 +600,6 @@ int ADC_client_connect(struct ADC_client* client, const char* address)
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ADC_client_send_handshake(struct ADC_client* client)
|
|
||||||
{
|
|
||||||
ADC_TRACE;
|
|
||||||
struct adc_message* handshake = adc_msg_create(ADC_HANDSHAKE);
|
|
||||||
|
|
||||||
client->callback(client, ADC_CLIENT_CONNECTED, 0);
|
|
||||||
net_con_update(client->con, NET_EVENT_READ);
|
|
||||||
ADC_client_send(client, handshake);
|
|
||||||
ADC_client_set_state(client, ps_protocol);
|
|
||||||
adc_msg_free(handshake);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void ADC_client_on_connected(struct ADC_client* client)
|
static void ADC_client_on_connected(struct ADC_client* client)
|
||||||
{
|
{
|
||||||
@@ -668,27 +610,32 @@ static void ADC_client_on_connected(struct ADC_client* client)
|
|||||||
net_con_update(client->con, NET_EVENT_READ | NET_EVENT_WRITE);
|
net_con_update(client->con, NET_EVENT_READ | NET_EVENT_WRITE);
|
||||||
client->callback(client, ADC_CLIENT_SSL_HANDSHAKE, 0);
|
client->callback(client, ADC_CLIENT_SSL_HANDSHAKE, 0);
|
||||||
ADC_client_set_state(client, ps_conn_ssl);
|
ADC_client_set_state(client, ps_conn_ssl);
|
||||||
net_con_ssl_handshake(client->con, net_con_ssl_mode_client, g_adc_client->ctx);
|
|
||||||
|
net_con_ssl_handshake(client->con, net_con_ssl_mode_client, NULL);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
#endif
|
#endif
|
||||||
ADC_client_send_handshake(client);
|
{
|
||||||
|
struct adc_message* handshake = adc_msg_create(ADC_HANDSHAKE);
|
||||||
|
net_con_update(client->con, NET_EVENT_READ);
|
||||||
|
client->callback(client, ADC_CLIENT_CONNECTED, 0);
|
||||||
|
ADC_client_send(client, handshake);
|
||||||
|
ADC_client_set_state(client, ps_protocol);
|
||||||
|
adc_msg_free(handshake);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef SSL_SUPPORT
|
#ifdef SSL_SUPPORT
|
||||||
static void ADC_client_on_connected_ssl(struct ADC_client* client)
|
static void ADC_client_on_connected_ssl(struct ADC_client* client)
|
||||||
{
|
{
|
||||||
ADC_TRACE;
|
ADC_TRACE;
|
||||||
struct ADC_client_callback_data data;
|
struct adc_message* handshake = adc_msg_create(ADC_HANDSHAKE);
|
||||||
struct ADC_client_tls_info tls_info;
|
client->callback(client, ADC_CLIENT_SSL_OK, 0);
|
||||||
data.tls_info = &tls_info;
|
client->callback(client, ADC_CLIENT_CONNECTED, 0);
|
||||||
|
net_con_update(client->con, NET_EVENT_READ);
|
||||||
tls_info.version = net_ssl_get_tls_version(client->con);
|
ADC_client_send(client, handshake);
|
||||||
tls_info.cipher = net_ssl_get_tls_cipher(client->con);
|
ADC_client_set_state(client, ps_protocol);
|
||||||
|
adc_msg_free(handshake);
|
||||||
client->callback(client, ADC_CLIENT_SSL_OK, &data);
|
|
||||||
ADC_client_send_handshake(client);
|
|
||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* uhub - A tiny ADC p2p connection hub
|
* uhub - A tiny ADC p2p connection hub
|
||||||
* Copyright (C) 2007-2014, Jan Vidar Krey
|
* Copyright (C) 2007-2012, Jan Vidar Krey
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@@ -95,11 +95,6 @@ struct ADC_client_quit_reason
|
|||||||
int flags;
|
int flags;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct ADC_client_tls_info
|
|
||||||
{
|
|
||||||
const char* cipher;
|
|
||||||
const char* version;
|
|
||||||
};
|
|
||||||
|
|
||||||
struct ADC_client_callback_data
|
struct ADC_client_callback_data
|
||||||
{
|
{
|
||||||
@@ -108,7 +103,6 @@ struct ADC_client_callback_data
|
|||||||
struct ADC_chat_message* chat;
|
struct ADC_chat_message* chat;
|
||||||
struct ADC_user* user;
|
struct ADC_user* user;
|
||||||
struct ADC_client_quit_reason* quit;
|
struct ADC_client_quit_reason* quit;
|
||||||
struct ADC_client_tls_info* tls_info;
|
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user