コンテンツにスキップ

dry-rb Ecosystem Plugins — Survey

ステータス: research, 2026-05-09。Rigorプラグインの観点からdry-rb gemファミリーのワンショットサーベイ。gem間の依存関係と各gemが公開する型シェイピングサーフェスをまとめ、後続の設計ドキュメントが単一のrigor-dryプラグイン、rigor-dry-*ファミリー、または中粒度の分割のいずれを出荷するかを証拠に基づいて決定できるようにする。

調査したコーパスはreferences/hanakai-rb/content/guides/dry/のhanakai-rbガイドツリーで、hanami/dry/rom組織がhanakai-rbに統合された後のdry-rbの権威ある公開ガイド。以下の依存関係エッジはそのコーパス内の散文的な記述から取得した。gemspecの検証とバージョンピンの決定はプラグインごとの作成ステップに延期する。

このドキュメントは参考情報。拘束力のあるプラグイン契約は各プラグインのREADME.mdとインテグレーションスペックに存在する。Railsプラグインロードマップの規律disciplinedocs/design/20260508-rails-plugins-roadmap.md)を踏襲する。

dry-rbがRigorにとって興味深い理由

Section titled “dry-rbがRigorにとって興味深い理由”

dry-rbは慣用的なRubyの中で最も型意識の高いDSLファミリー。3つの特性がRigorに関係する:

  1. 明示的な属性型付けdry-struct/dry-initializer/dry-schemaはプラグインが構文的に辿ることのできる実際のRubyオブジェクト(Types::StringTypes::Coercible::Integer等)を通じて属性ごとの型を宣言する。
  2. 構造化された戻り値シェイプdry-monadsdry-operationはメソッドに既知のモナドエンベロープ(Result[T, E]Maybe[T]等)を与え、メソッド境界を越えて生存する — ナローイングへの明確な恩恵。
  3. 合成的な基盤。高レベルなgem(dry-validationdry-operationdry-rails)は低レベルなgem(dry-typesdry-schemadry-monads)を合成する。ボトムアップのプラグイン順序は依存関係エッジに直接マッピングされる。クロスプラグインAPI(ADR-9)はまさにマルチgemのrigor-dry-*ファミリーがファクト共有に消費するものである。

このサーベイは特定のプラグインへの関与の前置。目標は後続のスコーピング決定を証拠に基づかせること。

各gemは静的ファクトの種類によって6つの階層のひとつに分類される。

階層この階層のプラグインが行うことメンバー
A — 型システム基盤型付き属性/リーダー/強制変換シェイプを宣言dry-types, dry-struct, dry-schema, dry-validation, dry-logic, dry-initializer
B — 制御フローシェイプ戻り型をRigorがナローイングできるモナドエンベロープにラップdry-monads, dry-operation, dry-effects
C — DI/設定コンテナまたはデフォルト値から戻り型が来るリーダー/クラスメソッドを生成dry-auto_inject, dry-configurable, dry-system, dry-container
D — ユーティリティ静的型シェイプへの影響なしdry-cli, dry-core, dry-events, dry-files, dry-inflector, dry-logger, dry-monitor
E — レガシー/置き換え済み完全性のためにリスト。置き換えが対応dry-equalizer, dry-matcher, dry-transaction, dry-view
F — フレームワーク統合階層A-CのgemをRailsに組み込むdry-rails

階層Aと階層BがRigorの主要ターゲット。階層CはADR-9がランディングした後に魅力的になるストレッチゴール。階層D/E/Fはスコープ外(D、E)か既存プラグイン上の薄いラッパー(F)のいずれか。

目的。強制変換、制約、合成コンビネータを持つ拡張可能な値型システム。

プラグイン関連DSL

  • Types = Dry.Types()は名前付き型のモジュールを開く。
  • Types::StringTypes::Coercible::IntegerTypes::Strict::*はビルド済みキャリアのレジストリに到達する。
  • T.optionalT.constrained(gteq: 18)T.constructor { … }T | UT.default(0)が日常的なコンビネータ。

