🤯

Goのnil panicを防ぐ静的解析ツール:nilaway

2024/10/24に公開

nilaway

https://github.com/uber-go/nilaway

どんなもの?

  • nil パニックによるランタイムエラーになる箇所を指摘してくれる静的解析ツール
  • スタンドアロンで動く。golangci-lintで使用するにはカスタマイズが必要。

先行技術やツールと比べてどこがすごい?

  • 標準パッケージに同梱されているnilness[1]ではチェックできないnil パニックとなりうる箇所を指摘する
    • 関数やパッケージの境界を跨いでnilフローを追跡
    • nil インターフェース値を報告

例えば以下のコードでは、nilnessはエラーを報告してくれないが、nilawayはnilパニックを報告してくれる。

var p *P
if someCondition {
      p = &P{}
}
print(p.f) // nilness reports NO error here, but nilaway does.

技術のキモはどこ?

Our main idea is that nilability flows in code can be modeled as a system of global typing constraints, which can then be solved using a 2-SAT algorithm to determine potential contradictions
Core Idea of nilaway[2]

コード内のnil可能性フローを型付け制約のシステムとしてモデル化し、2-SATアルゴリズムでnilable constraintからnonnil constaintにnil 値が流れている箇所を割り出しているところ。

どういうこと?

2-SATアルゴリズム[3]

  • いくつかの変数とそれらの変数に対する「AならばB」といった制約条件が与えられたときに、全ての制約条件を満たすような変数の値の組み合わせ(真偽)を見つけられるかどうかを判定する。
  • 例:「雨が降ったら傘をさす」という制約条件は、変数「雨」と「傘」が、「雨が真ならば傘も真」という関係になっていると考えることができる。2-SAT問題では、このような制約条件がたくさん与えられ、それらをすべて満たせるかを判定する。
  • nilawayでは、プログラム中のnil値のフロー(nil値がどこから来てどこへ行くのか)を、この2-SAT問題としてモデル化し、nilable constraintからnonnil constaintにnil値が流れている箇所(矛盾)を見つけている。

nilable constraint(以降、nilable 制約)

  • 変数や値がnilになる可能性があるということを表す制約
  • 以下コードでは、関数fooはstring型のポインタを返しているが、変数xは初期化されていない。そのため、xはnilである可能性があり、return xはnilable 制約となる。
func foo() *string {
  var x *string
  return x
}

nonnil constraint(以降、non nil制約)

  • 変数や値がnilではないということを表す制約
func bar(x *string) {
  fmt.Println(*x)
}

このコードでは、関数barはstring型のポインタを引数として受け取り、そのポインタが指す値を表示している。ここで、ポインタxがnilの場合、*xによる参照はnil パニックを引き起こすため、*xはnon nil制約となる。

具体例

以下のコードを実行するとnil パニックになる。objはnilになるにも関わらず、メソッドを呼び出そうとしているため。

type MyInterface interface {
	Method() int
}

func MyFunc() MyInterface {
	return nil
}

func main() {
	obj := MyFunc()
	result := obj.Method() // panic: runtime error: invalid memory address or nil pointer dereference
	fmt.Println(result)
}

ざっくり、筆者が理解した範囲で処理の流れを追ってみる。

  1. nilable constraintnonnil constraintの収集

    1. MyFunc() 関数の戻り値が MyInterface 型の nilインターフェース値を返しているため、nilable 制約としてマーク。
    2. main() 関数内の obj.Method() の呼び出しは、obj がnon nil制約であることをマーク。インターフェースのメソッドを呼び出すには、インターフェースが具体的な型の実装へのポインタを保持している必要があるため。
  2. 含意グラフの構築

    1. 収集した制約に基づいて、次のような含意グラフ[4]を構築する。
      1. ノード: MyFunc() の戻り値, obj
      2. エッジ: MyFunc() の戻り値 -> obj
  3. 含意グラフの走査:

    1. nilaway は、MyFunc() の戻り値が nil であるという情報から始め、含意グラフのエッジを介して obj に伝播する。その結果、obj も nil であると推論される。
  4. 矛盾の検出

    1. nilaway は、obj に対して nil 可能とnon nil の両方の制約があることを検出する。obj が nil の場合に obj.Method() を呼び出すと nil パニックが発生することを意味し、エラーとして報告する。

次に読むべき情報は?

Uber Blog -engineering

nilawayのアーキテクチャ

おわりに

注意点ですが、公式リポジトリで言及されている通り、nilaway は現在開発中というステータスです。ドキュメントが充実していない・一部欠けているなど注意が必要です。

とはいえ、気づきにくいnilパニックを報告してくれる実用的ツールです。皆さんもプロジェクトに組み込んで試してみてはいかがでしょうか。

脚注
  1. https://pkg.go.dev/golang.org/x/tools/go/analysis/passes/nilness ↩︎

  2. https://www.uber.com/en-JP/blog/nilaway-practical-nil-panic-detection-for-go/#:~:text=Our main idea is that nilability flows in code can be modeled as a system of global typing constraints%2C which can then be solved using a 2-SAT algorithm to determine potential contradictions ↩︎

  3. https://en.wikipedia.org/wiki/2-satisfiability?uclick_id=3a0e8c6c-f4fe-4f94-9448-eb11df90bde1 ↩︎

  4. https://en.wikipedia.org/wiki/Implication_graph ↩︎

GitHubで編集を提案

Discussion