async-io 1.27.7 → 1.30.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (57) hide show
  1. checksums.yaml +4 -4
  2. data/lib/async/io/host_endpoint.rb +1 -1
  3. data/lib/async/io/ssl_socket.rb +1 -1
  4. data/lib/async/io/stream.rb +16 -45
  5. data/{spec/async/io/socket/udp_spec.rb → lib/async/io/threads.rb} +56 -36
  6. data/lib/async/io/trap.rb +8 -3
  7. data/lib/async/io/version.rb +1 -1
  8. metadata +50 -99
  9. data/.editorconfig +0 -6
  10. data/.gitignore +0 -13
  11. data/.rspec +0 -3
  12. data/.travis.yml +0 -25
  13. data/.yardopts +0 -2
  14. data/Gemfile +0 -20
  15. data/README.md +0 -171
  16. data/async-io.gemspec +0 -30
  17. data/examples/allocations/byteslice.rb +0 -29
  18. data/examples/allocations/memory.rb +0 -16
  19. data/examples/allocations/read_chunks.rb +0 -18
  20. data/examples/chat/client.rb +0 -58
  21. data/examples/chat/server.rb +0 -83
  22. data/examples/defer/worker.rb +0 -29
  23. data/examples/echo/client.rb +0 -23
  24. data/examples/echo/server.rb +0 -58
  25. data/examples/issues/broken_ssl.rb +0 -15
  26. data/examples/issues/pipes.rb +0 -34
  27. data/examples/millions/client.rb +0 -44
  28. data/examples/millions/server.rb +0 -41
  29. data/examples/udp/client.rb +0 -14
  30. data/examples/udp/server.rb +0 -16
  31. data/gems/nio4r-2.3.gemfile +0 -3
  32. data/spec/addrinfo.rb +0 -16
  33. data/spec/async/io/buffer_spec.rb +0 -48
  34. data/spec/async/io/c10k_spec.rb +0 -138
  35. data/spec/async/io/echo_spec.rb +0 -75
  36. data/spec/async/io/endpoint_spec.rb +0 -105
  37. data/spec/async/io/generic_examples.rb +0 -73
  38. data/spec/async/io/generic_spec.rb +0 -107
  39. data/spec/async/io/notification_spec.rb +0 -46
  40. data/spec/async/io/protocol/line_spec.rb +0 -81
  41. data/spec/async/io/shared_endpoint/server_spec.rb +0 -72
  42. data/spec/async/io/shared_endpoint_spec.rb +0 -67
  43. data/spec/async/io/socket/tcp_spec.rb +0 -101
  44. data/spec/async/io/socket_spec.rb +0 -149
  45. data/spec/async/io/ssl_server_spec.rb +0 -133
  46. data/spec/async/io/ssl_socket_spec.rb +0 -96
  47. data/spec/async/io/standard_spec.rb +0 -47
  48. data/spec/async/io/stream_context.rb +0 -30
  49. data/spec/async/io/stream_spec.rb +0 -337
  50. data/spec/async/io/tcp_socket_spec.rb +0 -84
  51. data/spec/async/io/trap_spec.rb +0 -52
  52. data/spec/async/io/udp_socket_spec.rb +0 -56
  53. data/spec/async/io/unix_endpoint_spec.rb +0 -106
  54. data/spec/async/io/unix_socket_spec.rb +0 -66
  55. data/spec/async/io/wrap/http_rb_spec.rb +0 -47
  56. data/spec/async/io/wrap/tcp_spec.rb +0 -79
  57. data/spec/spec_helper.rb +0 -15
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: f36b84a41e617dbb5ffd863ecaf7b68761f6cdbc4969052372ef67071e98b85b
4
- data.tar.gz: 5c655fceda94aaae5b17166e7c039e60558dee99755875dbaa51e1f58244f2a0
3
+ metadata.gz: 61a14845c7d079931211e6ac68687686d7be2408f53967980594d13a7e526036
4
+ data.tar.gz: 6e6b66368d65a53f0ca0a313aab8a6857821d098c2c7316c8dae8d67329fa1c4
5
5
  SHA512:
