/**
 * (C) Copyright IBM Corp. 2015, 2016
 *
 * 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
 *
 *     http://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.
 *
 */

package com.ibm.stocator.fs.common;

import java.io.Closeable;
import java.io.IOException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URLDecoder;
import java.nio.charset.StandardCharsets;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import java.util.Locale;
import java.util.Properties;

import org.apache.hadoop.conf.Configuration;
import org.apache.hadoop.mapreduce.TaskAttemptID;
import org.apache.http.conn.util.InetAddressUtils;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import com.ibm.stocator.fs.common.exception.ConfigurationParseException;
import com.ibm.stocator.fs.common.exception.InvalidContainerNameException;
import static com.ibm.stocator.fs.common.Constants.HADOOP_ATTEMPT;

public class Utils {

  public static final String BAD_HOST = " hostname '%s' must be in the form container.service";

  /*
   * Logger
   */
  private static final Logger LOG = LoggerFactory.getLogger(Utils.class);

  /*
  * Time pattern
  */
  private static final String TIME_PATTERN = "EEE, d MMM yyyy hh:mm:ss zzz";

  /**
   * IOException if the host name is not comply with container.service
   *
   * @param hostname hostname
   * @return IOException
   */
  private static IOException badHostName(String hostname) {
    return new IOException(String.format(BAD_HOST, hostname));
  }

  /**
   * Extracts container name from the container.service
   *
   * @param hostname hostname to split
   * @return the container
   * @throws IOException if hostname is invalid
   */
  public static String getContainerName(String hostname) throws IOException {
    return getContainerName(hostname, true);
  }

  /**
   * Extracts container name from the container.service or container
   *
   * @param hostname hostname to split
   * @param serviceRequired flag if service name required as part of hostname
   * @return the container
   * @throws IOException if hostname is invalid
   */
  public static String getContainerName(String hostname,
      boolean serviceRequired) throws IOException {
    int i = hostname.lastIndexOf(".");
    if (i <= 0) {
      if (serviceRequired) {
        throw badHostName(hostname);
      }
      return hostname;
    }
    return hostname.substring(0, i);
  }

  /**
   * Extracts service name from the container.service
   *
   * @param hostname hostname
   * @return the separated out service name
   * @throws IOException if the hostname was invalid
   */
  public static String getServiceName(String hostname) throws IOException {
    int i = hostname.lastIndexOf(".");
    if (i <= 0) {
      throw badHostName(hostname);
    }
    String service = hostname.substring(i + 1);
    if (service.isEmpty() || service.contains(".")) {
      throw badHostName(hostname);
    }
    return service;
  }

  /**
   * Extracts service name from the container.service
   *
   * @param hostname hostname
   * @param defaultService default value
   * @return the separated out service name
   * @throws IOException if the hostname was invalid
   */
  public static String getServiceName(String hostname, String defaultService) throws IOException {
    int i = hostname.lastIndexOf(".");
    if (i <= 0) {
      throw badHostName(hostname);
    }
    String service = hostname.substring(i + 1);
    if (service.isEmpty() || service.contains(".")) {
      throw badHostName(hostname);
    }
    return service;
  }

  /**
   * Test if hostName of the form container.service
   *
   * @param uri schema URI
   * @return true if hostName of the form container.service
   * @throws IOException if error
   */
  public static boolean validSchema(URI uri) throws IOException {
    LOG.trace("Checking schema {}", uri.toString());
    String hostName = Utils.getHost(uri);
    LOG.trace("Got hostname as {}", hostName);
    int i = hostName.lastIndexOf(".");
    if (i < 0) {
      return false;
    }
    String service = hostName.substring(i + 1);
    LOG.trace("Got service as {}", service);
    if (service.isEmpty() || service.contains(".")) {
      return false;
    }
    return true;
  }

  public static boolean validSchema(String path) throws IOException {
    try {
      return validSchema(new URI(path));
    } catch (URISyntaxException e) {
      throw new IOException(e.getMessage());
    }
  }

  /**
   * Extract host name from the URI
   *
   * @param uri object store uri
   * @return host name
   * @throws IOException if error
   */
  public static String getHost(URI uri) throws IOException {
    String host = uri.getHost();
    if (host != null) {
      return host;
    }
    host = uri.toString();
    int sInd = host.indexOf("//") + 2;
    host = host.substring(sInd);
    int eInd = host.indexOf("/");
    if (eInd != -1) {
      host = host.substring(0,eInd);
    }
    host = URLDecoder.decode(host, StandardCharsets.UTF_8.toString());
    return host;
  }

  /**
   * Get a mandatory configuration option
   *
   * @param props property set
   * @param key key
   * @return value of the configuration
   * @throws IOException if there was no match for the key
   */
  public static String getOption(Properties props, String key) throws IOException {
    String val = props.getProperty(key);
    if (val == null) {
      throw new IOException("Undefined property: " + key);
    }
    return val;
  }

