Ruby 3.2.0 Preview 2 릴리스

Ruby 3.2.0-preview2 릴리스를 알리게 되어 기쁩니다. Ruby 3.2는 많은 기능과 성능 향상을 포함하고 있습니다.

WASI 기반 웹어셈블리 지원

WASI에 기반해 웹어셈블리를 지원하는 첫 이식판입니다. 이를 통해 CRuby 바이너리는 웹 브라우저, 서버리스 엣지 환경, 그 이외의 웹어셈블리/WASI를 사용 가능한 환경에서 동작할 수 있습니다. 현재 이 이식판은 스레드 API를 사용하지 않는 기본적인 테스트와 부트스트랩 테스트 스위트를 통과합니다.

배경

웹어셈블리(Wasm)는 본래 웹 브라우저에서 프로그램을 안전하고 빠르게 실행하기 위해서 만들어졌습니다. 하지만 그 목적 중 하나인 프로그램을 다양한 환경에서 안전하고 효율적으로 실행하는 것은 웹뿐만이 아니라 일반적인 애플리케이션도 바라던 것입니다.

WASI(The WebAssembly System Interface)는 이러한 용도를 위해 설계되었습니다. 이러한 애플리케이션은 운영체제와 통신해야 합니다만, 웹어셈블리는 시스템 인터페이스를 가지지 않는 가상 머신 위에서 동작합니다. WASI는 이 인터페이스를 표준화합니다.

Ruby의 웹어셈블리/WASI 지원은 이러한 프로젝트들을 활용하기 위함입니다. 이를 통해 Ruby 개발자들이 약속한 플랫폼에서 움직이는 애플리케이션을 작성할 수 있습니다.

사용 예시

이는 개발자가 웹어셈블리 환경에서 CRuby를 활용할 수 있도록 돕습니다. 하나의 예로, TryRuby playground의 CRuby 지원이 있습니다. 이제 CRuby를 웹 브라우저 상에서 직접 사용해볼 수 있습니다.

기술적인 부분

현재 WASI와 웹어셈블리 자체에는 Fiber, 예외 처리, GC를 구현하기 위한 일부 기능이 부족합니다. 이는 여전히 개발 중이라는 점도 있지만, 보안 때문이기도 합니다. 그래서 CRuby는 사용자 공간에서의 실행을 제어하기 위한 바이너리 변환 기술인 Asyncify를 사용해 그 차이를 메꿉니다.

나아가서 Ruby 앱을 간단하게 단일 .wasm 파일로 패키징할 수 있도록 WASI 상에 VFS를 구현했습니다. 이는 Ruby 앱의 배포를 쉽게 해줄 것입니다.

관련 링크

정규표현식 타임아웃

정규표현식 일치 처리에 타임아웃 기능을 도입했습니다.

Regexp.timeout = 1.0

/^a*b?a*$/ =~ "a" * 50000 + "x"
#=> 1초 후에 Regexp::TimeoutError 발생

정규표현식 일치는 기대와는 다르게 시간이 오래 걸리는 경우가 있습니다. 신뢰할 수 없는 입력에 대해서 비효율적일 가능성이 있는 정규표현식을 일치시키고 있다면, 공격자는 이를 이용해 효율적으로 서비스 거부 공격(이를 정규표현식 DoS, 또는 ReDoS라고 부릅니다)이 가능합니다.

Ruby 애플리케이션의 요구사항에 맞는 Regexp.timeout을 설정해 DoS에 대한 위험을 예방하거나 상당히 완화할 수 있습니다. 애플리케이션에 직접 설정해보세요. 피드백도 환영합니다.

Regexp.timeout은 전역 설정임에 주의하세요. 일부의 특수한 정규표현식에 대해서만 다른 타임아웃 설정을 사용하고 싶다면, Regexp.newtimeout 키워드를 사용할 수 있습니다.

Regexp.timeout = 1.0

# 이 정규표현식은 타임아웃 설정이 없습니다.
long_time_re = Regexp.new("^a*b?a*$", timeout: nil)

long_time_re =~ "a" * 50000 + "x" # never interrupted

기능 제안 티켓: https://bugs.ruby-lang.org/issues/17837