プラグインが発行する静的ファクト

  • 型式はRigorの型キャリアにマッピングされる: Types::StringStringTypes::Coercible::IntegerIntegerT.optionalT | nilT | U → union。
  • T.constrained(gteq: 18)は静的型をTに保ちつつ、ダウンストリームのナローイングが消費できる述語ファクトを追加する — v0.1.1の正規表現→リファインメント認識器が出荷されたらRigorのリファインメント名機構の候補となる。
  • カスタム型ビルダー(.constructor { ... })はキャリアをブロックの結果型にシフトする。そこでの精密な推論はv1のスコープ外であり、堅牢性の原則に従ってDynamic[T]に劣化できる。

ドキュメントに記載されたdry-*依存関係。なし — dry-typesは基盤。

プラグイン結合。基盤。dry-struct、dry-schema、dry-validation、dry-initializer、dry-monads(そのValidatedモナド)がすべてダウンストリームに位置する。

目的。型付き属性で定義された不変値オブジェクト。

プラグイン関連DSL

class User < Dry::Struct
attribute :name, Types::String
attribute :age, Types::Coercible::Integer
attribute :address do
attribute :city, Types::String
end
end

プラグインが発行する静的ファクト

  • attribute :name, Tはdry-typesプラグインがTから解決したキャリアを返すリーダー#nameを宣言する。
  • ブロック形式の属性(attribute :address do ... end)はネストした匿名のDry::Structサブクラスを定義する。プラグインは内部クラスのシェイプとそのクラスを返す外部リーダーの両方を発行する。
  • コンストラクタシグネチャは宣言済み属性のユニオンから導出される。
  • transform_keys(&:to_sym)等は静的属性セットを変更しない。

ドキュメントに記載されたdry-*依存関係。dry-typesの上に構築される。

プラグイン結合。dry-typesファクトのハードコンシューマー。

目的。ハッシュ形式入力のバリデーションと強制変換。2つのフレーバー: Schema.Params(Webフォーム強制変換: 文字列 → 整数/ブール値)、Schema.JSON(文字列強制変換なし)。_index.mdは明示的に「dry-schemadry-typesの強制変換型を使用する」と述べている。

プラグイン関連DSL

UserSchema = Dry::Schema.Params do
required(:name).filled(:string)
required(:age).value(:integer, gt?: 18)
required(:tags).array(:string)
required(:address).hash do
required(:street).filled(:string)
end
end

プラグインが発行する静的ファクト

  • スキーマ定数は型付き入力 → 出力契約にマッピングされる。
  • schema.call(input)の出力は、#to_h/[]のキーが宣言に従って型付けされた結果: :name → 非空文字列、:age → Integer、:tags → Array[String]、:address.street → 非空文字列。
  • 述語サフィックス(gt?: 18)はv0.1.1がランディングされたらRigorのリファインメント名カタログ(positive-int等)に供給される。
  • Params対JSONの区別が重要: Paramsのみが文字列を強制変換する — プラグインは強制変換型を解決する前にどのビルダーがスキーマを生成したかを記録しなければならない。

ドキュメントに記載されたdry-*依存関係。dry-types(強制変換バックエンド)、dry-logic(述語エンジン)。

プラグイン結合。dry-typesのハードコンシューマーであり、dry-logicの(軽量な)コンシューマー。

目的。ドメインバリデーション契約: 型付きのparams { ... }スキーマ(dry-schemaに委譲)にビジネスロジックのruleブロックを加えたもの。

プラグイン関連DSL

class NewUserContract < Dry::Validation::Contract
params do
required(:email).filled(:string)
required(:age).value(:integer)
end
rule(:email) do
key.failure('has invalid format') unless EMAIL_RE.match?(value)
end
end
contract.call(email: 'jane@doe.org', age: '17')
# => Dry::Validation::Result(型付きの:email/:age付き)

