/*
 * The Apache Software License, Version 1.1
 *
 *
 * Copyright (c) 2001-2003 The Apache Software Foundation.  All rights
 * reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 *
 * 3. The end-user documentation included with the redistribution,
 *    if any, must include the following acknowledgment:
 *       "This product includes software developed by the
 *        Apache Software Foundation (http://www.apache.org/)."
 *    Alternately, this acknowledgment may appear in the software itself,
 *    if and wherever such third-party acknowledgments normally appear.
 *
 * 4. The names "Axis" and "Apache Software Foundation" must
 *    not be used to endorse or promote products derived from this
 *    software without prior written permission. For written
 *    permission, please contact apache@apache.org.
 *
 * 5. Products derived from this software may not be called "Apache",
 *    nor may "Apache" appear in their name, without prior written
 *    permission of the Apache Software Foundation.
 *
 * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED
 * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED.  IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR
 * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
 * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF
 * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
 * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
 * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 * ====================================================================
 *
 * This software consists of voluntary contributions made by many
 * individuals on behalf of the Apache Software Foundation.  For more
 * information on the Apache Software Foundation, please see
 * <http://www.apache.org/>.
 */

package org.jboss.axis.utils;

import org.jboss.axis.AxisEngine;
import org.jboss.axis.AxisFault;
import org.jboss.axis.Constants;
import org.jboss.axis.EngineConfiguration;
import org.jboss.axis.Handler;
import org.jboss.axis.MessageContext;
import org.jboss.axis.WSDDEngineConfiguration;
import org.jboss.axis.client.AxisClient;
import org.jboss.axis.deployment.wsdd.WSDDConstants;
import org.jboss.axis.deployment.wsdd.WSDDDeployment;
import org.jboss.axis.deployment.wsdd.WSDDDocument;
import org.jboss.axis.encoding.SerializationContext;
import org.jboss.axis.encoding.SerializationContextImpl;
import org.jboss.axis.server.AxisServer;
import org.jboss.logging.Logger;
import org.w3c.dom.Document;
import org.w3c.dom.Element;
import org.xml.sax.InputSource;

import java.io.FileInputStream;
import java.io.StringReader;
import java.io.StringWriter;
import java.net.InetAddress;
import java.net.UnknownHostException;

/**
 * Handy static utility functions for turning XML into
 * Axis deployment operations.
 *
 * @author Doug Davis (dug@us.ibm.com)
 * @author Glen Daniels (gdaniels@macromedia.com)
 */
public class Admin
{
   private static Logger log = Logger.getLogger(Admin.class.getName());

   /**
    * Process a given XML document - needs cleanup.
    */
   public Element[] AdminService(Element[] xml)
           throws Exception
   {
      log.debug("Enter: Admin::AdminService");
      MessageContext msgContext = MessageContext.getCurrentContext();
      Document doc = process(msgContext, xml[0]);
      Element[] result = new Element[1];
      result[0] = doc.getDocumentElement();
      log.debug("Exit: Admin::AdminService");
      return result;
   }

   protected static Document processWSDD(MessageContext msgContext,
                                         AxisEngine engine,
                                         Element root)
           throws Exception
   {
      Document doc = null;

      String action = root.getLocalName();
      if (action.equals("passwd"))
      {
         String newPassword = root.getFirstChild().getNodeValue();
         engine.setAdminPassword(newPassword);
         doc = XMLUtils.newDocument();
         doc.appendChild(root = doc.createElementNS("", "Admin"));
         root.appendChild(doc.createTextNode(Messages.getMessage("done00")));
         return doc;
      }

      if (action.equals("quit"))
      {
         log.error(Messages.getMessage("quitRequest00"));
         if (msgContext != null)
         {
            // put a flag into message context so listener will exit after
            // sending response
            msgContext.setProperty(MessageContext.QUIT_REQUESTED, "true");
         }
         doc = XMLUtils.newDocument();
         doc.appendChild(root = doc.createElementNS("", "Admin"));
         root.appendChild(doc.createTextNode(Messages.getMessage("quit00", "")));
         return doc;
      }

      if (action.equals("list"))
      {
         return listConfig(engine);
      }

      if (action.equals("clientdeploy"))
      {
         // set engine to client engine
         engine = engine.getClientEngine();
      }

      WSDDDocument wsddDoc = new WSDDDocument(root);
      EngineConfiguration config = engine.getConfig();
      if (config instanceof WSDDEngineConfiguration)
      {
         WSDDDeployment deployment =
                 ((WSDDEngineConfiguration)config).getDeployment();
         wsddDoc.deploy(deployment);
      }
      engine.refreshGlobalOptions();

      engine.saveConfiguration();

      doc = XMLUtils.newDocument();
      doc.appendChild(root = doc.createElementNS("", "Admin"));
      root.appendChild(doc.createTextNode(Messages.getMessage("done00")));

      return doc;
   }

   protected void preProcessWSDD(MessageContext msgContext, AxisEngine engine, Element root)
   {
      // NoOp (for any extenders to implement any preprocessing desired
   }