  /**
   * Read key from core-site.xml and parse it to connector configuration
   *
   * @param conf source configuration
   * @param prefix configuration key prefix
   * @param altPrefix alternative prefix list. The last on the list wins
   * @param key key in the configuration file
   * @param props destination property set
   * @param propsKey key in the property set
   * @param required if the key is mandatory
   * @throws ConfigurationParseException if there was no match for the key
   */

  public static void updateProperty(Configuration conf, String prefix, String[] altPrefix,
      String key, Properties props, String propsKey,
      boolean required) throws ConfigurationParseException {
    String val = conf.get(prefix + key);
    String altKey = prefix + key;
    if (val == null) {
      // try alternative key
      for (String alternativePrefix : altPrefix) {
        altKey = alternativePrefix + key;
        val = conf.get(altKey);
      }
    }
    if (required && val == null) {
      throw new ConfigurationParseException("Missing mandatory configuration: " + key);
    }
    if (val != null) {
      LOG.trace("Found alternative key {} value {}", altKey, val);
      props.setProperty(propsKey, val.trim());
    }
  }

  public static int getInt(Configuration conf, String prefix, String[] altPrefix, String key,
      int defValue) {
    int result = -1;
    int notExistsValue = -2;
    result = conf.getInt(prefix + key, notExistsValue);
    if (result == notExistsValue) {
      result = -1;
      for (String alternativePrefix : altPrefix) {
        result = conf.getInt(alternativePrefix + key, defValue);
        if (result != defValue) {
          break;
        }
      }
    }
    if (result == -1) {
      result = defValue;
    }
    return result;
  }

  public static String getTrimmed(Configuration conf, String prefix, String[] altPrefix,
      String key, String defValue) {
    String result = null;
    String notExistsValue = "-2";
    result = conf.getTrimmed(prefix + key, notExistsValue);
    if (result.equals(notExistsValue)) {
      result = null;
      for (String alternativePrefix : altPrefix) {
        result = conf.getTrimmed(alternativePrefix + key, defValue);
        if (!result.equals(defValue)) {
          break;
        }
      }
    }
    if (result == null) {
      result = defValue;
    }
    return result;
  }

  public static String getTrimmed(Configuration conf, String prefix, String[] altPrefix,
      String key) {
    String result = null;
    String notExistsValue = "-2";
    result = conf.getTrimmed(prefix + key, notExistsValue);
    if (result.equals(notExistsValue)) {
      result = null;
      for (String alternativePrefix : altPrefix) {
        result = conf.getTrimmed(alternativePrefix + key, notExistsValue);
        if (!result.equals(notExistsValue)) {
          break;
        }
      }
    }
    if (!result.equals(notExistsValue)) {
      return result;
    }
    return null;
  }

  public static String getConfigKey(Configuration conf, String prefix, String[] altPrefix,
      String key) {
    String result = null;
    String notExistsValue = "-2";
    String keyName = prefix + key;
    result = conf.getTrimmed(keyName, notExistsValue);
    if (result.equals(notExistsValue)) {
      // try alternative keys, based on priority
      result = null;
      keyName = null;
      for (String alternativePrefix : altPrefix) {
        keyName = alternativePrefix + key;
        result = conf.getTrimmed(keyName, notExistsValue);
        if (!result.equals(notExistsValue)) {
          break;
        }
      }
    }
    if (!result.equals(notExistsValue)) {
      return keyName;
    }
    return null;
  }

  public static long getLong(Configuration conf, String prefix, String[] altPrefix,
      String key, long defValue) {
    long result = -1;
    long notExistsValue = -2;
    result = conf.getLong(prefix + key, notExistsValue);
    if (result == notExistsValue) {
      result = -1;
      for (String alternativePrefix : altPrefix) {
        result = stringToLong(conf.get(alternativePrefix + key), defValue);
        if (result != defValue) {
          break;
        }
      }
    }
    if (result == -1) {
      result = defValue;
    }
    return result;
  }

  public static boolean getBoolean(Configuration conf, String prefix, String[] altPrefix,
      String key, boolean defValue) {
    boolean found = false;
    boolean result = false;
    String notExistsValue = "-2";
    String res = conf.getTrimmed(prefix + key, notExistsValue);
    if (!res.equals(notExistsValue)) {
      found = true;
      result = conf.getBoolean(prefix + key, defValue);
    } else {
      for (String alternativePrefix : altPrefix) {
        found = true;
        result = conf.getBoolean(alternativePrefix + key, defValue);
      }
    }
    if (!found) {
      result = defValue;
    }
    return result;
  }

  /**
   * Extract Hadoop Task ID from path
   * @param path path to extract attempt id
   * @param identifier identifier to extract id
   * @return task id
   */
  public static String extractTaskID(String path, String identifier) {
    LOG.debug("extract task id for {}", path);
    if (path.contains(HADOOP_ATTEMPT)) {
      String prf = path.substring(path.indexOf(HADOOP_ATTEMPT));
      if (prf.contains("/")) {
        return TaskAttemptID.forName(prf.substring(0, prf.indexOf("/"))).toString();
      }
      return TaskAttemptID.forName(prf).toString();
    } else if (identifier != null && path.contains(identifier)) {
      int ind = path.indexOf(identifier);
      String prf = path.substring(ind + identifier.length());
      int boundary = prf.length();
      if (prf.indexOf("/") > 0) {
        boundary = prf.indexOf("/");
      }
      String taskID =  prf.substring(0, boundary);
      LOG.debug("extracted task id {} for {}", taskID, path);
      return taskID;
    }
    return null;
  }