プラグインが発行する静的ファクト

  • Contract#callDry::Validation::Resultを返す。#success?#failure?が結果をナローイングする。.to_hはスキーマ型付きハッシュを公開する。
  • params { ... }json { ... }ブロックは本質的にdry-schemaのスキーマ — プラグインは内部シェイプのためにdry-schemaプラグインに処理を委ねられる。
  • rule(:email) { ... }:emailの型を変更しない。ビジネスルールのファクトのみを追加する。

ドキュメントに記載されたdry-*依存関係。dry-schema(スキーマエンジン)、dry-types(強制変換)。

プラグイン結合。dry-schema(トランジティブにdry-types)のハードコンシューマー。

目的。述語合成プリミティブ — Rule::Predicate&/|コンビネータ、カリー化述語。dry-types(制約)とdry-schema(述語サフィックスDSL)が内部で使用する。

プラグイン関連DSL。ユーザーコードで一般的に手書きされない。ライブラリ基盤。

プラグインが発行する静的ファクト。ユーザーコードサーフェスではなし。専用のrigor-dry-logicプラグインは価値が低い可能性が高い。dry-typesとdry-schemaのプラグインが必要な述語対応ロジックを内部で持てばよい。

ドキュメントに記載されたdry-*依存関係。なし。

プラグイン結合。dry-typesとdry-schemaに埋め込み。

目的extend Dry::Initializer; param :foo, T; option :bar, Tで継承なしに型付きコンストラクタとアクセサを生成する。

プラグイン関連DSL

class User
extend Dry::Initializer
param :name, proc(&:to_s)
param :role, default: proc { 'customer' }
option :admin, default: proc { false }
option :phone, optional: true
option :emails, [] do
option :address, proc(&:to_s)
end
end

プラグインが発行する静的ファクト

  • param/optionはインスタンスリーダーを生成する。型は型制約引数。
  • 3種類のリーダー型ソースを処理する必要がある:
    • Dry::Types['…']制約 → dry-typesプラグインに委ねる。
    • proc(&:to_s)/類似の強制変換proc → 結果型はメソッドの戻り値(to_sStringto_iInteger等のビルトインでは既知のことが多い)。
    • 制約なし/default:のみ → リーダーはデフォルト式の型を返す。ない場合はuntyped
  • optional: trueT | nilに広げる(未セットのリーダーはDry::Initializer::UNDEFINEDをデフォルトにするが、ユーザー可視の境界ではnilがRigorの適切な近似)。
  • ネストしたoption ... do option ... endはリーダー自体が型付きの内部匿名struct風クラスを定義する。

ドキュメントに記載されたdry-*依存関係。ハードな依存なし。ユーザーがオプトインした場合はdry-typesと互換。

プラグイン結合。dry-typesファクトのオプショナルコンシューマー。スタンドアロンのrigor-dry-initializerプラグインはrigor-dry-typesなしでも有用。

目的。戻り値のための代数的データ型: Result(Success/Failure)、Maybe(Some/None)、Try(例外捕捉)、ListTaskValidatedUnit。バインドチェーンのためのdo記法も含む。

プラグイン関連DSL

include Dry::Monads[:result]
def call(input)
Success(input.upcase)
rescue ArgumentError => e
Failure(e)
end

プラグインが発行する静的ファクト

  • Success(x)またはFailure(e)を返すメソッドは、TSuccess引数型のunionでEFailure引数型のunionである戻り型Result[T, E]を持つ。
  • Maybe(x)Some[T] | None(== Maybe[T])を返す。
  • result.value_or(default)T | typeof(default)にナローイングされる。result.bind { |v| ... }Success上でフラットマップする。
  • case result; in Success[v]; ...; in Failure[k, v]; ...はパターンマッチングで、Rigorのナローイングはプリミティブレベルで既に理解している — プラグインはSuccess/Failureのデコンストラクションを教える必要がある。
  • do記法(yield Success(...))はFailureでの暗黙のショートサーキットと共に値をバインドする。推論においてはbindチェーンと等価。