   /**
    * The meat of the Admin service.  Process an xML document rooted with
    * a "deploy", "undeploy", "list", or "quit" element.
    *
    * @param msgContext the MessageContext we're processing
    * @param root       the root Element of the XML
    * @return an XML Document indicating the results.
    */
   public Document process(MessageContext msgContext, Element root)
           throws Exception
   {
      // Check security FIRST.

      /** Might do something like this once security is a little more
       * integrated.
       if (!engine.hasSafePassword() &&
       !action.equals("passwd"))
       throw new AxisFault("Server.MustSetPassword",
       "You must change the admin password before administering Axis!",
       null, null);
       */

      /** For now, though - make sure we can only admin from our own
       * IP, unless the remoteAdmin option is set.
       */
      Handler serviceHandler = msgContext.getService();
      if (serviceHandler != null &&
              !JavaUtils.isTrueExplicitly(serviceHandler.getOption("enableRemoteAdmin")))
      {

         String remoteIP = msgContext.getStrProp(Constants.MC_REMOTE_ADDR);
         if (remoteIP != null &&
                 !remoteIP.equals("127.0.0.1"))
         {

            try
            {
               InetAddress myAddr = InetAddress.getLocalHost();
               InetAddress remoteAddr =
                       InetAddress.getByName(remoteIP);
               if (log.isDebugEnabled())
               {
                  log.debug("Comparing remote caller " + remoteAddr + " to " + myAddr);
               }


               if (!myAddr.equals(remoteAddr))
               {
                  log.error(Messages.getMessage("noAdminAccess01",
                          remoteAddr.toString()));
                  throw new AxisFault("Server.Unauthorized",
                          Messages.getMessage("noAdminAccess00"),
                          null, null);
               }
            }
            catch (UnknownHostException e)
            {
               throw new AxisFault("Server.UnknownHost",
                       Messages.getMessage("unknownHost00"),
                       null, null);
            }
         }
      }

      String rootNS = root.getNamespaceURI();
      AxisEngine engine = msgContext.getAxisEngine();

      // If this is WSDD, process it correctly.
      if (rootNS != null && rootNS.equals(WSDDConstants.URI_WSDD))
      {
         preProcessWSDD(msgContext, engine, root);
         return processWSDD(msgContext, engine, root);
      }

      // Else fault
      // TODO: Better handling here
      throw new Exception("FIXME");
   }

   /**
    * Get an XML document representing this engine's configuration.
    * <p/>
    * This document is suitable for saving and reloading into the
    * engine.
    *
    * @param engine the AxisEngine to work with
    * @return an XML document holding the engine config
    * @throws AxisFault
    */
   public static Document listConfig(AxisEngine engine)
           throws AxisFault
   {
      StringWriter writer = new StringWriter();
      SerializationContext context = new SerializationContextImpl(writer, null);
      context.setPretty(true);
      try
      {
         EngineConfiguration config = engine.getConfig();

         if (config instanceof WSDDEngineConfiguration)
         {
            WSDDDeployment deployment =
                    ((WSDDEngineConfiguration)config).getDeployment();
            deployment.writeToContext(context);
         }
      }
      catch (Exception e)
      {
         // If the engine config isn't a FileProvider, or we have no
         // engine config for some odd reason, we'll end up here.

         throw new AxisFault(Messages.getMessage("noEngineWSDD"));
      }

      try
      {
         writer.close();
         return XMLUtils.newDocument(new InputSource(new StringReader(writer.getBuffer().toString())));
      }
      catch (Exception e)
      {
         log.error("exception00", e);
         return null;
      }
   }

   public static void main(String args[]) throws Exception
   {
      int i = 0;

      if (args.length < 2 || !(args[0].equals("client") ||
              args[0].equals("server")))
      {
         log.error(Messages.getMessage("usage00", "Admin client|server <xml-file>"));

         log.error(Messages.getMessage("where00", "<xml-file>"));
         log.error("<deploy>");
         /*
         log.error( "  <transport name=a request=\"a,b,c\" sender=\"s\"");
         log.error( "                    response=\"d,e\"/>" );
         */
         log.error("  <handler name=a class=className/>");
         log.error("  <chain name=a flow=\"a,b,c\" />");
         log.error("  <chain name=a request=\"a,b,c\" pivot=\"d\"");
         log.error("                  response=\"e,f,g\" />");
         log.error("  <service name=a handler=b />");
         log.error("</deploy>");
         log.error("<undeploy>");
         log.error("  <handler name=a/>");
         log.error("  <chain name=a/>");
         log.error("  <service name=a/>");
         log.error("</undeploy>");
         log.error("<list/>");


         // throw an Exception which will go uncaught!  this way, a test
         // suite can invoke main() and detect the exception
         throw new IllegalArgumentException(Messages.getMessage("usage00",
                 "Admin client|server <xml-file>"));
         // System.exit( 1 );
      }

      Admin admin = new Admin();

      AxisEngine engine;
      if (args[0].equals("client"))
         engine = new AxisClient();
      else
         engine = new AxisServer();
      engine.setShouldSaveConfig(true);
      engine.init();
      MessageContext msgContext = new MessageContext(engine);

      try
      {
         for (i = 1; i < args.length; i++)
         {
            if (log.isDebugEnabled())
               log.debug(Messages.getMessage("process00", args[i]));

            Document doc = XMLUtils.newDocument(new FileInputStream(args[i]));
            admin.process(msgContext, doc.getDocumentElement());
         }
      }
      catch (Exception e)
      {
         log.error(Messages.getMessage("errorProcess00", args[i]), e);
         //System.exit( 1 );
         throw e;
      }
   }
}