io-stream 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- checksums.yaml.gz.sig +0 -0
- data/lib/io/buffered.rb +63 -0
- data/lib/io/readable.rb +53 -0
- data/lib/io/stream/buffered_stream.rb +333 -0
- data/lib/io/stream/string_buffer.rb +28 -0
- data/lib/io/stream/version.rb +8 -0
- data/lib/io/stream.rb +12 -0
- data/license.md +21 -0
- data/readme.md +31 -0
- data.tar.gz.sig +0 -0
- metadata +81 -0
- metadata.gz.sig +1 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 259533ca29c9be8e0476777d9060a4aa6928fa856848396110b3f668628db636
|
4
|
+
data.tar.gz: ed074a9ff19c5f0f74223228d1bb9929ff7534f921c11e4adf34fe88c5f2824b
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: f90e4bca685926a8cc28b21b65ddceea17fd8d3ba25ccefd6daf9095403ef7446c71b9afdfe57367faccf2ee46635bcf939e14f8ac36e9ebc6b9ee55bdafaada
|
7
|
+
data.tar.gz: f7aaa56fb744df8cd057412083ad63c707165e3e1c404397dfa447f2135f7d11fef6d0b855cf88c23d28346ba96c320bc9566e7395520889d89ea7b63ea4fd1c
|
checksums.yaml.gz.sig
ADDED
Binary file
|
data/lib/io/buffered.rb
ADDED
@@ -0,0 +1,63 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Released under the MIT License.
|
4
|
+
# Copyright, 2023-2024, by Samuel Williams.
|
5
|
+
|
6
|
+
unless IO.method_defined?(:buffered?)
|
7
|
+
class IO
|
8
|
+
def buffered?
|
9
|
+
return !self.sync
|
10
|
+
end
|
11
|
+
|
12
|
+
def buffered=(value)
|
13
|
+
self.sync = !value
|
14
|
+
end
|
15
|
+
end
|
16
|
+
end
|
17
|
+
|
18
|
+
require 'socket'
|
19
|
+
|
20
|
+
unless BasicSocket.method_defined?(:buffered?)
|
21
|
+
class BasicSocket
|
22
|
+
# Is it likely that the socket is still connected?
|
23
|
+
# May return false positive, but won't return false negative.
|
24
|
+
def buffered?
|
25
|
+
return false unless super
|
26
|
+
|
27
|
+
case self.local_address.protocol
|
28
|
+
when IPPROTO_TCP
|
29
|
+
return !self.getsockopt(IPPROTO_TCP, TCP_NODELAY).bool
|
30
|
+
else
|
31
|
+
return true
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
def buffered=(value)
|
36
|
+
super
|
37
|
+
|
38
|
+
case self.local_address.socktype
|
39
|
+
when SOCK_STREAM
|
40
|
+
# When buffered is set to true, TCP_NODELAY shold be disabled.
|
41
|
+
self.setsockopt(IPPROTO_TCP, TCP_NODELAY, value ? 0 : 1)
|
42
|
+
end
|
43
|
+
rescue Errno::EINVAL
|
44
|
+
# On Darwin, sometimes occurs when the connection is not yet fully formed. Empirically, TCP_NODELAY is enabled despite this result.
|
45
|
+
rescue Errno::EOPNOTSUPP
|
46
|
+
# Some platforms may simply not support the operation.
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|
50
|
+
|
51
|
+
require 'stringio'
|
52
|
+
|
53
|
+
unless StringIO.method_defined?(:buffered?)
|
54
|
+
class StringIO
|
55
|
+
def buffered?
|
56
|
+
return !self.sync
|
57
|
+
end
|
58
|
+
|
59
|
+
def buffered=(value)
|
60
|
+
self.sync = !value
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
data/lib/io/readable.rb
ADDED
@@ -0,0 +1,53 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Released under the MIT License.
|
4
|
+
# Copyright, 2023-2024, by Samuel Williams.
|
5
|
+
|
6
|
+
class IO
|
7
|
+
unless method_defined?(:readable?)
|
8
|
+
def readable?
|
9
|
+
# Do not call `eof?` here as it is not concurrency-safe and it can block.
|
10
|
+
!closed?
|
11
|
+
end
|
12
|
+
end
|
13
|
+
end
|
14
|
+
|
15
|
+
require 'socket'
|
16
|
+
|
17
|
+
class BasicSocket
|
18
|
+
unless method_defined?(:readable?)
|
19
|
+
def readable?
|
20
|
+
# If we can wait for the socket to become readable, we know that the socket may still be open.
|
21
|
+
result = self.recv_nonblock(1, MSG_PEEK, exception: false)
|
22
|
+
|
23
|
+
# No data was available - newer Ruby can return nil instead of empty string:
|
24
|
+
return false if result.nil?
|
25
|
+
|
26
|
+
# Either there was some data available, or we can wait to see if there is data avaialble.
|
27
|
+
return !result.empty? || result == :wait_readable
|
28
|
+
rescue Errno::ECONNRESET, IOError
|
29
|
+
# This might be thrown by recv_nonblock.
|
30
|
+
return false
|
31
|
+
end
|
32
|
+
end
|
33
|
+
end
|
34
|
+
|
35
|
+
require 'stringio'
|
36
|
+
|
37
|
+
class StringIO
|
38
|
+
unless method_defined?(:readable?)
|
39
|
+
def readable?
|
40
|
+
!eof?
|
41
|
+
end
|
42
|
+
end
|
43
|
+
end
|
44
|
+
|
45
|
+
require 'openssl'
|
46
|
+
|
47
|
+
class OpenSSL::SSL::SSLSocket
|
48
|
+
unless method_defined?(:readable?)
|
49
|
+
def readable?
|
50
|
+
to_io.readable?
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
@@ -0,0 +1,333 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Released under the MIT License.
|
4
|
+
# Copyright, 2023-2024, by Samuel Williams.
|
5
|
+
|
6
|
+
require_relative 'string_buffer'
|
7
|
+
|
8
|
+
require_relative '../buffered'
|
9
|
+
require_relative '../readable'
|
10
|
+
|
11
|
+
module IO::Stream
|
12
|
+
# The default block size for IO buffers. Defaults to 64KB (typical pipe buffer size).
|
13
|
+
BLOCK_SIZE = ENV.fetch('IO_STREAM_BLOCK_SIZE', 1024*64).to_i
|
14
|
+
|
15
|
+
# The maximum read size when appending to IO buffers. Defaults to 8MB.
|
16
|
+
MAXIMUM_READ_SIZE = ENV.fetch('IO_STREAM_MAXIMUM_READ_SIZE', BLOCK_SIZE * 128).to_i
|
17
|
+
|
18
|
+
class BufferedStream
|
19
|
+
def self.open(path, mode = "r+", **options)
|
20
|
+
stream = self.new(File.open(path, mode), **options)
|
21
|
+
|
22
|
+
return stream unless block_given?
|
23
|
+
|
24
|
+
begin
|
25
|
+
yield stream
|
26
|
+
ensure
|
27
|
+
stream.close
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
def self.wrap(io, **options)
|
32
|
+
if io.respond_to?(:buffered=)
|
33
|
+
io.buffered = false
|
34
|
+
end
|
35
|
+
|
36
|
+
self.new(io, **options)
|
37
|
+
end
|
38
|
+
|
39
|
+
def initialize(io, block_size: BLOCK_SIZE, maximum_read_size: MAXIMUM_READ_SIZE)
|
40
|
+
@io = io
|
41
|
+
@eof = false
|
42
|
+
|
43
|
+
@writing = Thread::Mutex.new
|
44
|
+
|
45
|
+
@block_size = block_size
|
46
|
+
@maximum_read_size = maximum_read_size
|
47
|
+
|
48
|
+
@read_buffer = StringBuffer.new
|
49
|
+
@write_buffer = StringBuffer.new
|
50
|
+
@drain_buffer = StringBuffer.new
|
51
|
+
|
52
|
+
# Used as destination buffer for underlying reads.
|
53
|
+
@input_buffer = StringBuffer.new
|
54
|
+
end
|
55
|
+
|
56
|
+
attr :io
|
57
|
+
|
58
|
+
attr :block_size
|
59
|
+
|
60
|
+
# Reads `size` bytes from the stream. If size is not specified, read until end of file.
|
61
|
+
def read(size = nil)
|
62
|
+
return String.new(encoding: Encoding::BINARY) if size == 0
|
63
|
+
|
64
|
+
if size
|
65
|
+
until @eof or @read_buffer.bytesize >= size
|
66
|
+
# Compute the amount of data we need to read from the underlying stream:
|
67
|
+
read_size = size - @read_buffer.bytesize
|
68
|
+
|
69
|
+
# Don't read less than @block_size to avoid lots of small reads:
|
70
|
+
fill_read_buffer(read_size > @block_size ? read_size : @block_size)
|
71
|
+
end
|
72
|
+
else
|
73
|
+
until @eof
|
74
|
+
fill_read_buffer
|
75
|
+
end
|
76
|
+
end
|
77
|
+
|
78
|
+
return consume_read_buffer(size)
|
79
|
+
end
|
80
|
+
|
81
|
+
# Read at most `size` bytes from the stream. Will avoid reading from the underlying stream if possible.
|
82
|
+
def read_partial(size = nil)
|
83
|
+
return String.new(encoding: Encoding::BINARY) if size == 0
|
84
|
+
|
85
|
+
if !@eof and @read_buffer.empty?
|
86
|
+
fill_read_buffer
|
87
|
+
end
|
88
|
+
|
89
|
+
return consume_read_buffer(size)
|
90
|
+
end
|
91
|
+
|
92
|
+
def read_exactly(size, exception: EOFError)
|
93
|
+
if buffer = read(size)
|
94
|
+
if buffer.bytesize != size
|
95
|
+
raise exception, "could not read enough data"
|
96
|
+
end
|
97
|
+
|
98
|
+
return buffer
|
99
|
+
end
|
100
|
+
|
101
|
+
raise exception, "encountered eof while reading data"
|
102
|
+
end
|
103
|
+
|
104
|
+
# This is a compatibility shim for existing code:
|
105
|
+
def readpartial(size = nil)
|
106
|
+
read_partial(size) or raise EOFError, "Encountered eof while reading data!"
|
107
|
+
end
|
108
|
+
|
109
|
+
# Efficiently read data from the stream until encountering pattern.
|
110
|
+
# @param pattern [String] The pattern to match.
|
111
|
+
# @return [String] The contents of the stream up until the pattern, which is consumed but not returned.
|
112
|
+
def read_until(pattern, offset = 0, chomp: true)
|
113
|
+
# We don't want to split on the pattern, so we subtract the size of the pattern.
|
114
|
+
split_offset = pattern.bytesize - 1
|
115
|
+
|
116
|
+
until index = @read_buffer.index(pattern, offset)
|
117
|
+
offset = @read_buffer.bytesize - split_offset
|
118
|
+
|
119
|
+
offset = 0 if offset < 0
|
120
|
+
|
121
|
+
return unless fill_read_buffer
|
122
|
+
end
|
123
|
+
|
124
|
+
@read_buffer.freeze
|
125
|
+
matched = @read_buffer.byteslice(0, index+(chomp ? 0 : pattern.bytesize))
|
126
|
+
@read_buffer = @read_buffer.byteslice(index+pattern.bytesize, @read_buffer.bytesize)
|
127
|
+
|
128
|
+
return matched
|
129
|
+
end
|
130
|
+
|
131
|
+
def peek(size = nil)
|
132
|
+
if size
|
133
|
+
until @eof or @read_buffer.bytesize >= size
|
134
|
+
# Compute the amount of data we need to read from the underlying stream:
|
135
|
+
read_size = size - @read_buffer.bytesize
|
136
|
+
|
137
|
+
# Don't read less than @block_size to avoid lots of small reads:
|
138
|
+
fill_read_buffer(read_size > @block_size ? read_size : @block_size)
|
139
|
+
end
|
140
|
+
return @read_buffer[..([size, @read_buffer.size].min - 1)]
|
141
|
+
end
|
142
|
+
until (block_given? && yield(@read_buffer)) or @eof
|
143
|
+
fill_read_buffer
|
144
|
+
end
|
145
|
+
return @read_buffer
|
146
|
+
end
|
147
|
+
|
148
|
+
def gets(separator = $/, **options)
|
149
|
+
read_until(separator, **options)
|
150
|
+
end
|
151
|
+
|
152
|
+
# Flushes buffered data to the stream.
|
153
|
+
def flush
|
154
|
+
return if @write_buffer.empty?
|
155
|
+
|
156
|
+
@writing.synchronize do
|
157
|
+
# Flip the write buffer and drain buffer:
|
158
|
+
@write_buffer, @drain_buffer = @drain_buffer, @write_buffer
|
159
|
+
|
160
|
+
begin
|
161
|
+
@io.syswrite(@drain_buffer)
|
162
|
+
ensure
|
163
|
+
# If the write operation fails, we still need to clear this buffer, and the data is essentially lost.
|
164
|
+
@drain_buffer.clear
|
165
|
+
end
|
166
|
+
end
|
167
|
+
end
|
168
|
+
|
169
|
+
# Writes `string` to the buffer. When the buffer is full or #sync is true the
|
170
|
+
# buffer is flushed to the underlying `io`.
|
171
|
+
# @param string the string to write to the buffer.
|
172
|
+
# @return the number of bytes appended to the buffer.
|
173
|
+
def write(string)
|
174
|
+
@write_buffer << string
|
175
|
+
|
176
|
+
if @write_buffer.bytesize >= @block_size
|
177
|
+
flush
|
178
|
+
end
|
179
|
+
|
180
|
+
return string.bytesize
|
181
|
+
end
|
182
|
+
|
183
|
+
# Writes `string` to the stream and returns self.
|
184
|
+
def <<(string)
|
185
|
+
write(string)
|
186
|
+
|
187
|
+
return self
|
188
|
+
end
|
189
|
+
|
190
|
+
def puts(*arguments, separator: $/)
|
191
|
+
arguments.each do |argument|
|
192
|
+
@write_buffer << argument << separator
|
193
|
+
end
|
194
|
+
|
195
|
+
flush
|
196
|
+
end
|
197
|
+
|
198
|
+
def connected?
|
199
|
+
@io.connected?
|
200
|
+
end
|
201
|
+
|
202
|
+
def closed?
|
203
|
+
@io.closed?
|
204
|
+
end
|
205
|
+
|
206
|
+
def close_read
|
207
|
+
@io.close_read
|
208
|
+
end
|
209
|
+
|
210
|
+
def close_write
|
211
|
+
flush
|
212
|
+
ensure
|
213
|
+
@io.close_write
|
214
|
+
end
|
215
|
+
|
216
|
+
# Best effort to flush any unwritten data, and then close the underling IO.
|
217
|
+
def close
|
218
|
+
return if @io.closed?
|
219
|
+
|
220
|
+
begin
|
221
|
+
flush
|
222
|
+
rescue
|
223
|
+
# We really can't do anything here unless we want #close to raise exceptions.
|
224
|
+
ensure
|
225
|
+
@io.close
|
226
|
+
end
|
227
|
+
end
|
228
|
+
|
229
|
+
# Determins if the stream has consumed all available data. May block if the stream is not readable.
|
230
|
+
# See {readable?} for a non-blocking alternative.
|
231
|
+
#
|
232
|
+
# @returns [Boolean] If the stream is at file which means there is no more data to be read.
|
233
|
+
def eof?
|
234
|
+
if !@read_buffer.empty?
|
235
|
+
return false
|
236
|
+
elsif @eof
|
237
|
+
return true
|
238
|
+
else
|
239
|
+
return !self.fill_read_buffer
|
240
|
+
end
|
241
|
+
end
|
242
|
+
|
243
|
+
def eof!
|
244
|
+
@read_buffer.clear
|
245
|
+
@eof = true
|
246
|
+
|
247
|
+
raise EOFError
|
248
|
+
end
|
249
|
+
|
250
|
+
# Whether there is a chance that a read operation will succeed or not.
|
251
|
+
# @returns [Boolean] If the stream is readable, i.e. a `read` operation has a chance of success.
|
252
|
+
def readable?
|
253
|
+
# If we are at the end of the file, we can't read any more data:
|
254
|
+
if @eof
|
255
|
+
return false
|
256
|
+
end
|
257
|
+
|
258
|
+
# If the read buffer is not empty, we can read more data:
|
259
|
+
if !@read_buffer.empty?
|
260
|
+
return true
|
261
|
+
end
|
262
|
+
|
263
|
+
# If the underlying stream is readable, we can read more data:
|
264
|
+
return @io.readable?
|
265
|
+
end
|
266
|
+
|
267
|
+
private
|
268
|
+
|
269
|
+
# Reads data from the underlying stream as efficiently as possible.
|
270
|
+
def sysread(size, buffer)
|
271
|
+
# Come on Ruby, why couldn't this just return `nil`? EOF is not exceptional. Every file has one.
|
272
|
+
@io.sysread(size, buffer)
|
273
|
+
rescue EOFError
|
274
|
+
return false
|
275
|
+
end
|
276
|
+
|
277
|
+
# Fills the buffer from the underlying stream.
|
278
|
+
def fill_read_buffer(size = @block_size)
|
279
|
+
# We impose a limit because the underlying `read` system call can fail if we request too much data in one go.
|
280
|
+
if size > @maximum_read_size
|
281
|
+
size = @maximum_read_size
|
282
|
+
end
|
283
|
+
|
284
|
+
# This effectively ties the input and output stream together.
|
285
|
+
flush
|
286
|
+
|
287
|
+
if @read_buffer.empty?
|
288
|
+
if sysread(size, @read_buffer)
|
289
|
+
# Console.logger.debug(self, name: "read") {@read_buffer.inspect}
|
290
|
+
return true
|
291
|
+
end
|
292
|
+
else
|
293
|
+
if chunk = sysread(size, @input_buffer)
|
294
|
+
@read_buffer << chunk
|
295
|
+
# Console.logger.debug(self, name: "read") {@read_buffer.inspect}
|
296
|
+
|
297
|
+
return true
|
298
|
+
end
|
299
|
+
end
|
300
|
+
|
301
|
+
# else for both cases above:
|
302
|
+
@eof = true
|
303
|
+
return false
|
304
|
+
end
|
305
|
+
|
306
|
+
# Consumes at most `size` bytes from the buffer.
|
307
|
+
# @param size [Integer|nil] The amount of data to consume. If nil, consume entire buffer.
|
308
|
+
def consume_read_buffer(size = nil)
|
309
|
+
# If we are at eof, and the read buffer is empty, we can't consume anything.
|
310
|
+
return nil if @eof && @read_buffer.empty?
|
311
|
+
|
312
|
+
result = nil
|
313
|
+
|
314
|
+
if size.nil? or size >= @read_buffer.bytesize
|
315
|
+
# Consume the entire read buffer:
|
316
|
+
result = @read_buffer
|
317
|
+
@read_buffer = StringBuffer.new
|
318
|
+
else
|
319
|
+
# This approach uses more memory.
|
320
|
+
# result = @read_buffer.slice!(0, size)
|
321
|
+
|
322
|
+
# We know that we are not going to reuse the original buffer.
|
323
|
+
# But byteslice will generate a hidden copy. So let's freeze it first:
|
324
|
+
@read_buffer.freeze
|
325
|
+
|
326
|
+
result = @read_buffer.byteslice(0, size)
|
327
|
+
@read_buffer = @read_buffer.byteslice(size, @read_buffer.bytesize)
|
328
|
+
end
|
329
|
+
|
330
|
+
return result
|
331
|
+
end
|
332
|
+
end
|
333
|
+
end
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Released under the MIT License.
|
4
|
+
# Copyright, 2023-2024, by Samuel Williams.
|
5
|
+
|
6
|
+
module IO::Stream
|
7
|
+
class StringBuffer < String
|
8
|
+
BINARY = Encoding::BINARY
|
9
|
+
|
10
|
+
def initialize
|
11
|
+
super
|
12
|
+
|
13
|
+
force_encoding(BINARY)
|
14
|
+
end
|
15
|
+
|
16
|
+
def << string
|
17
|
+
if string.encoding == BINARY
|
18
|
+
super(string)
|
19
|
+
else
|
20
|
+
super(string.b)
|
21
|
+
end
|
22
|
+
|
23
|
+
return self
|
24
|
+
end
|
25
|
+
|
26
|
+
alias concat <<
|
27
|
+
end
|
28
|
+
end
|
data/lib/io/stream.rb
ADDED
data/license.md
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
# MIT License
|
2
|
+
|
3
|
+
Copyright, 2023-2024, by Samuel Williams.
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
13
|
+
copies or substantial portions of the Software.
|
14
|
+
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
21
|
+
SOFTWARE.
|
data/readme.md
ADDED
@@ -0,0 +1,31 @@
|
|
1
|
+
# IO::Stream
|
2
|
+
|
3
|
+
Provide a buffered stream implementation for Ruby, independent of the underlying IO.
|
4
|
+
|
5
|
+
[](https://github.com/socketry/io-stream/actions?workflow=Test)
|
6
|
+
|
7
|
+
## Usage
|
8
|
+
|
9
|
+
Please see the [project documentation](https://socketry.github.io/io-stream) for more details.
|
10
|
+
|
11
|
+
## Contributing
|
12
|
+
|
13
|
+
We welcome contributions to this project.
|
14
|
+
|
15
|
+
1. Fork it.
|
16
|
+
2. Create your feature branch (`git checkout -b my-new-feature`).
|
17
|
+
3. Commit your changes (`git commit -am 'Add some feature'`).
|
18
|
+
4. Push to the branch (`git push origin my-new-feature`).
|
19
|
+
5. Create new Pull Request.
|
20
|
+
|
21
|
+
### Developer Certificate of Origin
|
22
|
+
|
23
|
+
This project uses the [Developer Certificate of Origin](https://developercertificate.org/). All contributors to this project must agree to this document to have their contributions accepted.
|
24
|
+
|
25
|
+
### Contributor Covenant
|
26
|
+
|
27
|
+
This project is governed by the [Contributor Covenant](https://www.contributor-covenant.org/). All contributors and participants agree to abide by its terms.
|
28
|
+
|
29
|
+
## See Also
|
30
|
+
|
31
|
+
- [async-io](https://github.com/socketry/async-io) — Where this implementation originally came from.
|
data.tar.gz.sig
ADDED
Binary file
|
metadata
ADDED
@@ -0,0 +1,81 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: io-stream
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.1
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Samuel Williams
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain:
|
11
|
+
- |
|
12
|
+
-----BEGIN CERTIFICATE-----
|
13
|
+
MIIE2DCCA0CgAwIBAgIBATANBgkqhkiG9w0BAQsFADBhMRgwFgYDVQQDDA9zYW11
|
14
|
+
ZWwud2lsbGlhbXMxHTAbBgoJkiaJk/IsZAEZFg1vcmlvbnRyYW5zZmVyMRIwEAYK
|
15
|
+
CZImiZPyLGQBGRYCY28xEjAQBgoJkiaJk/IsZAEZFgJuejAeFw0yMjA4MDYwNDUz
|
16
|
+
MjRaFw0zMjA4MDMwNDUzMjRaMGExGDAWBgNVBAMMD3NhbXVlbC53aWxsaWFtczEd
|
17
|
+
MBsGCgmSJomT8ixkARkWDW9yaW9udHJhbnNmZXIxEjAQBgoJkiaJk/IsZAEZFgJj
|
18
|
+
bzESMBAGCgmSJomT8ixkARkWAm56MIIBojANBgkqhkiG9w0BAQEFAAOCAY8AMIIB
|
19
|
+
igKCAYEAomvSopQXQ24+9DBB6I6jxRI2auu3VVb4nOjmmHq7XWM4u3HL+pni63X2
|
20
|
+
9qZdoq9xt7H+RPbwL28LDpDNflYQXoOhoVhQ37Pjn9YDjl8/4/9xa9+NUpl9XDIW
|
21
|
+
sGkaOY0eqsQm1pEWkHJr3zn/fxoKPZPfaJOglovdxf7dgsHz67Xgd/ka+Wo1YqoE
|
22
|
+
e5AUKRwUuvaUaumAKgPH+4E4oiLXI4T1Ff5Q7xxv6yXvHuYtlMHhYfgNn8iiW8WN
|
23
|
+
XibYXPNP7NtieSQqwR/xM6IRSoyXKuS+ZNGDPUUGk8RoiV/xvVN4LrVm9upSc0ss
|
24
|
+
RZ6qwOQmXCo/lLcDUxJAgG95cPw//sI00tZan75VgsGzSWAOdjQpFM0l4dxvKwHn
|
25
|
+
tUeT3ZsAgt0JnGqNm2Bkz81kG4A2hSyFZTFA8vZGhp+hz+8Q573tAR89y9YJBdYM
|
26
|
+
zp0FM4zwMNEUwgfRzv1tEVVUEXmoFCyhzonUUw4nE4CFu/sE3ffhjKcXcY//qiSW
|
27
|
+
xm4erY3XAgMBAAGjgZowgZcwCQYDVR0TBAIwADALBgNVHQ8EBAMCBLAwHQYDVR0O
|
28
|
+
BBYEFO9t7XWuFf2SKLmuijgqR4sGDlRsMC4GA1UdEQQnMCWBI3NhbXVlbC53aWxs
|
29
|
+
aWFtc0BvcmlvbnRyYW5zZmVyLmNvLm56MC4GA1UdEgQnMCWBI3NhbXVlbC53aWxs
|
30
|
+
aWFtc0BvcmlvbnRyYW5zZmVyLmNvLm56MA0GCSqGSIb3DQEBCwUAA4IBgQB5sxkE
|
31
|
+
cBsSYwK6fYpM+hA5B5yZY2+L0Z+27jF1pWGgbhPH8/FjjBLVn+VFok3CDpRqwXCl
|
32
|
+
xCO40JEkKdznNy2avOMra6PFiQyOE74kCtv7P+Fdc+FhgqI5lMon6tt9rNeXmnW/
|
33
|
+
c1NaMRdxy999hmRGzUSFjozcCwxpy/LwabxtdXwXgSay4mQ32EDjqR1TixS1+smp
|
34
|
+
8C/NCWgpIfzpHGJsjvmH2wAfKtTTqB9CVKLCWEnCHyCaRVuKkrKjqhYCdmMBqCws
|
35
|
+
JkxfQWC+jBVeG9ZtPhQgZpfhvh+6hMhraUYRQ6XGyvBqEUe+yo6DKIT3MtGE2+CP
|
36
|
+
eX9i9ZWBydWb8/rvmwmX2kkcBbX0hZS1rcR593hGc61JR6lvkGYQ2MYskBveyaxt
|
37
|
+
Q2K9NVun/S785AP05vKkXZEFYxqG6EW012U4oLcFl5MySFajYXRYbuUpH6AY+HP8
|
38
|
+
voD0MPg1DssDLKwXyt1eKD/+Fq0bFWhwVM/1XiAXL7lyYUyOq24KHgQ2Csg=
|
39
|
+
-----END CERTIFICATE-----
|
40
|
+
date: 2024-04-22 00:00:00.000000000 Z
|
41
|
+
dependencies: []
|
42
|
+
description:
|
43
|
+
email:
|
44
|
+
executables: []
|
45
|
+
extensions: []
|
46
|
+
extra_rdoc_files: []
|
47
|
+
files:
|
48
|
+
- lib/io/buffered.rb
|
49
|
+
- lib/io/readable.rb
|
50
|
+
- lib/io/stream.rb
|
51
|
+
- lib/io/stream/buffered_stream.rb
|
52
|
+
- lib/io/stream/string_buffer.rb
|
53
|
+
- lib/io/stream/version.rb
|
54
|
+
- license.md
|
55
|
+
- readme.md
|
56
|
+
homepage: https://github.com/socketry/io-stream
|
57
|
+
licenses:
|
58
|
+
- MIT
|
59
|
+
metadata:
|
60
|
+
documentation_uri: https://socketry.github.io/io-stream
|
61
|
+
source_code_uri: https://github.com/socketry/io-stream.git
|
62
|
+
post_install_message:
|
63
|
+
rdoc_options: []
|
64
|
+
require_paths:
|
65
|
+
- lib
|
66
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
67
|
+
requirements:
|
68
|
+
- - ">="
|
69
|
+
- !ruby/object:Gem::Version
|
70
|
+
version: '3.1'
|
71
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
72
|
+
requirements:
|
73
|
+
- - ">="
|
74
|
+
- !ruby/object:Gem::Version
|
75
|
+
version: '0'
|
76
|
+
requirements: []
|
77
|
+
rubygems_version: 3.5.3
|
78
|
+
signing_key:
|
79
|
+
specification_version: 4
|
80
|
+
summary: Provides a generic stream wrapper for IO instances.
|
81
|
+
test_files: []
|
metadata.gz.sig
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
��f֩�����P�E�-z˶�oQ%�/*$(��Id��߄3$Jx�V�(;�$ewJ�x:��x�?︊w�����h����E���8�(��[�^}�u�ΙL�HZ1���!�g�t��z`�n��"�z�C��g��j=�4��L^/�NF�����pD�f�!���Y\�"٢q�s�ڀ4v[�|�~�m��[�0�+�`Y����هiV�V]ҽ�nT1�ۻ��e���ɓNX��C�P��&��nYJ�`q��b��� U��{6�ӳ'*=�}� =��5[��\�NGHΦ�f>���y2y�U�c���X�:��%2Zχ���e�݅���N������ĪΨ3�+��˥6R˽p�Oܸ,�t����
|