NSNetService публикует, но didAcceptConnectionWithInputStream никогда не вызывается

Я пытался сделать простое приложение на iOS, которое будет получать сообщение, а затем выполнять действие на основе этого сообщения (на данный момент я просто хочу показать его через NSLog). Я могу подключиться к службе, но служба, похоже, ничего не получает. Вот мой файл .h:

#import <Foundation/Foundation.h>
#import <arpa/inet.h>

@interface PalServiceController : NSObject <NSNetServiceDelegate>

@property (nonatomic, strong) NSNetService *ns;
@property (nonatomic, strong) NSOutputStream *ostream;

- (void)start;
+ (int)getPort;
- (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode;

@end

и мой .м:

#import "PalServiceController.h"

@implementation PalServiceController

- (void)start
{
    // Start a net service
    int port = [PalServiceController getPort];
    NSLog(@"Opening on port %i", port);
    self.ns = [[NSNetService alloc] initWithDomain:@""
                                              type:@"_TestingProtocol._tcp."
                                              name:@"Test Name For iPhone"
                                              port:port];
    if (self.ns) {
        [self.ns setDelegate:self];
        [self.ns publish];
        self.ns.delegate = self;
    } else {
        NSLog(@"Error starting service");
    }

}

/*
 * Code from: http://stackoverflow.com/a/11723158/657676
 */
+ (int) getPort
{
 ...   
}

- (void)netServiceWillResolve:(NSNetService *)sender
{
    NSLog(@"Resolving");
}

- (void)netServiceDidResolveAddress:(NSNetService *)sender
{
    NSLog(@"Resolved Address");
}

- (void)netService:(NSNetService *)sender didNotPublish:(NSDictionary *)errorDict
{
    NSLog(@"Error publishing");
}

- (void)netService:(NSNetService *)sender didAcceptConnectionWithInputStream:(NSInputStream *)inputStream outputStream:(NSOutputStream *)outputStream
{
    NSLog(@"Got a connection! (server)");
    // Close self down
    [self.ns stopMonitoring];
    [self.ns stop];
}

- (void)netService:(NSNetService *)sender didNotResolve:(NSDictionary *)errorDict
{
    NSLog(@"Error resolving");
}
- (void)netServiceDidPublish:(NSNetService *)sender
{
    NSLog(@"Published server");
}

- (void)netServiceDidStop:(NSNetService *)sender
{
    NSLog(@"Server stopped");
}

- (void)netService:(NSNetService *)sender didUpdateTXTRecordData:(NSData *)data
{
    NSLog(@"Updated TXT Record");
}

- (void)stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode
{
    NSLog(@"Event on server");
    switch(eventCode) {
        ...
    }
}

@end

Это вызывается через:

self.controller = [[PalServiceController alloc] init];
[self.controller start];

Когда я использую Bonjour Browser, я вижу службу, а когда использую собственную реализацию NSNetServiceBrowser или пример от Билла Дадни (сообщение в блоге, code), кажется, он подключается (т.е. моя собственная реализация получает NSStreamEventHasSpaceAvailable и NSStreamEventOpenCompletedevents через stream:(NSStream *)stream handleEvent:(NSStreamEvent)eventCode). Однако ни один из вышеперечисленных методов не вызывается на сервере.

Я все еще новичок в iOS, поэтому я надеюсь, что это просто какая-то глупая ошибка, которую можно легко исправить.


person Joseph Duffy    schedule 01.02.2014    source источник


Ответы (1)


-[NSNetService publish] просто публикует сервис; вы отвечаете за создание сокета и прослушивание порта, который вы предоставляете инициализатору NSNetService. Вы не получите netService:didAcceptConnectionWithInputStream:outputStream:, потому что NSNetService ничего не знает о сокете.

Если вы хотите, чтобы NSNetService управлял сокетом для вас и отправлял didAcceptConnection...use -[NSNetService publishWithOptions:]:

[self.ns publishWithOptions:NSNetServiceListenForConnections];

В этом случае вам нужно убедиться, что порт, с которым вы инициализируете NSNetService, не используется. Ваш метод getPort никогда не закрывает сокет (и приводит к утечке CFSocketRef), поэтому он будет использоваться при публикации службы, что приведет к ошибке.

Я бы порекомендовал вам удалить getPort и, как предлагает Apple в NSNetServices.h, «[s] указать нулевой номер порта для использования случайного порта».

person jonahb    schedule 02.02.2014
comment
Привет, спасибо за ответ! Кажется, что это делает 2 интересных вещи: 1. Он вызывает метод (void)netService:(NSNetService *)sender didNotPublish:(NSDictionary *)errorDict, а затем (void)netServiceDidPublish:(NSNetService *)sender (который также вызывается дважды при простом использовании [self.ns publish]. 2. Похоже, что он не прослушивает домен local. и явно устанавливает домен local. делает то же самое, но только с 1 вызовом netServiceDidPublish и, похоже, не публикуется.Есть идеи, что может быть причиной этого сейчас? - person Joseph Duffy; 02.02.2014
comment
Да это правильно! Я помню, как читал это где-то, а потом подумал, что не могу сделать это для NSNetService, так что спасибо, что указали на это. Теперь вроде все работает, большое спасибо - person Joseph Duffy; 02.02.2014
comment
@JosephDuffy, рад помочь. - person jonahb; 03.02.2014
comment
Это сработало для меня. К сожалению, я прочитал этот пост после того, как решил эту проблему сам. - person Sergey Stadnik; 08.09.2016