All Downloads are FREE. Search and download functionalities are using the official Maven repository.

mq5.1-source.src.win32.svc.jmqbrokersvc.c Maven / Gradle / Ivy

/*
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * Copyright (c) 2000-2010 Oracle and/or its affiliates. All rights reserved.
 *
 * The contents of this file are subject to the terms of either the GNU
 * General Public License Version 2 only ("GPL") or the Common Development
 * and Distribution License("CDDL") (collectively, the "License").  You
 * may not use this file except in compliance with the License.  You can
 * obtain a copy of the License at
 * https://glassfish.dev.java.net/public/CDDL+GPL_1_1.html
 * or packager/legal/LICENSE.txt.  See the License for the specific
 * language governing permissions and limitations under the License.
 *
 * When distributing the software, include this License Header Notice in each
 * file and include the License file at packager/legal/LICENSE.txt.
 *
 * GPL Classpath Exception:
 * Oracle designates this particular file as subject to the "Classpath"
 * exception as provided by Oracle in the GPL Version 2 section of the License
 * file that accompanied this code.
 *
 * Modifications:
 * If applicable, add the following below the License Header, with the fields
 * enclosed by brackets [] replaced by your own identifying information:
 * "Portions Copyright [year] [name of copyright owner]"
 *
 * Contributor(s):
 * If you wish your version of this file to be governed by only the CDDL or
 * only the GPL Version 2, indicate your decision by adding "[Contributor]
 * elects to include this software in this distribution under the [CDDL or GPL
 * Version 2] license."  If you don't indicate a single choice of license, a
 * recipient has the option to distribute your version of this file under
 * either the CDDL, the GPL Version 2 or to extend the choice of license to
 * its licensees as provided above.  However, if you add GPL Version 2 code
 * and therefore, elected the GPL Version 2 license, then the option applies
 * only if the new code is made subject to such option by the copyright
 * holder.
 */

/*
 * @(#)jmqbrokersvc.c	1.39 07/20/07
 */ 

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

#include "registry.h"
#include "mqapp.h"

#define MAX_OPTIONS 20

#define BROKER_MAIN                 "com/sun/messaging/jmq/jmsserver/Broker"
#define BROKER_SCM_EVENT_MANAGER    "com/sun/messaging/jmq/jmsserver/util/ntsvc/SCMEventManager"

// Main broker jar.
#define MAIN_JAR                    "imqbroker.jar"

// Class path entries. Relative paths are assumed to be relative to 
// imqhome/lib
char *classpath_entries[] = {
        "imqbroker.jar",
        "imqutil.jar",
        "jsse.jar",
        "jnet.jar",
        "jcert.jar",
        "..\\..\\share\\lib\\jdmkrt.jar",
        "..\\..\\share\\lib\\mfwk_instrum_tk.jar"
        };
int nclasspath_entries = sizeof (classpath_entries) / sizeof(char *);

int   application_argc = 0;
char *application_argv[128];
int   jvm_argc = 0;

// Argv list for jvm args. We skip first slot to reserv for "-server" if needed
char *jvm_argv_buffer[128];
char **jvm_argv = &(jvm_argv_buffer[1]);
char *broker_port = NULL;

char adminkey[80];

/* Information about the forked child process */
STARTUPINFO jsi;
PROCESS_INFORMATION jpi;

SERVICE_STATUS          ssStatus;
SERVICE_STATUS_HANDLE   sshStatusHandle;
BOOL bConsole = FALSE;
BOOL bVerbose = FALSE;
BOOL bManaged = FALSE;
char jmqhome[MAX_PATH];
char jmqbinhome[MAX_PATH];
char jmqbinhome[MAX_PATH];
char jmqvarhome[MAX_PATH];
char jmqetchome[MAX_PATH];
char jmqlibhome[MAX_PATH];
char jmqlibexthome[MAX_PATH];
char jrehome[MAX_PATH];
char classpath[65536];
DWORD dwErr = 0;
HANDLE hServerStopEvent = NULL;
int jmqBrokerExitCode = 0;

void WINAPI ServiceMain(DWORD argc, LPTSTR *argv);
void parseArgs (int argc, char *argv[]);
int stringToArgv(char *string, int argc, char *argv[], int maxn);
void dumpArgv(char *m, int argc, char *argv[]);
VOID WINAPI ControlHandler(DWORD ctrlCode);
BOOL WINAPI ConsoleCtrlHandler(DWORD dwCtrlType);
int Initialize(void);
VOID AddLibJars(const char *varhome, char *classpath);
VOID ServiceStart (DWORD dwArgc, LPTSTR *lpszArgv);
VOID ServiceStop(void);
BOOL ReportStatus(DWORD dwCurrentState,
                  DWORD dwWin32ExitCode,
                  DWORD dwWaitHint);
VOID AddToMessageLog(WORD wType, LPTSTR msg);
VOID LogError(LPTSTR msg);
VOID LogWarning(LPTSTR msg);
VOID LogInformation(LPTSTR msg);
int StartJavaApplication(
    DWORD argc,   LPSTR *argv,  /* Arguments to pass to JVM */
    LPSTR mainClass,            /* Main class to invoke.*/
    DWORD appArgc, LPSTR *appArgv   /* Arguments to pass to Main class */
    );
