raindrops 0.13.0 → 0.19.2
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.document +1 -2
- data/.gitattributes +4 -0
- data/.gitignore +1 -1
- data/.olddoc.yml +13 -0
- data/GIT-VERSION-GEN +1 -1
- data/GNUmakefile +1 -2
- data/LICENSE +3 -3
- data/README +28 -34
- data/TODO +2 -0
- data/archive/.gitignore +3 -0
- data/archive/slrnpull.conf +4 -0
- data/examples/linux-listener-stats.rb +1 -2
- data/examples/watcher_demo.ru +1 -1
- data/examples/yahns.conf.rb +30 -0
- data/examples/zbatery.conf.rb +4 -1
- data/ext/raindrops/extconf.rb +107 -2
- data/ext/raindrops/linux_inet_diag.c +94 -101
- data/ext/raindrops/raindrops.c +28 -7
- data/ext/raindrops/tcp_info.c +245 -0
- data/lib/raindrops.rb +1 -1
- data/lib/raindrops/aggregate.rb +1 -1
- data/lib/raindrops/aggregate/last_data_recv.rb +1 -5
- data/lib/raindrops/aggregate/pmq.rb +23 -17
- data/lib/raindrops/linux.rb +5 -6
- data/lib/raindrops/middleware.rb +4 -6
- data/lib/raindrops/middleware/proxy.rb +2 -2
- data/lib/raindrops/watcher.rb +13 -13
- data/pkg.mk +26 -50
- data/raindrops.gemspec +14 -21
- data/test/ipv6_enabled.rb +4 -4
- data/test/test_aggregate_pmq.rb +1 -1
- data/test/test_inet_diag_socket.rb +1 -1
- data/test/test_last_data_recv_unicorn.rb +1 -1
- data/test/test_linux.rb +10 -2
- data/test/test_linux_all_tcp_listen_stats_leak.rb +2 -2
- data/test/test_linux_ipv6.rb +8 -0
- data/test/test_raindrops.rb +1 -1
- data/test/{test_linux_tcp_info.rb → test_tcp_info.rb} +34 -14
- data/test/test_watcher.rb +15 -10
- metadata +59 -171
- data/.wrongdoc.yml +0 -6
- data/Rakefile +0 -28
- data/ext/raindrops/linux_tcp_info.c +0 -173
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 1e1444bc6e35e67993745b3627e848537490fd3b02a67e12be9721b031e22e70
|
4
|
+
data.tar.gz: 9ed2dff40418e89a71ecf18cfb74b72d28d9570ceef314ee1c81bdff19ed832a
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: ccb15ce6a96c1619d53d6090851ef8152ebd673b2db8aed9c6d4bb42757de2d9d051888d3f105d6d3461683eb33575d343e7d31a1ddb91a8f9914bb4e6abcb81
|
7
|
+
data.tar.gz: ce265e39bf3217f3b62931ead0fff78e7e118785466bc0eecff1858f9d808982c12b14f2645ecfffd96c8cb8e0a1a72ac03c0d6ad220c84950dfd2fe05d7f258
|
data/.document
CHANGED
data/.gitattributes
ADDED
data/.gitignore
CHANGED
data/.olddoc.yml
ADDED
@@ -0,0 +1,13 @@
|
|
1
|
+
---
|
2
|
+
cgit_url: https://yhbt.net/raindrops.git/
|
3
|
+
rdoc_url: https://yhbt.net/raindrops/
|
4
|
+
public_email: raindrops-public@yhbt.net
|
5
|
+
ml_url:
|
6
|
+
- https://yhbt.net/raindrops-public/
|
7
|
+
- http://7fh6tueqddpjyxjmgtdiueylzoqt6pt7hec3pukyptlmohoowvhde4yd.onion/raindrops-public
|
8
|
+
nntp_url:
|
9
|
+
- nntp://news.public-inbox.org/inbox.comp.lang.ruby.raindrops
|
10
|
+
- nntp://7fh6tueqddpjyxjmgtdiueylzoqt6pt7hec3pukyptlmohoowvhde4yd.onion/inbox.comp.lang.ruby.raindrops
|
11
|
+
source_code:
|
12
|
+
- git clone https://yhbt.net/raindrops.git
|
13
|
+
- torsocks git clone http://7fh6tueqddpjyxjmgtdiueylzoqt6pt7hec3pukyptlmohoowvhde4yd.onion/raindrops.git
|
data/GIT-VERSION-GEN
CHANGED
data/GNUmakefile
CHANGED
data/LICENSE
CHANGED
@@ -3,8 +3,8 @@ revision control for names and email addresses of all of them.
|
|
3
3
|
|
4
4
|
You can redistribute it and/or modify it under the terms of the GNU
|
5
5
|
Lesser General Public License (LGPL) as published by the Free Software
|
6
|
-
Foundation, version {2.1}[
|
7
|
-
later. Currently version {3}[
|
6
|
+
Foundation, version {2.1}[https://www.gnu.org/licenses/lgpl-2.1.txt] or
|
7
|
+
later. Currently version {3}[https://www.gnu.org/licenses/lgpl-3.0.txt],
|
8
8
|
is preferred (see link:COPYING).
|
9
9
|
|
10
10
|
raindrops is distributed in the hope that it will be useful, but WITHOUT
|
@@ -13,4 +13,4 @@ FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public
|
|
13
13
|
License for more details.
|
14
14
|
|
15
15
|
You should have received a copy of the GNU Lesser General Public License
|
16
|
-
along with the raindrops; if not, see <
|
16
|
+
along with the raindrops; if not, see <https://www.gnu.org/licenses/>
|
data/README
CHANGED
@@ -1,11 +1,10 @@
|
|
1
1
|
= raindrops - real-time stats for preforking Rack servers
|
2
2
|
|
3
|
-
|
4
|
-
servers. It is designed for preforking servers such as
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
multiple processes.
|
3
|
+
raindrops is a real-time stats toolkit to show statistics for Rack HTTP
|
4
|
+
servers. It is designed for preforking servers such as unicorn, but
|
5
|
+
should support any Rack HTTP server on platforms supporting POSIX shared
|
6
|
+
memory. It may also be used as a generic scoreboard for sharing atomic
|
7
|
+
counters across multiple processes.
|
9
8
|
|
10
9
|
== Features
|
11
10
|
|
@@ -38,12 +37,15 @@ and "tcp_diag" kernel modules are loaded as they do not autoload correctly
|
|
38
37
|
|
39
38
|
== Install
|
40
39
|
|
41
|
-
We recommend GCC 4+ (or compatible) to support the
|
42
|
-
{
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
40
|
+
We recommend GCC 4+ (or compatible) to support the __sync builtins
|
41
|
+
(__sync_{add,sub}_and_fetch()):
|
42
|
+
|
43
|
+
https://gcc.gnu.org/onlinedocs/gcc/_005f_005fsync-Builtins.html
|
44
|
+
|
45
|
+
For non-GCC 4+ users, we also support compilation with the libatomic_ops
|
46
|
+
package starting with Raindrops 0.4.0:
|
47
|
+
|
48
|
+
https://github.com/ivmai/libatomic_ops
|
47
49
|
|
48
50
|
If you're using a packaged Ruby distribution, make sure you have a C
|
49
51
|
compiler and the matching Ruby development libraries and headers.
|
@@ -52,30 +54,25 @@ If you use RubyGems:
|
|
52
54
|
|
53
55
|
gem install raindrops
|
54
56
|
|
55
|
-
Otherwise grab the latest tarball from:
|
56
|
-
|
57
|
-
http://raindrops.bogomips.org/files/
|
58
|
-
|
59
|
-
Unpack it, and run "ruby setup.rb"
|
60
|
-
|
61
57
|
== Usage
|
62
58
|
|
63
59
|
See Raindrops::Middleware and Raindrops::LastDataRecv documentation for
|
64
60
|
use Rack servers. The entire library is fully-documented and we are
|
65
|
-
responsive on the mailing list
|
66
|
-
|
61
|
+
responsive on the publically archived mailing list
|
62
|
+
(mailto:raindrops-public@yhbt.net) if
|
63
|
+
you have any questions or comments.
|
67
64
|
|
68
65
|
== Development
|
69
66
|
|
70
67
|
You can get the latest source via git from the following locations:
|
71
68
|
|
72
|
-
git://
|
69
|
+
git://yhbt.net/raindrops.git
|
73
70
|
git://repo.or.cz/raindrops.git (mirror)
|
74
71
|
|
75
72
|
You may browse the code from the web and download the latest snapshot
|
76
73
|
tarballs here:
|
77
74
|
|
78
|
-
*
|
75
|
+
* https://yhbt.net/raindrops.git
|
79
76
|
* http://repo.or.cz/w/raindrops.git (gitweb)
|
80
77
|
|
81
78
|
Inline patches (from "git format-patch") to the mailing list are
|
@@ -87,22 +84,19 @@ git itself. See the Documentation/SubmittingPatches document
|
|
87
84
|
distributed with git on on patch submission guidelines to follow. Just
|
88
85
|
don't email the git mailing list or maintainer with raindrops patches.
|
89
86
|
|
90
|
-
raindrops is
|
91
|
-
allow for a transition to future versions of the LGPL, contributors are
|
92
|
-
required to sign-off changes allowing allowing the project leader to
|
93
|
-
relicense raindrops under newer versions of the LGPL (which should be
|
94
|
-
similar in spirit to the existing LGPL).
|
87
|
+
raindrops is licensed under the LGPL-2.1+
|
95
88
|
|
96
89
|
== Contact
|
97
90
|
|
98
91
|
All feedback (bug reports, user/development discussion, patches, pull
|
99
|
-
requests) go to the mailing list:
|
100
|
-
|
101
|
-
The mailing list is mirrored to Gmane, all information about the
|
102
|
-
group is here:
|
92
|
+
requests) go to the publically archived mailing list:
|
93
|
+
mailto:raindrops-public@yhbt.net
|
103
94
|
|
104
|
-
|
95
|
+
Mailing list archives are available over HTTPS and NNTP:
|
105
96
|
|
106
|
-
|
97
|
+
* https://yhbt.net/raindrops-public/
|
98
|
+
* http://7fh6tueqddpjyxjmgtdiueylzoqt6pt7hec3pukyptlmohoowvhde4yd.onion/raindrops-public/
|
99
|
+
* nntp://news.public-inbox.org/inbox.comp.lang.ruby.raindrops
|
107
100
|
|
108
|
-
|
101
|
+
Since archives are public, scrub sensitive information and
|
102
|
+
use anonymity tools such as Tor or Mixmaster if you deem necessary.
|
data/TODO
CHANGED
data/archive/.gitignore
ADDED
@@ -15,7 +15,6 @@
|
|
15
15
|
usage = "Usage: #$0 [-d DELAY] [-t QUEUED_THRESHOLD] ADDR..."
|
16
16
|
ARGV.size > 0 or abort usage
|
17
17
|
delay = false
|
18
|
-
all = false
|
19
18
|
queued_thresh = -1
|
20
19
|
# "normal" exits when driven on the command-line
|
21
20
|
trap(:INT) { exit 130 }
|
@@ -25,7 +24,7 @@
|
|
25
24
|
opts.banner = usage
|
26
25
|
opts.on('-d', '--delay=DELAY', Float) { |n| delay = n }
|
27
26
|
opts.on('-t', '--queued-threshold=INT', Integer) { |n| queued_thresh = n }
|
28
|
-
opts.on('-a', '--all') {
|
27
|
+
opts.on('-a', '--all') { } # noop
|
29
28
|
opts.parse! ARGV
|
30
29
|
end
|
31
30
|
|
data/examples/watcher_demo.ru
CHANGED
@@ -0,0 +1,30 @@
|
|
1
|
+
# Inlined rack app using yahns server (git clone git://yhbt.net/yahns.git)
|
2
|
+
# Usage: yahns -c /path/to/this/file.conf.rb
|
3
|
+
# There is no separate config.ru file for this example,
|
4
|
+
# but rack_app may also be a string pointing to the path of a
|
5
|
+
# config.ru file
|
6
|
+
|
7
|
+
require 'rack'
|
8
|
+
rack_app = Rack::Builder.new do
|
9
|
+
use Rack::Head
|
10
|
+
addr = %w(0.0.0.0:9418 0.0.0.0:443 [::]:443 0.0.0.0:80 [::]:80
|
11
|
+
127.0.0.1:6081 127.0.0.1:280 0.0.0.0:119 [::]:119)
|
12
|
+
use Raindrops::Middleware, listeners: addr
|
13
|
+
run Raindrops::Watcher.new(listeners: addr)
|
14
|
+
end.to_app
|
15
|
+
# rack_app = '/path/to/config.ru' # a more standard config
|
16
|
+
|
17
|
+
app(:rack, rack_app) do
|
18
|
+
# I keep IPv4 and IPv6 on separate sockets to avoid ugly
|
19
|
+
# IPv4-mapped-IPv6 addresses:
|
20
|
+
listen 8080
|
21
|
+
listen '[::]:8080', ipv6only: true
|
22
|
+
client_max_body_size 0 # no POST or any uploads
|
23
|
+
client_timeout 5
|
24
|
+
output_buffering false # needed for /tail/ endpoint to avoid ENOSPC
|
25
|
+
queue { worker_threads 30 }
|
26
|
+
end
|
27
|
+
|
28
|
+
# logging is optional, but recommended for diagnosing problems
|
29
|
+
# stderr_path '/var/log/yahns/stderr-raindrops.log'
|
30
|
+
# stdout_path '/var/log/yahns/stdout-raindrops.log'
|
data/examples/zbatery.conf.rb
CHANGED
@@ -1,6 +1,9 @@
|
|
1
1
|
# Used for running Raindrops::Watcher, which requires a multi-threaded
|
2
2
|
# Rack server capable of streaming a response. Threads must be used,
|
3
|
-
# so
|
3
|
+
# so any multi-threaded Rack server may be used.
|
4
|
+
# zbatery was recommended in the past, but it is abandoned
|
5
|
+
# <http://zbatery.bogomip.org/>.
|
6
|
+
# yahns may work as an alternative (see yahns.conf.rb in this dir)
|
4
7
|
Rainbows! do
|
5
8
|
use :ThreadSpawn
|
6
9
|
end
|
data/ext/raindrops/extconf.rb
CHANGED
@@ -1,4 +1,5 @@
|
|
1
1
|
require 'mkmf'
|
2
|
+
require 'shellwords'
|
2
3
|
|
3
4
|
dir_config('atomic_ops')
|
4
5
|
have_func('mmap', 'sys/mman.h') or abort 'mmap() not found'
|
@@ -6,9 +7,112 @@
|
|
6
7
|
|
7
8
|
$CPPFLAGS += " -D_GNU_SOURCE "
|
8
9
|
have_func('mremap', 'sys/mman.h')
|
10
|
+
headers = %w(sys/types.h netdb.h string.h sys/socket.h netinet/in.h)
|
11
|
+
if have_header('linux/tcp.h')
|
12
|
+
headers << 'linux/tcp.h'
|
13
|
+
else
|
14
|
+
%w(netinet/tcp.h netinet/tcp_fsm.h).each { |h|
|
15
|
+
have_header(h, headers) and headers << h
|
16
|
+
}
|
17
|
+
end
|
9
18
|
|
10
19
|
$CPPFLAGS += " -D_BSD_SOURCE "
|
20
|
+
|
21
|
+
if have_type("struct tcp_info", headers)
|
22
|
+
%w(
|
23
|
+
tcpi_state
|
24
|
+
tcpi_ca_state
|
25
|
+
tcpi_retransmits
|
26
|
+
tcpi_probes
|
27
|
+
tcpi_backoff
|
28
|
+
tcpi_options
|
29
|
+
tcpi_snd_wscale
|
30
|
+
tcpi_rcv_wscale
|
31
|
+
tcpi_rto
|
32
|
+
tcpi_ato
|
33
|
+
tcpi_snd_mss
|
34
|
+
tcpi_rcv_mss
|
35
|
+
tcpi_unacked
|
36
|
+
tcpi_sacked
|
37
|
+
tcpi_lost
|
38
|
+
tcpi_retrans
|
39
|
+
tcpi_fackets
|
40
|
+
tcpi_last_data_sent
|
41
|
+
tcpi_last_ack_sent
|
42
|
+
tcpi_last_data_recv
|
43
|
+
tcpi_last_ack_recv
|
44
|
+
tcpi_pmtu
|
45
|
+
tcpi_rcv_ssthresh
|
46
|
+
tcpi_rtt
|
47
|
+
tcpi_rttvar
|
48
|
+
tcpi_snd_ssthresh
|
49
|
+
tcpi_snd_cwnd
|
50
|
+
tcpi_advmss
|
51
|
+
tcpi_reordering
|
52
|
+
tcpi_rcv_rtt
|
53
|
+
tcpi_rcv_space
|
54
|
+
tcpi_total_retrans
|
55
|
+
tcpi_snd_wnd
|
56
|
+
tcpi_snd_bwnd
|
57
|
+
tcpi_snd_nxt
|
58
|
+
tcpi_rcv_nxt
|
59
|
+
tcpi_toe_tid
|
60
|
+
tcpi_snd_rexmitpack
|
61
|
+
tcpi_rcv_ooopack
|
62
|
+
tcpi_snd_zerowin
|
63
|
+
).each do |field|
|
64
|
+
cfunc = "tcp_info_#{field}"
|
65
|
+
if have_struct_member('struct tcp_info', field, headers)
|
66
|
+
func_body = <<EOF
|
67
|
+
static VALUE #{cfunc}(VALUE self)
|
68
|
+
{
|
69
|
+
struct tcp_info *info = DATA_PTR(self);
|
70
|
+
return UINT2NUM((uint32_t)info->#{field});
|
71
|
+
}
|
72
|
+
EOF
|
73
|
+
func_body.delete!("\n")
|
74
|
+
$defs << "-DCFUNC_#{cfunc}=#{Shellwords.shellescape(func_body)}"
|
75
|
+
else
|
76
|
+
func_body = "static inline void #{cfunc}(void) {}"
|
77
|
+
$defs << "-DCFUNC_#{cfunc}=#{Shellwords.shellescape(func_body)}"
|
78
|
+
cfunc = 'rb_f_notimplement'.freeze
|
79
|
+
end
|
80
|
+
rbmethod = %Q("#{field.sub(/\Atcpi_/, ''.freeze)}")
|
81
|
+
$defs << "-DDEFINE_METHOD_tcp_info_#{field}=" \
|
82
|
+
"#{Shellwords.shellescape(
|
83
|
+
%Q[rb_define_method(cTCP_Info,#{rbmethod},#{cfunc},0)])}"
|
84
|
+
end
|
85
|
+
tcp_state_map = {
|
86
|
+
ESTABLISHED: %w(TCP_ESTABLISHED TCPS_ESTABLISHED),
|
87
|
+
SYN_SENT: %w(TCP_SYN_SENT TCPS_SYN_SENT),
|
88
|
+
SYN_RECV: %w(TCP_SYN_RECV TCPS_SYN_RECEIVED),
|
89
|
+
FIN_WAIT1: %w(TCP_FIN_WAIT1 TCPS_FIN_WAIT_1),
|
90
|
+
FIN_WAIT2: %w(TCP_FIN_WAIT2 TCPS_FIN_WAIT_2),
|
91
|
+
TIME_WAIT: %w(TCP_TIME_WAIT TCPS_TIME_WAIT),
|
92
|
+
CLOSE: %w(TCP_CLOSE TCPS_CLOSED),
|
93
|
+
CLOSE_WAIT: %w(TCP_CLOSE_WAIT TCPS_CLOSE_WAIT),
|
94
|
+
LAST_ACK: %w(TCP_LAST_ACK TCPS_LAST_ACK),
|
95
|
+
LISTEN: %w(TCP_LISTEN TCPS_LISTEN),
|
96
|
+
CLOSING: %w(TCP_CLOSING TCPS_CLOSING),
|
97
|
+
}
|
98
|
+
nstate = 0
|
99
|
+
tcp_state_map.each do |state, try|
|
100
|
+
try.each do |os_name|
|
101
|
+
have_const(os_name, headers) or next
|
102
|
+
tcp_state_map[state] = os_name
|
103
|
+
nstate += 1
|
104
|
+
end
|
105
|
+
end
|
106
|
+
if nstate == tcp_state_map.size
|
107
|
+
$defs << '-DRAINDROPS_TCP_STATES_ALL_KNOWN=1'
|
108
|
+
tcp_state_map.each do |state, name|
|
109
|
+
$defs << "-DRAINDROPS_TCP_#{state}=#{name}"
|
110
|
+
end
|
111
|
+
end
|
112
|
+
end
|
113
|
+
|
11
114
|
have_func("getpagesize", "unistd.h")
|
115
|
+
have_func('rb_thread_call_without_gvl')
|
12
116
|
have_func('rb_thread_blocking_region')
|
13
117
|
have_func('rb_thread_io_blocking_region')
|
14
118
|
|
@@ -39,17 +143,18 @@
|
|
39
143
|
$defs.push(format("-DHAVE_GCC_ATOMIC_BUILTINS"))
|
40
144
|
true
|
41
145
|
else
|
42
|
-
|
146
|
+
$CFLAGS = prev_cflags
|
43
147
|
false
|
44
148
|
end
|
45
149
|
end
|
46
150
|
end or have_header('atomic_ops.h') or abort <<-SRC
|
47
151
|
|
48
152
|
libatomic_ops is required if GCC 4+ is not used.
|
49
|
-
See
|
153
|
+
See https://github.com/ivmai/libatomic_ops
|
50
154
|
|
51
155
|
Users of Debian-based distros may run:
|
52
156
|
|
53
157
|
apt-get install libatomic-ops-dev
|
54
158
|
SRC
|
159
|
+
create_header # generate extconf.h to avoid excessively long command-line
|
55
160
|
create_makefile('raindrops_ext')
|
@@ -1,45 +1,24 @@
|
|
1
1
|
#include <ruby.h>
|
2
|
-
#
|
3
|
-
#
|
4
|
-
#else
|
5
|
-
# include <st.h>
|
6
|
-
#endif
|
2
|
+
#include <stdarg.h>
|
3
|
+
#include <ruby/st.h>
|
7
4
|
#include "my_fileno.h"
|
8
5
|
#ifdef __linux__
|
9
6
|
|
10
|
-
/* Ruby 1.8.6+ macros (for compatibility with Ruby 1.9) */
|
11
|
-
#ifndef RSTRING_LEN
|
12
|
-
# define RSTRING_LEN(s) (RSTRING(s)->len)
|
13
|
-
#endif
|
14
|
-
|
15
|
-
/* partial emulation of the 1.9 rb_thread_blocking_region under 1.8 */
|
16
|
-
#if !defined(HAVE_RB_THREAD_BLOCKING_REGION) && \
|
17
|
-
!defined(HAVE_RB_THREAD_IO_BLOCKING_REGION)
|
18
|
-
# include <rubysig.h>
|
19
|
-
# define RUBY_UBF_IO ((rb_unblock_function_t *)-1)
|
20
|
-
typedef void rb_unblock_function_t(void *);
|
21
|
-
typedef VALUE rb_blocking_function_t(void *);
|
22
|
-
static VALUE
|
23
|
-
rb_thread_blocking_region(
|
24
|
-
rb_blocking_function_t *func, void *data1,
|
25
|
-
rb_unblock_function_t *ubf, void *data2)
|
26
|
-
{
|
27
|
-
VALUE rv;
|
28
|
-
|
29
|
-
TRAP_BEG;
|
30
|
-
rv = func(data1);
|
31
|
-
TRAP_END;
|
32
|
-
|
33
|
-
return rv;
|
34
|
-
}
|
35
|
-
#endif /* ! HAVE_RB_THREAD_BLOCKING_REGION */
|
36
|
-
|
37
7
|
#ifdef HAVE_RB_THREAD_IO_BLOCKING_REGION
|
8
|
+
/* Ruby 1.9.3 and 2.0.0 */
|
38
9
|
VALUE rb_thread_io_blocking_region(rb_blocking_function_t *, void *, int);
|
10
|
+
# define rd_fd_region(fn,data,fd) \
|
11
|
+
rb_thread_io_blocking_region((fn),(data),(fd))
|
12
|
+
#elif defined(HAVE_RB_THREAD_CALL_WITHOUT_GVL) && \
|
13
|
+
defined(HAVE_RUBY_THREAD_H) && HAVE_RUBY_THREAD_H
|
14
|
+
/* in case Ruby 2.0+ ever drops rb_thread_io_blocking_region: */
|
15
|
+
# include <ruby/thread.h>
|
16
|
+
# define COMPAT_FN (void *(*)(void *))
|
17
|
+
# define rd_fd_region(fn,data,fd) \
|
18
|
+
rb_thread_call_without_gvl(COMPAT_FN(fn),(data),RUBY_UBF_IO,NULL)
|
39
19
|
#else
|
40
|
-
#
|
41
|
-
|
42
|
-
#endif /* HAVE_RB_THREAD_IO_BLOCKING_REGION */
|
20
|
+
# error Ruby <= 1.8 not supported
|
21
|
+
#endif
|
43
22
|
|
44
23
|
#include <assert.h>
|
45
24
|
#include <errno.h>
|
@@ -213,91 +192,89 @@ static const char *addr_any(sa_family_t family)
|
|
213
192
|
return ipv6;
|
214
193
|
}
|
215
194
|
|
216
|
-
|
195
|
+
#ifdef __GNUC__
|
196
|
+
static void bug_warn_nogvl(const char *, ...)
|
197
|
+
__attribute__((format(printf,1,2)));
|
198
|
+
#endif
|
199
|
+
static void bug_warn_nogvl(const char *fmt, ...)
|
217
200
|
{
|
201
|
+
va_list ap;
|
202
|
+
|
203
|
+
va_start(ap, fmt);
|
204
|
+
vfprintf(stderr, fmt, ap);
|
205
|
+
va_end(ap);
|
206
|
+
|
218
207
|
fprintf(stderr, "Please report how you produced this at "\
|
219
|
-
"raindrops@
|
208
|
+
"raindrops-public@yhbt.net\n");
|
220
209
|
fflush(stderr);
|
221
210
|
}
|
222
211
|
|
223
212
|
static struct listen_stats *stats_for(st_table *table, struct inet_diag_msg *r)
|
224
213
|
{
|
225
|
-
char *key, *port, *old_key;
|
214
|
+
char *host, *key, *port, *old_key;
|
226
215
|
size_t alloca_len;
|
227
216
|
struct listen_stats *stats;
|
228
|
-
|
229
|
-
|
230
|
-
|
231
|
-
|
232
|
-
|
233
|
-
|
234
|
-
|
235
|
-
switch ((sa.ss.ss_family = r->idiag_family)) {
|
217
|
+
socklen_t hostlen;
|
218
|
+
socklen_t portlen = (socklen_t)sizeof("65535");
|
219
|
+
int n;
|
220
|
+
const void *src = r->id.idiag_src;
|
221
|
+
|
222
|
+
switch (r->idiag_family) {
|
236
223
|
case AF_INET: {
|
237
|
-
|
238
|
-
|
239
|
-
|
240
|
-
alloca_len = keylen + 1 + portlen;
|
241
|
-
key = alloca(alloca_len);
|
242
|
-
key[keylen] = 0; /* will be ':' later */
|
243
|
-
port = key + keylen + 1;
|
244
|
-
rc = getnameinfo(&sa.sa, len,
|
245
|
-
key, keylen, port, portlen, flags);
|
224
|
+
hostlen = INET_ADDRSTRLEN;
|
225
|
+
alloca_len = hostlen + portlen;
|
226
|
+
host = key = alloca(alloca_len);
|
246
227
|
break;
|
247
228
|
}
|
248
229
|
case AF_INET6: {
|
249
|
-
|
250
|
-
|
251
|
-
keylen = INET6_ADDRSTRLEN;
|
252
|
-
/* [ ] */
|
253
|
-
alloca_len = 1 + keylen + 1 + 1 + portlen;
|
230
|
+
hostlen = INET6_ADDRSTRLEN;
|
231
|
+
alloca_len = 1 + hostlen + 1 + portlen;
|
254
232
|
key = alloca(alloca_len);
|
255
|
-
|
256
|
-
key[1 + keylen + 1] = 0; /* will be ':' later */
|
257
|
-
port = 1 + key + keylen + 1 + 1;
|
258
|
-
rc = getnameinfo(&sa.sa, len,
|
259
|
-
key + 1, keylen, port, portlen, flags);
|
233
|
+
host = key + 1;
|
260
234
|
break;
|
261
235
|
}
|
262
236
|
default:
|
263
237
|
assert(0 && "unsupported address family, could that be IPv7?!");
|
264
238
|
}
|
265
|
-
if (
|
266
|
-
|
267
|
-
|
268
|
-
*
|
239
|
+
if (!inet_ntop(r->idiag_family, src, host, hostlen)) {
|
240
|
+
bug_warn_nogvl("BUG: inet_ntop: %s\n", strerror(errno));
|
241
|
+
*key = '\0';
|
242
|
+
*host = '\0';
|
269
243
|
}
|
270
|
-
|
271
|
-
|
272
|
-
portlen = strlen(port);
|
273
|
-
|
274
|
-
switch (sa.ss.ss_family) {
|
244
|
+
hostlen = (socklen_t)strlen(host);
|
245
|
+
switch (r->idiag_family) {
|
275
246
|
case AF_INET:
|
276
|
-
|
277
|
-
|
247
|
+
host[hostlen] = ':';
|
248
|
+
port = host + hostlen + 1;
|
278
249
|
break;
|
279
250
|
case AF_INET6:
|
280
|
-
key[
|
281
|
-
|
282
|
-
|
283
|
-
|
251
|
+
key[0] = '[';
|
252
|
+
host[hostlen] = ']';
|
253
|
+
host[hostlen + 1] = ':';
|
254
|
+
port = host + hostlen + 2;
|
284
255
|
break;
|
285
256
|
default:
|
286
257
|
assert(0 && "unsupported address family, could that be IPv7?!");
|
287
258
|
}
|
288
259
|
|
260
|
+
n = snprintf(port, portlen, "%u", ntohs(r->id.idiag_sport));
|
261
|
+
if (n <= 0) {
|
262
|
+
bug_warn_nogvl("BUG: snprintf port: %d\n", n);
|
263
|
+
*key = '\0';
|
264
|
+
}
|
265
|
+
|
289
266
|
if (st_lookup(table, (st_data_t)key, (st_data_t *)&stats))
|
290
267
|
return stats;
|
291
268
|
|
292
269
|
old_key = key;
|
293
270
|
|
294
271
|
if (r->idiag_state == TCP_ESTABLISHED) {
|
295
|
-
|
296
|
-
addr_any(
|
272
|
+
n = snprintf(key, alloca_len, "%s:%u",
|
273
|
+
addr_any(r->idiag_family),
|
297
274
|
ntohs(r->id.idiag_sport));
|
298
275
|
if (n <= 0) {
|
299
|
-
|
300
|
-
|
276
|
+
bug_warn_nogvl("BUG: snprintf: %d\n", n);
|
277
|
+
*key = '\0';
|
301
278
|
}
|
302
279
|
if (st_lookup(table, (st_data_t)key, (st_data_t *)&stats))
|
303
280
|
return stats;
|
@@ -310,8 +287,9 @@ static struct listen_stats *stats_for(st_table *table, struct inet_diag_msg *r)
|
|
310
287
|
memcpy(key, old_key, n + 1);
|
311
288
|
}
|
312
289
|
} else {
|
313
|
-
|
314
|
-
|
290
|
+
size_t old_len = strlen(old_key) + 1;
|
291
|
+
key = xmalloc(old_len);
|
292
|
+
memcpy(key, old_key, old_len);
|
315
293
|
}
|
316
294
|
stats = xcalloc(1, sizeof(struct listen_stats));
|
317
295
|
st_insert(table, (st_data_t)key, (st_data_t)stats);
|
@@ -391,8 +369,8 @@ static void prep_diag_args(
|
|
391
369
|
|
392
370
|
nladdr->nl_family = AF_NETLINK;
|
393
371
|
|
394
|
-
req->nlh.nlmsg_len = sizeof(struct diag_req) +
|
395
|
-
RTA_LENGTH(args->iov[2].iov_len);
|
372
|
+
req->nlh.nlmsg_len = (unsigned int)(sizeof(struct diag_req) +
|
373
|
+
RTA_LENGTH(args->iov[2].iov_len));
|
396
374
|
req->nlh.nlmsg_type = TCPDIAG_GETSOCK;
|
397
375
|
req->nlh.nlmsg_flags = NLM_F_ROOT | NLM_F_MATCH | NLM_F_REQUEST;
|
398
376
|
req->nlh.nlmsg_pid = getpid();
|
@@ -465,12 +443,12 @@ static VALUE diag(void *ptr)
|
|
465
443
|
}
|
466
444
|
}
|
467
445
|
out:
|
468
|
-
|
446
|
+
/* prepare to raise, free memory before reacquiring GVL */
|
447
|
+
if (err && args->table) {
|
469
448
|
int save_errno = errno;
|
470
|
-
|
471
|
-
|
472
|
-
|
473
|
-
}
|
449
|
+
|
450
|
+
st_foreach(args->table, st_free_data, 0);
|
451
|
+
st_free_table(args->table);
|
474
452
|
errno = save_errno;
|
475
453
|
}
|
476
454
|
return (VALUE)err;
|
@@ -584,6 +562,10 @@ static void gen_bytecode(struct iovec *iov, union any_addr *inet)
|
|
584
562
|
}
|
585
563
|
}
|
586
564
|
|
565
|
+
/*
|
566
|
+
* n.b. we may safely raise here because an error will cause diag()
|
567
|
+
* to free args->table
|
568
|
+
*/
|
587
569
|
static void nl_errcheck(VALUE r)
|
588
570
|
{
|
589
571
|
const char *err = (const char *)r;
|
@@ -604,11 +586,18 @@ static VALUE tcp_stats(struct nogvl_args *args, VALUE addr)
|
|
604
586
|
gen_bytecode(&args->iov[2], &query_addr);
|
605
587
|
|
606
588
|
memset(&args->stats, 0, sizeof(struct listen_stats));
|
607
|
-
nl_errcheck(
|
589
|
+
nl_errcheck(rd_fd_region(diag, args, args->fd));
|
608
590
|
|
609
591
|
return rb_listen_stats(&args->stats);
|
610
592
|
}
|
611
593
|
|
594
|
+
static int drop_placeholders(st_data_t k, st_data_t v, st_data_t ign)
|
595
|
+
{
|
596
|
+
if ((VALUE)v == Qtrue)
|
597
|
+
return ST_DELETE;
|
598
|
+
return ST_CONTINUE;
|
599
|
+
}
|
600
|
+
|
612
601
|
/*
|
613
602
|
* call-seq:
|
614
603
|
* Raindrops::Linux.tcp_listener_stats([addrs[, sock]]) => hash
|
@@ -649,10 +638,9 @@ static VALUE tcp_listener_stats(int argc, VALUE *argv, VALUE self)
|
|
649
638
|
case T_ARRAY: {
|
650
639
|
long i;
|
651
640
|
long len = RARRAY_LEN(addrs);
|
652
|
-
VALUE cur;
|
653
641
|
|
654
642
|
if (len == 1) {
|
655
|
-
cur = rb_ary_entry(addrs, 0);
|
643
|
+
VALUE cur = rb_ary_entry(addrs, 0);
|
656
644
|
|
657
645
|
rb_hash_aset(rv, cur, tcp_stats(&args, cur));
|
658
646
|
return rv;
|
@@ -662,7 +650,7 @@ static VALUE tcp_listener_stats(int argc, VALUE *argv, VALUE self)
|
|
662
650
|
VALUE cur = rb_ary_entry(addrs, i);
|
663
651
|
|
664
652
|
parse_addr(&check, cur);
|
665
|
-
rb_hash_aset(rv, cur, Qtrue);
|
653
|
+
rb_hash_aset(rv, cur, Qtrue /* placeholder */);
|
666
654
|
}
|
667
655
|
/* fall through */
|
668
656
|
}
|
@@ -675,11 +663,14 @@ static VALUE tcp_listener_stats(int argc, VALUE *argv, VALUE self)
|
|
675
663
|
"addr must be an array of strings, a string, or nil");
|
676
664
|
}
|
677
665
|
|
678
|
-
nl_errcheck(
|
666
|
+
nl_errcheck(rd_fd_region(diag, &args, args.fd));
|
679
667
|
|
680
668
|
st_foreach(args.table, NIL_P(addrs) ? st_to_hash : st_AND_hash, rv);
|
681
669
|
st_free_table(args.table);
|
682
670
|
|
671
|
+
if (RHASH_SIZE(rv) > 1)
|
672
|
+
rb_hash_foreach(rv, drop_placeholders, Qfalse);
|
673
|
+
|
683
674
|
/* let GC deal with corner cases */
|
684
675
|
if (argc < 2) rb_io_close(sock);
|
685
676
|
return rv;
|
@@ -687,11 +678,12 @@ static VALUE tcp_listener_stats(int argc, VALUE *argv, VALUE self)
|
|
687
678
|
|
688
679
|
void Init_raindrops_linux_inet_diag(void)
|
689
680
|
{
|
690
|
-
VALUE cRaindrops =
|
681
|
+
VALUE cRaindrops = rb_define_class("Raindrops", rb_cObject);
|
691
682
|
VALUE mLinux = rb_define_module_under(cRaindrops, "Linux");
|
683
|
+
VALUE Socket;
|
692
684
|
|
693
685
|
rb_require("socket");
|
694
|
-
|
686
|
+
Socket = rb_const_get(rb_cObject, rb_intern("Socket"));
|
695
687
|
id_new = rb_intern("new");
|
696
688
|
|
697
689
|
/*
|
@@ -700,10 +692,11 @@ void Init_raindrops_linux_inet_diag(void)
|
|
700
692
|
* This is a subclass of +Socket+ specifically for talking
|
701
693
|
* to the inet_diag facility of Netlink.
|
702
694
|
*/
|
703
|
-
cIDSock = rb_define_class_under(cRaindrops, "InetDiagSocket",
|
695
|
+
cIDSock = rb_define_class_under(cRaindrops, "InetDiagSocket", Socket);
|
704
696
|
rb_define_singleton_method(cIDSock, "new", ids_s_new, 0);
|
705
697
|
|
706
698
|
cListenStats = rb_const_get(cRaindrops, rb_intern("ListenStats"));
|
699
|
+
rb_gc_register_mark_object(cListenStats); /* pin */
|
707
700
|
|
708
701
|
rb_define_module_function(mLinux, "tcp_listener_stats",
|
709
702
|
tcp_listener_stats, -1);
|