コンテンツにスキップ

rigor-survey project-init baseline sweep — 2026-05-29

~/repo/ruby/rigor-survey/内のすべてのプロジェクトについて、rigor-project-init直後のスナップショットをrigortype 0.1.14で取得したもの。

目的は、プロジェクトがオンボーディングされた瞬間に、ベースラインやpre_eval:のチューニングを行う前にrigor checkが何を顕在化させるかを見ることにある — すなわち、Rigorがコールドなプロジェクトで「回収」できるundefined-method / possible-nil-receiver / フロー問題の生の集合である。これは意図的に未削減の状態だ: .rigor-baseline.ymlは一切生成しておらず、したがって何も抑制されていない。

1つのドライバ(_reports/init/run-init.sh、Flakeシェル内で実行)が、各プロジェクトについて次を行った:

  1. リセット — 既存の.rigor.dist.yml / .rigor.yml / .rigor-baseline.yml / .rigor/キャッシュを削除する。(以前にチューニング済みだったmastodonとredmineの設定は、先に_reports/init/_backup/へバックアップした。)
  2. 新しい.rigor.dist.ymlを書く — project-initワークフローに従う:
    • paths: = ライブラリはlib(protobufはruby/lib、2つのRailsアプリはapp+lib)、exclude: [vendor, tmp]
    • target_ruby:.ruby-versionが宣言されていればそこから取得。
    • ローカルのsig/は存在すればsignature_paths:で配線(herb、redmine)。
    • プラグインは2つの本物のRailsアプリ(mastodon、redmine)にのみ: Railsセット + rigor-activesupport-core-extseverity_profile: lenient。それ以外のプロジェクトはすべて素のRuby → プラグインなし、デフォルト(balanced)プロファイル。これにより、ライブラリのシグナルをプラグインRBSで事前フィルタせず生のまま保つ。
  3. 生のrigor check_reports/init/<proj>.check.txt)とrigor triage --format json_reports/init/<proj>.triage.json)をキャプチャする。

集計は_reports/init/aggregate.rbによる。

注 — ベースラインなし / pre_eval:なし。実際のオンボーディング(Phase 6a/7)であれば、証明済みのモンキーパッチ箇所にpre_eval:を追加し、残りをベースラインへスナップショットするだろう。したがって以下のredmineとmastodonの数値は膨張したコールドの数値である。以前にチューニングされたredmineの設定(pre_eval: + ベースライン付き)は、顕在化された件数をここに示す4101のごく一部にまで下げていた。

ProjectFilesTotalErrWarnInfoTop rulesTriage hints
algorithms14373331possible-nil-receiver:33, always-truthy:3systemic-file-cluster, genuine-bugs×3
concurrent-ruby178131030possible-nil-receiver:4, undefined-method:3, arg-type-mismatch:2genuine-bugs×13
erubi33300possible-nil-receiver:2, undefined-method:1genuine-bugs×3
faraday338800undefined-method:5, possible-nil-receiver:3genuine-bugs×8
haml51121020possible-nil-receiver:5, undefined-method:5activesupport-core-ext, unresolved-toplevel, genuine-bugs×8
hamlit61141310possible-nil-receiver:10, undefined-method:3activesupport-core-ext, genuine-bugs×1
herb4211641undefined-method:5, unresolved-toplevel:2, return-type-mismatch:2gem-without-rbs, unresolved-toplevel, genuine-bugs×8
jbuilder143120ivar-write-mismatch:2, undefined-method:1activesupport-core-ext, genuine-bugs×2
kramdown55141220undefined-method:8, possible-nil-receiver:4genuine-bugs×6
liquid64131030undefined-method:7, possible-nil-receiver:3genuine-bugs×6
mail111205150unreachable-branch:11, possible-nil-receiver:4genuine-bugs×9
mastodon121919203231894activerecord.model-call:555, actionpack.filter-call:527, routes.helper:350, i18n.translation:233systemic-file-cluster, genuine-bugs×10
net-ssh972713140always-truthy:11, possible-nil-receiver:6, undefined-method:5genuine-bugs×10
numo-narray26510possible-nil-receiver:5, always-truthy:1genuine-bugs×6
oj115500undefined-method:3, possible-nil-receiver:2genuine-bugs×5
ox15121200possible-nil-receiver:10, arg-type-mismatch:2systemic-file-cluster, genuine-bugs×2
parser567340always-truthy:4, possible-nil-receiver:3genuine-bugs×7
protobuf24161600undefined-method:13, possible-nil-receiver:2, wrong-arity:1systemic-file-cluster, genuine-bugs×6
pycall22100100unresolved-toplevel:10unresolved-toplevel×10
rbnacl370000(clean)
redmine3314101335231718undefined-method:3319, routes.helper:313, actionpack.helper:180project-monkey-patch×1665, systemic-file-cluster×103
rgl280000(clean)
rubocop-ast994130always-truthy:2, undefined-method:1, ivar-write:1genuine-bugs×4
slim277340undefined-method:2, ivar-write-mismatch:2genuine-bugs×7
tdiary-core6924852421unresolved-toplevel:234, possible-nil-receiver:4unresolved-toplevel×234, genuine-bugs×13
RuleTotal
call.undefined-method3384
plugin.actionpack.filter-call684
plugin.rails-routes.helper663
plugin.activerecord.model-call555
plugin.actionpack.helper-call398
plugin.rails-i18n.translation-call258
call.unresolved-toplevel247
call.possible-nil-receiver141
flow.always-truthy-condition67
call.wrong-arity28
flow.dead-assignment24
flow.unreachable-branch11
plugin.rails-routes.unknown-helper11
def.return-type-mismatch10
call.argument-type-mismatch9
def.ivar-write-mismatch8
plugin.actionmailer.mailer-call7
rbs.coverage.missing-gem4
plugin.actionmailer.missing-view1
plugin.activerecord.load-error1