6
- metadata.gz: d3fa287b88b45a084bf750cbd5343f02ad0848c0340ba653937933aeec0c980a61e779c6322afcbcb9ada9fd263d92a35fd9b8d5fb5616df3e9d535d06cb6a8b
7
- data.tar.gz: ca1ea7afbee6741820cd9b058fc1bcaa2b1b1f89352280a484ca1ae5732fd2447ec71f4b06deb0d5d0d965e80a00bb17c567ea3188a11c1368eaa26c976a66a4
6
+ metadata.gz: c49e189437e5c89c011323cbc73c24c799d08e479b85fbf513a96f58b5732262085755750474a048e415a43e703bb78f28100f7a87ac7d506b966357d5d20d6a
7
+ data.tar.gz: b65bfd71f89d9659dc5f30464c820df28a13b14294d7e4cb8ef432cd698b19fc09208a57f3ae2206c4c28db61ddbebfd0a334b9de50d0b89b3f28e621949cc63
@@ -77,7 +77,7 @@ module Async
77
77
  # @yield [Socket] the bound socket
78
78
  # @return [Array<Socket>] an array of bound sockets
79
79
  def bind(&block)
80
- Addrinfo.foreach(*@specification).collect do |address|
80
+ Addrinfo.foreach(*@specification).map do |address|
81
81
  Socket.bind(address, **@options, &block)
82
82
  end
83
83
  end
@@ -30,7 +30,7 @@ module Async
30
30
 
31
31
  # Asynchronous TCP socket wrapper.
32
32
  class SSLSocket < Generic
33
- wraps OpenSSL::SSL::SSLSocket, :alpn_protocol, :cert, :cipher, :client_ca, :context, :getsockopt, :hostname, :hostname=, :npn_protocol, :peer_cert, :peer_cert_chain, :pending, :post_connection_check, :setsockopt, :session, :session=, :session_reused?, :ssl_version, :state, :sync_close, :sync_close=, :sysclose, :verify_result, :tmp_key
33
+ wraps OpenSSL::SSL::SSLSocket, :alpn_protocol, :cert, :cipher, :client_ca, :context, :finished_message, :peer_finished_message, :getsockopt, :hostname, :hostname=, :npn_protocol, :peer_cert, :peer_cert_chain, :pending, :post_connection_check, :setsockopt, :session, :session=, :session_reused?, :ssl_version, :state, :sync_close, :sync_close=, :sysclose, :verify_result, :tmp_key
34
34
 
35
35
  wrap_blocking_method :accept, :accept_nonblock
36
36
  wrap_blocking_method :connect, :connect_nonblock
@@ -46,8 +46,9 @@ module Async
46
46
  @io = io
47
47
  @eof = false
48
48
 
49
- @deferred = deferred
50
49
  @pending = 0
50
+ # This field is ignored, but used to mean, try to buffer packets in a single iteration of the reactor.
51
+ # @deferred = deferred
51
52
 
52
53
  @writing = Async::Semaphore.new(1)
53
54
 
@@ -148,23 +149,19 @@ module Async
148
149
  end
149
150
 
150
151
  # Flushes buffered data to the stream.
151
- def flush(deferred: @deferred)
152
- if deferred and task = Task.current?
153
- # Despite how it looks, this field is not actually directly related to whether or not writes should occur. It's actually used to control the logic of deferred flushing. Therefore, it should NOT be modified outside this method.
154
- @pending += 1
152
+ def flush
153
+ return if @write_buffer.empty?
154
+
155
+ @writing.acquire do
156
+ # Flip the write buffer and drain buffer:
157
+ @write_buffer, @drain_buffer = @drain_buffer, @write_buffer
155
158
 
156
- if @pending == 1
157
- task.yield
158
-
159
- begin
160
- drain_write_buffer unless @write_buffer.empty?
161
- ensure
162
- # The write buffer no longer contains pending writes
163
- @pending = 0
164
- end
159
+ begin
160
+ @io.write(@drain_buffer)
161
+ ensure
162
+ # If the write operation fails, we still need to clear this buffer, and the data is essentially lost.
163
+ @drain_buffer.clear
165
164
  end
