package test;

import org.eclipse.core.resources.IFile;
import org.eclipse.core.resources.IMarker;
import org.eclipse.core.runtime.CoreException;
import org.eclipse.jface.preference.IPreferenceStore;
import org.eclipse.ui.texteditor.MarkerUtilities;
import net.sourceforge.phpeclipse.PHPeclipsePlugin;
import net.sourceforge.phpeclipse.actions.PHPStartApacheAction;
import net.sourceforge.phpdt.internal.compiler.parser.PHPOutlineInfo;
import net.sourceforge.phpdt.internal.compiler.parser.Parser;

import java.text.MessageFormat;
import java.util.Hashtable;

/**
 * The superclass for our PHP parsers.
 * @author Matthieu Casanova
 */
public abstract class PHPParserSuperclass {
  // strings for external parser call
  private static final String PARSE_ERROR_STRING = "Parse error"; //$NON-NLS-1$
  private static final String PARSE_WARNING_STRING = "Warning"; //$NON-NLS-1$
  public static final int ERROR = 2;
  public static final int WARNING = 1;
  public static final int INFO = 0;

  /**
   * Call the php parse command ( php -l -f <filename> )
   * and create markers according to the external parser output.
   * @param file the file that will be parsed
   */
  public static void phpExternalParse(final IFile file) {
    //IFile file = (IFile) resource;
  //  final IPath path = file.getFullPath();
    final IPreferenceStore store = PHPeclipsePlugin.getDefault().getPreferenceStore();
    final String filename = file.getLocation().toString();

    final String[] arguments = { filename };
    final MessageFormat form =
      new MessageFormat(store.getString(PHPeclipsePlugin.EXTERNAL_PARSER_PREF));
    final String command = form.format(arguments);

    final String parserResult =
      PHPStartApacheAction.getParserOutput(command, "External parser: ");

    try {
      // parse the buffer to find the errors and warnings
      createMarkers(parserResult, file);
    } catch (CoreException e) {
    }
  }

  /**
   * Create markers according to the external parser output.
   */
  private static void createMarkers(final String output, final IFile file)
    throws CoreException {
    // delete all markers
    file.deleteMarkers(IMarker.PROBLEM, false, 0);

    int indx = 0;
    int brIndx;
    boolean flag = true;
    while ((brIndx = output.indexOf("<br />", indx)) != -1) {
      // newer php error output (tested with 4.2.3)
      scanLine(output, file, indx, brIndx);
      indx = brIndx + 6;
      flag = false;
    }
    if (flag) {
      while ((brIndx = output.indexOf("<br>", indx)) != -1) {
        // older php error output (tested with 4.2.3)
        scanLine(output, file, indx, brIndx);
        indx = brIndx + 4;
      }
    }
  }