void getParamsFromRegistry();
BOOL createKeyFile(char *keyfile, const char *key);
char *GetNumericPropertyValue(const char *s);
char *FindLine(const char *file, const char *s);
char *FindBrokerPort( int argc, char *argv[], char *jmqvarhome, char *jmqlibhome);
BOOL argInArgv(const char *s, const char **argv, int argc);
BOOL hotspotServerAvailable(const char *jhome);

extern int FindJavaRuntime(const char*home, const char *varhome, char *path);
extern int mymkdir(const *path);

VOID main(int argc, char** argv)
{
    SERVICE_TABLE_ENTRY DispatchTable[] = {
        { TEXT(SERVICE_NAME), (LPSERVICE_MAIN_FUNCTION) ServiceMain } ,
        { NULL, NULL}
    };

    jmqhome[0] = '\0';
    jmqbinhome[0] = '\0';
    jmqvarhome[0] = '\0';
    jmqetchome[0] = '\0';
    jmqlibhome[0] = '\0';
    jmqlibexthome[0] = '\0';
    jrehome[0] = '\0';
    classpath[0] = '\0';
    adminkey[0] = '\0';

    parseArgs(argc, argv);

    if (bConsole) {
        if (Initialize() < 0) {
            exit (1);
        };
        ServiceStart(application_argc, application_argv);
        exit (jmqBrokerExitCode);
    } else {
        // NTService
        if (!StartServiceCtrlDispatcher(DispatchTable)) {
            LogError("StartServiceCtrlDispatcher failed");
        }
    }
}

// Handles console events that are generate on shutdown,
// logoff, Ctrl-C, Ctrl-Break and console close.`
BOOL WINAPI ConsoleCtrlHandler(DWORD dwCtrlType) {
    switch(dwCtrlType)
    {
        case CTRL_SHUTDOWN_EVENT:
            // Called when the system is shutdown. We want to shutdown the svc
            ServiceStop();
            return TRUE;

        // For other events let the default handler take care of them.
        // Even though the documentation says the default handler exits
        // the process, it appears as though a different handler is used
        // for services in which case LOGOFF_EVENT does not exit
        case CTRL_LOGOFF_EVENT:
        case CTRL_C_EVENT:
        case CTRL_BREAK_EVENT:
        case CTRL_CLOSE_EVENT:
        default:
            return FALSE;
    }
}

void WINAPI ServiceMain(DWORD argc, LPTSTR *argv)
{
    // Get parameters from the registry. This will set
    // jrehome, jvm_argc/argv and application_argc/argv to whatever
    // was installed in the registry (if any). These may then be
    // overwritten in parseArgs() by what the user entered in the SCM panel.
    getParamsFromRegistry();
    parseArgs(argc, argv);

    // Call RegisterServiceCtrlHandler immediately to register a service
    // control handler function. The returned SERVICE_STATUS_HANDLE is
    // saved with global scope, and used as a service id in calls to
    // SetServiceStatus.
    sshStatusHandle = RegisterServiceCtrlHandler( TEXT(SERVICE_NAME),
                                                  ControlHandler);

    if (!sshStatusHandle)
        goto finally;

    // The global ssStatus SERVICE_STATUS structure contains information
    // about the service, and is used throughout the program in calls made
    // to SetStatus through the ReportStatus function.
    ssStatus.dwServiceType = SERVICE_WIN32_OWN_PROCESS;
    ssStatus.dwServiceSpecificExitCode = 0;


    // If we could guarantee that all initialization would occur in less
    // than one second, we would not have to report our status to the
    // service control manager. For good measure, we will assign
    // SERVICE_START_PENDING to the current service state and inform the
    // service control manager through our ReportStatus function.
    if (!ReportStatus(SERVICE_START_PENDING, NO_ERROR, 3000))
        goto finally;

    // Do it! In ServiceStart, we'll send additional status reports to the
    // service control manager, especially the SERVICE_RUNNING report once 
    // our JVM is initiallized and ready to be invoked.
    if (Initialize() == 0) {
        ServiceStart(application_argc, application_argv);
    }

finally:

    // Report the stopped status to the service control manager, if we have
    // a valid server status handle.
     if (sshStatusHandle)
        (VOID)ReportStatus( SERVICE_STOPPED, dwErr, 0);
}


// Each Win32 service must have a control handler to respond to
// control requests from the dispatcher.

VOID WINAPI ControlHandler(DWORD ctrlCode)
{

    switch(ctrlCode)
    {
        case SERVICE_CONTROL_STOP:
            // Request to stop the service. Report SERVICE_STOP_PENDING
            // to the service control manager before calling ServiceStop()
            // to avoid a "Service did not respond" error.
            ReportStatus(SERVICE_STOP_PENDING, NO_ERROR, 0);
            ServiceStop();
            return;

        case SERVICE_CONTROL_SHUTDOWN:
            ReportStatus(SERVICE_STOP_PENDING, NO_ERROR, 0);
            ServiceStop();
            return;

        case SERVICE_CONTROL_PAUSE:
            //ReportStatus(SERVICE_PAUSE_PENDING, NO_ERROR, 0);
            return;

        case SERVICE_CONTROL_CONTINUE:
            //ReportStatus(SERVICE_CONTINUE_PENDING, NO_ERROR, 0);
            return;

        case SERVICE_CONTROL_INTERROGATE:
            // This case MUST be processed, even though we are not
            // obligated to do anything substantial in the process.
            break;

         default:
            // Any other cases...
            break;

    }

    // After invocation of this function, we MUST call the SetServiceStatus
    // function, which is accomplished through our ReportStatus function. We
    // must do this even if the current status has not changed.
    ReportStatus(ssStatus.dwCurrentState, NO_ERROR, 0);
}