166
- else
167
- drain_write_buffer unless @write_buffer.empty?
168
165
  end
169
166
  end
170
167
 
@@ -210,7 +207,7 @@ module Async
210
207
  end
211
208
 
212
209
  def close_write
213
- drain_write_buffer unless @write_buffer.empty?
210
+ flush
214
211
  ensure
215
212
  @io.close_write
216
213
  end
@@ -220,7 +217,7 @@ module Async
220
217
  return if @io.closed?
221
218
 
222
219
  begin
223
- drain_write_buffer unless @write_buffer.empty?
220
+ flush
224
221
  rescue
225
222
  # We really can't do anything here unless we want #close to raise exceptions.
226
223
  ensure
@@ -250,32 +247,6 @@ module Async
250
247
 
251
248
  private
252
249
 
253
- # For efficiency purposes, only call this method when the write buffer is not empty.
254
- def drain_write_buffer
255
- @writing.acquire do
256
- Async.logger.debug(self) do |buffer|
257
- if @pending > 0
258
- buffer.puts "Draining #{@pending} writes (#{@write_buffer.bytesize} bytes)..."
259
- else
260
- buffer.puts "Draining immediate write (#{@write_buffer.bytesize} bytes)..."
261
- end
262
-
263
- # buffer.puts "@write_buffer = #{@write_buffer.inspect}"
264
- # buffer.puts "@drain_buffer = #{@drain_buffer.inspect}"
265
- end
266
-
267
- # Flip the write buffer and drain buffer:
268
- @write_buffer, @drain_buffer = @drain_buffer, @write_buffer
269
-
270
- begin
271
- @io.write(@drain_buffer)
272
- ensure
273
- # If the write operation fails, we still need to clear this buffer, and the data is essentially lost.
274
- @drain_buffer.clear
275
- end
276
- end
277
- end
278
-
279
250
  # Fills the buffer from the underlying stream.
280
251
  def fill_read_buffer(size = @block_size)
281
252
  # We impose a limit because the underlying `read` system call can fail if we request too much data in one go.
@@ -284,7 +255,7 @@ module Async
284
255
  end
285
256
 
286
257
  # This effectively ties the input and output stream together.
287
- drain_write_buffer unless @write_buffer.empty?
258
+ flush
288
259
 
289
260
  if @read_buffer.empty?
290
261
  if @io.read_nonblock(size, @read_buffer, exception: false)
@@ -1,6 +1,6 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- # Copyright, 2017, by Samuel G. D. Williams. <http://www.codeotaku.com>
3
+ # Copyright, 2020, by Samuel G. D. Williams. <http://www.codeotaku.com>
4
4
  #
5
5
  # Permission is hereby granted, free of charge, to any person obtaining a copy
6
6
  # of this software and associated documentation files (the "Software"), to deal
@@ -20,44 +20,64 @@
20
20
  # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
21
21
  # THE SOFTWARE.
22
22
 
23
- require 'async/io/udp_socket'
23
+ require_relative 'notification'
24
24
 
25
- RSpec.describe Async::IO::Socket do
26
- include_context Async::RSpec::Reactor
27
-
28
- # Shared port for localhost network tests.
29
- let(:server_address) {Async::IO::Address.udp("127.0.0.1", 6778)}
30
- let(:data) {"The quick brown fox jumped over the lazy dog."}
31
-
32
- let!(:server_task) do
33
- reactor.async do
34
- Async::IO::Socket.bind(server_address) do |server|
35
- packet, address = server.recvfrom(512)
36
-
37
- server.send(packet, 0, address)
25
+ module Async
26
+ module IO
27
+ class Threads
28
+ def initialize(parent: nil)
29
+ @parent = parent
38
30
  end