ドキュメントに記載されたdry-*依存関係。なし。

プラグイン結合。dry-operationの基盤。rigor-dry-monadsがランディングされるとResult/Maybeはすべてのdryユーザーのファーストクラスのナローイングターゲットになる。

目的。ビジネスオペレーションのステップベースDSL。各step ...Resultをアンラップし、Failureでショートサーキットする。

プラグイン関連DSL

class CreateUser < Dry::Operation
def call(input)
attrs = step validate(input)
user = step persist(attrs)
step notify(user)
user
end
end

プラグインが発行する静的ファクト

  • Dry::Operation#callは常にResult[T, E]を返す。Tcall内の最終の非step式の型。Eは内部のstep呼び出しからの失敗型のunion。
  • step exprexprResult[T, E]からTSuccessペイロード)にナローイングする。
  • call末尾のベア値(上記のuser)は暗黙的にSuccessにラップされる。

ドキュメントに記載されたdry-*依存関係。dry-monads(ガイドが「dry-monadsを中心とした軽量DSL」で始まる)。

プラグイン結合。dry-monadsファクトのハードコンシューマー。

目的。代数的エフェクト — Dry::Effects.State(:counter)Dry::Effects::Handler.State(:counter)等。合成可能なハンドラを持つ副作用トラッキング。

プラグイン関連DSL。エフェクトはinclude Dry::Effects.X(...)でミックスインし、include Dry::Effects::Handler.X(...)でハンドルする。

プラグインが発行する静的ファクト。エフェクトはメソッドの戻り型を変更しない。ケイパビリティ要件を課す(マッチするハンドラが呼び出しサイトのスコープ内にあること)。Rigorの型ラティスでのモデリングは可能だが、v0.1.xキャリアとは直接整合しない。推奨: 延期。Rigorに明示的なエフェクトロウキャリアが現れた場合(現時点でこのADRはない)に再検討。

ドキュメントに記載されたdry-*依存関係。なし。

プラグイン結合。なし — ファミリーの残りに対して直交。

目的Import = Dry::AutoInject(Container); class X; include Import["users.repo"]; endattr_reader :users_repoとコンストラクタの配線を自動生成する。

プラグインが発行する静的ファクト

  • include Import["x.y.z"]は、名前がキーのリーフコンポーネント(またはその正規化形式)であり、コンテナのそのキーに登録された型が戻り型であるインスタンスリーダーを宣言する。
  • リーダーの型はコンテナのイントロスペクションなしには解決できない → コンパニオンのrigor-dry-container/rigor-dry-systemプラグイン、またはコンテナファクトをFactStoreとして消費するクロスプラグインAPI(ADR-9)が必要。

ドキュメントに記載されたdry-*依存関係Dry::ContainerDry::Systemのコンテナと互換。

プラグイン結合。クロスプラグイン(コンテナファクトを消費)。

目的。クラスまたはモジュールスコープの設定のためにextend Dry::Configurable; setting :foo, default: 1, reader: trueで、オプションのクラスレベルリーダー生成を行う。

プラグインが発行する静的ファクト

  • setting :foo, default: 1, reader: trueInteger(デフォルトの型)を返すKlass.foo/instance.fooを生成する。
  • ネストしたsetting :db do setting :dsn, default: '…' endKlass.config.db.dsnとしてアクセスできるネストした設定オブジェクトを生成する。
  • setting :foo, constructor: Types::String(サポートされる場合)はdry-typesにフィードバックする。

ドキュメントに記載されたdry-*依存関係。なし。

プラグイン結合。スタンドアロン。コンストラクタ形式が使用される場合はdry-typesへのフック付き。

目的。コンポーネントディレクトリからの自動登録を持つ依存関係コンテナ — Hanamiスライスの基盤。