/*
 * Do some initialization. This basically locates JMQ, the JDK, determines
 * our CLASSPATH, and dynamically loads jvm.dll
 */
int Initialize()
{
    int n;
    char buffer[512];
    HANDLE hLib = NULL;
    MqEnv me;

    MqAppInitMqEnv(&me, "");
    MqAppInitializeNoJava(&me);

    strcpy(jmqhome, me.imqhome);

    if (jmqlibhome[0] == '\0') {
        strcpy(jmqlibhome, me.imqlibhome);
    }
    if (jmqlibexthome[0] == '\0') {
        strcpy(jmqlibexthome, jmqlibhome);
        strcat(jmqlibexthome, "\\ext");
    }

    // Determine jmqvarhome. This can be set via an environment variable
    if (jmqvarhome[0] == '\0') {
        strcpy(jmqvarhome, me.imqvarhome);
    }

    if (_access(jmqvarhome, 00) < 0) {
        if (NULL != jmqvarhome) {
            if (0 != mymkdir(jmqvarhome)){
                sprintf(buffer, TEXT("Unable to make directory to %s"),
                                jmqvarhome);
                LogError(buffer);
		return -1;
	    }
	}
    }

    if (jmqetchome[0] == '\0' && me.imqetchome != '\0') {
        strcpy(jmqetchome, me.imqetchome);
    }
    if (jmqetchome[0] != '\0' && _access(jmqetchome, 00) < 0) {
        sprintf(buffer, "Couldn't find IMQ_ETCHOME '%s'", jmqetchome);
        LogError(buffer);
        return -1;
    }

    // Make sure IMQ_HOME is set in our environment. This will be
    // needed by any commands we execute in IMQ_HOME\bin
    if (!SetEnvironmentVariable("IMQ_HOME", jmqhome)) {
        sprintf(buffer, "Could not set IMQ_HOME to '%s'", jmqhome);
        LogError(buffer);
    }

    if (*jrehome == '\0') {
        /* Locate a java runtime. */
        FindJavaRuntime(jmqhome, jmqvarhome, jrehome);
    }

    if (*jrehome == '\0') {
        sprintf(buffer, "Please specify a Java runtime using the IMQ_JAVAHOME\nenvironment variable, or -javahome command line option\n");
        LogInformation(buffer);
        return -1;
    }

    if (_access(jrehome, 00) < 0) {
        sprintf(buffer, "Invalid Java Runtime '%s'", jrehome);
        LogError(buffer);
        return -1;
    }

    // Change working directory to jmqhome
    if (NULL != jmqhome){
	strcpy(jmqbinhome, jmqhome);
	strcat(jmqbinhome, "\\bin");
        if (0 != _chdir(jmqbinhome)){
            sprintf(buffer, TEXT("Unable to change working directory to %s"),
                         jmqbinhome);
            LogError(buffer);
	}
    }

    // Initialize classpath
    *classpath = '\0';
    for (n = 0; n < nclasspath_entries; n++) {
        if (n != 0) strcat(classpath, ";");
        if (*classpath_entries[n] != '\\') {
            strcat(classpath, jmqhome);
            strcat(classpath, "\\lib\\");
        }
        strcat(classpath, classpath_entries[n]);
    }

    /*
     * If IMQ_DEFAULT_EXT_JARS is set, append it to the value of 'classpath'
     */
    if (me.imqextjars[0] != '\0') {
        strcat(classpath, ";");
        strcat(classpath, me.imqextjars);
    }

    AddLibJars(jmqlibexthome, classpath);

    if (bVerbose && bConsole) {
        printf("Starting " DISPLAY_NAME "\n");
        printf("   imqhome=%s\n", jmqhome);
        printf("imqvarhome=%s\n", jmqvarhome);
        printf("imqetchome=%s\n", jmqetchome);
        printf("   jrehome=%s\n", jrehome);
    }

    broker_port = FindBrokerPort(application_argc, application_argv,
                            jmqvarhome, jmqlibhome);

    return 0;
}

/**
 * Add all the jar files in the $IMQ_VARHOME/lib directory to classpath.
 * These are user defined jar files needed for things like JDBC connectors
 */
VOID AddLibJars(const char *exthome, char *classpath) {

    WIN32_FIND_DATA dir;
    char lib[MAX_PATH];
    HANDLE handle;

    /* Added ./lib/ext/classes directory */
    strcat(classpath, ";");
    strcat(classpath, exthome);

    sprintf(lib, "%s\\*", exthome);

    /* Add jars if any */
    handle = FindFirstFile(lib, &dir);
    if (handle == INVALID_HANDLE_VALUE) {
	return;
    }
    do {
	int len = strlen(dir.cFileName);

	if (len <= 4) {
            // not a .jar file
	} else if (strcmp(dir.cFileName + len - 4, ".jar") != 0 &&
                   strcmp(dir.cFileName + len - 4, ".zip") != 0) {
            // File doesn't end in .jar or .zip
	} else {
            strcat(classpath, ";");
            strcat(classpath, exthome);
            strcat(classpath, "\\");
            strcat(classpath, dir.cFileName);
	}
    } while (FindNextFile(handle, &dir));
}