call.undefined-methodの合計(3384)は97%がredmine(3319)である。2つのRailsアプリを取り除けば、ライブラリのコーパスは小さく扱いやすい: 23のライブラリにわたって診断は数百件で、大半はpossible-nil-receiverundefined-method、そしてフロー警告である。

数値の意味 — シグナル対オンボーディングのノイズ

Section titled “数値の意味 — シグナル対オンボーディングのノイズ”

生の合計はきれいに3つのバケツに分かれる。

1. 実際のinitなら解消されるオンボーディングのノイズ(大きな数値)

Section titled “1. 実際のinitなら解消されるオンボーディングのノイズ(大きな数値)”
  • redmine — 3319件のundefined-method / 1665件のproject-monkey-patch。ほぼすべてがRedmineによるコア / stdlib / モデルクラスの再オープンと、クロスファイルでのメソッド追加である: User.currentMailer.deliver_*Setting.lost_password?Token.find_tokenUser.find_by_mail、……。これらはまさに以前にチューニングされたredmineの設定がpre_eval:に列挙していた箇所である(そしてベースラインへスナップショットしていた)。トリアージのproject-monkey-patchヒント(件数1665)がこれを正しくフラグしている。これはコールドの数値であり、Phase 6aのpre_eval:がこれを潰す
  • mastodon — 1894件のinfoerrorはわずか3件。大半は情報的なプラグイン認識である: plugin.activerecord.model-call(555)、plugin.actionpack.filter-call(527)、plugin.rails-routes.helper(350)、plugin.rails-i18n.translation-call(233)。これらはRigorがRailsのDSL呼び出しを認識しているものであって、バグをフラグしているのではない。mastodonの実際のエラー件数は3件(後述)。
  • tdiary-core / pycall — unresolved-toplevel(234 / 10)。tdiaryのプラグインシステムは、プラグインファイルをまたいで消費されるヘルパー(hto_nativebot?)をトップレベル / Kernelメソッドとして定義する。pycallのiruby_helperregister_python_object_formatterを呼ぶ。ADR-34の診断は設計どおりちょうど発火し、メッセージはすでに修正法(ADR-17に従うpre_eval:)を指し示している。すべてbalancedの下で警告となる。

2. 「genuine-bugs」ヒント — ライブラリレビューの山

Section titled “2. 「genuine-bugs」ヒント — ライブラリレビューの山”

