# Copyright 2020 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#      https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.

from absl import app
import numpy as np
from pyiree.tf.support import tf_test_utils
from pyiree.tf.support import tf_utils
import tensorflow.compat.v2 as tf


class GatherModule(tf.Module):

  @tf.function(input_signature=[
      tf.TensorSpec([4, 8], tf.float32),
      tf.TensorSpec([], tf.int32)
  ])
  def gather_axis0_scalar(self, params, indices):
    return tf.gather(params, indices)

  @tf.function(input_signature=[
      tf.TensorSpec([4, 8], tf.float32),
      tf.TensorSpec([2], tf.int32)
  ])
  def gather_axis0_batch0(self, params, indices):
    return tf.gather(params, indices)

  @tf.function(input_signature=[
      tf.TensorSpec([4, 7, 8], tf.float32),
      tf.TensorSpec([2], tf.int32)
  ])
  def gather_axis1_batch0(self, params, indices):
    return tf.gather(params, indices, axis=1)

  @tf.function(input_signature=[
      tf.TensorSpec([4, 7, 8, 2], tf.float32),
      tf.TensorSpec([4, 1], tf.int32)
  ])
  def gather_axis2_batch1(self, params, indices):
    return tf.gather(params, indices, axis=2, batch_dims=1)


  @tf.function(input_signature=[
      tf.TensorSpec([4, 7, 8, 2], tf.float32),
      tf.TensorSpec([4, 1], tf.int32)
  ])
  def gather_axis1_batch1(self, params, indices):
    return tf.gather(params, indices, axis=1, batch_dims=1)


  @tf.function(input_signature=[
      tf.TensorSpec([2, 4], tf.int32),
      tf.TensorSpec([2, 4], tf.int32)
  ])
  def gather_axis2_batch2(self, params, indices):
    return tf.gather(params, indices, axis=1, batch_dims=1)


class GatherTest(tf_test_utils.TracedModuleTestCase):

  def __init__(self, methodName="runTest"):
    super(GatherTest, self).__init__(methodName)
    self._modules = tf_test_utils.compile_tf_module(GatherModule)

  def test_gather_axis0_scalar(self):

    def gather_axis0_scalar(module):
      indices = np.array(2, dtype=np.int32)
      params = tf_utils.ndarange([4, 8])
      module.gather_axis0_scalar(params, indices)

    self.compare_backends(gather_axis0_scalar, self._modules)

  def test_gather_axis0_batch0(self):

    def gather_axis0_batch0(module):
      indices = np.array([2, 3], dtype=np.int32)
      params = tf_utils.ndarange([4, 8])
      module.gather_axis0_batch0(params, indices)

    self.compare_backends(gather_axis0_batch0, self._modules)

  def test_gather_axis1_batch0(self):

    def gather_axis1_batch0(module):
      indices = np.array([2, 3], dtype=np.int32)
      params = tf_utils.ndarange([4, 7, 8])
      module.gather_axis1_batch0(params, indices)

    self.compare_backends(gather_axis1_batch0, self._modules)

  def test_gather_axis2_batch1(self):

    def gather_axis2_batch1(module):
      indices = np.array([[2], [3], [0], [1]], dtype=np.int32)
      params = tf_utils.ndarange([4, 7, 8, 2])
      module.gather_axis2_batch1(params, indices)

    self.compare_backends(gather_axis2_batch1, self._modules)

  def test_gather_axis1_batch1(self):

    def gather_axis1_batch1(module):
      indices = np.array([[2], [3], [0], [1]], dtype=np.int32)
      params = tf_utils.ndarange([4, 7, 8, 2])
      module.gather_axis1_batch1(params, indices)

    self.compare_backends(gather_axis1_batch1, self._modules)

  def test_gather_axis2_batch2(self):

    def gather_axis2_batch2(module):
      indices = np.array([[0, 1, 2, 3], [3, 2, 1, 0]], dtype=np.int32)
      values = np.array([[0, 1, 2, 3], [9, 8, 7, 0]], dtype=np.int32)
      module.gather_axis2_batch2(values, indices)

    self.compare_backends(gather_axis2_batch2, self._modules)



def main(argv):
  del argv  # Unused
  if hasattr(tf, 'enable_v2_behavior'):
    tf.enable_v2_behavior()
  tf.test.main()


if __name__ == '__main__':
  app.run(main)