// This method is called from ServiceMain() when NT starts the service

VOID ServiceStart (DWORD service_argc, LPTSTR *service_argv)
{
    int  status;
    int  n;
    char keyfile[MAX_PATH + 1];

    // Create a Stop Event
    hServerStopEvent = CreateEvent(
        NULL,    
        TRUE,    
        FALSE,   
        NULL);

    if ( hServerStopEvent == NULL)
        goto cleanup;

    if (!ReportStatus(SERVICE_RUNNING,NO_ERROR,0)){
        goto cleanup;
    }

    jvm_argv[jvm_argc] = malloc(sizeof(classpath) + 80);
    sprintf(jvm_argv[jvm_argc], "-cp");
    jvm_argc++;

    jvm_argv[jvm_argc] = malloc(sizeof(classpath) + 80);
    sprintf(jvm_argv[jvm_argc], "%s", classpath);
    jvm_argc++;

    jvm_argv[jvm_argc] = malloc(sizeof(jmqhome) + 80);
    sprintf(jvm_argv[jvm_argc], "-Dimq.home=%s", jmqhome);
    jvm_argc++;

    jvm_argv[jvm_argc] = malloc(sizeof(jmqvarhome) + 80);
    sprintf(jvm_argv[jvm_argc], "-Dimq.varhome=%s", jmqvarhome);
    jvm_argc++;

    if (jmqetchome[0] != '\0') {
        jvm_argv[jvm_argc] = malloc(sizeof(jmqetchome) + 80);
        sprintf(jvm_argv[jvm_argc], "-Dimq.etchome=%s", jmqetchome);
        jvm_argc++;
    }

    // If -Xms has not been specified yet, then specify it.
    if (!argInArgv("-Xms", jvm_argv, jvm_argc)) {
        jvm_argv[jvm_argc] = _strdup("-Xms16m");
        jvm_argc++;
    }

    // If -Xmx has not been specified yet, then specify it.
    if (!argInArgv("-Xmx", jvm_argv, jvm_argc)) {
        jvm_argv[jvm_argc] = _strdup("-Xmx192m");
        jvm_argc++;
    }

    if (!bConsole) {
        // this prevents this process from getting whacked upon the logoff event
        // We need this to cleanly terminate service on shutdown
        SetConsoleCtrlHandler(ConsoleCtrlHandler, TRUE);
        // this prevents the VM from getting whacked upon the logoff event
        jvm_argv[jvm_argc] = _strdup("-Xrs");
        jvm_argc++;
    }

    // If there is a hotspot server vm available, use it
    if (!argInArgv("-server", jvm_argv, jvm_argc) &&
         hotspotServerAvailable(jrehome)) {
        // Point to start of argv buffer (we left an empty slot there)
        // and insert -server at the start
        jvm_argv = jvm_argv_buffer;
        jvm_argv[0] = _strdup("-server");
        jvm_argc++;
    }

    if (bVerbose && bConsole) {
        dumpArgv("Java VM Arguments:", jvm_argc, jvm_argv);
        dumpArgv("Broker Arguments", application_argc, application_argv);
    }

    srand(time(0));

    do {
        if (!bConsole) {
            // Generate a random admin session key. We need this
            // when running as a service to shutdown the broker
            // using jmqcmd
            sprintf(adminkey, "%d%d%d%d%d%d",
                rand(), rand(), rand(), rand(), rand(), rand());
            createKeyFile(keyfile, adminkey);
            service_argv[service_argc] = _strdup("-adminkeyfile");
            service_argc++;
            service_argv[service_argc] = _strdup(keyfile);
            service_argc++;
        } else {
            keyfile[0] = '\0';
            adminkey[0] = '\0';
        }


        // After the initialization is complete (we've checked for
        // arguments) and the service control manager has been told
        // the service is running, invoke the Java application.
        // This call will block until the broker exits`
        status = StartJavaApplication(jvm_argc, jvm_argv,
            "com.sun.messaging.jmq.jmsserver.Broker",
            service_argc, service_argv);

        jmqBrokerExitCode = status;

        // Broker may have exited due to a restart command from JMQ
        // admin. If this is the case it would have set its exit
        // code to 255 which means we need to restart it. We sleep
        // for a bit to avoid spinning in a tight loop if we get
        // here unintentionally 
        if (status == 255 && !bManaged) {
            Sleep(1000 * 1);
        } else {
            // Report the stopped status in ServiceMain
            //if (sshStatusHandle)
            //   (VOID)ReportStatus( SERVICE_STOPPED, NO_ERROR, 0);
        }

        if (!bConsole) {
            // Remove -adminkeyfile options.
            for (n = 0; n < 2; n++) {
                service_argc--;
                if (service_argv[service_argc] != NULL) {
                    free(service_argv[service_argc]);
                    service_argv[service_argc] = NULL;
                }
            }
        }

        // If -managed was supplied, the restart will be performed by Glassfish
        // If -managed was not supplied, then loop round and restart

    } while (status == 255 && !bManaged);


cleanup:

    if (hServerStopEvent) {
        CloseHandle(hServerStopEvent);
    }
    LogInformation(TEXT("Broker process is stopped"));
}

