Ruby 3.2.0 Preview 3 Dirilis

Kami dengan senang hati mengumumkan rilis dari Ruby 3.2.0-preview3. Ruby 3.2 menambahkan banyak fitur dan memperbaiki performa.

Dukungan WebAssembly berbasis WASI

Ini adalah sebuah port awal dari dukungan WebAssembly berbasis WASI. Port ini memperbolehkan sebuah CRuby binary agar tersedia pada web browser, Serverless Edge environment, dan WebAssembly/WASI embedder lainnya. Saat ini, port melewatkan rangkaian tes dasar dan bootstrap dengan tidak menggunakan Thread API.

Latar Belakang

WebAssembly (Wasm) semula diperkenalkan untuk menjalankan program dengan aman dan cepat pada web browser. Sasaran - menjalankan program secara efisien dengan keamanan pada berbagai macam environment - diinginkan sejak lama tidak hanya oleh web, akan tetapi oleh aplikasi umum.

WASI (The WebAssembly System Interface) didesain untuk berbagai macam kasus penggunaan. Walaupun aplikasi perlu berkomunikasi dengan sistem operasi, WebAssembly berjalan pada sebuah virtual machine yang mana tidak memiliki sebuah system interface. WASI menstandarkannya.

Dukungan WebAssembly/WASI pada Ruby bermaksud untuk memanfaatkan proyek tersebut. Ini memperbolehkan pengembang Ruby untuk memprogram aplikasi yang berjalan pada platform yang menjanjikan.

Kasus penggunaan

Dukungan ini mendorong pengembang untuk memanfaatkan CRuby pada WebAssembly environment. Sebuah contoh kasus penggunaan adalah dukungan CRuby pada TryRuby playground. Sekarang Anda bisa mencoba CRuby pada web browser.

Poin teknis

WASI dan WebAssembly saat ini memiliki beberapa fitur yang tidak didukung, seperti mengimplementasikan Fiber, exception, dan GC karena ini masih terus berkembang dan juga untuk alasan keamanan. Oleh sebab itu, CRuby mengisi gap tersebut dengan menggunakan Asyncify, yang mana adalah sebuah teknik transformasi binary untuk mengendalikan eksekusi pada lingkungan pengguna.

Sebagai tambahan, kami membangun sebuah VFS di atas WASI sehingga kami dapat mengemas aplikasi Ruby ke dalam sebuah berkas .wasm dengan mudah. Ini membuat distribusi aplikasi Ruby sedikit lebih mudah.

Tautan terkait

Peningkatan Regexp melawan ReDoS

Diketahui bahwa Regexp matching mungkin membutuhkan waktu yang lama. Jika kode Anda mencoba untuk mencocokkan sebuah Regexp yang tidak efisien dengan sebuah masukan yang tidak terpercaya, seorang penyerang bisa mengeksploitasi ini untuk Denial of Service yang efisien (disebut Regular expression DoS atau ReDoS).

Kami telah memperkenalkan dua peningkatan untuk memitigasi ReDoS secara signifikan.

Peningkatan algoritma Regexp matching

Sejak Ruby 3.2, algoritma Regexp matching telah ditingkatkan dengan menggunakan teknik memoization.

# Regexp matching ini membutuhkan waktu 10 detik pada Ruby 3.1 dan 0.003 detik pada Ruby 3.2

/^a*b?a*$/ =~ "a" * 50000 + "x"

Peningkatan algoritma matching memperbolehkan eksekusi (sekitar 90% dari eksperimen kami) selesai dalam waktu linier.

(Untuk preview user: peningkatan ini mungkin akan mengonsumsi memori sebanding dengan panjang masukan untuk setiap matching. Kami mengharapkan tidak ada masalah muncul karena alokasi memori biasanya terlambat dan sebuah Regexp matching biasa mengonsumsi paling banyak 10 kali memori sepanjang masukan. Jika Anda kehabisan memori ketika mengeksekusi pada sebuah aplikasi, mohon laporkan).

Proposal asli di https://bugs.ruby-lang.org/issues/19104

Regexp timeout

Peningkatan di atas tidak dapat diterapkan pada regular expression tertentu, seperti mengandung fitur lanjutan (contoh, back-references atau look-around), atau dengan pengulangan dalam jumlah tetap yang sangat besar. Sebagai penanganan, Sebuah fitur timeout pada Regexp matching juga diperkenalkan.

Regexp.timeout = 1.0

/^a*b?a*()\1$/ =~ "a" * 50000 + "x"
#=> Regexp::TimeoutError akan muncul dalam waktu satu detik

Catat bahwa Regexp.timeout adalah sebuah pengaturan global. Jika Anda ingin menggunakan pengaturan yang berbeda untuk Regexp tertentu, Anda bisa menggunakan kata kunci timeout pada Regexp.new.

Regexp.timeout = 1.0

