// Copyright (c) by respective owners including Yahoo!, Microsoft, and
// individual contributors. All rights reserved. Released under a BSD (revised)
// license as described in the file LICENSE.
#include "vw/core/unique_sort.h"

#include "vw/core/example.h"
#include "vw/core/feature_group.h"

#include <algorithm>

/**
 * \brief Remove all non unique features from a feature group.
 * - Uniqueness is determined by feature index.
 * - This function requires the feature group to be sorted. For sorting see features::sort.
 *
 * \param fs Feature group to remove non-unique features from
 * \param max The maximum number of unique features to keep. -1 to keep all unique features.
 */
void unique_features(features& fs, int max)
{
  if (fs.indices.empty()) { return; }
  if (max == 0)
  {
    fs.clear();
    return;
  }
  if (max == 1)
  {
    fs.truncate_to(1);
    return;
  }

  auto flat_extents = VW::details::flatten_namespace_extents(fs.namespace_extents, fs.indices.size());

  auto last_index = std::size_t{0};
  for (auto i = std::size_t{1}; i != fs.size(); ++i)
  {
    if (fs.indices[i] != fs.indices[last_index])
    {
      if (i != ++last_index)
      {
        fs.values[last_index] = fs.values[i];
        fs.indices[last_index] = fs.indices[i];
        flat_extents[last_index] = flat_extents[i];
        if (!fs.space_names.empty()) { fs.space_names[last_index] = std::move(fs.space_names[i]); }
      }

      const auto unique_items_found = last_index + 1;
      // Rely on a negative integer to wrap around and be larger if a user passed in a negative value.
      if (unique_items_found >= static_cast<size_t>(max)) { break; }
    }
  }
  fs.namespace_extents = VW::details::unflatten_namespace_extents(flat_extents);
  ++last_index;
  fs.truncate_to(last_index);
}

void unique_sort_features(uint64_t parse_mask, VW::example* ae)
{
  for (features& fs : *ae)
  {
    if (fs.sort(parse_mask)) { unique_features(fs); }
  }

  ae->sorted = true;
}