/*
 *  Create a key file and write the specified key string to it.
 *  char *keyfile   Buffer to hold name of key file. Allocated by caller.
 *  char *key       Key  string to write to file
 */
BOOL createKeyFile(char *keyfile, const char *key) {

    char buf[MAX_PATH];
    int  nbytes = 0;
    HANDLE hFile;

    *keyfile = '\0';

    // Generate a temp file  in IMQ_VARHOME
    if (GetTempFileName(jmqvarhome, "imqkey", 0, keyfile) == 0) {
        LogError("Could not get a temp file");
        sprintf(keyfile, "%s\\%s", jmqvarhome, "imqkey.txt");
    }

    // Create file
    hFile = CreateFile(keyfile,
                    GENERIC_WRITE,              // open for writing
                    0,                          // do not share
                    NULL,                       // no security
                    CREATE_ALWAYS,              // overwirte existing
                    FILE_ATTRIBUTE_NORMAL,      // normal file
                    NULL);                      // no attr. template
    if (hFile == INVALID_HANDLE_VALUE) {
        sprintf(buf, "Could not open admin key file %s", keyfile);
        LogError(buf);
        return 0;
    }

    // Write admin key to temp file
    if (!WriteFile(
                hFile,              // File handle to write to
                adminkey,           // buffer to write
                strlen(adminkey),   // number of bytes to write
                &nbytes,            // [out] number of written
                NULL                // Overlap not needed (not async)
                )) { 
        sprintf(buf, "Could not write to admin key file %s", keyfile);
        LogError(buf);
        return 0;
    }
    CloseHandle(hFile);
    return 1;
}

/*
 * Stop the service means we need to stop the broker.
 * We do this by running imqcmd
 */
VOID ServiceStop()
{
    char cmdLine[1024];
    char  buf[512];
    DWORD exitCode = 0;
    STARTUPINFO si;
    PROCESS_INFORMATION pi;

    /* Signal the stop event. */
    if ( hServerStopEvent ){
        SetEvent(hServerStopEvent);
    }

    /* Run imqcmd to stop broker */
    sprintf(cmdLine, "\"%s\\bin\\imqcmd\"", jmqhome);
    //strcat(cmdLine, " shutdown bkr -f -s -u admin -p admin");
    strcat(cmdLine, " shutdown bkr -f -s -adminkey -u admin -pw ");
    strcat(cmdLine, adminkey);
    if (broker_port != NULL) {
        strcat(cmdLine, " -b localhost:");
        strcat(cmdLine, broker_port);
    }

    ZeroMemory(&si, sizeof(si));
    si.cb = sizeof(si);
    ZeroMemory(&pi, sizeof(pi));

    if (bVerbose) {
        sprintf(buf, "Shutting down service by executing %s", cmdLine);
        LogInformation(buf);
    }

    /* Execute the child process */
    if ( !CreateProcess(
        NULL,               // No module name. Use command line.
        cmdLine,            // Command line
        NULL,               // Prcess handle not inheritable.
        NULL,               // Thread handle not inheritable.
        FALSE,              // Set handle inheritance to FALSE.
        0,                  // No creation flags, use parent's console.
        NULL,               // Use parent's environment block
        NULL,               // Use parent's current directory
        &si,                // Pointer to STARTUPINFO struct [in]
        &pi                 // Pointer to PROCESS_INFORMATION struct [out]
        )
       ) {
        sprintf(buf, "Stopping service failed. Could not execute %s", cmdLine);
        LogError(buf);
        return;
    }

    if (sshStatusHandle) {
        // Make sure service control manager doesn't time us out
        ReportStatus(SERVICE_STOP_PENDING, NO_ERROR, 0);
    }

    // Wait for child to exit
    WaitForSingleObject (pi.hProcess, INFINITE);

    // Get exit code from child process
    if (!GetExitCodeProcess(pi.hProcess, &exitCode)) {
        LogError(TEXT("Unable to get child's exit code."));
    }

    /*
     * Can't rely on imqcmd exit status yet. 
    if (exitCode != 0) {
        sprintf(buf, "Error stopping service. %s returned %d", cmdLine, exitCode);
        LogError(buf);
    }
    */

    // Close process and thread handles
    CloseHandle(pi.hProcess);
    CloseHandle(pi.hThread);

    /**
     * Let the ServiceMain thread to set STOPPED status
     */
    //(VOID)ReportStatus( SERVICE_STOPPED, NO_ERROR, 0);

    return;
}

// Throughout the program, calls to SetServiceStatus are required
// which are handled by calling this function. Here, the non-constant
// members of the SERVICE_STATUS struct are assigned and SetServiceStatus
// is called with the struct. Note that we will not report to the service
// control manager if we are running as  console application.