プラグインが発行する静的ファクト

  • container.register(:key, instance)とコンポーネントディレクトリの自動登録がコンテナのkey→型マップを埋める。
  • container[:key]/container.resolve(:key)は登録された型を返す。
  • Hanamiスライス(Hanami.app["users.create"])は親のdry-systemコンテナを通じて解決される。

ドキュメントに記載されたdry-*依存関係。dry-core(Container)、dry-auto_inject。

プラグイン結合。dry-auto_injectプラグインが消費するファクトのプロデューサー。クロスプラグインAPI(ADR-9)が必要になる可能性が高い。

目的。スタンドアロンのスレッドセーフDIコンテナ。現在dry-coreにバンドルされている。dry-container gem自体は薄い再エクスポート。

静的ファクト。dry-systemのコンテナと同一のサーフェス(自動登録を除く)。

プラグイン結合。実際にはdry-systemに包含される。

階層D — ユーティリティ(静的型シェイプへの影響なし)

Section titled “階層D — ユーティリティ(静的型シェイプへの影響なし)”

これらのgemは型付きアクセサを宣言せず、DSLを通じてシェイプを持つ値を返さず、設定によって戻り型が変わるメソッドを生成しない。Rigorプラグインは基となるRBSが既にカバーするもの以上のものを発行することがない。

  • dry-cli — コマンドクラスへの引数パース。引数はランタイムに関係なく文字列型。
  • dry-core — 各種ヘルパー(キャッシュ、クラス属性、equalizer、コンテナ — レガシー分割については階層Eを参照)。各ヘルパーはその使用が現れるプラグインで最もよく処理される(例: dry-struct内のEqualizerコンシューマー)。
  • dry-events — pub/subバス。サブスクライバーはシェイプがアプリケーション定義のイベントハッシュを受け取る。
  • dry-files — ファイルシステム操作。
  • dry-inflector — 文字列変換。
  • dry-logger — 構造化ログ。
  • dry-monitor — インストゥルメンテーションフック。

階層E — レガシー/置き換え済み

Section titled “階層E — レガシー/置き換え済み”

一度のみリスト。現役の置き換えがプラグイン作業を担う。

  • dry-equalizerdry-coreに統合(Dry::Core::Equalizer)。
  • dry-matcherdry-monadsのパターンマッチングに置き換え済み。
  • dry-transactiondry-operationに置き換え済み。
  • dry-view → Hanami Viewにリネーム(dryファミリーのスコープ外。Hanamiプラグイントラックが開かれた場合は別途対応)。

dry-railsはdry-rb gemをRailsアプリに組み込むRailsのrailstie:

  • strong parametersを置き換えるコントローラーヘルパーsafe_params(dry-schemaで動作)。
  • ApplicationContract(dry-validationで動作)。
  • 自動インジェクション用のDepsミックスイン(dry-auto_injectで動作)。
  • 自動登録されるアプリケーションコンテナ(dry-systemで動作)。

rigor-dry-railsプラグインは新しいDSLサーフェスを追加しない。規約を宣言する — 「このアプリのコントローラーはdry-schemaのsafe_paramsを使用する」 — ためRigorはRailsレイアウトでdry-rbプラグインのファクトをどこで見つけるかを知る。プラグイン作者の観点からは、基底のrigor-dry-schema/rigor-dry-validation/rigor-dry-auto_inject/rigor-dry-systemプラグイン上の薄いコーディネーター。Railsプラグインロードマップとも重複する。将来の決定としてrigor-dry-railsrigor-rails-routes等のピアかあるいは両ファミリーに依存するグルーレイヤーかを解決すべき。

2種類のエッジが混在しており、それぞれラベル付き:

  • runtime — gemspecまたはガイドが直接requireを記述。
  • plugin — ソースのRigorプラグインはターゲットのRigorプラグインが生成したファクトを消費しなければならない。
