You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
tdelibs/tdehtml/java/kjavaprocess.cpp

398 lines
10 KiB

/* This file is part of the KDE project
*
* Copyright (C) 2000 Richard Moore <rich@kde.org>
* 2000 Wynn Wilkes <wynnw@caldera.com>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "kjavaprocess.h"
#include <kdebug.h>
#include <tdeio/kprotocolmanager.h>
#include <tqtextstream.h>
#include <tqmap.h>
#include <config.h>
#include <unistd.h>
#include <tqptrlist.h>
class KJavaProcessPrivate
{
friend class KJavaProcess;
private:
TQString jvmPath;
TQString classPath;
TQString mainClass;
TQString extraArgs;
TQString classArgs;
TQPtrList<TQByteArray> BufferList;
TQMap<TQString, TQString> systemProps;
bool processKilled;
};
KJavaProcess::KJavaProcess() : TDEProcess()
{
d = new KJavaProcessPrivate;
d->BufferList.setAutoDelete( true );
d->processKilled = false;
javaProcess = this; //new TDEProcess();
connect( javaProcess, TQT_SIGNAL( wroteStdin( TDEProcess * ) ),
this, TQT_SLOT( slotWroteData() ) );
connect( javaProcess, TQT_SIGNAL( receivedStdout( int, int& ) ),
this, TQT_SLOT( slotReceivedData(int, int&) ) );
connect( javaProcess, TQT_SIGNAL( processExited (TDEProcess *) ),
this, TQT_SLOT( slotExited (TDEProcess *) ) );
d->jvmPath = "java";
d->mainClass = "-help";
}
KJavaProcess::~KJavaProcess()
{
if ( isRunning() )
{
kdDebug(6100) << "stopping java process" << endl;
stopJava();
}
//delete javaProcess;
delete d;
}
bool KJavaProcess::isRunning()
{
return javaProcess->isRunning();
}
bool KJavaProcess::startJava()
{
return invokeJVM();
}
void KJavaProcess::stopJava()
{
killJVM();
}
void KJavaProcess::setJVMPath( const TQString& path )
{
d->jvmPath = path;
}
void KJavaProcess::setClasspath( const TQString& classpath )
{
d->classPath = classpath;
}
void KJavaProcess::setSystemProperty( const TQString& name,
const TQString& value )
{
d->systemProps.insert( name, value );
}
void KJavaProcess::setMainClass( const TQString& className )
{
d->mainClass = className;
}
void KJavaProcess::setExtraArgs( const TQString& args )
{
d->extraArgs = args;
}
void KJavaProcess::setClassArgs( const TQString& args )
{
d->classArgs = args;
}
//Private Utility Functions used by the two send() methods
TQByteArray* KJavaProcess::addArgs( char cmd_code, const TQStringList& args )
{
//the buffer to store stuff, etc.
TQByteArray* const buff = new TQByteArray();
TQTextOStream output( *buff );
const char sep = 0;
//make space for the command size: 8 characters...
const TQCString space( " " );
output << space;
//write command code
output << cmd_code;
//store the arguments...
if( args.isEmpty() )
{
output << sep;
}
else
{
TQStringList::ConstIterator it = args.begin();
const TQStringList::ConstIterator itEnd = args.end();
for( ; it != itEnd; ++it )
{
if( !(*it).isEmpty() )
{
output << (*it).local8Bit();
}
output << sep;
}
}
return buff;
}
void KJavaProcess::storeSize( TQByteArray* buff )
{
const int size = buff->size() - 8; //subtract out the length of the size_str
const TQString size_str = TQString("%1").arg( size, 8 );
kdDebug(6100) << "KJavaProcess::storeSize, size = " << size_str << endl;
const char* size_ptr = size_str.latin1();
for( int i = 0; i < 8; ++i )
buff->at(i) = size_ptr[i];
}
void KJavaProcess::sendBuffer( TQByteArray* buff )
{
d->BufferList.append( buff );
if( d->BufferList.count() == 1)
{
popBuffer();
}
}
void KJavaProcess::send( char cmd_code, const TQStringList& args )
{
if( isRunning() )
{
TQByteArray* const buff = addArgs( cmd_code, args );
storeSize( buff );
kdDebug(6100) << "<KJavaProcess::send " << (int)cmd_code << endl;
sendBuffer( buff );
}
}
void KJavaProcess::send( char cmd_code, const TQStringList& args,
const TQByteArray& data )
{
if( isRunning() )
{
kdDebug(6100) << "KJavaProcess::send, qbytearray is size = " << data.size() << endl;
TQByteArray* const buff = addArgs( cmd_code, args );
const int cur_size = buff->size();
const int data_size = data.size();
buff->resize( cur_size + data_size );
memcpy( buff->data() + cur_size, data.data(), data_size );
storeSize( buff );
sendBuffer( buff );
}
}
void KJavaProcess::popBuffer()
{
TQByteArray* const buf = d->BufferList.first();
if( buf )
{
// DEBUG stuff...
// kdDebug(6100) << "Sending buffer to java, buffer = >>";
// for( unsigned int i = 0; i < buf->size(); i++ )
// {
// if( buf->at(i) == (char)0 )
// kdDebug(6100) << "<SEP>";
// else if( buf->at(i) > 0 && buf->at(i) < 10 )
// kdDebug(6100) << "<CMD " << (int) buf->at(i) << ">";
// else
// kdDebug(6100) << buf->at(i);
// }
// kdDebug(6100) << "<<" << endl;
//write the data
if ( !javaProcess->writeStdin( buf->data(),
buf->size() ) )
{
kdError(6100) << "Could not write command" << endl;
}
}
}
void KJavaProcess::slotWroteData( )
{
//do this here- we can't free the data until we know it went through
d->BufferList.removeFirst(); //this should delete it since we setAutoDelete(true)
kdDebug(6100) << "slotWroteData " << d->BufferList.count() << endl;
if ( !d->BufferList.isEmpty() )
{
popBuffer();
}
}
bool KJavaProcess::invokeJVM()
{
*javaProcess << d->jvmPath;
if( !d->classPath.isEmpty() )
{
*javaProcess << "-classpath";
*javaProcess << d->classPath;
}
//set the system properties, iterate through the qmap of system properties
TQMap<TQString,TQString>::ConstIterator it = d->systemProps.begin();
const TQMap<TQString,TQString>::ConstIterator itEnd = d->systemProps.end();
for( ; it != itEnd; ++it )
{
TQString currarg;
if( !it.key().isEmpty() )
{
currarg = "-D" + it.key();
if( !it.data().isEmpty() )
currarg += "=" + it.data();
}
if( !currarg.isEmpty() )
*javaProcess << currarg;
}
//load the extra user-defined arguments
if( !d->extraArgs.isEmpty() )
{
// BUG HERE: if an argument contains space (-Dname="My name")
// this parsing will fail. Need more sophisticated parsing -- use KShell?
const TQStringList args = TQStringList::split( " ", d->extraArgs );
TQStringList::ConstIterator it = args.begin();
const TQStringList::ConstIterator itEnd = args.end();
for ( ; it != itEnd; ++it )
*javaProcess << *it;
}
*javaProcess << d->mainClass;
if ( !d->classArgs.isNull() )
*javaProcess << d->classArgs;
kdDebug(6100) << "Invoking JVM now...with arguments = " << endl;
TQString argStr;
TQTextOStream stream( &argStr );
const TQValueList<TQCString> args = javaProcess->args();
tqCopy( args.begin(), args.end(), TQTextOStreamIterator<TQCString>( stream, " " ) );
kdDebug(6100) << argStr << endl;
TDEProcess::Communication flags = (TDEProcess::Communication)
(TDEProcess::Stdin | TDEProcess::Stdout |
TDEProcess::NoRead);
const bool rval = javaProcess->start( TDEProcess::NotifyOnExit, flags );
if( rval )
javaProcess->resume(); //start processing stdout on the java process
else
killJVM();
return rval;
}
void KJavaProcess::killJVM()
{
d->processKilled = true;
disconnect( javaProcess, TQT_SIGNAL( receivedStdout( int, int& ) ),
this, TQT_SLOT( slotReceivedData(int, int&) ) );
javaProcess->kill();
}
void KJavaProcess::flushBuffers()
{
while ( !d->BufferList.isEmpty() ) {
if (innot)
slotSendData(0);
else
d->BufferList.removeFirst(); //note: AutoDelete is true
}
}
/* In this method, read one command and send it to the d->appletServer
* then return, so we don't block the event handling
*/
void KJavaProcess::slotReceivedData( int fd, int& len )
{
//read out the length of the message,
//read the message and send it to the applet server
char length[9] = { 0 };
const int num_bytes = ::read( fd, length, 8 );
if( !num_bytes )
{
len = 0;
return;
}
if( num_bytes == -1 )
{
kdError(6100) << "could not read 8 characters for the message length!!!!" << endl;
len = 0;
return;
}
const TQString lengthstr( length );
bool ok;
const int num_len = lengthstr.toInt( &ok );
if( !ok )
{
kdError(6100) << "could not parse length out of: " << lengthstr << endl;
len = num_bytes;
return;
}
//now parse out the rest of the message.
char* const msg = new char[num_len];
const int num_bytes_msg = ::read( fd, msg, num_len );
if( num_bytes_msg == -1 || num_bytes_msg != num_len )
{
kdError(6100) << "could not read the msg, num_bytes_msg = " << num_bytes_msg << endl;
delete[] msg;
len = num_bytes;
return;
}
TQByteArray qb;
emit received( qb.duplicate( msg, num_len ) );
delete[] msg;
len = num_bytes + num_bytes_msg;
}
void KJavaProcess::slotExited( TDEProcess *process )
{
if (process == javaProcess) {
int status = -1;
if (!d->processKilled) {
status = javaProcess->exitStatus();
}
kdDebug(6100) << "jvm exited with status " << status << endl;
emit exited(status);
}
}
#include "kjavaprocess.moc"