ConnectTestViewController.m 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238
  1. //
  2. // ConnectTestViewController.m
  3. // RayProxyDemo
  4. //
  5. // Created by Sim Tsai on 2023/8/28.
  6. //
  7. #import "ConnectTestViewController.h"
  8. #import <RayProxy/RaylinkProxy.h>
  9. #import <CocoaAsyncSocket/GCDAsyncSocket.h>
  10. @interface ConnectTestViewController ()<
  11. GCDAsyncSocketDelegate,
  12. RaylinkProxyDelegate
  13. > {
  14. dispatch_queue_t initProxyQueue;
  15. }
  16. @property (weak, nonatomic) IBOutlet UILabel *sdnConnectStateLab;
  17. @property (weak, nonatomic) IBOutlet UILabel *sdnIDLab;
  18. @property (weak, nonatomic) IBOutlet UITextField *sdnIdTF;
  19. @property (weak, nonatomic) IBOutlet UIButton *connectBtn;
  20. @property (weak, nonatomic) IBOutlet UILabel *serverPortLab;
  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. @property (weak, nonatomic) IBOutlet UIButton *sendBtn;
  28. @property (weak, nonatomic) IBOutlet UILabel *revcLab;
  29. @property (weak, nonatomic) IBOutlet UITextField *peerPortTF;
  30. @property (weak, nonatomic) IBOutlet UITextField *messageTF;
  31. @end
  32. @implementation ConnectTestViewController
  33. - (void)viewDidLoad {
  34. [super viewDidLoad];
  35. self.sdnConnected = NO;
  36. self.peerConnected = NO;
  37. self.sendBtn.enabled = NO;
  38. self.sdnConnectStateLab.text = @"SDN state: Connectting";
  39. self.sdnIDLab.text = @"My SDN ID:";
  40. RaylinkProxy.sharedManager.delegate = self;
  41. initProxyQueue = dispatch_queue_create("init_proxy", 0);
  42. NSURL *logUrl = [[NSFileManager.defaultManager URLsForDirectory:NSDocumentDirectory inDomains:NSUserDomainMask].firstObject URLByAppendingPathComponent:@"logs"];
  43. if (![NSFileManager.defaultManager fileExistsAtPath:logUrl.path]) {
  44. [NSFileManager.defaultManager createDirectoryAtURL:logUrl withIntermediateDirectories:YES attributes:NULL error:NULL];
  45. }
  46. dispatch_async(initProxyQueue, ^{
  47. NSData *sdnInfo = [NSData dataWithContentsOfURL:[[NSBundle.mainBundle bundleURL] URLByAppendingPathComponent:@"planet.1ali_3ry_peer"]];
  48. /// 1. 初始化代理库
  49. [RaylinkProxy.sharedManager initProxy:logUrl.path rootSdnInfo:sdnInfo];
  50. });
  51. /// 作为服务端时,启动端口监听
  52. self.serverSocket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
  53. if ([self.serverSocket acceptOnPort:0 error:NULL]) {
  54. self.serverPortLab.text = [NSString stringWithFormat: @"server port: %d", self.serverSocket.localPort];
  55. NSLog(@"accept ok");
  56. }
  57. /// 启动定时器监听 SND 连接状态
  58. self.connectCheckTimer = [NSTimer scheduledTimerWithTimeInterval:0.5 repeats:YES block:^(NSTimer * _Nonnull timer) {
  59. /// 2. 监听SDN 连接状态
  60. bool connected = [RaylinkProxy.sharedManager isSdnConnected];
  61. if (self.sdnConnected == connected) {
  62. return;
  63. }
  64. self.sdnConnected = connected;
  65. if (self.sdnConnected) {
  66. self.sdnConnectStateLab.text = @"SDN state: Connected";
  67. self.sdnIDLab.text = [@"My SDN ID: " stringByAppendingString:[RaylinkProxy.sharedManager getSdnId]];
  68. } else {
  69. self.sdnConnectStateLab.text = @"SDN state: Connectting";
  70. }
  71. }];
  72. }
  73. - (IBAction)onConnect:(id)sender {
  74. if (self.peerConnected == YES) {
  75. [self disconnect];
  76. return;
  77. }
  78. if (self.sdnIdTF.text.length <= 0) {
  79. return;
  80. }
  81. if (self.peerPortTF.text.length <= 0) {
  82. return;
  83. }
  84. /// 3. 根据对端的 SDN ID 创建连接
  85. [RaylinkProxy.sharedManager createNewConnection:self.sdnIdTF.text];
  86. /// 4. 创建 HttpService
  87. self.tcpPort = [RaylinkProxy.sharedManager createHttpService:self.sdnIdTF.text];
  88. self.socket = [[GCDAsyncSocket alloc] initWithDelegate:self delegateQueue:dispatch_get_main_queue()];
  89. /// 6. 添加端口防火墙
  90. [RaylinkProxy.sharedManager addSdnId:self.sdnIdTF.text allowPort:[self.peerPortTF.text intValue]];
  91. [self.connectBtn setEnabled:NO];
  92. [self.connectBtn setTitle:@"连接中..." forState:UIControlStateDisabled];
  93. }
  94. - (void)disconnect {
  95. [RaylinkProxy.sharedManager closeConnection:self.sdnIdTF.text];
  96. [RaylinkProxy.sharedManager closeHttpService:self.sdnIdTF.text];
  97. [self.connectBtn setTitle:@"连接" forState:UIControlStateNormal];
  98. self.peerConnected = NO;
  99. }
  100. - (IBAction)onClickSend:(id)sender {
  101. if (self.messageTF.text.length <= 0) {
  102. return;
  103. }
  104. [self.socket writeData:[self.messageTF.text dataUsingEncoding:NSUTF8StringEncoding] withTimeout:-1 tag:0];
  105. }
  106. - (void)socket:(GCDAsyncSocket *)sock didAcceptNewSocket:(GCDAsyncSocket *)newSocket {
  107. NSLog(@"didAcceptNewSocket");
  108. [newSocket setDelegate:self delegateQueue:dispatch_get_main_queue()];
  109. NSThread *revcThread = [[NSThread alloc] initWithBlock:^{
  110. while (true) {
  111. [newSocket readDataWithTimeout:-1 tag:0];
  112. [NSThread sleepForTimeInterval:0.1];
  113. }
  114. }];
  115. [revcThread start];
  116. }
  117. - (void)onProxyConnected:(NSString *)sdnId status:(int)status {
  118. if (status == 0) {
  119. /// 6. 等待连接成功回调,创建 TCP socket 连接 HttpService 的端口
  120. [self.socket connectToHost:@"127.0.0.1" onPort:self.tcpPort error:NULL];
  121. [self.connectBtn setTitle:@"连接成功" forState:UIControlStateDisabled];
  122. [self.sendBtn setEnabled:YES];
  123. } else {
  124. [self.connectBtn setTitle:@"连接中..." forState:UIControlStateDisabled];
  125. [self.sendBtn setEnabled:NO];
  126. }
  127. }
  128. - (void)socket:(GCDAsyncSocket *)sock didConnectToHost:(NSString *)host port:(uint16_t)port {
  129. NSLog(@"didConnectToHost %d", port);
  130. NSThread *thread = [[NSThread alloc] initWithBlock:^{
  131. while (true) {
  132. [self.socket readDataWithTimeout:-1 tag:0];
  133. [NSThread sleepForTimeInterval:0.1];
  134. }
  135. }];
  136. [thread start];
  137. /// 7. Socket 连接成功后,通过 socket 发送, self.peerPortTF 为对端服务端口
  138. [sock writeData:[[NSString stringWithFormat:@"CONNECT 127.0.0.1:%@ HTTP/1.0\r\n\r\n", self.peerPortTF.text] dataUsingEncoding:NSUTF8StringEncoding] withTimeout:-1 tag:0];
  139. NSLog(@"readDataToData");
  140. }
  141. - (void)socket:(GCDAsyncSocket *)sock didWriteDataWithTag:(long)tag
  142. {
  143. NSLog(@"socket:%p didWriteDataWithTag:%ld", sock, tag);
  144. }
  145. - (void)socket:(GCDAsyncSocket *)sock didReadData:(NSData *)data withTag:(long)tag {
  146. if (tag == 0) {
  147. NSString *str = [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding];
  148. /// 8. 收到包含 `HTTP/1.0 200`的字符串,则认为连接成功。可以正常给对端发消息了
  149. if ([str containsString:@"HTTP/1.0 200"]) {
  150. NSLog(@"Connected");
  151. self.peerConnected = YES;
  152. self.connectBtn.enabled = YES;
  153. [self.connectBtn setTitle:@"断开连接" forState:UIControlStateNormal];
  154. }
  155. }
  156. self.revcLab.text = [NSString stringWithFormat:@"接收:%@", [[NSString alloc] initWithData:data encoding:NSUTF8StringEncoding]];
  157. }
  158. @end