// file      : libbuild2/utility.ixx -*- C++ -*-
// license   : MIT; see accompanying LICENSE file

#include <cstring> // strlen() strchr()

namespace build2
{
  inline bool
  run_wait (const cstrings& args, process& pr, const location& loc)
  {
    return run_wait (args.data (), pr, loc);
  }

  // Note: these functions are also used in the run() implementations.
  //
  LIBBUILD2_SYMEXPORT bool
  run_finish_impl (const char* const*,
                   process&,
                   bool fail,
                   const string&,
                   uint16_t,
                   bool = false,
                   const location& = {});

  LIBBUILD2_SYMEXPORT bool
  run_finish_impl (diag_buffer&,
                   const char* const*,
                   process&,
                   bool fail,
                   uint16_t,
                   bool = false,
                   const location& = {});

  inline void
  run_finish (const char* const* args,
              process& pr,
              uint16_t v,
              bool on,
              const location& loc)
  {
    run_finish_impl (args, pr, true /* fail */, string (), v, on, loc);
  }

  inline void
  run_finish (const cstrings& args,
              process& pr,
              uint16_t v,
              bool on,
              const location& loc)
  {
    run_finish (args.data (), pr, v, on, loc);
  }

  inline void
  run_finish (const char* const* args,
              process& pr,
              const string& l,
              uint16_t v,
              bool on,
              const location& loc)
  {
    run_finish_impl (args, pr, true, l, v, on, loc);
  }

  inline bool
  run_finish_code (const char* const* args,
                   process& pr,
                   uint16_t v,
                   bool on,
                   const location& loc)
  {
    return run_finish_impl (args, pr, false, string (), v, on, loc);
  }

  inline bool
  run_finish_code (const cstrings& args,
                   process& pr,
                   uint16_t v,
                   bool on,
                   const location& loc)
  {
    return run_finish_code (args.data (), pr, v, on, loc);
  }

  inline bool
  run_finish_code (const char* const* args,
                   process& pr,
                   const string& l,
                   uint16_t v,
                   bool on,
                   const location& loc)
  {
    return run_finish_impl (args, pr, false, l, v, on, loc);
  }

  inline void
  run_finish (diag_buffer& dbuf,
              const char* const* args,
              process& pr,
              uint16_t v,
              bool on,
              const location& loc)
  {
    run_finish_impl (dbuf, args, pr, true /* fail */, v, on, loc);
  }

  inline void
  run_finish (diag_buffer& dbuf,
              const cstrings& args,
              process& pr,
              uint16_t v,
              bool on,
              const location& loc)
  {
    run_finish_impl (dbuf, args.data (), pr, true, v, on, loc);
  }

  inline bool
  run_finish_code (diag_buffer& dbuf,
                   const char* const* args,
                   process& pr,
                   uint16_t v,
                   bool on,
                   const location& loc)
  {
    return run_finish_impl (dbuf, args, pr, false, v, on, loc);
  }

  inline bool
  run_finish_code (diag_buffer& dbuf,
                   const cstrings& args,
                   process& pr,
                   uint16_t v,
                   bool on,
                   const location& loc)
  {
    return run_finish_impl (dbuf, args.data (), pr, false, v, on, loc);
  }

  template <typename T, typename F>
  inline T
  run (context& ctx,
       uint16_t verbosity,
       const process_env& pe,
       const char* const* args,
       F&& f,
       bool err,
       bool ignore_exit,
       sha256* checksum)
  {
    T r;
    if (!run (ctx,
              verbosity,
              pe, args,
              verbosity - 1,
              [&r, &f] (string& l, bool last) // Small function optimmization.
              {
                r = f (l, last);
                return r.empty ();
              },
              true /* trim */,
              err,
              ignore_exit,
              checksum))
      r = T ();

    return r;
  }

