Source code for graph_tool.inference.mcmc

#! /usr/bin/env python
# -*- coding: utf-8 -*-
#
# graph_tool -- a general graph manipulation python module
#
# Copyright (C) 2006-2025 Tiago de Paula Peixoto <tiago@skewed.de>
#
# This program is free software; you can redistribute it and/or modify it under
# the terms of the GNU Lesser General Public License as published by the Free
# Software Foundation; either version 3 of the License, or (at your option) any
# later version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
# details.
#
# You should have received a copy of the GNU Lesser General Public License
# along with this program. If not, see <http://www.gnu.org/licenses/>.

from .. import Vector_size_t, Vector_double

import numpy as np
from . util import *
from . base_states import *
from . nested_blockmodel import NestedBlockState

[docs] def mcmc_equilibrate(state, wait=1000, nbreaks=2, max_niter=np.inf, force_niter=None, epsilon=0, gibbs=False, multiflip=True, mcmc_args={}, entropy_args={}, history=False, callback=None, verbose=False): r"""Equilibrate a MCMC with a given starting state. Parameters ---------- state : Any state class (e.g. :class:`~graph_tool.inference.BlockState`) Initial state. This state will be modified during the algorithm. wait : ``int`` (optional, default: ``1000``) Number of iterations to wait for a record-breaking event. nbreaks : ``int`` (optional, default: ``2``) Number of iteration intervals (of size ``wait``) without record-breaking events necessary to stop the algorithm. max_niter : ``int`` (optional, default: ``numpy.inf``) Maximum number of iterations. force_niter : ``int`` (optional, default: ``None``) If given, will force the algorithm to run this exact number of iterations. epsilon : ``float`` (optional, default: ``0``) Relative changes in entropy smaller than epsilon will not be considered as record-breaking. gibbs : ``bool`` (optional, default: ``False``) If ``True``, each step will call ``state.gibbs_sweep`` instead of ``state.mcmc_sweep``. multiflip : ``bool`` (optional, default: ``True``) If ``True`` and ``state`` is an instance of :class:`~graph_tool.inference.MultiflipMCMCState`, each step will call ``state.multiflip_mcmc_sweep`` instead of ``state.mcmc_sweep``. mcmc_args : ``dict`` (optional, default: ``{}``) Arguments to be passed to ``state.mcmc_sweep`` (or ``state.gibbs_sweep``). history : ``bool`` (optional, default: ``False``) If ``True``, a list of tuples of the form ``(nattempts, nmoves, entropy)`` will be kept and returned, where ``entropy`` is the current entropy and ``nmoves`` is the number of vertices moved. callback : ``function`` (optional, default: ``None``) If given, this function will be called after each iteration. The function must accept the current state as an argument, and its return value must be either `None` or a (possibly empty) list of values that will be append to the history, if ``history == True``. verbose : ``bool`` or ``tuple`` (optional, default: ``False``) If ``True``, progress information will be shown. Optionally, this accepts arguments of the type ``tuple`` of the form ``(level, prefix)`` where ``level`` is a positive integer that specifies the level of detail, and ``prefix`` is a string that is prepended to the all output messages. Notes ----- The MCMC equilibration is attempted by keeping track of the maximum and minimum values, and waiting sufficiently long without a record-breaking event. This function calls ``state.mcmc_sweep`` (or ``state.gibbs_sweep``) at each iteration (e.g. :meth:`graph_tool.inference.BlockState.mcmc_sweep` and :meth:`graph_tool.inference.BlockState.gibbs_sweep`), and keeps track of the value of ``state.entropy(**args)`` with ``args`` corresponding to ``mcmc_args["entropy_args"]``. Returns ------- history : list of tuples of the form ``(nattempts, nmoves, entropy)`` Summary of the MCMC run. This is returned only if ``history == True``. entropy : ``float`` Current entropy value after run. This is returned only if ``history == False``. nattempts : ``int`` Number of node move attempts. nmoves : ``int`` Number of node moves. References ---------- .. [peixoto-efficient-2014] Tiago P. Peixoto, "Efficient Monte Carlo and greedy heuristic for the inference of stochastic block models", Phys. Rev. E 89, 012804 (2014), :doi:`10.1103/PhysRevE.89.012804`, :arxiv:`1310.4378` """ count = 0 break_count = 0 niter = 0 total_nmoves = 0 total_nattempts = 0 S = state.entropy(**mcmc_args.get("entropy_args", {})) min_S = max_S = S m_eps = 1e-6 hist = [] while count < wait: if gibbs: delta, nattempts, nmoves = state.gibbs_sweep(**mcmc_args) elif multiflip and hasattr(state, "multiflip_mcmc_sweep"): delta, nattempts, nmoves = state.multiflip_mcmc_sweep(**mcmc_args) else: delta, nattempts, nmoves = state.mcmc_sweep(**mcmc_args) S += delta niter += 1 total_nmoves += nmoves total_nattempts += nattempts if force_niter is not None: max_S = max((S, max_S)) min_S = min((S, min_S)) if niter >= force_niter: break else: if abs(delta) >= (S - delta) * epsilon: if S > max_S + m_eps: max_S = S count = 0 elif S < min_S - m_eps: min_S = S count = 0 else: count += 1 else: count += 1 if count >= wait: break_count += 1 if break_count < nbreaks: count = 0 min_S = max_S = S extra = [] if callback is not None: extra = callback(state) if extra is None: extra = [] if check_verbose(verbose): print((verbose_pad(verbose) + u"niter: %5d count: %4d breaks: %2d min_S: %#8.8g " + u"max_S: %#8.8g S: %#8.8g ΔS: %#12.6g moves: %5d %s") % (niter, count, break_count, min_S, max_S, S, delta, nmoves, str(extra) if len(extra) > 0 else "")) if history: hist.append(tuple([nattempts, nmoves, S] + extra)) if niter >= max_niter: break if history: return hist else: return (S, total_nattempts, total_nmoves)
[docs] def mcmc_anneal(state, beta_range=(1., 10.), niter=100, history=False, mcmc_equilibrate_args={}, verbose=False): r"""Equilibrate a MCMC at a specified target temperature by performing simulated annealing. Parameters ---------- state : Any state class (e.g. :class:`~graph_tool.inference.BlockState`) Initial state. This state will be modified during the algorithm. beta_range : ``tuple`` of two floats (optional, default: ``(1., 10.)``) Inverse temperature range. niter : ``int`` (optional, default: ``100``) Number of steps (in logspace) from the starting temperature to the final one. history : ``bool`` (optional, default: ``False``) If ``True``, a list of tuples of the form ``(nattempts, nmoves, beta, entropy)`` mcmc_equilibrate_args : ``dict`` (optional, default: ``{}``) Arguments to be passed to :func:`~graph_tool.inference.mcmc_equilibrate`. verbose : ``bool`` or ``tuple`` (optional, default: ``False``) If ``True``, progress information will be shown. Optionally, this accepts arguments of the type ``tuple`` of the form ``(level, prefix)`` where ``level`` is a positive integer that specifies the level of detail, and ``prefix`` is a string that is prepended to the all output messages. Notes ----- This algorithm employs exponential cooling, where the value of beta is multiplied by a constant at each iteration, so that starting from `beta_range[0]` the value of `beta_range[1]` is reached after `niter` iterations. At each iteration, the function :func:`~graph_tool.inference.mcmc_equilibrate` is called with the current value of `beta` (via the ``mcmc_args`` parameter). Returns ------- history : list of tuples of the form ``(nattempts, nmoves, beta, entropy)`` Summary of the MCMC run. This is returned only if ``history == True``. entropy : ``float`` Current entropy value after run. This is returned only if ``history == False``. nattempts : ``int`` Number of node move attempts. nmoves : ``int`` Number of node moves. References ---------- .. [peixoto-efficient-2014] Tiago P. Peixoto, "Efficient Monte Carlo and greedy heuristic for the inference of stochastic block models", Phys. Rev. E 89, 012804 (2014), :doi:`10.1103/PhysRevE.89.012804`, :arxiv:`1310.4378` """ beta = beta_range[0] hist = ([], [], [], []) nattempts = 0 nmoves = 0 speed = np.exp((np.log(beta_range[1]) - np.log(beta_range[0])) / niter) mcmc_args = mcmc_equilibrate_args.get("mcmc_args", {}) while beta < beta_range[1] * speed: ret = mcmc_equilibrate(state, **dict(mcmc_equilibrate_args, mcmc_args=dict(mcmc_args, beta=beta), history=history, verbose=verbose_push(verbose, ("β: %#8.6g " % beta)))) if history: ret = list(zip(*ret)) hist[0].extend(ret[0]) hist[1].extend(ret[1]) hist[2].extend([beta] * len(ret[0])) hist[3].extend(ret[2]) S = ret[0][-1] else: S = ret[0] nattempts += ret[1] nmoves += ret[2] beta *= speed if history: return list(zip(hist)) else: return S, nattempts, nmoves
[docs] class MulticanonicalState(object): r"""The density of states of a multicanonical Monte Carlo algorithm. It is used by :func:`graph_tool.inference.multicanonical_equilibrate`. Parameters ---------- state : :class:`~graph_tool.inference.BlockState` or :class:`~graph_tool.inference.OverlapBlockState` or :class:`~graph_tool.inference.NestedBlockState` Block state to be used. S_min : ``float`` Minimum energy. S_max : ``float`` Maximum energy. nbins : ``int`` (optional, default: ``1000``) Number of bins. """ def __init__(self, state, S_min, S_max, nbins=1000): self._state = state self._g = state.g self._N = self._g.num_vertices() self._S_min = S_min self._S_max = S_max self._density = Vector_double() self._density.resize(nbins) self._hist = Vector_size_t() self._hist.resize(nbins) self._perm_hist = np.zeros(nbins, dtype=self._hist.a.dtype) self._f = None def __getstate__(self): state = [self._state, self._S_min, self._S_max, np.array(self._density.a), np.array(self._hist.a), np.array(self._perm_hist), self._f] return state def __setstate__(self, state): bstate, S_min, S_max, density, hist, phist, self._f = state self.__init__(bstate, S_min, S_max, len(hist)) self._density.a[:] = density self._hist.a[:] = hist self._perm_hist[:] = phist
[docs] def sweep(self, **kwargs): self._state.multicanonical_sweep(self, **kwargs)
[docs] def get_energies(self): "Get energy bounds." return self._S_min, self._S_max
[docs] def get_allowed_energies(self): "Get allowed energy bounds." h = self._hist.a.copy() h += self._perm_hist Ss = self.get_range() Ss = Ss[h > 0] return Ss[0], Ss[-1]
[docs] def get_range(self): "Get energy range." return np.linspace(self._S_min, self._S_max, len(self._hist))
[docs] def get_density(self, B=None): """Get density of states, normalized so that total sum is :math:`B^N`, where :math:`B` is the number of groups, and :math:`N` is the number of nodes. If not supplied :math:`B=N` is assumed. """ r = np.array(self._density.a) r -= r.max() r -= np.log(np.exp(r).sum()) if B is None: B = self._g.num_vertices() r += self._g.num_vertices() * np.log(B) return r
[docs] def get_entropy(self, S, B=None): r = self.get_density(B) j = self.get_bin() return r[j]
[docs] def get_bin(self, S): return int(round((len(self._hist) - 1) * ((S - self._S_min) / (self._S_max - self._S_min))))
[docs] def get_hist(self): "Get energy histogram." return np.array(self._hist.a)
[docs] def get_perm_hist(self): "Get permanent energy histogram." return self._perm_hist
[docs] def get_flatness(self, h=None, allow_gaps=True): "Get energy histogram flatness." if h is None: h = self._hist.a if h.sum() == 0: return 0 if allow_gaps: idx = (h + self._perm_hist) > 0 else: Ss = self.get_range() S_min, S_max = self.get_allowed_energies() idx = np.logical_and(Ss >= S_min, Ss <= S_max) h = array(h[idx], dtype="float") if len(h) == 1: h = array([1e-6] + list(h)) h_mean = h.mean() return min((h.min() / h_mean, h_mean / h.max()))
[docs] def get_posterior(self, N=None): "Get posterior probability." r = self.get_density(N) Ss = np.linspace(self._S_min, self._S_max, len(r)) y = -Ss + r y_max = y.max() y -= y_max return y_max + np.log(np.exp(y).sum())
[docs] def reset_hist(self): "Reset energy histogram." self._perm_hist += self._hist.a self._hist.a = 0
[docs] def multicanonical_equilibrate(m_state, f_range=(1., 1e-6), r=2, flatness=.95, allow_gaps=True, callback=None, multicanonical_args={}, verbose=False): r"""Equilibrate a multicanonical Monte Carlo sampling using the Wang-Landau algorithm. Parameters ---------- m_state : :class:`~graph_tool.inference.MulticanonicalState` Initial multicanonical state, where the state density will be stored. f_range : ``tuple`` of two floats (optional, default: ``(1., 1e-6)``) Range of density updates. r : ``float`` (optional, default: ``2.``) Greediness of convergence. At each iteration, the density updates will be reduced by a factor ``r``. flatness : ``float`` (optional, default: ``.95``) Sufficient histogram flatness threshold used to continue the algorithm. allow_gaps : ``bool`` (optional, default: ``True``) If ``True``, gaps in the histogram (regions with zero count) will be ignored when computing the flatness. callback : ``function`` (optional, default: ``None``) If given, this function will be called after each iteration. The function must accept the current ``state`` and ``m_state`` as arguments. multicanonical_args : ``dict`` (optional, default: ``{}``) Arguments to be passed to ``state.multicanonical_sweep`` (e.g. :meth:`graph_tool.inference.BlockState.multicanonical_sweep`). verbose : ``bool`` or ``tuple`` (optional, default: ``False``) If ``True``, progress information will be shown. Optionally, this accepts arguments of the type ``tuple`` of the form ``(level, prefix)`` where ``level`` is a positive integer that specifies the level of detail, and ``prefix`` is a string that is prepended to the all output messages. Returns ------- niter : ``int`` Number of iterations required for convergence. References ---------- .. [wang-efficient-2001] Fugao Wang, D. P. Landau, "An efficient, multiple range random walk algorithm to calculate the density of states", Phys. Rev. Lett. 86, 2050 (2001), :doi:`10.1103/PhysRevLett.86.2050`, :arxiv:`cond-mat/0011174` .. [belardinelli-wang-2007] R. E. Belardinelli, V. D. Pereyra, "Wang-Landau algorithm: A theoretical analysis of the saturation of the error", J. Chem. Phys. 127, 184105 (2007), :doi:`10.1063/1.2803061`, :arxiv:`cond-mat/0702414` """ count = 0 if m_state._f is None: m_state._f = f_range[0] while m_state._f >= f_range[1]: m_state.sweep(**multicanonical_args) hf = m_state.get_flatness(allow_gaps=allow_gaps) if callback is not None: callback(m_state) if check_verbose(verbose): print(verbose_pad(verbose) + "count: %d f: %#8.8g flatness: %#8.8g nonempty bins: %d S: %#8.8g B: %d" % \ (count, m_state._f, hf, (m_state._hist.a > 0).sum(), m_state._state.entropy(**multicanonical_args.get("entropy_args", {})), m_state._state.get_nonempty_B())) if hf > flatness: m_state._f /= r if m_state._f >= f_range[1]: m_state.reset_hist() count += 1 return count
[docs] class TemperingState(object): """This class aggregates several state classes and corresponding inverse-temperature values to implement `parallel tempering MCMC <https://en.wikipedia.org/wiki/Parallel_tempering>`_. This is meant to be used with :func:`~graph_tool.inference.mcmc_equilibrate`. Parameters ---------- states : list of state objects (e.g. :class:`~graph_tool.inference.BlockState`) Initial parallel states. betas : list of floats Inverse temperature values. """ def __init__(self, states, betas, idx=None, beta_dl=False): if not (len(states) == len(betas)): raise ValueError("states and betas must be of the same size") self.states = states self.betas = betas if idx is None: self.idx = list(range(len(betas))) self.beta_dl = beta_dl self.swap_attempts = 0 self.swaps = np.zeros(len(states), dtype="int")
[docs] def entropy(self, **kwargs): """Returns the sum of the entropy of the parallel states. All keyword arguments are propagated to the individual states' `entropy()` method. """ if self.beta_dl: return sum(s.entropy(beta_dl=beta, **kwargs) for s, beta in zip(self.states, self.betas)) else: return sum(s.entropy(**kwargs) * beta for s, beta in zip(self.states, self.betas))
[docs] def entropies(self, **kwargs): """Returns the entropies of the parallel states. All keyword arguments are propagated to the individual states' `entropy()` method. """ if self.beta_dl: return [s.entropy(beta_dl=beta, **kwargs) for s, beta in zip(self.states, self.betas)] else: return [s.entropy(**kwargs) * beta for s, beta in zip(self.states, self.betas)]
[docs] def states_swap(self, **kwargs): """Perform a full sweep of the parallel states, where swaps are attempted. All relevant keyword arguments are propagated to the individual states' `entropy()` method.""" verbose = kwargs.get("verbose", False) eargs = kwargs.get("entropy_args", {}) self.swap_attempts += 1 idx = np.arange(len(self.states) - 1) np.random.shuffle(idx) nswaps = 0 dS = 0 for i in idx: j = i + 1 s1 = self.states[i] s2 = self.states[j] b1 = self.betas[i] b2 = self.betas[j] if self.beta_dl: P1_f = -s1.entropy(beta_dl=b2, **eargs) P2_f = -s2.entropy(beta_dl=b1, **eargs) P1_b = -s1.entropy(beta_dl=b1, **eargs) P2_b = -s2.entropy(beta_dl=b2, **eargs) else: S1 = s1.entropy(**eargs) S2 = s2.entropy(**eargs) P1_f = -S1 * b2 P2_f = -S2 * b1 P1_b = -S1 * b1 P2_b = -S2 * b2 ddS = -(P1_f + P2_f - P1_b - P2_b) if ddS < 0 or np.random.random() < exp(-ddS): self.states[j], self.states[i], self.idx[j], self.idx[i] = \ self.states[i], self.states[j], self.idx[i], self.idx[j] nswaps += 1 self.swaps[i] += 1 dS += ddS if check_verbose(verbose): print(verbose_pad(verbose) + u"swapped states: %d [β = %g] <-> %d [β = %g], a: %g" % \ (i, b1, j, b2, exp(-ddS))) return dS, nswaps
[docs] def states_move(self, sweep_algo, **kwargs): """Perform a full sweep of the parallel states, where state moves are attempted by calling `sweep_algo(state, beta=beta, **kwargs)`.""" algo_states = [] if isinstance(self.states[0], NestedBlockState): ls = list(kwargs.pop("ls", range(len(self.states[0].levels)))) if kwargs.pop("ls_shuffle", True): np.random.shuffle(ls) kwargs["ls"] = ls kwargs["ls_shuffle"] = False for state, beta in zip(self.states, self.betas): entropy_args = dict(kwargs.get("entropy_args", {})) if self.beta_dl: algo_state = sweep_algo[0](state, dispatch=False, **dict(kwargs, entropy_args=dict(entropy_args, beta_dl=beta))) else: algo_state = sweep_algo[0](state, dispatch=False, **dict(kwargs, beta=beta)) algo_states.append(algo_state) return sweep_algo[1](self.states, algo_states)
def _sweep(self, algo, r=0.1, adjacent=True, **kwargs): if np.random.random() < r: return self.states_swap(adjacent=adjacent, **kwargs) else: return self.states_move(algo, **dict(kwargs, verbose=False))
[docs] def mcmc_sweep(self, **kwargs): """Perform a full mcmc sweep of the parallel states, where swap or moves are chosen randomly. It accepts an keyword argument ``r`` (default: ``0.1``) specifying the relative probability with which state swaps are performed with respect to node moves. All remaining keyword arguments are propagated to the individual states' `mcmc_sweep()` method. """ algo = (lambda s, **kw: s.mcmc_sweep(**kw), lambda states, sweeps: type(self.states[0])._mcmc_sweep_parallel_dispatch(states, sweeps)) return self._sweep(algo, **kwargs)
[docs] def multiflip_mcmc_sweep(self, **kwargs): """Perform a full mcmc sweep of the parallel states, where swap or moves are chosen randomly. It accepts an keyword argument ``r`` (default: ``0.1``) specifying the relative probability with which state swaps are performed with respect to node moves. All remaining keyword arguments are propagated to the individual states' `mcmc_sweep()` method. """ algo = (lambda s, **kw: s.multiflip_mcmc_sweep(**kw), lambda states, sweeps: type(self.states[0])._multiflip_mcmc_sweep_parallel_dispatch(states, sweeps)) return self._sweep(algo, **kwargs)
[docs] def gibbs_sweep(self, **kwargs): """Perform a full Gibbs mcmc sweep of the parallel states, where swap or moves are chosen randomly. It accepts an keyword argument ``r`` (default: ``0.1``) specifying the relative probability with which state swaps are performed with respect to node moves. All remaining keyword arguments are propagated to the individual states' `gibbs_sweep()` method. """ algo = (lambda s, **kw: s.gibbs_sweep(**kw), lambda states, sweeps: type(self.states[0])._gibbs_sweep_parallel_dispatch(states, sweeps)) return self._sweep(algo, **kwargs)