すべてのライブラリがgenuine-bugsヒント(少数で散在するルール)を得た。繰り返し現れる形:

  • call.possible-nil-receiver — 単独で最も多いライブラリの所見(algorithms 33、ox 10、hamlit 10、numo-narray 5)。[feedback_false_positive_discipline]に従えば、大半はテストスイートが安全だと証明しているコードに対する最悪ケース健全なT | nilの読み取りであり、強制的な修正ではなくベースライン材料である。少数(例: nil許容のインデックスに対するox/element.rbの比較チェーン)は人間が見る価値がある。
  • stdlib / ネイティブレシーバー上のcall.undefined-method — 大半はRBSカバレッジのギャップであり、バグではない:
    • oj: singleton(JSON::Ext::Generator::State)上のfrom_state(ネイティブ)。
    • faraday: URI上のfind_proxyClass上のoptions_for / member_set
    • protobuf: Numeric上のto_i / to_f(×13 — 抽象的なNumericto_iを持たない。値は実行時には具体的にInteger/Floatである)。
    • erubi: Object上のescapeHTML(CGIがmixinされている)。
    • rubocop-ast: Binding上のunion_bind
    • mastodon: Resolv::DNS::Resource上のexchange。 これらは組み込み / stdlibのRBSカバレッジ不足、またはネイティブ拡張メソッドを指している — プロジェクトのバグではなくrigor-builtin-import作業の候補である。
  • フロー警告always-truthy/falseydead-assignmentunreachable-branch)— mailの11件のunreachable-branch、net-sshの11件のalways-truthy。たいていは冗長なガード / 防御的な死コードであり、重大度は低いが、ときに本物のロジックの臭いになる。
  • def.ivar-write-mismatch(concurrent-rubyの@backend Concurrent::Hash→Hash、rubocop-astの@cur_index Symbol→Integer)— インスタンス変数が異なる型で再代入されている。軽度の一貫性の所見である。

3. 純粋にクリーンなプロジェクト

Section titled “3. 純粋にクリーンなプロジェクト”

rbnacl、rglは診断ゼロを報告する。pycallはエラーがゼロ(トップレベル警告のみ)である。特筆すべきは、rglが以前(5月19日のサーベイ、0.1.14以前)は暗黙的self呼び出しに対してObject上のundefined-methodを2件報告していたことだ — それらはいまやADR-24のself-method-call解決によって沈黙させられている。コーパスに見える具体的な精度向上である。

最も価値の高いフォローアップ(実行可能な「undefined」の所見)

Section titled “最も価値の高いフォローアップ(実行可能な「undefined」の所見)”