# regexp ini tidak memiliki timeout
long_time_re = Regexp.new('^a*b?a*()\1$', timeout: Float::INFINITY)

long_time_re =~ "a" * 50000 + "x" # tidak pernah diinterupsi

Proposal asli di https://bugs.ruby-lang.org/issues/17837

Fitur penting lainnya

Tidak lagi membundel sumber pihak ketiga

  • Kami tidak lagi membundel sumber pihak ketiga, seperti libyaml, libffi.

    • libyaml telah dihapus dari psych. Anda mungkin butuh untuk memasang libyaml-dev pada platform Ubuntu/Debian. Nama package bisa berbeda pada platform lain.

    • libffi akan dihapus dari fiddle

Bahasa

  • Argumen dari anonymous rest dan keyword rest sekarang dapat dilewatkan sebagai argumen daripada sekadar digunakan pada parameter method. [Feature #18351]

      def foo(*)
        bar(*)
      end
      def baz(**)
        quux(**)
      end
    
  • Sebuah proc yang menerima sebuah positional argument dan keyword tidak akan lagi autosplat. [Bug #18633]

    proc{|a, **k| a}.call([1, 2])
    # Ruby 3.1 dan sebelumnya
    # => 1
    # Ruby 3.2 dan setelahnya
    # => [1, 2]
    
  • Urutan evaluasi dari constant assignment pada objek eksplisit telah dibuat konsisten dengan urutan evaluasi tunggal. Pada kode berikut:

      foo::BAR = baz
    

    foo sekarang dipanggil sebelum baz. Demikian pula, untuk multiple assignment pada constant, urutan evaluasi left-to-right digunakan. Pada kode berikut:

        foo1::BAR1, foo2::BAR2 = baz1, baz2
    

    Urutan evaluasi berikut sekarang digunakan:

    1. foo1
    2. foo2
    3. baz1
    4. baz2

    [Bug #15928]

  • Pencarian pattern tidak lagi eksperimental. [Feature #18585]

  • Method mengambil sebuah parameter lain (seperti *args) dan mengharapkan untuk mendelegasikan keyword argument melalui foo(*args) sekarang harus ditandai dengan ruby2_keywords (jika belum terjadi). Dengan kata lain, semua method mengharapkan untuk mendelegasikan keyword argument melalui *args sekarang harus ditandai dengan ruby2_keywords, tanpa kecuali. Ini akan membuat transisi lebih mudah ke bentuk delegasi lain saat sebuah pustaka memerlukan Ruby 3+. Sebelumnya, penanda ruby2_keywords dijaga jika method penerima mengambil *args, tetapi ini adalah sebuah bug dan ketidakkonsistenan. Sebuah teknik yang bagus untuk menemukan potensi hilang dari ruby2_keywords adalah menjalankan rangkaian tes, dimana pun tes gagal cari method terakhir yang harus menerima keyword argument, menggunakan puts nil, caller, nil, dan mengecek setiap method/block pada rantai pemanggilan yang harus mendelegasikan keyword ditandai dengan benar sebagai ruby2_keywords. [Bug #18625] [Bug #16466]

      def target(**kw)
      end
    
      # Secara tidak sengaja, berjalan tanpa ruby2_keywords pada Ruby 2.7-3.1,
      # ruby2_keywords dibutuhkan pada 3.2+. Sama halnya dengan (*args, **kwargs)
      # atau (...) yang juga dibutuhkan baik #foo maupun #bar ketika migrasi
      # dari ruby2_keywords.
      ruby2_keywords def bar(*args)
        target(*args)
      end
    
      ruby2_keywords def foo(*args)
        bar(*args)
      end
    
      foo(k: 1)
    

Perbaikan performa

YJIT

  • Dukungan arm64 / aarch64 pada platform UNIX.
  • Membangun YJIT memerlukan Rust 1.58.1+. [Feature #18481]

Perubahan penting lainnya sejak 3.1

  • Hash
    • Hash#shift sekarang selalu mengembalikan nil jika hash kosong dibandingkan mengembalikan nilai default atau memanggil default proc. [Bug #16908]
  • MatchData
  • Module
  • Proc
    • Proc#dup mengembalikan sebuah instance dari subclass. [Bug #17545]
    • Proc#parameters sekarang menerima kata kunci lambda. [Feature #15357]
  • Refinement
  • Set
    • Set sekarang tersedia sebagai sebuah builtin class tanpa membutuhkan require "set". [Feature #16989] Ini dimuatkan secara otomatis melalui constant Set atau sebuah pemanggilan Enumerable#to_set.
  • String
    • String#byteindex dan String#byterindex telah ditambahkan. [Feature #13110]
    • Memperbarui versi Unicode ke 14.0.0 dan Emoji ke 14.0. [Feature #18037] (juga diaplikasikan pada Regexp)
    • String#bytesplice telah ditambahkan. [Feature #18598]
  • Struct
    • Sebuah class Struct juga dapat diinisialisasi dengan argumen keyword tanpa keyword_init: true pada Struct.new [Feature #16806]

Isu kompatibilitas

Catatan: tidak termasuk perbaikan bug.

Constant terhapus

Berikut ini adalah constant yang telah dihapus.

Method terhapus

Berikut ini adalah method yang telah dihapus.

Isu kompatibilitas Stdlib

  • Psych tidak lagi membundel libyaml. Pengguna perlu memasang pustakan libyaml sendiri melalui package system. [Feature #18571]

Pembaruan C API

C API diperbarui

Berikut ini adalah API yang telah diperbarui.

  • Pembaruan PRNG rb_random_interface_t diperbarui dan diberikan versi. Pustaka extension yang menggunakan interface ini dan untuk versi lama. init_int32 juga perlu didefinisikan.

C API terhapus

Berikut ini adalah API yang telah dihapus.

  • Variabel rb_cData.
  • Function “taintedness” dan “trustedness”. [Feature #16131]

Pembaruan pustaka standar

  • SyntaxSuggest

    • Fitur syntax_suggest sebelumnya dead_end diintegrasikan dengan Ruby. [Feature #18159]
  • ErrorHighlight

    • Sekarang, ini menunjuk pada sebuah argumen dari TypeError dan ArgumentError
test.rb:2:in `+': nil tidak dapat dipaksa menjadi Integer (TypeError)

sum = ary[0] + ary[1]
               ^^^^^^
  • Berikut ini adalah default gem yang telah diperbarui.
    • RubyGems 3.4.0.dev
    • bigdecimal 3.1.2
    • bundler 2.4.0.dev
    • cgi 0.3.2
    • date 3.2.3
    • error_highlight 0.4.0
    • etc 1.4.0
    • io-console 0.5.11
    • io-nonblock 0.1.1
    • io-wait 0.3.0.pre
    • ipaddr 1.2.4
    • json 2.6.2
    • logger 1.5.1
    • net-http 0.2.2
    • net-protocol 0.1.3
    • ostruct 0.5.5
    • psych 5.0.0.dev
    • reline 0.3.1
    • securerandom 0.2.0
    • set 1.0.3
    • stringio 3.0.3
    • syntax_suggest 0.0.1
    • timeout 0.3.0
  • Berikut ini adalah bundled gem yang telah diperbarui.
    • minitest 5.16.3
    • net-imap 0.2.3
    • rbs 2.6.0
    • typeprof 0.21.3
    • debug 1.6.2
  • Berikut ini adalah default gem yang sekarang menjadi bundled gem.

Lihat NEWS atau commit logs untuk lebih detail.

Dengan perubah tersebut, 2719 berkas berubah, 191269 sisipan(+), 120315 terhapus(-) sejak Ruby 3.1.0!

Unduh

  • https://cache.ruby-lang.org/pub/ruby/3.2/ruby-3.2.0-preview3.tar.gz

    SIZE: 20086542
    SHA1: dafca8116d36ceaa32482ab38359768de8c3ae5e
    SHA256: c041d1488e62730d3a10dbe7cf7a3b3e4268dc867ec20ec991e7d16146640487
    SHA512: 860634d95e4b9c48f18d38146dfbdc3c389666d45454248a4ccdfc3a5d3cd0c71c73533aabf359558117de9add1472af228d8eaec989c9336b1a3a6f03f1ae88
    
  • https://cache.ruby-lang.org/pub/ruby/3.2/ruby-3.2.0-preview3.tar.xz

    SIZE: 14799804
    SHA1: c94e2add05502cb5c39afffc995b7c8f000f7df0
    SHA256: d3f5619de544240d92a5d03aa289e71bd1103379622c523a0e80ed029a74b3bb
    SHA512: c1864e2e07c3711eaa17d0f85dfbcc6e0682b077782bb1c155315af45139ae66dc4567c73682d326975b0f472111eb0a70f949811cb54bed0b3a816ed6ac34df
    
  • https://cache.ruby-lang.org/pub/ruby/3.2/ruby-3.2.0-preview3.zip

    SIZE: 24426893
    SHA1: 346c051c4be7ab8d0b551fd2ff8169785697db62
    SHA256: cf49aa70e7ebd8abebffd5e49cd3bd92e5b9f3782d587cc7ed88c98dd5f17069
    SHA512: 4f22b5ea91be17ef5f68cf0acb1e3a226dcc549ad71cc9b40e623220087c4065ca9bea942710f668e5c94ca0323da8d2ccd565f95a9085c1a0e38e9c0543b22f
    

Apa itu Ruby

Ruby pertama kali dikembangkan oleh Matz (Yukihiro Matsumoto) pada 1993 dan sekarang dikembangkan sebagai Open Source. Ruby berjalan pada berbagai platform dan digunakan di seluruh dunia, khususnya pengembangan web.