39
- end
40
- end
41
-
42
- describe 'basic udp server' do
43
- it "should echo data back to peer" do
44
- reactor.async do
45
- Async::IO::Socket.connect(server_address) do |client|
46
- client.send(data)
47
- response = client.recv(512)
48
-
49
- expect(response).to be == data
31
+
32
+ if Async::Scheduler.supported?
33
+ def async(parent: (@parent or Task.current))
34
+ parent.async do
35
+ thread = ::Thread.new do
36
+ yield
37
+ end
38
+
39
+ thread.join
40
+ rescue Stop
41
+ if thread&.alive?
42
+ thread.raise(Stop)
43
+ end
44
+
45
+ begin
46
+ thread.join
47
+ rescue Stop
48
+ # Ignore.
49
+ end
50
+ end
50
51
  end
51
- end
52
- end
53
-
54
- it "should use unconnected socket" do
55
- reactor.async do
56
- Async::IO::UDPSocket.wrap(server_address.afamily) do |client|
57
- client.send(data, 0, server_address)
58
- response, address = client.recvfrom(512)
59
-
60
- expect(response).to be == data
52
+ else
53
+ def async(parent: (@parent or Task.current))
54
+ parent.async do |task|
55
+ notification = Async::IO::Notification.new
56
+
57
+ thread = ::Thread.new do
58
+ yield
59
+ ensure
60
+ notification.signal
61
+ end
62
+
63
+ task.annotate "Waiting for thread to finish..."
64
+
65
+ notification.wait
66
+
67
+ thread.value
68
+ ensure
69
+ if thread&.alive?
70
+ thread.raise(Stop)
71
+
72
+ begin
73
+ thread.join
74
+ rescue Stop
75
+ # Ignore.
76
+ end
77
+ end
78
+
79
+ notification&.close
80
+ end
61
81
  end
62
82
  end
63
83
  end
data/lib/async/io/trap.rb CHANGED
@@ -36,6 +36,10 @@ module Async
36
36
  @mutex = Mutex.new
37
37
  end
38
38
 
39
+ def to_s
40
+ "\#<#{self.class} #{@name}>"
41
+ end
42
+
39
43
  # Ignore the trap within the current process. Can be invoked before forking and/or invoking `install!` to assert default behaviour.
40
44
  def ignore!
41
45
  Signal.trap(@name, :IGNORE)
@@ -87,9 +91,10 @@ module Async
87
91
  # Deprecated.
88
92
  alias trap wait
89
93
 
90
- def async(parent: Task.current, &block)
91
- parent.async do |task|
92
- self.trap(task: task, &block)
94
+ # In order to avoid blocking the reactor, specify `transient: true` as an option.
95
+ def async(parent: Task.current, **options, &block)
96
+ parent.async(**options) do |task|
97
+ self.wait(task: task, &block)
93
98
  end
94
99
  end
95
100
 
@@ -22,6 +22,6 @@
22
22
 
23
23
  module Async
24
24
  module IO
25
- VERSION = "1.27.7"
25
+ VERSION = "1.30.2"
26
26
  end
27
27
  end
metadata CHANGED
@@ -1,14 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: async-io
3
3
  version: !ruby/object:Gem::Version
4
- version: 1.27.7
4
+ version: 1.30.2
5
5
  platform: ruby
6
6
  authors:
7
7
  - Samuel Williams
8
- autorequire:
8
+ autorequire:
9
9
  bindir: bin
10
10
  cert_chain: []
11
- date: 2020-03-30 00:00:00.000000000 Z
11
+ date: 2021-02-09 00:00:00.000000000 Z
12
12
  dependencies:
13
13
  - !ruby/object:Gem::Dependency
14
14
  name: async
@@ -24,6 +24,20 @@ dependencies:
24
24
  - - "~>"
25
25
  - !ruby/object:Gem::Version
26
26
  version: '1.14'
27
+ - !ruby/object:Gem::Dependency
28
+ name: async-container
29
+ requirement: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - "~>"
32
+ - !ruby/object:Gem::Version
33
+ version: '0.15'
34
+ type: :development
35
+ prerelease: false
36
+ version_requirements: !ruby/object:Gem::Requirement
37
+ requirements:
38
+ - - "~>"
39
+ - !ruby/object:Gem::Version
40
+ version: '0.15'
27
41
  - !ruby/object:Gem::Dependency
28
42
  name: async-rspec
29
43
  requirement: !ruby/object:Gem::Requirement