BOOL ReportStatus(DWORD dwCurrentState,
                  DWORD dwWin32ExitCode,
                  DWORD dwWaitHint)
{
    static DWORD dwCheckPoint = 1;
    BOOL bResult = TRUE;

    if ( !bConsole ) 
    {
        if (dwCurrentState == SERVICE_START_PENDING)
            ssStatus.dwControlsAccepted = 0;
        else {
            ssStatus.dwControlsAccepted =
                SERVICE_ACCEPT_STOP;
        }

        ssStatus.dwCurrentState = dwCurrentState;
        ssStatus.dwWin32ExitCode = dwWin32ExitCode;
        ssStatus.dwWaitHint = dwWaitHint;

        if ( ( dwCurrentState == SERVICE_RUNNING ) ||
             ( dwCurrentState == SERVICE_STOPPED ) )
            ssStatus.dwCheckPoint = 0;
        else
            ssStatus.dwCheckPoint = dwCheckPoint++;

        if (!(bResult = SetServiceStatus( sshStatusHandle, &ssStatus))) {
            LogError(TEXT("SetServiceStatus"));
        }
    }

    return bResult;
}

VOID LogError(LPTSTR msg) {
    AddToMessageLog(EVENTLOG_ERROR_TYPE, msg);
}

VOID LogWarning(LPTSTR msg) {
    AddToMessageLog(EVENTLOG_WARNING_TYPE, msg);
}

VOID LogInformation(LPTSTR msg) {
    AddToMessageLog(EVENTLOG_INFORMATION_TYPE, msg);
}

// If running as a service, use event logging to post a message
// If not, display the message on the console.
// wType should be one of:
//  EVENTLOG_ERROR_TYPE  
//  EVENTLOG_WARNING_TYPE
//  EVENTLOG_INFORMATION_TYPE

VOID AddToMessageLog(WORD wType, LPTSTR msg )
{
    TCHAR   *szMsg = NULL;
    HANDLE  hEventSource;
    LPTSTR  lpszStrings[2];
    LPVOID lpMsgBuf = NULL;
    int len = 24;

    if (wType == EVENTLOG_ERROR_TYPE) {
        dwErr = GetLastError();
        FormatMessage(
            FORMAT_MESSAGE_ALLOCATE_BUFFER |
            FORMAT_MESSAGE_FROM_SYSTEM |
            FORMAT_MESSAGE_IGNORE_INSERTS ,
            NULL,
            dwErr,
            MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
            (LPTSTR)&lpMsgBuf,
            0,
            NULL
        );
        len += strlen(TEXT(": :  ")) + strlen(TEXT(DISPLAY_NAME));
        len += strlen(msg) + 24 + strlen(lpMsgBuf);
        szMsg = (TCHAR *)malloc(len*sizeof(TCHAR));
        _stprintf(szMsg, TEXT("%s: %s: %d %s"), TEXT(DISPLAY_NAME), msg, dwErr,
            lpMsgBuf);
        LocalFree(lpMsgBuf);
    } else {
        len += strlen(TEXT(": ")) + strlen(TEXT(DISPLAY_NAME)) + strlen(msg);
        szMsg = (TCHAR *)malloc(len*sizeof(TCHAR));
        _stprintf(szMsg, TEXT("%s: %s"), TEXT(DISPLAY_NAME), msg);
    }

    if (!bConsole) {
        hEventSource = RegisterEventSource(NULL, TEXT(SERVICE_NAME));

        lpszStrings[0] = szMsg;

        if (hEventSource != NULL) {
            ReportEvent(hEventSource, 
                wType,  
                0,                    
                0,                    
                NULL,                 
                1,                    
                0,                    
                lpszStrings,          
                NULL);                

            DeregisterEventSource(hEventSource);
        }
    } else{
        _tprintf(TEXT("%s\n"), szMsg);
    }
    if (szMsg != NULL) free(szMsg);
}

/*
 * Run a Java application and block until it exits.
 * Return the exit status from the Java application
 */
