From 6717e47b00d32d7bcd8e9dfa100e5a92df87853a Mon Sep 17 00:00:00 2001 From: Schaechtle Date: Tue, 9 Apr 2024 17:03:10 -0400 Subject: [PATCH 1/2] fix: Make importance resampling work when some samples have weight log(0) --- src/gen/inference/importance.cljc | 11 ++++++++--- 1 file changed, 8 insertions(+), 3 deletions(-) diff --git a/src/gen/inference/importance.cljc b/src/gen/inference/importance.cljc index 1fb04bb..6e9b577 100644 --- a/src/gen/inference/importance.cljc +++ b/src/gen/inference/importance.cljc @@ -22,6 +22,10 @@ (+ (math/log nr) x) (recur rst nr (double x))))))) +(defn- neg-inf? + [v] + (= v ##-Inf)) + (defn resampling [gf args observations n-samples] ;; https://github.com/probcomp/Gen.jl/blob/master/src/inference/importance.jl#L77...L95 (let [result (gf/generate gf args observations) @@ -31,9 +35,10 @@ (let [candidate (gf/generate gf args observations) candidate-model-trace (:trace candidate) log-weight (:weight candidate)] - (vswap! log-total-weight #(logsumexp [log-weight %])) - (when (dist/bernoulli (math/exp (- log-weight @log-total-weight))) - (vreset! model-trace candidate-model-trace)))) + (when-not (neg-inf? log-weight) + (vswap! log-total-weight #(logsumexp [log-weight %])) + (when (dist/bernoulli (math/exp (- log-weight @log-total-weight))) + (vreset! model-trace candidate-model-trace))))) (let [log-ml-estimate (- @log-total-weight (math/log n-samples))] {:trace @model-trace :weight log-ml-estimate}))) From e6b6fa6a09d21dcd5a01d05d10456cb3ef1a1bcc Mon Sep 17 00:00:00 2001 From: Schaechtle Date: Tue, 9 Apr 2024 17:05:03 -0400 Subject: [PATCH 2/2] test: Test importance sampling with a model and constraints resulting in samples with weight 0 This crashed previously. --- test/gen/inference/importance_test.cljc | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) create mode 100644 test/gen/inference/importance_test.cljc diff --git a/test/gen/inference/importance_test.cljc b/test/gen/inference/importance_test.cljc new file mode 100644 index 0000000..d3be762 --- /dev/null +++ b/test/gen/inference/importance_test.cljc @@ -0,0 +1,23 @@ +(ns gen.inference.importance-test + (:require [clojure.test :refer [deftest is testing]] + [gen.choicemap :as choicemap :refer [choicemap get-value]] + [gen.distribution.kixi :as dist] + [gen.dynamic :as dynamic :refer [gen]] + [gen.inference.importance :as importance] + [gen.trace :as trace])) + +(def model-causing-rejection-sampling + (gen + [] + (if (dynamic/trace! :foo dist/bernoulli 0.5) + (dynamic/trace! :bar dist/bernoulli 1.0) + (dynamic/trace! :bar dist/bernoulli 0.0)))) + +(deftest rejection + (testing "Robustness in the presence of importance samples with weight log(0)." + (is {:foo true :bar true} + ;; Needs a couple of samples to trigger previous bug here. + (-> (importance/resampling model-causing-rejection-sampling [] (choicemap {:bar true}) 10) + (:trace) + (trace/get-choices) + (get-value)))))