@@ -39,21 +53,35 @@ dependencies:
39
53
  - !ruby/object:Gem::Version
40
54
  version: '1.10'
41
55
  - !ruby/object:Gem::Dependency
42
- name: async-container
56
+ name: bake
43
57
  requirement: !ruby/object:Gem::Requirement
44
58
  requirements:
45
- - - "~>"
59
+ - - ">="
46
60
  - !ruby/object:Gem::Version
47
- version: '0.15'
61
+ version: '0'
48
62
  type: :development
49
63
  prerelease: false
50
64
  version_requirements: !ruby/object:Gem::Requirement
51
65
  requirements:
52
- - - "~>"
66
+ - - ">="
53
67
  - !ruby/object:Gem::Version
54
- version: '0.15'
68
+ version: '0'
55
69
  - !ruby/object:Gem::Dependency
56
- name: covered
70
+ name: bake-bundler
71
+ requirement: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: '0'
76
+ type: :development
77
+ prerelease: false
78
+ version_requirements: !ruby/object:Gem::Requirement
79
+ requirements:
80
+ - - ">="
81
+ - !ruby/object:Gem::Version
82
+ version: '0'
83
+ - !ruby/object:Gem::Dependency
84
+ name: bake-modernize
57
85
  requirement: !ruby/object:Gem::Requirement
58
86
  requirements:
59
87
  - - ">="
@@ -81,19 +109,19 @@ dependencies:
81
109
  - !ruby/object:Gem::Version
82
110
  version: '0'
83
111
  - !ruby/object:Gem::Dependency
84
- name: rake
112
+ name: covered
85
113
  requirement: !ruby/object:Gem::Requirement
86
114
  requirements:
87
- - - "~>"
115
+ - - ">="
88
116
  - !ruby/object:Gem::Version
89
- version: '10.0'
117
+ version: '0'
90
118
  type: :development
91
119
  prerelease: false
92
120
  version_requirements: !ruby/object:Gem::Requirement
93
121
  requirements:
94
- - - "~>"
122
+ - - ">="
95
123
  - !ruby/object:Gem::Version
96
- version: '10.0'
124
+ version: '0'
97
125
  - !ruby/object:Gem::Dependency
98
126
  name: rspec
99
127
  requirement: !ruby/object:Gem::Requirement
@@ -108,36 +136,12 @@ dependencies:
108
136
  - - "~>"
109
137
  - !ruby/object:Gem::Version
110
138
  version: '3.0'
111
- description:
139
+ description:
112
140
  email:
113
- - samuel.williams@oriontransfer.co.nz
114
141
  executables: []
115
142
  extensions: []
116
143
  extra_rdoc_files: []
117
144
  files:
118
- - ".editorconfig"
119
- - ".gitignore"
120
- - ".rspec"
121
- - ".travis.yml"
122
- - ".yardopts"
123
- - Gemfile
124
- - README.md
125
- - async-io.gemspec
126
- - examples/allocations/byteslice.rb
127
- - examples/allocations/memory.rb
128
- - examples/allocations/read_chunks.rb
129
- - examples/chat/client.rb
130
- - examples/chat/server.rb
131
- - examples/defer/worker.rb
132
- - examples/echo/client.rb
133
- - examples/echo/server.rb
134
- - examples/issues/broken_ssl.rb
135
- - examples/issues/pipes.rb
136
- - examples/millions/client.rb
137
- - examples/millions/server.rb
138
- - examples/udp/client.rb
139
- - examples/udp/server.rb
140
- - gems/nio4r-2.3.gemfile
141
145
  - lib/async/io.rb
142
146
  - lib/async/io/address.rb
143
147
  - lib/async/io/address_endpoint.rb
@@ -160,86 +164,33 @@ files:
160
164
  - lib/async/io/standard.rb
161
165
  - lib/async/io/stream.rb
162
166
  - lib/async/io/tcp_socket.rb
167
+ - lib/async/io/threads.rb
163
168
  - lib/async/io/trap.rb
164
169
  - lib/async/io/udp_socket.rb
165
170
  - lib/async/io/unix_endpoint.rb
166
171
  - lib/async/io/unix_socket.rb