dry-types — 基盤。dry-* 依存なし。
dry-logic — 基盤。dry-* 依存なし。
dry-monads — 基盤。dry-* 依存なし。
dry-effects — 基盤。dry-* 依存なし。
dry-configurable — 基盤。dry-* 依存なし。
dry-struct -> dry-types (runtime, plugin)
dry-schema -> dry-types (runtime, plugin)
-> dry-logic (runtime; 述語ファクトが浮上した場合のみplugin)
dry-validation -> dry-schema (runtime, plugin)
-> dry-types (runtime; pluginではdry-schema経由でトランジティブ)
dry-initializer -> dry-types (runtime依存なし; ユーザーがTypesをオプトインした場合plugin)
dry-operation -> dry-monads (runtime, plugin)
dry-container -> dry-core (runtime; plugin: 低影響)
dry-auto_inject -> dry-container OR dry-system (runtime; plugin: コンテナファクト)
dry-system -> dry-container (runtime, plugin)
-> dry-auto_inject (runtime; plugin: そのプロデューサー)
dry-rails -> dry-system (runtime, plugin)
-> dry-schema (runtime, plugin)
-> dry-validation (runtime, plugin)
-> dry-auto_inject (runtime, plugin)

フラグを立てるふたつのサイクル: dry-system ↔ dry-auto_inject(各gemのガイドが相手を参照)とdry-types ↔ dry-schema ↔ dry-validation(バリデーション契約がdry-schemaがdry-types経由で解釈するアドホック型をインラインで宣言できる)。両者はプラグイン作者が順序付けで解決できる依存関係の方向: まずプロデューサープラグイン(dry-typesdry-system)を構築し、次にコンシューマー(dry-schema/dry-validationdry-auto_inject)を構築する。

3つの現実的な切り分け案。それぞれのトレードオフを伴う。このドキュメントでは推奨しない — 選択は後続の設計ステップに委ねる。以下はトレードオフの説明であり、支持表明ではない。

階層AとB(オプションでC)をカバーするひとつのプラグインgem。

  • 賛成。単一のGemfileエントリー、単一のsemver、プラグイン間ファクトプロトコル不要(すべてが1つのプラグインのプロセスに収まる)。
  • 賛成。シンプルな初期作成 — v0.1.0プラグイン契約は実績がある。新しいクロスプラグインAPIサーフェスは不要。
  • 反対。リリースが無関係な変更を結合する(dry-monadの微調整がdry-structの修正と一緒に出荷される)。
  • 反対。部分的なdry-rb採用のユーザー(例: dry-struct + dry-typesのみ)がプラグインを明示的に内部でモジュラー化しない限り、使用していないgemの解析コストを負担する。
  • 反対。上流のgemのバージョンロックステップが乖離した場合(そして実際に乖離する — dry-types 1.7対1.8は異なるケイデンスで出荷される)、モノリシックなプラグインは最も遅いgemに追従しなければならない。

戦略2 — フルのrigor-dry-*ファミリー

Section titled “戦略2 — フルのrigor-dry-*ファミリー”

上流のgemごとにひとつのプラグイン(階層A: 5プラグイン、階層B: 2、階層C: 3 = dry-railsを除いて10個のgem)。

  • 賛成。各プラグインが上流gemのバージョンケイデンスをきれいに追跡する。
  • 賛成。ユーザーはアラカルトでオプトイン。Gemfileには実際に依存するdry-rbサーフェスのみをリスト。
  • 賛成。dry-rbの組織原則を反映 — 合成する小さく焦点を絞ったユニット。
  • 反対rigor-dry-validation/rigor-dry-auto_inject等のプラグインがジョブをこなすには、クロスプラグインAPIが必要(ADR-9)。ADR-9はv0.1.xにキューイングされているが未実装。
  • 反対。10のプラグインリポジトリ、10のCIパイプライン、10のCHANGELOG、10のsubtreeスプリット。