  template <typename T, typename F>
  inline T
  run (diag_buffer& dbuf,
       uint16_t verbosity,
       const process_env& pe,
       const char* const* args,
       F&& f,
       sha256* checksum)
  {
    T r;
    run (dbuf,
         verbosity,
         pe, args,
         verbosity - 1,
         [&r, &f] (string& l, bool last) // Small function optimmization.
         {
           r = f (l, last);
           return r.empty ();
         },
         true /* trim */,
         checksum);
    return r;
  }

  template <typename T, typename F>
  inline T
  run (context& ctx,
       const process_env& pe,
       const char* const* args,
       uint16_t finish_verbosity,
       F&& f,
       bool err,
       bool ignore_exit,
       sha256* checksum)
  {
    T r;
    if (!run (ctx,
              verb_never,
              pe, args,
              finish_verbosity,
              [&r, &f] (string& l, bool last)
              {
                r = f (l, last);
                return r.empty ();
              },
              true /* trim */,
              err,
              ignore_exit,
              checksum))
      r = T ();

    return r;
  }

  template <typename T, typename F>
  inline T
  run (diag_buffer& dbuf,
       const process_env& pe,
       const char* const* args,
       uint16_t finish_verbosity,
       F&& f,
       sha256* checksum)
  {
    T r;
    run (dbuf,
         verb_never,
         pe, args,
         finish_verbosity,
         [&r, &f] (string& l, bool last)
         {
           r = f (l, last);
           return r.empty ();
         },
         true /* trim */,
         checksum);
    return r;
  }

  inline void
  hash_path (sha256& cs, const path& p, const dir_path& prefix)
  {
    // Note: for efficiency we don't use path::leaf() and "skip" the prefix
    // without copying.
    //
    const char* s (p.string ().c_str ());

    if (!prefix.empty () && p.sub (prefix))
    {
      s += prefix.size (); // Does not include trailing slash except for root.
      if (path::traits_type::is_separator (*s))
        ++s;
    }

    cs.append (s);
  }

  template <typename T>
  inline void
  append_options (cstrings& args, T& s, const variable& var, const char* e)
  {
    append_options (args, s[var], e);
  }

  template <typename T>
  inline void
  append_options (strings& args, T& s, const variable& var, const char* e)
  {
    append_options (args, s[var], e);
  }

  template <typename T>
  inline void
  append_options (sha256& csum, T& s, const variable& var)
  {
    append_options (csum, s[var]);
  }

  template <typename T>
  inline void
  append_options (cstrings& args, T& s, const char* var, const char* e)
  {
    append_options (args, s[var], e);
  }

  template <typename T>
  inline void
  append_options (strings& args, T& s, const char* var, const char* e)
  {
    append_options (args, s[var], e);
  }

  template <typename T>
  inline void
  append_options (sha256& csum, T& s, const char* var)
  {
    append_options (csum, s[var]);
  }

  inline void
  append_options (cstrings& args, const strings& sv, const char* e)
  {
    if (size_t n = sv.size ())
      append_options (args, sv, n, e);
  }

  inline void
  append_options (strings& args, const strings& sv, const char* e)
  {
    if (size_t n = sv.size ())
      append_options (args, sv, n, e);
  }

  inline void
  append_options (sha256& csum, const strings& sv)
  {
    if (size_t n = sv.size ())
      append_options (csum, sv, n);
  }

  template <typename T>
  inline bool
  find_option (const char* o, T& s, const variable& var, bool ic)
  {
    return find_option (o, s[var], ic);
  }

  template <typename T>
  inline bool
  find_option (const char* o, T& s, const char* var, bool ic)
  {
    return find_option (o, s[var], ic);
  }

  inline bool
  compare_option (const char* o, const char* s, bool ic)
  {
    return s != nullptr && (ic ? icasecmp (s, o) : strcmp (s, o)) == 0;
  }

  inline bool
  compare_option (const char* o, const string& s, bool ic)
  {
    return ic ? icasecmp (s, o) == 0 : s == o;
  }

  template <typename I>
  inline I
  find_option (const char* o, I b, I e, bool ic)
  {
    for (; b != e; ++b)
      if (compare_option (o, *b, ic))
        return b;

    return e;
  }