int StartJavaApplication(
    DWORD argc,   LPSTR *argv,  /* Arguments to pass to JVM */
    LPSTR mainClass,            /* Main class to invoke. */
    DWORD appArgc, LPSTR *appArgv   /* Arguments to pass to Main class */
    )
{
    char *buf = NULL;
    char *cmdLine = NULL;
    char javaCmd[512];
    DWORD exitCode = 0;
    UINT i;
    int len = 24;

    sprintf(javaCmd, "\"%s\\bin\\java\"", jrehome);
    len += strlen(javaCmd);
    
    for (i = 0; i < argc; i++) {
        len += strlen(argv[i]) + 8;
    }

    len += strlen(mainClass);

    for (i = 0; i < appArgc; i++) {
        len += strlen(appArgv[i]) + 8;
    }

    cmdLine = (char *)malloc(len * sizeof(char));
    memset(cmdLine, '\0', len);

    /* Copy Java command and command line arguments into command line */
    strcpy(cmdLine, javaCmd);
    for (i = 0; i < argc; i++) {
        strcat(cmdLine, " ");
        strcat(cmdLine, "\"");
        strcat(cmdLine, argv[i]);
        strcat(cmdLine, "\"");
    }

    /* Copy main class into command line */
    strcat(cmdLine, " ");
    strcat(cmdLine, mainClass);
    strcat(cmdLine, " ");

    /* Copy main class arguments into command line */
    for (i = 0; i < appArgc; i++) {
        strcat(cmdLine, " ");
        strcat(cmdLine, "\"");
        strcat(cmdLine, appArgv[i]);
        strcat(cmdLine, "\"");
    }

    ZeroMemory(&jsi, sizeof(jsi));
    jsi.cb = sizeof(jsi);
    ZeroMemory(&jpi, sizeof(jpi));

    len = strlen(cmdLine) + 24;
    buf = (char *)malloc(len * sizeof(char));
    memset(buf, '\0', len);
    if (bVerbose || !bConsole) {
        sprintf(buf, "Starting %s", cmdLine);
        LogInformation(buf);
    }
;
    // Execute the child process
    if ( !CreateProcess(
        NULL,               // Use cmdLine
        cmdLine,            // Command line
        NULL,               // Prcess handle not inheritable.
        NULL,               // Thread handle not inheritable.
        TRUE,               // Set handle inheritance to TRUE.
        0,                  // No creation flags, use parent's console.
        NULL,               // Use parent's environment block
        NULL,               // Use parent's current directory
        &jsi,                // Pointer to STARTUPINFO struct [in]
        &jpi                 // Pointer to PROCESS_INFORMATION struct [out]
        )
       ) {
        sprintf(buf, "Starting service failed. Could not execute %s", cmdLine);
        LogError(buf);
        free(buf);
        return 1;
    }
    free(buf);

    // Wait for child to exit
    WaitForSingleObject (jpi.hProcess, INFINITE);

    // Get exit code from child process
    if (!GetExitCodeProcess(jpi.hProcess, &exitCode)) {
        LogError(TEXT("Unable to get child's exit code."));
        exitCode = 1;
    }

    // Close process and thread handles
    CloseHandle(jpi.hProcess);
    CloseHandle(jpi.hThread);

    /*
    sprintf(buf, "Application exited with code %d\n", exitCode);
    LogInformation(buf);
    */

    return exitCode;
}

void parseArgs (int argc, char *argv[]) {

    argv++; argc--;
    while (argc > 0) {
        if (strcmp(*argv, "-console") == 0) {
            bConsole = TRUE;
        } else if (strcmp(*argv, "-verbose") == 0) {
            bVerbose = TRUE;
        } else if (strcmp(*argv, "-managed") == 0) {
            bManaged = TRUE;            
        } else if (strcmp(*argv, "-javahome") == 0) {
            argv++; argc--;
            if (argc > 0) {
                /* use it to set jrehome */
                strncpy(jrehome, *argv, sizeof(jrehome) - 32);
                if (!_stricmp((jrehome+(strlen(*argv)-4)),"\\jre") == 0) {
                   /* doesn't end in \jre, so append \jre on the end */     
                   strcat(jrehome, "\\jre");
                }
            }
        } else if (strcmp(*argv, "-jrehome") == 0) {
            argv++; argc--;
            if (argc > 0) {
                strncpy(jrehome, *argv, sizeof(jrehome));
            }
        } else if (strcmp(*argv, "-varhome") == 0) {
            argv++; argc--;
            if (argc > 0) {
                strncpy(jmqvarhome, *argv, sizeof(jmqvarhome));
            }
        } else if (strcmp(*argv, "-vmargs") == 0) {
            argv++; argc--;
            if (argc > 0) {
                jvm_argc = stringToArgv(*argv, jvm_argc, jvm_argv,  32);
            }
        } else {
            // We don't recognize the option, pass it on to application
            application_argv[application_argc] = _strdup(*argv);
            application_argc++;
        }

        argv++; argc--;
    }

    if (!bConsole) {
        // We are running as an NT service. Pass info to broker.
        application_argv[application_argc] = _strdup("-ntservice");
        application_argc++;
    }

}

/*
 * Convert a string to argc/argv format. If the passed argc is
 * greater than 0 then the arguments are appended starting at the
 * location indexed by argc.
 *
 * Returns the new argc.
 */
int stringToArgv(char *string, int argc, char *argv[], int maxn) {

    int  n;
    char **p;
    char *s;

    for (n = argc, p = argv, s = string; *s != '\0' && n < maxn; n++, p++) {
        argv[n] = s;
        while (*s != ' ' && *s != '\0') {
            s++;
        }

        while (*s == ' ') {
            *s = '\0';
            s++;
        }
    }
    argv[n] = NULL;

    return(n);
}

void dumpArgv(char *m, int argc, char *argv[]) {

    int  n;
    if (m != NULL) {
        printf("%s\n", m);
    }

    for (n = 0; n < argc; n++) {
        printf("argv[%d]=%s\n", n, argv[n]);
    }
}



/*
 * Set 'jrehome', 'jvm_argc/argv' and 'application_argc/argv' to
 * values stored in the registry at install time (if any)
 */
