Compare commits
4 Commits
v5.0.6
...
user/bserg
Author | SHA1 | Date | |
---|---|---|---|
067c128474 | |||
a127d9ef0d | |||
4a845e3cc4 | |||
633afa5bed |
@ -1,46 +0,0 @@
|
|||||||
# https://releases.llvm.org/7.0.0/tools/clang/docs/ClangFormatStyleOptions.html
|
|
||||||
|
|
||||||
---
|
|
||||||
Language: Cpp
|
|
||||||
|
|
||||||
BasedOnStyle: WebKit
|
|
||||||
|
|
||||||
AlignAfterOpenBracket: Align
|
|
||||||
AlignOperands: true
|
|
||||||
AlignTrailingComments: true
|
|
||||||
AllowAllParametersOfDeclarationOnNextLine: true
|
|
||||||
AllowShortBlocksOnASingleLine: false
|
|
||||||
AllowShortCaseLabelsOnASingleLine: true
|
|
||||||
AllowShortFunctionsOnASingleLine: InlineOnly
|
|
||||||
AllowShortIfStatementsOnASingleLine: true
|
|
||||||
AllowShortLoopsOnASingleLine: false
|
|
||||||
AlwaysBreakTemplateDeclarations: true
|
|
||||||
BinPackArguments: false
|
|
||||||
BinPackParameters: false
|
|
||||||
BreakBeforeBinaryOperators: None
|
|
||||||
BreakBeforeBraces: Allman
|
|
||||||
BreakConstructorInitializersBeforeComma: true
|
|
||||||
ColumnLimit: 100
|
|
||||||
ConstructorInitializerAllOnOneLineOrOnePerLine: false
|
|
||||||
Cpp11BracedListStyle: true
|
|
||||||
FixNamespaceComments: true
|
|
||||||
IncludeBlocks: Regroup
|
|
||||||
IncludeCategories:
|
|
||||||
- Regex: '^["<](stdafx|pch)\.h[">]$'
|
|
||||||
Priority: -1
|
|
||||||
- Regex: '^<Windows\.h>$'
|
|
||||||
Priority: 3
|
|
||||||
- Regex: '^<(WinIoCtl|winhttp|Shellapi)\.h>$'
|
|
||||||
Priority: 4
|
|
||||||
- Regex: '.*'
|
|
||||||
Priority: 2
|
|
||||||
IndentCaseLabels: true
|
|
||||||
IndentWidth: 4
|
|
||||||
KeepEmptyLinesAtTheStartOfBlocks: false
|
|
||||||
MaxEmptyLinesToKeep: 2
|
|
||||||
NamespaceIndentation: All
|
|
||||||
PenaltyReturnTypeOnItsOwnLine: 1000
|
|
||||||
PointerAlignment: Left
|
|
||||||
SpaceAfterTemplateKeyword: false
|
|
||||||
Standard: Cpp11
|
|
||||||
UseTab: Never
|
|
@ -1,4 +1,3 @@
|
|||||||
build
|
build
|
||||||
CMakeCache.txt
|
CMakeCache.txt
|
||||||
ws/CMakeCache.txt
|
ws/CMakeCache.txt
|
||||||
test/build
|
|
||||||
|
34
.travis.yml
34
.travis.yml
@ -1,25 +1,17 @@
|
|||||||
language: bash
|
language: bash
|
||||||
|
|
||||||
# See https://github.com/amaiorano/vectrexy/blob/master/.travis.yml
|
|
||||||
# for ideas on installing vcpkg
|
|
||||||
|
|
||||||
matrix:
|
matrix:
|
||||||
include:
|
include:
|
||||||
# macOS
|
# macOS
|
||||||
- os: osx
|
- os: osx
|
||||||
env:
|
|
||||||
- HOMEBREW_NO_AUTO_UPDATE=1
|
|
||||||
compiler: clang
|
compiler: clang
|
||||||
script:
|
script:
|
||||||
- brew install mbedtls
|
|
||||||
- python test/run.py
|
- python test/run.py
|
||||||
- make ws
|
- make ws
|
||||||
|
|
||||||
# Linux
|
# Linux
|
||||||
- os: linux
|
- os: linux
|
||||||
dist: bionic
|
dist: xenial
|
||||||
before_install:
|
|
||||||
- sudo apt-get install -y libmbedtls-dev
|
|
||||||
script:
|
script:
|
||||||
- python test/run.py
|
- python test/run.py
|
||||||
- make ws
|
- make ws
|
||||||
@ -36,21 +28,9 @@ matrix:
|
|||||||
# - CXX=clang++
|
# - CXX=clang++
|
||||||
|
|
||||||
# Windows
|
# Windows
|
||||||
# - os: windows
|
- os: windows
|
||||||
# env:
|
env:
|
||||||
# - CMAKE_PATH="/c/Program Files/CMake/bin"
|
- CMAKE_PATH="/c/Program Files/CMake/bin"
|
||||||
# script:
|
script:
|
||||||
# - cd third_party/zlib
|
- export PATH=$CMAKE_PATH:$PATH
|
||||||
# - cmake .
|
- python test/run.py
|
||||||
# - cmake --build . --target install
|
|
||||||
# - cd ../..
|
|
||||||
# # - cd third_party/mbedtls
|
|
||||||
# # - cmake .
|
|
||||||
# # - cmake --build . --target install
|
|
||||||
# # - cd ../..
|
|
||||||
# - export PATH=$CMAKE_PATH:$PATH
|
|
||||||
# - cd test
|
|
||||||
# - cmake .
|
|
||||||
# - cmake --build --parallel .
|
|
||||||
# - ixwebsocket_unittest.exe
|
|
||||||
# # - python test/run.py
|
|
||||||
|
52
CHANGELOG.md
52
CHANGELOG.md
@ -1,52 +0,0 @@
|
|||||||
# Changelog
|
|
||||||
All notable changes to this project will be documented in this file.
|
|
||||||
|
|
||||||
## [5.0.5] - 2019-08-22
|
|
||||||
- Windows: use select instead of WSAPoll, through a poll wrapper
|
|
||||||
|
|
||||||
## [5.0.4] - 2019-08-20
|
|
||||||
- Windows build fixes (there was a problem with the use of ::poll that has a different name on Windows (WSAPoll))
|
|
||||||
|
|
||||||
## [5.0.3] - 2019-08-14
|
|
||||||
- CobraMetricThreadedPublisher _enable flag is an atomic, and CobraMetricsPublisher is enabled by default
|
|
||||||
|
|
||||||
## [5.0.2] - 2019-08-01
|
|
||||||
- ws cobra_subscribe has a new -q (quiet) option
|
|
||||||
- ws cobra_subscribe knows to and display msg stats (count and # of messages received per second)
|
|
||||||
- ws cobra_subscribe, cobra_to_statsd and cobra_to_sentry commands have a new option, --filter to restrict the events they want to receive
|
|
||||||
|
|
||||||
## [5.0.1] - 2019-07-25
|
|
||||||
- ws connect command has a new option to send in binary mode (still default to text)
|
|
||||||
- ws connect command has readline history thanks to libnoise-cpp. Now ws connect one can use using arrows to lookup previous sent messages and edit them
|
|
||||||
|
|
||||||
## [5.0.0] - 2019-06-23
|
|
||||||
### Changed
|
|
||||||
- New HTTP server / still very early. ws gained a new command, httpd can run a simple webserver serving local files.
|
|
||||||
- IXDNSLookup. Uses weak pointer + smart_ptr + shared_from_this instead of static sets + mutex to handle object going away before dns lookup has resolved
|
|
||||||
- cobra_to_sentry / backtraces are reversed and line number is not extracted correctly
|
|
||||||
- mbedtls and zlib are searched with find_package, and we use the vendored version if nothing is found
|
|
||||||
- travis CI uses g++ on Linux
|
|
||||||
|
|
||||||
## [4.0.0] - 2019-06-09
|
|
||||||
### Changed
|
|
||||||
- WebSocket::send() sends message in TEXT mode by default
|
|
||||||
- WebSocketMessage sets a new binary field, which tells whether the received incoming message is binary or text
|
|
||||||
- WebSocket::send takes a third arg, binary which default to true (can be text too)
|
|
||||||
- WebSocket callback only take one object, a const ix::WebSocketMessagePtr& msg
|
|
||||||
- Add explicit WebSocket::sendBinary method
|
|
||||||
- New headers + WebSocketMessage class to hold message data, still not used across the board
|
|
||||||
- Add test/compatibility folder with small servers and clients written in different languages and different libraries to test compatibility.
|
|
||||||
- ws echo_server has a -g option to print a greeting message on connect
|
|
||||||
- IXSocketMbedTLS: better error handling in close and connect
|
|
||||||
|
|
||||||
## [3.1.2] - 2019-06-06
|
|
||||||
### Added
|
|
||||||
- ws connect has a -x option to disable per message deflate
|
|
||||||
- Add WebSocket::disablePerMessageDeflate() option.
|
|
||||||
|
|
||||||
## [3.0.0] - 2019-06-xx
|
|
||||||
### Changed
|
|
||||||
- TLS, aka SSL works on Windows (websocket and http clients)
|
|
||||||
- ws command line tool build on Windows
|
|
||||||
- Async API for HttpClient
|
|
||||||
- HttpClient API changed to use shared_ptr for response and request
|
|
@ -1,13 +0,0 @@
|
|||||||
find_path(MBEDTLS_INCLUDE_DIRS mbedtls/ssl.h)
|
|
||||||
|
|
||||||
find_library(MBEDTLS_LIBRARY mbedtls)
|
|
||||||
find_library(MBEDX509_LIBRARY mbedx509)
|
|
||||||
find_library(MBEDCRYPTO_LIBRARY mbedcrypto)
|
|
||||||
|
|
||||||
set(MBEDTLS_LIBRARIES "${MBEDTLS_LIBRARY}" "${MBEDX509_LIBRARY}" "${MBEDCRYPTO_LIBRARY}")
|
|
||||||
|
|
||||||
include(FindPackageHandleStandardArgs)
|
|
||||||
find_package_handle_standard_args(MBEDTLS DEFAULT_MSG
|
|
||||||
MBEDTLS_INCLUDE_DIRS MBEDTLS_LIBRARY MBEDX509_LIBRARY MBEDCRYPTO_LIBRARY)
|
|
||||||
|
|
||||||
mark_as_advanced(MBEDTLS_INCLUDE_DIRS MBEDTLS_LIBRARY MBEDX509_LIBRARY MBEDCRYPTO_LIBRARY)
|
|
125
CMakeLists.txt
125
CMakeLists.txt
@ -4,8 +4,6 @@
|
|||||||
#
|
#
|
||||||
|
|
||||||
cmake_minimum_required(VERSION 3.4.1)
|
cmake_minimum_required(VERSION 3.4.1)
|
||||||
set(CMAKE_MODULE_PATH "${CMAKE_CURRENT_SOURCE_DIR}/CMake;${CMAKE_MODULE_PATH}")
|
|
||||||
|
|
||||||
project(ixwebsocket C CXX)
|
project(ixwebsocket C CXX)
|
||||||
|
|
||||||
set (CMAKE_CXX_STANDARD 14)
|
set (CMAKE_CXX_STANDARD 14)
|
||||||
@ -22,68 +20,60 @@ if ("${CMAKE_CXX_COMPILER_ID}" MATCHES "Clang")
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
set( IXWEBSOCKET_SOURCES
|
set( IXWEBSOCKET_SOURCES
|
||||||
ixwebsocket/IXCancellationRequest.cpp
|
|
||||||
ixwebsocket/IXConnectionState.cpp
|
|
||||||
ixwebsocket/IXDNSLookup.cpp
|
|
||||||
ixwebsocket/IXHttp.cpp
|
|
||||||
ixwebsocket/IXHttpClient.cpp
|
|
||||||
ixwebsocket/IXHttpServer.cpp
|
|
||||||
ixwebsocket/IXNetSystem.cpp
|
|
||||||
ixwebsocket/IXSelectInterrupt.cpp
|
|
||||||
ixwebsocket/IXSelectInterruptFactory.cpp
|
|
||||||
ixwebsocket/IXSocket.cpp
|
ixwebsocket/IXSocket.cpp
|
||||||
|
ixwebsocket/IXSocketServer.cpp
|
||||||
ixwebsocket/IXSocketConnect.cpp
|
ixwebsocket/IXSocketConnect.cpp
|
||||||
ixwebsocket/IXSocketFactory.cpp
|
ixwebsocket/IXSocketFactory.cpp
|
||||||
ixwebsocket/IXSocketServer.cpp
|
ixwebsocket/IXDNSLookup.cpp
|
||||||
ixwebsocket/IXUrlParser.cpp
|
ixwebsocket/IXCancellationRequest.cpp
|
||||||
|
ixwebsocket/IXNetSystem.cpp
|
||||||
ixwebsocket/IXWebSocket.cpp
|
ixwebsocket/IXWebSocket.cpp
|
||||||
ixwebsocket/IXWebSocketCloseConstants.cpp
|
ixwebsocket/IXWebSocketServer.cpp
|
||||||
|
ixwebsocket/IXWebSocketTransport.cpp
|
||||||
ixwebsocket/IXWebSocketHandshake.cpp
|
ixwebsocket/IXWebSocketHandshake.cpp
|
||||||
ixwebsocket/IXWebSocketHttpHeaders.cpp
|
|
||||||
ixwebsocket/IXWebSocketMessageQueue.cpp
|
|
||||||
ixwebsocket/IXWebSocketPerMessageDeflate.cpp
|
ixwebsocket/IXWebSocketPerMessageDeflate.cpp
|
||||||
ixwebsocket/IXWebSocketPerMessageDeflateCodec.cpp
|
ixwebsocket/IXWebSocketPerMessageDeflateCodec.cpp
|
||||||
ixwebsocket/IXWebSocketPerMessageDeflateOptions.cpp
|
ixwebsocket/IXWebSocketPerMessageDeflateOptions.cpp
|
||||||
ixwebsocket/IXWebSocketServer.cpp
|
ixwebsocket/IXWebSocketHttpHeaders.cpp
|
||||||
ixwebsocket/IXWebSocketTransport.cpp
|
ixwebsocket/IXHttpClient.cpp
|
||||||
|
ixwebsocket/IXUrlParser.cpp
|
||||||
ixwebsocket/LUrlParser.cpp
|
ixwebsocket/LUrlParser.cpp
|
||||||
|
ixwebsocket/IXSelectInterrupt.cpp
|
||||||
|
ixwebsocket/IXSelectInterruptFactory.cpp
|
||||||
|
ixwebsocket/IXConnectionState.cpp
|
||||||
|
ixwebsocket/IXWebSocketCloseConstants.cpp
|
||||||
|
ixwebsocket/IXWebSocketMessageQueue.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
set( IXWEBSOCKET_HEADERS
|
set( IXWEBSOCKET_HEADERS
|
||||||
ixwebsocket/IXCancellationRequest.h
|
|
||||||
ixwebsocket/IXConnectionState.h
|
|
||||||
ixwebsocket/IXDNSLookup.h
|
|
||||||
ixwebsocket/IXHttp.h
|
|
||||||
ixwebsocket/IXHttpClient.h
|
|
||||||
ixwebsocket/IXHttpServer.h
|
|
||||||
ixwebsocket/IXNetSystem.h
|
|
||||||
ixwebsocket/IXProgressCallback.h
|
|
||||||
ixwebsocket/IXSelectInterrupt.h
|
|
||||||
ixwebsocket/IXSelectInterruptFactory.h
|
|
||||||
ixwebsocket/IXSetThreadName.h
|
|
||||||
ixwebsocket/IXSocket.h
|
ixwebsocket/IXSocket.h
|
||||||
|
ixwebsocket/IXSocketServer.h
|
||||||
ixwebsocket/IXSocketConnect.h
|
ixwebsocket/IXSocketConnect.h
|
||||||
ixwebsocket/IXSocketFactory.h
|
ixwebsocket/IXSocketFactory.h
|
||||||
ixwebsocket/IXSocketServer.h
|
ixwebsocket/IXSetThreadName.h
|
||||||
ixwebsocket/IXUrlParser.h
|
ixwebsocket/IXDNSLookup.h
|
||||||
|
ixwebsocket/IXCancellationRequest.h
|
||||||
|
ixwebsocket/IXNetSystem.h
|
||||||
|
ixwebsocket/IXProgressCallback.h
|
||||||
ixwebsocket/IXWebSocket.h
|
ixwebsocket/IXWebSocket.h
|
||||||
ixwebsocket/IXWebSocketCloseConstants.h
|
ixwebsocket/IXWebSocketServer.h
|
||||||
ixwebsocket/IXWebSocketCloseInfo.h
|
ixwebsocket/IXWebSocketTransport.h
|
||||||
ixwebsocket/IXWebSocketErrorInfo.h
|
|
||||||
ixwebsocket/IXWebSocketHandshake.h
|
ixwebsocket/IXWebSocketHandshake.h
|
||||||
ixwebsocket/IXWebSocketHttpHeaders.h
|
ixwebsocket/IXWebSocketSendInfo.h
|
||||||
ixwebsocket/IXWebSocketMessage.h
|
ixwebsocket/IXWebSocketErrorInfo.h
|
||||||
ixwebsocket/IXWebSocketMessageQueue.h
|
|
||||||
ixwebsocket/IXWebSocketMessageType.h
|
|
||||||
ixwebsocket/IXWebSocketOpenInfo.h
|
|
||||||
ixwebsocket/IXWebSocketPerMessageDeflate.h
|
ixwebsocket/IXWebSocketPerMessageDeflate.h
|
||||||
ixwebsocket/IXWebSocketPerMessageDeflateCodec.h
|
ixwebsocket/IXWebSocketPerMessageDeflateCodec.h
|
||||||
ixwebsocket/IXWebSocketPerMessageDeflateOptions.h
|
ixwebsocket/IXWebSocketPerMessageDeflateOptions.h
|
||||||
ixwebsocket/IXWebSocketSendInfo.h
|
ixwebsocket/IXWebSocketHttpHeaders.h
|
||||||
ixwebsocket/IXWebSocketServer.h
|
|
||||||
ixwebsocket/IXWebSocketTransport.h
|
|
||||||
ixwebsocket/LUrlParser.h
|
|
||||||
ixwebsocket/libwshandshake.hpp
|
ixwebsocket/libwshandshake.hpp
|
||||||
|
ixwebsocket/IXHttpClient.h
|
||||||
|
ixwebsocket/IXUrlParser.h
|
||||||
|
ixwebsocket/LUrlParser.h
|
||||||
|
ixwebsocket/IXSelectInterrupt.h
|
||||||
|
ixwebsocket/IXSelectInterruptFactory.h
|
||||||
|
ixwebsocket/IXConnectionState.h
|
||||||
|
ixwebsocket/IXWebSocketCloseConstants.h
|
||||||
|
ixwebsocket/IXWebSocketMessageQueue.h
|
||||||
)
|
)
|
||||||
|
|
||||||
if (UNIX)
|
if (UNIX)
|
||||||
@ -103,26 +93,17 @@ else()
|
|||||||
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSelectInterruptEventFd.h)
|
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSelectInterruptEventFd.h)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (WIN32)
|
|
||||||
set(USE_MBED_TLS TRUE)
|
|
||||||
endif()
|
|
||||||
|
|
||||||
set(USE_OPEN_SSL FALSE)
|
set(USE_OPEN_SSL FALSE)
|
||||||
if (USE_TLS)
|
if (USE_TLS)
|
||||||
add_definitions(-DIXWEBSOCKET_USE_TLS)
|
add_definitions(-DIXWEBSOCKET_USE_TLS)
|
||||||
|
|
||||||
if (USE_MBED_TLS)
|
if (APPLE)
|
||||||
add_definitions(-DIXWEBSOCKET_USE_MBED_TLS)
|
|
||||||
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSocketMbedTLS.h)
|
|
||||||
list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSocketMbedTLS.cpp)
|
|
||||||
elseif (APPLE)
|
|
||||||
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSocketAppleSSL.h)
|
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSocketAppleSSL.h)
|
||||||
list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSocketAppleSSL.cpp)
|
list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSocketAppleSSL.cpp)
|
||||||
elseif (WIN32)
|
elseif (WIN32)
|
||||||
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSocketSChannel.h)
|
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSocketSChannel.h)
|
||||||
list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSocketSChannel.cpp)
|
list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSocketSChannel.cpp)
|
||||||
else()
|
else()
|
||||||
add_definitions(-DIXWEBSOCKET_USE_OPEN_SSL)
|
|
||||||
set(USE_OPEN_SSL TRUE)
|
set(USE_OPEN_SSL TRUE)
|
||||||
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSocketOpenSSL.h)
|
list( APPEND IXWEBSOCKET_HEADERS ixwebsocket/IXSocketOpenSSL.h)
|
||||||
list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSocketOpenSSL.cpp)
|
list( APPEND IXWEBSOCKET_SOURCES ixwebsocket/IXSocketOpenSSL.cpp)
|
||||||
@ -134,16 +115,11 @@ add_library( ixwebsocket STATIC
|
|||||||
${IXWEBSOCKET_HEADERS}
|
${IXWEBSOCKET_HEADERS}
|
||||||
)
|
)
|
||||||
|
|
||||||
if (APPLE AND USE_TLS AND NOT USE_MBED_TLS)
|
if (APPLE AND USE_TLS)
|
||||||
target_link_libraries(ixwebsocket "-framework foundation" "-framework security")
|
target_link_libraries(ixwebsocket "-framework foundation" "-framework security")
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (UNIX)
|
if (USE_OPEN_SSL)
|
||||||
find_package(Threads)
|
|
||||||
target_link_libraries(ixwebsocket ${CMAKE_THREAD_LIBS_INIT})
|
|
||||||
endif()
|
|
||||||
|
|
||||||
if (USE_TLS AND USE_OPEN_SSL)
|
|
||||||
find_package(OpenSSL REQUIRED)
|
find_package(OpenSSL REQUIRED)
|
||||||
add_definitions(${OPENSSL_DEFINITIONS})
|
add_definitions(${OPENSSL_DEFINITIONS})
|
||||||
message(STATUS "OpenSSL: " ${OPENSSL_VERSION})
|
message(STATUS "OpenSSL: " ${OPENSSL_VERSION})
|
||||||
@ -151,29 +127,18 @@ if (USE_TLS AND USE_OPEN_SSL)
|
|||||||
target_link_libraries(ixwebsocket ${OPENSSL_LIBRARIES})
|
target_link_libraries(ixwebsocket ${OPENSSL_LIBRARIES})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
if (USE_TLS AND USE_MBED_TLS)
|
if (WIN32)
|
||||||
if (USE_VENDORED_THIRD_PARTY)
|
|
||||||
set (ENABLE_PROGRAMS OFF)
|
|
||||||
add_subdirectory(third_party/mbedtls)
|
|
||||||
include_directories(third_party/mbedtls/include)
|
|
||||||
|
|
||||||
target_link_libraries(ixwebsocket mbedtls)
|
|
||||||
else()
|
|
||||||
find_package(MbedTLS REQUIRED)
|
|
||||||
include_directories(${MBEDTLS_INCLUDE_DIRS})
|
|
||||||
target_link_libraries(ixwebsocket ${MBEDTLS_LIBRARIES})
|
|
||||||
endif()
|
|
||||||
endif()
|
|
||||||
|
|
||||||
find_package(ZLIB REQUIRED)
|
|
||||||
if (ZLIB_FOUND)
|
|
||||||
include_directories(${ZLIB_INCLUDE_DIRS})
|
|
||||||
target_link_libraries(ixwebsocket ${ZLIB_LIBRARIES})
|
|
||||||
else()
|
|
||||||
add_subdirectory(third_party/zlib)
|
add_subdirectory(third_party/zlib)
|
||||||
include_directories(third_party/zlib ${CMAKE_CURRENT_BINARY_DIR}/third_party/zlib)
|
include_directories(third_party/zlib ${CMAKE_CURRENT_BINARY_DIR}/third_party/zlib)
|
||||||
target_link_libraries(ixwebsocket zlibstatic wsock32 ws2_32)
|
target_link_libraries(ixwebsocket zlibstatic wsock32 ws2_32)
|
||||||
add_definitions(-D_CRT_SECURE_NO_WARNINGS)
|
add_definitions(-D_CRT_SECURE_NO_WARNINGS)
|
||||||
|
|
||||||
|
else()
|
||||||
|
# gcc/Linux needs -pthread
|
||||||
|
find_package(Threads)
|
||||||
|
|
||||||
|
target_link_libraries(ixwebsocket
|
||||||
|
z ${CMAKE_THREAD_LIBS_INIT})
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
set( IXWEBSOCKET_INCLUDE_DIRS
|
set( IXWEBSOCKET_INCLUDE_DIRS
|
||||||
@ -185,7 +150,7 @@ if (CMAKE_CXX_COMPILER_ID MATCHES "MSVC")
|
|||||||
target_compile_options(ixwebsocket PRIVATE /MP)
|
target_compile_options(ixwebsocket PRIVATE /MP)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
target_include_directories(ixwebsocket PUBLIC ${IXWEBSOCKET_INCLUDE_DIRS})
|
target_include_directories( ixwebsocket PUBLIC ${IXWEBSOCKET_INCLUDE_DIRS} )
|
||||||
|
|
||||||
set_target_properties(ixwebsocket PROPERTIES PUBLIC_HEADER "${IXWEBSOCKET_HEADERS}")
|
set_target_properties(ixwebsocket PROPERTIES PUBLIC_HEADER "${IXWEBSOCKET_HEADERS}")
|
||||||
|
|
||||||
|
@ -1 +1 @@
|
|||||||
5.0.4
|
2.0.0
|
||||||
|
@ -1 +1 @@
|
|||||||
docker/Dockerfile.alpine
|
docker/Dockerfile.ubuntu_xenial
|
278
README.md
278
README.md
@ -4,44 +4,21 @@
|
|||||||
|
|
||||||
## Introduction
|
## Introduction
|
||||||
|
|
||||||
[*WebSocket*](https://en.wikipedia.org/wiki/WebSocket) is a computer communications protocol, providing full-duplex and bi-directionnal communication channels over a single TCP connection. *IXWebSocket* is a C++ library for client and server Websocket communication, and for client and server HTTP communication. The code is derived from [easywsclient](https://github.com/dhbaird/easywsclient) and from the [Satori C SDK](https://github.com/satori-com/satori-rtm-sdk-c). It has been tested on the following platforms.
|
[*WebSocket*](https://en.wikipedia.org/wiki/WebSocket) is a computer communications protocol, providing full-duplex and bi-directionnal communication channels over a single TCP connection. *IXWebSocket* is a C++ library for client and server Websocket communication, and for client HTTP communication. The code is derived from [easywsclient](https://github.com/dhbaird/easywsclient) and from the [Satori C SDK](https://github.com/satori-com/satori-rtm-sdk-c). It has been tested on the following platforms.
|
||||||
|
|
||||||
* macOS
|
* macOS
|
||||||
* iOS
|
* iOS
|
||||||
* Linux
|
* Linux
|
||||||
* Android
|
* Android
|
||||||
* Windows
|
* Windows (no TLS)
|
||||||
|
|
||||||
## Examples
|
## Examples
|
||||||
|
|
||||||
The [*ws*](https://github.com/machinezone/IXWebSocket/tree/master/ws) folder countains many interactive programs for chat, [file transfers](https://github.com/machinezone/IXWebSocket/blob/master/ws/ws_send.cpp), [curl like](https://github.com/machinezone/IXWebSocket/blob/master/ws/ws_http_client.cpp) http clients, demonstrating client and server usage.
|
The [*ws*](https://github.com/machinezone/IXWebSocket/tree/master/ws) folder countains many interactive programs for chat, [file transfers](https://github.com/machinezone/IXWebSocket/blob/master/ws/ws_send.cpp), [curl like](https://github.com/machinezone/IXWebSocket/blob/master/ws/ws_http_client.cpp) http clients, demonstrating client and server usage.
|
||||||
|
|
||||||
### Windows note
|
Here is what the client API looks like.
|
||||||
|
|
||||||
To use the network system on Windows, you need to initialize it once with *WSAStartup()* and clean it up with *WSACleanup()*. We have helpers for that which you can use, see below. This init would typically take place in your main function.
|
|
||||||
|
|
||||||
```
|
```
|
||||||
#include <ixwebsocket/IXNetSystem.h>
|
|
||||||
|
|
||||||
int main()
|
|
||||||
{
|
|
||||||
ix::initNetSystem();
|
|
||||||
|
|
||||||
...
|
|
||||||
|
|
||||||
ix::uninitNetSystem();
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
### WebSocket client API
|
|
||||||
|
|
||||||
```
|
|
||||||
#include <ixwebsocket/IXWebSocket.h>
|
|
||||||
|
|
||||||
...
|
|
||||||
|
|
||||||
# Our websocket object
|
|
||||||
ix::WebSocket webSocket;
|
ix::WebSocket webSocket;
|
||||||
|
|
||||||
std::string url("ws://localhost:8080/");
|
std::string url("ws://localhost:8080/");
|
||||||
@ -51,27 +28,29 @@ webSocket.setUrl(url);
|
|||||||
// to make sure that load balancers do not kill an idle connection.
|
// to make sure that load balancers do not kill an idle connection.
|
||||||
webSocket.setHeartBeatPeriod(45);
|
webSocket.setHeartBeatPeriod(45);
|
||||||
|
|
||||||
// Per message deflate connection is enabled by default. You can tweak its parameters or disable it
|
|
||||||
webSocket.disablePerMessageDeflate();
|
|
||||||
|
|
||||||
// Setup a callback to be fired when a message or an event (open, close, error) is received
|
// Setup a callback to be fired when a message or an event (open, close, error) is received
|
||||||
webSocket.setOnMessageCallback([](const ix::WebSocketMessagePtr& msg)
|
webSocket.setOnMessageCallback(
|
||||||
|
[](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Message)
|
if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
std::cout << msg->str << std::endl;
|
std::cout << str << std::endl;
|
||||||
}
|
}
|
||||||
}
|
});
|
||||||
);
|
|
||||||
|
|
||||||
// Now that our callback is setup, we can start our background thread and receive messages
|
// Now that our callback is setup, we can start our background thread and receive messages
|
||||||
webSocket.start();
|
webSocket.start();
|
||||||
|
|
||||||
// Send a message to the server (default to TEXT mode)
|
// Send a message to the server (default to BINARY mode)
|
||||||
webSocket.send("hello world");
|
webSocket.send("hello world");
|
||||||
|
|
||||||
// The message can be sent in BINARY mode (useful if you send MsgPack data for example)
|
// The message can be sent in TEXT mode
|
||||||
webSocket.sendBinary("some serialized binary data");
|
webSocket.sendText("hello again");
|
||||||
|
|
||||||
// ... finally ...
|
// ... finally ...
|
||||||
|
|
||||||
@ -79,13 +58,9 @@ webSocket.sendBinary("some serialized binary data");
|
|||||||
webSocket.stop()
|
webSocket.stop()
|
||||||
```
|
```
|
||||||
|
|
||||||
### WebSocket server API
|
Here is what the server API looks like. Note that server support is very recent and subject to changes.
|
||||||
|
|
||||||
```
|
```
|
||||||
#include <ixwebsocket/IXWebSocketServer.h>
|
|
||||||
|
|
||||||
...
|
|
||||||
|
|
||||||
// Run a server on localhost at a given port.
|
// Run a server on localhost at a given port.
|
||||||
// Bound host name, max connections and listen backlog can also be passed in as parameters.
|
// Bound host name, max connections and listen backlog can also be passed in as parameters.
|
||||||
ix::WebSocketServer server(port);
|
ix::WebSocketServer server(port);
|
||||||
@ -95,9 +70,14 @@ server.setOnConnectionCallback(
|
|||||||
std::shared_ptr<ConnectionState> connectionState)
|
std::shared_ptr<ConnectionState> connectionState)
|
||||||
{
|
{
|
||||||
webSocket->setOnMessageCallback(
|
webSocket->setOnMessageCallback(
|
||||||
[webSocket, connectionState, &server](const ix::WebSocketMessagePtr msg)
|
[webSocket, connectionState, &server](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
std::cerr << "New connection" << std::endl;
|
std::cerr << "New connection" << std::endl;
|
||||||
|
|
||||||
@ -108,21 +88,19 @@ server.setOnConnectionCallback(
|
|||||||
std::cerr << "id: " << connectionState->getId() << std::endl;
|
std::cerr << "id: " << connectionState->getId() << std::endl;
|
||||||
|
|
||||||
// The uri the client did connect to.
|
// The uri the client did connect to.
|
||||||
std::cerr << "Uri: " << msg->openInfo.uri << std::endl;
|
std::cerr << "Uri: " << openInfo.uri << std::endl;
|
||||||
|
|
||||||
std::cerr << "Headers:" << std::endl;
|
std::cerr << "Headers:" << std::endl;
|
||||||
for (auto it : msg->openInfo.headers)
|
for (auto it : openInfo.headers)
|
||||||
{
|
{
|
||||||
std::cerr << it.first << ": " << it.second << std::endl;
|
std::cerr << it.first << ": " << it.second << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
// For an echo server, we just send back to the client whatever was received by the server
|
// For an echo server, we just send back to the client whatever was received by the server
|
||||||
// All connected clients are available in an std::set. See the broadcast cpp example.
|
// All connected clients are available in an std::set. See the broadcast cpp example.
|
||||||
// Second parameter tells whether we are sending the message in binary or text mode.
|
webSocket->send(str);
|
||||||
// Here we send it in the same mode as it was received.
|
|
||||||
webSocket->send(msg->str, msg->binary);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -144,44 +122,40 @@ server.wait();
|
|||||||
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### HTTP client API
|
Here is what the HTTP client API looks like. Note that HTTP client support is very recent and subject to changes.
|
||||||
|
|
||||||
```
|
```
|
||||||
#include <ixwebsocket/IXHttpClient.h>
|
|
||||||
|
|
||||||
...
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Preparation
|
// Preparation
|
||||||
//
|
//
|
||||||
HttpClient httpClient;
|
HttpClient httpClient;
|
||||||
HttpRequestArgsPtr args = httpClient.createRequest();
|
HttpRequestArgs args;
|
||||||
|
|
||||||
// Custom headers can be set
|
// Custom headers can be set
|
||||||
WebSocketHttpHeaders headers;
|
WebSocketHttpHeaders headers;
|
||||||
headers["Foo"] = "bar";
|
headers["Foo"] = "bar";
|
||||||
args->extraHeaders = headers;
|
args.extraHeaders = headers;
|
||||||
|
|
||||||
// Timeout options
|
// Timeout options
|
||||||
args->connectTimeout = connectTimeout;
|
args.connectTimeout = connectTimeout;
|
||||||
args->transferTimeout = transferTimeout;
|
args.transferTimeout = transferTimeout;
|
||||||
|
|
||||||
// Redirect options
|
// Redirect options
|
||||||
args->followRedirects = followRedirects;
|
args.followRedirects = followRedirects;
|
||||||
args->maxRedirects = maxRedirects;
|
args.maxRedirects = maxRedirects;
|
||||||
|
|
||||||
// Misc
|
// Misc
|
||||||
args->compress = compress; // Enable gzip compression
|
args.compress = compress; // Enable gzip compression
|
||||||
args->verbose = verbose;
|
args.verbose = verbose;
|
||||||
args->logger = [](const std::string& msg)
|
args.logger = [](const std::string& msg)
|
||||||
{
|
{
|
||||||
std::cout << msg;
|
std::cout << msg;
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
// Synchronous Request
|
// Request
|
||||||
//
|
//
|
||||||
HttpResponsePtr out;
|
HttpResponse out;
|
||||||
std::string url = "https://www.google.com";
|
std::string url = "https://www.google.com";
|
||||||
|
|
||||||
// HEAD request
|
// HEAD request
|
||||||
@ -201,120 +175,34 @@ out = httpClient.post(url, std::string("foo=bar"), args);
|
|||||||
//
|
//
|
||||||
// Result
|
// Result
|
||||||
//
|
//
|
||||||
auto statusCode = response->statusCode; // Can be HttpErrorCode::Ok, HttpErrorCode::UrlMalformed, etc...
|
auto statusCode = std::get<0>(out);
|
||||||
auto errorCode = response->errorCode; // 200, 404, etc...
|
auto errorCode = std::get<1>(out);
|
||||||
auto responseHeaders = response->headers; // All the headers in a special case-insensitive unordered_map of (string, string)
|
auto responseHeaders = std::get<2>(out);
|
||||||
auto payload = response->payload; // All the bytes from the response as an std::string
|
auto payload = std::get<3>(out);
|
||||||
auto errorMsg = response->errorMsg; // Descriptive error message in case of failure
|
auto errorMsg = std::get<4>(out);
|
||||||
auto uploadSize = response->uploadSize; // Byte count of uploaded data
|
auto uploadSize = std::get<5>(out);
|
||||||
auto downloadSize = response->downloadSize; // Byte count of downloaded data
|
auto downloadSize = std::get<6>(out);
|
||||||
|
|
||||||
//
|
|
||||||
// Asynchronous Request
|
|
||||||
//
|
|
||||||
bool async = true;
|
|
||||||
HttpClient httpClient(async);
|
|
||||||
auto args = httpClient.createRequest(url, HttpClient::kGet);
|
|
||||||
|
|
||||||
// Push the request to a queue,
|
|
||||||
bool ok = httpClient.performRequest(args, [](const HttpResponsePtr& response)
|
|
||||||
{
|
|
||||||
// This callback execute in a background thread. Make sure you uses appropriate protection such as mutex
|
|
||||||
auto statusCode = response->statusCode; // acess results
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
// ok will be false if your httpClient is not async
|
|
||||||
```
|
|
||||||
|
|
||||||
### HTTP server API
|
|
||||||
|
|
||||||
```
|
|
||||||
#include <ixwebsocket/IXHttpServer.h>
|
|
||||||
|
|
||||||
ix::HttpServer server(port, hostname);
|
|
||||||
|
|
||||||
auto res = server.listen();
|
|
||||||
if (!res.first)
|
|
||||||
{
|
|
||||||
std::cerr << res.second << std::endl;
|
|
||||||
return 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
server.start();
|
|
||||||
server.wait();
|
|
||||||
```
|
|
||||||
|
|
||||||
If you want to handle how requests are processed, implement the setOnConnectionCallback callback, which takes an HttpRequestPtr as input, and returns an HttpResponsePtr. You can look at HttpServer::setDefaultConnectionCallback for a slightly more advanced callback example.
|
|
||||||
|
|
||||||
```
|
|
||||||
setOnConnectionCallback(
|
|
||||||
[this](HttpRequestPtr request,
|
|
||||||
std::shared_ptr<ConnectionState> /*connectionState*/) -> HttpResponsePtr
|
|
||||||
{
|
|
||||||
// Build a string for the response
|
|
||||||
std::stringstream ss;
|
|
||||||
ss << request->method
|
|
||||||
<< " "
|
|
||||||
<< request->uri;
|
|
||||||
|
|
||||||
std::string content = ss.str();
|
|
||||||
|
|
||||||
return std::make_shared<HttpResponse>(200, "OK",
|
|
||||||
HttpErrorCode::Ok,
|
|
||||||
WebSocketHttpHeaders(),
|
|
||||||
content);
|
|
||||||
}
|
|
||||||
```
|
```
|
||||||
|
|
||||||
## Build
|
## Build
|
||||||
|
|
||||||
### CMake
|
|
||||||
|
|
||||||
CMakefiles for the library and the examples are available. This library has few dependencies, so it is possible to just add the source files into your project. Otherwise the usual way will suffice.
|
CMakefiles for the library and the examples are available. This library has few dependencies, so it is possible to just add the source files into your project. Otherwise the usual way will suffice.
|
||||||
|
|
||||||
```
|
```
|
||||||
mkdir build # make a build dir so that you can build out of tree.
|
mkdir build # make a build dir so that you can build out of tree.
|
||||||
cd build
|
cd build
|
||||||
cmake -DUSE_TLS=1 ..
|
cmake ..
|
||||||
make -j
|
make -j
|
||||||
make install # will install to /usr/local on Unix, on macOS it is a good idea to sudo chown -R `whoami`:staff /usr/local
|
make install # will install to /usr/local on Unix, on macOS it is a good idea to sudo chown -R `whoami`:staff /usr/local
|
||||||
```
|
```
|
||||||
|
|
||||||
Headers and a static library will be installed to the target dir.
|
Headers and a static library will be installed to the target dir.
|
||||||
|
|
||||||
|
A [conan](https://conan.io/) file is available at [conan-IXWebSocket](https://github.com/Zinnion/conan-IXWebSocket).
|
||||||
|
|
||||||
There is a unittest which can be executed by typing `make test`.
|
There is a unittest which can be executed by typing `make test`.
|
||||||
|
|
||||||
Options for building:
|
There is a Dockerfile for running some code on Linux. To use docker-compose you must make a docker container first.
|
||||||
|
|
||||||
* `-DUSE_TLS=1` will enable TLS support
|
|
||||||
* `-DUSE_MBED_TLS=1` will use [mbedlts](https://tls.mbed.org/) for the TLS support (default on Windows)
|
|
||||||
* `-DUSE_WS=1` will build the ws interactive command line tool
|
|
||||||
|
|
||||||
### vcpkg
|
|
||||||
|
|
||||||
It is possible to get IXWebSocket through Microsoft [vcpkg](https://github.com/microsoft/vcpkg).
|
|
||||||
|
|
||||||
```
|
|
||||||
vcpkg install ixwebsocket
|
|
||||||
```
|
|
||||||
|
|
||||||
### Conan
|
|
||||||
|
|
||||||
Support for building with conan was contributed by Olivia Zoe (thanks !). The package name to reference is `IXWebSocket/5.0.0@LunarWatcher/stable`. The package is in the process to be published to the official conan package repo, but in the meantime, it can be accessed by adding a new remote
|
|
||||||
|
|
||||||
```
|
|
||||||
conan remote add remote_name_here https://api.bintray.com/conan/oliviazoe0/conan-packages
|
|
||||||
```
|
|
||||||
|
|
||||||
### Docker
|
|
||||||
|
|
||||||
There is a Dockerfile for running the unittest on Linux, and to run the `ws` tool. It is also available on the docker registry.
|
|
||||||
|
|
||||||
```
|
|
||||||
docker run bsergean/ws
|
|
||||||
```
|
|
||||||
|
|
||||||
To use docker-compose you must make a docker container first.
|
|
||||||
|
|
||||||
```
|
```
|
||||||
$ make docker
|
$ make docker
|
||||||
@ -327,6 +215,12 @@ ws is a websocket tool
|
|||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
|
Finally you can build and install the `ws command line tool` with Homebrew. The homebrew version might be slightly out of date.
|
||||||
|
|
||||||
|
```
|
||||||
|
brew tap bsergean/IXWebSocket
|
||||||
|
brew install IXWebSocket
|
||||||
|
```
|
||||||
|
|
||||||
## Implementation details
|
## Implementation details
|
||||||
|
|
||||||
@ -336,7 +230,7 @@ The per message deflate compression option is supported. It can lead to very nic
|
|||||||
|
|
||||||
### TLS/SSL
|
### TLS/SSL
|
||||||
|
|
||||||
Connections can be optionally secured and encrypted with TLS/SSL when using a wss:// endpoint, or using normal un-encrypted socket with ws:// endpoints. AppleSSL is used on iOS and macOS, OpenSSL is used on Android and Linux, mbedTLS is used on Windows.
|
Connections can be optionally secured and encrypted with TLS/SSL when using a wss:// endpoint, or using normal un-encrypted socket with ws:// endpoints. AppleSSL is used on iOS and macOS, and OpenSSL is used on Android and Linux.
|
||||||
|
|
||||||
### Polling and background thread work
|
### Polling and background thread work
|
||||||
|
|
||||||
@ -344,7 +238,7 @@ No manual polling to fetch data is required. Data is sent and received instantly
|
|||||||
|
|
||||||
### Automatic reconnection
|
### Automatic reconnection
|
||||||
|
|
||||||
If the remote end (server) breaks the connection, the code will try to perpetually reconnect, by using an exponential backoff strategy, capped at one retry every 10 seconds. This behavior can be disabled.
|
If the remote end (server) breaks the connection, the code will try to perpetually reconnect, by using an exponential backoff strategy, capped at one retry every 10 seconds.
|
||||||
|
|
||||||
### Large messages
|
### Large messages
|
||||||
|
|
||||||
@ -352,8 +246,6 @@ Large frames are broken up into smaller chunks or messages to avoid filling up t
|
|||||||
|
|
||||||
## Limitations
|
## Limitations
|
||||||
|
|
||||||
* On Windows TLS is not setup yet to validate certificates.
|
|
||||||
* There is no convenient way to embed a ca cert.
|
|
||||||
* No utf-8 validation is made when sending TEXT message with sendText()
|
* No utf-8 validation is made when sending TEXT message with sendText()
|
||||||
* Automatic reconnection works at the TCP socket level, and will detect remote end disconnects. However, if the device/computer network become unreachable (by turning off wifi), it is quite hard to reliably and timely detect it at the socket level using `recv` and `send` error codes. [Here](https://stackoverflow.com/questions/14782143/linux-socket-how-to-detect-disconnected-network-in-a-client-program) is a good discussion on the subject. This behavior is consistent with other runtimes such as node.js. One way to detect a disconnected device with low level C code is to do a name resolution with DNS but this can be expensive. Mobile devices have good and reliable API to do that.
|
* Automatic reconnection works at the TCP socket level, and will detect remote end disconnects. However, if the device/computer network become unreachable (by turning off wifi), it is quite hard to reliably and timely detect it at the socket level using `recv` and `send` error codes. [Here](https://stackoverflow.com/questions/14782143/linux-socket-how-to-detect-disconnected-network-in-a-client-program) is a good discussion on the subject. This behavior is consistent with other runtimes such as node.js. One way to detect a disconnected device with low level C code is to do a name resolution with DNS but this can be expensive. Mobile devices have good and reliable API to do that.
|
||||||
* The server code is using select to detect incoming data, and creates one OS thread per connection. This is not as scalable as strategies using epoll or kqueue.
|
* The server code is using select to detect incoming data, and creates one OS thread per connection. This is not as scalable as strategies using epoll or kqueue.
|
||||||
@ -419,27 +311,33 @@ If the connection was closed and sending failed, the return value will be set to
|
|||||||
The onMessage event will be fired when the connection is opened or closed. This is similar to the [Javascript browser API](https://developer.mozilla.org/en-US/docs/Web/API/WebSocket), which has `open` and `close` events notification that can be registered with the browser `addEventListener`.
|
The onMessage event will be fired when the connection is opened or closed. This is similar to the [Javascript browser API](https://developer.mozilla.org/en-US/docs/Web/API/WebSocket), which has `open` and `close` events notification that can be registered with the browser `addEventListener`.
|
||||||
|
|
||||||
```
|
```
|
||||||
webSocket.setOnMessageCallback([](const ix::WebSocketMessagePtr& msg)
|
webSocket.setOnMessageCallback(
|
||||||
|
[](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
std::cout << "send greetings" << std::endl;
|
std::cout << "send greetings" << std::endl;
|
||||||
|
|
||||||
// Headers can be inspected (pairs of string/string)
|
// Headers can be inspected (pairs of string/string)
|
||||||
std::cout << "Handshake Headers:" << std::endl;
|
std::cout << "Handshake Headers:" << std::endl;
|
||||||
for (auto it : msg->headers)
|
for (auto it : headers)
|
||||||
{
|
{
|
||||||
std::cout << it.first << ": " << it.second << std::endl;
|
std::cout << it.first << ": " << it.second << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
std::cout << "disconnected" << std::endl;
|
std::cout << "disconnected" << std::endl;
|
||||||
|
|
||||||
// The server can send an explicit code and reason for closing.
|
// The server can send an explicit code and reason for closing.
|
||||||
// This data can be accessed through the closeInfo object.
|
// This data can be accessed through the closeInfo object.
|
||||||
std::cout << msg->closeInfo.code << std::endl;
|
std::cout << closeInfo.code << std::endl;
|
||||||
std::cout << msg->closeInfo.reason << std::endl;
|
std::cout << closeInfo.reason << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -450,15 +348,21 @@ webSocket.setOnMessageCallback([](const ix::WebSocketMessagePtr& msg)
|
|||||||
A message will be fired when there is an error with the connection. The message type will be `ix::WebSocketMessageType::Error`. Multiple fields will be available on the event to describe the error.
|
A message will be fired when there is an error with the connection. The message type will be `ix::WebSocketMessageType::Error`. Multiple fields will be available on the event to describe the error.
|
||||||
|
|
||||||
```
|
```
|
||||||
webSocket.setOnMessageCallback([](const ix::WebSocketMessagePtr& msg)
|
webSocket.setOnMessageCallback(
|
||||||
|
[](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Error)
|
if (messageType == ix::WebSocketMessageType::Error)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Error: " << msg->errorInfo.reason << std::endl;
|
ss << "Error: " << error.reason << std::endl;
|
||||||
ss << "#retries: " << msg->eventInfo.retries << std::endl;
|
ss << "#retries: " << event.retries << std::endl;
|
||||||
ss << "Wait time(ms): " << msg->eventInfo.wait_time << std::endl;
|
ss << "Wait time(ms): " << event.wait_time << std::endl;
|
||||||
ss << "HTTP Status: " << msg->eventInfo.http_status << std::endl;
|
ss << "HTTP Status: " << event.http_status << std::endl;
|
||||||
std::cout << ss.str() << std::endl;
|
std::cout << ss.str() << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -484,12 +388,18 @@ websocket.configure(url);
|
|||||||
Ping/pong messages are used to implement keep-alive. 2 message types exists to identify ping and pong messages. Note that when a ping message is received, a pong is instantly send back as requested by the WebSocket spec.
|
Ping/pong messages are used to implement keep-alive. 2 message types exists to identify ping and pong messages. Note that when a ping message is received, a pong is instantly send back as requested by the WebSocket spec.
|
||||||
|
|
||||||
```
|
```
|
||||||
webSocket.setOnMessageCallback([](const ix::WebSocketMessagePtr& msg)
|
webSocket.setOnMessageCallback(
|
||||||
|
[](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Ping ||
|
if (messageType == ix::WebSocketMessageType::Ping ||
|
||||||
msg->type == ix::WebSocketMessageType::Pong)
|
messageType == ix::WebSocketMessageType::Pong)
|
||||||
{
|
{
|
||||||
std::cout << "pong data: " << msg->str << std::endl;
|
std::cout << "pong data: " << str << std::endl;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
@ -29,15 +29,5 @@ services:
|
|||||||
networks:
|
networks:
|
||||||
- ws-net
|
- ws-net
|
||||||
|
|
||||||
statsd:
|
|
||||||
image: jaconel/statsd
|
|
||||||
ports:
|
|
||||||
- "8125:8125"
|
|
||||||
environment:
|
|
||||||
- STATSD_DUMP_MSG=true
|
|
||||||
- GRAPHITE_HOST=127.0.0.1
|
|
||||||
networks:
|
|
||||||
- ws-net
|
|
||||||
|
|
||||||
networks:
|
networks:
|
||||||
ws-net:
|
ws-net:
|
||||||
|
@ -1,33 +0,0 @@
|
|||||||
FROM alpine as build
|
|
||||||
|
|
||||||
RUN apk add --no-cache gcc g++ musl-dev linux-headers cmake openssl-dev
|
|
||||||
RUN apk add --no-cache make
|
|
||||||
RUN apk add --no-cache zlib-dev
|
|
||||||
|
|
||||||
RUN addgroup -S app && adduser -S -G app app
|
|
||||||
RUN chown -R app:app /opt
|
|
||||||
RUN chown -R app:app /usr/local
|
|
||||||
|
|
||||||
# There is a bug in CMake where we cannot build from the root top folder
|
|
||||||
# So we build from /opt
|
|
||||||
COPY --chown=app:app . /opt
|
|
||||||
WORKDIR /opt
|
|
||||||
|
|
||||||
USER app
|
|
||||||
RUN [ "make" ]
|
|
||||||
|
|
||||||
FROM alpine as runtime
|
|
||||||
|
|
||||||
RUN apk add --no-cache libstdc++
|
|
||||||
|
|
||||||
RUN addgroup -S app && adduser -S -G app app
|
|
||||||
COPY --chown=app:app --from=build /usr/local/bin/ws /usr/local/bin/ws
|
|
||||||
RUN chmod +x /usr/local/bin/ws
|
|
||||||
RUN ldd /usr/local/bin/ws
|
|
||||||
|
|
||||||
# Now run in usermode
|
|
||||||
USER app
|
|
||||||
WORKDIR /home/app
|
|
||||||
|
|
||||||
ENTRYPOINT ["ws"]
|
|
||||||
CMD ["--help"]
|
|
@ -1,23 +0,0 @@
|
|||||||
# Build time
|
|
||||||
FROM ubuntu:bionic as build
|
|
||||||
|
|
||||||
ENV DEBIAN_FRONTEND noninteractive
|
|
||||||
RUN apt-get update
|
|
||||||
RUN apt-get -y install wget
|
|
||||||
RUN mkdir -p /tmp/cmake
|
|
||||||
WORKDIR /tmp/cmake
|
|
||||||
RUN wget https://github.com/Kitware/CMake/releases/download/v3.14.0/cmake-3.14.0-Linux-x86_64.tar.gz
|
|
||||||
RUN tar zxf cmake-3.14.0-Linux-x86_64.tar.gz
|
|
||||||
|
|
||||||
RUN apt-get -y install g++
|
|
||||||
RUN apt-get -y install libssl-dev
|
|
||||||
RUN apt-get -y install libz-dev
|
|
||||||
RUN apt-get -y install make
|
|
||||||
RUN apt-get -y install python
|
|
||||||
|
|
||||||
COPY . .
|
|
||||||
|
|
||||||
ARG CMAKE_BIN_PATH=/tmp/cmake/cmake-3.14.0-Linux-x86_64/bin
|
|
||||||
ENV PATH="${CMAKE_BIN_PATH}:${PATH}"
|
|
||||||
|
|
||||||
RUN ["make", "ws"]
|
|
@ -1,23 +0,0 @@
|
|||||||
# Build time
|
|
||||||
FROM ubuntu:disco as build
|
|
||||||
|
|
||||||
ENV DEBIAN_FRONTEND noninteractive
|
|
||||||
RUN apt-get update
|
|
||||||
RUN apt-get -y install wget
|
|
||||||
RUN mkdir -p /tmp/cmake
|
|
||||||
WORKDIR /tmp/cmake
|
|
||||||
RUN wget https://github.com/Kitware/CMake/releases/download/v3.14.0/cmake-3.14.0-Linux-x86_64.tar.gz
|
|
||||||
RUN tar zxf cmake-3.14.0-Linux-x86_64.tar.gz
|
|
||||||
|
|
||||||
RUN apt-get -y install g++
|
|
||||||
RUN apt-get -y install libssl-dev
|
|
||||||
RUN apt-get -y install libz-dev
|
|
||||||
RUN apt-get -y install make
|
|
||||||
RUN apt-get -y install python
|
|
||||||
|
|
||||||
COPY . .
|
|
||||||
|
|
||||||
ARG CMAKE_BIN_PATH=/tmp/cmake/cmake-3.14.0-Linux-x86_64/bin
|
|
||||||
ENV PATH="${CMAKE_BIN_PATH}:${PATH}"
|
|
||||||
|
|
||||||
RUN ["make", "test"]
|
|
@ -6,13 +6,14 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
#include <functional>
|
#include <functional>
|
||||||
|
#include <atomic>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
using CancellationRequest = std::function<bool()>;
|
using CancellationRequest = std::function<bool()>;
|
||||||
|
|
||||||
CancellationRequest makeCancellationRequestWithTimeout(
|
CancellationRequest makeCancellationRequestWithTimeout(int seconds,
|
||||||
int seconds, std::atomic<bool>& requestInitCancellation);
|
std::atomic<bool>& requestInitCancellation);
|
||||||
} // namespace ix
|
}
|
||||||
|
|
||||||
|
@ -6,15 +6,14 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
#include <memory>
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <atomic>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class ConnectionState
|
class ConnectionState {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
ConnectionState();
|
ConnectionState();
|
||||||
virtual ~ConnectionState() = default;
|
virtual ~ConnectionState() = default;
|
||||||
@ -33,4 +32,6 @@ namespace ix
|
|||||||
|
|
||||||
static std::atomic<uint64_t> _globalId;
|
static std::atomic<uint64_t> _globalId;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -9,20 +9,30 @@
|
|||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
const int64_t DNSLookup::kDefaultWait = 1; // ms
|
const int64_t DNSLookup::kDefaultWait = 10; // ms
|
||||||
|
|
||||||
|
std::atomic<uint64_t> DNSLookup::_nextId(0);
|
||||||
|
std::set<uint64_t> DNSLookup::_activeJobs;
|
||||||
|
std::mutex DNSLookup::_activeJobsMutex;
|
||||||
|
|
||||||
DNSLookup::DNSLookup(const std::string& hostname, int port, int64_t wait) :
|
DNSLookup::DNSLookup(const std::string& hostname, int port, int64_t wait) :
|
||||||
_hostname(hostname),
|
|
||||||
_port(port),
|
_port(port),
|
||||||
_wait(wait),
|
_wait(wait),
|
||||||
_res(nullptr),
|
_res(nullptr),
|
||||||
_done(false)
|
_done(false),
|
||||||
|
_id(_nextId++)
|
||||||
{
|
{
|
||||||
;
|
setHostname(hostname);
|
||||||
|
}
|
||||||
|
|
||||||
|
DNSLookup::~DNSLookup()
|
||||||
|
{
|
||||||
|
// Remove this job from the active jobs list
|
||||||
|
std::lock_guard<std::mutex> lock(_activeJobsMutex);
|
||||||
|
_activeJobs.erase(_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct addrinfo* DNSLookup::getAddrInfo(const std::string& hostname,
|
struct addrinfo* DNSLookup::getAddrInfo(const std::string& hostname,
|
||||||
@ -50,14 +60,14 @@ namespace ix
|
|||||||
|
|
||||||
struct addrinfo* DNSLookup::resolve(std::string& errMsg,
|
struct addrinfo* DNSLookup::resolve(std::string& errMsg,
|
||||||
const CancellationRequest& isCancellationRequested,
|
const CancellationRequest& isCancellationRequested,
|
||||||
bool cancellable)
|
bool blocking)
|
||||||
{
|
{
|
||||||
return cancellable ? resolveCancellable(errMsg, isCancellationRequested)
|
return blocking ? resolveBlocking(errMsg, isCancellationRequested)
|
||||||
: resolveUnCancellable(errMsg, isCancellationRequested);
|
: resolveAsync(errMsg, isCancellationRequested);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct addrinfo* DNSLookup::resolveUnCancellable(std::string& errMsg,
|
struct addrinfo* DNSLookup::resolveBlocking(std::string& errMsg,
|
||||||
const CancellationRequest& isCancellationRequested)
|
const CancellationRequest& isCancellationRequested)
|
||||||
{
|
{
|
||||||
errMsg = "no error";
|
errMsg = "no error";
|
||||||
|
|
||||||
@ -68,11 +78,11 @@ namespace ix
|
|||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
return getAddrInfo(_hostname, _port, errMsg);
|
return getAddrInfo(getHostname(), _port, errMsg);
|
||||||
}
|
}
|
||||||
|
|
||||||
struct addrinfo* DNSLookup::resolveCancellable(std::string& errMsg,
|
struct addrinfo* DNSLookup::resolveAsync(std::string& errMsg,
|
||||||
const CancellationRequest& isCancellationRequested)
|
const CancellationRequest& isCancellationRequested)
|
||||||
{
|
{
|
||||||
errMsg = "no error";
|
errMsg = "no error";
|
||||||
|
|
||||||
@ -84,28 +94,30 @@ namespace ix
|
|||||||
// if you need a second lookup.
|
// if you need a second lookup.
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Record job in the active Job set
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(_activeJobsMutex);
|
||||||
|
_activeJobs.insert(_id);
|
||||||
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// Good resource on thread forced termination
|
// Good resource on thread forced termination
|
||||||
// https://www.bo-yang.net/2017/11/19/cpp-kill-detached-thread
|
// https://www.bo-yang.net/2017/11/19/cpp-kill-detached-thread
|
||||||
//
|
//
|
||||||
auto ptr = shared_from_this();
|
_thread = std::thread(&DNSLookup::run, this, _id, getHostname(), _port);
|
||||||
std::weak_ptr<DNSLookup> self(ptr);
|
_thread.detach();
|
||||||
|
|
||||||
int port = _port;
|
std::unique_lock<std::mutex> lock(_conditionVariableMutex);
|
||||||
std::string hostname(_hostname);
|
|
||||||
|
|
||||||
// We make the background thread doing the work a shared pointer
|
|
||||||
// instead of a member variable, because it can keep running when
|
|
||||||
// this object goes out of scope, in case of cancellation
|
|
||||||
auto t = std::make_shared<std::thread>(&DNSLookup::run, this, self, hostname, port);
|
|
||||||
t->detach();
|
|
||||||
|
|
||||||
while (!_done)
|
while (!_done)
|
||||||
{
|
{
|
||||||
// Wait for 1 milliseconds, to see if the bg thread has terminated.
|
// Wait for 10 milliseconds on the condition variable, to see
|
||||||
// We do not use a condition variable to wait, as destroying this one
|
// if the bg thread has terminated.
|
||||||
// if the bg thread is alive can cause undefined behavior.
|
if (_condition.wait_for(lock, std::chrono::milliseconds(_wait)) == std::cv_status::no_timeout)
|
||||||
std::this_thread::sleep_for(std::chrono::milliseconds(_wait));
|
{
|
||||||
|
// Background thread has terminated, so we can break of this loop
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
// Were we cancelled ?
|
// Were we cancelled ?
|
||||||
if (isCancellationRequested && isCancellationRequested())
|
if (isCancellationRequested && isCancellationRequested())
|
||||||
@ -126,7 +138,7 @@ namespace ix
|
|||||||
return getRes();
|
return getRes();
|
||||||
}
|
}
|
||||||
|
|
||||||
void DNSLookup::run(std::weak_ptr<DNSLookup> self, std::string hostname, int port) // thread runner
|
void DNSLookup::run(uint64_t id, const std::string& hostname, int port) // thread runner
|
||||||
{
|
{
|
||||||
// We don't want to read or write into members variables of an object that could be
|
// We don't want to read or write into members variables of an object that could be
|
||||||
// gone, so we use temporary variables (res) or we pass in by copy everything that
|
// gone, so we use temporary variables (res) or we pass in by copy everything that
|
||||||
@ -134,14 +146,33 @@ namespace ix
|
|||||||
std::string errMsg;
|
std::string errMsg;
|
||||||
struct addrinfo* res = getAddrInfo(hostname, port, errMsg);
|
struct addrinfo* res = getAddrInfo(hostname, port, errMsg);
|
||||||
|
|
||||||
if (self.lock())
|
// if this isn't an active job, and the control thread is gone
|
||||||
|
// there is nothing to do, and we don't want to touch the defunct
|
||||||
|
// object data structure such as _errMsg or _condition
|
||||||
|
std::lock_guard<std::mutex> lock(_activeJobsMutex);
|
||||||
|
if (_activeJobs.count(id) == 0)
|
||||||
{
|
{
|
||||||
// Copy result into the member variables
|
return;
|
||||||
setRes(res);
|
|
||||||
setErrMsg(errMsg);
|
|
||||||
|
|
||||||
_done = true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Copy result into the member variables
|
||||||
|
setRes(res);
|
||||||
|
setErrMsg(errMsg);
|
||||||
|
|
||||||
|
_condition.notify_one();
|
||||||
|
_done = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void DNSLookup::setHostname(const std::string& hostname)
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(_hostnameMutex);
|
||||||
|
_hostname = hostname;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& DNSLookup::getHostname()
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(_hostnameMutex);
|
||||||
|
return _hostname;
|
||||||
}
|
}
|
||||||
|
|
||||||
void DNSLookup::setErrMsg(const std::string& errMsg)
|
void DNSLookup::setErrMsg(const std::string& errMsg)
|
||||||
|
@ -11,37 +11,42 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXCancellationRequest.h"
|
#include "IXCancellationRequest.h"
|
||||||
#include <atomic>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <set>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <thread>
|
||||||
|
#include <atomic>
|
||||||
|
#include <condition_variable>
|
||||||
|
#include <set>
|
||||||
|
|
||||||
struct addrinfo;
|
struct addrinfo;
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class DNSLookup : public std::enable_shared_from_this<DNSLookup>
|
class DNSLookup {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
DNSLookup(const std::string& hostname, int port, int64_t wait = DNSLookup::kDefaultWait);
|
DNSLookup(const std::string& hostname,
|
||||||
~DNSLookup() = default;
|
int port,
|
||||||
|
int64_t wait = DNSLookup::kDefaultWait);
|
||||||
|
~DNSLookup();
|
||||||
|
|
||||||
struct addrinfo* resolve(std::string& errMsg,
|
struct addrinfo* resolve(std::string& errMsg,
|
||||||
const CancellationRequest& isCancellationRequested,
|
const CancellationRequest& isCancellationRequested,
|
||||||
bool cancellable = true);
|
bool blocking = false);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
struct addrinfo* resolveCancellable(std::string& errMsg,
|
struct addrinfo* resolveAsync(std::string& errMsg,
|
||||||
const CancellationRequest& isCancellationRequested);
|
const CancellationRequest& isCancellationRequested);
|
||||||
struct addrinfo* resolveUnCancellable(std::string& errMsg,
|
struct addrinfo* resolveBlocking(std::string& errMsg,
|
||||||
const CancellationRequest& isCancellationRequested);
|
const CancellationRequest& isCancellationRequested);
|
||||||
|
|
||||||
static struct addrinfo* getAddrInfo(const std::string& hostname,
|
static struct addrinfo* getAddrInfo(const std::string& hostname,
|
||||||
int port,
|
int port,
|
||||||
std::string& errMsg);
|
std::string& errMsg);
|
||||||
|
|
||||||
void run(std::weak_ptr<DNSLookup> self, std::string hostname, int port); // thread runner
|
void run(uint64_t id, const std::string& hostname, int port); // thread runner
|
||||||
|
|
||||||
|
void setHostname(const std::string& hostname);
|
||||||
|
const std::string& getHostname();
|
||||||
|
|
||||||
void setErrMsg(const std::string& errMsg);
|
void setErrMsg(const std::string& errMsg);
|
||||||
const std::string& getErrMsg();
|
const std::string& getErrMsg();
|
||||||
@ -50,9 +55,10 @@ namespace ix
|
|||||||
struct addrinfo* getRes();
|
struct addrinfo* getRes();
|
||||||
|
|
||||||
std::string _hostname;
|
std::string _hostname;
|
||||||
|
std::mutex _hostnameMutex;
|
||||||
int _port;
|
int _port;
|
||||||
|
|
||||||
int64_t _wait;
|
int64_t _wait;
|
||||||
const static int64_t kDefaultWait;
|
|
||||||
|
|
||||||
struct addrinfo* _res;
|
struct addrinfo* _res;
|
||||||
std::mutex _resMutex;
|
std::mutex _resMutex;
|
||||||
@ -61,5 +67,15 @@ namespace ix
|
|||||||
std::mutex _errMsgMutex;
|
std::mutex _errMsgMutex;
|
||||||
|
|
||||||
std::atomic<bool> _done;
|
std::atomic<bool> _done;
|
||||||
|
std::thread _thread;
|
||||||
|
std::condition_variable _condition;
|
||||||
|
std::mutex _conditionVariableMutex;
|
||||||
|
|
||||||
|
uint64_t _id;
|
||||||
|
static std::atomic<uint64_t> _nextId;
|
||||||
|
static std::set<uint64_t> _activeJobs;
|
||||||
|
static std::mutex _activeJobsMutex;
|
||||||
|
|
||||||
|
const static int64_t kDefaultWait;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -1,138 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXHttp.cpp
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "IXHttp.h"
|
|
||||||
#include "IXCancellationRequest.h"
|
|
||||||
#include "IXSocket.h"
|
|
||||||
|
|
||||||
#include <sstream>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
std::string Http::trim(const std::string& str)
|
|
||||||
{
|
|
||||||
std::string out;
|
|
||||||
for (auto c : str)
|
|
||||||
{
|
|
||||||
if (c != ' ' && c != '\n' && c != '\r')
|
|
||||||
{
|
|
||||||
out += c;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return out;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::tuple<std::string, std::string, std::string> Http::parseRequestLine(const std::string& line)
|
|
||||||
{
|
|
||||||
// Request-Line = Method SP Request-URI SP HTTP-Version CRLF
|
|
||||||
std::string token;
|
|
||||||
std::stringstream tokenStream(line);
|
|
||||||
std::vector<std::string> tokens;
|
|
||||||
|
|
||||||
// Split by ' '
|
|
||||||
while (std::getline(tokenStream, token, ' '))
|
|
||||||
{
|
|
||||||
tokens.push_back(token);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string method;
|
|
||||||
if (tokens.size() >= 1)
|
|
||||||
{
|
|
||||||
method = trim(tokens[0]);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string requestUri;
|
|
||||||
if (tokens.size() >= 2)
|
|
||||||
{
|
|
||||||
requestUri = trim(tokens[1]);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string httpVersion;
|
|
||||||
if (tokens.size() >= 3)
|
|
||||||
{
|
|
||||||
httpVersion = trim(tokens[2]);
|
|
||||||
}
|
|
||||||
|
|
||||||
return std::make_tuple(method, requestUri, httpVersion);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::tuple<bool, std::string, HttpRequestPtr> Http::parseRequest(std::shared_ptr<Socket> socket)
|
|
||||||
{
|
|
||||||
HttpRequestPtr httpRequest;
|
|
||||||
|
|
||||||
std::atomic<bool> requestInitCancellation(false);
|
|
||||||
|
|
||||||
int timeoutSecs = 5; // FIXME
|
|
||||||
|
|
||||||
auto isCancellationRequested =
|
|
||||||
makeCancellationRequestWithTimeout(timeoutSecs, requestInitCancellation);
|
|
||||||
|
|
||||||
// Read first line
|
|
||||||
auto lineResult = socket->readLine(isCancellationRequested);
|
|
||||||
auto lineValid = lineResult.first;
|
|
||||||
auto line = lineResult.second;
|
|
||||||
|
|
||||||
if (!lineValid)
|
|
||||||
{
|
|
||||||
return std::make_tuple(false, "Error reading HTTP request line", httpRequest);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Parse request line (GET /foo HTTP/1.1\r\n)
|
|
||||||
auto requestLine = Http::parseRequestLine(line);
|
|
||||||
auto method = std::get<0>(requestLine);
|
|
||||||
auto uri = std::get<1>(requestLine);
|
|
||||||
auto httpVersion = std::get<2>(requestLine);
|
|
||||||
|
|
||||||
// Retrieve and validate HTTP headers
|
|
||||||
auto result = parseHttpHeaders(socket, isCancellationRequested);
|
|
||||||
auto headersValid = result.first;
|
|
||||||
auto headers = result.second;
|
|
||||||
|
|
||||||
if (!headersValid)
|
|
||||||
{
|
|
||||||
return std::make_tuple(false, "Error parsing HTTP headers", httpRequest);
|
|
||||||
}
|
|
||||||
|
|
||||||
httpRequest = std::make_shared<HttpRequest>(uri, method, httpVersion, headers);
|
|
||||||
return std::make_tuple(true, "", httpRequest);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Http::sendResponse(HttpResponsePtr response, std::shared_ptr<Socket> socket)
|
|
||||||
{
|
|
||||||
// Write the response to the socket
|
|
||||||
std::stringstream ss;
|
|
||||||
ss << "HTTP/1.1 ";
|
|
||||||
ss << response->statusCode;
|
|
||||||
ss << " ";
|
|
||||||
ss << response->description;
|
|
||||||
ss << "\r\n";
|
|
||||||
|
|
||||||
if (!socket->writeBytes(ss.str(), nullptr))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Write headers
|
|
||||||
ss.str("");
|
|
||||||
ss << "Content-Length: " << response->payload.size() << "\r\n";
|
|
||||||
for (auto&& it : response->headers)
|
|
||||||
{
|
|
||||||
ss << it.first << ": " << it.second << "\r\n";
|
|
||||||
}
|
|
||||||
ss << "\r\n";
|
|
||||||
|
|
||||||
if (!socket->writeBytes(ss.str(), nullptr))
|
|
||||||
{
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return response->payload.empty()
|
|
||||||
? true
|
|
||||||
: socket->writeBytes(response->payload, nullptr);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,122 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXHttp.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "IXProgressCallback.h"
|
|
||||||
#include "IXWebSocketHttpHeaders.h"
|
|
||||||
#include <tuple>
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
enum class HttpErrorCode : int
|
|
||||||
{
|
|
||||||
Ok = 0,
|
|
||||||
CannotConnect = 1,
|
|
||||||
Timeout = 2,
|
|
||||||
Gzip = 3,
|
|
||||||
UrlMalformed = 4,
|
|
||||||
CannotCreateSocket = 5,
|
|
||||||
SendError = 6,
|
|
||||||
ReadError = 7,
|
|
||||||
CannotReadStatusLine = 8,
|
|
||||||
MissingStatus = 9,
|
|
||||||
HeaderParsingError = 10,
|
|
||||||
MissingLocation = 11,
|
|
||||||
TooManyRedirects = 12,
|
|
||||||
ChunkReadError = 13,
|
|
||||||
CannotReadBody = 14,
|
|
||||||
Invalid = 100
|
|
||||||
};
|
|
||||||
|
|
||||||
struct HttpResponse
|
|
||||||
{
|
|
||||||
int statusCode;
|
|
||||||
std::string description;
|
|
||||||
HttpErrorCode errorCode;
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
std::string payload;
|
|
||||||
std::string errorMsg;
|
|
||||||
uint64_t uploadSize;
|
|
||||||
uint64_t downloadSize;
|
|
||||||
|
|
||||||
HttpResponse(int s = 0,
|
|
||||||
const std::string& des = std::string(),
|
|
||||||
const HttpErrorCode& c = HttpErrorCode::Ok,
|
|
||||||
const WebSocketHttpHeaders& h = WebSocketHttpHeaders(),
|
|
||||||
const std::string& p = std::string(),
|
|
||||||
const std::string& e = std::string(),
|
|
||||||
uint64_t u = 0,
|
|
||||||
uint64_t d = 0)
|
|
||||||
: statusCode(s)
|
|
||||||
, description(des)
|
|
||||||
, errorCode(c)
|
|
||||||
, headers(h)
|
|
||||||
, payload(p)
|
|
||||||
, errorMsg(e)
|
|
||||||
, uploadSize(u)
|
|
||||||
, downloadSize(d)
|
|
||||||
{
|
|
||||||
;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using HttpResponsePtr = std::shared_ptr<HttpResponse>;
|
|
||||||
using HttpParameters = std::map<std::string, std::string>;
|
|
||||||
using Logger = std::function<void(const std::string&)>;
|
|
||||||
using OnResponseCallback = std::function<void(const HttpResponsePtr&)>;
|
|
||||||
|
|
||||||
struct HttpRequestArgs
|
|
||||||
{
|
|
||||||
std::string url;
|
|
||||||
std::string verb;
|
|
||||||
WebSocketHttpHeaders extraHeaders;
|
|
||||||
std::string body;
|
|
||||||
int connectTimeout;
|
|
||||||
int transferTimeout;
|
|
||||||
bool followRedirects;
|
|
||||||
int maxRedirects;
|
|
||||||
bool verbose;
|
|
||||||
bool compress;
|
|
||||||
Logger logger;
|
|
||||||
OnProgressCallback onProgressCallback;
|
|
||||||
};
|
|
||||||
|
|
||||||
using HttpRequestArgsPtr = std::shared_ptr<HttpRequestArgs>;
|
|
||||||
|
|
||||||
struct HttpRequest
|
|
||||||
{
|
|
||||||
std::string uri;
|
|
||||||
std::string method;
|
|
||||||
std::string version;
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
|
|
||||||
HttpRequest(const std::string& u,
|
|
||||||
const std::string& m,
|
|
||||||
const std::string& v,
|
|
||||||
const WebSocketHttpHeaders& h = WebSocketHttpHeaders())
|
|
||||||
: uri(u)
|
|
||||||
, method(m)
|
|
||||||
, version(v)
|
|
||||||
, headers(h)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using HttpRequestPtr = std::shared_ptr<HttpRequest>;
|
|
||||||
|
|
||||||
class Http
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
static std::tuple<bool, std::string, HttpRequestPtr> parseRequest(
|
|
||||||
std::shared_ptr<Socket> socket);
|
|
||||||
static bool sendResponse(HttpResponsePtr response, std::shared_ptr<Socket> socket);
|
|
||||||
|
|
||||||
static std::tuple<std::string, std::string, std::string> parseRequestLine(
|
|
||||||
const std::string& line);
|
|
||||||
static std::string trim(const std::string& str);
|
|
||||||
};
|
|
||||||
} // namespace ix
|
|
@ -14,7 +14,6 @@
|
|||||||
#include <vector>
|
#include <vector>
|
||||||
#include <cstring>
|
#include <cstring>
|
||||||
|
|
||||||
#include <assert.h>
|
|
||||||
#include <zlib.h>
|
#include <zlib.h>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
@ -22,106 +21,29 @@ namespace ix
|
|||||||
const std::string HttpClient::kPost = "POST";
|
const std::string HttpClient::kPost = "POST";
|
||||||
const std::string HttpClient::kGet = "GET";
|
const std::string HttpClient::kGet = "GET";
|
||||||
const std::string HttpClient::kHead = "HEAD";
|
const std::string HttpClient::kHead = "HEAD";
|
||||||
const std::string HttpClient::kDel = "DEL";
|
|
||||||
const std::string HttpClient::kPut = "PUT";
|
|
||||||
|
|
||||||
HttpClient::HttpClient(bool async) : _async(async), _stop(false)
|
HttpClient::HttpClient()
|
||||||
{
|
{
|
||||||
if (!_async) return;
|
|
||||||
|
|
||||||
_thread = std::thread(&HttpClient::run, this);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpClient::~HttpClient()
|
HttpClient::~HttpClient()
|
||||||
{
|
{
|
||||||
if (!_thread.joinable()) return;
|
|
||||||
|
|
||||||
_stop = true;
|
|
||||||
_condition.notify_one();
|
|
||||||
_thread.join();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpRequestArgsPtr HttpClient::createRequest(const std::string& url,
|
HttpResponse HttpClient::request(
|
||||||
const std::string& verb)
|
|
||||||
{
|
|
||||||
auto request = std::make_shared<HttpRequestArgs>();
|
|
||||||
request->url = url;
|
|
||||||
request->verb = verb;
|
|
||||||
return request;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool HttpClient::performRequest(HttpRequestArgsPtr args,
|
|
||||||
const OnResponseCallback& onResponseCallback)
|
|
||||||
{
|
|
||||||
assert(_async && "HttpClient needs its async parameter set to true "
|
|
||||||
"in order to call performRequest");
|
|
||||||
if (!_async) return false;
|
|
||||||
|
|
||||||
// Enqueue the task
|
|
||||||
{
|
|
||||||
// acquire lock
|
|
||||||
std::unique_lock<std::mutex> lock(_queueMutex);
|
|
||||||
|
|
||||||
// add the task
|
|
||||||
_queue.push(std::make_pair(args, onResponseCallback));
|
|
||||||
} // release lock
|
|
||||||
|
|
||||||
// wake up one thread
|
|
||||||
_condition.notify_one();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HttpClient::run()
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
HttpRequestArgsPtr args;
|
|
||||||
OnResponseCallback onResponseCallback;
|
|
||||||
|
|
||||||
{
|
|
||||||
std::unique_lock<std::mutex> lock(_queueMutex);
|
|
||||||
|
|
||||||
while (!_stop && _queue.empty())
|
|
||||||
{
|
|
||||||
_condition.wait(lock);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (_stop) return;
|
|
||||||
|
|
||||||
auto p = _queue.front();
|
|
||||||
_queue.pop();
|
|
||||||
|
|
||||||
args = p.first;
|
|
||||||
onResponseCallback = p.second;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (_stop) return;
|
|
||||||
|
|
||||||
HttpResponsePtr response = request(args->url, args->verb, args->body, args);
|
|
||||||
onResponseCallback(response);
|
|
||||||
|
|
||||||
if (_stop) return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::request(
|
|
||||||
const std::string& url,
|
const std::string& url,
|
||||||
const std::string& verb,
|
const std::string& verb,
|
||||||
const std::string& body,
|
const std::string& body,
|
||||||
HttpRequestArgsPtr args,
|
const HttpRequestArgs& args,
|
||||||
int redirects)
|
int redirects)
|
||||||
{
|
{
|
||||||
// We only have one socket connection, so we cannot
|
|
||||||
// make multiple requests concurrently.
|
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
|
||||||
|
|
||||||
uint64_t uploadSize = 0;
|
uint64_t uploadSize = 0;
|
||||||
uint64_t downloadSize = 0;
|
uint64_t downloadSize = 0;
|
||||||
int code = 0;
|
int code = 0;
|
||||||
WebSocketHttpHeaders headers;
|
WebSocketHttpHeaders headers;
|
||||||
std::string payload;
|
std::string payload;
|
||||||
std::string description;
|
|
||||||
|
|
||||||
std::string protocol, host, path, query;
|
std::string protocol, host, path, query;
|
||||||
int port;
|
int port;
|
||||||
@ -130,9 +52,9 @@ namespace ix
|
|||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Cannot parse url: " << url;
|
ss << "Cannot parse url: " << url;
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::UrlMalformed,
|
return std::make_tuple(code, HttpErrorCode::UrlMalformed,
|
||||||
headers, payload, ss.str(),
|
headers, payload, ss.str(),
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool tls = protocol == "https";
|
bool tls = protocol == "https";
|
||||||
@ -141,45 +63,35 @@ namespace ix
|
|||||||
|
|
||||||
if (!_socket)
|
if (!_socket)
|
||||||
{
|
{
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::CannotCreateSocket,
|
return std::make_tuple(code, HttpErrorCode::CannotCreateSocket,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build request string
|
// Build request string
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << verb << " " << path << " HTTP/1.1\r\n";
|
ss << verb << " " << path << " HTTP/1.1\r\n";
|
||||||
ss << "Host: " << host << "\r\n";
|
ss << "Host: " << host << "\r\n";
|
||||||
|
ss << "User-Agent: ixwebsocket/1.0.0" << "\r\n";
|
||||||
|
ss << "Accept: */*" << "\r\n";
|
||||||
|
|
||||||
if (args->compress)
|
if (args.compress)
|
||||||
{
|
{
|
||||||
ss << "Accept-Encoding: gzip" << "\r\n";
|
ss << "Accept-Encoding: gzip" << "\r\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Append extra headers
|
// Append extra headers
|
||||||
for (auto&& it : args->extraHeaders)
|
for (auto&& it : args.extraHeaders)
|
||||||
{
|
{
|
||||||
ss << it.first << ": " << it.second << "\r\n";
|
ss << it.first << ": " << it.second << "\r\n";
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set a default Accept header if none is present
|
if (verb == kPost)
|
||||||
if (headers.find("Accept") == headers.end())
|
|
||||||
{
|
|
||||||
ss << "Accept: */*" << "\r\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set a default User agent if none is present
|
|
||||||
if (headers.find("User-Agent") == headers.end())
|
|
||||||
{
|
|
||||||
ss << "User-Agent: ixwebsocket" << "\r\n";
|
|
||||||
}
|
|
||||||
|
|
||||||
if (verb == kPost || verb == kPut)
|
|
||||||
{
|
{
|
||||||
ss << "Content-Length: " << body.size() << "\r\n";
|
ss << "Content-Length: " << body.size() << "\r\n";
|
||||||
|
|
||||||
// Set default Content-Type if unspecified
|
// Set default Content-Type if unspecified
|
||||||
if (args->extraHeaders.find("Content-Type") == args->extraHeaders.end())
|
if (args.extraHeaders.find("Content-Type") == args.extraHeaders.end())
|
||||||
{
|
{
|
||||||
ss << "Content-Type: application/x-www-form-urlencoded" << "\r\n";
|
ss << "Content-Type: application/x-www-form-urlencoded" << "\r\n";
|
||||||
}
|
}
|
||||||
@ -197,23 +109,23 @@ namespace ix
|
|||||||
|
|
||||||
// Make a cancellation object dealing with connection timeout
|
// Make a cancellation object dealing with connection timeout
|
||||||
auto isCancellationRequested =
|
auto isCancellationRequested =
|
||||||
makeCancellationRequestWithTimeout(args->connectTimeout, requestInitCancellation);
|
makeCancellationRequestWithTimeout(args.connectTimeout, requestInitCancellation);
|
||||||
|
|
||||||
bool success = _socket->connect(host, port, errMsg, isCancellationRequested);
|
bool success = _socket->connect(host, port, errMsg, isCancellationRequested);
|
||||||
if (!success)
|
if (!success)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Cannot connect to url: " << url << " / error : " << errMsg;
|
ss << "Cannot connect to url: " << url;
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::CannotConnect,
|
return std::make_tuple(code, HttpErrorCode::CannotConnect,
|
||||||
headers, payload, ss.str(),
|
headers, payload, ss.str(),
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make a new cancellation object dealing with transfer timeout
|
// Make a new cancellation object dealing with transfer timeout
|
||||||
isCancellationRequested =
|
isCancellationRequested =
|
||||||
makeCancellationRequestWithTimeout(args->transferTimeout, requestInitCancellation);
|
makeCancellationRequestWithTimeout(args.transferTimeout, requestInitCancellation);
|
||||||
|
|
||||||
if (args->verbose)
|
if (args.verbose)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Sending " << verb << " request "
|
ss << "Sending " << verb << " request "
|
||||||
@ -230,9 +142,9 @@ namespace ix
|
|||||||
if (!_socket->writeBytes(req, isCancellationRequested))
|
if (!_socket->writeBytes(req, isCancellationRequested))
|
||||||
{
|
{
|
||||||
std::string errorMsg("Cannot send request");
|
std::string errorMsg("Cannot send request");
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::SendError,
|
return std::make_tuple(code, HttpErrorCode::SendError,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
uploadSize = req.size();
|
uploadSize = req.size();
|
||||||
@ -244,12 +156,12 @@ namespace ix
|
|||||||
if (!lineValid)
|
if (!lineValid)
|
||||||
{
|
{
|
||||||
std::string errorMsg("Cannot retrieve status line");
|
std::string errorMsg("Cannot retrieve status line");
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::CannotReadStatusLine,
|
return std::make_tuple(code, HttpErrorCode::CannotReadStatusLine,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (args->verbose)
|
if (args.verbose)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Status line " << line;
|
ss << "Status line " << line;
|
||||||
@ -259,9 +171,9 @@ namespace ix
|
|||||||
if (sscanf(line.c_str(), "HTTP/1.1 %d", &code) != 1)
|
if (sscanf(line.c_str(), "HTTP/1.1 %d", &code) != 1)
|
||||||
{
|
{
|
||||||
std::string errorMsg("Cannot parse response code from status line");
|
std::string errorMsg("Cannot parse response code from status line");
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::MissingStatus,
|
return std::make_tuple(code, HttpErrorCode::MissingStatus,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
auto result = parseHttpHeaders(_socket, isCancellationRequested);
|
auto result = parseHttpHeaders(_socket, isCancellationRequested);
|
||||||
@ -271,29 +183,29 @@ namespace ix
|
|||||||
if (!headersValid)
|
if (!headersValid)
|
||||||
{
|
{
|
||||||
std::string errorMsg("Cannot parse http headers");
|
std::string errorMsg("Cannot parse http headers");
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::HeaderParsingError,
|
return std::make_tuple(code, HttpErrorCode::HeaderParsingError,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Redirect ?
|
// Redirect ?
|
||||||
if ((code >= 301 && code <= 308) && args->followRedirects)
|
if ((code >= 301 && code <= 308) && args.followRedirects)
|
||||||
{
|
{
|
||||||
if (headers.find("Location") == headers.end())
|
if (headers.find("Location") == headers.end())
|
||||||
{
|
{
|
||||||
std::string errorMsg("Missing location header for redirect");
|
std::string errorMsg("Missing location header for redirect");
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::MissingLocation,
|
return std::make_tuple(code, HttpErrorCode::MissingLocation,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (redirects >= args->maxRedirects)
|
if (redirects >= args.maxRedirects)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Too many redirects: " << redirects;
|
ss << "Too many redirects: " << redirects;
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::TooManyRedirects,
|
return std::make_tuple(code, HttpErrorCode::TooManyRedirects,
|
||||||
headers, payload, ss.str(),
|
headers, payload, ss.str(),
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Recurse
|
// Recurse
|
||||||
@ -303,9 +215,9 @@ namespace ix
|
|||||||
|
|
||||||
if (verb == "HEAD")
|
if (verb == "HEAD")
|
||||||
{
|
{
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::Ok,
|
return std::make_tuple(code, HttpErrorCode::Ok,
|
||||||
headers, payload, std::string(),
|
headers, payload, std::string(),
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Parse response:
|
// Parse response:
|
||||||
@ -319,14 +231,14 @@ namespace ix
|
|||||||
payload.reserve(contentLength);
|
payload.reserve(contentLength);
|
||||||
|
|
||||||
auto chunkResult = _socket->readBytes(contentLength,
|
auto chunkResult = _socket->readBytes(contentLength,
|
||||||
args->onProgressCallback,
|
args.onProgressCallback,
|
||||||
isCancellationRequested);
|
isCancellationRequested);
|
||||||
if (!chunkResult.first)
|
if (!chunkResult.first)
|
||||||
{
|
{
|
||||||
errorMsg = "Cannot read chunk";
|
errorMsg = "Cannot read chunk";
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::ChunkReadError,
|
return std::make_tuple(code, HttpErrorCode::ChunkReadError,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
payload += chunkResult.second;
|
payload += chunkResult.second;
|
||||||
}
|
}
|
||||||
@ -342,9 +254,9 @@ namespace ix
|
|||||||
|
|
||||||
if (!lineResult.first)
|
if (!lineResult.first)
|
||||||
{
|
{
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::ChunkReadError,
|
return std::make_tuple(code, HttpErrorCode::ChunkReadError,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
uint64_t chunkSize;
|
uint64_t chunkSize;
|
||||||
@ -352,7 +264,7 @@ namespace ix
|
|||||||
ss << std::hex << line;
|
ss << std::hex << line;
|
||||||
ss >> chunkSize;
|
ss >> chunkSize;
|
||||||
|
|
||||||
if (args->verbose)
|
if (args.verbose)
|
||||||
{
|
{
|
||||||
std::stringstream oss;
|
std::stringstream oss;
|
||||||
oss << "Reading " << chunkSize << " bytes"
|
oss << "Reading " << chunkSize << " bytes"
|
||||||
@ -360,18 +272,18 @@ namespace ix
|
|||||||
log(oss.str(), args);
|
log(oss.str(), args);
|
||||||
}
|
}
|
||||||
|
|
||||||
payload.reserve(payload.size() + (size_t) chunkSize);
|
payload.reserve(payload.size() + chunkSize);
|
||||||
|
|
||||||
// Read a chunk
|
// Read a chunk
|
||||||
auto chunkResult = _socket->readBytes((size_t) chunkSize,
|
auto chunkResult = _socket->readBytes(chunkSize,
|
||||||
args->onProgressCallback,
|
args.onProgressCallback,
|
||||||
isCancellationRequested);
|
isCancellationRequested);
|
||||||
if (!chunkResult.first)
|
if (!chunkResult.first)
|
||||||
{
|
{
|
||||||
errorMsg = "Cannot read chunk";
|
errorMsg = "Cannot read chunk";
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::ChunkReadError,
|
return std::make_tuple(code, HttpErrorCode::ChunkReadError,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
payload += chunkResult.second;
|
payload += chunkResult.second;
|
||||||
|
|
||||||
@ -380,9 +292,9 @@ namespace ix
|
|||||||
|
|
||||||
if (!lineResult.first)
|
if (!lineResult.first)
|
||||||
{
|
{
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::ChunkReadError,
|
return std::make_tuple(code, HttpErrorCode::ChunkReadError,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (chunkSize == 0) break;
|
if (chunkSize == 0) break;
|
||||||
@ -395,9 +307,9 @@ namespace ix
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
std::string errorMsg("Cannot read http body");
|
std::string errorMsg("Cannot read http body");
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::CannotReadBody,
|
return std::make_tuple(code, HttpErrorCode::CannotReadBody,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
downloadSize = payload.size();
|
downloadSize = payload.size();
|
||||||
@ -409,64 +321,44 @@ namespace ix
|
|||||||
if (!gzipInflate(payload, decompressedPayload))
|
if (!gzipInflate(payload, decompressedPayload))
|
||||||
{
|
{
|
||||||
std::string errorMsg("Error decompressing payload");
|
std::string errorMsg("Error decompressing payload");
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::Gzip,
|
return std::make_tuple(code, HttpErrorCode::Gzip,
|
||||||
headers, payload, errorMsg,
|
headers, payload, errorMsg,
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
payload = decompressedPayload;
|
payload = decompressedPayload;
|
||||||
}
|
}
|
||||||
|
|
||||||
return std::make_shared<HttpResponse>(code, description, HttpErrorCode::Ok,
|
return std::make_tuple(code, HttpErrorCode::Ok,
|
||||||
headers, payload, std::string(),
|
headers, payload, std::string(),
|
||||||
uploadSize, downloadSize);
|
uploadSize, downloadSize);
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::get(const std::string& url,
|
HttpResponse HttpClient::get(const std::string& url,
|
||||||
HttpRequestArgsPtr args)
|
const HttpRequestArgs& args)
|
||||||
{
|
{
|
||||||
return request(url, kGet, std::string(), args);
|
return request(url, kGet, std::string(), args);
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::head(const std::string& url,
|
HttpResponse HttpClient::head(const std::string& url,
|
||||||
HttpRequestArgsPtr args)
|
const HttpRequestArgs& args)
|
||||||
{
|
{
|
||||||
return request(url, kHead, std::string(), args);
|
return request(url, kHead, std::string(), args);
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::del(const std::string& url,
|
HttpResponse HttpClient::post(const std::string& url,
|
||||||
HttpRequestArgsPtr args)
|
const HttpParameters& httpParameters,
|
||||||
{
|
const HttpRequestArgs& args)
|
||||||
return request(url, kDel, std::string(), args);
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::post(const std::string& url,
|
|
||||||
const HttpParameters& httpParameters,
|
|
||||||
HttpRequestArgsPtr args)
|
|
||||||
{
|
{
|
||||||
return request(url, kPost, serializeHttpParameters(httpParameters), args);
|
return request(url, kPost, serializeHttpParameters(httpParameters), args);
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::post(const std::string& url,
|
HttpResponse HttpClient::post(const std::string& url,
|
||||||
const std::string& body,
|
const std::string& body,
|
||||||
HttpRequestArgsPtr args)
|
const HttpRequestArgs& args)
|
||||||
{
|
{
|
||||||
return request(url, kPost, body, args);
|
return request(url, kPost, body, args);
|
||||||
}
|
}
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::put(const std::string& url,
|
|
||||||
const HttpParameters& httpParameters,
|
|
||||||
HttpRequestArgsPtr args)
|
|
||||||
{
|
|
||||||
return request(url, kPut, serializeHttpParameters(httpParameters), args);
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpResponsePtr HttpClient::put(const std::string& url,
|
|
||||||
const std::string& body,
|
|
||||||
const HttpRequestArgsPtr args)
|
|
||||||
{
|
|
||||||
return request(url, kPut, body, args);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string HttpClient::urlEncode(const std::string& value)
|
std::string HttpClient::urlEncode(const std::string& value)
|
||||||
{
|
{
|
||||||
std::ostringstream escaped;
|
std::ostringstream escaped;
|
||||||
@ -564,11 +456,11 @@ namespace ix
|
|||||||
}
|
}
|
||||||
|
|
||||||
void HttpClient::log(const std::string& msg,
|
void HttpClient::log(const std::string& msg,
|
||||||
HttpRequestArgsPtr args)
|
const HttpRequestArgs& args)
|
||||||
{
|
{
|
||||||
if (args->logger)
|
if (args.logger)
|
||||||
{
|
{
|
||||||
args->logger(msg);
|
args.logger(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,85 +6,102 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXHttp.h"
|
#include <algorithm>
|
||||||
|
#include <functional>
|
||||||
|
#include <mutex>
|
||||||
|
#include <atomic>
|
||||||
|
#include <tuple>
|
||||||
|
#include <memory>
|
||||||
|
#include <map>
|
||||||
|
|
||||||
#include "IXSocket.h"
|
#include "IXSocket.h"
|
||||||
#include "IXWebSocketHttpHeaders.h"
|
#include "IXWebSocketHttpHeaders.h"
|
||||||
#include <algorithm>
|
|
||||||
#include <atomic>
|
|
||||||
#include <condition_variable>
|
|
||||||
#include <functional>
|
|
||||||
#include <map>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <queue>
|
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class HttpClient
|
enum class HttpErrorCode : int
|
||||||
{
|
{
|
||||||
|
Ok = 0,
|
||||||
|
CannotConnect = 1,
|
||||||
|
Timeout = 2,
|
||||||
|
Gzip = 3,
|
||||||
|
UrlMalformed = 4,
|
||||||
|
CannotCreateSocket = 5,
|
||||||
|
SendError = 6,
|
||||||
|
ReadError = 7,
|
||||||
|
CannotReadStatusLine = 8,
|
||||||
|
MissingStatus = 9,
|
||||||
|
HeaderParsingError = 10,
|
||||||
|
MissingLocation = 11,
|
||||||
|
TooManyRedirects = 12,
|
||||||
|
ChunkReadError = 13,
|
||||||
|
CannotReadBody = 14
|
||||||
|
};
|
||||||
|
|
||||||
|
using HttpResponse = std::tuple<int, // status
|
||||||
|
HttpErrorCode, // error code
|
||||||
|
WebSocketHttpHeaders,
|
||||||
|
std::string, // payload
|
||||||
|
std::string, // error msg
|
||||||
|
uint64_t, // upload size
|
||||||
|
uint64_t>; // download size
|
||||||
|
|
||||||
|
using HttpParameters = std::map<std::string, std::string>;
|
||||||
|
using Logger = std::function<void(const std::string&)>;
|
||||||
|
|
||||||
|
struct HttpRequestArgs
|
||||||
|
{
|
||||||
|
std::string url;
|
||||||
|
WebSocketHttpHeaders extraHeaders;
|
||||||
|
std::string body;
|
||||||
|
int connectTimeout;
|
||||||
|
int transferTimeout;
|
||||||
|
bool followRedirects;
|
||||||
|
int maxRedirects;
|
||||||
|
bool verbose;
|
||||||
|
bool compress;
|
||||||
|
Logger logger;
|
||||||
|
OnProgressCallback onProgressCallback;
|
||||||
|
};
|
||||||
|
|
||||||
|
class HttpClient {
|
||||||
public:
|
public:
|
||||||
HttpClient(bool async = false);
|
HttpClient();
|
||||||
~HttpClient();
|
~HttpClient();
|
||||||
|
|
||||||
HttpResponsePtr get(const std::string& url, HttpRequestArgsPtr args);
|
HttpResponse get(const std::string& url,
|
||||||
HttpResponsePtr head(const std::string& url, HttpRequestArgsPtr args);
|
const HttpRequestArgs& args);
|
||||||
HttpResponsePtr del(const std::string& url, HttpRequestArgsPtr args);
|
HttpResponse head(const std::string& url,
|
||||||
|
const HttpRequestArgs& args);
|
||||||
|
|
||||||
HttpResponsePtr post(const std::string& url,
|
HttpResponse post(const std::string& url,
|
||||||
const HttpParameters& httpParameters,
|
const HttpParameters& httpParameters,
|
||||||
HttpRequestArgsPtr args);
|
const HttpRequestArgs& args);
|
||||||
HttpResponsePtr post(const std::string& url,
|
HttpResponse post(const std::string& url,
|
||||||
|
const std::string& body,
|
||||||
|
const HttpRequestArgs& args);
|
||||||
|
|
||||||
|
private:
|
||||||
|
HttpResponse request(const std::string& url,
|
||||||
|
const std::string& verb,
|
||||||
const std::string& body,
|
const std::string& body,
|
||||||
HttpRequestArgsPtr args);
|
const HttpRequestArgs& args,
|
||||||
|
int redirects = 0);
|
||||||
HttpResponsePtr put(const std::string& url,
|
|
||||||
const HttpParameters& httpParameters,
|
|
||||||
HttpRequestArgsPtr args);
|
|
||||||
HttpResponsePtr put(const std::string& url,
|
|
||||||
const std::string& body,
|
|
||||||
HttpRequestArgsPtr args);
|
|
||||||
|
|
||||||
HttpResponsePtr request(const std::string& url,
|
|
||||||
const std::string& verb,
|
|
||||||
const std::string& body,
|
|
||||||
HttpRequestArgsPtr args,
|
|
||||||
int redirects = 0);
|
|
||||||
|
|
||||||
// Async API
|
|
||||||
HttpRequestArgsPtr createRequest(const std::string& url = std::string(),
|
|
||||||
const std::string& verb = HttpClient::kGet);
|
|
||||||
|
|
||||||
bool performRequest(HttpRequestArgsPtr request,
|
|
||||||
const OnResponseCallback& onResponseCallback);
|
|
||||||
|
|
||||||
std::string serializeHttpParameters(const HttpParameters& httpParameters);
|
std::string serializeHttpParameters(const HttpParameters& httpParameters);
|
||||||
|
|
||||||
std::string urlEncode(const std::string& value);
|
std::string urlEncode(const std::string& value);
|
||||||
|
|
||||||
|
void log(const std::string& msg, const HttpRequestArgs& args);
|
||||||
|
|
||||||
|
bool gzipInflate(
|
||||||
|
const std::string& in,
|
||||||
|
std::string& out);
|
||||||
|
|
||||||
|
std::shared_ptr<Socket> _socket;
|
||||||
|
|
||||||
const static std::string kPost;
|
const static std::string kPost;
|
||||||
const static std::string kGet;
|
const static std::string kGet;
|
||||||
const static std::string kHead;
|
const static std::string kHead;
|
||||||
const static std::string kDel;
|
|
||||||
const static std::string kPut;
|
|
||||||
|
|
||||||
private:
|
|
||||||
void log(const std::string& msg, HttpRequestArgsPtr args);
|
|
||||||
|
|
||||||
bool gzipInflate(const std::string& in, std::string& out);
|
|
||||||
|
|
||||||
// Async API background thread runner
|
|
||||||
void run();
|
|
||||||
|
|
||||||
// Async API
|
|
||||||
bool _async;
|
|
||||||
std::queue<std::pair<HttpRequestArgsPtr, OnResponseCallback>> _queue;
|
|
||||||
mutable std::mutex _queueMutex;
|
|
||||||
std::condition_variable _condition;
|
|
||||||
std::atomic<bool> _stop;
|
|
||||||
std::thread _thread;
|
|
||||||
|
|
||||||
std::shared_ptr<Socket> _socket;
|
|
||||||
std::mutex _mutex; // to protect accessing the _socket (only one socket per client)
|
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -1,158 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXHttpServer.cpp
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "IXHttpServer.h"
|
|
||||||
#include "IXSocketConnect.h"
|
|
||||||
#include "IXSocketFactory.h"
|
|
||||||
#include "IXNetSystem.h"
|
|
||||||
|
|
||||||
#include <iostream>
|
|
||||||
#include <sstream>
|
|
||||||
#include <fstream>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
namespace
|
|
||||||
{
|
|
||||||
std::pair<bool, std::vector<uint8_t>> load(const std::string& path)
|
|
||||||
{
|
|
||||||
std::vector<uint8_t> memblock;
|
|
||||||
|
|
||||||
std::ifstream file(path);
|
|
||||||
if (!file.is_open()) return std::make_pair(false, memblock);
|
|
||||||
|
|
||||||
file.seekg(0, file.end);
|
|
||||||
std::streamoff size = file.tellg();
|
|
||||||
file.seekg(0, file.beg);
|
|
||||||
|
|
||||||
memblock.resize((size_t) size);
|
|
||||||
file.read((char*)&memblock.front(), static_cast<std::streamsize>(size));
|
|
||||||
|
|
||||||
return std::make_pair(true, memblock);
|
|
||||||
}
|
|
||||||
|
|
||||||
std::pair<bool, std::string> readAsString(const std::string& path)
|
|
||||||
{
|
|
||||||
auto res = load(path);
|
|
||||||
auto vec = res.second;
|
|
||||||
return std::make_pair(res.first, std::string(vec.begin(), vec.end()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
HttpServer::HttpServer(int port,
|
|
||||||
const std::string& host,
|
|
||||||
int backlog,
|
|
||||||
size_t maxConnections) : SocketServer(port, host, backlog, maxConnections),
|
|
||||||
_connectedClientsCount(0)
|
|
||||||
{
|
|
||||||
setDefaultConnectionCallback();
|
|
||||||
}
|
|
||||||
|
|
||||||
HttpServer::~HttpServer()
|
|
||||||
{
|
|
||||||
stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
void HttpServer::stop()
|
|
||||||
{
|
|
||||||
stopAcceptingConnections();
|
|
||||||
|
|
||||||
// FIXME: cancelling / closing active clients ...
|
|
||||||
|
|
||||||
SocketServer::stop();
|
|
||||||
}
|
|
||||||
|
|
||||||
void HttpServer::setOnConnectionCallback(const OnConnectionCallback& callback)
|
|
||||||
{
|
|
||||||
_onConnectionCallback = callback;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HttpServer::handleConnection(
|
|
||||||
int fd,
|
|
||||||
std::shared_ptr<ConnectionState> connectionState)
|
|
||||||
{
|
|
||||||
_connectedClientsCount++;
|
|
||||||
|
|
||||||
std::string errorMsg;
|
|
||||||
auto socket = createSocket(fd, errorMsg);
|
|
||||||
|
|
||||||
// Set the socket to non blocking mode + other tweaks
|
|
||||||
SocketConnect::configure(fd);
|
|
||||||
|
|
||||||
auto ret = Http::parseRequest(socket);
|
|
||||||
// FIXME: handle errors in parseRequest
|
|
||||||
|
|
||||||
if (std::get<0>(ret))
|
|
||||||
{
|
|
||||||
auto response = _onConnectionCallback(std::get<2>(ret), connectionState);
|
|
||||||
if (!Http::sendResponse(response, socket))
|
|
||||||
{
|
|
||||||
logError("Cannot send response");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
connectionState->setTerminated();
|
|
||||||
|
|
||||||
_connectedClientsCount--;
|
|
||||||
}
|
|
||||||
|
|
||||||
size_t HttpServer::getConnectedClientsCount()
|
|
||||||
{
|
|
||||||
return _connectedClientsCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
void HttpServer::setDefaultConnectionCallback()
|
|
||||||
{
|
|
||||||
setOnConnectionCallback(
|
|
||||||
[this](HttpRequestPtr request,
|
|
||||||
std::shared_ptr<ConnectionState> /*connectionState*/) -> HttpResponsePtr
|
|
||||||
{
|
|
||||||
std::string uri(request->uri);
|
|
||||||
if (uri.empty() || uri == "/")
|
|
||||||
{
|
|
||||||
uri = "/index.html";
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string path("." + uri);
|
|
||||||
auto res = readAsString(path);
|
|
||||||
bool found = res.first;
|
|
||||||
if (!found)
|
|
||||||
{
|
|
||||||
return std::make_shared<HttpResponse>(404, "Not Found",
|
|
||||||
HttpErrorCode::Ok,
|
|
||||||
WebSocketHttpHeaders(),
|
|
||||||
std::string());
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string content = res.second;
|
|
||||||
|
|
||||||
// Log request
|
|
||||||
std::stringstream ss;
|
|
||||||
ss << request->method
|
|
||||||
<< " "
|
|
||||||
<< request->uri
|
|
||||||
<< " "
|
|
||||||
<< content.size();
|
|
||||||
logInfo(ss.str());
|
|
||||||
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
// FIXME: check extensions to set the content type
|
|
||||||
// headers["Content-Type"] = "application/octet-stream";
|
|
||||||
headers["Accept-Ranges"] = "none";
|
|
||||||
|
|
||||||
for (auto&& it : request->headers)
|
|
||||||
{
|
|
||||||
headers[it.first] = it.second;
|
|
||||||
}
|
|
||||||
|
|
||||||
return std::make_shared<HttpResponse>(200, "OK",
|
|
||||||
HttpErrorCode::Ok,
|
|
||||||
headers,
|
|
||||||
content);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,49 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXHttpServer.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2018 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "IXHttp.h"
|
|
||||||
#include "IXSocketServer.h"
|
|
||||||
#include "IXWebSocket.h"
|
|
||||||
#include <functional>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <set>
|
|
||||||
#include <string>
|
|
||||||
#include <thread>
|
|
||||||
#include <utility> // pair
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
class HttpServer final : public SocketServer
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
using OnConnectionCallback =
|
|
||||||
std::function<HttpResponsePtr(HttpRequestPtr, std::shared_ptr<ConnectionState>)>;
|
|
||||||
|
|
||||||
HttpServer(int port = SocketServer::kDefaultPort,
|
|
||||||
const std::string& host = SocketServer::kDefaultHost,
|
|
||||||
int backlog = SocketServer::kDefaultTcpBacklog,
|
|
||||||
size_t maxConnections = SocketServer::kDefaultMaxConnections);
|
|
||||||
virtual ~HttpServer();
|
|
||||||
virtual void stop() final;
|
|
||||||
|
|
||||||
void setOnConnectionCallback(const OnConnectionCallback& callback);
|
|
||||||
|
|
||||||
private:
|
|
||||||
// Member variables
|
|
||||||
OnConnectionCallback _onConnectionCallback;
|
|
||||||
std::atomic<int> _connectedClientsCount;
|
|
||||||
|
|
||||||
// Methods
|
|
||||||
virtual void handleConnection(int fd,
|
|
||||||
std::shared_ptr<ConnectionState> connectionState) final;
|
|
||||||
virtual size_t getConnectedClientsCount() final;
|
|
||||||
|
|
||||||
void setDefaultConnectionCallback();
|
|
||||||
};
|
|
||||||
} // namespace ix
|
|
@ -15,8 +15,9 @@ namespace ix
|
|||||||
WSADATA wsaData;
|
WSADATA wsaData;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
// Use the MAKEWORD(lowbyte, highbyte) macro declared in Windef.h
|
/* Use the MAKEWORD(lowbyte, highbyte) macro declared in Windef.h */
|
||||||
wVersionRequested = MAKEWORD(2, 2);
|
wVersionRequested = MAKEWORD(2, 2);
|
||||||
|
|
||||||
err = WSAStartup(wVersionRequested, &wsaData);
|
err = WSAStartup(wVersionRequested, &wsaData);
|
||||||
|
|
||||||
return err == 0;
|
return err == 0;
|
||||||
@ -29,83 +30,10 @@ namespace ix
|
|||||||
{
|
{
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
int err = WSACleanup();
|
int err = WSACleanup();
|
||||||
|
|
||||||
return err == 0;
|
return err == 0;
|
||||||
#else
|
#else
|
||||||
return true;
|
return true;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// This function should be in the global namespace
|
|
||||||
#ifdef _WIN32
|
|
||||||
//
|
|
||||||
// That function could 'return WSAPoll(pfd, nfds, timeout);'
|
|
||||||
// but WSAPoll is said to have weird behaviors on the internet
|
|
||||||
// (the curl folks have had problems with it).
|
|
||||||
//
|
|
||||||
// So we make it a select wrapper
|
|
||||||
//
|
|
||||||
int poll(struct pollfd *fds, nfds_t nfds, int timeout)
|
|
||||||
{
|
|
||||||
int maxfd = 0;
|
|
||||||
fd_set readfds, writefds, errorfds;
|
|
||||||
FD_ZERO(&readfds);
|
|
||||||
FD_ZERO(&writefds);
|
|
||||||
FD_ZERO(&errorfds);
|
|
||||||
|
|
||||||
for (nfds_t i = 0; i < nfds; ++i)
|
|
||||||
{
|
|
||||||
struct pollfd *fd = &fds[i];
|
|
||||||
|
|
||||||
if (fd->fd > maxfd)
|
|
||||||
{
|
|
||||||
maxfd = fd->fd;
|
|
||||||
}
|
|
||||||
if ((fd->events & POLLIN))
|
|
||||||
{
|
|
||||||
FD_SET(fd->fd, &readfds);
|
|
||||||
}
|
|
||||||
if ((fd->events & POLLOUT))
|
|
||||||
{
|
|
||||||
FD_SET(fd->fd, &writefds);
|
|
||||||
}
|
|
||||||
if ((fd->events & POLLERR))
|
|
||||||
{
|
|
||||||
FD_SET(fd->fd, &errorfds);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
struct timeval tv;
|
|
||||||
tv.tv_sec = timeout / 1000;
|
|
||||||
tv.tv_usec = (timeout % 1000) * 1000;
|
|
||||||
|
|
||||||
int ret = select(maxfd + 1, &readfds, &writefds, &errorfds,
|
|
||||||
timeout != -1 ? &tv : NULL);
|
|
||||||
|
|
||||||
if (ret < 0)
|
|
||||||
{
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (nfds_t i = 0; i < nfds; ++i)
|
|
||||||
{
|
|
||||||
struct pollfd *fd = &fds[i];
|
|
||||||
fd->revents = 0;
|
|
||||||
|
|
||||||
if (FD_ISSET(fd->fd, &readfds))
|
|
||||||
{
|
|
||||||
fd->revents |= POLLIN;
|
|
||||||
}
|
|
||||||
if (FD_ISSET(fd->fd, &writefds))
|
|
||||||
{
|
|
||||||
fd->revents |= POLLOUT;
|
|
||||||
}
|
|
||||||
if (FD_ISSET(fd->fd, &errorfds))
|
|
||||||
{
|
|
||||||
fd->revents |= POLLERR;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return ret;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
@ -7,32 +7,25 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
#include <WS2tcpip.h>
|
# include <WS2tcpip.h>
|
||||||
#include <WinSock2.h>
|
# include <WinSock2.h>
|
||||||
#include <basetsd.h>
|
# include <basetsd.h>
|
||||||
#include <io.h>
|
# include <io.h>
|
||||||
#include <ws2def.h>
|
# include <ws2def.h>
|
||||||
|
|
||||||
// Define our own poll on Windows
|
|
||||||
typedef unsigned long int nfds_t;
|
|
||||||
|
|
||||||
int poll(struct pollfd* fds, nfds_t nfds, int timeout);
|
|
||||||
|
|
||||||
#else
|
#else
|
||||||
#include <arpa/inet.h>
|
# include <arpa/inet.h>
|
||||||
#include <errno.h>
|
# include <errno.h>
|
||||||
#include <netdb.h>
|
# include <netdb.h>
|
||||||
#include <netinet/tcp.h>
|
# include <netinet/tcp.h>
|
||||||
#include <poll.h>
|
# include <sys/select.h>
|
||||||
#include <sys/select.h>
|
# include <sys/socket.h>
|
||||||
#include <sys/socket.h>
|
# include <sys/stat.h>
|
||||||
#include <sys/stat.h>
|
# include <sys/time.h>
|
||||||
#include <sys/time.h>
|
# include <unistd.h>
|
||||||
#include <unistd.h>
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
bool initNetSystem();
|
bool initNetSystem();
|
||||||
bool uninitNetSystem();
|
bool uninitNetSystem();
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -11,8 +11,7 @@
|
|||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class SelectInterrupt
|
class SelectInterrupt {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
SelectInterrupt();
|
SelectInterrupt();
|
||||||
virtual ~SelectInterrupt();
|
virtual ~SelectInterrupt();
|
||||||
@ -24,4 +23,6 @@ namespace ix
|
|||||||
virtual uint64_t read();
|
virtual uint64_t read();
|
||||||
virtual int getFd() const;
|
virtual int getFd() const;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -7,13 +7,13 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXSelectInterrupt.h"
|
#include "IXSelectInterrupt.h"
|
||||||
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class SelectInterruptEventFd final : public SelectInterrupt
|
class SelectInterruptEventFd final : public SelectInterrupt {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
SelectInterruptEventFd();
|
SelectInterruptEventFd();
|
||||||
virtual ~SelectInterruptEventFd();
|
virtual ~SelectInterruptEventFd();
|
||||||
@ -28,4 +28,5 @@ namespace ix
|
|||||||
private:
|
private:
|
||||||
int _eventfd;
|
int _eventfd;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
|
||||||
|
@ -12,4 +12,4 @@ namespace ix
|
|||||||
{
|
{
|
||||||
class SelectInterrupt;
|
class SelectInterrupt;
|
||||||
std::shared_ptr<SelectInterrupt> createSelectInterrupt();
|
std::shared_ptr<SelectInterrupt> createSelectInterrupt();
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -7,14 +7,14 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXSelectInterrupt.h"
|
#include "IXSelectInterrupt.h"
|
||||||
#include <mutex>
|
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <mutex>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class SelectInterruptPipe final : public SelectInterrupt
|
class SelectInterruptPipe final : public SelectInterrupt {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
SelectInterruptPipe();
|
SelectInterruptPipe();
|
||||||
virtual ~SelectInterruptPipe();
|
virtual ~SelectInterruptPipe();
|
||||||
@ -37,4 +37,5 @@ namespace ix
|
|||||||
static const int kPipeReadIndex;
|
static const int kPipeReadIndex;
|
||||||
static const int kPipeWriteIndex;
|
static const int kPipeWriteIndex;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
|
||||||
|
@ -10,3 +10,4 @@ namespace ix
|
|||||||
{
|
{
|
||||||
void setThreadName(const std::string& name);
|
void setThreadName(const std::string& name);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -44,42 +44,43 @@ namespace ix
|
|||||||
close();
|
close();
|
||||||
}
|
}
|
||||||
|
|
||||||
PollResultType Socket::poll(bool readyToRead,
|
PollResultType Socket::poll(int timeoutMs)
|
||||||
int timeoutMs,
|
|
||||||
int sockfd,
|
|
||||||
std::shared_ptr<SelectInterrupt> selectInterrupt)
|
|
||||||
{
|
{
|
||||||
//
|
if (_sockfd == -1)
|
||||||
// We used to use ::select to poll but on Android 9 we get large fds out of ::connect
|
|
||||||
// which crash in FD_SET as they are larger than FD_SETSIZE.
|
|
||||||
// Switching to ::poll does fix that.
|
|
||||||
//
|
|
||||||
// However poll isn't as portable as select and has bugs on Windows, so we should write a
|
|
||||||
// shim to fallback to select on those platforms.
|
|
||||||
// See https://github.com/mpv-player/mpv/pull/5203/files for such a select wrapper.
|
|
||||||
//
|
|
||||||
nfds_t nfds = 1;
|
|
||||||
struct pollfd fds[2];
|
|
||||||
|
|
||||||
fds[0].fd = sockfd;
|
|
||||||
fds[0].events = (readyToRead) ? POLLIN : POLLOUT;
|
|
||||||
fds[0].events |= POLLERR;
|
|
||||||
|
|
||||||
// File descriptor used to interrupt select when needed
|
|
||||||
int interruptFd = -1;
|
|
||||||
if (selectInterrupt)
|
|
||||||
{
|
{
|
||||||
interruptFd = selectInterrupt->getFd();
|
return PollResultType::Error;
|
||||||
|
|
||||||
if (interruptFd != -1)
|
|
||||||
{
|
|
||||||
nfds = 2;
|
|
||||||
fds[1].fd = interruptFd;
|
|
||||||
fds[1].events = POLLIN;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int ret = ::poll(fds, nfds, timeoutMs);
|
return isReadyToRead(timeoutMs);
|
||||||
|
}
|
||||||
|
|
||||||
|
PollResultType Socket::select(bool readyToRead, int timeoutMs)
|
||||||
|
{
|
||||||
|
fd_set rfds;
|
||||||
|
fd_set wfds;
|
||||||
|
FD_ZERO(&rfds);
|
||||||
|
FD_ZERO(&wfds);
|
||||||
|
|
||||||
|
fd_set* fds = (readyToRead) ? &rfds : & wfds;
|
||||||
|
FD_SET(_sockfd, fds);
|
||||||
|
|
||||||
|
// File descriptor used to interrupt select when needed
|
||||||
|
int interruptFd = _selectInterrupt->getFd();
|
||||||
|
if (interruptFd != -1)
|
||||||
|
{
|
||||||
|
FD_SET(interruptFd, fds);
|
||||||
|
}
|
||||||
|
|
||||||
|
struct timeval timeout;
|
||||||
|
timeout.tv_sec = timeoutMs / 1000;
|
||||||
|
timeout.tv_usec = 1000 * (timeoutMs % 1000);
|
||||||
|
|
||||||
|
// Compute the highest fd.
|
||||||
|
int sockfd = _sockfd;
|
||||||
|
int nfds = (std::max)(sockfd, interruptFd);
|
||||||
|
|
||||||
|
int ret = ::select(nfds + 1, &rfds, &wfds, nullptr,
|
||||||
|
(timeoutMs < 0) ? nullptr : &timeout);
|
||||||
|
|
||||||
PollResultType pollResult = PollResultType::ReadyForRead;
|
PollResultType pollResult = PollResultType::ReadyForRead;
|
||||||
if (ret < 0)
|
if (ret < 0)
|
||||||
@ -90,9 +91,9 @@ namespace ix
|
|||||||
{
|
{
|
||||||
pollResult = PollResultType::Timeout;
|
pollResult = PollResultType::Timeout;
|
||||||
}
|
}
|
||||||
else if (interruptFd != -1 && fds[1].revents & POLLIN)
|
else if (interruptFd != -1 && FD_ISSET(interruptFd, &rfds))
|
||||||
{
|
{
|
||||||
uint64_t value = selectInterrupt->read();
|
uint64_t value = _selectInterrupt->read();
|
||||||
|
|
||||||
if (value == kSendRequest)
|
if (value == kSendRequest)
|
||||||
{
|
{
|
||||||
@ -103,36 +104,13 @@ namespace ix
|
|||||||
pollResult = PollResultType::CloseRequest;
|
pollResult = PollResultType::CloseRequest;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (sockfd != -1 && readyToRead && fds[0].revents & POLLIN)
|
else if (sockfd != -1 && readyToRead && FD_ISSET(sockfd, &rfds))
|
||||||
{
|
{
|
||||||
pollResult = PollResultType::ReadyForRead;
|
pollResult = PollResultType::ReadyForRead;
|
||||||
}
|
}
|
||||||
else if (sockfd != -1 && !readyToRead && fds[0].revents & POLLOUT)
|
else if (sockfd != -1 && !readyToRead && FD_ISSET(sockfd, &wfds))
|
||||||
{
|
{
|
||||||
pollResult = PollResultType::ReadyForWrite;
|
pollResult = PollResultType::ReadyForWrite;
|
||||||
|
|
||||||
#ifdef _WIN32
|
|
||||||
// On connect error, in async mode, windows will write to the exceptions fds
|
|
||||||
if (fds[0].revents & POLLERR)
|
|
||||||
{
|
|
||||||
pollResult = PollResultType::Error;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
int optval = -1;
|
|
||||||
socklen_t optlen = sizeof(optval);
|
|
||||||
|
|
||||||
// getsockopt() puts the errno value for connect into optval so 0
|
|
||||||
// means no-error.
|
|
||||||
if (getsockopt(sockfd, SOL_SOCKET, SO_ERROR, &optval, &optlen) == -1 ||
|
|
||||||
optval != 0)
|
|
||||||
{
|
|
||||||
pollResult = PollResultType::Error;
|
|
||||||
|
|
||||||
// set errno to optval so that external callers can have an
|
|
||||||
// appropriate error description when calling strerror
|
|
||||||
errno = optval;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return pollResult;
|
return pollResult;
|
||||||
@ -140,24 +118,14 @@ namespace ix
|
|||||||
|
|
||||||
PollResultType Socket::isReadyToRead(int timeoutMs)
|
PollResultType Socket::isReadyToRead(int timeoutMs)
|
||||||
{
|
{
|
||||||
if (_sockfd == -1)
|
|
||||||
{
|
|
||||||
return PollResultType::Error;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool readyToRead = true;
|
bool readyToRead = true;
|
||||||
return poll(readyToRead, timeoutMs, _sockfd, _selectInterrupt);
|
return select(readyToRead, timeoutMs);
|
||||||
}
|
}
|
||||||
|
|
||||||
PollResultType Socket::isReadyToWrite(int timeoutMs)
|
PollResultType Socket::isReadyToWrite(int timeoutMs)
|
||||||
{
|
{
|
||||||
if (_sockfd == -1)
|
|
||||||
{
|
|
||||||
return PollResultType::Error;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool readyToRead = false;
|
bool readyToRead = false;
|
||||||
return poll(readyToRead, timeoutMs, _sockfd, _selectInterrupt);
|
return select(readyToRead, timeoutMs);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Wake up from poll/select by writing to the pipe which is watched by select
|
// Wake up from poll/select by writing to the pipe which is watched by select
|
||||||
@ -256,28 +224,19 @@ namespace ix
|
|||||||
bool Socket::writeBytes(const std::string& str,
|
bool Socket::writeBytes(const std::string& str,
|
||||||
const CancellationRequest& isCancellationRequested)
|
const CancellationRequest& isCancellationRequested)
|
||||||
{
|
{
|
||||||
int offset = 0;
|
|
||||||
int len = (int) str.size();
|
|
||||||
|
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
if (isCancellationRequested && isCancellationRequested()) return false;
|
if (isCancellationRequested && isCancellationRequested()) return false;
|
||||||
|
|
||||||
ssize_t ret = send((char*)&str[offset], len);
|
char* buffer = const_cast<char*>(str.c_str());
|
||||||
|
int len = (int) str.size();
|
||||||
|
|
||||||
|
ssize_t ret = send(buffer, len);
|
||||||
|
|
||||||
// We wrote some bytes, as needed, all good.
|
// We wrote some bytes, as needed, all good.
|
||||||
if (ret > 0)
|
if (ret > 0)
|
||||||
{
|
{
|
||||||
if (ret == len)
|
return ret == len;
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
offset += ret;
|
|
||||||
len -= ret;
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
// There is possibly something to be writen, try again
|
// There is possibly something to be writen, try again
|
||||||
else if (ret < 0 && Socket::isWaitNeeded())
|
else if (ret < 0 && Socket::isWaitNeeded())
|
||||||
|
@ -6,12 +6,12 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <atomic>
|
|
||||||
#include <functional>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <functional>
|
||||||
|
#include <mutex>
|
||||||
|
#include <atomic>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
#include <BaseTsd.h>
|
#include <BaseTsd.h>
|
||||||
@ -24,11 +24,11 @@ typedef SSIZE_T ssize_t;
|
|||||||
#undef EINVAL
|
#undef EINVAL
|
||||||
|
|
||||||
// map to WSA error codes
|
// map to WSA error codes
|
||||||
#define EWOULDBLOCK WSAEWOULDBLOCK
|
#define EWOULDBLOCK WSAEWOULDBLOCK
|
||||||
#define EAGAIN WSATRY_AGAIN
|
#define EAGAIN WSATRY_AGAIN
|
||||||
#define EINPROGRESS WSAEINPROGRESS
|
#define EINPROGRESS WSAEINPROGRESS
|
||||||
#define EBADF WSAEBADF
|
#define EBADF WSAEBADF
|
||||||
#define EINVAL WSAEINVAL
|
#define EINVAL WSAEINVAL
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@ -41,16 +41,15 @@ namespace ix
|
|||||||
|
|
||||||
enum class PollResultType
|
enum class PollResultType
|
||||||
{
|
{
|
||||||
ReadyForRead = 0,
|
ReadyForRead = 0,
|
||||||
ReadyForWrite = 1,
|
ReadyForWrite = 1,
|
||||||
Timeout = 2,
|
Timeout = 2,
|
||||||
Error = 3,
|
Error = 3,
|
||||||
SendRequest = 4,
|
SendRequest = 4,
|
||||||
CloseRequest = 5
|
CloseRequest = 5
|
||||||
};
|
};
|
||||||
|
|
||||||
class Socket
|
class Socket {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
Socket(int fd = -1);
|
Socket(int fd = -1);
|
||||||
virtual ~Socket();
|
virtual ~Socket();
|
||||||
@ -76,24 +75,22 @@ namespace ix
|
|||||||
|
|
||||||
// Blocking and cancellable versions, working with socket that can be set
|
// Blocking and cancellable versions, working with socket that can be set
|
||||||
// to non blocking mode. Used during HTTP upgrade.
|
// to non blocking mode. Used during HTTP upgrade.
|
||||||
bool readByte(void* buffer, const CancellationRequest& isCancellationRequested);
|
bool readByte(void* buffer,
|
||||||
bool writeBytes(const std::string& str, const CancellationRequest& isCancellationRequested);
|
const CancellationRequest& isCancellationRequested);
|
||||||
|
bool writeBytes(const std::string& str,
|
||||||
|
const CancellationRequest& isCancellationRequested);
|
||||||
|
|
||||||
std::pair<bool, std::string> readLine(const CancellationRequest& isCancellationRequested);
|
std::pair<bool, std::string> readLine(
|
||||||
std::pair<bool, std::string> readBytes(size_t length,
|
const CancellationRequest& isCancellationRequested);
|
||||||
const OnProgressCallback& onProgressCallback,
|
std::pair<bool, std::string> readBytes(
|
||||||
const CancellationRequest& isCancellationRequested);
|
size_t length,
|
||||||
|
const OnProgressCallback& onProgressCallback,
|
||||||
|
const CancellationRequest& isCancellationRequested);
|
||||||
|
|
||||||
static int getErrno();
|
static int getErrno();
|
||||||
static bool isWaitNeeded();
|
static bool isWaitNeeded();
|
||||||
static void closeSocket(int fd);
|
static void closeSocket(int fd);
|
||||||
|
|
||||||
static PollResultType poll(bool readyToRead,
|
|
||||||
int timeoutMs,
|
|
||||||
int sockfd,
|
|
||||||
std::shared_ptr<SelectInterrupt> selectInterrupt = nullptr);
|
|
||||||
|
|
||||||
|
|
||||||
// Used as special codes for pipe communication
|
// Used as special codes for pipe communication
|
||||||
static const uint64_t kSendRequest;
|
static const uint64_t kSendRequest;
|
||||||
static const uint64_t kCloseRequest;
|
static const uint64_t kCloseRequest;
|
||||||
@ -103,6 +100,8 @@ namespace ix
|
|||||||
std::mutex _socketMutex;
|
std::mutex _socketMutex;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
PollResultType select(bool readyToRead, int timeoutMs);
|
||||||
|
|
||||||
static const int kDefaultPollTimeout;
|
static const int kDefaultPollTimeout;
|
||||||
static const int kDefaultPollNoTimeout;
|
static const int kDefaultPollNoTimeout;
|
||||||
|
|
||||||
@ -112,4 +111,4 @@ namespace ix
|
|||||||
|
|
||||||
std::shared_ptr<SelectInterrupt> _selectInterrupt;
|
std::shared_ptr<SelectInterrupt> _selectInterrupt;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -6,10 +6,12 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXCancellationRequest.h"
|
|
||||||
#include "IXSocket.h"
|
#include "IXSocket.h"
|
||||||
#include <Security/SecureTransport.h>
|
#include "IXCancellationRequest.h"
|
||||||
|
|
||||||
#include <Security/Security.h>
|
#include <Security/Security.h>
|
||||||
|
#include <Security/SecureTransport.h>
|
||||||
|
|
||||||
#include <mutex>
|
#include <mutex>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
@ -32,7 +34,7 @@ namespace ix
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
SSLContextRef _sslContext;
|
SSLContextRef _sslContext;
|
||||||
mutable std::mutex _mutex; // AppleSSL routines are not thread-safe
|
mutable std::mutex _mutex; // AppleSSL routines are not thread-safe
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -63,31 +63,55 @@ namespace ix
|
|||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
int timeoutMs = 10;
|
// On Linux the timeout needs to be re-initialized everytime
|
||||||
bool readyToRead = false;
|
// http://man7.org/linux/man-pages/man2/select.2.html
|
||||||
PollResultType pollResult = Socket::poll(readyToRead, timeoutMs, fd);
|
struct timeval timeout;
|
||||||
|
timeout.tv_sec = 0;
|
||||||
|
timeout.tv_usec = 10 * 1000; // 10ms timeout
|
||||||
|
|
||||||
if (pollResult == PollResultType::Timeout)
|
fd_set wfds;
|
||||||
{
|
fd_set efds;
|
||||||
continue;
|
|
||||||
}
|
FD_ZERO(&wfds);
|
||||||
else if (pollResult == PollResultType::Error)
|
FD_SET(fd, &wfds);
|
||||||
|
FD_ZERO(&efds);
|
||||||
|
FD_SET(fd, &efds);
|
||||||
|
|
||||||
|
// Use select to check the status of the new connection
|
||||||
|
res = select(fd + 1, nullptr, &wfds, &efds, &timeout);
|
||||||
|
|
||||||
|
if (res < 0 && (Socket::getErrno() == EBADF || Socket::getErrno() == EINVAL))
|
||||||
{
|
{
|
||||||
Socket::closeSocket(fd);
|
Socket::closeSocket(fd);
|
||||||
errMsg = std::string("Connect error: ") +
|
errMsg = std::string("Connect error, select error: ") + strerror(Socket::getErrno());
|
||||||
strerror(Socket::getErrno());
|
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
else if (pollResult == PollResultType::ReadyForWrite)
|
|
||||||
|
// Nothing was written to the socket, wait again.
|
||||||
|
if (!FD_ISSET(fd, &wfds)) continue;
|
||||||
|
|
||||||
|
// Something was written to the socket. Check for errors.
|
||||||
|
int optval = -1;
|
||||||
|
socklen_t optlen = sizeof(optval);
|
||||||
|
|
||||||
|
#ifdef _WIN32
|
||||||
|
// On connect error, in async mode, windows will write to the exceptions fds
|
||||||
|
if (FD_ISSET(fd, &efds))
|
||||||
|
#else
|
||||||
|
// getsockopt() puts the errno value for connect into optval so 0
|
||||||
|
// means no-error.
|
||||||
|
if (getsockopt(fd, SOL_SOCKET, SO_ERROR, &optval, &optlen) == -1 ||
|
||||||
|
optval != 0)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
return fd;
|
Socket::closeSocket(fd);
|
||||||
|
errMsg = strerror(optval);
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
Socket::closeSocket(fd);
|
// Success !
|
||||||
errMsg = std::string("Connect error: ") +
|
return fd;
|
||||||
strerror(Socket::getErrno());
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -104,8 +128,8 @@ namespace ix
|
|||||||
//
|
//
|
||||||
// First do DNS resolution
|
// First do DNS resolution
|
||||||
//
|
//
|
||||||
auto dnsLookup = std::make_shared<DNSLookup>(hostname, port);
|
DNSLookup dnsLookup(hostname, port);
|
||||||
struct addrinfo *res = dnsLookup->resolve(errMsg, isCancellationRequested);
|
struct addrinfo *res = dnsLookup.resolve(errMsg, isCancellationRequested);
|
||||||
if (res == nullptr)
|
if (res == nullptr)
|
||||||
{
|
{
|
||||||
return -1;
|
return -1;
|
||||||
|
@ -7,14 +7,14 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXCancellationRequest.h"
|
#include "IXCancellationRequest.h"
|
||||||
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
|
||||||
struct addrinfo;
|
struct addrinfo;
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class SocketConnect
|
class SocketConnect {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
static int connect(const std::string& hostname,
|
static int connect(const std::string& hostname,
|
||||||
int port,
|
int port,
|
||||||
@ -24,8 +24,9 @@ namespace ix
|
|||||||
static void configure(int sockfd);
|
static void configure(int sockfd);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static int connectToAddress(const struct addrinfo* address,
|
static int connectToAddress(const struct addrinfo *address,
|
||||||
std::string& errMsg,
|
std::string& errMsg,
|
||||||
const CancellationRequest& isCancellationRequested);
|
const CancellationRequest& isCancellationRequested);
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
|
||||||
|
@ -8,13 +8,11 @@
|
|||||||
|
|
||||||
#ifdef IXWEBSOCKET_USE_TLS
|
#ifdef IXWEBSOCKET_USE_TLS
|
||||||
|
|
||||||
# ifdef IXWEBSOCKET_USE_MBED_TLS
|
# ifdef __APPLE__
|
||||||
# include <ixwebsocket/IXSocketMbedTLS.h>
|
|
||||||
# elif __APPLE__
|
|
||||||
# include <ixwebsocket/IXSocketAppleSSL.h>
|
# include <ixwebsocket/IXSocketAppleSSL.h>
|
||||||
# elif defined(_WIN32)
|
# elif defined(_WIN32)
|
||||||
# include <ixwebsocket/IXSocketSChannel.h>
|
# include <ixwebsocket/IXSocketSChannel.h>
|
||||||
# elif defined(IXWEBSOCKET_USE_OPEN_SSL)
|
# else
|
||||||
# include <ixwebsocket/IXSocketOpenSSL.h>
|
# include <ixwebsocket/IXSocketOpenSSL.h>
|
||||||
# endif
|
# endif
|
||||||
|
|
||||||
@ -39,9 +37,7 @@ namespace ix
|
|||||||
else
|
else
|
||||||
{
|
{
|
||||||
#ifdef IXWEBSOCKET_USE_TLS
|
#ifdef IXWEBSOCKET_USE_TLS
|
||||||
# if defined(IXWEBSOCKET_USE_MBED_TLS)
|
# ifdef __APPLE__
|
||||||
socket = std::make_shared<SocketMbedTLS>();
|
|
||||||
# elif defined(__APPLE__)
|
|
||||||
socket = std::make_shared<SocketAppleSSL>();
|
socket = std::make_shared<SocketAppleSSL>();
|
||||||
# elif defined(_WIN32)
|
# elif defined(_WIN32)
|
||||||
socket = std::make_shared<SocketSChannel>();
|
socket = std::make_shared<SocketSChannel>();
|
||||||
|
@ -13,7 +13,9 @@
|
|||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class Socket;
|
class Socket;
|
||||||
std::shared_ptr<Socket> createSocket(bool tls, std::string& errorMsg);
|
std::shared_ptr<Socket> createSocket(bool tls,
|
||||||
|
std::string& errorMsg);
|
||||||
|
|
||||||
std::shared_ptr<Socket> createSocket(int fd, std::string& errorMsg);
|
std::shared_ptr<Socket> createSocket(int fd,
|
||||||
} // namespace ix
|
std::string& errorMsg);
|
||||||
|
}
|
||||||
|
@ -1,178 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXSocketMbedTLS.cpp
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*
|
|
||||||
* Some code taken from
|
|
||||||
* https://github.com/rottor12/WsClientLib/blob/master/lib/src/WsClientLib.cpp
|
|
||||||
* and mini_client.c example from mbedtls
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "IXSocketMbedTLS.h"
|
|
||||||
#include "IXSocketConnect.h"
|
|
||||||
#include "IXNetSystem.h"
|
|
||||||
#include "IXSocket.h"
|
|
||||||
|
|
||||||
#include <string.h>
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
SocketMbedTLS::~SocketMbedTLS()
|
|
||||||
{
|
|
||||||
close();
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SocketMbedTLS::init(const std::string& host, std::string& errMsg)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
|
||||||
|
|
||||||
mbedtls_ssl_init(&_ssl);
|
|
||||||
mbedtls_ssl_config_init(&_conf);
|
|
||||||
mbedtls_ctr_drbg_init(&_ctr_drbg);
|
|
||||||
|
|
||||||
const char *pers = "IXSocketMbedTLS";
|
|
||||||
|
|
||||||
mbedtls_entropy_init(&_entropy);
|
|
||||||
if (mbedtls_ctr_drbg_seed(&_ctr_drbg,
|
|
||||||
mbedtls_entropy_func,
|
|
||||||
&_entropy,
|
|
||||||
(const unsigned char *) pers,
|
|
||||||
strlen(pers)) != 0)
|
|
||||||
{
|
|
||||||
errMsg = "Setting entropy seed failed";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mbedtls_ssl_config_defaults(&_conf,
|
|
||||||
MBEDTLS_SSL_IS_CLIENT,
|
|
||||||
MBEDTLS_SSL_TRANSPORT_STREAM,
|
|
||||||
MBEDTLS_SSL_PRESET_DEFAULT ) != 0)
|
|
||||||
{
|
|
||||||
errMsg = "Setting config default failed";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
mbedtls_ssl_conf_rng(&_conf, mbedtls_ctr_drbg_random, &_ctr_drbg);
|
|
||||||
|
|
||||||
// FIXME: cert verification is disabled
|
|
||||||
mbedtls_ssl_conf_authmode(&_conf, MBEDTLS_SSL_VERIFY_NONE);
|
|
||||||
|
|
||||||
if (mbedtls_ssl_setup(&_ssl, &_conf) != 0)
|
|
||||||
{
|
|
||||||
errMsg = "SSL setup failed";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mbedtls_ssl_set_hostname(&_ssl, host.c_str()) != 0)
|
|
||||||
{
|
|
||||||
errMsg = "SNI setup failed";
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool SocketMbedTLS::connect(const std::string& host,
|
|
||||||
int port,
|
|
||||||
std::string& errMsg,
|
|
||||||
const CancellationRequest& isCancellationRequested)
|
|
||||||
{
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
|
||||||
_sockfd = SocketConnect::connect(host, port, errMsg, isCancellationRequested);
|
|
||||||
if (_sockfd == -1) return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!init(host, errMsg))
|
|
||||||
{
|
|
||||||
close();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
mbedtls_ssl_set_bio(&_ssl, &_sockfd, mbedtls_net_send, mbedtls_net_recv, NULL);
|
|
||||||
|
|
||||||
int res;
|
|
||||||
do
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
|
||||||
res = mbedtls_ssl_handshake(&_ssl);
|
|
||||||
}
|
|
||||||
while (res == MBEDTLS_ERR_SSL_WANT_READ || res == MBEDTLS_ERR_SSL_WANT_WRITE);
|
|
||||||
|
|
||||||
if (res != 0)
|
|
||||||
{
|
|
||||||
char buf[256];
|
|
||||||
mbedtls_strerror(res, buf, sizeof(buf));
|
|
||||||
|
|
||||||
errMsg = "error in handshake : ";
|
|
||||||
errMsg += buf;
|
|
||||||
|
|
||||||
close();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void SocketMbedTLS::close()
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
|
||||||
|
|
||||||
mbedtls_ssl_free(&_ssl);
|
|
||||||
mbedtls_ssl_config_free(&_conf);
|
|
||||||
mbedtls_ctr_drbg_free(&_ctr_drbg);
|
|
||||||
mbedtls_entropy_free(&_entropy);
|
|
||||||
|
|
||||||
Socket::close();
|
|
||||||
}
|
|
||||||
|
|
||||||
ssize_t SocketMbedTLS::send(char* buf, size_t nbyte)
|
|
||||||
{
|
|
||||||
ssize_t sent = 0;
|
|
||||||
|
|
||||||
while (nbyte > 0)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
|
||||||
|
|
||||||
ssize_t res = mbedtls_ssl_write(&_ssl, (unsigned char*) buf, nbyte);
|
|
||||||
|
|
||||||
if (res > 0) {
|
|
||||||
nbyte -= res;
|
|
||||||
sent += res;
|
|
||||||
} else if (res == MBEDTLS_ERR_SSL_WANT_READ || res == MBEDTLS_ERR_SSL_WANT_WRITE) {
|
|
||||||
errno = EWOULDBLOCK;
|
|
||||||
return -1;
|
|
||||||
} else {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return sent;
|
|
||||||
}
|
|
||||||
|
|
||||||
ssize_t SocketMbedTLS::send(const std::string& buffer)
|
|
||||||
{
|
|
||||||
return send((char*)&buffer[0], buffer.size());
|
|
||||||
}
|
|
||||||
|
|
||||||
ssize_t SocketMbedTLS::recv(void* buf, size_t nbyte)
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
|
||||||
|
|
||||||
ssize_t res = mbedtls_ssl_read(&_ssl, (unsigned char*) buf, (int) nbyte);
|
|
||||||
|
|
||||||
if (res > 0)
|
|
||||||
{
|
|
||||||
return res;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (res == MBEDTLS_ERR_SSL_WANT_READ || res == MBEDTLS_ERR_SSL_WANT_WRITE)
|
|
||||||
{
|
|
||||||
errno = EWOULDBLOCK;
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
@ -1,47 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXSocketMbedTLS.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "IXSocket.h"
|
|
||||||
#include <mbedtls/ctr_drbg.h>
|
|
||||||
#include <mbedtls/debug.h>
|
|
||||||
#include <mbedtls/entropy.h>
|
|
||||||
#include <mbedtls/error.h>
|
|
||||||
#include <mbedtls/net.h>
|
|
||||||
#include <mbedtls/platform.h>
|
|
||||||
#include <mutex>
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
class SocketMbedTLS final : public Socket
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
SocketMbedTLS() = default;
|
|
||||||
~SocketMbedTLS();
|
|
||||||
|
|
||||||
virtual bool connect(const std::string& host,
|
|
||||||
int port,
|
|
||||||
std::string& errMsg,
|
|
||||||
const CancellationRequest& isCancellationRequested) final;
|
|
||||||
virtual void close() final;
|
|
||||||
|
|
||||||
virtual ssize_t send(char* buffer, size_t length) final;
|
|
||||||
virtual ssize_t send(const std::string& buffer) final;
|
|
||||||
virtual ssize_t recv(void* buffer, size_t length) final;
|
|
||||||
|
|
||||||
private:
|
|
||||||
mbedtls_ssl_context _ssl;
|
|
||||||
mbedtls_ssl_config _conf;
|
|
||||||
mbedtls_entropy_context _entropy;
|
|
||||||
mbedtls_ctr_drbg_context _ctr_drbg;
|
|
||||||
|
|
||||||
std::mutex _mutex;
|
|
||||||
|
|
||||||
bool init(const std::string& host, std::string& errMsg);
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace ix
|
|
@ -240,6 +240,7 @@ namespace ix
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// No wait support
|
||||||
bool SocketOpenSSL::connect(const std::string& host,
|
bool SocketOpenSSL::connect(const std::string& host,
|
||||||
int port,
|
int port,
|
||||||
std::string& errMsg,
|
std::string& errMsg,
|
||||||
@ -360,6 +361,7 @@ namespace ix
|
|||||||
return send((char*)&buffer[0], buffer.size());
|
return send((char*)&buffer[0], buffer.size());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// No wait support
|
||||||
ssize_t SocketOpenSSL::recv(void* buf, size_t nbyte)
|
ssize_t SocketOpenSSL::recv(void* buf, size_t nbyte)
|
||||||
{
|
{
|
||||||
while (true)
|
while (true)
|
||||||
|
@ -6,15 +6,17 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXCancellationRequest.h"
|
|
||||||
#include "IXSocket.h"
|
#include "IXSocket.h"
|
||||||
#include <mutex>
|
#include "IXCancellationRequest.h"
|
||||||
|
|
||||||
#include <openssl/bio.h>
|
#include <openssl/bio.h>
|
||||||
|
#include <openssl/hmac.h>
|
||||||
#include <openssl/conf.h>
|
#include <openssl/conf.h>
|
||||||
#include <openssl/err.h>
|
#include <openssl/err.h>
|
||||||
#include <openssl/hmac.h>
|
|
||||||
#include <openssl/ssl.h>
|
#include <openssl/ssl.h>
|
||||||
|
|
||||||
|
#include <mutex>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class SocketOpenSSL final : public Socket
|
class SocketOpenSSL final : public Socket
|
||||||
@ -38,16 +40,18 @@ namespace ix
|
|||||||
std::string getSSLError(int ret);
|
std::string getSSLError(int ret);
|
||||||
SSL_CTX* openSSLCreateContext(std::string& errMsg);
|
SSL_CTX* openSSLCreateContext(std::string& errMsg);
|
||||||
bool openSSLHandshake(const std::string& hostname, std::string& errMsg);
|
bool openSSLHandshake(const std::string& hostname, std::string& errMsg);
|
||||||
bool openSSLCheckServerCert(SSL* ssl, const std::string& hostname, std::string& errMsg);
|
bool openSSLCheckServerCert(SSL *ssl,
|
||||||
bool checkHost(const std::string& host, const char* pattern);
|
const std::string& hostname,
|
||||||
|
std::string& errMsg);
|
||||||
|
bool checkHost(const std::string& host, const char *pattern);
|
||||||
|
|
||||||
SSL* _ssl_connection;
|
SSL* _ssl_connection;
|
||||||
SSL_CTX* _ssl_context;
|
SSL_CTX* _ssl_context;
|
||||||
const SSL_METHOD* _ssl_method;
|
const SSL_METHOD* _ssl_method;
|
||||||
mutable std::mutex _mutex; // OpenSSL routines are not thread-safe
|
mutable std::mutex _mutex; // OpenSSL routines are not thread-safe
|
||||||
|
|
||||||
static std::once_flag _openSSLInitFlag;
|
static std::once_flag _openSSLInitFlag;
|
||||||
static std::atomic<bool> _openSSLInitializationSuccessful;
|
static std::atomic<bool> _openSSLInitializationSuccessful;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -16,7 +16,9 @@ namespace ix
|
|||||||
SocketSChannel();
|
SocketSChannel();
|
||||||
~SocketSChannel();
|
~SocketSChannel();
|
||||||
|
|
||||||
virtual bool connect(const std::string& host, int port, std::string& errMsg) final;
|
virtual bool connect(const std::string& host,
|
||||||
|
int port,
|
||||||
|
std::string& errMsg) final;
|
||||||
virtual void close() final;
|
virtual void close() final;
|
||||||
|
|
||||||
// The important override
|
// The important override
|
||||||
@ -29,4 +31,4 @@ namespace ix
|
|||||||
private:
|
private:
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -11,6 +11,7 @@
|
|||||||
|
|
||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
#include <future>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <assert.h>
|
#include <assert.h>
|
||||||
|
|
||||||
@ -213,12 +214,17 @@ namespace ix
|
|||||||
{
|
{
|
||||||
if (_stop) return;
|
if (_stop) return;
|
||||||
|
|
||||||
// Use poll to check whether a new connection is in progress
|
// Use select to check whether a new connection is in progress
|
||||||
int timeoutMs = 10;
|
fd_set rfds;
|
||||||
bool readyToRead = true;
|
struct timeval timeout;
|
||||||
PollResultType pollResult = Socket::poll(readyToRead, timeoutMs, _serverFd);
|
timeout.tv_sec = 0;
|
||||||
|
timeout.tv_usec = 10 * 1000; // 10ms timeout
|
||||||
|
|
||||||
if (pollResult == PollResultType::Error)
|
FD_ZERO(&rfds);
|
||||||
|
FD_SET(_serverFd, &rfds);
|
||||||
|
|
||||||
|
if (select(_serverFd + 1, &rfds, nullptr, nullptr, &timeout) < 0 &&
|
||||||
|
(errno == EBADF || errno == EINVAL))
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "SocketServer::run() error in select: "
|
ss << "SocketServer::run() error in select: "
|
||||||
@ -227,8 +233,9 @@ namespace ix
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (pollResult != PollResultType::ReadyForRead)
|
if (!FD_ISSET(_serverFd, &rfds))
|
||||||
{
|
{
|
||||||
|
// We reached the select timeout, and no new connections are pending
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7,28 +7,28 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXConnectionState.h"
|
#include "IXConnectionState.h"
|
||||||
|
|
||||||
|
#include <utility> // pair
|
||||||
|
#include <string>
|
||||||
|
#include <set>
|
||||||
|
#include <thread>
|
||||||
|
#include <list>
|
||||||
|
#include <mutex>
|
||||||
|
#include <functional>
|
||||||
|
#include <memory>
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <condition_variable>
|
#include <condition_variable>
|
||||||
#include <functional>
|
|
||||||
#include <list>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <set>
|
|
||||||
#include <string>
|
|
||||||
#include <thread>
|
|
||||||
#include <utility> // pair
|
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class SocketServer
|
class SocketServer {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
using ConnectionStateFactory = std::function<std::shared_ptr<ConnectionState>()>;
|
using ConnectionStateFactory = std::function<std::shared_ptr<ConnectionState>()>;
|
||||||
|
|
||||||
// Each connection is handled by its own worker thread.
|
// Each connection is handled by its own worker thread.
|
||||||
// We use a list as we only care about remove and append operations.
|
// We use a list as we only care about remove and append operations.
|
||||||
using ConnectionThreads =
|
using ConnectionThreads = std::list<std::pair<std::shared_ptr<ConnectionState>,
|
||||||
std::list<std::pair<std::shared_ptr<ConnectionState>, std::thread>>;
|
std::thread>>;
|
||||||
|
|
||||||
SocketServer(int port = SocketServer::kDefaultPort,
|
SocketServer(int port = SocketServer::kDefaultPort,
|
||||||
const std::string& host = SocketServer::kDefaultHost,
|
const std::string& host = SocketServer::kDefaultHost,
|
||||||
@ -52,6 +52,7 @@ namespace ix
|
|||||||
void wait();
|
void wait();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
|
|
||||||
// Logging
|
// Logging
|
||||||
void logError(const std::string& str);
|
void logError(const std::string& str);
|
||||||
void logInfo(const std::string& str);
|
void logInfo(const std::string& str);
|
||||||
@ -92,11 +93,12 @@ namespace ix
|
|||||||
// the factory to create ConnectionState objects
|
// the factory to create ConnectionState objects
|
||||||
ConnectionStateFactory _connectionStateFactory;
|
ConnectionStateFactory _connectionStateFactory;
|
||||||
|
|
||||||
virtual void handleConnection(int fd, std::shared_ptr<ConnectionState> connectionState) = 0;
|
virtual void handleConnection(int fd,
|
||||||
|
std::shared_ptr<ConnectionState> connectionState) = 0;
|
||||||
virtual size_t getConnectedClientsCount() = 0;
|
virtual size_t getConnectedClientsCount() = 0;
|
||||||
|
|
||||||
// Returns true if all connection threads are joined
|
// Returns true if all connection threads are joined
|
||||||
void closeTerminatedThreads();
|
void closeTerminatedThreads();
|
||||||
size_t getConnectionsThreadsCount();
|
size_t getConnectionsThreadsCount();
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -20,4 +20,4 @@ namespace ix
|
|||||||
std::string& query,
|
std::string& query,
|
||||||
int& port);
|
int& port);
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -51,11 +51,9 @@ namespace ix
|
|||||||
_ws.setOnCloseCallback(
|
_ws.setOnCloseCallback(
|
||||||
[this](uint16_t code, const std::string& reason, size_t wireSize, bool remote)
|
[this](uint16_t code, const std::string& reason, size_t wireSize, bool remote)
|
||||||
{
|
{
|
||||||
_onMessageCallback(
|
_onMessageCallback(WebSocketMessageType::Close, "", wireSize,
|
||||||
std::make_shared<WebSocketMessage>(
|
WebSocketErrorInfo(), WebSocketOpenInfo(),
|
||||||
WebSocketMessageType::Close, "", wireSize,
|
WebSocketCloseInfo(code, reason, remote));
|
||||||
WebSocketErrorInfo(), WebSocketOpenInfo(),
|
|
||||||
WebSocketCloseInfo(code, reason, remote)));
|
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
@ -137,13 +135,6 @@ namespace ix
|
|||||||
_enablePong = false;
|
_enablePong = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void WebSocket::disablePerMessageDeflate()
|
|
||||||
{
|
|
||||||
std::lock_guard<std::mutex> lock(_configMutex);
|
|
||||||
WebSocketPerMessageDeflateOptions perMessageDeflateOptions(false);
|
|
||||||
_perMessageDeflateOptions = perMessageDeflateOptions;
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebSocket::start()
|
void WebSocket::start()
|
||||||
{
|
{
|
||||||
if (_thread.joinable()) return; // we've already been started
|
if (_thread.joinable()) return; // we've already been started
|
||||||
@ -182,12 +173,10 @@ namespace ix
|
|||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
_onMessageCallback(
|
_onMessageCallback(WebSocketMessageType::Open, "", 0,
|
||||||
std::make_shared<WebSocketMessage>(
|
WebSocketErrorInfo(),
|
||||||
WebSocketMessageType::Open, "", 0,
|
WebSocketOpenInfo(status.uri, status.headers),
|
||||||
WebSocketErrorInfo(),
|
WebSocketCloseInfo());
|
||||||
WebSocketOpenInfo(status.uri, status.headers),
|
|
||||||
WebSocketCloseInfo()));
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -207,12 +196,10 @@ namespace ix
|
|||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
_onMessageCallback(
|
_onMessageCallback(WebSocketMessageType::Open, "", 0,
|
||||||
std::make_shared<WebSocketMessage>(
|
WebSocketErrorInfo(),
|
||||||
WebSocketMessageType::Open, "", 0,
|
WebSocketOpenInfo(status.uri, status.headers),
|
||||||
WebSocketErrorInfo(),
|
WebSocketCloseInfo());
|
||||||
WebSocketOpenInfo(status.uri, status.headers),
|
|
||||||
WebSocketCloseInfo()));
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -237,7 +224,7 @@ namespace ix
|
|||||||
using millis = std::chrono::duration<double, std::milli>;
|
using millis = std::chrono::duration<double, std::milli>;
|
||||||
|
|
||||||
uint32_t retries = 0;
|
uint32_t retries = 0;
|
||||||
millis duration(0);
|
millis duration;
|
||||||
|
|
||||||
// Try to connect perpertually
|
// Try to connect perpertually
|
||||||
while (true)
|
while (true)
|
||||||
@ -280,11 +267,9 @@ namespace ix
|
|||||||
connectErr.reason = status.errorStr;
|
connectErr.reason = status.errorStr;
|
||||||
connectErr.http_status = status.http_status;
|
connectErr.http_status = status.http_status;
|
||||||
|
|
||||||
_onMessageCallback(
|
_onMessageCallback(WebSocketMessageType::Error, "", 0,
|
||||||
std::make_shared<WebSocketMessage>(
|
connectErr, WebSocketOpenInfo(),
|
||||||
WebSocketMessageType::Error, "", 0,
|
WebSocketCloseInfo());
|
||||||
connectErr, WebSocketOpenInfo(),
|
|
||||||
WebSocketCloseInfo()));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -325,8 +310,8 @@ namespace ix
|
|||||||
WebSocketMessageType webSocketMessageType;
|
WebSocketMessageType webSocketMessageType;
|
||||||
switch (messageKind)
|
switch (messageKind)
|
||||||
{
|
{
|
||||||
case WebSocketTransport::MessageKind::MSG_TEXT:
|
default:
|
||||||
case WebSocketTransport::MessageKind::MSG_BINARY:
|
case WebSocketTransport::MessageKind::MSG:
|
||||||
{
|
{
|
||||||
webSocketMessageType = WebSocketMessageType::Message;
|
webSocketMessageType = WebSocketMessageType::Message;
|
||||||
} break;
|
} break;
|
||||||
@ -350,13 +335,9 @@ namespace ix
|
|||||||
WebSocketErrorInfo webSocketErrorInfo;
|
WebSocketErrorInfo webSocketErrorInfo;
|
||||||
webSocketErrorInfo.decompressionError = decompressionError;
|
webSocketErrorInfo.decompressionError = decompressionError;
|
||||||
|
|
||||||
bool binary = messageKind == WebSocketTransport::MessageKind::MSG_BINARY;
|
_onMessageCallback(webSocketMessageType, msg, wireSize,
|
||||||
|
webSocketErrorInfo, WebSocketOpenInfo(),
|
||||||
_onMessageCallback(
|
WebSocketCloseInfo());
|
||||||
std::make_shared<WebSocketMessage>(
|
|
||||||
webSocketMessageType, msg, wireSize,
|
|
||||||
webSocketErrorInfo, WebSocketOpenInfo(),
|
|
||||||
WebSocketCloseInfo(), binary));
|
|
||||||
|
|
||||||
WebSocket::invokeTrafficTrackerCallback(msg.size(), true);
|
WebSocket::invokeTrafficTrackerCallback(msg.size(), true);
|
||||||
});
|
});
|
||||||
@ -387,18 +368,9 @@ namespace ix
|
|||||||
}
|
}
|
||||||
|
|
||||||
WebSocketSendInfo WebSocket::send(const std::string& data,
|
WebSocketSendInfo WebSocket::send(const std::string& data,
|
||||||
bool binary,
|
|
||||||
const OnProgressCallback& onProgressCallback)
|
const OnProgressCallback& onProgressCallback)
|
||||||
{
|
{
|
||||||
return sendMessage(data,
|
return sendMessage(data, SendMessageKind::Binary, onProgressCallback);
|
||||||
(binary) ? SendMessageKind::Binary: SendMessageKind::Text,
|
|
||||||
onProgressCallback);
|
|
||||||
}
|
|
||||||
|
|
||||||
WebSocketSendInfo WebSocket::sendBinary(const std::string& text,
|
|
||||||
const OnProgressCallback& onProgressCallback)
|
|
||||||
{
|
|
||||||
return sendMessage(text, SendMessageKind::Binary, onProgressCallback);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
WebSocketSendInfo WebSocket::sendText(const std::string& text,
|
WebSocketSendInfo WebSocket::sendText(const std::string& text,
|
||||||
|
@ -9,31 +9,78 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXProgressCallback.h"
|
|
||||||
#include "IXWebSocketCloseConstants.h"
|
|
||||||
#include "IXWebSocketErrorInfo.h"
|
|
||||||
#include "IXWebSocketHttpHeaders.h"
|
|
||||||
#include "IXWebSocketMessage.h"
|
|
||||||
#include "IXWebSocketPerMessageDeflateOptions.h"
|
|
||||||
#include "IXWebSocketSendInfo.h"
|
|
||||||
#include "IXWebSocketTransport.h"
|
|
||||||
#include <atomic>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <thread>
|
#include <thread>
|
||||||
|
#include <mutex>
|
||||||
|
#include <atomic>
|
||||||
|
|
||||||
|
#include "IXWebSocketTransport.h"
|
||||||
|
#include "IXWebSocketErrorInfo.h"
|
||||||
|
#include "IXWebSocketSendInfo.h"
|
||||||
|
#include "IXWebSocketPerMessageDeflateOptions.h"
|
||||||
|
#include "IXWebSocketHttpHeaders.h"
|
||||||
|
#include "IXWebSocketCloseConstants.h"
|
||||||
|
#include "IXProgressCallback.h"
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
// https://developer.mozilla.org/en-US/docs/Web/API/WebSocket#Ready_state_constants
|
// https://developer.mozilla.org/en-US/docs/Web/API/WebSocket#Ready_state_constants
|
||||||
enum class ReadyState
|
enum class ReadyState
|
||||||
{
|
{
|
||||||
Connecting = 0,
|
Connecting = 0,
|
||||||
Open = 1,
|
Open = 1,
|
||||||
Closing = 2,
|
Closing = 2,
|
||||||
Closed = 3
|
Closed = 3
|
||||||
};
|
};
|
||||||
|
|
||||||
using OnMessageCallback = std::function<void(const WebSocketMessagePtr&)>;
|
enum class WebSocketMessageType
|
||||||
|
{
|
||||||
|
Message = 0,
|
||||||
|
Open = 1,
|
||||||
|
Close = 2,
|
||||||
|
Error = 3,
|
||||||
|
Ping = 4,
|
||||||
|
Pong = 5,
|
||||||
|
Fragment = 6
|
||||||
|
};
|
||||||
|
|
||||||
|
struct WebSocketOpenInfo
|
||||||
|
{
|
||||||
|
std::string uri;
|
||||||
|
WebSocketHttpHeaders headers;
|
||||||
|
|
||||||
|
WebSocketOpenInfo(const std::string& u = std::string(),
|
||||||
|
const WebSocketHttpHeaders& h = WebSocketHttpHeaders())
|
||||||
|
: uri(u)
|
||||||
|
, headers(h)
|
||||||
|
{
|
||||||
|
;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
struct WebSocketCloseInfo
|
||||||
|
{
|
||||||
|
uint16_t code;
|
||||||
|
std::string reason;
|
||||||
|
bool remote;
|
||||||
|
|
||||||
|
WebSocketCloseInfo(uint16_t c = 0,
|
||||||
|
const std::string& r = std::string(),
|
||||||
|
bool rem = false)
|
||||||
|
: code(c)
|
||||||
|
, reason(r)
|
||||||
|
, remote(rem)
|
||||||
|
{
|
||||||
|
;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
using OnMessageCallback = std::function<void(WebSocketMessageType,
|
||||||
|
const std::string&,
|
||||||
|
size_t wireSize,
|
||||||
|
const WebSocketErrorInfo&,
|
||||||
|
const WebSocketOpenInfo&,
|
||||||
|
const WebSocketCloseInfo&)>;
|
||||||
|
|
||||||
using OnTrafficTrackerCallback = std::function<void(size_t size, bool incoming)>;
|
using OnTrafficTrackerCallback = std::function<void(size_t size, bool incoming)>;
|
||||||
|
|
||||||
@ -44,14 +91,12 @@ namespace ix
|
|||||||
~WebSocket();
|
~WebSocket();
|
||||||
|
|
||||||
void setUrl(const std::string& url);
|
void setUrl(const std::string& url);
|
||||||
void setPerMessageDeflateOptions(
|
void setPerMessageDeflateOptions(const WebSocketPerMessageDeflateOptions& perMessageDeflateOptions);
|
||||||
const WebSocketPerMessageDeflateOptions& perMessageDeflateOptions);
|
|
||||||
void setHeartBeatPeriod(int heartBeatPeriodSecs);
|
void setHeartBeatPeriod(int heartBeatPeriodSecs);
|
||||||
void setPingInterval(int pingIntervalSecs); // alias of setHeartBeatPeriod
|
void setPingInterval(int pingIntervalSecs); // alias of setHeartBeatPeriod
|
||||||
void setPingTimeout(int pingTimeoutSecs);
|
void setPingTimeout(int pingTimeoutSecs);
|
||||||
void enablePong();
|
void enablePong();
|
||||||
void disablePong();
|
void disablePong();
|
||||||
void disablePerMessageDeflate();
|
|
||||||
|
|
||||||
// Run asynchronously, by calling start and stop.
|
// Run asynchronously, by calling start and stop.
|
||||||
void start();
|
void start();
|
||||||
@ -64,18 +109,15 @@ namespace ix
|
|||||||
WebSocketInitResult connect(int timeoutSecs);
|
WebSocketInitResult connect(int timeoutSecs);
|
||||||
void run();
|
void run();
|
||||||
|
|
||||||
// send is in binary mode by default
|
// send binary data
|
||||||
WebSocketSendInfo send(const std::string& data,
|
WebSocketSendInfo send(const std::string& data,
|
||||||
bool binary = false,
|
|
||||||
const OnProgressCallback& onProgressCallback = nullptr);
|
const OnProgressCallback& onProgressCallback = nullptr);
|
||||||
WebSocketSendInfo sendBinary(const std::string& text,
|
|
||||||
const OnProgressCallback& onProgressCallback = nullptr);
|
|
||||||
WebSocketSendInfo sendText(const std::string& text,
|
WebSocketSendInfo sendText(const std::string& text,
|
||||||
const OnProgressCallback& onProgressCallback = nullptr);
|
const OnProgressCallback& onProgressCallback = nullptr);
|
||||||
WebSocketSendInfo ping(const std::string& text);
|
WebSocketSendInfo ping(const std::string& text);
|
||||||
|
|
||||||
void close(uint16_t code = WebSocketCloseConstants::kNormalClosureCode,
|
void close(uint16_t code = 1000,
|
||||||
const std::string& reason = WebSocketCloseConstants::kNormalClosureMessage);
|
const std::string& reason = "Normal closure");
|
||||||
|
|
||||||
void setOnMessageCallback(const OnMessageCallback& callback);
|
void setOnMessageCallback(const OnMessageCallback& callback);
|
||||||
static void setTrafficTrackerCallback(const OnTrafficTrackerCallback& callback);
|
static void setTrafficTrackerCallback(const OnTrafficTrackerCallback& callback);
|
||||||
@ -96,6 +138,7 @@ namespace ix
|
|||||||
bool isAutomaticReconnectionEnabled() const;
|
bool isAutomaticReconnectionEnabled() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
WebSocketSendInfo sendMessage(const std::string& text,
|
WebSocketSendInfo sendMessage(const std::string& text,
|
||||||
SendMessageKind sendMessageKind,
|
SendMessageKind sendMessageKind,
|
||||||
const OnProgressCallback& callback = nullptr);
|
const OnProgressCallback& callback = nullptr);
|
||||||
@ -129,7 +172,7 @@ namespace ix
|
|||||||
bool _enablePong;
|
bool _enablePong;
|
||||||
static const bool kDefaultEnablePong;
|
static const bool kDefaultEnablePong;
|
||||||
|
|
||||||
// Optional ping and pong timeout
|
// Optional ping and ping timeout
|
||||||
int _pingIntervalSecs;
|
int _pingIntervalSecs;
|
||||||
int _pingTimeoutSecs;
|
int _pingTimeoutSecs;
|
||||||
static const int kDefaultPingIntervalSecs;
|
static const int kDefaultPingIntervalSecs;
|
||||||
@ -137,4 +180,4 @@ namespace ix
|
|||||||
|
|
||||||
friend class WebSocketServer;
|
friend class WebSocketServer;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -26,4 +26,4 @@ namespace ix
|
|||||||
static const std::string kProtocolErrorMessage;
|
static const std::string kProtocolErrorMessage;
|
||||||
static const std::string kNoStatusCodeErrorMessage;
|
static const std::string kNoStatusCodeErrorMessage;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -1,25 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXWebSocketCloseInfo.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2017-2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
struct WebSocketCloseInfo
|
|
||||||
{
|
|
||||||
uint16_t code;
|
|
||||||
std::string reason;
|
|
||||||
bool remote;
|
|
||||||
|
|
||||||
WebSocketCloseInfo(uint16_t c = 0, const std::string& r = std::string(), bool rem = false)
|
|
||||||
: code(c)
|
|
||||||
, reason(r)
|
|
||||||
, remote(rem)
|
|
||||||
{
|
|
||||||
;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
} // namespace ix
|
|
@ -18,4 +18,4 @@ namespace ix
|
|||||||
std::string reason;
|
std::string reason;
|
||||||
bool decompressionError = false;
|
bool decompressionError = false;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -7,7 +7,6 @@
|
|||||||
#include "IXWebSocketHandshake.h"
|
#include "IXWebSocketHandshake.h"
|
||||||
#include "IXSocketConnect.h"
|
#include "IXSocketConnect.h"
|
||||||
#include "IXUrlParser.h"
|
#include "IXUrlParser.h"
|
||||||
#include "IXHttp.h"
|
|
||||||
|
|
||||||
#include "libwshandshake.hpp"
|
#include "libwshandshake.hpp"
|
||||||
|
|
||||||
@ -32,6 +31,15 @@ namespace ix
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::string WebSocketHandshake::trim(const std::string& str)
|
||||||
|
{
|
||||||
|
std::string out(str);
|
||||||
|
out.erase(std::remove(out.begin(), out.end(), ' '), out.end());
|
||||||
|
out.erase(std::remove(out.begin(), out.end(), '\r'), out.end());
|
||||||
|
out.erase(std::remove(out.begin(), out.end(), '\n'), out.end());
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
|
||||||
bool WebSocketHandshake::insensitiveStringCompare(const std::string& a, const std::string& b)
|
bool WebSocketHandshake::insensitiveStringCompare(const std::string& a, const std::string& b)
|
||||||
{
|
{
|
||||||
return std::equal(a.begin(), a.end(),
|
return std::equal(a.begin(), a.end(),
|
||||||
@ -42,6 +50,40 @@ namespace ix
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::tuple<std::string, std::string, std::string> WebSocketHandshake::parseRequestLine(const std::string& line)
|
||||||
|
{
|
||||||
|
// Request-Line = Method SP Request-URI SP HTTP-Version CRLF
|
||||||
|
std::string token;
|
||||||
|
std::stringstream tokenStream(line);
|
||||||
|
std::vector<std::string> tokens;
|
||||||
|
|
||||||
|
// Split by ' '
|
||||||
|
while (std::getline(tokenStream, token, ' '))
|
||||||
|
{
|
||||||
|
tokens.push_back(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string method;
|
||||||
|
if (tokens.size() >= 1)
|
||||||
|
{
|
||||||
|
method = trim(tokens[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string requestUri;
|
||||||
|
if (tokens.size() >= 2)
|
||||||
|
{
|
||||||
|
requestUri = trim(tokens[1]);
|
||||||
|
}
|
||||||
|
|
||||||
|
std::string httpVersion;
|
||||||
|
if (tokens.size() >= 3)
|
||||||
|
{
|
||||||
|
httpVersion = trim(tokens[2]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return std::make_tuple(method, requestUri, httpVersion);
|
||||||
|
}
|
||||||
|
|
||||||
std::string WebSocketHandshake::genRandomString(const int len)
|
std::string WebSocketHandshake::genRandomString(const int len)
|
||||||
{
|
{
|
||||||
std::string alphanum =
|
std::string alphanum =
|
||||||
@ -252,7 +294,7 @@ namespace ix
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Validate request line (GET /foo HTTP/1.1\r\n)
|
// Validate request line (GET /foo HTTP/1.1\r\n)
|
||||||
auto requestLine = Http::parseRequestLine(line);
|
auto requestLine = parseRequestLine(line);
|
||||||
auto method = std::get<0>(requestLine);
|
auto method = std::get<0>(requestLine);
|
||||||
auto uri = std::get<1>(requestLine);
|
auto uri = std::get<1>(requestLine);
|
||||||
auto httpVersion = std::get<2>(requestLine);
|
auto httpVersion = std::get<2>(requestLine);
|
||||||
|
@ -7,14 +7,15 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXCancellationRequest.h"
|
#include "IXCancellationRequest.h"
|
||||||
#include "IXSocket.h"
|
|
||||||
#include "IXWebSocketHttpHeaders.h"
|
#include "IXWebSocketHttpHeaders.h"
|
||||||
#include "IXWebSocketPerMessageDeflate.h"
|
#include "IXWebSocketPerMessageDeflate.h"
|
||||||
#include "IXWebSocketPerMessageDeflateOptions.h"
|
#include "IXWebSocketPerMessageDeflateOptions.h"
|
||||||
|
#include "IXSocket.h"
|
||||||
|
|
||||||
|
#include <string>
|
||||||
#include <atomic>
|
#include <atomic>
|
||||||
#include <chrono>
|
#include <chrono>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
|
||||||
#include <tuple>
|
#include <tuple>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
@ -41,8 +42,7 @@ namespace ix
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
class WebSocketHandshake
|
class WebSocketHandshake {
|
||||||
{
|
|
||||||
public:
|
public:
|
||||||
WebSocketHandshake(std::atomic<bool>& requestInitCancellation,
|
WebSocketHandshake(std::atomic<bool>& requestInitCancellation,
|
||||||
std::shared_ptr<Socket> _socket,
|
std::shared_ptr<Socket> _socket,
|
||||||
@ -56,7 +56,8 @@ namespace ix
|
|||||||
int port,
|
int port,
|
||||||
int timeoutSecs);
|
int timeoutSecs);
|
||||||
|
|
||||||
WebSocketInitResult serverHandshake(int fd, int timeoutSecs);
|
WebSocketInitResult serverHandshake(int fd,
|
||||||
|
int timeoutSecs);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::string genRandomString(const int len);
|
std::string genRandomString(const int len);
|
||||||
@ -64,6 +65,8 @@ namespace ix
|
|||||||
// Parse HTTP headers
|
// Parse HTTP headers
|
||||||
WebSocketInitResult sendErrorResponse(int code, const std::string& reason);
|
WebSocketInitResult sendErrorResponse(int code, const std::string& reason);
|
||||||
|
|
||||||
|
std::tuple<std::string, std::string, std::string> parseRequestLine(const std::string& line);
|
||||||
|
std::string trim(const std::string& str);
|
||||||
bool insensitiveStringCompare(const std::string& a, const std::string& b);
|
bool insensitiveStringCompare(const std::string& a, const std::string& b);
|
||||||
|
|
||||||
std::atomic<bool>& _requestInitCancellation;
|
std::atomic<bool>& _requestInitCancellation;
|
||||||
@ -72,4 +75,4 @@ namespace ix
|
|||||||
WebSocketPerMessageDeflateOptions& _perMessageDeflateOptions;
|
WebSocketPerMessageDeflateOptions& _perMessageDeflateOptions;
|
||||||
std::atomic<bool>& _enablePerMessageDeflate;
|
std::atomic<bool>& _enablePerMessageDeflate;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -7,9 +7,10 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXCancellationRequest.h"
|
#include "IXCancellationRequest.h"
|
||||||
|
|
||||||
|
#include <string>
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <string>
|
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
@ -20,14 +21,15 @@ namespace ix
|
|||||||
// Case Insensitive compare_less binary function
|
// Case Insensitive compare_less binary function
|
||||||
struct NocaseCompare
|
struct NocaseCompare
|
||||||
{
|
{
|
||||||
bool operator()(const unsigned char& c1, const unsigned char& c2) const;
|
bool operator() (const unsigned char& c1, const unsigned char& c2) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
bool operator()(const std::string& s1, const std::string& s2) const;
|
bool operator() (const std::string & s1, const std::string & s2) const;
|
||||||
};
|
};
|
||||||
|
|
||||||
using WebSocketHttpHeaders = std::map<std::string, std::string, CaseInsensitiveLess>;
|
using WebSocketHttpHeaders = std::map<std::string, std::string, CaseInsensitiveLess>;
|
||||||
|
|
||||||
std::pair<bool, WebSocketHttpHeaders> parseHttpHeaders(
|
std::pair<bool, WebSocketHttpHeaders> parseHttpHeaders(
|
||||||
std::shared_ptr<Socket> socket, const CancellationRequest& isCancellationRequested);
|
std::shared_ptr<Socket> socket,
|
||||||
} // namespace ix
|
const CancellationRequest& isCancellationRequested);
|
||||||
|
}
|
||||||
|
@ -1,49 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXWebSocketMessage.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2017-2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include "IXWebSocketCloseInfo.h"
|
|
||||||
#include "IXWebSocketErrorInfo.h"
|
|
||||||
#include "IXWebSocketMessageType.h"
|
|
||||||
#include "IXWebSocketOpenInfo.h"
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
struct WebSocketMessage
|
|
||||||
{
|
|
||||||
WebSocketMessageType type;
|
|
||||||
std::string str;
|
|
||||||
size_t wireSize;
|
|
||||||
WebSocketErrorInfo errorInfo;
|
|
||||||
WebSocketOpenInfo openInfo;
|
|
||||||
WebSocketCloseInfo closeInfo;
|
|
||||||
bool binary;
|
|
||||||
|
|
||||||
WebSocketMessage(WebSocketMessageType t,
|
|
||||||
const std::string& s,
|
|
||||||
size_t w,
|
|
||||||
WebSocketErrorInfo e,
|
|
||||||
WebSocketOpenInfo o,
|
|
||||||
WebSocketCloseInfo c,
|
|
||||||
bool b = false)
|
|
||||||
: type(t)
|
|
||||||
, str(std::move(s))
|
|
||||||
, wireSize(w)
|
|
||||||
, errorInfo(e)
|
|
||||||
, openInfo(o)
|
|
||||||
, closeInfo(c)
|
|
||||||
, binary(b)
|
|
||||||
{
|
|
||||||
;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
using WebSocketMessagePtr = std::shared_ptr<WebSocketMessage>;
|
|
||||||
} // namespace ix
|
|
@ -32,7 +32,14 @@ namespace ix
|
|||||||
if (_websocket)
|
if (_websocket)
|
||||||
{
|
{
|
||||||
// set dummy callback just to avoid crash
|
// set dummy callback just to avoid crash
|
||||||
_websocket->setOnMessageCallback([](const WebSocketMessagePtr&) {});
|
_websocket->setOnMessageCallback([](
|
||||||
|
WebSocketMessageType,
|
||||||
|
const std::string&,
|
||||||
|
size_t,
|
||||||
|
const WebSocketErrorInfo&,
|
||||||
|
const WebSocketOpenInfo&,
|
||||||
|
const WebSocketCloseInfo&)
|
||||||
|
{});
|
||||||
}
|
}
|
||||||
|
|
||||||
_websocket = websocket;
|
_websocket = websocket;
|
||||||
@ -40,10 +47,27 @@ namespace ix
|
|||||||
// bind new
|
// bind new
|
||||||
if (_websocket)
|
if (_websocket)
|
||||||
{
|
{
|
||||||
_websocket->setOnMessageCallback([this](const WebSocketMessagePtr& msg)
|
_websocket->setOnMessageCallback([this](
|
||||||
|
WebSocketMessageType type,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const WebSocketErrorInfo& errorInfo,
|
||||||
|
const WebSocketOpenInfo& openInfo,
|
||||||
|
const WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lock(_messagesMutex);
|
MessagePtr message(new Message());
|
||||||
_messages.emplace_back(std::move(msg));
|
|
||||||
|
message->type = type;
|
||||||
|
message->str = str;
|
||||||
|
message->wireSize = wireSize;
|
||||||
|
message->errorInfo = errorInfo;
|
||||||
|
message->openInfo = openInfo;
|
||||||
|
message->closeInfo = closeInfo;
|
||||||
|
|
||||||
|
{
|
||||||
|
std::lock_guard<std::mutex> lock(_messagesMutex);
|
||||||
|
_messages.emplace_back(std::move(message));
|
||||||
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -57,10 +81,10 @@ namespace ix
|
|||||||
{
|
{
|
||||||
_onMessageUserCallback = std::move(callback);
|
_onMessageUserCallback = std::move(callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
WebSocketMessagePtr WebSocketMessageQueue::popMessage()
|
WebSocketMessageQueue::MessagePtr WebSocketMessageQueue::popMessage()
|
||||||
{
|
{
|
||||||
WebSocketMessagePtr message;
|
MessagePtr message;
|
||||||
std::lock_guard<std::mutex> lock(_messagesMutex);
|
std::lock_guard<std::mutex> lock(_messagesMutex);
|
||||||
|
|
||||||
if (!_messages.empty())
|
if (!_messages.empty())
|
||||||
@ -77,11 +101,19 @@ namespace ix
|
|||||||
if (!_onMessageUserCallback)
|
if (!_onMessageUserCallback)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
WebSocketMessagePtr message;
|
MessagePtr message;
|
||||||
|
|
||||||
while (count > 0 && (message = popMessage()))
|
while (count > 0 && (message = popMessage()))
|
||||||
{
|
{
|
||||||
_onMessageUserCallback(message);
|
_onMessageUserCallback(
|
||||||
|
message->type,
|
||||||
|
message->str,
|
||||||
|
message->wireSize,
|
||||||
|
message->errorInfo,
|
||||||
|
message->openInfo,
|
||||||
|
message->closeInfo
|
||||||
|
);
|
||||||
|
|
||||||
--count;
|
--count;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -7,9 +7,9 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXWebSocket.h"
|
#include "IXWebSocket.h"
|
||||||
|
#include <thread>
|
||||||
#include <list>
|
#include <list>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <thread>
|
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
@ -30,12 +30,24 @@ namespace ix
|
|||||||
void poll(int count = 512);
|
void poll(int count = 512);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
WebSocketMessagePtr popMessage();
|
struct Message
|
||||||
|
{
|
||||||
|
WebSocketMessageType type;
|
||||||
|
std::string str;
|
||||||
|
size_t wireSize;
|
||||||
|
WebSocketErrorInfo errorInfo;
|
||||||
|
WebSocketOpenInfo openInfo;
|
||||||
|
WebSocketCloseInfo closeInfo;
|
||||||
|
};
|
||||||
|
|
||||||
|
using MessagePtr = std::shared_ptr<Message>;
|
||||||
|
|
||||||
|
MessagePtr popMessage();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
WebSocket* _websocket = nullptr;
|
WebSocket* _websocket = nullptr;
|
||||||
OnMessageCallback _onMessageUserCallback;
|
OnMessageCallback _onMessageUserCallback;
|
||||||
std::mutex _messagesMutex;
|
std::mutex _messagesMutex;
|
||||||
std::list<WebSocketMessagePtr> _messages;
|
std::list<MessagePtr> _messages;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -1,21 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXWebSocketMessageType.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2017-2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
enum class WebSocketMessageType
|
|
||||||
{
|
|
||||||
Message = 0,
|
|
||||||
Open = 1,
|
|
||||||
Close = 2,
|
|
||||||
Error = 3,
|
|
||||||
Ping = 4,
|
|
||||||
Pong = 5,
|
|
||||||
Fragment = 6
|
|
||||||
};
|
|
||||||
}
|
|
@ -1,24 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXWebSocketOpenInfo.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2017-2019 Machine Zone, Inc. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
struct WebSocketOpenInfo
|
|
||||||
{
|
|
||||||
std::string uri;
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
|
|
||||||
WebSocketOpenInfo(const std::string& u = std::string(),
|
|
||||||
const WebSocketHttpHeaders& h = WebSocketHttpHeaders())
|
|
||||||
: uri(u)
|
|
||||||
, headers(h)
|
|
||||||
{
|
|
||||||
;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
} // namespace ix
|
|
@ -34,8 +34,8 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
@ -57,4 +57,4 @@ namespace ix
|
|||||||
std::unique_ptr<WebSocketPerMessageDeflateCompressor> _compressor;
|
std::unique_ptr<WebSocketPerMessageDeflateCompressor> _compressor;
|
||||||
std::unique_ptr<WebSocketPerMessageDeflateDecompressor> _decompressor;
|
std::unique_ptr<WebSocketPerMessageDeflateDecompressor> _decompressor;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -7,8 +7,8 @@
|
|||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "zlib.h"
|
#include "zlib.h"
|
||||||
#include <memory>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
@ -46,4 +46,5 @@ namespace ix
|
|||||||
z_stream _inflateState;
|
z_stream _inflateState;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace ix
|
}
|
||||||
|
|
||||||
|
@ -42,4 +42,4 @@ namespace ix
|
|||||||
int _clientMaxWindowBits;
|
int _clientMaxWindowBits;
|
||||||
int _serverMaxWindowBits;
|
int _serverMaxWindowBits;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -15,7 +15,8 @@ namespace ix
|
|||||||
size_t payloadSize;
|
size_t payloadSize;
|
||||||
size_t wireSize;
|
size_t wireSize;
|
||||||
|
|
||||||
WebSocketSendInfo(bool s = false, bool c = false, size_t p = 0, size_t w = 0)
|
WebSocketSendInfo(bool s = false, bool c = false,
|
||||||
|
size_t p = 0, size_t w = 0)
|
||||||
: success(s)
|
: success(s)
|
||||||
, compressionError(c)
|
, compressionError(c)
|
||||||
, payloadSize(p)
|
, payloadSize(p)
|
||||||
@ -24,4 +25,4 @@ namespace ix
|
|||||||
;
|
;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -6,25 +6,25 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXSocketServer.h"
|
#include <utility> // pair
|
||||||
#include "IXWebSocket.h"
|
#include <string>
|
||||||
#include <condition_variable>
|
#include <set>
|
||||||
|
#include <thread>
|
||||||
|
#include <mutex>
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
#include <mutex>
|
#include <condition_variable>
|
||||||
#include <set>
|
|
||||||
#include <string>
|
#include "IXWebSocket.h"
|
||||||
#include <thread>
|
#include "IXSocketServer.h"
|
||||||
#include <utility> // pair
|
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
class WebSocketServer final : public SocketServer
|
using OnConnectionCallback = std::function<void(std::shared_ptr<WebSocket>,
|
||||||
{
|
std::shared_ptr<ConnectionState>)>;
|
||||||
public:
|
|
||||||
using OnConnectionCallback =
|
|
||||||
std::function<void(std::shared_ptr<WebSocket>, std::shared_ptr<ConnectionState>)>;
|
|
||||||
|
|
||||||
|
class WebSocketServer final : public SocketServer {
|
||||||
|
public:
|
||||||
WebSocketServer(int port = SocketServer::kDefaultPort,
|
WebSocketServer(int port = SocketServer::kDefaultPort,
|
||||||
const std::string& host = SocketServer::kDefaultHost,
|
const std::string& host = SocketServer::kDefaultHost,
|
||||||
int backlog = SocketServer::kDefaultTcpBacklog,
|
int backlog = SocketServer::kDefaultTcpBacklog,
|
||||||
@ -59,4 +59,4 @@ namespace ix
|
|||||||
std::shared_ptr<ConnectionState> connectionState) final;
|
std::shared_ptr<ConnectionState> connectionState) final;
|
||||||
virtual size_t getConnectedClientsCount() final;
|
virtual size_t getConnectedClientsCount() final;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -330,7 +330,7 @@ namespace ix
|
|||||||
}
|
}
|
||||||
|
|
||||||
// poll the socket
|
// poll the socket
|
||||||
PollResultType pollResult = _socket->isReadyToRead(lastingTimeoutDelayInMs);
|
PollResultType pollResult = _socket->poll(lastingTimeoutDelayInMs);
|
||||||
|
|
||||||
// Make sure we send all the buffered data
|
// Make sure we send all the buffered data
|
||||||
// there can be a lot of it for large messages.
|
// there can be a lot of it for large messages.
|
||||||
@ -542,17 +542,12 @@ namespace ix
|
|||||||
) {
|
) {
|
||||||
unmaskReceiveBuffer(ws);
|
unmaskReceiveBuffer(ws);
|
||||||
|
|
||||||
MessageKind messageKind =
|
|
||||||
(ws.opcode == wsheader_type::TEXT_FRAME)
|
|
||||||
? MessageKind::MSG_TEXT
|
|
||||||
: MessageKind::MSG_BINARY;
|
|
||||||
|
|
||||||
//
|
//
|
||||||
// Usual case. Small unfragmented messages
|
// Usual case. Small unfragmented messages
|
||||||
//
|
//
|
||||||
if (ws.fin && _chunks.empty())
|
if (ws.fin && _chunks.empty())
|
||||||
{
|
{
|
||||||
emitMessage(messageKind,
|
emitMessage(MessageKind::MSG,
|
||||||
std::string(_rxbuf.begin()+ws.header_size,
|
std::string(_rxbuf.begin()+ws.header_size,
|
||||||
_rxbuf.begin()+ws.header_size+(size_t) ws.N),
|
_rxbuf.begin()+ws.header_size+(size_t) ws.N),
|
||||||
ws,
|
ws,
|
||||||
@ -572,7 +567,7 @@ namespace ix
|
|||||||
_rxbuf.begin()+ws.header_size+(size_t)ws.N));
|
_rxbuf.begin()+ws.header_size+(size_t)ws.N));
|
||||||
if (ws.fin)
|
if (ws.fin)
|
||||||
{
|
{
|
||||||
emitMessage(messageKind, getMergedChunks(), ws, onMessageCallback);
|
emitMessage(MessageKind::MSG, getMergedChunks(), ws, onMessageCallback);
|
||||||
_chunks.clear();
|
_chunks.clear();
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -755,7 +750,7 @@ namespace ix
|
|||||||
{
|
{
|
||||||
if (_readyState != ReadyState::OPEN && _readyState != ReadyState::CLOSING)
|
if (_readyState != ReadyState::OPEN && _readyState != ReadyState::CLOSING)
|
||||||
{
|
{
|
||||||
return WebSocketSendInfo(false);
|
return WebSocketSendInfo();
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t payloadSize = message.size();
|
size_t payloadSize = message.size();
|
||||||
@ -1046,7 +1041,7 @@ namespace ix
|
|||||||
_requestInitCancellation = true;
|
_requestInitCancellation = true;
|
||||||
|
|
||||||
if (_readyState == ReadyState::CLOSING || _readyState == ReadyState::CLOSED) return;
|
if (_readyState == ReadyState::CLOSING || _readyState == ReadyState::CLOSED) return;
|
||||||
|
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lock(_closeDataMutex);
|
std::lock_guard<std::mutex> lock(_closeDataMutex);
|
||||||
_closeCode = code;
|
_closeCode = code;
|
||||||
|
@ -10,21 +10,22 @@
|
|||||||
// Adapted from https://github.com/dhbaird/easywsclient
|
// Adapted from https://github.com/dhbaird/easywsclient
|
||||||
//
|
//
|
||||||
|
|
||||||
#include "IXCancellationRequest.h"
|
|
||||||
#include "IXProgressCallback.h"
|
|
||||||
#include "IXWebSocketCloseConstants.h"
|
|
||||||
#include "IXWebSocketHandshake.h"
|
|
||||||
#include "IXWebSocketHttpHeaders.h"
|
|
||||||
#include "IXWebSocketPerMessageDeflate.h"
|
|
||||||
#include "IXWebSocketPerMessageDeflateOptions.h"
|
|
||||||
#include "IXWebSocketSendInfo.h"
|
|
||||||
#include <atomic>
|
|
||||||
#include <functional>
|
|
||||||
#include <list>
|
|
||||||
#include <memory>
|
|
||||||
#include <mutex>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <functional>
|
||||||
|
#include <memory>
|
||||||
|
#include <mutex>
|
||||||
|
#include <atomic>
|
||||||
|
#include <list>
|
||||||
|
|
||||||
|
#include "IXWebSocketSendInfo.h"
|
||||||
|
#include "IXWebSocketPerMessageDeflate.h"
|
||||||
|
#include "IXWebSocketPerMessageDeflateOptions.h"
|
||||||
|
#include "IXWebSocketHttpHeaders.h"
|
||||||
|
#include "IXCancellationRequest.h"
|
||||||
|
#include "IXWebSocketHandshake.h"
|
||||||
|
#include "IXProgressCallback.h"
|
||||||
|
#include "IXWebSocketCloseConstants.h"
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
@ -50,8 +51,7 @@ namespace ix
|
|||||||
|
|
||||||
enum class MessageKind
|
enum class MessageKind
|
||||||
{
|
{
|
||||||
MSG_TEXT,
|
MSG,
|
||||||
MSG_BINARY,
|
|
||||||
PING,
|
PING,
|
||||||
PONG,
|
PONG,
|
||||||
FRAGMENT
|
FRAGMENT
|
||||||
@ -63,9 +63,14 @@ namespace ix
|
|||||||
AbnormalClose
|
AbnormalClose
|
||||||
};
|
};
|
||||||
|
|
||||||
using OnMessageCallback =
|
using OnMessageCallback = std::function<void(const std::string&,
|
||||||
std::function<void(const std::string&, size_t, bool, MessageKind)>;
|
size_t,
|
||||||
using OnCloseCallback = std::function<void(uint16_t, const std::string&, size_t, bool)>;
|
bool,
|
||||||
|
MessageKind)>;
|
||||||
|
using OnCloseCallback = std::function<void(uint16_t,
|
||||||
|
const std::string&,
|
||||||
|
size_t,
|
||||||
|
bool)>;
|
||||||
|
|
||||||
WebSocketTransport();
|
WebSocketTransport();
|
||||||
~WebSocketTransport();
|
~WebSocketTransport();
|
||||||
@ -77,7 +82,7 @@ namespace ix
|
|||||||
|
|
||||||
WebSocketInitResult connectToUrl(const std::string& url, // Client
|
WebSocketInitResult connectToUrl(const std::string& url, // Client
|
||||||
int timeoutSecs);
|
int timeoutSecs);
|
||||||
WebSocketInitResult connectToSocket(int fd, // Server
|
WebSocketInitResult connectToSocket(int fd, // Server
|
||||||
int timeoutSecs);
|
int timeoutSecs);
|
||||||
|
|
||||||
PollResult poll();
|
PollResult poll();
|
||||||
@ -98,26 +103,25 @@ namespace ix
|
|||||||
ReadyState getReadyState() const;
|
ReadyState getReadyState() const;
|
||||||
void setReadyState(ReadyState readyState);
|
void setReadyState(ReadyState readyState);
|
||||||
void setOnCloseCallback(const OnCloseCallback& onCloseCallback);
|
void setOnCloseCallback(const OnCloseCallback& onCloseCallback);
|
||||||
void dispatch(PollResult pollResult, const OnMessageCallback& onMessageCallback);
|
void dispatch(PollResult pollResult,
|
||||||
|
const OnMessageCallback& onMessageCallback);
|
||||||
size_t bufferedAmount() const;
|
size_t bufferedAmount() const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::string _url;
|
std::string _url;
|
||||||
|
|
||||||
struct wsheader_type
|
struct wsheader_type {
|
||||||
{
|
|
||||||
unsigned header_size;
|
unsigned header_size;
|
||||||
bool fin;
|
bool fin;
|
||||||
bool rsv1;
|
bool rsv1;
|
||||||
bool mask;
|
bool mask;
|
||||||
enum opcode_type
|
enum opcode_type {
|
||||||
{
|
|
||||||
CONTINUATION = 0x0,
|
CONTINUATION = 0x0,
|
||||||
TEXT_FRAME = 0x1,
|
TEXT_FRAME = 0x1,
|
||||||
BINARY_FRAME = 0x2,
|
BINARY_FRAME = 0x2,
|
||||||
CLOSE = 8,
|
CLOSE = 8,
|
||||||
PING = 9,
|
PING = 9,
|
||||||
PONG = 0xa,
|
PONG = 0xa,
|
||||||
} opcode;
|
} opcode;
|
||||||
int N0;
|
int N0;
|
||||||
uint64_t N;
|
uint64_t N;
|
||||||
@ -172,7 +176,7 @@ namespace ix
|
|||||||
std::atomic<bool> _requestInitCancellation;
|
std::atomic<bool> _requestInitCancellation;
|
||||||
|
|
||||||
mutable std::mutex _closingTimePointMutex;
|
mutable std::mutex _closingTimePointMutex;
|
||||||
std::chrono::time_point<std::chrono::steady_clock> _closingTimePoint;
|
std::chrono::time_point<std::chrono::steady_clock>_closingTimePoint;
|
||||||
static const int kClosingMaximumWaitingDelayInMs;
|
static const int kClosingMaximumWaitingDelayInMs;
|
||||||
|
|
||||||
// enable auto response to ping
|
// enable auto response to ping
|
||||||
@ -207,8 +211,7 @@ namespace ix
|
|||||||
// No PONG data was received through the socket for longer than ping timeout delay
|
// No PONG data was received through the socket for longer than ping timeout delay
|
||||||
bool pingTimeoutExceeded();
|
bool pingTimeoutExceeded();
|
||||||
|
|
||||||
// after calling close(), if no CLOSE frame answer is received back from the remote, we
|
// after calling close(), if no CLOSE frame answer is received back from the remote, we should close the connexion
|
||||||
// should close the connexion
|
|
||||||
bool closingDelayExceeded();
|
bool closingDelayExceeded();
|
||||||
|
|
||||||
void initTimePointsAndGCDAfterConnect();
|
void initTimePointsAndGCDAfterConnect();
|
||||||
@ -249,4 +252,4 @@ namespace ix
|
|||||||
|
|
||||||
std::string getMergedChunks() const;
|
std::string getMergedChunks() const;
|
||||||
};
|
};
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -31,51 +31,48 @@
|
|||||||
|
|
||||||
namespace LUrlParser
|
namespace LUrlParser
|
||||||
{
|
{
|
||||||
enum LUrlParserError
|
enum LUrlParserError
|
||||||
{
|
{
|
||||||
LUrlParserError_Ok = 0,
|
LUrlParserError_Ok = 0,
|
||||||
LUrlParserError_Uninitialized = 1,
|
LUrlParserError_Uninitialized = 1,
|
||||||
LUrlParserError_NoUrlCharacter = 2,
|
LUrlParserError_NoUrlCharacter = 2,
|
||||||
LUrlParserError_InvalidSchemeName = 3,
|
LUrlParserError_InvalidSchemeName = 3,
|
||||||
LUrlParserError_NoDoubleSlash = 4,
|
LUrlParserError_NoDoubleSlash = 4,
|
||||||
LUrlParserError_NoAtSign = 5,
|
LUrlParserError_NoAtSign = 5,
|
||||||
LUrlParserError_UnexpectedEndOfLine = 6,
|
LUrlParserError_UnexpectedEndOfLine = 6,
|
||||||
LUrlParserError_NoSlash = 7,
|
LUrlParserError_NoSlash = 7,
|
||||||
};
|
};
|
||||||
|
|
||||||
class clParseURL
|
class clParseURL
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
LUrlParserError m_ErrorCode;
|
LUrlParserError m_ErrorCode;
|
||||||
std::string m_Scheme;
|
std::string m_Scheme;
|
||||||
std::string m_Host;
|
std::string m_Host;
|
||||||
std::string m_Port;
|
std::string m_Port;
|
||||||
std::string m_Path;
|
std::string m_Path;
|
||||||
std::string m_Query;
|
std::string m_Query;
|
||||||
std::string m_Fragment;
|
std::string m_Fragment;
|
||||||
std::string m_UserName;
|
std::string m_UserName;
|
||||||
std::string m_Password;
|
std::string m_Password;
|
||||||
|
|
||||||
clParseURL()
|
clParseURL()
|
||||||
: m_ErrorCode(LUrlParserError_Uninitialized)
|
: m_ErrorCode( LUrlParserError_Uninitialized )
|
||||||
{
|
{}
|
||||||
}
|
|
||||||
|
|
||||||
/// return 'true' if the parsing was successful
|
/// return 'true' if the parsing was successful
|
||||||
bool IsValid() const { return m_ErrorCode == LUrlParserError_Ok; }
|
bool IsValid() const { return m_ErrorCode == LUrlParserError_Ok; }
|
||||||
|
|
||||||
/// helper to convert the port number to int, return 'true' if the port is valid (within the
|
/// helper to convert the port number to int, return 'true' if the port is valid (within the 0..65535 range)
|
||||||
/// 0..65535 range)
|
bool GetPort( int* OutPort ) const;
|
||||||
bool GetPort(int* OutPort) const;
|
|
||||||
|
|
||||||
/// parse the URL
|
/// parse the URL
|
||||||
static clParseURL ParseURL(const std::string& URL);
|
static clParseURL ParseURL( const std::string& URL );
|
||||||
|
|
||||||
private:
|
private:
|
||||||
explicit clParseURL(LUrlParserError ErrorCode)
|
explicit clParseURL( LUrlParserError ErrorCode )
|
||||||
: m_ErrorCode(ErrorCode)
|
: m_ErrorCode( ErrorCode )
|
||||||
{
|
{}
|
||||||
}
|
};
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace LUrlParser
|
} // namespace LUrlParser
|
||||||
|
15
makefile
15
makefile
@ -9,17 +9,14 @@ install: brew
|
|||||||
# on osx it is good practice to make /usr/local user writable
|
# on osx it is good practice to make /usr/local user writable
|
||||||
# sudo chown -R `whoami`/staff /usr/local
|
# sudo chown -R `whoami`/staff /usr/local
|
||||||
brew:
|
brew:
|
||||||
mkdir -p build && (cd build ; cmake -DCMAKE_BUILD_TYPE=Debug -DUSE_TLS=1 -DUSE_WS=1 .. ; make -j install)
|
mkdir -p build && (cd build ; cmake -DUSE_TLS=1 -DUSE_WS=1 .. ; make -j install)
|
||||||
|
|
||||||
ws:
|
ws:
|
||||||
mkdir -p build && (cd build ; cmake -DCMAKE_BUILD_TYPE=Debug -DUSE_TLS=1 -DUSE_WS=1 -DUSE_MBED_TLS=1 .. ; make -j)
|
mkdir -p build && (cd build ; cmake -DUSE_TLS=1 -DUSE_WS=1 .. ; make -j)
|
||||||
|
|
||||||
uninstall:
|
uninstall:
|
||||||
xargs rm -fv < build/install_manifest.txt
|
xargs rm -fv < build/install_manifest.txt
|
||||||
|
|
||||||
tag:
|
|
||||||
git tag v"`cat DOCKER_VERSION`"
|
|
||||||
|
|
||||||
.PHONY: docker
|
.PHONY: docker
|
||||||
|
|
||||||
NAME := bsergean/ws
|
NAME := bsergean/ws
|
||||||
@ -28,9 +25,6 @@ IMG := ${NAME}:${TAG}
|
|||||||
LATEST := ${NAME}:latest
|
LATEST := ${NAME}:latest
|
||||||
BUILD := ${NAME}:build
|
BUILD := ${NAME}:build
|
||||||
|
|
||||||
docker_test:
|
|
||||||
docker build -f docker/Dockerfile.debian -t bsergean/ixwebsocket_test:build .
|
|
||||||
|
|
||||||
docker:
|
docker:
|
||||||
docker build -t ${IMG} .
|
docker build -t ${IMG} .
|
||||||
docker tag ${IMG} ${BUILD}
|
docker tag ${IMG} ${BUILD}
|
||||||
@ -40,15 +34,12 @@ docker_push:
|
|||||||
docker push ${LATEST}
|
docker push ${LATEST}
|
||||||
|
|
||||||
run:
|
run:
|
||||||
docker run --cap-add sys_ptrace --entrypoint=sh -it bsergean/ws:build
|
docker run --cap-add sys_ptrace --entrypoint=bash -it bsergean/ws:build
|
||||||
|
|
||||||
# this is helpful to remove trailing whitespaces
|
# this is helpful to remove trailing whitespaces
|
||||||
trail:
|
trail:
|
||||||
sh third_party/remote_trailing_whitespaces.sh
|
sh third_party/remote_trailing_whitespaces.sh
|
||||||
|
|
||||||
format:
|
|
||||||
find test ixwebsocket ws -name '*.cpp' -o -name '*.h' -exec clang-format -i {} \;
|
|
||||||
|
|
||||||
# That target is used to start a node server, but isn't required as we have
|
# That target is used to start a node server, but isn't required as we have
|
||||||
# a builtin C++ server started in the unittest now
|
# a builtin C++ server started in the unittest now
|
||||||
test_server:
|
test_server:
|
||||||
|
@ -7,7 +7,7 @@ project (ixwebsocket_unittest)
|
|||||||
|
|
||||||
set (CMAKE_CXX_STANDARD 14)
|
set (CMAKE_CXX_STANDARD 14)
|
||||||
|
|
||||||
if (MAC)
|
if (NOT WIN32)
|
||||||
set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/../third_party/sanitizers-cmake/cmake" ${CMAKE_MODULE_PATH})
|
set(CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/../third_party/sanitizers-cmake/cmake" ${CMAKE_MODULE_PATH})
|
||||||
find_package(Sanitizers)
|
find_package(Sanitizers)
|
||||||
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=thread")
|
set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -fsanitize=thread")
|
||||||
@ -28,7 +28,6 @@ include_directories(
|
|||||||
set (SOURCES
|
set (SOURCES
|
||||||
test_runner.cpp
|
test_runner.cpp
|
||||||
IXTest.cpp
|
IXTest.cpp
|
||||||
IXGetFreePort.cpp
|
|
||||||
../third_party/msgpack11/msgpack11.cpp
|
../third_party/msgpack11/msgpack11.cpp
|
||||||
../ws/ixcore/utils/IXCoreLogger.cpp
|
../ws/ixcore/utils/IXCoreLogger.cpp
|
||||||
|
|
||||||
@ -38,9 +37,6 @@ set (SOURCES
|
|||||||
IXWebSocketTestConnectionDisconnection.cpp
|
IXWebSocketTestConnectionDisconnection.cpp
|
||||||
IXUrlParserTest.cpp
|
IXUrlParserTest.cpp
|
||||||
IXWebSocketServerTest.cpp
|
IXWebSocketServerTest.cpp
|
||||||
IXHttpClientTest.cpp
|
|
||||||
IXHttpServerTest.cpp
|
|
||||||
IXUnityBuildsTest.cpp
|
|
||||||
)
|
)
|
||||||
|
|
||||||
# Some unittest don't work on windows yet
|
# Some unittest don't work on windows yet
|
||||||
@ -49,6 +45,8 @@ if (UNIX)
|
|||||||
IXDNSLookupTest.cpp
|
IXDNSLookupTest.cpp
|
||||||
cmd_websocket_chat.cpp
|
cmd_websocket_chat.cpp
|
||||||
IXWebSocketCloseTest.cpp
|
IXWebSocketCloseTest.cpp
|
||||||
|
IXWebSocketPingTest.cpp
|
||||||
|
IXWebSocketPingTimeoutTest.cpp
|
||||||
)
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
@ -59,15 +57,11 @@ if (MAC OR WIN32)
|
|||||||
)
|
)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
# Ping test fails intermittently, disabling them for now
|
|
||||||
# IXWebSocketPingTest.cpp
|
|
||||||
# IXWebSocketPingTimeoutTest.cpp
|
|
||||||
|
|
||||||
# Disable tests for now that are failing or not reliable
|
# Disable tests for now that are failing or not reliable
|
||||||
|
|
||||||
add_executable(ixwebsocket_unittest ${SOURCES})
|
add_executable(ixwebsocket_unittest ${SOURCES})
|
||||||
|
|
||||||
if (MAC)
|
if (NOT WIN32)
|
||||||
add_sanitizers(ixwebsocket_unittest)
|
add_sanitizers(ixwebsocket_unittest)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
@ -17,33 +17,33 @@ TEST_CASE("dns", "[net]")
|
|||||||
{
|
{
|
||||||
SECTION("Test resolving a known hostname")
|
SECTION("Test resolving a known hostname")
|
||||||
{
|
{
|
||||||
auto dnsLookup = std::make_shared<DNSLookup>("www.google.com", 80);
|
DNSLookup dnsLookup("www.google.com", 80);
|
||||||
|
|
||||||
std::string errMsg;
|
std::string errMsg;
|
||||||
struct addrinfo* res;
|
struct addrinfo* res;
|
||||||
|
|
||||||
res = dnsLookup->resolve(errMsg, [] { return false; });
|
res = dnsLookup.resolve(errMsg, [] { return false; });
|
||||||
std::cerr << "Error message: " << errMsg << std::endl;
|
std::cerr << "Error message: " << errMsg << std::endl;
|
||||||
REQUIRE(res != nullptr);
|
REQUIRE(res != nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION("Test resolving a non-existing hostname")
|
SECTION("Test resolving a non-existing hostname")
|
||||||
{
|
{
|
||||||
auto dnsLookup = std::make_shared<DNSLookup>("wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww", 80);
|
DNSLookup dnsLookup("wwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwwww", 80);
|
||||||
|
|
||||||
std::string errMsg;
|
std::string errMsg;
|
||||||
struct addrinfo* res = dnsLookup->resolve(errMsg, [] { return false; });
|
struct addrinfo* res = dnsLookup.resolve(errMsg, [] { return false; });
|
||||||
std::cerr << "Error message: " << errMsg << std::endl;
|
std::cerr << "Error message: " << errMsg << std::endl;
|
||||||
REQUIRE(res == nullptr);
|
REQUIRE(res == nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
SECTION("Test resolving a good hostname, with cancellation")
|
SECTION("Test resolving a good hostname, with cancellation")
|
||||||
{
|
{
|
||||||
auto dnsLookup = std::make_shared<DNSLookup>("www.google.com", 80, 1);
|
DNSLookup dnsLookup("www.google.com", 80, 1);
|
||||||
|
|
||||||
std::string errMsg;
|
std::string errMsg;
|
||||||
// The callback returning true means we are requesting cancellation
|
// The callback returning true means we are requesting cancellation
|
||||||
struct addrinfo* res = dnsLookup->resolve(errMsg, [] { return true; });
|
struct addrinfo* res = dnsLookup.resolve(errMsg, [] { return true; });
|
||||||
std::cerr << "Error message: " << errMsg << std::endl;
|
std::cerr << "Error message: " << errMsg << std::endl;
|
||||||
REQUIRE(res == nullptr);
|
REQUIRE(res == nullptr);
|
||||||
}
|
}
|
||||||
|
@ -1,93 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXGetFreePort.cpp
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "IXGetFreePort.h"
|
|
||||||
#include <ixwebsocket/IXNetSystem.h>
|
|
||||||
#include <ixwebsocket/IXSocket.h>
|
|
||||||
|
|
||||||
#include <string>
|
|
||||||
#include <random>
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
int getAnyFreePortRandom()
|
|
||||||
{
|
|
||||||
std::random_device rd;
|
|
||||||
std::uniform_int_distribution<int> dist(1024 + 1, 65535);
|
|
||||||
|
|
||||||
return dist(rd);
|
|
||||||
}
|
|
||||||
|
|
||||||
int getAnyFreePort()
|
|
||||||
{
|
|
||||||
int defaultPort = 8090;
|
|
||||||
int sockfd;
|
|
||||||
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
|
|
||||||
{
|
|
||||||
return getAnyFreePortRandom();
|
|
||||||
}
|
|
||||||
|
|
||||||
int enable = 1;
|
|
||||||
if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR,
|
|
||||||
(char*) &enable, sizeof(enable)) < 0)
|
|
||||||
{
|
|
||||||
return getAnyFreePortRandom();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Bind to port 0. This is the standard way to get a free port.
|
|
||||||
struct sockaddr_in server; // server address information
|
|
||||||
server.sin_family = AF_INET;
|
|
||||||
server.sin_port = htons(0);
|
|
||||||
server.sin_addr.s_addr = inet_addr("127.0.0.1");
|
|
||||||
|
|
||||||
if (bind(sockfd, (struct sockaddr *)&server, sizeof(server)) < 0)
|
|
||||||
{
|
|
||||||
Socket::closeSocket(sockfd);
|
|
||||||
return getAnyFreePortRandom();
|
|
||||||
}
|
|
||||||
|
|
||||||
struct sockaddr_in sa; // server address information
|
|
||||||
socklen_t len = sizeof(sa);
|
|
||||||
if (getsockname(sockfd, (struct sockaddr *) &sa, &len) < 0)
|
|
||||||
{
|
|
||||||
Socket::closeSocket(sockfd);
|
|
||||||
return getAnyFreePortRandom();
|
|
||||||
}
|
|
||||||
|
|
||||||
int port = ntohs(sa.sin_port);
|
|
||||||
Socket::closeSocket(sockfd);
|
|
||||||
|
|
||||||
return port;
|
|
||||||
}
|
|
||||||
|
|
||||||
int getFreePort()
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
#if defined(__has_feature)
|
|
||||||
# if __has_feature(address_sanitizer)
|
|
||||||
int port = getAnyFreePortRandom();
|
|
||||||
# else
|
|
||||||
int port = getAnyFreePort();
|
|
||||||
# endif
|
|
||||||
#else
|
|
||||||
int port = getAnyFreePort();
|
|
||||||
#endif
|
|
||||||
//
|
|
||||||
// Only port above 1024 can be used by non root users, but for some
|
|
||||||
// reason I got port 7 returned with macOS when binding on port 0...
|
|
||||||
//
|
|
||||||
if (port > 1024)
|
|
||||||
{
|
|
||||||
return port;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
} // namespace ix
|
|
||||||
|
|
||||||
|
|
@ -1,12 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXGetFreePort.h
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
|
|
||||||
namespace ix
|
|
||||||
{
|
|
||||||
int getFreePort();
|
|
||||||
} // namespace ix
|
|
@ -1,233 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXSocketTest.cpp
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <iostream>
|
|
||||||
#include <ixwebsocket/IXHttpClient.h>
|
|
||||||
|
|
||||||
#include "catch.hpp"
|
|
||||||
|
|
||||||
using namespace ix;
|
|
||||||
|
|
||||||
TEST_CASE("http client", "[http]")
|
|
||||||
{
|
|
||||||
SECTION("Connect to a remote HTTP server")
|
|
||||||
{
|
|
||||||
HttpClient httpClient;
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
|
|
||||||
std::string url("http://httpbin.org/");
|
|
||||||
auto args = httpClient.createRequest(url);
|
|
||||||
|
|
||||||
args->extraHeaders = headers;
|
|
||||||
args->connectTimeout = 60;
|
|
||||||
args->transferTimeout = 60;
|
|
||||||
args->followRedirects = true;
|
|
||||||
args->maxRedirects = 10;
|
|
||||||
args->verbose = true;
|
|
||||||
args->compress = true;
|
|
||||||
args->logger = [](const std::string& msg)
|
|
||||||
{
|
|
||||||
std::cout << msg;
|
|
||||||
};
|
|
||||||
args->onProgressCallback = [](int current, int total) -> bool
|
|
||||||
{
|
|
||||||
std::cerr << "\r" << "Downloaded "
|
|
||||||
<< current << " bytes out of " << total;
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
auto response = httpClient.get(url, args);
|
|
||||||
|
|
||||||
for (auto it : response->headers)
|
|
||||||
{
|
|
||||||
std::cerr << it.first << ": " << it.second << std::endl;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::cerr << "Upload size: " << response->uploadSize << std::endl;
|
|
||||||
std::cerr << "Download size: " << response->downloadSize << std::endl;
|
|
||||||
std::cerr << "Status: " << response->statusCode << std::endl;
|
|
||||||
std::cerr << "Error message: " << response->errorMsg << std::endl;
|
|
||||||
|
|
||||||
REQUIRE(response->errorCode == HttpErrorCode::Ok);
|
|
||||||
REQUIRE(response->statusCode == 200);
|
|
||||||
}
|
|
||||||
|
|
||||||
SECTION("Connect to a remote HTTPS server")
|
|
||||||
{
|
|
||||||
HttpClient httpClient;
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
|
|
||||||
std::string url("https://httpbin.org/");
|
|
||||||
auto args = httpClient.createRequest(url);
|
|
||||||
|
|
||||||
args->extraHeaders = headers;
|
|
||||||
args->connectTimeout = 60;
|
|
||||||
args->transferTimeout = 60;
|
|
||||||
args->followRedirects = true;
|
|
||||||
args->maxRedirects = 10;
|
|
||||||
args->verbose = true;
|
|
||||||
args->compress = true;
|
|
||||||
args->logger = [](const std::string& msg)
|
|
||||||
{
|
|
||||||
std::cout << msg;
|
|
||||||
};
|
|
||||||
args->onProgressCallback = [](int current, int total) -> bool
|
|
||||||
{
|
|
||||||
std::cerr << "\r" << "Downloaded "
|
|
||||||
<< current << " bytes out of " << total;
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
auto response = httpClient.get(url, args);
|
|
||||||
|
|
||||||
for (auto it : response->headers)
|
|
||||||
{
|
|
||||||
std::cerr << it.first << ": " << it.second << std::endl;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::cerr << "Upload size: " << response->uploadSize << std::endl;
|
|
||||||
std::cerr << "Download size: " << response->downloadSize << std::endl;
|
|
||||||
std::cerr << "Status: " << response->statusCode << std::endl;
|
|
||||||
std::cerr << "Error message: " << response->errorMsg << std::endl;
|
|
||||||
|
|
||||||
REQUIRE(response->errorCode == HttpErrorCode::Ok);
|
|
||||||
REQUIRE(response->statusCode == 200);
|
|
||||||
}
|
|
||||||
|
|
||||||
SECTION("Async API, one call")
|
|
||||||
{
|
|
||||||
bool async = true;
|
|
||||||
HttpClient httpClient(async);
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
|
|
||||||
std::string url("https://httpbin.org/");
|
|
||||||
auto args = httpClient.createRequest(url);
|
|
||||||
|
|
||||||
args->extraHeaders = headers;
|
|
||||||
args->connectTimeout = 60;
|
|
||||||
args->transferTimeout = 60;
|
|
||||||
args->followRedirects = true;
|
|
||||||
args->maxRedirects = 10;
|
|
||||||
args->verbose = true;
|
|
||||||
args->compress = true;
|
|
||||||
args->logger = [](const std::string& msg)
|
|
||||||
{
|
|
||||||
std::cout << msg;
|
|
||||||
};
|
|
||||||
args->onProgressCallback = [](int current, int total) -> bool
|
|
||||||
{
|
|
||||||
std::cerr << "\r" << "Downloaded "
|
|
||||||
<< current << " bytes out of " << total;
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
std::atomic<bool> requestCompleted(false);
|
|
||||||
std::atomic<int> statusCode(0);
|
|
||||||
|
|
||||||
httpClient.performRequest(args, [&requestCompleted, &statusCode]
|
|
||||||
(const HttpResponsePtr& response)
|
|
||||||
{
|
|
||||||
std::cerr << "Upload size: " << response->uploadSize << std::endl;
|
|
||||||
std::cerr << "Download size: " << response->downloadSize << std::endl;
|
|
||||||
std::cerr << "Status: " << response->statusCode << std::endl;
|
|
||||||
std::cerr << "Error message: " << response->errorMsg << std::endl;
|
|
||||||
|
|
||||||
// In case of failure, print response->errorMsg
|
|
||||||
statusCode = response->statusCode;
|
|
||||||
requestCompleted = true;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
int wait = 0;
|
|
||||||
while (wait < 5000)
|
|
||||||
{
|
|
||||||
if (requestCompleted) break;
|
|
||||||
|
|
||||||
std::chrono::duration<double, std::milli> duration(10);
|
|
||||||
std::this_thread::sleep_for(duration);
|
|
||||||
wait += 10;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::cerr << "Done" << std::endl;
|
|
||||||
REQUIRE(statusCode == 200);
|
|
||||||
}
|
|
||||||
|
|
||||||
SECTION("Async API, multiple calls")
|
|
||||||
{
|
|
||||||
bool async = true;
|
|
||||||
HttpClient httpClient(async);
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
|
|
||||||
std::string url("http://httpbin.org/");
|
|
||||||
auto args = httpClient.createRequest(url);
|
|
||||||
|
|
||||||
args->extraHeaders = headers;
|
|
||||||
args->connectTimeout = 60;
|
|
||||||
args->transferTimeout = 60;
|
|
||||||
args->followRedirects = true;
|
|
||||||
args->maxRedirects = 10;
|
|
||||||
args->verbose = true;
|
|
||||||
args->compress = true;
|
|
||||||
args->logger = [](const std::string& msg)
|
|
||||||
{
|
|
||||||
std::cout << msg;
|
|
||||||
};
|
|
||||||
args->onProgressCallback = [](int current, int total) -> bool
|
|
||||||
{
|
|
||||||
std::cerr << "\r" << "Downloaded "
|
|
||||||
<< current << " bytes out of " << total;
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
std::atomic<bool> requestCompleted(false);
|
|
||||||
std::atomic<int> statusCode0(0);
|
|
||||||
std::atomic<int> statusCode1(0);
|
|
||||||
std::atomic<int> statusCode2(0);
|
|
||||||
|
|
||||||
for (int i = 0; i < 3; ++i)
|
|
||||||
{
|
|
||||||
httpClient.performRequest(args, [i, &requestCompleted, &statusCode0, &statusCode1, &statusCode2]
|
|
||||||
(const HttpResponsePtr& response)
|
|
||||||
{
|
|
||||||
std::cerr << "Upload size: " << response->uploadSize << std::endl;
|
|
||||||
std::cerr << "Download size: " << response->downloadSize << std::endl;
|
|
||||||
std::cerr << "Status: " << response->statusCode << std::endl;
|
|
||||||
std::cerr << "Error message: " << response->errorMsg << std::endl;
|
|
||||||
|
|
||||||
// In case of failure, print response->errorMsg
|
|
||||||
if (i == 0)
|
|
||||||
{
|
|
||||||
statusCode0 = response->statusCode;
|
|
||||||
}
|
|
||||||
else if (i == 1)
|
|
||||||
{
|
|
||||||
statusCode1 = response->statusCode;
|
|
||||||
}
|
|
||||||
else if (i == 2)
|
|
||||||
{
|
|
||||||
statusCode2 = response->statusCode;
|
|
||||||
requestCompleted = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
int wait = 0;
|
|
||||||
while (wait < 10000)
|
|
||||||
{
|
|
||||||
if (requestCompleted) break;
|
|
||||||
|
|
||||||
std::chrono::duration<double, std::milli> duration(10);
|
|
||||||
std::this_thread::sleep_for(duration);
|
|
||||||
wait += 10;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::cerr << "Done" << std::endl;
|
|
||||||
REQUIRE(statusCode0 == 200);
|
|
||||||
REQUIRE(statusCode1 == 200);
|
|
||||||
REQUIRE(statusCode2 == 200);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,70 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXSocketTest.cpp
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <iostream>
|
|
||||||
#include <ixwebsocket/IXHttpClient.h>
|
|
||||||
#include <ixwebsocket/IXHttpServer.h>
|
|
||||||
#include "IXGetFreePort.h"
|
|
||||||
|
|
||||||
#include "catch.hpp"
|
|
||||||
|
|
||||||
using namespace ix;
|
|
||||||
|
|
||||||
TEST_CASE("http server", "[httpd]")
|
|
||||||
{
|
|
||||||
SECTION("Connect to a local HTTP server")
|
|
||||||
{
|
|
||||||
int port = getFreePort();
|
|
||||||
ix::HttpServer server(port, "127.0.0.1");
|
|
||||||
|
|
||||||
auto res = server.listen();
|
|
||||||
REQUIRE(res.first);
|
|
||||||
server.start();
|
|
||||||
|
|
||||||
HttpClient httpClient;
|
|
||||||
WebSocketHttpHeaders headers;
|
|
||||||
|
|
||||||
std::string url("http://127.0.0.1:");
|
|
||||||
url += std::to_string(port);
|
|
||||||
url += "/data/foo.txt";
|
|
||||||
auto args = httpClient.createRequest(url);
|
|
||||||
|
|
||||||
args->extraHeaders = headers;
|
|
||||||
args->connectTimeout = 60;
|
|
||||||
args->transferTimeout = 60;
|
|
||||||
args->followRedirects = true;
|
|
||||||
args->maxRedirects = 10;
|
|
||||||
args->verbose = true;
|
|
||||||
args->compress = true;
|
|
||||||
args->logger = [](const std::string& msg)
|
|
||||||
{
|
|
||||||
std::cout << msg;
|
|
||||||
};
|
|
||||||
args->onProgressCallback = [](int current, int total) -> bool
|
|
||||||
{
|
|
||||||
std::cerr << "\r" << "Downloaded "
|
|
||||||
<< current << " bytes out of " << total;
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
auto response = httpClient.get(url, args);
|
|
||||||
|
|
||||||
for (auto it : response->headers)
|
|
||||||
{
|
|
||||||
std::cerr << it.first << ": " << it.second << std::endl;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::cerr << "Upload size: " << response->uploadSize << std::endl;
|
|
||||||
std::cerr << "Download size: " << response->downloadSize << std::endl;
|
|
||||||
std::cerr << "Status: " << response->statusCode << std::endl;
|
|
||||||
std::cerr << "Error message: " << response->errorMsg << std::endl;
|
|
||||||
|
|
||||||
REQUIRE(response->errorCode == HttpErrorCode::Ok);
|
|
||||||
REQUIRE(response->statusCode == 200);
|
|
||||||
|
|
||||||
server.stop();
|
|
||||||
}
|
|
||||||
}
|
|
@ -77,7 +77,7 @@ TEST_CASE("socket", "[socket]")
|
|||||||
testSocket(host, port, request, socket, expectedStatus, timeoutSecs);
|
testSocket(host, port, request, socket, expectedStatus, timeoutSecs);
|
||||||
}
|
}
|
||||||
|
|
||||||
#if defined(IXWEBSOCKET_USE_TLS)
|
#if defined(__APPLE__) || defined(__linux__)
|
||||||
SECTION("Connect to google HTTPS server over port 443. Send GET request without header. Should return 200")
|
SECTION("Connect to google HTTPS server over port 443. Send GET request without header. Should return 200")
|
||||||
{
|
{
|
||||||
std::string errMsg;
|
std::string errMsg;
|
||||||
|
101
test/IXTest.cpp
101
test/IXTest.cpp
@ -72,6 +72,88 @@ namespace ix
|
|||||||
Logger() << msg;
|
Logger() << msg;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int getAnyFreePortRandom()
|
||||||
|
{
|
||||||
|
std::random_device rd;
|
||||||
|
std::uniform_int_distribution<int> dist(1024 + 1, 65535);
|
||||||
|
|
||||||
|
return dist(rd);
|
||||||
|
}
|
||||||
|
|
||||||
|
int getAnyFreePort()
|
||||||
|
{
|
||||||
|
int defaultPort = 8090;
|
||||||
|
int sockfd;
|
||||||
|
if ((sockfd = socket(AF_INET, SOCK_STREAM, 0)) < 0)
|
||||||
|
{
|
||||||
|
log("Cannot compute a free port. socket error.");
|
||||||
|
return getAnyFreePortRandom();
|
||||||
|
}
|
||||||
|
|
||||||
|
int enable = 1;
|
||||||
|
if (setsockopt(sockfd, SOL_SOCKET, SO_REUSEADDR,
|
||||||
|
(char*) &enable, sizeof(enable)) < 0)
|
||||||
|
{
|
||||||
|
log("Cannot compute a free port. setsockopt error.");
|
||||||
|
return getAnyFreePortRandom();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Bind to port 0. This is the standard way to get a free port.
|
||||||
|
struct sockaddr_in server; // server address information
|
||||||
|
server.sin_family = AF_INET;
|
||||||
|
server.sin_port = htons(0);
|
||||||
|
server.sin_addr.s_addr = inet_addr("127.0.0.1");
|
||||||
|
|
||||||
|
if (bind(sockfd, (struct sockaddr *)&server, sizeof(server)) < 0)
|
||||||
|
{
|
||||||
|
log("Cannot compute a free port. bind error.");
|
||||||
|
|
||||||
|
Socket::closeSocket(sockfd);
|
||||||
|
return getAnyFreePortRandom();
|
||||||
|
}
|
||||||
|
|
||||||
|
struct sockaddr_in sa; // server address information
|
||||||
|
socklen_t len = sizeof(sa);
|
||||||
|
if (getsockname(sockfd, (struct sockaddr *) &sa, &len) < 0)
|
||||||
|
{
|
||||||
|
log("Cannot compute a free port. getsockname error.");
|
||||||
|
|
||||||
|
Socket::closeSocket(sockfd);
|
||||||
|
return getAnyFreePortRandom();
|
||||||
|
}
|
||||||
|
|
||||||
|
int port = ntohs(sa.sin_port);
|
||||||
|
Socket::closeSocket(sockfd);
|
||||||
|
|
||||||
|
return port;
|
||||||
|
}
|
||||||
|
|
||||||
|
int getFreePort()
|
||||||
|
{
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
#if defined(__has_feature)
|
||||||
|
# if __has_feature(address_sanitizer)
|
||||||
|
int port = getAnyFreePortRandom();
|
||||||
|
# else
|
||||||
|
int port = getAnyFreePort();
|
||||||
|
# endif
|
||||||
|
#else
|
||||||
|
int port = getAnyFreePort();
|
||||||
|
#endif
|
||||||
|
//
|
||||||
|
// Only port above 1024 can be used by non root users, but for some
|
||||||
|
// reason I got port 7 returned with macOS when binding on port 0...
|
||||||
|
//
|
||||||
|
if (port > 1024)
|
||||||
|
{
|
||||||
|
return port;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
void hexDump(const std::string& prefix,
|
void hexDump(const std::string& prefix,
|
||||||
const std::string& s)
|
const std::string& s)
|
||||||
{
|
{
|
||||||
@ -96,29 +178,34 @@ namespace ix
|
|||||||
std::shared_ptr<ConnectionState> connectionState)
|
std::shared_ptr<ConnectionState> connectionState)
|
||||||
{
|
{
|
||||||
webSocket->setOnMessageCallback(
|
webSocket->setOnMessageCallback(
|
||||||
[webSocket, connectionState, &server](const ix::WebSocketMessagePtr& msg)
|
[webSocket, connectionState, &server](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
Logger() << "New connection";
|
Logger() << "New connection";
|
||||||
Logger() << "Uri: " << msg->openInfo.uri;
|
Logger() << "Uri: " << openInfo.uri;
|
||||||
Logger() << "Headers:";
|
Logger() << "Headers:";
|
||||||
for (auto it : msg->openInfo.headers)
|
for (auto it : openInfo.headers)
|
||||||
{
|
{
|
||||||
Logger() << it.first << ": " << it.second;
|
Logger() << it.first << ": " << it.second;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
Logger() << "Closed connection";
|
Logger() << "Closed connection";
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
for (auto&& client : server.getClients())
|
for (auto&& client : server.getClients())
|
||||||
{
|
{
|
||||||
if (client != webSocket)
|
if (client != webSocket)
|
||||||
{
|
{
|
||||||
client->send(msg->str, msg->binary);
|
client->send(str);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,14 +6,13 @@
|
|||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include "IXGetFreePort.h"
|
|
||||||
#include <iostream>
|
|
||||||
#include <ixwebsocket/IXWebSocketServer.h>
|
|
||||||
#include <mutex>
|
|
||||||
#include <spdlog/spdlog.h>
|
|
||||||
#include <sstream>
|
|
||||||
#include <string>
|
#include <string>
|
||||||
#include <vector>
|
#include <vector>
|
||||||
|
#include <sstream>
|
||||||
|
#include <iostream>
|
||||||
|
#include <mutex>
|
||||||
|
#include <spdlog/spdlog.h>
|
||||||
|
#include <ixwebsocket/IXWebSocketServer.h>
|
||||||
|
|
||||||
namespace ix
|
namespace ix
|
||||||
{
|
{
|
||||||
@ -29,23 +28,25 @@ namespace ix
|
|||||||
|
|
||||||
struct Logger
|
struct Logger
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
template<typename T>
|
template <typename T>
|
||||||
Logger& operator<<(T const& obj)
|
Logger& operator<<(T const& obj)
|
||||||
{
|
{
|
||||||
std::lock_guard<std::mutex> lock(_mutex);
|
std::lock_guard<std::mutex> lock(_mutex);
|
||||||
|
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << obj;
|
ss << obj;
|
||||||
spdlog::info(ss.str());
|
spdlog::info(ss.str());
|
||||||
return *this;
|
return *this;
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
static std::mutex _mutex;
|
static std::mutex _mutex;
|
||||||
};
|
};
|
||||||
|
|
||||||
void log(const std::string& msg);
|
void log(const std::string& msg);
|
||||||
|
|
||||||
|
int getFreePort();
|
||||||
|
|
||||||
bool startWebSocketEchoServer(ix::WebSocketServer& server);
|
bool startWebSocketEchoServer(ix::WebSocketServer& server);
|
||||||
} // namespace ix
|
}
|
||||||
|
@ -1,52 +0,0 @@
|
|||||||
/*
|
|
||||||
* IXUnityBuildsTest.cpp
|
|
||||||
* Author: Benjamin Sergeant
|
|
||||||
* Copyright (c) 2019 Machine Zone. All rights reserved.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include <ixwebsocket/IXCancellationRequest.h>
|
|
||||||
#include <ixwebsocket/IXConnectionState.h>
|
|
||||||
#include <ixwebsocket/IXDNSLookup.h>
|
|
||||||
#include <ixwebsocket/IXHttp.h>
|
|
||||||
#include <ixwebsocket/IXHttpClient.h>
|
|
||||||
#include <ixwebsocket/IXHttpServer.h>
|
|
||||||
#include <ixwebsocket/IXNetSystem.h>
|
|
||||||
#include <ixwebsocket/IXProgressCallback.h>
|
|
||||||
#include <ixwebsocket/IXSelectInterrupt.h>
|
|
||||||
#include <ixwebsocket/IXSelectInterruptFactory.h>
|
|
||||||
#include <ixwebsocket/IXSetThreadName.h>
|
|
||||||
#include <ixwebsocket/IXSocket.h>
|
|
||||||
#include <ixwebsocket/IXSocketConnect.h>
|
|
||||||
#include <ixwebsocket/IXSocketFactory.h>
|
|
||||||
#include <ixwebsocket/IXSocketServer.h>
|
|
||||||
#include <ixwebsocket/IXUrlParser.h>
|
|
||||||
#include <ixwebsocket/IXWebSocket.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketCloseConstants.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketCloseInfo.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketErrorInfo.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketHandshake.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketHttpHeaders.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketMessage.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketMessageQueue.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketMessageType.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketOpenInfo.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketPerMessageDeflate.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketPerMessageDeflateCodec.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketPerMessageDeflateOptions.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketSendInfo.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketServer.h>
|
|
||||||
#include <ixwebsocket/IXWebSocketTransport.h>
|
|
||||||
#include <ixwebsocket/LUrlParser.h>
|
|
||||||
#include <ixwebsocket/libwshandshake.hpp>
|
|
||||||
|
|
||||||
#include "catch.hpp"
|
|
||||||
|
|
||||||
using namespace ix;
|
|
||||||
|
|
||||||
TEST_CASE("unity build", "[unity_build]")
|
|
||||||
{
|
|
||||||
SECTION("dummy test")
|
|
||||||
{
|
|
||||||
REQUIRE(true);
|
|
||||||
}
|
|
||||||
}
|
|
@ -34,8 +34,6 @@ namespace
|
|||||||
const std::string& getCloseReason();
|
const std::string& getCloseReason();
|
||||||
bool getCloseRemote();
|
bool getCloseRemote();
|
||||||
|
|
||||||
bool hasConnectionError() const;
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
ix::WebSocket _webSocket;
|
ix::WebSocket _webSocket;
|
||||||
int _port;
|
int _port;
|
||||||
@ -44,7 +42,6 @@ namespace
|
|||||||
uint16_t _closeCode;
|
uint16_t _closeCode;
|
||||||
std::string _closeReason;
|
std::string _closeReason;
|
||||||
bool _closeRemote;
|
bool _closeRemote;
|
||||||
std::atomic<bool> _connectionError;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
WebSocketClient::WebSocketClient(int port)
|
WebSocketClient::WebSocketClient(int port)
|
||||||
@ -52,16 +49,10 @@ namespace
|
|||||||
, _closeCode(0)
|
, _closeCode(0)
|
||||||
, _closeReason(std::string(""))
|
, _closeReason(std::string(""))
|
||||||
, _closeRemote(false)
|
, _closeRemote(false)
|
||||||
, _connectionError(false)
|
|
||||||
{
|
{
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool WebSocketClient::hasConnectionError() const
|
|
||||||
{
|
|
||||||
return _connectionError;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool WebSocketClient::isReady() const
|
bool WebSocketClient::isReady() const
|
||||||
{
|
{
|
||||||
return _webSocket.getReadyState() == ix::ReadyState::Open;
|
return _webSocket.getReadyState() == ix::ReadyState::Open;
|
||||||
@ -117,48 +108,52 @@ namespace
|
|||||||
log(std::string("Connecting to url: ") + url);
|
log(std::string("Connecting to url: ") + url);
|
||||||
|
|
||||||
_webSocket.setOnMessageCallback(
|
_webSocket.setOnMessageCallback(
|
||||||
[this](const ix::WebSocketMessagePtr& msg)
|
[this](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
log("client connected");
|
log("client connected");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "client disconnected("
|
ss << "client disconnected("
|
||||||
<< msg->closeInfo.code
|
<< closeInfo.code
|
||||||
<< ","
|
<< ","
|
||||||
<< msg->closeInfo.reason
|
<< closeInfo.reason
|
||||||
<< ")";
|
<< ")";
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
|
|
||||||
std::lock_guard<std::mutex> lck(_mutexCloseData);
|
std::lock_guard<std::mutex> lck(_mutexCloseData);
|
||||||
|
|
||||||
_closeCode = msg->closeInfo.code;
|
_closeCode = closeInfo.code;
|
||||||
_closeReason = std::string(msg->closeInfo.reason);
|
_closeReason = std::string(closeInfo.reason);
|
||||||
_closeRemote = msg->closeInfo.remote;
|
_closeRemote = closeInfo.remote;
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Error)
|
else if (messageType == ix::WebSocketMessageType::Error)
|
||||||
{
|
{
|
||||||
_connectionError = true;
|
ss << "Error ! " << error.reason;
|
||||||
ss << "Error ! " << msg->errorInfo.reason;
|
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Pong)
|
else if (messageType == ix::WebSocketMessageType::Pong)
|
||||||
{
|
{
|
||||||
ss << "Received pong message " << msg->str;
|
ss << "Received pong message " << str;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Ping)
|
else if (messageType == ix::WebSocketMessageType::Ping)
|
||||||
{
|
{
|
||||||
ss << "Received ping message " << msg->str;
|
ss << "Received ping message " << str;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
ss << "Received message " << msg->str;
|
ss << "Received message " << str;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -188,34 +183,39 @@ namespace
|
|||||||
std::shared_ptr<ConnectionState> connectionState)
|
std::shared_ptr<ConnectionState> connectionState)
|
||||||
{
|
{
|
||||||
webSocket->setOnMessageCallback(
|
webSocket->setOnMessageCallback(
|
||||||
[webSocket, connectionState, &server, &receivedCloseCode, &receivedCloseReason, &receivedCloseRemote, &mutexWrite](const ix::WebSocketMessagePtr& msg)
|
[webSocket, connectionState, &server, &receivedCloseCode, &receivedCloseReason, &receivedCloseRemote, &mutexWrite](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
Logger() << "New server connection";
|
Logger() << "New server connection";
|
||||||
Logger() << "id: " << connectionState->getId();
|
Logger() << "id: " << connectionState->getId();
|
||||||
Logger() << "Uri: " << msg->openInfo.uri;
|
Logger() << "Uri: " << openInfo.uri;
|
||||||
Logger() << "Headers:";
|
Logger() << "Headers:";
|
||||||
for (auto it : msg->openInfo.headers)
|
for (auto it : openInfo.headers)
|
||||||
{
|
{
|
||||||
Logger() << it.first << ": " << it.second;
|
Logger() << it.first << ": " << it.second;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
ss << "Server closed connection("
|
ss << "Server closed connection("
|
||||||
<< msg->closeInfo.code
|
<< closeInfo.code
|
||||||
<< ","
|
<< ","
|
||||||
<< msg->closeInfo.reason
|
<< closeInfo.reason
|
||||||
<< ")";
|
<< ")";
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
|
|
||||||
std::lock_guard<std::mutex> lck(mutexWrite);
|
std::lock_guard<std::mutex> lck(mutexWrite);
|
||||||
|
|
||||||
receivedCloseCode = msg->closeInfo.code;
|
receivedCloseCode = closeInfo.code;
|
||||||
receivedCloseReason = std::string(msg->closeInfo.reason);
|
receivedCloseReason = std::string(closeInfo.reason);
|
||||||
receivedCloseRemote = msg->closeInfo.remote;
|
receivedCloseRemote = closeInfo.remote;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@ -258,7 +258,6 @@ TEST_CASE("Websocket_client_close_default", "[close]")
|
|||||||
// Wait for all chat instance to be ready
|
// Wait for all chat instance to be ready
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
REQUIRE(!webSocketClient.hasConnectionError());
|
|
||||||
if (webSocketClient.isReady()) break;
|
if (webSocketClient.isReady()) break;
|
||||||
ix::msleep(10);
|
ix::msleep(10);
|
||||||
}
|
}
|
||||||
|
@ -20,34 +20,39 @@ namespace
|
|||||||
{
|
{
|
||||||
server.setOnConnectionCallback(
|
server.setOnConnectionCallback(
|
||||||
[&server](std::shared_ptr<ix::WebSocket> webSocket,
|
[&server](std::shared_ptr<ix::WebSocket> webSocket,
|
||||||
std::shared_ptr<ConnectionState> connectionState)
|
std::shared_ptr<ConnectionState> connectionState)
|
||||||
{
|
{
|
||||||
webSocket->setOnMessageCallback(
|
webSocket->setOnMessageCallback(
|
||||||
[connectionState, &server](const WebSocketMessagePtr& msg)
|
[connectionState, &server](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string & str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo & error,
|
||||||
|
const ix::WebSocketOpenInfo & openInfo,
|
||||||
|
const ix::WebSocketCloseInfo & closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
Logger() << "New connection";
|
Logger() << "New connection";
|
||||||
connectionState->computeId();
|
connectionState->computeId();
|
||||||
Logger() << "id: " << connectionState->getId();
|
Logger() << "id: " << connectionState->getId();
|
||||||
Logger() << "Uri: " << msg->openInfo.uri;
|
Logger() << "Uri: " << openInfo.uri;
|
||||||
Logger() << "Headers:";
|
Logger() << "Headers:";
|
||||||
for (auto&& it : msg->openInfo.headers)
|
for (auto it : openInfo.headers)
|
||||||
{
|
{
|
||||||
Logger() << it.first << ": " << it.second;
|
Logger() << it.first << ": " << it.second;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
Logger() << "Closed connection";
|
Logger() << "Closed connection";
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
Logger() << "Message received: " << msg->str;
|
Logger() << "Message received: " << str;
|
||||||
|
|
||||||
for (auto&& client : server.getClients())
|
for (auto&& client : server.getClients())
|
||||||
{
|
{
|
||||||
client->send(msg->str);
|
client->send(str);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -73,41 +78,46 @@ namespace
|
|||||||
{
|
{
|
||||||
msgQ.bindWebsocket(&ws);
|
msgQ.bindWebsocket(&ws);
|
||||||
|
|
||||||
msgQ.setOnMessageCallback([this](const WebSocketMessagePtr& msg)
|
msgQ.setOnMessageCallback([this](WebSocketMessageType messageType,
|
||||||
|
const std::string & str,
|
||||||
|
size_t wireSize,
|
||||||
|
const WebSocketErrorInfo & error,
|
||||||
|
const WebSocketOpenInfo & openInfo,
|
||||||
|
const WebSocketCloseInfo & closeInfo)
|
||||||
{
|
{
|
||||||
REQUIRE(mainThreadId == std::this_thread::get_id());
|
REQUIRE(mainThreadId == std::this_thread::get_id());
|
||||||
|
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
if (msg->type == WebSocketMessageType::Open)
|
if (messageType == WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
log("client connected");
|
log("client connected");
|
||||||
sendNextMessage();
|
sendNextMessage();
|
||||||
}
|
}
|
||||||
else if (msg->type == WebSocketMessageType::Close)
|
else if (messageType == WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
log("client disconnected");
|
log("client disconnected");
|
||||||
}
|
}
|
||||||
else if (msg->type == WebSocketMessageType::Error)
|
else if (messageType == WebSocketMessageType::Error)
|
||||||
{
|
{
|
||||||
ss << "Error ! " << msg->errorInfo.reason;
|
ss << "Error ! " << error.reason;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
testDone = true;
|
testDone = true;
|
||||||
}
|
}
|
||||||
else if (msg->type == WebSocketMessageType::Pong)
|
else if (messageType == WebSocketMessageType::Pong)
|
||||||
{
|
{
|
||||||
ss << "Received pong message " << msg->str;
|
ss << "Received pong message " << str;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == WebSocketMessageType::Ping)
|
else if (messageType == WebSocketMessageType::Ping)
|
||||||
{
|
{
|
||||||
ss << "Received ping message " << msg->str;
|
ss << "Received ping message " << str;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == WebSocketMessageType::Message)
|
else if (messageType == WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
REQUIRE(msg->str.compare("Hey dude!") == 0);
|
REQUIRE(str.compare("Hey dude!") == 0);
|
||||||
++receivedCount;
|
++receivedCount;
|
||||||
ss << "Received message " << msg->str;
|
ss << "Received message " << str;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
sendNextMessage();
|
sendNextMessage();
|
||||||
}
|
}
|
||||||
@ -179,4 +189,5 @@ TEST_CASE("Websocket_message_queue", "[websocket_message_q]")
|
|||||||
|
|
||||||
server.stop();
|
server.stop();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -427,7 +427,6 @@ TEST_CASE("Websocket_ping_no_data_sent_setHeartBeatPeriod", "[setHeartBeatPeriod
|
|||||||
REQUIRE(server.getClients().size() == 0);
|
REQUIRE(server.getClients().size() == 0);
|
||||||
|
|
||||||
ix::reportWebSocketTraffic();
|
ix::reportWebSocketTraffic();
|
||||||
server.stop();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -478,6 +477,5 @@ TEST_CASE("Websocket_ping_data_sent_setHeartBeatPeriod", "[setHeartBeatPeriod]")
|
|||||||
REQUIRE(server.getClients().size() == 0);
|
REQUIRE(server.getClients().size() == 0);
|
||||||
|
|
||||||
ix::reportWebSocketTraffic();
|
ix::reportWebSocketTraffic();
|
||||||
server.stop();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -106,7 +106,7 @@ namespace
|
|||||||
{
|
{
|
||||||
log("client disconnected");
|
log("client disconnected");
|
||||||
|
|
||||||
if (msg->closeInfo.code == 1011)
|
if (closeInfo.code == 1011)
|
||||||
{
|
{
|
||||||
_closedDueToPingTimeout = true;
|
_closedDueToPingTimeout = true;
|
||||||
}
|
}
|
||||||
|
@ -39,37 +39,42 @@ namespace ix
|
|||||||
|
|
||||||
server.setOnConnectionCallback(
|
server.setOnConnectionCallback(
|
||||||
[&server, &connectionId](std::shared_ptr<ix::WebSocket> webSocket,
|
[&server, &connectionId](std::shared_ptr<ix::WebSocket> webSocket,
|
||||||
std::shared_ptr<ConnectionState> connectionState)
|
std::shared_ptr<ConnectionState> connectionState)
|
||||||
{
|
{
|
||||||
webSocket->setOnMessageCallback(
|
webSocket->setOnMessageCallback(
|
||||||
[webSocket, connectionState,
|
[webSocket, connectionState,
|
||||||
&connectionId, &server](const ix::WebSocketMessagePtr& msg)
|
&connectionId, &server](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
Logger() << "New connection";
|
Logger() << "New connection";
|
||||||
connectionState->computeId();
|
connectionState->computeId();
|
||||||
Logger() << "id: " << connectionState->getId();
|
Logger() << "id: " << connectionState->getId();
|
||||||
Logger() << "Uri: " << msg->openInfo.uri;
|
Logger() << "Uri: " << openInfo.uri;
|
||||||
Logger() << "Headers:";
|
Logger() << "Headers:";
|
||||||
for (auto it : msg->openInfo.headers)
|
for (auto it : openInfo.headers)
|
||||||
{
|
{
|
||||||
Logger() << it.first << ": " << it.second;
|
Logger() << it.first << ": " << it.second;
|
||||||
}
|
}
|
||||||
|
|
||||||
connectionId = connectionState->getId();
|
connectionId = connectionState->getId();
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
Logger() << "Closed connection";
|
Logger() << "Closed connection";
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
for (auto&& client : server.getClients())
|
for (auto&& client : server.getClients())
|
||||||
{
|
{
|
||||||
if (client != webSocket)
|
if (client != webSocket)
|
||||||
{
|
{
|
||||||
client->send(msg->str);
|
client->send(str);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -52,36 +52,41 @@ namespace
|
|||||||
log(std::string("Connecting to url: ") + url);
|
log(std::string("Connecting to url: ") + url);
|
||||||
|
|
||||||
_webSocket.setOnMessageCallback(
|
_webSocket.setOnMessageCallback(
|
||||||
[](const ix::WebSocketMessagePtr& msg)
|
[](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
log("TestConnectionDisconnection: connected !");
|
log("TestConnectionDisconnection: connected !");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
log("TestConnectionDisconnection: disconnected !");
|
log("TestConnectionDisconnection: disconnected !");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Error)
|
else if (messageType == ix::WebSocketMessageType::Error)
|
||||||
{
|
{
|
||||||
ss << "TestConnectionDisconnection: Error! ";
|
ss << "TestConnectionDisconnection: Error! ";
|
||||||
ss << msg->errorInfo.reason;
|
ss << error.reason;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
log("TestConnectionDisconnection: received message.!");
|
log("TestConnectionDisconnection: received message.!");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Ping)
|
else if (messageType == ix::WebSocketMessageType::Ping)
|
||||||
{
|
{
|
||||||
log("TestConnectionDisconnection: received ping message.!");
|
log("TestConnectionDisconnection: received ping message.!");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Pong)
|
else if (messageType == ix::WebSocketMessageType::Pong)
|
||||||
{
|
{
|
||||||
log("TestConnectionDisconnection: received pong message.!");
|
log("TestConnectionDisconnection: received pong message.!");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Fragment)
|
else if (messageType == ix::WebSocketMessageType::Fragment)
|
||||||
{
|
{
|
||||||
log("TestConnectionDisconnection: received fragment.!");
|
log("TestConnectionDisconnection: received fragment.!");
|
||||||
}
|
}
|
||||||
|
@ -114,26 +114,31 @@ namespace
|
|||||||
log(std::string("Connecting to url: ") + url);
|
log(std::string("Connecting to url: ") + url);
|
||||||
|
|
||||||
_webSocket.setOnMessageCallback(
|
_webSocket.setOnMessageCallback(
|
||||||
[this](const ix::WebSocketMessagePtr& msg)
|
[this](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
std::stringstream ss;
|
std::stringstream ss;
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
ss << "cmd_websocket_chat: user "
|
ss << "cmd_websocket_chat: user "
|
||||||
<< _user
|
<< _user
|
||||||
<< " Connected !";
|
<< " Connected !";
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
ss << "cmd_websocket_chat: user "
|
ss << "cmd_websocket_chat: user "
|
||||||
<< _user
|
<< _user
|
||||||
<< " disconnected !";
|
<< " disconnected !";
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
auto result = decodeMessage(msg->str);
|
auto result = decodeMessage(str);
|
||||||
|
|
||||||
// Our "chat" / "broacast" node.js server does not send us
|
// Our "chat" / "broacast" node.js server does not send us
|
||||||
// the messages we send, so we don't need to have a msg_user != user
|
// the messages we send, so we don't need to have a msg_user != user
|
||||||
@ -154,20 +159,20 @@ namespace
|
|||||||
<< _user << " > ";
|
<< _user << " > ";
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Error)
|
else if (messageType == ix::WebSocketMessageType::Error)
|
||||||
{
|
{
|
||||||
ss << "cmd_websocket_chat: Error ! " << msg->errorInfo.reason;
|
ss << "cmd_websocket_chat: Error ! " << error.reason;
|
||||||
log(ss.str());
|
log(ss.str());
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Ping)
|
else if (messageType == ix::WebSocketMessageType::Ping)
|
||||||
{
|
{
|
||||||
log("cmd_websocket_chat: received ping message");
|
log("cmd_websocket_chat: received ping message");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Pong)
|
else if (messageType == ix::WebSocketMessageType::Pong)
|
||||||
{
|
{
|
||||||
log("cmd_websocket_chat: received pong message");
|
log("cmd_websocket_chat: received pong message");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Fragment)
|
else if (messageType == ix::WebSocketMessageType::Fragment)
|
||||||
{
|
{
|
||||||
log("cmd_websocket_chat: received message fragment");
|
log("cmd_websocket_chat: received message fragment");
|
||||||
}
|
}
|
||||||
@ -216,30 +221,35 @@ namespace
|
|||||||
std::shared_ptr<ConnectionState> connectionState)
|
std::shared_ptr<ConnectionState> connectionState)
|
||||||
{
|
{
|
||||||
webSocket->setOnMessageCallback(
|
webSocket->setOnMessageCallback(
|
||||||
[webSocket, connectionState, &server](const ix::WebSocketMessagePtr& msg)
|
[webSocket, connectionState, &server](ix::WebSocketMessageType messageType,
|
||||||
|
const std::string& str,
|
||||||
|
size_t wireSize,
|
||||||
|
const ix::WebSocketErrorInfo& error,
|
||||||
|
const ix::WebSocketOpenInfo& openInfo,
|
||||||
|
const ix::WebSocketCloseInfo& closeInfo)
|
||||||
{
|
{
|
||||||
if (msg->type == ix::WebSocketMessageType::Open)
|
if (messageType == ix::WebSocketMessageType::Open)
|
||||||
{
|
{
|
||||||
Logger() << "New connection";
|
Logger() << "New connection";
|
||||||
Logger() << "id: " << connectionState->getId();
|
Logger() << "id: " << connectionState->getId();
|
||||||
Logger() << "Uri: " << msg->openInfo.uri;
|
Logger() << "Uri: " << openInfo.uri;
|
||||||
Logger() << "Headers:";
|
Logger() << "Headers:";
|
||||||
for (auto it : msg->openInfo.headers)
|
for (auto it : openInfo.headers)
|
||||||
{
|
{
|
||||||
Logger() << it.first << ": " << it.second;
|
Logger() << it.first << ": " << it.second;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Close)
|
else if (messageType == ix::WebSocketMessageType::Close)
|
||||||
{
|
{
|
||||||
log("Closed connection");
|
log("Closed connection");
|
||||||
}
|
}
|
||||||
else if (msg->type == ix::WebSocketMessageType::Message)
|
else if (messageType == ix::WebSocketMessageType::Message)
|
||||||
{
|
{
|
||||||
for (auto&& client : server.getClients())
|
for (auto&& client : server.getClients())
|
||||||
{
|
{
|
||||||
if (client != webSocket)
|
if (client != webSocket)
|
||||||
{
|
{
|
||||||
client->send(msg->str);
|
client->send(str);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,30 +0,0 @@
|
|||||||
# Clients
|
|
||||||
|
|
||||||
## ws
|
|
||||||
|
|
||||||
```
|
|
||||||
$ ws connect ws://127.0.0.1:8765
|
|
||||||
Type Ctrl-D to exit prompt...
|
|
||||||
Connecting to url: ws://127.0.0.1:8765
|
|
||||||
> ws_connect: connected
|
|
||||||
Uri: /
|
|
||||||
Handshake Headers:
|
|
||||||
Connection: Upgrade
|
|
||||||
Date: Sat, 08 Jun 2019 16:43:29 GMT
|
|
||||||
Sec-WebSocket-Accept: kPCNwGa97y+7NWdAvHi/7/rA8AE=
|
|
||||||
Sec-WebSocket-Extensions: permessage-deflate; server_max_window_bits=15; client_max_window_bits=15
|
|
||||||
Server: Python/3.7 websockets/7.0
|
|
||||||
Upgrade: websocket
|
|
||||||
Received 13 bytes
|
|
||||||
ws_connect: received message: > Welcome !
|
|
||||||
ws_connect: connection closed: code 1006 reason Abnormal closure
|
|
||||||
```
|
|
||||||
|
|
||||||
## wscat
|
|
||||||
|
|
||||||
```
|
|
||||||
$ ./node_modules/.bin/wscat -c ws://127.0.0.1:8765
|
|
||||||
connected (press CTRL+C to quit)
|
|
||||||
< > Welcome !
|
|
||||||
disconnected (code: 1006)
|
|
||||||
```
|
|
@ -1,22 +0,0 @@
|
|||||||
#!/usr/bin/env python
|
|
||||||
|
|
||||||
# WS server example
|
|
||||||
|
|
||||||
import asyncio
|
|
||||||
import websockets
|
|
||||||
|
|
||||||
async def hello(websocket, path):
|
|
||||||
await websocket.send(f"> Welcome !")
|
|
||||||
|
|
||||||
name = await websocket.recv()
|
|
||||||
print(f"< {name}")
|
|
||||||
|
|
||||||
greeting = f"Hello {name}!"
|
|
||||||
|
|
||||||
await websocket.send(greeting)
|
|
||||||
print(f"> {greeting}")
|
|
||||||
|
|
||||||
start_server = websockets.serve(hello, 'localhost', 8765)
|
|
||||||
|
|
||||||
asyncio.get_event_loop().run_until_complete(start_server)
|
|
||||||
asyncio.get_event_loop().run_forever()
|
|
@ -1 +0,0 @@
|
|||||||
Hello world
|
|
@ -99,10 +99,8 @@ def runCMake(sanitizer, buildDir):
|
|||||||
#generator = '"NMake Makefiles"'
|
#generator = '"NMake Makefiles"'
|
||||||
#generator = '"Visual Studio 16 2019"'
|
#generator = '"Visual Studio 16 2019"'
|
||||||
generator = '"Visual Studio 15 2017"'
|
generator = '"Visual Studio 15 2017"'
|
||||||
USE_VENDORED_THIRD_PARTY = 'ON'
|
|
||||||
else:
|
else:
|
||||||
generator = '"Unix Makefiles"'
|
generator = '"Unix Makefiles"'
|
||||||
USE_VENDORED_THIRD_PARTY = 'ON'
|
|
||||||
|
|
||||||
CMAKE_BUILD_TYPE = BUILD_TYPE
|
CMAKE_BUILD_TYPE = BUILD_TYPE
|
||||||
|
|
||||||
@ -112,7 +110,6 @@ def runCMake(sanitizer, buildDir):
|
|||||||
-DCMAKE_BUILD_TYPE={CMAKE_BUILD_TYPE} \
|
-DCMAKE_BUILD_TYPE={CMAKE_BUILD_TYPE} \
|
||||||
-DUSE_TLS=1 \
|
-DUSE_TLS=1 \
|
||||||
-DCMAKE_EXPORT_COMPILE_COMMANDS=ON \
|
-DCMAKE_EXPORT_COMPILE_COMMANDS=ON \
|
||||||
-DUSE_VENDORED_THIRD_PARTY={USE_VENDORED_THIRD_PARTY} \
|
|
||||||
-G{generator}'
|
-G{generator}'
|
||||||
|
|
||||||
cmakeCmd = fmt.format(**locals())
|
cmakeCmd = fmt.format(**locals())
|
||||||
|
6
third_party/README.md
vendored
6
third_party/README.md
vendored
@ -1,7 +1,3 @@
|
|||||||
# Note
|
# Note
|
||||||
|
|
||||||
Except *zlib* and *mbedtls* on Windows, all dependencies here are for the ws command line tools, not for the IXWebSockets library which is standalone.
|
Except *zlib* on Windows, all dependencies here are for the ws command line tools, not for the IXWebSockets library which is standalone.
|
||||||
|
|
||||||
## MbedTLS
|
|
||||||
|
|
||||||
A small CMakeLists.txt fix had to be done so that the library can be included with "include_directory" in the top level CMakeLists.txt file. See https://github.com/ARMmbed/mbedtls/issues/2609
|
|
||||||
|
44
third_party/cpp-linenoise/.gitignore
vendored
44
third_party/cpp-linenoise/.gitignore
vendored
@ -1,44 +0,0 @@
|
|||||||
# Compiled Object files
|
|
||||||
*.slo
|
|
||||||
*.lo
|
|
||||||
*.o
|
|
||||||
*.obj
|
|
||||||
|
|
||||||
# Precompiled Headers
|
|
||||||
*.gch
|
|
||||||
*.pch
|
|
||||||
|
|
||||||
# Compiled Dynamic libraries
|
|
||||||
*.so
|
|
||||||
*.dylib
|
|
||||||
*.dll
|
|
||||||
|
|
||||||
# Fortran module files
|
|
||||||
*.mod
|
|
||||||
|
|
||||||
# Compiled Static libraries
|
|
||||||
*.lai
|
|
||||||
*.la
|
|
||||||
*.a
|
|
||||||
*.lib
|
|
||||||
|
|
||||||
# Executables
|
|
||||||
*.exe
|
|
||||||
*.out
|
|
||||||
*.app
|
|
||||||
|
|
||||||
# Others
|
|
||||||
*.dSYM
|
|
||||||
*.swp
|
|
||||||
Debug
|
|
||||||
Release
|
|
||||||
*.suo
|
|
||||||
*.sdf
|
|
||||||
*.user
|
|
||||||
xcuserdata
|
|
||||||
*.xcworkspace
|
|
||||||
Makefile
|
|
||||||
CMakeFiles
|
|
||||||
CMakeCache.txt
|
|
||||||
*.cmake
|
|
||||||
history.txt
|
|
22
third_party/cpp-linenoise/LICENSE
vendored
22
third_party/cpp-linenoise/LICENSE
vendored
@ -1,22 +0,0 @@
|
|||||||
Copyright (c) 2015 yhirose
|
|
||||||
All rights reserved.
|
|
||||||
|
|
||||||
Redistribution and use in source and binary forms, with or without
|
|
||||||
modification, are permitted provided that the following conditions are met:
|
|
||||||
|
|
||||||
1. Redistributions of source code must retain the above copyright notice, this
|
|
||||||
list of conditions and the following disclaimer.
|
|
||||||
2. Redistributions in binary form must reproduce the above copyright notice,
|
|
||||||
this list of conditions and the following disclaimer in the documentation
|
|
||||||
and/or other materials provided with the distribution.
|
|
||||||
|
|
||||||
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
|
|
||||||
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
|
|
||||||
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
|
||||||
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR
|
|
||||||
ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
||||||
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
|
||||||
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
|
|
||||||
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
||||||
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
|
|
||||||
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
95
third_party/cpp-linenoise/README.md
vendored
95
third_party/cpp-linenoise/README.md
vendored
@ -1,95 +0,0 @@
|
|||||||
cpp-linenoise
|
|
||||||
=============
|
|
||||||
|
|
||||||
Multi-platfrom (Unix, Windows) C++ header-only linenoise-based readline library.
|
|
||||||
|
|
||||||
This library gathered code from following excellent libraries, clean it up, and put it into a C++ header file for convenience.
|
|
||||||
|
|
||||||
* `linenoise.h` and `linenose.c` ([antirez/linenoise](https://github.com/antirez/linenoise))
|
|
||||||
* `ANSI.c` ([adoxa/ansicon](https://github.com/adoxa/ansicon))
|
|
||||||
* `Win32_ANSI.h` and `Win32_ANSI.c` ([MSOpenTech/redis](https://github.com/MSOpenTech/redis))
|
|
||||||
|
|
||||||
The licenses for the libraries are included in `linenoise.hpp`.
|
|
||||||
|
|
||||||
Usage
|
|
||||||
-----
|
|
||||||
|
|
||||||
```c++
|
|
||||||
#include "linenoise.hpp"
|
|
||||||
|
|
||||||
...
|
|
||||||
|
|
||||||
const auto path = "history.txt";
|
|
||||||
|
|
||||||
// Setup completion words every time when a user types
|
|
||||||
linenoise::SetCompletionCallback([](const char* editBuffer, std::vector<std::string>& completions) {
|
|
||||||
if (editBuffer[0] == 'h') {
|
|
||||||
completions.push_back("hello");
|
|
||||||
completions.push_back("hello there");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Enable the multi-line mode
|
|
||||||
linenoise::SetMultiLine(true);
|
|
||||||
|
|
||||||
// Set max length of the history
|
|
||||||
linenoise::SetHistoryMaxLen(4);
|
|
||||||
|
|
||||||
// Load history
|
|
||||||
linenoise::LoadHistory(path);
|
|
||||||
|
|
||||||
while (true) {
|
|
||||||
// Read line
|
|
||||||
std::string line;
|
|
||||||
auto quit = linenoise::Readline("hello> ", line);
|
|
||||||
|
|
||||||
if (quit) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
cout << "echo: '" << line << "'" << endl;
|
|
||||||
|
|
||||||
// Add text to history
|
|
||||||
linenoise::AddHistory(line.c_str());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save history
|
|
||||||
linenoise::SaveHistory(path);
|
|
||||||
```
|
|
||||||
|
|
||||||
API
|
|
||||||
---
|
|
||||||
|
|
||||||
```c++
|
|
||||||
namespace linenoise;
|
|
||||||
|
|
||||||
std::string Readline(const char* prompt);
|
|
||||||
|
|
||||||
void SetMultiLine(bool multiLineMode);
|
|
||||||
|
|
||||||
typedef std::function<void (const char* editBuffer, std::vector<std::string>& completions)> CompletionCallback;
|
|
||||||
|
|
||||||
void SetCompletionCallback(CompletionCallback fn);
|
|
||||||
|
|
||||||
bool SetHistoryMaxLen(size_t len);
|
|
||||||
|
|
||||||
bool LoadHistory(const char* path);
|
|
||||||
|
|
||||||
bool SaveHistory(const char* path);
|
|
||||||
|
|
||||||
bool AddHistory(const char* line);
|
|
||||||
|
|
||||||
const std::vector<std::string>& GetHistory();
|
|
||||||
```
|
|
||||||
|
|
||||||
Tested compilers
|
|
||||||
----------------
|
|
||||||
|
|
||||||
* Visual Studio 2015
|
|
||||||
* Clang 3.5
|
|
||||||
* GCC 6.3.1
|
|
||||||
|
|
||||||
License
|
|
||||||
-------
|
|
||||||
|
|
||||||
BSD license (© 2015 Yuji Hirose)
|
|
@ -1,5 +0,0 @@
|
|||||||
cmake_minimum_required(VERSION 3.0)
|
|
||||||
include_directories(.)
|
|
||||||
add_definitions("-std=c++1y")
|
|
||||||
|
|
||||||
add_executable(example example.cpp)
|
|
54
third_party/cpp-linenoise/example/example.cpp
vendored
54
third_party/cpp-linenoise/example/example.cpp
vendored
@ -1,54 +0,0 @@
|
|||||||
#include <iostream>
|
|
||||||
#include "../linenoise.hpp"
|
|
||||||
|
|
||||||
using namespace std;
|
|
||||||
|
|
||||||
int main(int argc, const char** argv)
|
|
||||||
{
|
|
||||||
const auto path = "history.txt";
|
|
||||||
|
|
||||||
// Enable the multi-line mode
|
|
||||||
linenoise::SetMultiLine(true);
|
|
||||||
|
|
||||||
// Set max length of the history
|
|
||||||
linenoise::SetHistoryMaxLen(4);
|
|
||||||
|
|
||||||
// Setup completion words every time when a user types
|
|
||||||
linenoise::SetCompletionCallback([](const char* editBuffer, std::vector<std::string>& completions) {
|
|
||||||
if (editBuffer[0] == 'h') {
|
|
||||||
#ifdef _WIN32
|
|
||||||
completions.push_back("hello こんにちは");
|
|
||||||
completions.push_back("hello こんにちは there");
|
|
||||||
#else
|
|
||||||
completions.push_back("hello");
|
|
||||||
completions.push_back("hello there");
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
// Load history
|
|
||||||
linenoise::LoadHistory(path);
|
|
||||||
|
|
||||||
while (true) {
|
|
||||||
std::string line;
|
|
||||||
#ifdef _WIN32
|
|
||||||
auto quit = linenoise::Readline("hello> ", line);
|
|
||||||
#else
|
|
||||||
auto quit = linenoise::Readline("\033[32mこんにちは\x1b[0m> ", line);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (quit) {
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
cout << "echo: '" << line << "'" << endl;
|
|
||||||
|
|
||||||
// Add line to history
|
|
||||||
linenoise::AddHistory(line.c_str());
|
|
||||||
|
|
||||||
// Save history
|
|
||||||
linenoise::SaveHistory(path);
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
|
||||||
}
|
|
28
third_party/cpp-linenoise/example/example.sln
vendored
28
third_party/cpp-linenoise/example/example.sln
vendored
@ -1,28 +0,0 @@
|
|||||||
|
|
||||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
|
||||||
# Visual Studio 14
|
|
||||||
VisualStudioVersion = 14.0.23107.0
|
|
||||||
MinimumVisualStudioVersion = 10.0.40219.1
|
|
||||||
Project("{8BC9CEB8-8B4A-11D0-8D11-00A0C91BC942}") = "example", "example.vcxproj", "{563BF990-4217-439F-92A4-F8A285052772}"
|
|
||||||
EndProject
|
|
||||||
Global
|
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
|
||||||
Debug|x64 = Debug|x64
|
|
||||||
Debug|x86 = Debug|x86
|
|
||||||
Release|x64 = Release|x64
|
|
||||||
Release|x86 = Release|x86
|
|
||||||
EndGlobalSection
|
|
||||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Debug|x64.ActiveCfg = Debug|x64
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Debug|x64.Build.0 = Debug|x64
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Debug|x86.ActiveCfg = Debug|Win32
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Debug|x86.Build.0 = Debug|Win32
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Release|x64.ActiveCfg = Release|x64
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Release|x64.Build.0 = Release|x64
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Release|x86.ActiveCfg = Release|Win32
|
|
||||||
{563BF990-4217-439F-92A4-F8A285052772}.Release|x86.Build.0 = Release|Win32
|
|
||||||
EndGlobalSection
|
|
||||||
GlobalSection(SolutionProperties) = preSolution
|
|
||||||
HideSolutionNode = FALSE
|
|
||||||
EndGlobalSection
|
|
||||||
EndGlobal
|
|
153
third_party/cpp-linenoise/example/example.vcxproj
vendored
153
third_party/cpp-linenoise/example/example.vcxproj
vendored
@ -1,153 +0,0 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
|
||||||
<Project DefaultTargets="Build" ToolsVersion="14.0" xmlns="http://schemas.microsoft.com/developer/msbuild/2003">
|
|
||||||
<ItemGroup Label="ProjectConfigurations">
|
|
||||||
<ProjectConfiguration Include="Debug|Win32">
|
|
||||||
<Configuration>Debug</Configuration>
|
|
||||||
<Platform>Win32</Platform>
|
|
||||||
</ProjectConfiguration>
|
|
||||||
<ProjectConfiguration Include="Release|Win32">
|
|
||||||
<Configuration>Release</Configuration>
|
|
||||||
<Platform>Win32</Platform>
|
|
||||||
</ProjectConfiguration>
|
|
||||||
<ProjectConfiguration Include="Debug|x64">
|
|
||||||
<Configuration>Debug</Configuration>
|
|
||||||
<Platform>x64</Platform>
|
|
||||||
</ProjectConfiguration>
|
|
||||||
<ProjectConfiguration Include="Release|x64">
|
|
||||||
<Configuration>Release</Configuration>
|
|
||||||
<Platform>x64</Platform>
|
|
||||||
</ProjectConfiguration>
|
|
||||||
</ItemGroup>
|
|
||||||
<PropertyGroup Label="Globals">
|
|
||||||
<ProjectGuid>{563BF990-4217-439F-92A4-F8A285052772}</ProjectGuid>
|
|
||||||
<Keyword>Win32Proj</Keyword>
|
|
||||||
<RootNamespace>example</RootNamespace>
|
|
||||||
<WindowsTargetPlatformVersion>8.1</WindowsTargetPlatformVersion>
|
|
||||||
</PropertyGroup>
|
|
||||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.Default.props" />
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'" Label="Configuration">
|
|
||||||
<ConfigurationType>Application</ConfigurationType>
|
|
||||||
<UseDebugLibraries>true</UseDebugLibraries>
|
|
||||||
<PlatformToolset>v140</PlatformToolset>
|
|
||||||
<CharacterSet>Unicode</CharacterSet>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'" Label="Configuration">
|
|
||||||
<ConfigurationType>Application</ConfigurationType>
|
|
||||||
<UseDebugLibraries>false</UseDebugLibraries>
|
|
||||||
<PlatformToolset>v140</PlatformToolset>
|
|
||||||
<WholeProgramOptimization>true</WholeProgramOptimization>
|
|
||||||
<CharacterSet>Unicode</CharacterSet>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'" Label="Configuration">
|
|
||||||
<ConfigurationType>Application</ConfigurationType>
|
|
||||||
<UseDebugLibraries>true</UseDebugLibraries>
|
|
||||||
<PlatformToolset>v140</PlatformToolset>
|
|
||||||
<CharacterSet>Unicode</CharacterSet>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'" Label="Configuration">
|
|
||||||
<ConfigurationType>Application</ConfigurationType>
|
|
||||||
<UseDebugLibraries>false</UseDebugLibraries>
|
|
||||||
<PlatformToolset>v140</PlatformToolset>
|
|
||||||
<WholeProgramOptimization>true</WholeProgramOptimization>
|
|
||||||
<CharacterSet>Unicode</CharacterSet>
|
|
||||||
</PropertyGroup>
|
|
||||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.props" />
|
|
||||||
<ImportGroup Label="ExtensionSettings">
|
|
||||||
</ImportGroup>
|
|
||||||
<ImportGroup Label="Shared">
|
|
||||||
</ImportGroup>
|
|
||||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
|
||||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
|
||||||
</ImportGroup>
|
|
||||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
|
||||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
|
||||||
</ImportGroup>
|
|
||||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
|
||||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
|
||||||
</ImportGroup>
|
|
||||||
<ImportGroup Label="PropertySheets" Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
|
||||||
<Import Project="$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props" Condition="exists('$(UserRootDir)\Microsoft.Cpp.$(Platform).user.props')" Label="LocalAppDataPlatform" />
|
|
||||||
</ImportGroup>
|
|
||||||
<PropertyGroup Label="UserMacros" />
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
|
||||||
<LinkIncremental>true</LinkIncremental>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
|
||||||
<LinkIncremental>true</LinkIncremental>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
|
||||||
<LinkIncremental>false</LinkIncremental>
|
|
||||||
</PropertyGroup>
|
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
|
||||||
<LinkIncremental>false</LinkIncremental>
|
|
||||||
</PropertyGroup>
|
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|Win32'">
|
|
||||||
<ClCompile>
|
|
||||||
<PrecompiledHeader>
|
|
||||||
</PrecompiledHeader>
|
|
||||||
<WarningLevel>Level3</WarningLevel>
|
|
||||||
<Optimization>Disabled</Optimization>
|
|
||||||
<PreprocessorDefinitions>WIN32;_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
|
||||||
</ClCompile>
|
|
||||||
<Link>
|
|
||||||
<SubSystem>Console</SubSystem>
|
|
||||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
|
||||||
</Link>
|
|
||||||
</ItemDefinitionGroup>
|
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Debug|x64'">
|
|
||||||
<ClCompile>
|
|
||||||
<PrecompiledHeader>
|
|
||||||
</PrecompiledHeader>
|
|
||||||
<WarningLevel>Level3</WarningLevel>
|
|
||||||
<Optimization>Disabled</Optimization>
|
|
||||||
<PreprocessorDefinitions>_DEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
|
||||||
</ClCompile>
|
|
||||||
<Link>
|
|
||||||
<SubSystem>Console</SubSystem>
|
|
||||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
|
||||||
</Link>
|
|
||||||
</ItemDefinitionGroup>
|
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|Win32'">
|
|
||||||
<ClCompile>
|
|
||||||
<WarningLevel>Level3</WarningLevel>
|
|
||||||
<PrecompiledHeader>
|
|
||||||
</PrecompiledHeader>
|
|
||||||
<Optimization>MaxSpeed</Optimization>
|
|
||||||
<FunctionLevelLinking>true</FunctionLevelLinking>
|
|
||||||
<IntrinsicFunctions>true</IntrinsicFunctions>
|
|
||||||
<PreprocessorDefinitions>WIN32;NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
|
||||||
</ClCompile>
|
|
||||||
<Link>
|
|
||||||
<SubSystem>Console</SubSystem>
|
|
||||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
|
||||||
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
|
||||||
<OptimizeReferences>true</OptimizeReferences>
|
|
||||||
</Link>
|
|
||||||
</ItemDefinitionGroup>
|
|
||||||
<ItemDefinitionGroup Condition="'$(Configuration)|$(Platform)'=='Release|x64'">
|
|
||||||
<ClCompile>
|
|
||||||
<WarningLevel>Level3</WarningLevel>
|
|
||||||
<PrecompiledHeader>
|
|
||||||
</PrecompiledHeader>
|
|
||||||
<Optimization>MaxSpeed</Optimization>
|
|
||||||
<FunctionLevelLinking>true</FunctionLevelLinking>
|
|
||||||
<IntrinsicFunctions>true</IntrinsicFunctions>
|
|
||||||
<PreprocessorDefinitions>NDEBUG;_CONSOLE;%(PreprocessorDefinitions)</PreprocessorDefinitions>
|
|
||||||
</ClCompile>
|
|
||||||
<Link>
|
|
||||||
<SubSystem>Console</SubSystem>
|
|
||||||
<GenerateDebugInformation>true</GenerateDebugInformation>
|
|
||||||
<EnableCOMDATFolding>true</EnableCOMDATFolding>
|
|
||||||
<OptimizeReferences>true</OptimizeReferences>
|
|
||||||
</Link>
|
|
||||||
</ItemDefinitionGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<ClCompile Include="example.cpp" />
|
|
||||||
</ItemGroup>
|
|
||||||
<ItemGroup>
|
|
||||||
<ClInclude Include="..\linenoise.hpp" />
|
|
||||||
</ItemGroup>
|
|
||||||
<Import Project="$(VCTargetsPath)\Microsoft.Cpp.targets" />
|
|
||||||
<ImportGroup Label="ExtensionTargets">
|
|
||||||
</ImportGroup>
|
|
||||||
</Project>
|
|
2421
third_party/cpp-linenoise/linenoise.hpp
vendored
2421
third_party/cpp-linenoise/linenoise.hpp
vendored
File diff suppressed because it is too large
Load Diff
160
third_party/jsoncpp/json/json-forwards.h
vendored
160
third_party/jsoncpp/json/json-forwards.h
vendored
@ -7,32 +7,32 @@
|
|||||||
// //////////////////////////////////////////////////////////////////////
|
// //////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
/*
|
/*
|
||||||
The JsonCpp library's source code, including accompanying documentation,
|
The JsonCpp library's source code, including accompanying documentation,
|
||||||
tests and demonstration applications, are licensed under the following
|
tests and demonstration applications, are licensed under the following
|
||||||
conditions...
|
conditions...
|
||||||
|
|
||||||
The author (Baptiste Lepilleur) explicitly disclaims copyright in all
|
Baptiste Lepilleur and The JsonCpp Authors explicitly disclaim copyright in all
|
||||||
jurisdictions which recognize such a disclaimer. In such jurisdictions,
|
jurisdictions which recognize such a disclaimer. In such jurisdictions,
|
||||||
this software is released into the Public Domain.
|
this software is released into the Public Domain.
|
||||||
|
|
||||||
In jurisdictions which do not recognize Public Domain property (e.g. Germany as of
|
In jurisdictions which do not recognize Public Domain property (e.g. Germany as of
|
||||||
2010), this software is Copyright (c) 2007-2010 by Baptiste Lepilleur, and is
|
2010), this software is Copyright (c) 2007-2010 by Baptiste Lepilleur and
|
||||||
released under the terms of the MIT License (see below).
|
The JsonCpp Authors, and is released under the terms of the MIT License (see below).
|
||||||
|
|
||||||
In jurisdictions which recognize Public Domain property, the user of this
|
In jurisdictions which recognize Public Domain property, the user of this
|
||||||
software may choose to accept it either as 1) Public Domain, 2) under the
|
software may choose to accept it either as 1) Public Domain, 2) under the
|
||||||
conditions of the MIT License (see below), or 3) under the terms of dual
|
conditions of the MIT License (see below), or 3) under the terms of dual
|
||||||
Public Domain/MIT License conditions described here, as they choose.
|
Public Domain/MIT License conditions described here, as they choose.
|
||||||
|
|
||||||
The MIT License is about as close to Public Domain as a license can get, and is
|
The MIT License is about as close to Public Domain as a license can get, and is
|
||||||
described in clear, concise terms at:
|
described in clear, concise terms at:
|
||||||
|
|
||||||
http://en.wikipedia.org/wiki/MIT_License
|
http://en.wikipedia.org/wiki/MIT_License
|
||||||
|
|
||||||
The full text of the MIT License follows:
|
The full text of the MIT License follows:
|
||||||
|
|
||||||
========================================================================
|
========================================================================
|
||||||
Copyright (c) 2007-2010 Baptiste Lepilleur
|
Copyright (c) 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person
|
Permission is hereby granted, free of charge, to any person
|
||||||
obtaining a copy of this software and associated documentation
|
obtaining a copy of this software and associated documentation
|
||||||
@ -83,13 +83,16 @@ license you like.
|
|||||||
// Beginning of content of file: include/json/config.h
|
// Beginning of content of file: include/json/config.h
|
||||||
// //////////////////////////////////////////////////////////////////////
|
// //////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
// Copyright 2007-2010 Baptiste Lepilleur
|
// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
|
||||||
// Distributed under MIT license, or public domain if desired and
|
// Distributed under MIT license, or public domain if desired and
|
||||||
// recognized in your jurisdiction.
|
// recognized in your jurisdiction.
|
||||||
// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
||||||
|
|
||||||
#ifndef JSON_CONFIG_H_INCLUDED
|
#ifndef JSON_CONFIG_H_INCLUDED
|
||||||
#define JSON_CONFIG_H_INCLUDED
|
#define JSON_CONFIG_H_INCLUDED
|
||||||
|
#include <stddef.h>
|
||||||
|
#include <string> //typedef String
|
||||||
|
#include <stdint.h> //typedef int64_t, uint64_t
|
||||||
|
|
||||||
/// If defined, indicates that json library is embedded in CppTL library.
|
/// If defined, indicates that json library is embedded in CppTL library.
|
||||||
//# define JSON_IN_CPPTL 1
|
//# define JSON_IN_CPPTL 1
|
||||||
@ -122,12 +125,12 @@ license you like.
|
|||||||
#ifdef JSON_IN_CPPTL
|
#ifdef JSON_IN_CPPTL
|
||||||
#define JSON_API CPPTL_API
|
#define JSON_API CPPTL_API
|
||||||
#elif defined(JSON_DLL_BUILD)
|
#elif defined(JSON_DLL_BUILD)
|
||||||
#if defined(_MSC_VER)
|
#if defined(_MSC_VER) || defined(__MINGW32__)
|
||||||
#define JSON_API __declspec(dllexport)
|
#define JSON_API __declspec(dllexport)
|
||||||
#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
|
#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
|
||||||
#endif // if defined(_MSC_VER)
|
#endif // if defined(_MSC_VER)
|
||||||
#elif defined(JSON_DLL)
|
#elif defined(JSON_DLL)
|
||||||
#if defined(_MSC_VER)
|
#if defined(_MSC_VER) || defined(__MINGW32__)
|
||||||
#define JSON_API __declspec(dllimport)
|
#define JSON_API __declspec(dllimport)
|
||||||
#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
|
#define JSONCPP_DISABLE_DLL_INTERFACE_WARNING
|
||||||
#endif // if defined(_MSC_VER)
|
#endif // if defined(_MSC_VER)
|
||||||
@ -136,49 +139,101 @@ license you like.
|
|||||||
#define JSON_API
|
#define JSON_API
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#if !defined(JSON_HAS_UNIQUE_PTR)
|
|
||||||
#if __cplusplus >= 201103L
|
|
||||||
#define JSON_HAS_UNIQUE_PTR (1)
|
|
||||||
#elif _MSC_VER >= 1600
|
|
||||||
#define JSON_HAS_UNIQUE_PTR (1)
|
|
||||||
#else
|
|
||||||
#define JSON_HAS_UNIQUE_PTR (0)
|
|
||||||
#endif
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// If JSON_NO_INT64 is defined, then Json only support C++ "int" type for
|
// If JSON_NO_INT64 is defined, then Json only support C++ "int" type for
|
||||||
// integer
|
// integer
|
||||||
// Storages, and 64 bits integer support is disabled.
|
// Storages, and 64 bits integer support is disabled.
|
||||||
// #define JSON_NO_INT64 1
|
// #define JSON_NO_INT64 1
|
||||||
|
|
||||||
#if defined(_MSC_VER) && _MSC_VER <= 1200 // MSVC 6
|
#if defined(_MSC_VER) // MSVC
|
||||||
// Microsoft Visual Studio 6 only support conversion from __int64 to double
|
# if _MSC_VER <= 1200 // MSVC 6
|
||||||
// (no conversion from unsigned __int64).
|
// Microsoft Visual Studio 6 only support conversion from __int64 to double
|
||||||
#define JSON_USE_INT64_DOUBLE_CONVERSION 1
|
// (no conversion from unsigned __int64).
|
||||||
// Disable warning 4786 for VS6 caused by STL (identifier was truncated to '255'
|
# define JSON_USE_INT64_DOUBLE_CONVERSION 1
|
||||||
// characters in the debug information)
|
// Disable warning 4786 for VS6 caused by STL (identifier was truncated to '255'
|
||||||
// All projects I've ever seen with VS6 were using this globally (not bothering
|
// characters in the debug information)
|
||||||
// with pragma push/pop).
|
// All projects I've ever seen with VS6 were using this globally (not bothering
|
||||||
#pragma warning(disable : 4786)
|
// with pragma push/pop).
|
||||||
#endif // if defined(_MSC_VER) && _MSC_VER < 1200 // MSVC 6
|
# pragma warning(disable : 4786)
|
||||||
|
# endif // MSVC 6
|
||||||
|
|
||||||
#if defined(_MSC_VER) && _MSC_VER >= 1500 // MSVC 2008
|
# if _MSC_VER >= 1500 // MSVC 2008
|
||||||
/// Indicates that the following function is deprecated.
|
/// Indicates that the following function is deprecated.
|
||||||
#define JSONCPP_DEPRECATED(message) __declspec(deprecated(message))
|
# define JSONCPP_DEPRECATED(message) __declspec(deprecated(message))
|
||||||
#elif defined(__clang__) && defined(__has_feature)
|
# endif
|
||||||
#if __has_feature(attribute_deprecated_with_message)
|
|
||||||
#define JSONCPP_DEPRECATED(message) __attribute__ ((deprecated(message)))
|
#endif // defined(_MSC_VER)
|
||||||
|
|
||||||
|
// In c++11 the override keyword allows you to explicity define that a function
|
||||||
|
// is intended to override the base-class version. This makes the code more
|
||||||
|
// managable and fixes a set of common hard-to-find bugs.
|
||||||
|
#if __cplusplus >= 201103L
|
||||||
|
# define JSONCPP_OVERRIDE override
|
||||||
|
# define JSONCPP_NOEXCEPT noexcept
|
||||||
|
#elif defined(_MSC_VER) && _MSC_VER > 1600 && _MSC_VER < 1900
|
||||||
|
# define JSONCPP_OVERRIDE override
|
||||||
|
# define JSONCPP_NOEXCEPT throw()
|
||||||
|
#elif defined(_MSC_VER) && _MSC_VER >= 1900
|
||||||
|
# define JSONCPP_OVERRIDE override
|
||||||
|
# define JSONCPP_NOEXCEPT noexcept
|
||||||
|
#else
|
||||||
|
# define JSONCPP_OVERRIDE
|
||||||
|
# define JSONCPP_NOEXCEPT throw()
|
||||||
#endif
|
#endif
|
||||||
#elif defined(__GNUC__) && (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5))
|
|
||||||
#define JSONCPP_DEPRECATED(message) __attribute__ ((deprecated(message)))
|
#ifndef JSON_HAS_RVALUE_REFERENCES
|
||||||
#elif defined(__GNUC__) && (__GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 1))
|
|
||||||
#define JSONCPP_DEPRECATED(message) __attribute__((__deprecated__))
|
#if defined(_MSC_VER) && _MSC_VER >= 1600 // MSVC >= 2010
|
||||||
|
#define JSON_HAS_RVALUE_REFERENCES 1
|
||||||
|
#endif // MSVC >= 2010
|
||||||
|
|
||||||
|
#ifdef __clang__
|
||||||
|
#if __has_feature(cxx_rvalue_references)
|
||||||
|
#define JSON_HAS_RVALUE_REFERENCES 1
|
||||||
|
#endif // has_feature
|
||||||
|
|
||||||
|
#elif defined __GNUC__ // not clang (gcc comes later since clang emulates gcc)
|
||||||
|
#if defined(__GXX_EXPERIMENTAL_CXX0X__) || (__cplusplus >= 201103L)
|
||||||
|
#define JSON_HAS_RVALUE_REFERENCES 1
|
||||||
|
#endif // GXX_EXPERIMENTAL
|
||||||
|
|
||||||
|
#endif // __clang__ || __GNUC__
|
||||||
|
|
||||||
|
#endif // not defined JSON_HAS_RVALUE_REFERENCES
|
||||||
|
|
||||||
|
#ifndef JSON_HAS_RVALUE_REFERENCES
|
||||||
|
#define JSON_HAS_RVALUE_REFERENCES 0
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifdef __clang__
|
||||||
|
# if __has_extension(attribute_deprecated_with_message)
|
||||||
|
# define JSONCPP_DEPRECATED(message) __attribute__ ((deprecated(message)))
|
||||||
|
# endif
|
||||||
|
#elif defined __GNUC__ // not clang (gcc comes later since clang emulates gcc)
|
||||||
|
# if (__GNUC__ > 4 || (__GNUC__ == 4 && __GNUC_MINOR__ >= 5))
|
||||||
|
# define JSONCPP_DEPRECATED(message) __attribute__ ((deprecated(message)))
|
||||||
|
# elif (__GNUC__ > 3 || (__GNUC__ == 3 && __GNUC_MINOR__ >= 1))
|
||||||
|
# define JSONCPP_DEPRECATED(message) __attribute__((__deprecated__))
|
||||||
|
# endif // GNUC version
|
||||||
|
#endif // __clang__ || __GNUC__
|
||||||
|
|
||||||
#if !defined(JSONCPP_DEPRECATED)
|
#if !defined(JSONCPP_DEPRECATED)
|
||||||
#define JSONCPP_DEPRECATED(message)
|
#define JSONCPP_DEPRECATED(message)
|
||||||
#endif // if !defined(JSONCPP_DEPRECATED)
|
#endif // if !defined(JSONCPP_DEPRECATED)
|
||||||
|
|
||||||
|
#if __GNUC__ >= 6
|
||||||
|
# define JSON_USE_INT64_DOUBLE_CONVERSION 1
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#if !defined(JSON_IS_AMALGAMATION)
|
||||||
|
|
||||||
|
# include "version.h"
|
||||||
|
|
||||||
|
# if JSONCPP_USING_SECURE_MEMORY
|
||||||
|
# include "allocator.h" //typedef Allocator
|
||||||
|
# endif
|
||||||
|
|
||||||
|
#endif // if !defined(JSON_IS_AMALGAMATION)
|
||||||
|
|
||||||
namespace Json {
|
namespace Json {
|
||||||
typedef int Int;
|
typedef int Int;
|
||||||
typedef unsigned int UInt;
|
typedef unsigned int UInt;
|
||||||
@ -192,13 +247,26 @@ typedef unsigned int LargestUInt;
|
|||||||
typedef __int64 Int64;
|
typedef __int64 Int64;
|
||||||
typedef unsigned __int64 UInt64;
|
typedef unsigned __int64 UInt64;
|
||||||
#else // if defined(_MSC_VER) // Other platforms, use long long
|
#else // if defined(_MSC_VER) // Other platforms, use long long
|
||||||
typedef long long int Int64;
|
typedef int64_t Int64;
|
||||||
typedef unsigned long long int UInt64;
|
typedef uint64_t UInt64;
|
||||||
#endif // if defined(_MSC_VER)
|
#endif // if defined(_MSC_VER)
|
||||||
typedef Int64 LargestInt;
|
typedef Int64 LargestInt;
|
||||||
typedef UInt64 LargestUInt;
|
typedef UInt64 LargestUInt;
|
||||||
#define JSON_HAS_INT64
|
#define JSON_HAS_INT64
|
||||||
#endif // if defined(JSON_NO_INT64)
|
#endif // if defined(JSON_NO_INT64)
|
||||||
|
#if JSONCPP_USING_SECURE_MEMORY
|
||||||
|
#define JSONCPP_STRING std::basic_string<char, std::char_traits<char>, Json::SecureAllocator<char> >
|
||||||
|
#define JSONCPP_OSTRINGSTREAM std::basic_ostringstream<char, std::char_traits<char>, Json::SecureAllocator<char> >
|
||||||
|
#define JSONCPP_OSTREAM std::basic_ostream<char, std::char_traits<char>>
|
||||||
|
#define JSONCPP_ISTRINGSTREAM std::basic_istringstream<char, std::char_traits<char>, Json::SecureAllocator<char> >
|
||||||
|
#define JSONCPP_ISTREAM std::istream
|
||||||
|
#else
|
||||||
|
#define JSONCPP_STRING std::string
|
||||||
|
#define JSONCPP_OSTRINGSTREAM std::ostringstream
|
||||||
|
#define JSONCPP_OSTREAM std::ostream
|
||||||
|
#define JSONCPP_ISTRINGSTREAM std::istringstream
|
||||||
|
#define JSONCPP_ISTREAM std::istream
|
||||||
|
#endif // if JSONCPP_USING_SECURE_MEMORY
|
||||||
} // end namespace Json
|
} // end namespace Json
|
||||||
|
|
||||||
#endif // JSON_CONFIG_H_INCLUDED
|
#endif // JSON_CONFIG_H_INCLUDED
|
||||||
@ -216,7 +284,7 @@ typedef UInt64 LargestUInt;
|
|||||||
// Beginning of content of file: include/json/forwards.h
|
// Beginning of content of file: include/json/forwards.h
|
||||||
// //////////////////////////////////////////////////////////////////////
|
// //////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
// Copyright 2007-2010 Baptiste Lepilleur
|
// Copyright 2007-2010 Baptiste Lepilleur and The JsonCpp Authors
|
||||||
// Distributed under MIT license, or public domain if desired and
|
// Distributed under MIT license, or public domain if desired and
|
||||||
// recognized in your jurisdiction.
|
// recognized in your jurisdiction.
|
||||||
// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
// See file LICENSE for detail or copy at http://jsoncpp.sourceforge.net/LICENSE
|
||||||
|
530
third_party/jsoncpp/json/json.h
vendored
530
third_party/jsoncpp/json/json.h
vendored
File diff suppressed because it is too large
Load Diff
1180
third_party/jsoncpp/jsoncpp.cpp
vendored
1180
third_party/jsoncpp/jsoncpp.cpp
vendored
File diff suppressed because it is too large
Load Diff
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user