2022-06-25 00:23:03 +08:00

379 lines
9.7 KiB
Plaintext

/****************************************************************************
Copyright (c) 2017-2018 Xiamen Yaji Software Co., Ltd.
http://www.cocos.com
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated engine source code (the "Software"), a limited,
worldwide, royalty-free, non-assignable, revocable and non-exclusive license
to use Cocos Creator solely to develop games on your target platforms. You shall
not use Cocos Creator software for developing other software or tools that's
used for developing games. You are not granted to publish, distribute,
sublicense, and/or sell copies of Cocos Creator.
The software or tools in this License Agreement are licensed, not sold.
Xiamen Yaji Software Co., Ltd. reserves all rights not expressly granted to you.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
****************************************************************************/
#include "network/WebSocket.h"
#include "base/CCData.h"
#import "SocketRocket/SocketRocket.h"
#if !__has_feature(objc_arc)
#error WebSocket must be compiled with ARC enabled
#endif
static std::vector<cocos2d::network::WebSocket*>* __websocketInstances = nullptr;
@interface WebSocketImpl : NSObject<SRWebSocketDelegate>
{
}
@end
//
@implementation WebSocketImpl
{
SRWebSocket* _ws;
cocos2d::network::WebSocket* _ccws;
cocos2d::network::WebSocket::Delegate* _delegate;
std::string _url;
std::string _selectedProtocol;
bool _isDestroyed;
}
-(id) initWithURL:(const std::string&) url protocols:(NSArray<NSString *> *)protocols allowsUntrustedSSLCertificates:(BOOL)allowsUntrustedSSLCertificates ws:(cocos2d::network::WebSocket*) ccws delegate:(const cocos2d::network::WebSocket::Delegate&) delegate
{
if (self = [super init])
{
_ccws = ccws;
_delegate = const_cast<cocos2d::network::WebSocket::Delegate*>(&delegate);
_url = url;
NSURL* nsUrl = [[NSURL alloc] initWithString:[[NSString alloc] initWithUTF8String:_url.c_str()]];
_ws = [[SRWebSocket alloc] initWithURL:nsUrl protocols:protocols allowsUntrustedSSLCertificates:allowsUntrustedSSLCertificates];
_ws.delegate = self;
[_ws open];
_isDestroyed = false;
}
return self;
}
-(void) dealloc
{
// NSLog(@"WebSocketImpl-apple dealloc: %p, SRWebSocket ref: %ld", self, CFGetRetainCount((__bridge CFTypeRef)_ws));
}
-(void) sendString:(NSString*) message
{
[_ws sendString:message error:nil];
}
-(void) sendData:(NSData*) data
{
[_ws sendData:data error:nil];
}
-(void) close
{
_isDestroyed = true;
_ccws->retain();
_delegate->onClose(_ccws);
[_ws close];
_ccws->release();
}
-(void) closeAsync
{
[_ws close];
}
-(cocos2d::network::WebSocket::State) getReadyState
{
cocos2d::network::WebSocket::State ret;
SRReadyState state = _ws.readyState;
switch (state) {
case SR_OPEN:
ret = cocos2d::network::WebSocket::State::OPEN;
break;
case SR_CONNECTING:
ret = cocos2d::network::WebSocket::State::CONNECTING;
break;
case SR_CLOSING:
ret = cocos2d::network::WebSocket::State::CLOSING;
break;
default:
ret = cocos2d::network::WebSocket::State::CLOSED;
break;
}
return ret;
}
-(const std::string&) getUrl
{
return _url;
}
-(const std::string&) getProtocol
{
return _selectedProtocol;
}
-(cocos2d::network::WebSocket::Delegate*) getDelegate
{
return (cocos2d::network::WebSocket::Delegate*)_delegate;
}
// Delegate methods
-(void)webSocketDidOpen:(SRWebSocket *)webSocket;
{
if (!_isDestroyed)
{
// NSLog(@"Websocket Connected");
if (webSocket.protocol != nil)
_selectedProtocol = [webSocket.protocol UTF8String];
_delegate->onOpen(_ccws);
}
else
{
NSLog(@"WebSocketImpl webSocketDidOpen was destroyed!");
}
}
- (void)webSocket:(SRWebSocket *)webSocket didFailWithError:(NSError *)error;
{
if (!_isDestroyed)
{
NSLog(@":( Websocket Failed With Error %@", error);
_delegate->onError(_ccws, cocos2d::network::WebSocket::ErrorCode::UNKNOWN);
[self webSocket:webSocket didCloseWithCode:0 reason:@"onerror" wasClean:YES];
}
else
{
NSLog(@"WebSocketImpl didFailWithError was destroyed!");
}
}
- (void)webSocket:(SRWebSocket *)webSocket didReceiveMessageWithString:(nonnull NSString *)string
{
if (!_isDestroyed)
{
cocos2d::network::WebSocket::Data data;
data.bytes = (char*) [string cStringUsingEncoding:NSUTF8StringEncoding];
data.len = [string lengthOfBytesUsingEncoding:NSUTF8StringEncoding];
data.isBinary = false;
data.issued = 0;
data.ext = nullptr;
_delegate->onMessage(_ccws, data);
}
else
{
NSLog(@"WebSocketImpl didReceiveMessageWithString was destroyed!");
}
}
- (void)webSocket:(SRWebSocket *)webSocket didReceiveMessageWithData:(NSData *)nsData
{
if (!_isDestroyed)
{
cocos2d::network::WebSocket::Data data;
data.bytes = (char*)nsData.bytes;
data.len = nsData.length;
data.isBinary = true;
data.issued = 0;
data.ext = nullptr;
_delegate->onMessage(_ccws, data);
}
else
{
NSLog(@"WebSocketImpl didReceiveMessageWithData was destroyed!");
}
}
- (void)webSocket:(SRWebSocket *)webSocket didCloseWithCode:(NSInteger)code reason:(NSString *)reason wasClean:(BOOL)wasClean
{
if (!_isDestroyed)
_delegate->onClose(_ccws);
else
NSLog(@"WebSocketImpl didCloseWithCode was destroyed!");
}
@end
NS_CC_BEGIN
namespace network {
void WebSocket::closeAllConnections()
{
if (__websocketInstances != nullptr)
{
ssize_t count = __websocketInstances->size();
for (ssize_t i = count-1; i >=0 ; i--)
{
WebSocket* instance = __websocketInstances->at(i);
instance->close();
}
__websocketInstances->clear();
delete __websocketInstances;
__websocketInstances = nullptr;
}
}
WebSocket::WebSocket()
: _impl(nil)
{
if (__websocketInstances == nullptr)
{
__websocketInstances = new (std::nothrow) std::vector<WebSocket*>();
}
__websocketInstances->push_back(this);
}
WebSocket::~WebSocket()
{
// NSLog(@"In the destructor of WebSocket-apple (%p).", this);
if (__websocketInstances != nullptr)
{
auto iter = std::find(__websocketInstances->begin(), __websocketInstances->end(), this);
if (iter != __websocketInstances->end())
{
__websocketInstances->erase(iter);
}
else
{
NSLog(@"ERROR: WebSocket instance wasn't added to the container which saves websocket instances!");
}
}
}
bool WebSocket::init(const Delegate& delegate,
const std::string& url,
const std::vector<std::string>* protocols/* = nullptr*/,
const std::string& caFilePath/* = ""*/)
{
if (url.empty())
return false;
NSMutableArray* nsProtocols = [[NSMutableArray alloc] init];
if (protocols != nullptr)
{
for (const auto& protocol : *protocols)
{
[nsProtocols addObject:[[NSString alloc] initWithUTF8String:protocol.c_str()]];
}
}
_impl = [[WebSocketImpl alloc] initWithURL: url protocols:nsProtocols allowsUntrustedSSLCertificates:NO ws: this delegate:delegate];
return _impl != nil;
}
void WebSocket::send(const std::string& message)
{
if ([_impl getReadyState] == State::OPEN)
{
NSString* str = [[NSString alloc] initWithBytes:message.data() length:message.length() encoding:NSUTF8StringEncoding];
[_impl sendString:str];
}
else
{
NSLog(@"Couldn't send message since websocket wasn't opened!");
}
}
void WebSocket::send(const unsigned char* binaryMsg, unsigned int len)
{
if ([_impl getReadyState] == State::OPEN)
{
NSData* data = [[NSData alloc] initWithBytes:binaryMsg length:(NSUInteger)len];
[_impl sendData:data];
}
else
{
NSLog(@"Couldn't send message since websocket wasn't opened!");
}
}
void WebSocket::close()
{
if ([_impl getReadyState] == State::CLOSING || [_impl getReadyState] == State::CLOSED)
{
NSLog(@"WebSocket (%p) was closed, no need to close it again!", this);
return;
}
[_impl close];
}
void WebSocket::closeAsync()
{
if ([_impl getReadyState] == State::CLOSING || [_impl getReadyState] == State::CLOSED)
{
NSLog(@"WebSocket (%p) was closed, no need to close it again!", this);
return;
}
[_impl closeAsync];
}
void WebSocket::closeAsync(int code, const std::string &reason)
{
//lws_close_reason() replacement required
closeAsync();
}
std::string WebSocket::getExtensions() const
{
//TODO websocket extensions
return "";
}
size_t WebSocket::getBufferedAmount() const
{
//TODO pending send bytes
return 0;
}
WebSocket::State WebSocket::getReadyState() const
{
return [_impl getReadyState];
}
const std::string& WebSocket::getUrl() const
{
return [_impl getUrl];
}
const std::string& WebSocket::getProtocol() const
{
return [_impl getProtocol];
}
WebSocket::Delegate* WebSocket::getDelegate() const
{
return [_impl getDelegate];
}
} // namespace network {
NS_CC_END