  private static void scanLine(final String output, final IFile file, final int indx, final int brIndx)
    throws CoreException {
    String current;
  //  String outLineNumberString; never used
    final StringBuffer lineNumberBuffer = new StringBuffer(10);
    char ch;
    current = output.substring(indx, brIndx);

    if (current.indexOf(PARSE_WARNING_STRING) != -1
      || current.indexOf(PARSE_ERROR_STRING) != -1) {
      final int onLine = current.indexOf("on line <b>");
      if (onLine != -1) {
        lineNumberBuffer.delete(0, lineNumberBuffer.length());
        for (int i = onLine; i < current.length(); i++) {
          ch = current.charAt(i);
          if ('0' <= ch && '9' >= ch) {
            lineNumberBuffer.append(ch);
          }
        }

        final int lineNumber = Integer.parseInt(lineNumberBuffer.toString());

        final Hashtable attributes = new Hashtable();

        current = current.replaceAll("\n", "");
        current = current.replaceAll("<b>", "");
        current = current.replaceAll("</b>", "");
        MarkerUtilities.setMessage(attributes, current);

        if (current.indexOf(PARSE_ERROR_STRING) != -1)
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_ERROR));
        else if (current.indexOf(PARSE_WARNING_STRING) != -1)
          attributes.put(
            IMarker.SEVERITY,
            new Integer(IMarker.SEVERITY_WARNING));
        else
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_INFO));
        MarkerUtilities.setLineNumber(attributes, lineNumber);
        MarkerUtilities.createMarker(file, attributes, IMarker.PROBLEM);
      }
    }
  }

  /**
   * This will parse the file and generate the outline info
   * @param parent the parent object
   * @param s the string that should be parsed
   * @return the outline info
   */
  public abstract PHPOutlineInfo parseInfo(Object parent, String s);

  /**
   * This will change the file to parse.
   * @param fileToParse the file that should be parsed
   */
  public abstract void setFileToParse(IFile fileToParse);

  /**
   * This will parse the given string
   * @param s the string to parse
   * @throws CoreException an exception that can be launched
   */
  public abstract void parse(String s) throws CoreException;

  /**
   * This will set a marker.
   * @param file the file that generated the marker
   * @param message the message
   * @param charStart the starting character
   * @param charEnd the end character
   * @param errorLevel the error level ({@link PHPParserSuperclass#ERROR},
   *        {@link PHPParserSuperclass#INFO},{@link PHPParserSuperclass#WARNING})
   * @throws CoreException an exception throwed by the MarkerUtilities
   */
  public static void setMarker(final IFile file,
                               final String message,
                               final int charStart,
                               final int charEnd,
                               final int errorLevel)
    throws CoreException {
    if (file != null) {
      final Hashtable attributes = new Hashtable();
      MarkerUtilities.setMessage(attributes, message);
      switch (errorLevel) {
        case Parser.ERROR :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_ERROR));
          break;
        case Parser.WARNING :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_WARNING));
          break;
        case Parser.INFO :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_INFO));
          break;
      }
      MarkerUtilities.setCharStart(attributes, charStart);
      MarkerUtilities.setCharEnd(attributes, charEnd);
      MarkerUtilities.createMarker(file, attributes, IMarker.PROBLEM);
    }
  }

  /**
   * This will set a marker.
   * @param file the file that generated the marker
   * @param message the message
   * @param line the line number
   * @param errorLevel the error level ({@link PHPParserSuperclass#ERROR},
   *        {@link PHPParserSuperclass#INFO},{@link PHPParserSuperclass#WARNING})
   * @throws CoreException an exception throwed by the MarkerUtilities
   */
  public static void setMarker(final IFile file,
                               final String message,
                               final int line,
                               final int errorLevel,
                               final String location)
    throws CoreException {
    if (file != null) {
      final Hashtable attributes = new Hashtable();
      MarkerUtilities.setMessage(attributes, message);
      switch (errorLevel) {
        case Parser.ERROR :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_ERROR));
          break;
        case Parser.WARNING :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_WARNING));
          break;
        case Parser.INFO :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_INFO));
          break;
      }
      attributes.put(IMarker.LOCATION,location);
      MarkerUtilities.setLineNumber(attributes, line);
      MarkerUtilities.createMarker(file, attributes, IMarker.PROBLEM);
    }
  }
  /**
   * This will set a marker.
   * @param file the file that generated the marker
   * @param message the message
   * @param charStart the starting character
   * @param charEnd the end character
   * @param errorLevel the error level ({@link PHPParserSuperclass#ERROR},
   *        {@link PHPParserSuperclass#INFO},{@link PHPParserSuperclass#WARNING})
   * @param location the location of the error
   * @throws CoreException an exception throwed by the MarkerUtilities
   */
  public static void setMarker(final IFile file,
                               final String message,
                               final int charStart,
                               final int charEnd,
                               final int errorLevel,
                               final String location)
    throws CoreException {
    if (file != null) {
      final Hashtable attributes = new Hashtable();
      MarkerUtilities.setMessage(attributes, message);
      switch (errorLevel) {
        case Parser.ERROR :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_ERROR));
          break;
        case Parser.WARNING :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_WARNING));
          break;
        case Parser.INFO :
          attributes.put(IMarker.SEVERITY, new Integer(IMarker.SEVERITY_INFO));
          break;
      }
      attributes.put(IMarker.LOCATION,location);
      MarkerUtilities.setCharStart(attributes, charStart);
      MarkerUtilities.setCharEnd(attributes, charEnd);
      MarkerUtilities.createMarker(file, attributes, IMarker.PROBLEM);
    }
  }
}