#!/usr/bin/env python3

import unittest

import torch

from gpytorch.likelihoods import _OneDimensionalLikelihood, LaplaceLikelihood
from gpytorch.test.base_likelihood_test_case import BaseLikelihoodTestCase


class TestLaplaceLikelihood(BaseLikelihoodTestCase, unittest.TestCase):
    seed = 1

    def _create_targets(self, batch_shape=torch.Size([])):
        res = torch.sigmoid(torch.randn(*batch_shape, 5)).float()
        return res

    def create_likelihood(self, **kwargs):
        return LaplaceLikelihood(**kwargs)

    def _test_log_marginal(self, batch_shape):
        likelihood = self.create_likelihood()
        input = self._create_marginal_input(batch_shape)
        target = self._create_targets(batch_shape)
        output = likelihood.log_marginal(target, input)

        self.assertTrue(torch.is_tensor(output))
        self.assertEqual(output.shape, batch_shape + torch.Size([5]))
        default_log_prob = _OneDimensionalLikelihood.log_marginal(likelihood, target, input)
        self.assertAllClose(output.sum(-1), default_log_prob.sum(-1), rtol=0.25, atol=0.1)

    def _test_log_prob(self, batch_shape):
        likelihood = self.create_likelihood()
        input = self._create_marginal_input(batch_shape)
        target = self._create_targets(batch_shape)
        output = likelihood.expected_log_prob(target, input)

        self.assertTrue(torch.is_tensor(output))
        self.assertEqual(output.shape, batch_shape + torch.Size([5]))
        default_log_prob = _OneDimensionalLikelihood.expected_log_prob(likelihood, target, input)
        self.assertAllClose(output.sum(-1), default_log_prob.sum(-1), rtol=0.25, atol=0.1)

    def _test_marginal(self, batch_shape):
        # Likelihood uses the default marginal behavior, no point testing a set of samples against a set of samples.
        pass
