为什么Gio.Socket.create_source()返回null?

时间:2017-09-08 07:38:33

标签: sockets gio gjs

我是套接字的新手,并尝试在GJS / Gio中完成一些套接字编程,然后在墙上创建一个GLib.Source来处理从套接字接收。相关代码(我认为)是:

const DeviceChannel = new Lang.Class({
    Name: "DeviceChannel",

    _init: function (device) {
        this.device = device;

        this.connection = null;
        this.inStream = null;
        this.outStream = null;
        this.socket = null;
        this.sock_source = 0;
    },

    open: function () {
        let client = new Gio.SocketClient();

        this.addr = new Gio.InetSocketAddress({
            address: this.device.tcpHost,
            port: this.device.tcpPort
        });

        let conn = client.connect_async(
            this.addr,
            null,
            Lang.bind(this, this.opened)
        );
    },

    opened: function (client, res) {
        this.connection = client.connect_finish(res);

        // Streams
        this.inStream = new Gio.DataInputStream({
            base_stream: this.connection.get_input_stream()
        });

        this.outStream = new Gio.DataOutputStream({
            base_stream: this.connection.get_output_stream()
        });

        // Socket
        this.socket = this.connection.get_socket();
        this.socket.set_option(6, 4, 10);   // TCP_KEEPIDLE
        this.socket.set_option(6, 5, 5);    // TCP_KEEPINTVL
        this.socket.set_option(6, 6, 3);    // TCP_KEEPCNT
        this.socket.set_keepalive(true);

        this.sock_source = this.socket.create_source(GLib.IOCondition.IN, null);
        this.sock_source.set_callback(Lang.bind(this, this._io_ready));
        this.sock_source.attach(null);
    },

    _io_ready: function (condition) {
        return true;
    }
});

一切顺利,直到我收到错误时致电this.sock_source.set_callback()

(JSConnect:15118): Gjs-WARNING **: JS ERROR: TypeError: this.sock_source is null
DeviceChannel<.opened@application.js:184:9
wrapper@resource:///org/gnome/gjs/modules/lang.js:178:22
@application.js:427:2

我在代码的另一部分中的另一个套接字(尽管是UDP)上调用了Gio.Socket.create_source(),它工作正常。调用create_source()本身不会抛出任何错误(即使我使用G_MESSAGES_DEBUG=all运行我的脚本)并且没有提及函数返回null in the documentation所以我'我感到困惑的是我做错了什么。

修改

有3年的评论here表明:

  

这是工作,因为1)Socket.create_source不存在   typelib,因为它被标记       with(skip)in Glib / gio / gsocket.c

但我假设这已不再适用,因为我在UDP套接字上创建了一个源代码,尽管该套接字是“手工制作的”,并非使用Gio.SocketClient()构建。

1 个答案:

答案 0 :(得分:1)

您可能无意中调用了Gio.DatagramBased.create_source()方法。查看源代码,最终将调用g_socket_create_source(),但先进行一些检查,如果失败则返回null。以下是支票:https://github.com/GNOME/glib/blob/master/gio/gsocket.c#L1114

它似乎是the method will simply return null的一个小错误,甚至没有打印check_datagram_based()的错误。