ruby-core@ruby-lang.org archive (unofficial mirror)
 help / color / mirror / Atom feed
* [ruby-core:105660] [Ruby master Feature#18254] Add an `offset` parameter to String#unpack and String#unpack1
@ 2021-10-18  8:04 byroot (Jean Boussier)
  2021-10-18  8:24 ` [ruby-core:105661] " znz (Kazuhiro NISHIYAMA)
                   ` (11 more replies)
  0 siblings, 12 replies; 13+ messages in thread
From: byroot (Jean Boussier) @ 2021-10-18  8:04 UTC (permalink / raw)
  To: ruby-core

Issue #18254 has been reported by byroot (Jean Boussier).

----------------------------------------
Feature #18254: Add an `offset` parameter to String#unpack and String#unpack1
https://bugs.ruby-lang.org/issues/18254

* Author: byroot (Jean Boussier)
* Status: Open
* Priority: Normal
----------------------------------------
When working with binary protocols it's common to have to first unpack some kind of header or type prefix, and then based on that unpack another part of the string.

For instance here's [a code snippet from Dalli, the most common Memcached client](https://github.com/petergoldstein/dalli/blob/76b79d78cda13562da17bc99f92edcedf1873994/lib/dalli/protocol/binary.rb#L156-L184):

```ruby
while buf.bytesize - pos >= 24
  header = buf.slice(pos, 24)
  (key_length, _, body_length, cas) = header.unpack(KV_HEADER)

  if key_length == 0
    # all done!
    @multi_buffer = nil
    @position = nil
    @inprogress = false
    break

  elsif buf.bytesize - pos >= 24 + body_length
    flags = buf.slice(pos + 24, 4).unpack1("N")
    key = buf.slice(pos + 24 + 4, key_length)
    value = buf.slice(pos + 24 + 4 + key_length, body_length - key_length - 4) if body_length - key_length - 4 > 0

    pos = pos + 24 + body_length

    begin
      values[key] = [deserialize(value, flags), cas]
    rescue DalliError
    end

  else
    # not enough data yet, wait for more
    break
  end
end
@position = pos
```

### Proposal

If `unpack` and `unpack1` had an `offset:` parameter, it would allow this kind of code to extract the fields it needs without allocating and copying as much strings, e.g.:

```ruby
flags = buf.slice(pos + 24, 4).unpack1("N")
```

could be:

```ruby
buf.unpack1("N", offset: pos + 24)
```




-- 
https://bugs.ruby-lang.org/

^ permalink raw reply	[flat|nested] 13+ messages in thread

end of thread, other threads:[~2021-10-25 17:36 UTC | newest]

Thread overview: 13+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2021-10-18  8:04 [ruby-core:105660] [Ruby master Feature#18254] Add an `offset` parameter to String#unpack and String#unpack1 byroot (Jean Boussier)
2021-10-18  8:24 ` [ruby-core:105661] " znz (Kazuhiro NISHIYAMA)
2021-10-18  8:32 ` [ruby-core:105662] " byroot (Jean Boussier)
2021-10-18 14:26 ` [ruby-core:105663] " byroot (Jean Boussier)
2021-10-25  2:08 ` [ruby-core:105766] " matz (Yukihiro Matsumoto)
2021-10-25  2:36 ` [ruby-core:105769] " mame (Yusuke Endoh)
2021-10-25  5:39 ` [ruby-core:105774] " nobu (Nobuyoshi Nakada)
2021-10-25  6:56 ` [ruby-core:105775] " byroot (Jean Boussier)
2021-10-25  9:26 ` [ruby-core:105783] " duerst
2021-10-25  9:41 ` [ruby-core:105784] " byroot (Jean Boussier)
2021-10-25 14:34 ` [ruby-core:105788] " byroot (Jean Boussier)
2021-10-25 16:55 ` [ruby-core:105792] " mame (Yusuke Endoh)
2021-10-25 17:36 ` [ruby-core:105794] " byroot (Jean Boussier)

This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for read-only IMAP folder(s) and NNTP newsgroup(s).