2015-04-09 15:42:02 +01:00
|
|
|
# Copyright (c) 2012-2015 The GPy authors (see AUTHORS.txt)
|
2013-12-04 11:32:12 +00:00
|
|
|
# Licensed under the BSD 3-clause license (see LICENSE.txt)
|
|
|
|
|
|
|
|
|
|
import numpy as np
|
2015-04-09 15:42:02 +01:00
|
|
|
from ..util.univariate_Gaussian import std_norm_cdf, std_norm_pdf
|
2013-12-04 11:32:12 +00:00
|
|
|
import scipy as sp
|
2015-04-10 14:58:02 +01:00
|
|
|
from ..util.misc import safe_exp, safe_square, safe_cube, safe_quad, safe_three_times
|
2014-03-14 12:42:36 +00:00
|
|
|
|
2013-12-04 11:32:12 +00:00
|
|
|
class GPTransformation(object):
|
|
|
|
|
"""
|
|
|
|
|
Link function class for doing non-Gaussian likelihoods approximation
|
|
|
|
|
|
|
|
|
|
:param Y: observed output (Nx1 numpy.darray)
|
|
|
|
|
|
|
|
|
|
.. note:: Y values allowed depend on the likelihood_function used
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
def __init__(self):
|
|
|
|
|
pass
|
|
|
|
|
|
|
|
|
|
def transf(self,f):
|
|
|
|
|
"""
|
|
|
|
|
Gaussian process tranformation function, latent space -> output space
|
|
|
|
|
"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
def dtransf_df(self,f):
|
|
|
|
|
"""
|
|
|
|
|
derivative of transf(f) w.r.t. f
|
|
|
|
|
"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
def d2transf_df2(self,f):
|
|
|
|
|
"""
|
|
|
|
|
second derivative of transf(f) w.r.t. f
|
|
|
|
|
"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
def d3transf_df3(self,f):
|
|
|
|
|
"""
|
|
|
|
|
third derivative of transf(f) w.r.t. f
|
|
|
|
|
"""
|
|
|
|
|
raise NotImplementedError
|
|
|
|
|
|
|
|
|
|
class Identity(GPTransformation):
|
|
|
|
|
"""
|
|
|
|
|
.. math::
|
|
|
|
|
|
|
|
|
|
g(f) = f
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
def transf(self,f):
|
|
|
|
|
return f
|
|
|
|
|
|
|
|
|
|
def dtransf_df(self,f):
|
|
|
|
|
return np.ones_like(f)
|
|
|
|
|
|
|
|
|
|
def d2transf_df2(self,f):
|
|
|
|
|
return np.zeros_like(f)
|
|
|
|
|
|
|
|
|
|
def d3transf_df3(self,f):
|
|
|
|
|
return np.zeros_like(f)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
class Probit(GPTransformation):
|
|
|
|
|
"""
|
|
|
|
|
.. math::
|
|
|
|
|
|
|
|
|
|
g(f) = \\Phi^{-1} (mu)
|
2015-04-16 12:45:04 +01:00
|
|
|
|
2013-12-04 11:32:12 +00:00
|
|
|
"""
|
|
|
|
|
def transf(self,f):
|
|
|
|
|
return std_norm_cdf(f)
|
|
|
|
|
|
|
|
|
|
def dtransf_df(self,f):
|
|
|
|
|
return std_norm_pdf(f)
|
|
|
|
|
|
|
|
|
|
def d2transf_df2(self,f):
|
|
|
|
|
return -f * std_norm_pdf(f)
|
|
|
|
|
|
|
|
|
|
def d3transf_df3(self,f):
|
2015-04-16 12:45:04 +01:00
|
|
|
return (safe_square(f)-1.)*std_norm_pdf(f)
|
2013-12-04 11:32:12 +00:00
|
|
|
|
2014-10-16 16:38:32 +01:00
|
|
|
|
|
|
|
|
class Cloglog(GPTransformation):
|
|
|
|
|
"""
|
|
|
|
|
Complementary log-log link
|
|
|
|
|
.. math::
|
|
|
|
|
|
|
|
|
|
p(f) = 1 - e^{-e^f}
|
|
|
|
|
|
|
|
|
|
or
|
|
|
|
|
|
|
|
|
|
f = \log (-\log(1-p))
|
2015-04-09 15:42:02 +01:00
|
|
|
|
2014-10-16 16:38:32 +01:00
|
|
|
"""
|
|
|
|
|
def transf(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
ef = safe_exp(f)
|
|
|
|
|
return 1-np.exp(-ef)
|
2014-10-16 16:38:32 +01:00
|
|
|
|
|
|
|
|
def dtransf_df(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
ef = safe_exp(f)
|
|
|
|
|
return np.exp(f-ef)
|
2014-10-16 16:38:32 +01:00
|
|
|
|
|
|
|
|
def d2transf_df2(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
ef = safe_exp(f)
|
2014-10-16 16:38:32 +01:00
|
|
|
return -np.exp(f-ef)*(ef-1.)
|
|
|
|
|
|
|
|
|
|
def d3transf_df3(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
ef = safe_exp(f)
|
|
|
|
|
ef2 = safe_square(ef)
|
|
|
|
|
three_times_ef = safe_three_times(ef)
|
|
|
|
|
r_val = np.exp(f-ef)*(1.-three_times_ef + ef2)
|
|
|
|
|
return r_val
|
2014-10-16 16:38:32 +01:00
|
|
|
|
2013-12-04 11:32:12 +00:00
|
|
|
class Log(GPTransformation):
|
|
|
|
|
"""
|
|
|
|
|
.. math::
|
|
|
|
|
|
|
|
|
|
g(f) = \\log(\\mu)
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
def transf(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
return safe_exp(f)
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
def dtransf_df(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
return safe_exp(f)
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
def d2transf_df2(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
return safe_exp(f)
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
def d3transf_df3(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
return safe_exp(f)
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
class Log_ex_1(GPTransformation):
|
|
|
|
|
"""
|
|
|
|
|
.. math::
|
|
|
|
|
|
|
|
|
|
g(f) = \\log(\\exp(\\mu) - 1)
|
|
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
def transf(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
return np.log1p(safe_exp(f))
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
def dtransf_df(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
ef = safe_exp(f)
|
|
|
|
|
return ef/(1.+ef)
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
def d2transf_df2(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
ef = safe_exp(f)
|
|
|
|
|
aux = ef/(1.+ef)
|
2013-12-04 11:32:12 +00:00
|
|
|
return aux*(1.-aux)
|
|
|
|
|
|
|
|
|
|
def d3transf_df3(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
ef = safe_exp(f)
|
|
|
|
|
aux = ef/(1.+ef)
|
2013-12-04 11:32:12 +00:00
|
|
|
daux_df = aux*(1.-aux)
|
|
|
|
|
return daux_df - (2.*aux*daux_df)
|
|
|
|
|
|
|
|
|
|
class Reciprocal(GPTransformation):
|
|
|
|
|
def transf(self,f):
|
|
|
|
|
return 1./f
|
|
|
|
|
|
2015-04-10 14:58:02 +01:00
|
|
|
def dtransf_df(self, f):
|
|
|
|
|
f2 = safe_square(f)
|
|
|
|
|
return -1./f2
|
2013-12-04 11:32:12 +00:00
|
|
|
|
2015-04-10 14:58:02 +01:00
|
|
|
def d2transf_df2(self, f):
|
|
|
|
|
f3 = safe_cube(f)
|
|
|
|
|
return 2./f3
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
def d3transf_df3(self,f):
|
2015-04-10 14:58:02 +01:00
|
|
|
f4 = safe_quad(f)
|
|
|
|
|
return -6./f4
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
class Heaviside(GPTransformation):
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
.. math::
|
|
|
|
|
|
2015-04-09 15:42:02 +01:00
|
|
|
g(f) = I_{x \\geq 0}
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
def transf(self,f):
|
|
|
|
|
#transformation goes here
|
|
|
|
|
return np.where(f>0, 1, 0)
|
|
|
|
|
|
|
|
|
|
def dtransf_df(self,f):
|
2015-02-26 13:43:55 +00:00
|
|
|
raise NotImplementedError("This function is not differentiable!")
|
2013-12-04 11:32:12 +00:00
|
|
|
|
|
|
|
|
def d2transf_df2(self,f):
|
2015-02-26 13:43:55 +00:00
|
|
|
raise NotImplementedError("This function is not differentiable!")
|