  /**
   * Transform http://hostname/v1/auth_id/container/object to
   * http://hostname/v1/auth_id
   *
   * @param publicURL public url
   * @param scheme URL scheme
   * @return accessURL access url
   * @throws IOException if path is malformed
   */
  public static String extractAccessURL(String publicURL, String scheme) throws IOException {
    if (publicURL != null && !publicURL.startsWith("http")) {
      String startScheme = scheme + "://";
      int end = publicURL.length();
      if (publicURL.indexOf("/", startScheme.length()) >= 0) {
        end = publicURL.indexOf("/", startScheme.length());
      }
      return publicURL.substring(0, end);
    }
    try {
      String hostName = new URI(publicURL).getAuthority();
      int  start = publicURL.indexOf("//") + 2 + hostName.length() + 1;
      for (int i = 0; i < 2; i++) {
        start = publicURL.indexOf("/", start) + 1;
      }
      String authURL = publicURL.substring(0, start);
      if (authURL.endsWith("/")) {
        authURL = authURL.substring(0, authURL.length() - 1);
      }
      return authURL;
    } catch (URISyntaxException e) {
      throw new IOException("Public URL: " + publicURL + " is not valid");
    }
  }

  /**
   * Extracts container name from http://hostname/v1/auth_id/container/object
   *
   * @param publicURL public url
   * @param accessURL access url
   * @return container name
   */
  public static String extractDataRoot(String publicURL, String accessURL) {
    if (publicURL != null && !publicURL.startsWith("http")) {
      return "";
    }
    String reminder = publicURL.substring(accessURL.length() + 1);
    String container = null;
    if (reminder.indexOf("/") > 0) {
      container =  reminder.substring(0, reminder.indexOf("/"));
    } else {
      container = reminder;
    }
    if (container.endsWith("/")) {
      container = container.substring(0, container.length() - 1);
    }
    return container;
  }

  /**
   * Extracts container/object  from http://hostname/v1/auth_id/container/object
   *
   * @param publicURL pubic url
   * @param accessURL access url
   * @return reminder of the URI
   */
  public static String extractReminder(String publicURL, String accessURL) {
    return publicURL.substring(accessURL.length());
  }

  public static void closeWithoutException(Closeable is) {
    if (is != null) {
      try {
        is.close();
      } catch (IOException ex) {
        LOG.debug("Ignore failure in closing the Closeable", ex);
      }
    }
  }

  public static boolean shouldAbort() {
    return Thread.interrupted();
  }

  /**
   * Transforms last modified time stamp from String to the long format
   *
   * @param strTime time in string format as returned from Swift
   * @return time in long format
   * @throws IOException if failed to parse time stamp
   */
  public static long lastModifiedAsLong(String strTime) throws IOException {
    final SimpleDateFormat simpleDateFormat = new SimpleDateFormat(TIME_PATTERN,
        Locale.US);
    try {
      long lastModified = simpleDateFormat.parse(strTime).getTime();
      if (lastModified == 0) {
        lastModified = System.currentTimeMillis();
      }
      return lastModified;
    } catch (ParseException e) {
      throw new IOException("Failed to parse " + strTime, e);
    }
  }

  public static boolean validContainer(String container) throws InvalidContainerNameException {
    if (container != null && container.length() < 4) {
      throw new InvalidContainerNameException("Container " + container
          + " length must be at least 3 letters");
    }
    if (InetAddressUtils.isIPv4Address(container)) {
      throw new InvalidContainerNameException("Container " + container
          + " is of IP address pattern");
    }
    return true;
  }

  /**
   * Close the Closeable objects and <b>ignore</b> any Exception or null
   * pointers. (This is the SLF4J equivalent of that in {@code IOUtils}).
   *
   * @param log the log to log at debug level. Can be null
   * @param closeables the objects to close
   */
  public static void closeAll(Logger log, java.io.Closeable... closeables) {
    for (java.io.Closeable c : closeables) {
      if (c != null) {
        try {
          if (log != null) {
            log.debug("Closing {}", c);
          }
          c.close();
        } catch (Exception e) {
          if (log != null && log.isDebugEnabled()) {
            log.debug("Exception in closing {}", c, e);
          }
        }
      }
    }
  }

  private static long stringToLong(String value, long defValue) {
    long res = defValue;
    if (value == null) {
      return res;
    }
    try {
      res = Long.valueOf(value);
    } catch (NumberFormatException ex) {
      if (value.endsWith("K")) {
        res = Long.valueOf(value.substring(0, value.length() - 1)) * 1024;
      }
    }
    return res;
  }
}