그 이외의 주목할 만한 새 기능

서드파티 소스 코드를 더 이상 내장하지 않음

  • libyaml이나 libffi와 같은 서드파티의 소스 코드를 내장하지 않기로 결정했습니다.

    • psych에 포함되어 있던 libyaml의 소스 코드는 삭제되었습니다. Ubuntu/Debian 환경이라면 직접 libyaml-dev를 설치해야합니다. 이 패키지의 이름은 각 환경마다 다를 수 있습니다.

    • fiddle에 포함되어 있던 libffi는 preview2에서 삭제될 예정입니다.

언어

  • 익명 나머지 인수, 익명 나머지 키워드 인수가 파라미터로서뿐만 아니라, 인수로서도 사용할 수 있게 됩니다. [Feature #18351]

      def foo(*)
        bar(*)
      end
      def baz(**)
        quux(**)
      end
    
  • 1개의 위치 인수와 나머지를 키워드로 받는 프록은 인수를 자동으로 전개하지 않습니다. [Bug #18633]

    proc{|a, **k| a}.call([1, 2])
    # Ruby 3.1 and before
    # => 1
    # Ruby 3.2 and after
    # => [1, 2]
    
  • 상수 대입에서의 상수 평가 순서가 단일 속성 대입 시의 평가 순서와 일관성을 가지게 됩니다. 다음 코드의 경우,

      foo::BAR = baz
    

    foo는 이제 baz보다 먼저 호출됩니다. 마찬가지로 상수의 다중 대입에서도 왼쪽에서 오른쪽으로 평가됩니다. 다음 코드의 경우,

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

    아래와 같은 순서로 평가됩니다.

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

    [Bug #15928]

  • 검색 패턴은 이제 정식 기능입니다. [Feature #18585]

  • *args와 같은 나머지 파라미터를 받는 메서드에서 foo(*args)를 통해 키워드 인수를 위임하고 싶은 경우, 반드시 ruby2_keywords를 사용해야 합니다. 다르게 말하면, *args를 사용해 키워드 인수를 위임하고 싶은 모든 메서드는 예외 없이 ruby2_keywords를 사용해야 합니다. 이 변경으로 라이브러리가 Ruby 3 이상을 요구하게 되었을 때 다른 위임 방식으로의 마이그레이션이 간단해집니다. 지금까지 메서드가 *args를 넘겨받았을 때, ruby2_keywords 플래그가 유지되었습니다만, 이는 의도치 않은 동작이었으며, 일관성이 없었습니다. 빠져있었던 ruby2_keywords를 찾아내기 위한 좋은 방법 중 한 가지로 테스트를 실행한 뒤, 실패하는 각각의 테스트에서 키워드 인수를 받는 마지막 메서드를 찾고, 그곳에서 puts nil, caller, nil를 사용하세요. 그리고 나서 호출 체인의 각 메서드/블록이 키워드를 위임할 때 ruby2_keywords를 올바르게 사용하고 있는지 확인하세요. [Bug #18625] [Bug #16466]

      def target(**kw)
      end
    
      # 의도치 않게 Ruby 2.7-3.1에서 ruby2_keywords 없이 동작했습니다만,
      # Ruby 3.2+에서는 ruby2_keywords가 필요합니다. ruby2_keywords를 사용하지 않는 경우,
      # #foo, #bar 양쪽에 (*args, **kwargs)나 (...)이 필요합니다.
      ruby2_keywords def bar(*args)
        target(*args)
      end
    
      ruby2_keywords def foo(*args)
        bar(*args)
      end
    
      foo(k: 1)
    

성능 향상

YJIT

  • arm64, aarch64 상의 UNIX 환경을 지원합니다.
  • YJIT 빌드하기 위해서는 Rust 1.58.1 이상을 요구합니다. [Feature #18481]

3.1 이후로 주목할 만한 변경

  • Hash
    • Hash#shift는 이제 해시가 비어있다면 기본 값이나 기본 프록을 호출한 결과 값을 반환하는 대신 언제나 nil을 반환합니다. [Bug #16908]
  • MatchData
  • Module
  • Proc
    • Proc#dup은 서브클래스의 인스턴스를 반환합니다. [Bug #17545]
    • Proc#parameters는 이제 람다 키워드를 받습니다. [Feature #15357]
  • Refinement
  • Set
    • Set은 이제 require "set"할 필요 없이 사용 가능한 내장 클래스입니다. [Feature #16989] 현재는 Set 상수를 사용하거나 Enumerable#to_set을 호출하면 자동으로 로드됩니다.
  • String
    • String#byteindexString#byterindex가 추가되었습니다. [Feature #13110]
    • 유니코드 버전이 14.0.0, 에모지 버전이 14.0으로 갱신되었습니다. [Feature #18037] (이는 정규표현식에도 적용됩니다)
    • String#bytesplice가 추가되었습니다. [Feature #18598]
  • Struct
    • keyword_init: true 없이 Struct.new에 키워드 인수를 넘겨 Struct 클래스를 초기화할 수 있습니다. [Feature #16806]

호환성 문제

주의: 기능 버그 수정은 포함되어 있지 않습니다.

삭제된 상수

폐기 예정이었던 상수가 삭제됩니다.

삭제된 메서드

폐기 예정이었던 메서드가 삭제됩니다.

Stdlib 호환성 문제

  • Psych는 더 이상 libyaml 소스 코드를 포함하지 않습니다. 이용자는 패키지 매니저를 사용해서 스스로 libyaml 라이브러리를 설치해야합니다. [Feature #18571]

C API 변경

삭제된 C API

다음 폐기 예정인 API가 삭제됩니다.

  • rb_cData 변수.
  • “taintedness”와 “trustedness” 함수. [Feature #16131]

표준 라이브러리 갱신

  • 다음 기본 gem이 갱신되었습니다.

    • 미정
  • 다음 내장 gem이 갱신되었습니다.

    • 미정
  • 다음 기본 gem은 이제 내장 gem입니다. bundler 환경에서는 Gemfile에 다음 라이브러리를 추가해야 합니다.

    • 미정

더 자세한 내용은 NEWS커밋 로그를 확인해 주세요.

이러한 변경사항에 따라, Ruby 3.1.0 이후로 파일 2393개 수정, 168931줄 추가(+), 113411줄 삭제(-)가 이루어졌습니다!

다운로드

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

    SIZE: 19816780
    SHA1: 2106c77fc1600daf41ae137ecc4cf7937e27f67f
    SHA256: 8a78fd7a221b86032f96f25c1d852954c94d193b9d21388a9b434e160b7ed891
    SHA512: 5e9ddcb1a43cff449b0062cc716bfb80a9ebbb14a1b063f34005e2998c2c5033badb44e882232db9b2fceda9376f6615986e983511fda2575d60894752b605cc
    
  • https://cache.ruby-lang.org/pub/ruby/3.2/ruby-3.2.0-preview2.tar.xz

    SIZE: 14578112
    SHA1: 538b3ea4dc0d99f60f8bd6f71e65a56ceeb41c18
    SHA256: 01fac0929dccdabc0686c1109da6c187897a401da9ff8851242befa92f7fd430
    SHA512: 0f4cc919284fdfa1a42b6381760d1b3a4660da4b0fcdd2adf01ea04a425548b3c5ac090866915675db73964a1055090e54dd97cf4628cbb69403e541c71c28ff
    
  • https://cache.ruby-lang.org/pub/ruby/3.2/ruby-3.2.0-preview2.zip

    SIZE: 24150109
    SHA1: 69ffffc52cad626166f73f21f25c29c9d73fe0e8
    SHA256: 67f9ad3110be1975b3ce547c0a6e2c910dfc1945fd6e9bb1bd340568897c6554
    SHA512: 1447e099e7a8da0ff206fda6f4e466640d6e86e9da8148315ab0154684b1fd22c02c0022b5a2f4d3fc00103b4e8cef8e35a770174921fd8c6abeca9ad41c1818
    

Ruby는

Ruby는 1993년에 Matz(마츠모토 유키히로) 씨가 처음 개발했고, 현재는 오픈 소스로서 개발되고 있습니다. 여러 플랫폼에서 동작하며, 특히 웹 개발에서 전 세계적으로 이용되고 있습니다.