// file      : libbuild2/target-key.hxx -*- C++ -*-
// license   : MIT; see accompanying LICENSE file

#ifndef LIBBUILD2_TARGET_KEY_HXX
#define LIBBUILD2_TARGET_KEY_HXX

#include <cstring> // strcmp()

#include <libbuild2/types.hxx>
#include <libbuild2/forward.hxx>
#include <libbuild2/utility.hxx>

#include <libbuild2/target-type.hxx>

#include <libbuild2/export.hxx>

namespace build2
{
  // Light-weight (by being shallow-pointing) target key.
  //
  class LIBBUILD2_SYMEXPORT target_key
  {
  public:
    const target_type*       type;
    const dir_path*          dir; // Can be relative if part of prerequisite_key.
    const dir_path*          out; // Can be relative if part of prerequisite_key.
    const string*            name;
    mutable optional<string> ext; // Absent - unspecified, empty - none.

    template <typename T>
    bool is_a () const {return type->is_a<T> ();}
    bool is_a (const target_type& tt) const {return type->is_a (tt);}

    // Return an "effective" name, for example, for pattern matching, that
    // includes the extension where appropriate.
    //
    const string&
    effective_name (string& storage, bool force_ext = false) const;

    // Append/return the target name or a pair of names if out-qualified.
    //
    // See also target::as_name() for the returned name stability guarantees.
    //
    void
    as_name (names&) const;

    names
    as_name () const
    {
      names r;
      as_name (r);
      return r;
    }
  };

  inline bool
  operator== (const target_key& x, const target_key& y)
  {
    if (x.type  != y.type ||
        *x.dir  != *y.dir ||
        *x.out  != *y.out ||
        *x.name != *y.name)
      return false;

    // Unless fixed, unspecified and specified extensions are assumed equal.
    //
    const target_type& tt (*x.type);

    if (tt.fixed_extension == nullptr)
      return !x.ext || !y.ext || *x.ext == *y.ext;
    else
    {
      // Note that for performance reasons here we use the specified extension
      // without calling fixed_extension() to verify it matches.
      //
      const char* xe (x.ext
                      ? x.ext->c_str ()
                      : tt.fixed_extension (x, nullptr /* root scope */));

      const char* ye (y.ext
                      ? y.ext->c_str ()
                      : tt.fixed_extension (y, nullptr /* root scope */));

      return strcmp (xe, ye) == 0;
    }
  }

  inline bool
  operator!= (const target_key& x, const target_key& y) {return !(x == y);}

  // If the target type has a custom print function, call that. Otherwise,
  // call to_stream(). Both are defined in target.cxx.
  //
  LIBBUILD2_SYMEXPORT ostream&
  operator<< (ostream&, const target_key&);

  // If name_only is true, then only print the target name (and extension, if
  // necessary), without the directory or type.
  //
  // Return true if the result is regular, that is, in the
  // <dir>/<type>{<name>}@<out>/ form with the individual components
  // corresponding directly to the target_key members (that is, without moving
  // parts around as would be the case for directories). This information is
  // used when trying to print several targets in a combined form (for
  // example, {hxx cxx}{foo}) in print_diag().
  //
  LIBBUILD2_SYMEXPORT bool
  to_stream (ostream&,
             const target_key&,
             optional<stream_verbosity> = nullopt,
             bool name_only = false);
}

namespace std
{
  // Note that we ignore the extension when calculating the hash because of
  // its special "unspecified" logic (see operator== above).
  //
  template <>
  struct hash<build2::target_key>
  {
    using argument_type = build2::target_key;
    using result_type = size_t;

    size_t
    operator() (const build2::target_key& k) const noexcept
    {
      return build2::combine_hash (
        hash<const build2::target_type*> () (k.type),
        hash<build2::dir_path> () (*k.dir),
        hash<build2::dir_path> () (*k.out),
        hash<string> () (*k.name));
    }
  };
}

#endif // LIBBUILD2_TARGET_KEY_HXX