オンボーディングのノイズを捨てたあと、人間が一度目を通す価値が最も高い診断:

  1. mastodonの3件の本物のエラーmedium_player_urlルートヘルパーが不明(×2、plugin.rails-routes.unknown-helper)とResolv::DNS::Resource#exchange(RBSギャップ)。ルートヘルパーのものは、1219ファイルの中でアプリレベルの問題の可能性に見える唯一のもの(動的な / 削除されたルート)である。
  2. herbHerb.diff / Herb.leak_check / Herb.arena_stats(singleton(Herb))とHerb::AST::Node#tag_openingに対するundefined-method、加えて2件のdef.return-type-mismatch。herbは自前のsig/を出荷しているので、これらはプロジェクト自身のRBSにおけるsig完全性のギャップである — rigor sig-genパスで解消できるだろう。
  3. ライブラリのpossible-nil-receiverクラスタ(ox、numo-narray、algorithms)— nil許容のインデックス / shapeチェーンをレビューする。大半はベースライン、いくつかは本物かもしれない。
  4. RBSカバレッジ起因のundefined-method(oj、faraday、protobuf、erubi、rubocop-ast)— コア / stdlibのRBSカバレッジ(Numeric#to_iURI#find_proxyBinding、CGIのescapeHTML)へ送り込む。プロジェクトの修正ではない。

コールド対pre_eval:チューニング済みのinit — 2つのRailsアプリ

Section titled “コールド対pre_eval:チューニング済みのinit — 2つのRailsアプリ”

コールドの数値を受けて、適切なPhase-6a init(トリアージのproject-monkey-patchヒントに従い、名指しされたdef箇所をpre_eval:へ追加する)が実際に何をもたらすかを調べた。run-tuned-rails.sh経由で_reports/init/<proj>.tuned.*としてキャプチャした。

Apptotalerrorwarninfoundefined-method
redminecold41013352317183319
redmine+pre_eval34882739317182706
mastodoncold192032318941
mastodon+pre_eval192032318941

redmine — pre_eval:静的に定義されたパッチのみを解消する(−613)

Section titled “redmine — pre_eval:は静的に定義されたパッチのみを解消する(−613)”

pre_eval:には、トリアージのヒントが指した6つのファイル(app/models/{user,setting,mailer,token}.rblib/redmine/{configuration,twofa}.rb)を列挙した。結果: undefined-methodが−613件(3319 → 2706)、project-monkey-patchヒントは1665 → 1108に縮小した。

レシーバー別に、何が解消され何が残ったか:

  • 解消された — User.current(486 → 0)user.rb内にリテラルなdef self.currentとして定義されている。pre_eval:のウォーカーがそれを見て、すべての呼び出し元が解決する。この単一のクラスタが削減の約80%である。Mailer.deliver_*Token.find_*Redmine::Configuration[]も同様。
  • 解消されなかった — Setting.default_language / app_title / …(依然それぞれ約20/18件)Settingはそのアクセサを静的なdefではなくmethod_missing経由で公開する — そのためファイルの事前パスを歩いても登録すべきものが見つからない。これはエスカレーションパスAの境界である: method_missingによるDSLはpre_eval:ではなくプロジェクトプラグインを必要とする。
  • 解消されなかった — モデルのシングルトン上のtable_name / where / visible / スコープ(table_name単独でIssue/Project/TimeEntry/…にまたがり約470件)。これらはActiveRecordのクラスメソッドである。rigor-activerecordは現状これらをsingleton(Model)上に供給しない。モンキーパッチではなく、プラグイン/RBSのカバレッジギャップである。これが支配的な残余であり、redmineがpre_eval:後も4桁にとどまる真の理由である。
  • 解消されなかった — Redmine::Export::PDF::ITCPDF上のSetFontStyle / RDMCell / ln。RBSのないrbpdf/TCPDF gemのサブクラス → gem-without-rbs

したがって率直な読み方はこうだ: pre_eval:証明済みで静的に定義されたモンキーパッチのクラスタに対する正しいツールであり(それをきれいに除去する)、しかしredmineではそれはコールド件数の少数派である。多数派はActiveRecordのクラスメソッドのカバレッジ + method_missingによるDSL + RBSのないPDF gemであり — これらをacknowledgeモードのワークフローはベースラインへ送り込む(以前のチューニング済みredmine設定がまさにそれを行っていた: pre_eval: + 約37 KBのベースラインが、顕在化された件数をほぼゼロに近づけた)。pre_eval:はベースラインを縮小するのであって、それを置き換えるのではない。

mastodon — pre_eval:は無効(すでに底に達している)

Section titled “mastodon — pre_eval:は無効(すでに底に達している)”

前後で診断レベルまで同一である。トリアージはmastodonに対してproject-monkey-patchヒントを報告しなかったので、登録すべきdef箇所がなかった。その1894件のinfoplugin.*によるRails-DSLの認識(エラーではない)であり、実際のサーフェスは3件のエラー + 23件の警告 — すでに最小である。mastodonの忠実なinitはpre_eval:を追加しない。そこでのレバーは、RBSのない325個のgemに対するrbs collection installinfo認識ノイズを縮小する)であって、モンキーパッチの登録ではない。

pre_eval:ステップは、プロジェクトの「undefined」サーフェスのうち、どれだけがプロジェクト内で静的に定義された再オープンであるかに比例して効果を発揮する — redmineのUser.currentスタイルのパッチには大きく、mastodonにはゼロである。それはmethod_missingによるDSL(→ プロジェクトプラグイン)、ActiveRecordのクラスメソッドのギャップ(→ プラグイン/RBSカバレッジ)、RBSのないgem(→ rbs collection install / source_inference:)には触れない。acknowledgeモードでは残余はベースライン材料であり、pre_eval:は単にそのベースラインをより小さく、より率直にするだけである。

  • コールドの数値pre_eval:なし、ベースラインなし。redmine/mastodonの合計は未オンボーディングの状態を反映する。上記のファミリー別分析は、それを勘案したあとの率直なシグナルである。
  • ライブラリのプラグインは意図的に省略。いくつかのライブラリはActiveSupportに依存しており(jbuilder、haml、hamlit、mail)、トリアージはそれらにactivesupport-core-extをフラグした。そのプラグインを追加すればいくつかのundefined-method件数は縮小する。生のコアシグナルを保つためにここでは外した。
  • 0.1.14のスナップショット。後のエンジンで再実行すればドリフトする(rglのADR-24の勝利を参照)。キャプチャはdiff用に_reports/init/配下に置いてある。
  • 以前のチューニング済み設定のバックアップ: _reports/init/_backup/

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