connectDeviceManager.m 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378
  1. //
  2. // connectDeviceManager.m
  3. // 隐私保护
  4. //
  5. // Created by xd h on 2023/10/9.
  6. //
  7. //第三方 瑞云 的链接服务
  8. #import "connectDeviceManager.h"
  9. #import <RayProxy/RaylinkProxy.h>
  10. #import <CocoaAsyncSocket/GCDAsyncSocket.h>
  11. #import "WHPingTester.h"
  12. @interface connectDeviceManager ()<
  13. GCDAsyncSocketDelegate,
  14. RaylinkProxyDelegate,
  15. WHPingDelegate
  16. >
  17. {
  18. dispatch_queue_t initProxyQueue;
  19. WHPingTester *whPingTester;
  20. }
  21. @property (nonatomic, strong) NSTimer *connectCheckTimer;
  22. @property (nonatomic, strong) GCDAsyncSocket *socket;
  23. @property (nonatomic, strong) GCDAsyncSocket *serverSocket;
  24. @property (nonatomic, assign) BOOL sdnConnected;
  25. @property (nonatomic, assign) BOOL peerConnected;
  26. @property (nonatomic, assign) NSInteger tcpPort;
  27. @end
  28. @implementation connectDeviceManager
  29. static connectDeviceManager *connectDeviceManagerInstance = nil;
  30. +(connectDeviceManager *)shareInstance;
  31. {
  32. static dispatch_once_t onceToken;
  33. dispatch_once(&onceToken, ^{
  34. connectDeviceManagerInstance = [[connectDeviceManager alloc] init];
  35. connectDeviceManagerInstance.curConnectDeviceState = DeviceConnectUnknown;
  36. });
  37. return connectDeviceManagerInstance;
  38. }
  39. #pragma mark 根据扫码的sn获取第三方信息
  40. -(void)getThridMsgBySN:(NSString*)snStr
  41. {
  42. _curConnectDeviceState = DeviceConnectGetThridMsging;
  43. NSMutableDictionary *paraDict = [NSMutableDictionary new];
  44. [paraDict setValue:snStr forKey:@"sn"];
  45. KWeakSelf
  46. [[netWorkManager shareInstance] CommonGetWithCallBackCode:getThirdIdBySn Parameters:paraDict success:^(id _Nonnull responseObject) {
  47. weakSelf.DeviceThirdIdMod = [[DeviceThirdIdModel alloc] initWithDictionary:responseObject error:nil];
  48. if(weakSelf.DeviceThirdIdMod.status == 0)
  49. {
  50. weakSelf.curConnectDeviceState = DeviceConnectGetThridOK;
  51. NSString *ipStr = weakSelf.DeviceThirdIdMod.data.ip;
  52. [weakSelf startPingDeviceIp:ipStr];
  53. [weakSelf initRuiyunSDKFun];
  54. }
  55. } failure:^(NSError * _Nonnull error) {
  56. HLog("网络报错");
  57. }];
  58. }
  59. #pragma mark 初始化瑞云的SDK等
  60. - (void)initRuiyunSDKFun{
  61. self.sdnConnected = NO;
  62. self.peerConnected = NO;
  63. //self.sendBtn.enabled = NO;
  64. //self.sdnConnectStateLab.text = @"SDN state: Connectting";
  65. //self.sdnIDLab.text = @"My SDN ID:";
  66. RaylinkProxy.sharedManager.delegate = self;
  67. initProxyQueue = dispatch_queue_create("init_proxy", 0);
  68. NSURL *logUrl = [[NSFileManager.defaultManager URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask].firstObject URLByAppendingPathComponent:@"logs"];
  69. if (![NSFileManager.defaultManager fileExistsAtPath:logUrl.path]) {
  70. [NSFileManager.defaultManager createDirectoryAtURL:logUrl withIntermediateDirectories:YES attributes:NULL error:NULL];
  71. }
  72. dispatch_async(initProxyQueue, ^{
  73. NSData *sdnInfo = [NSData dataWithContentsOfURL:[[NSBundle.mainBundle bundleURL] URLByAppendingPathComponent:@"planet.1ali_3ry_peer"]];
  74. /// 1. 初始化代理库
  75. [RaylinkProxy.sharedManager initProxy:logUrl.path rootSdnInfo:sdnInfo];
  76. });
  77. /// 作为服务端时,启动端口监听
  78. self.serverSocket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
  79. if ([self.serverSocket acceptOnPort:0 error:NULL]) {
  80. //self.serverPortLab.text = [NSString stringWithFormat: @"server port: %d", self.serverSocket.localPort];
  81. NSLog(@"accept ok %d",self.serverSocket.localPort);
  82. }
  83. /// 启动定时器监听 SND 连接状态
  84. self.connectCheckTimer = [NSTimer scheduledTimerWithTimeInterval:0.5 repeats:YES block:^(NSTimer * _Nonnull timer) {
  85. /// 2. 监听SDN 连接状态
  86. bool connected = [RaylinkProxy.sharedManager isSdnConnected];
  87. if (self.sdnConnected == connected) {
  88. return;
  89. }
  90. self.sdnConnected = connected;
  91. if (self.sdnConnected) {
  92. //self.sdnConnectStateLab.text = @"SDN state: Connected";
  93. //self.sdnIDLab.text = [@"My SDN ID: " stringByAppendingString:[RaylinkProxy.sharedManager getSdnId]];
  94. } else {
  95. //self.sdnConnectStateLab.text = @"SDN state: Connectting";
  96. }
  97. }];
  98. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  99. mainBlock(^{
  100. [self onConnectFun];
  101. });
  102. });
  103. }
  104. - (void)onConnectFun{
  105. if (self.peerConnected == YES) {
  106. [self disconnect];
  107. return;
  108. }
  109. _curConnectDeviceState = DeviceConnectDeciceing;
  110. NSString *curSdnId = self.DeviceThirdIdMod.data.sdnId; //@"3dfe7c1f";
  111. /// 3. 根据对端的 SDN ID 创建连接
  112. [RaylinkProxy.sharedManager createNewConnection:curSdnId];
  113. /// 4. 创建 HttpService
  114. self.tcpPort = [RaylinkProxy.sharedManager createHttpService:curSdnId];
  115. self.tcpPortStr = [[NSString alloc] initWithFormat:@"%ld",self.tcpPort];
  116. self.socket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
  117. /// 6. 添加端口防火墙
  118. [RaylinkProxy.sharedManager addSdnId:curSdnId allowPort:9300];
  119. [RaylinkProxy.sharedManager addSdnId:curSdnId allowPort:9100];
  120. //[self.connectBtn setEnabled:NO];
  121. //[self.connectBtn setTitle:@"连接中..." forState:UIControlStateDisabled];
  122. }
  123. - (void)disconnect {
  124. // [RaylinkProxy.sharedManager closeConnection:self.sdnIdTF.text];
  125. // [RaylinkProxy.sharedManager closeHttpService:self.sdnIdTF.text];
  126. // [self.connectBtn setTitle:@"连接" forState:UIControlStateNormal];
  127. self.peerConnected = NO;
  128. }
  129. - (IBAction)onClickSend:(id)sender {
  130. // if (self.messageTF.text.length <= 0) {
  131. // return;
  132. // }
  133. [self.socket writeData:[@"111" dataUsingEncoding:NSUTF8StringEncoding] withTimeout:-1 tag:0];
  134. }
  135. - (void)socket:(GCDAsyncSocket *)sock didAcceptNewSocket:(GCDAsyncSocket *)newSocket {
  136. NSLog(@"didAcceptNewSocket");
  137. [newSocket setDelegate:self delegateQueue:dispatch_get_main_queue()];
  138. NSThread *revcThread = [[NSThread alloc] initWithBlock:^{
  139. while (true) {
  140. [newSocket readDataWithTimeout:-1 tag:0];
  141. [NSThread sleepForTimeInterval:0.1];
  142. }
  143. }];
  144. [revcThread start];
  145. }
  146. - (void)onProxyConnected:(NSString *)sdnId status:(int)status {
  147. if (status == 0) {
  148. /// 6. 等待连接成功回调,创建 TCP socket 连接 HttpService 的端口
  149. [self.socket connectToHost:@"127.0.0.1" onPort:self.tcpPort error:NULL];
  150. _curConnectDeviceState = DeviceConnectDeciceOk;
  151. // [self.connectBtn setTitle:@"连接成功" forState:UIControlStateDisabled];
  152. // [self.sendBtn setEnabled:YES];
  153. } else {
  154. // [self.connectBtn setTitle:@"连接中..." forState:UIControlStateDisabled];
  155. // [self.sendBtn setEnabled:NO];
  156. }
  157. }
  158. - (void)socket:(GCDAsyncSocket *)sock didConnectToHost:(NSString *)host port:(uint16_t)port {
  159. NSLog(@"didConnectToHost %d", port);
  160. NSThread *thread = [[NSThread alloc] initWithBlock:^{
  161. while (true) {
  162. [self.socket readDataWithTimeout:-1 tag:0];
  163. [NSThread sleepForTimeInterval:0.1];
  164. }
  165. }];
  166. [thread start];
  167. /// 7. Socket 连接成功后,通过 socket 发送, self.peerPortTF 为对端服务端口
  168. [sock writeData:[[NSString stringWithFormat:@"CONNECT 127.0.0.1:%@ HTTP/1.0\r\n\r\n", @"9100"] dataUsingEncoding:NSUTF8StringEncoding] withTimeout:-1 tag:0];
  169. NSLog(@"readDataToData");
  170. }
  171. - (void)socket:(GCDAsyncSocket *)sock didWriteDataWithTag:(long)tag
  172. {
  173. NSLog(@"socket:%p didWriteDataWithTag:%ld", sock, tag);
  174. }
  175. - (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag {
  176. if (tag == 0) {
  177. NSString *str = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
  178. /// 8. 收到包含 `HTTP/1.0 200`的字符串,则认为连接成功。可以正常给对端发消息了
  179. if ([str containsString:@"HTTP/1.0 200"]) {
  180. NSLog(@"Connected");
  181. self.peerConnected = YES;
  182. // self.connectBtn.enabled = YES;
  183. // [self.connectBtn setTitle:@"断开连接" forState:UIControlStateNormal];
  184. }
  185. }
  186. //self.revcLab.text = [NSString stringWithFormat:@"接收:%@", [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]];
  187. }
  188. #pragma mark 保持卡密码
  189. -(void)updateCardInfoBySN:(NSString*)snStr withPwdStr:(NSString*)pwd
  190. {
  191. NSMutableDictionary *paraDict = [NSMutableDictionary new];
  192. [paraDict setValue:snStr forKey:@"sn"];
  193. [paraDict setValue:pwd forKey:@"password"];
  194. KWeakSelf
  195. [[netWorkManager shareInstance] CommonGetWithCallBackCode:updateCardInfo Parameters:paraDict success:^(id _Nonnull responseObject) {
  196. SuperModel*model = [[SuperModel alloc] initWithDictionary:responseObject error:nil];
  197. if(model.status == 0)
  198. {
  199. }
  200. } failure:^(NSError * _Nonnull error) {
  201. HLog("网络报错");
  202. }];
  203. }
  204. #pragma mark ---ping-----站点 判断是否是局域网
  205. - (void)startPingDeviceIp:(NSString *)roomInternetIp
  206. {
  207. //roomInternetIp = @"www.baidu.com";
  208. if(!roomInternetIp || roomInternetIp.length < 7){
  209. self.DeviceThirdIdMod.data.isPingOk = NO;
  210. return;
  211. }
  212. if (self->whPingTester)
  213. {
  214. [self->whPingTester stopPing];
  215. self->whPingTester = nil;
  216. }
  217. self->whPingTester = [[WHPingTester alloc] initWithHostName:roomInternetIp];
  218. self->whPingTester.delegate = self;
  219. //self->whPingTester.countdownTime = 1; //1秒ping一次
  220. [self->whPingTester startPing];
  221. /// 启动定时器监听 10秒后 whPingTester的状态
  222. [NSTimer scheduledTimerWithTimeInterval:5 repeats:NO block:^(NSTimer * _Nonnull timer) {
  223. if(self->whPingTester){
  224. [self didGetPingStateIsOK:NO];
  225. }
  226. }];
  227. }
  228. #pragma mark ---ping---- 代理回调
  229. - (void) didPingSucccessWithTime:(float)time withError:(NSError*) error
  230. {
  231. HLog(@"time:%f \n error: %@",time,error)
  232. //if(error.code == 111 || time > 1000) //超时或者延时超过100ms,就要记录数据
  233. if(error != nil)
  234. {
  235. [self didGetPingStateIsOK:NO];
  236. }
  237. else if(time > 0){
  238. [self didGetPingStateIsOK:YES];
  239. }
  240. }
  241. - (void) didPingfialewithError:(NSError*)error
  242. {
  243. HLog(@" error error: %@",error)
  244. [self didGetPingStateIsOK:NO];
  245. }
  246. -(void)didGetPingStateIsOK:(BOOL)isOk
  247. {
  248. self.DeviceThirdIdMod.data.isPingOk = isOk;
  249. [self->whPingTester stopPing];
  250. self->whPingTester = nil;
  251. }
  252. @end