From: naruse@... Date: 2017-04-11T06:13:33+00:00 Subject: [ruby-core:80652] [Ruby trunk Bug#13409] UDPSocket#send breaks when using sockaddr Issue #13409 has been updated by naruse (Yui NARUSE). naruse (Yui NARUSE) wrote: > `UDPSocket.new` receives an optional argument address family, whose default is AF_INET. > If you use IPv6, you need to specify AF_INET6 like `UDPSocket.new(Socket::AF_INET6)`. FYI, [Socket#udp_server_loop](https://ruby-doc.org/stdlib-2.4.1/libdoc/socket/rdoc/Socket.html#method-c-udp_server_loop) is useful to implement UDP server which supports both IPv4 and IPv6. ---------------------------------------- Bug #13409: UDPSocket#send breaks when using sockaddr https://bugs.ruby-lang.org/issues/13409#change-64173 * Author: ioquatix (Samuel Williams) * Status: Rejected * Priority: Normal * Assignee: * Target version: * ruby -v: 2.4.0 * Backport: 2.2: UNKNOWN, 2.3: UNKNOWN, 2.4: UNKNOWN ---------------------------------------- Here is the working example. ~~~ruby #!/usr/bin/env ruby require 'socket' port = 6778 server = UDPSocket.new.tap{|socket| socket.bind("localhost", port)} client = UDPSocket.new data = "Matz is nice so we are nice." t1 = Thread.new do packet, (_, remote_port, remote_host) = server.recvfrom(512) server.send(packet, 0, remote_host, remote_port) end t2 = Thread.new do client.send(data, 0, "localhost", port) response, _ = client.recvfrom(512) puts "Got response: #{response.inspect}" end [t1, t2].each(&:join) puts "Finished." ~~~ Here is one that fails with EINVAL: ~~~ruby #!/usr/bin/env ruby Thread.abort_on_exception = true require 'socket' port = 6778 server = UDPSocket.new.tap{|socket| socket.bind("localhost", port)} client = UDPSocket.new data = "Matz is nice so we are nice." t1 = Thread.new do puts "Server waiting for packet..." packet, (_, remote_port, remote_host) = server.recvfrom(512) server.send(packet, 0, remote_host, remote_port) end t2 = Thread.new do address = Addrinfo.udp("localhost", port) puts "Sending data to #{address.inspect}" # Should call ssize_t sendto(int sockfd, const void *buf, size_t len, int flags, const struct sockaddr *dest_addr, socklen_t addrlen), but calls send which fails with EINVAL. result = client.send(data, 0, address.to_sockaddr) response, _ = client.recvfrom(512) puts "Got response: #{response.inspect}" end [t1, t2].each(&:join) puts "Finished." ~~~ -- https://bugs.ruby-lang.org/ Unsubscribe: