kde-workspace/kioslave/cgi/cgi.cpp
Ivailo Monev 260c6f3dd8 generic: misc cleanups
Signed-off-by: Ivailo Monev <xakepa10@gmail.com>
2016-04-28 22:27:02 +00:00

227 lines
5.6 KiB
C++

/*
Copyright (C) 2002 Cornelius Schumacher <schumacher@kde.org>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program 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 General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
*/
#include "cgi.h"
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h> // getpid()
#include <kdebug.h>
#include <kshell.h>
#include <kstandarddirs.h>
#include <kcomponentdata.h>
#include <klocale.h>
#include <kconfig.h>
#include <kconfiggroup.h>
#include <kdemacros.h>
#include <QFile>
using namespace KIO;
CgiProtocol::CgiProtocol( const QByteArray &pool, const QByteArray &app )
: SlaveBase( "cgi", pool, app )
{
kDebug(7124) << "CgiProtocol::CgiProtocol";
KConfig _cfg( "kcmcgirc" );
KConfigGroup cfg(&_cfg, "General" );
mCgiPaths = cfg.readEntry( "Paths" , QStringList() );
}
CgiProtocol::~CgiProtocol()
{
kDebug(7124) << "CgiProtocol::~CgiProtocol";
}
void CgiProtocol::get( const KUrl& url )
{
kDebug(7124) << "CgiProtocol::get()";
kDebug(7124) << " URL: " << url.url();
#if 0
kDebug(7124) << " Path: " << url.path();
kDebug(7124) << " Query: " << url.query();
kDebug(7124) << " Protocol: " << url.protocol();
kDebug(7124) << " Filename: " << url.filename();
#endif
QByteArray protocol = "SERVER_PROTOCOL=HTTP";
putenv( protocol.data() );
QByteArray requestMethod = "REQUEST_METHOD=GET";
putenv( requestMethod.data() );
QByteArray query = url.query().mid( 1 ).toLocal8Bit();
query.prepend( "QUERY_STRING=" );
putenv( query.data() );
QString path = url.path();
QString file;
int pos = path.lastIndexOf('/');
if ( pos >= 0 ) file = path.mid( pos + 1 );
else file = path;
QString cmd;
bool stripHeader = false;
bool forwardFile = true;
QStringList::ConstIterator it;
for( it = mCgiPaths.constBegin(); it != mCgiPaths.constEnd(); ++it ) {
cmd = *it;
if ( !(*it).endsWith('/') )
cmd += '/';
cmd += file;
if ( KGlobal::dirs()->exists( cmd ) ) {
forwardFile = false;
stripHeader = true;
break;
}
}
FILE *fd;
if ( forwardFile ) {
kDebug(7124) << "Forwarding to '" << path << "'";
QByteArray filepath = QFile::encodeName( path );
fd = fopen( filepath.data(), "r" );
if ( !fd ) {
kDebug(7124) << "Error opening '" << filepath << "'";
error(KIO::ERR_CANNOT_OPEN_FOR_READING, path);
return;
}
} else {
kDebug(7124) << "Cmd: " << cmd;
fd = popen( QFile::encodeName(KShell::quoteArg( cmd )).data(), "r" );
if ( !fd ) {
kDebug(7124) << "Error running '" << cmd << "'";
error( KIO::ERR_CANNOT_OPEN_FOR_READING, cmd );
return;
}
}
char buffer[ 4090 ];
while ( !feof( fd ) )
{
int n = fread( buffer, 1, 2048, fd );
if ( n == -1 )
{
// ERROR
if ( forwardFile ) {
fclose( fd );
} else {
pclose( fd );
}
return;
}
buffer[n] = 0;
if ( stripHeader ) {
QByteArray output = buffer; // this assumes buffer is text and not binary
int colon = output.indexOf( ':' );
int newline = output.indexOf( '\n' );
int semicolon = output.lastIndexOf( ';', newline );
int end;
if ( semicolon < 0 ) end = newline;
else end = semicolon;
#if 0
kDebug(7124) << " colon: " << colon;
kDebug(7124) << " newline: " << newline;
kDebug(7124) << " semicolon: " << semicolon;
kDebug(7124) << " end: " << end;
#endif
// Set the charset meta data if one exists (BR# 241364).
if (end == semicolon) {
const int assignOp = output.indexOf('=', semicolon + 1);
if (assignOp != -1) {
const QByteArray charset (output.mid(assignOp+1, newline- assignOp -1).trimmed());
kDebug(7124) << "Charset:" << charset;
setMetaData(QLatin1String("charset"), charset.toLower());
}
}
QByteArray contentType = output.mid( colon + 1, end - colon - 1 );
contentType = contentType.trimmed().toLower();
kDebug(7124) << "ContentType: '" << contentType << "'";
mimeType( contentType );
int start = output.indexOf( "\r\n\r\n" );
if ( start >= 0 ) start += 4;
else {
start = output.indexOf( "\n\n" );
if ( start >= 0 ) start += 2;
}
if ( start >= 0 ) output = output.mid( start );
stripHeader = false;
data( output );
} else {
data( QByteArray::fromRawData( buffer, n ) );
}
}
if ( forwardFile ) {
fclose( fd );
} else {
pclose( fd );
}
finished();
kDebug(7124) << "CgiProtocol::get - done";
}
extern "C" { int KDE_EXPORT kdemain( int argc, char **argv ); }
/*! The kdemain function generates an instance of the ioslave and starts its
* dispatch loop. */
int kdemain( int argc, char **argv )
{
KComponentData componentData( "kio_cgi" );
kDebug(7124) << "kio_cgi starting " << getpid();
if (argc != 4)
{
fprintf(stderr, "Usage: kio_cgi protocol domain-socket1 domain-socket2\n");
exit(-1);
}
CgiProtocol slave( argv[2], argv[3] );
slave.dispatchLoop();
return 0;
}