階層でグループ化した3〜4つのプラグイン:

  • rigor-dry-types-family — dry-types、dry-struct、dry-schema、dry-validation、dry-initializerをカバー(dry-logicを除く階層A)。

  • rigor-dry-monads-family — dry-monads、dry-operationをカバー(dry-effectsを除く階層B、延期)。

  • rigor-dry-system-family — dry-container、dry-auto_inject、dry-system、dry-configurableをカバー(階層C)。

  • rigor-dry-rails — 3つすべてに依存するコーディネーターgem。

  • 賛成。各バンドルは内部的に結束している: 共有された内部ファクトバス、ひとつのリリースサイクル、ひとりの作者がバンドル全体を頭の中に入れられる。

  • 賛成。バンドル間の受け渡し(dry-railsでの階層A ↔ 階層C、またはdry-typesからプルするdry-validation)だけがクロスプラグインAPIを必要とする — 戦略2よりADR-9への依存が少ない。

  • 反対。バンドル境界は部分的に慣習的。dry-structのみを使うユーザーでもschema/validationプラグインコードを引き込む。

  • 反対。各バンドル内部のモジュラリティはまだ設計する必要がある — そうしなければバンドルはより小さいスケールで戦略1と同じバージョニングリスクを持つミニモノリスになる。

パッケージング選択は2つの今後のアナライザー作業に影響される:

  • ADR-9クロスプラグインAPIdocs/adr/9-cross-plugin-api.md) — 別のプラグインのファクトを消費するプラグイン(dry-schemaの強制変換シェイプを必要とするdry-validation、dry-systemのコンテナマップを必要とするdry-auto_inject)には必須。戦略1はそれを回避する。戦略2と3はクロスプラグインの受け渡しにブロックされる。
  • v0.1.1正規表現→リファインメント名認識器docs/ROADMAP.md参照) — スライス1は未リリースでランディング済み。フルの認識器が出荷されると、gt?: 18format?: /\A.../等のdry-schemaの述語がビルトインのリファインメント名にきれいにマッピングされる。それまでは述語ファクトは記録されるが型ナローイングはされない。

どちらも、ドライを限定するMVP(dry-types + dry-struct + dry-monadsの3つのgem: 別のプラグインからファクトを消費する必要なくローカルにファクトを生成するもの)をブロックしない。その密接なサブセットは3つの戦略いずれでも妥当なv1である。

2026-05-09のこのサーベイのランディング後の議論でキャプチャされた解決事項。オープンアイテムは真に未決定。

  1. MVPタイミング — RESOLVED。dryプラグインを急がない。まずADR-9クロスプラグインAPIをランディングし、その後パッケージングを再検討する。これによりADR-9以前の唯一の実行可能なパスとして戦略1を強いたプレッシャーが取り除かれる。戦略2と3はADR-9が出荷された後のライブ候補になる。
  2. rigor-dry-railsの配置 — DELEGATED。強い好みなし — 作成が容易なファミリー下で実装する。プラグインがスキャフォールドされる時点で決定できる。事前にコミットしない。
  3. dry-effects — DEFERRED。エフェクトシステムサポートは原則として望ましいが具体的な計画はない。Rigorの型ラティスにエフェクトロウキャリアまたは類似のものが現れた場合に再検討。
  4. Hanami/romプラグイン — QUEUED。dry-rbプラグインがランディングした後のバージョンを対象とする。Hanamiプラグインはdry-systemプラグインを引き込む。romプラグインのスコープはここでスコープ未定。
  5. dry-rb gemspecの検証 — OPEN。まだ強い意見なし。ADR-12でパッケージング戦略を確定する前に価値がある可能性が高いが、ブロッキングではない。

ADR-9クロスプラグインAPIをランディングする。その後dry-rbパッケージング戦略の選択をキャプチャするADR-12を提出する。最初に作成するプラグイン — どの戦略でも — はrigor-dry-types。他のすべての階層Aプラグインがそれに依存するため。

© 2026 TypedDuck. Licensed under CC BY-SA 4.0.