void getParamsFromRegistry() {

    int n = 0;
    char tmp_buf[1024];

    n = sizeof(tmp_buf);
    tmp_buf[0] = '\0';
    getStringFromRegistry(tmp_buf, &n, JREHOME_KEY);
    if (tmp_buf[0] != '\0') {
        strncpy(jrehome, tmp_buf, sizeof(jrehome) - 32);
    }

    n = sizeof(tmp_buf);
    tmp_buf[0] = '\0';
    getStringFromRegistry(tmp_buf, &n, JVMARGS_KEY);
    if (tmp_buf[0] != '\0') {
        jvm_argc = stringToArgv(_strdup(tmp_buf), jvm_argc, jvm_argv, 32);
    }

    n = sizeof(tmp_buf);
    tmp_buf[0] = '\0';
    getStringFromRegistry(tmp_buf, &n, SERVICEARGS_KEY);
    if (tmp_buf[0] != '\0') {
        application_argc = stringToArgv(_strdup(tmp_buf), application_argc, application_argv, 32);
    }
}

/*
 * This is a gross, gross hack to try to figure out what port the
 * broker (in particular the broker portmapper) is running on.
 *
 * Parameters:
 *    argc, *argv[]	The command line arguments that will be passed to broker
 *    jmqhome		The value of $IMQ_HOME
 *    jmqvarhome        The value of $IMQ_VARHOME
 *
 * Returns:
 *    NULL if no port property value was found
 *    Or a string representation of the port number
 */
char *FindBrokerPort(
	int argc, char *argv[],
	char *jmqvarhome,
	char *jmqlibhome
	) {

    const char *port_prop      = "-Dimq.portmapper.port=";
    const char *port_prop_name = "imq.portmapper.port=";
    char buf[MAX_PATH + 1];
    char *name = NULL;
    char *s = NULL;
    int  rcode = 0;

    /* First check the application command line arguments for
     * -port  or -Dimq.portmapper.port=
     */
    while (argc > 0) {
        if (strcmp(*argv, "-port") == 0) {
            argv++; argc--;
	    if (argc > 0) {
		return _strdup(*argv);
	    }
         } else if (strncmp(*argv, port_prop, strlen(port_prop)) == 0) {
	    /* Found -Dimq.portmapper.port=" */
	    return GetNumericPropertyValue(*argv);
	 } else if (strcmp(*argv, "-name") == 0) {
	    /* We need instance name to check property files */
	    argv++; argc--;
	    if (argc > 0) {
		name = _strdup(*argv);
            }
	 }
	argv++; argc--;
    }

    if (name == NULL) {
	name = _strdup("imqbroker");
    }

    /* Next check instance property file if it exists */
    sprintf(buf, "%s\\instances\\%s\\props\\config.properties",
	jmqvarhome, name);
    free(name); name = NULL;
    if (_access(buf, 00) == 0) {
        if ((s = FindLine(buf, port_prop_name)) != NULL) {
	    return GetNumericPropertyValue(s);
        }
    }

    /* Finally check default configuration */
    sprintf(buf, "%s\\props\\broker\\default.properties",
	jmqlibhome);
    if ((s = FindLine(buf, port_prop_name)) != NULL) {
	return GetNumericPropertyValue(s);
    }

    return NULL;
}

/*
 * Takes something of the form:
 * blah-blah-blah=12345
 * and returns the "12345".
 * Caller is responsible for freeing returned buffer.
 */
char *GetNumericPropertyValue(const char *s)  {

    char *result;
    char *p;

    /* Skip to '=' (or ':') sign */
    while (*s != '\0' && *s != '=' && *s != ':') {
	s++;
    }

    if (*s == '\0') return NULL;

    /* Make s point to first byte in value */
    s++;
    result = _strdup(s);
    p = result;

    /* Find end of value */
    while (*p >= '0' && *p <= '9') {
	p++;
    }

    if (p == result) {
	free(result);
	return NULL;
    }

    *p = '\0';

    return result;
}


/*
 * Return the last line from a file that starts with the specified string.
 * We skip leading whitespace before doing comparision.
 */
char *FindLine(const char *file, const char *s) {

    char buf[MAX_PATH];
    FILE *fp;
    char *p;
    char *match_p = NULL;

    if ((fp = fopen(file, "r")) == NULL) {
        sprintf(buf, "MQ NT service couldn't search file %s", file);
	perror(buf);
	return NULL;
    }

    memset(buf, '\0', sizeof(buf));

    while (fgets(buf, sizeof(buf), fp) != NULL) {
	/* Skip leading white space */
	p = buf;
	while (*p != '\0' && (*p == ' ' || *p == '\t')) {
	    p++;
        }
	if (strncmp(s, p, strlen(s)) == 0) {
	    /* We have a match */
	    if (match_p != NULL) free(match_p);
	    match_p = _strdup(p);
        }
    }

    fclose(fp);
    return match_p;
}

/*
 * Check if an argument appears in the passed argument list. We
 * basically do this by checking if any string in argv starts with
 * the passed string.
 */
BOOL argInArgv(const char *s, const char **argv, int argc) {

    int n;
    for (n = 0; n < argc; n++) {
        if (strncmp(s, argv[n], sizeof(s)) == 0) {
            return 1;
        }
    }
    return 0;
}

/*
 * Check if the hotspot server VM is installed in the JRE we are running
 * against
 */
BOOL hotspotServerAvailable(const char *jhome) {

    char buf[MAX_PATH];

    sprintf(buf, "%s\\bin\\server", jhome);
    if (_access(buf, 00) < 0) {
        return 0;
    } else {
        return 1;
    }
}





© 2015 - 2025 Weber Informatics LLC | Privacy Policy