  template <typename T>
  inline bool
  find_options (const initializer_list<const char*>& os,
                T& s,
                const variable& var,
                bool ic)
  {
    return find_options (os, s[var], ic);
  }

  template <typename T>
  inline bool
  find_options (const initializer_list<const char*>& os,
                T& s,
                const char* var,
                bool ic)
  {
    return find_options (os, s[var], ic);
  }

  template <typename T>
  inline const string*
  find_option_prefix (const char* p, T& s, const variable& var, bool ic)
  {
    return find_option_prefix (p, s[var], ic);
  }

  template <typename T>
  inline const string*
  find_option_prefix (const char* p, T& s, const char* var, bool ic)
  {
    return find_option_prefix (p, s[var], ic);
  }

  inline bool
  compare_option_prefix (const char* p, size_t n, const char* s, bool ic)
  {
    return s != nullptr && (ic ? icasecmp (s, p, n) : strncmp (s, p, n)) == 0;
  }

  inline bool
  compare_option_prefix (const char* p, size_t n, const string& s, bool ic)
  {
    return (ic ? icasecmp (s, p, n) : s.compare (0, n, p)) == 0;
  }

  template <typename I>
  inline I
  find_option_prefix (const char* p, I b, I e, bool ic)
  {
    size_t n (strlen (p));

    for (; b != e; ++b)
      if (compare_option_prefix (p, n, *b, ic))
        return b;

    return e;
  }

  template <typename T>
  inline const string*
  find_option_prefixes (const initializer_list<const char*>& ps,
                        T& s,
                        const variable& var,
                        bool ic)
  {
    return find_option_prefixes (ps, s[var], ic);
  }

  template <typename T>
  inline const string*
  find_option_prefixes (const initializer_list<const char*>& ps,
                        T& s,
                        const char* var,
                        bool ic)
  {
    return find_option_prefixes (ps, s[var], ic);
  }

  // hash_environment()
  //
  inline void
  hash_environment (sha256& cs, const char* n)
  {
    cs.append (n);

    if (optional<string> v = getenv (n))
      cs.append (*v);
  }

  inline void
  hash_environment (sha256& cs, const string& n)
  {
    hash_environment (cs, n.c_str ());
  }

  inline void
  hash_environment (sha256& cs, initializer_list<const char*> ns)
  {
    for (const char* n: ns)
      hash_environment (cs, n);
  }

  inline string
  hash_environment (initializer_list<const char*> ns)
  {
    sha256 cs;
    hash_environment (cs, ns);
    return cs.string ();
  }

  inline void
  hash_environment (sha256& cs, const cstrings& ns)
  {
    for (const char* n: ns)
      hash_environment (cs, n);
  }

  inline string
  hash_environment (const cstrings& ns)
  {
    sha256 cs;
    hash_environment (cs, ns);
    return cs.string ();
  }

  inline void
  hash_environment (sha256& cs, const strings& ns)
  {
    for (const string& n: ns)
      hash_environment (cs, n);
  }

  inline string
  hash_environment (const strings& ns)
  {
    sha256 cs;
    hash_environment (cs, ns);
    return cs.string ();
  }

  inline void
  hash_environment (sha256& cs, const char* const* ns)
  {
    if (ns != nullptr)
    {
      for (; *ns != nullptr; ++ns)
        hash_environment (cs, *ns);
    }
  }

  inline string
  hash_environment (const char* const* ns)
  {
    sha256 cs;
    hash_environment (cs, ns);
    return cs.string ();
  }

  // find_stem()
  //
  inline size_t
  find_stem (const string& s, size_t s_p, size_t s_n,
             const char* stem, const char* seps)
  {
    auto sep = [seps] (char c) -> bool
    {
      return strchr (seps, c) != nullptr;
    };

    size_t m (strlen (stem));
    size_t p (s.find (stem, s_p, m));

    return (p != string::npos &&
            (      p == s_p || sep (s[p - 1])) && // Separated beginning.
            ((p + m) == s_n || sep (s[p + m])))   // Separated end.
      ? p
      : string::npos;
  }
}