停止Avahi服务并返回元素列表

1 投票
2 回答
942 浏览
提问于 2025-04-16 12:34

根据这个例子,我想在获取到所有元素后停止服务,并把它们放在一个列表里。

要停止这个无限循环,可以调用gobject.MainLoop().quit(),但我不知道应该在什么时候或什么地方调用它。

2 个回答

1

你还需要关注一下 AllForNow 这个信号。同时也要考虑 CacheExhaustedFailure 这两个情况。关于 org.freedesktop.Avahi.ServiceBrowser 的接口说明可以在这里找到。

在我的电脑上运行以下代码,大约一秒钟就能返回结果,并打印出:

browsing ...
cache exhausted
New service: name = Lexmark X264dn; address = 10.9.0.93; port = 80
that's all for now
done
0:00:01.004258

希望这对你有帮助!


from _dbus_glib_bindings import DBusGMainLoop
import dbus, avahi
import gobject

class ServiceBrowser():
    def __init__(self, service):
        loop = DBusGMainLoop(set_as_default=True)

        bus = dbus.SystemBus(mainloop=loop)

        self._server = dbus.Interface(bus.get_object(avahi.DBUS_NAME, avahi.DBUS_PATH_SERVER), avahi.DBUS_INTERFACE_SERVER)

        browser = self._server.ServiceBrowserNew(avahi.IF_UNSPEC, avahi.PROTO_INET, service, '', 0)
        listener = dbus.Interface(bus.get_object(avahi.DBUS_NAME, browser), avahi.DBUS_INTERFACE_SERVICE_BROWSER)

        listener.connect_to_signal("ItemNew", self.item_new_handler)
        listener.connect_to_signal("ItemRemove", self.item_remove_handler)
        listener.connect_to_signal("AllForNow", self.all_for_now_handler)
        listener.connect_to_signal("CacheExhausted", self.cache_exhausted_handler)
        listener.connect_to_signal("Failure", self.failure_handler)

        self._mainloop = gobject.MainLoop()


    def service_resolved(self, *args):
        print 'New service: name = %s; address = %s; port = %s' % (args[2], args[7], args[8])


    def print_error(self, *args):
        print 'error_handler'
        print args[0]

    def item_new_handler(self, interface, protocol, name, stype, domain, flags):
        self._server.ResolveService(interface, protocol, name, stype, domain, avahi.PROTO_UNSPEC, dbus.UInt32(0), reply_handler=self.service_resolved, error_handler=self.print_error)


    def item_remove_handler(self, interface, protocol, name, stype, domain, flags):
        print "Removed service: %s" % name


    def all_for_now_handler(self):
        print "that's all for now"
        self._mainloop.quit()


    def cache_exhausted_handler(self):
        print "cache exhausted"


    def failure_handler(self, error):
        print "failure: %s" % error


    def browse(self):
        self._mainloop.run()


def browse_avahi(service):
    browser = ServiceBrowser(service)
    browser.browse()

def main():
    t0 = datetime.utcnow()
    print "browsing ..."
    browse_avahi('_http._tcp')
    print "done"
    print datetime.utcnow()-t0

if __name__ == '__main__':
    main()
2

当最后一个服务到达时,你需要停止服务浏览。这样你就能看到当前可用的服务列表。

在C语言层面,有一个标志可以让你检查是否还有更多服务即将到来。这个标志叫做AVAHI_BROWSER_ALL_FOR_NOW。我对python的绑定不太熟悉,但也许你可以在回调的stypeflags参数中检查这个标志?

当你看到这个标志被设置时,就调用gobject.MainLoop().quit()来结束循环。

撰写回答