167
172
  - lib/async/io/version.rb
168
- - spec/addrinfo.rb
169
- - spec/async/io/buffer_spec.rb
170
- - spec/async/io/c10k_spec.rb
171
- - spec/async/io/echo_spec.rb
172
- - spec/async/io/endpoint_spec.rb
173
- - spec/async/io/generic_examples.rb
174
- - spec/async/io/generic_spec.rb
175
- - spec/async/io/notification_spec.rb
176
- - spec/async/io/protocol/line_spec.rb
177
- - spec/async/io/shared_endpoint/server_spec.rb
178
- - spec/async/io/shared_endpoint_spec.rb
179
- - spec/async/io/socket/tcp_spec.rb
180
- - spec/async/io/socket/udp_spec.rb
181
- - spec/async/io/socket_spec.rb
182
- - spec/async/io/ssl_server_spec.rb
183
- - spec/async/io/ssl_socket_spec.rb
184
- - spec/async/io/standard_spec.rb
185
- - spec/async/io/stream_context.rb
186
- - spec/async/io/stream_spec.rb
187
- - spec/async/io/tcp_socket_spec.rb
188
- - spec/async/io/trap_spec.rb
189
- - spec/async/io/udp_socket_spec.rb
190
- - spec/async/io/unix_endpoint_spec.rb
191
- - spec/async/io/unix_socket_spec.rb
192
- - spec/async/io/wrap/http_rb_spec.rb
193
- - spec/async/io/wrap/tcp_spec.rb
194
- - spec/spec_helper.rb
195
173
  homepage: https://github.com/socketry/async-io
196
174
  licenses:
197
175
  - MIT
198
176
  metadata: {}
199
- post_install_message:
177
+ post_install_message:
200
178
  rdoc_options: []
201
179
  require_paths:
202
180
  - lib
203
181
  required_ruby_version: !ruby/object:Gem::Requirement
204
182
  requirements:
205
- - - "~>"
183
+ - - ">="
206
184
  - !ruby/object:Gem::Version
207
- version: '2.4'
185
+ version: '2.5'
208
186
  required_rubygems_version: !ruby/object:Gem::Requirement
209
187
  requirements:
210
188
  - - ">="
211
189
  - !ruby/object:Gem::Version
212
190
  version: '0'
213
191
  requirements: []
214
- rubygems_version: 3.1.2
215
- signing_key:
192
+ rubygems_version: 3.2.3
193
+ signing_key:
216
194
  specification_version: 4
217
195
  summary: Provides support for asynchonous TCP, UDP, UNIX and SSL sockets.
218
- test_files:
219
- - spec/addrinfo.rb
220
- - spec/async/io/buffer_spec.rb
221
- - spec/async/io/c10k_spec.rb
222
- - spec/async/io/echo_spec.rb
223
- - spec/async/io/endpoint_spec.rb
224
- - spec/async/io/generic_examples.rb
225
- - spec/async/io/generic_spec.rb
226
- - spec/async/io/notification_spec.rb
227
- - spec/async/io/protocol/line_spec.rb
228
- - spec/async/io/shared_endpoint/server_spec.rb
229
- - spec/async/io/shared_endpoint_spec.rb
230
- - spec/async/io/socket/tcp_spec.rb
231
- - spec/async/io/socket/udp_spec.rb
232
- - spec/async/io/socket_spec.rb
233
- - spec/async/io/ssl_server_spec.rb
234
- - spec/async/io/ssl_socket_spec.rb
235
- - spec/async/io/standard_spec.rb
236
- - spec/async/io/stream_context.rb
237
- - spec/async/io/stream_spec.rb
238
- - spec/async/io/tcp_socket_spec.rb
239
- - spec/async/io/trap_spec.rb
240
- - spec/async/io/udp_socket_spec.rb
241
- - spec/async/io/unix_endpoint_spec.rb
242
- - spec/async/io/unix_socket_spec.rb
243
- - spec/async/io/wrap/http_rb_spec.rb
244
- - spec/async/io/wrap/tcp_spec.rb
245
- - spec/spec_helper.rb
196
+ test_files: []