connectDeviceManager.m 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  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. #import "RSATool.h"
  13. @interface connectDeviceManager ()<
  14. GCDAsyncSocketDelegate,
  15. RaylinkProxyDelegate,
  16. WHPingDelegate
  17. >
  18. {
  19. dispatch_queue_t initProxyQueue;
  20. WHPingTester *whPingTester;
  21. }
  22. @property (nonatomic, strong) NSTimer *connectCheckTimer;
  23. @property (nonatomic, strong) GCDAsyncSocket *socket;
  24. @property (nonatomic, strong) GCDAsyncSocket *serverSocket;
  25. @property (nonatomic, assign) BOOL sdnConnected;
  26. @property (nonatomic, assign) BOOL peerConnected;
  27. @property (nonatomic, assign) NSInteger tcpPort;
  28. @property (nonatomic, copy) NSString *Pre_sdnId;//上一个sdn好
  29. @property (nonatomic, assign) BOOL isGetThridMsgType;
  30. @end
  31. @implementation connectDeviceManager
  32. static connectDeviceManager *connectDeviceManagerInstance = nil;
  33. +(connectDeviceManager *)shareInstance;
  34. {
  35. static dispatch_once_t onceToken;
  36. dispatch_once(&onceToken, ^{
  37. connectDeviceManagerInstance = [[connectDeviceManager alloc] init];
  38. connectDeviceManagerInstance.curConnectDeviceState = DeviceConnectUnknown;
  39. });
  40. return connectDeviceManagerInstance;
  41. }
  42. #pragma mark 根据扫码的sn获取第三方信息
  43. -(void)getThridMsgBySN:(NSString*)snStr needReconnect:(BOOL)needReconnect didNetEnd:(netWork_DidEndByOK)didNetEndIsOK
  44. {
  45. if(_isGetThridMsgType){
  46. didNetEndIsOK(-1);
  47. return;
  48. }
  49. if(needReconnect){
  50. _curConnectDeviceState = DeviceConnectGetThridMsging;
  51. //更换设备
  52. // NSString *curSdnId = self.DeviceThirdIdMod.data.sdnId;
  53. // if(curSdnId){
  54. // _Pre_sdnId = curSdnId;
  55. // }
  56. }
  57. NSMutableDictionary *paraDict = [NSMutableDictionary new];
  58. [paraDict setValue:snStr forKey:@"sn"];
  59. _isGetThridMsgType = YES;
  60. KWeakSelf
  61. [[netWorkManager shareInstance] CommonGetWithCallBackCode:getThirdIdBySn Parameters:paraDict success:^(id _Nonnull responseObject) {
  62. self->_isGetThridMsgType = NO;
  63. DeviceThirdIdModel *model = [[DeviceThirdIdModel alloc] initWithDictionary:responseObject error:nil];
  64. if(model.status == 0
  65. && model.data)
  66. {
  67. weakSelf.DeviceThirdIdMod = model;
  68. NSString*desPwdstr = [RSATool AES128Decrypt:weakSelf.DeviceThirdIdMod.data.password key:AESCODEKEEYY];
  69. didNetEndIsOK(YES);
  70. /*密码保持在本地*/
  71. NSDictionary *deviceDict = [HWDataManager getObjectWithKey:Const_Have_Add_Device];
  72. NSMutableDictionary *dict = [[NSMutableDictionary alloc] initWithDictionary:deviceDict];
  73. [dict setObject:desPwdstr forKey:Const_Have_Add_Device_PWD];
  74. NSString *sdnid = model.data.sdnId;
  75. if(sdnid){
  76. [dict setObject:sdnid forKey:Const_Have_Add_Device_sdnid];
  77. }
  78. [HWDataManager setObjectWithKey:Const_Have_Add_Device value:dict];
  79. //
  80. bool connected = [RaylinkProxy.sharedManager isSdnConnected];
  81. if(!connected || needReconnect){
  82. weakSelf.curConnectDeviceState = DeviceConnectGetThridOK;
  83. // NSString *ipStr = weakSelf.DeviceThirdIdMod.data.ip;
  84. // [weakSelf startPingDeviceIp:ipStr];
  85. [weakSelf initRuiyunSDKFun];
  86. }
  87. }
  88. else{
  89. didNetEndIsOK(NO);
  90. }
  91. } failure:^(NSError * _Nonnull error) {
  92. HLog("网络报错");
  93. self->_isGetThridMsgType = NO;
  94. didNetEndIsOK(NO);
  95. }];
  96. }
  97. #pragma mark 刷新扫码的sn获取第三方信息 是否需要重连
  98. -(void)RefreshThridMsg
  99. {
  100. /*先判断本地有无设备 无设备时需要先扫码添加设备*/
  101. NSDictionary *deviceDict = [HWDataManager getObjectWithKey:Const_Have_Add_Device];
  102. if (deviceDict && [[deviceDict allKeys] containsObject:Const_Have_Add_Device_SN]){
  103. //有设备了先去做链接准备 // 80bec9c5
  104. NSString *SNStr = deviceDict[@"Const_Have_Add_Device_SN"];
  105. if(SNStr){
  106. [[connectDeviceManager shareInstance] getThridMsgBySN:SNStr needReconnect:NO didNetEnd:^(NSInteger didSuc) {
  107. }];
  108. }
  109. }
  110. }
  111. #pragma mark 初始化瑞云的SDK等
  112. - (void)initRuiyunSDKFun{
  113. bool connected = [RaylinkProxy.sharedManager isSdnConnected];
  114. if(connected){
  115. [self onConnectFun];
  116. return;
  117. }
  118. HLog(@"initRuiyunSDKFun");
  119. self.sdnConnected = NO;
  120. self.peerConnected = NO;
  121. //self.sendBtn.enabled = NO;
  122. //self.sdnConnectStateLab.text = @"SDN state: Connectting";
  123. //self.sdnIDLab.text = @"My SDN ID:";
  124. RaylinkProxy.sharedManager.delegate = self;
  125. initProxyQueue = dispatch_queue_create("init_proxy", 0);
  126. NSURL *logUrl = [[NSFileManager.defaultManager URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask].firstObject URLByAppendingPathComponent:@"logs"];
  127. if (![NSFileManager.defaultManager fileExistsAtPath:logUrl.path]) {
  128. [NSFileManager.defaultManager createDirectoryAtURL:logUrl withIntermediateDirectories:YES attributes:NULL error:NULL];
  129. }
  130. dispatch_async(initProxyQueue, ^{
  131. NSData *sdnInfo = [NSData dataWithContentsOfURL:[[NSBundle.mainBundle bundleURL] URLByAppendingPathComponent:@"planet.1ali_3ry_peer"]];
  132. /// 1. 初始化代理库
  133. [RaylinkProxy.sharedManager initProxy:logUrl.path rootSdnInfo:sdnInfo];
  134. });
  135. /// 作为服务端时,启动端口监听
  136. self.serverSocket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
  137. if ([self.serverSocket acceptOnPort:0 error:NULL]) {
  138. //self.serverPortLab.text = [NSString stringWithFormat: @"server port: %d", self.serverSocket.localPort];
  139. NSLog(@"accept ok %d",self.serverSocket.localPort);
  140. }
  141. KWeakSelf
  142. /// 启动定时器监听 SND 连接状态
  143. self.connectCheckTimer = [NSTimer scheduledTimerWithTimeInterval:0.5 repeats:YES block:^(NSTimer * _Nonnull timer) {
  144. /// 2. 监听SDN 连接状态
  145. bool connected = [RaylinkProxy.sharedManager isSdnConnected];
  146. if (self.sdnConnected == connected) {
  147. return;
  148. }
  149. self.sdnConnected = connected;
  150. if (self.sdnConnected) {
  151. [weakSelf onConnectFun];
  152. [self->_connectCheckTimer invalidate];
  153. //self.sdnConnectStateLab.text = @"SDN state: Connected";
  154. //self.sdnIDLab.text = [@"My SDN ID: " stringByAppendingString:[RaylinkProxy.sharedManager getSdnId]];
  155. } else {
  156. //self.sdnConnectStateLab.text = @"SDN state: Connectting";
  157. }
  158. }];
  159. // dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(2.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  160. // mainBlock(^{
  161. // [self onConnectFun];
  162. // });
  163. // });
  164. }
  165. - (void)onConnectFun{
  166. // 2. 监听SDN 连接状态
  167. // bool connected = [RaylinkProxy.sharedManager isSdnConnected];
  168. //
  169. // HLog(@"协助瑞云看日志onConnectFun isSdnConnected:%d",connected);
  170. //
  171. // if(!connected){
  172. // dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(1.0 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  173. // mainBlock(^{
  174. // [self onConnectFun];
  175. // });
  176. // });
  177. //
  178. // return;
  179. // }
  180. if (self.peerConnected == YES) {
  181. //[self disconnect];
  182. //return;
  183. }
  184. _curConnectDeviceState = DeviceConnectDeciceing;
  185. NSString *curSdnId = self.DeviceThirdIdMod.data.sdnId; //@"3dfe7c1f";
  186. if(!curSdnId){
  187. NSDictionary *deviceDict = [HWDataManager getObjectWithKey:Const_Have_Add_Device];
  188. if(deviceDict && [[deviceDict allKeys] containsObject:Const_Have_Add_Device_sdnid])
  189. {
  190. curSdnId = deviceDict[Const_Have_Add_Device_sdnid];
  191. }
  192. }
  193. /// 3. 根据对端的 SDN ID 创建连接
  194. [RaylinkProxy.sharedManager createNewConnection:curSdnId];
  195. /// 4. 创建 HttpService
  196. self.tcpPort = [RaylinkProxy.sharedManager createHttpService:curSdnId];
  197. self.tcpPortStr = [[NSString alloc] initWithFormat:@"%ld",self.tcpPort];
  198. self.socket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
  199. /// 6. 添加端口防火墙
  200. [RaylinkProxy.sharedManager addSdnId:curSdnId allowPort:9300];
  201. [RaylinkProxy.sharedManager addSdnId:curSdnId allowPort:9100];
  202. //[self.connectBtn setEnabled:NO];
  203. //[self.connectBtn setTitle:@"连接中..." forState:UIControlStateDisabled];
  204. }
  205. - (void)disconnect {
  206. NSString *curSdnId = self.DeviceThirdIdMod.data.sdnId;
  207. if(_Pre_sdnId){
  208. curSdnId = _Pre_sdnId;
  209. }
  210. if(curSdnId){
  211. [RaylinkProxy.sharedManager closeConnection:curSdnId];
  212. [RaylinkProxy.sharedManager closeHttpService:curSdnId];
  213. //[self.connectBtn setTitle:@"连接" forState:UIControlStateNormal];
  214. self.peerConnected = NO;
  215. }
  216. }
  217. - (IBAction)onClickSend:(id)sender {
  218. // if (self.messageTF.text.length <= 0) {
  219. // return;
  220. // }
  221. [self.socket writeData:[@"111" dataUsingEncoding:NSUTF8StringEncoding] withTimeout:-1 tag:0];
  222. }
  223. - (void)socket:(GCDAsyncSocket *)sock didAcceptNewSocket:(GCDAsyncSocket *)newSocket {
  224. NSLog(@"didAcceptNewSocket");
  225. [newSocket setDelegate:self delegateQueue:dispatch_get_main_queue()];
  226. NSThread *revcThread = [[NSThread alloc] initWithBlock:^{
  227. while (true) {
  228. [newSocket readDataWithTimeout:-1 tag:0];
  229. [NSThread sleepForTimeInterval:0.1];
  230. }
  231. }];
  232. [revcThread start];
  233. }
  234. - (void)onProxyConnected:(NSString *)sdnId status:(int)status {
  235. if (status == 0) {
  236. /// 6. 等待连接成功回调,创建 TCP socket 连接 HttpService 的端口
  237. [self.socket connectToHost:@"127.0.0.1" onPort:self.tcpPort error:NULL];
  238. _curConnectDeviceState = DeviceConnectDeciceOk;
  239. _Pre_sdnId = self.DeviceThirdIdMod.data.sdnId;
  240. // [self.connectBtn setTitle:@"连接成功" forState:UIControlStateDisabled];
  241. // [self.sendBtn setEnabled:YES];
  242. } else {
  243. // [self.connectBtn setTitle:@"连接中..." forState:UIControlStateDisabled];
  244. // [self.sendBtn setEnabled:NO];
  245. HLog(@"瑞云链接断开");
  246. }
  247. }
  248. - (void)socket:(GCDAsyncSocket *)sock didConnectToHost:(NSString *)host port:(uint16_t)port {
  249. NSLog(@"didConnectToHost %d", port);
  250. NSThread *thread = [[NSThread alloc] initWithBlock:^{
  251. while (true) {
  252. [self.socket readDataWithTimeout:-1 tag:0];
  253. [NSThread sleepForTimeInterval:0.1];
  254. }
  255. }];
  256. [thread start];
  257. /// 7. Socket 连接成功后,通过 socket 发送, self.peerPortTF 为对端服务端口
  258. [sock writeData:[[NSString stringWithFormat:@"CONNECT 127.0.0.1:%@ HTTP/1.0\r\n\r\n", @"9100"] dataUsingEncoding:NSUTF8StringEncoding] withTimeout:-1 tag:0];
  259. NSLog(@"readDataToData");
  260. }
  261. - (void)socket:(GCDAsyncSocket *)sock didWriteDataWithTag:(long)tag
  262. {
  263. NSLog(@"socket:%p didWriteDataWithTag:%ld", sock, tag);
  264. }
  265. - (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag {
  266. if (tag == 0) {
  267. NSString *str = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
  268. /// 8. 收到包含 `HTTP/1.0 200`的字符串,则认为连接成功。可以正常给对端发消息了
  269. if ([str containsString:@"HTTP/1.0 200"]) {
  270. NSLog(@"Connected");
  271. self.peerConnected = YES;
  272. // self.connectBtn.enabled = YES;
  273. // [self.connectBtn setTitle:@"断开连接" forState:UIControlStateNormal];
  274. }
  275. }
  276. //self.revcLab.text = [NSString stringWithFormat:@"接收:%@", [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]];
  277. }
  278. #pragma mark 保持卡密码
  279. -(void)updateCardInfoBySN:(NSString*)snStr withPwdStr:(NSString*)pwd didNetEnd:(netWork_DidEndByOK)didNetEndIsOK
  280. {
  281. NSMutableDictionary *paraDict = [NSMutableDictionary new];
  282. [paraDict setValue:snStr forKey:@"sn"];
  283. NSString* enPwdStr = [RSATool AES128Encrypt:pwd key:AESCODEKEEYY];
  284. if(enPwdStr){
  285. [paraDict setValue:enPwdStr forKey:@"password"];
  286. }
  287. KWeakSelf
  288. [[netWorkManager shareInstance] CommonPostCallBackCode:updateCardInfo Parameters:paraDict success:^(id _Nonnull responseObject) {
  289. SuperModel*model = [[SuperModel alloc] initWithDictionary:responseObject error:nil];
  290. if(model && model.status == 0)
  291. {
  292. didNetEndIsOK(YES);
  293. [connectDeviceManager shareInstance].DeviceThirdIdMod.data.password = enPwdStr;
  294. /*密码保持在本地*/
  295. NSDictionary *deviceDict = [HWDataManager getObjectWithKey:Const_Have_Add_Device];
  296. NSMutableDictionary *dict = [[NSMutableDictionary alloc] initWithDictionary:deviceDict];
  297. [dict setObject:pwd forKey:Const_Have_Add_Device_PWD];
  298. [HWDataManager setObjectWithKey:Const_Have_Add_Device value:dict];
  299. }
  300. else{
  301. didNetEndIsOK(NO);
  302. }
  303. } failure:^(NSError * _Nonnull error) {
  304. HLog("网络报错");
  305. didNetEndIsOK(NO);
  306. }];
  307. }
  308. #pragma mark ---ping-----站点 判断是否是局域网
  309. - (void)startPingDeviceIp:(NSString *)roomInternetIp
  310. {
  311. //roomInternetIp = @"www.baidu.com";
  312. if(!roomInternetIp || roomInternetIp.length < 7){
  313. self.DeviceThirdIdMod.data.isPingOk = NO;
  314. return;
  315. }
  316. if (self->whPingTester)
  317. {
  318. [self->whPingTester stopPing];
  319. self->whPingTester = nil;
  320. }
  321. self->whPingTester = [[WHPingTester alloc] initWithHostName:roomInternetIp];
  322. self->whPingTester.delegate = self;
  323. //self->whPingTester.countdownTime = 1; //1秒ping一次
  324. [self->whPingTester startPing];
  325. /// 启动定时器监听 10秒后 whPingTester的状态
  326. [NSTimer scheduledTimerWithTimeInterval:5 repeats:NO block:^(NSTimer * _Nonnull timer) {
  327. if(self->whPingTester){
  328. [self didGetPingStateIsOK:NO];
  329. }
  330. }];
  331. }
  332. #pragma mark ---ping---- 代理回调
  333. - (void) didPingSucccessWithTime:(float)time withError:(NSError*) error
  334. {
  335. HLog(@"time:%f \n error: %@",time,error)
  336. //if(error.code == 111 || time > 1000) //超时或者延时超过100ms,就要记录数据
  337. if(error != nil)
  338. {
  339. [self didGetPingStateIsOK:NO];
  340. }
  341. else if(time > 0){
  342. [self didGetPingStateIsOK:YES];
  343. }
  344. }
  345. - (void) didPingfialewithError:(NSError*)error
  346. {
  347. HLog(@" error error: %@",error)
  348. [self didGetPingStateIsOK:NO];
  349. }
  350. -(void)didGetPingStateIsOK:(BOOL)isOk
  351. {
  352. self.DeviceThirdIdMod.data.isPingOk = isOk;
  353. [self->whPingTester stopPing];
  354. self->whPingTester = nil;
  355. }
  356. @end