diff options
Diffstat (limited to 'kio_gopher.cpp')
-rw-r--r-- | kio_gopher.cpp | 352 |
1 files changed, 352 insertions, 0 deletions
diff --git a/kio_gopher.cpp b/kio_gopher.cpp new file mode 100644 index 0000000..baf5457 --- /dev/null +++ b/kio_gopher.cpp @@ -0,0 +1,352 @@ +/**************************************************************************** + * Copyright (C) 2003-2008 by Albert Astals Cid * + * [email protected] * + * * + * 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. * + ****************************************************************************/ + +#include "kio_gopher.h" + +#include <kcodecs.h> +#include <kcomponentdata.h> +#include <kdebug.h> +#include <klocale.h> +#include <kmimetype.h> + +#include <qbuffer.h> +#include <qfile.h> + +using namespace KIO; + +extern "C" +{ + int KDE_EXPORT kdemain( int argc, char **argv ) + { + KComponentData instance( "kio_gopher" ); + + if (argc != 4) + { + fprintf(stderr, "Usage: kio_gopher protocol domain-socket1 domain-socket2\n"); + exit(-1); + } + + gopher slave(argv[2], argv[3]); + slave.dispatchLoop(); + return 0; + } +} + +/* gopher */ + +gopher::gopher(const QByteArray &pool_socket, const QByteArray &app_socket) : TCPSlaveBase("gopher", pool_socket, app_socket) +{ +} + +void gopher::get(const KUrl& url ) +{ + // gopher urls are + // gopher://<host>:<port>/<gopher-path> + // + // where <gopher-path> is one of + // + // <gophertype><selector> + // <gophertype><selector>%09<search> + // <gophertype><selector>%09<search>%09<gopher+_string> + int port; + QChar type; + QString path(url.path()); + QString query(url.query()); + + // determine the type + if (path != "/" && path != "") type = path[1]; + else type = '1'; + + // determine the port + if (url.port() > 0) port = url.port(); + else port = 70; + + // connect to the host + if (!connectToHost("gopher", url.host(), port)) return; + + setBlocking(true); + + if (type == '7' && query.isNull()) + { + disconnectFromHost(); + handleSearch(url.host(), path, port); + } + else + { + int i, bytes; + char aux[10240]; + QBuffer received; + received.open(QIODevice::WriteOnly); + + infoMessage(i18n("Connecting to %1...", url.host())); + infoMessage(i18n("%1 contacted. Retrieving data...", url.host())); + bytes = 0; + + // send the selector + path.remove(0, 2); + write(path.toLatin1(), path.length()); + write(query.toLatin1(), query.length()); + write("\r\n", 2); + + // read the data + while((i = read(aux, 10240)) > 0) + { + bytes += i; + received.write(aux, i); + processedSize(bytes); + infoMessage(i18n("Retrieved %1 bytes from %2...", bytes, url.host())); + } + + if (type == '1' || type =='7') processDirectory(new QByteArray(received.buffer().data(), bytes + 1), url.host(), url.path()); + else + { + KMimeType::Ptr result = KMimeType::findByContent(received.buffer()); + mimeType(result->name()); + data(received.buffer()); + } + disconnectFromHost(); + } + finished(); +} + +void gopher::processDirectory(QByteArray *received, const QString &host, const QString &path) +{ + int i, remove; + QString pathToShow; + QByteArray show; + QByteArray info; + if (path == "/" || path == "/1") pathToShow = ""; + else pathToShow = path; + mimeType("text/html"); + show.append("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.1//EN\" \"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd\">\n"); + show.append("<html xmlns=\"http://www.w3.org/1999/xhtml\">\n"); + show.append("\t<head>\n"); + show.append("\t\t<title>"); + show.append(host.toUtf8()); + show.append(pathToShow.toUtf8()); + show.append("</title>\n"); + show.append("\t\t<meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\" />\n"); + show.append("\t\t<style type=\"text/css\">\n"); + show.append("\t\t\t.info{ font-size : small; display : block; font-family : monospace; white-space : pre; margin-left : 18px; }\n"); + show.append("\t\t</style>\n"); + show.append("\t</head>\n"); + show.append("\t<body>\n"); + show.append("\t\t<h1>"); + show.append(host.toUtf8()); + show.append(pathToShow.toUtf8()); + show.append("</h1>\n"); + findLine(received, &i, &remove); + while(i != -1) + { + processDirectoryLine(received -> left(i), show, info); + received -> remove(0, i + remove); + findLine(received, &i, &remove); + } + show.append("\t</body>\n"); + show.append("</html>\n"); + data(show); + delete received; +} + +void gopher::processDirectoryLine(const QByteArray &d, QByteArray &show, QByteArray &info) +{ + // gopher <type><display><tab><selector><tab><server><tab><port><\r><\n> + // gopher+ <type><display><tab><selector><tab><server><tab><port><tab><things><\r><\n> + int i; + QByteArray type, name, url, server, port; + QByteArray data = d; + + type = data.left(1); + data.remove(0, 1); + + i = data.indexOf("\t"); + name = data.left(i); + data.remove(0, i + 1); + + i = data.indexOf("\t"); + url = data.left(i); + data.remove(0, i + 1); + + i = data.indexOf("\t"); + server = data.left(i); + data.remove(0, i + 1); + + port = parsePort(&data); + + if (type == "i") + { + if (!info.isEmpty()) + info.append("\n"); + info.append(name); + } + else + { + if (!info.isEmpty()) + { + show.append("\t\t<div class=\"info\">"); + show.append(info); + show.append("</div>\n"); + info = ""; + } + // it's the final line, ignore it + if (type == ".") return; + // those are the standard gopher types defined in the rfc + // 0 Item is a file + // 1 Item is a directory + // 2 Item is a CSO phone-book server + // 3 Error + // 4 Item is a BinHexed Macintosh file. + // 5 Item is DOS binary archive of some sort. Client must read until the TCP connection closes. Beware. + // 6 Item is a UNIX uuencoded file. + // 7 Item is an Index-Search server. + // 8 Item points to a text-based telnet session. + // 9 Item is a binary file! Client must read until the TCP connection closes. Beware. + // + Item is a redundant server + // T Item points to a text-based tn3270 session. + // g Item is a GIF format graphics file. + // I Item is some kind of image file. Client decides how to display. + show.append("\t\t\t<div>"); + // support the non-standard extension for URL to external sites + // in this case, url begins with 'URL:' + QByteArray finalUrl; + QByteArray iconUrl; + if (url.startsWith("URL:")) + { + finalUrl = url.mid(4); + iconUrl = finalUrl; + } + else + { + finalUrl = "gopher://" + server; + if (port != "70") + { + finalUrl.append(":"); + finalUrl.append(port); + } + finalUrl.append('/' + type + url); + iconUrl = url; + } + show.append("\t\t\t\t<a href=\""); + show.append(finalUrl); + show.append("\">"); + addIcon(type, iconUrl, show); + show.append(name); + show.append("</a><br />\n"); + show.append("\t\t\t</div>"); + } +} + +QByteArray gopher::parsePort(QByteArray *received) +{ + int i = 0; + QByteArray port; + bool found = false; + QChar c; + while (!found && i < received -> size()) + { + c = received -> at(i); + if (c.isDigit()) i++; + else found = true; + } + port = received -> left(i); + received -> remove(0, i); + return port; +} + +void gopher::findLine(QByteArray *received, int *i, int *remove) +{ + // it's not in the rfc but most servers don't follow the spec + // find lines ending only in \n and in \r\n + int aux, aux2; + aux = received -> indexOf("\r\n"); + aux2 = received -> indexOf("\n"); + + if (aux == -1) + { + *i = aux2; + *remove = 1; + } + else + { + if (aux2 < aux) + { + *remove = 1; + *i = aux2; + } + else + { + *remove = 2; + *i = aux; + } + } +} + +void gopher::handleSearch(const QString &host, const QString &path, int port) +{ + QByteArray show; + QString sPort; + if (port != 70) sPort = ':' + QString::number(port); + mimeType("text/html"); + show.append("<!DOCTYPE html PUBLIC \"-//W3C//DTD XHTML 1.1//EN\" \"http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd\">\n"); + show.append("<html xmlns=\"http://www.w3.org/1999/xhtml\">\n"); + show.append("\t<head>\n"); + show.append("\t\t<title>"); + show.append(host.toUtf8()); + show.append(path.toUtf8()); + show.append("</title>\n"); + show.append("\t\t<meta http-equiv=\"Content-Type\" content=\"text/html; charset=utf-8\" />\n"); + show.append("\t\t<script type=\"text/javascript\">\n"); + show.append("\t\t\tfunction search()\n"); + show.append("\t\t\t{\n"); + show.append("\t\t\t\tdocument.location = 'gopher://"); + show.append(host.toUtf8()); + show.append(sPort.toUtf8()); + show.append(path.toUtf8()); + show.append("?' + document.getElementById('what').value;\n"); + show.append("\t\t\t}\n"); + show.append("\t\t</script>\n"); + show.append("\t</head>\n"); + show.append("\t<body>\n"); + show.append("\t\t<h1>"); + show.append(host.toUtf8()); + show.append(path.toUtf8()); + show.append("</h1>\n"); + show.append("\t\t"); + show.append(i18n("Enter a search term:").toUtf8()); + show.append("<br />\n"); + show.append("\t\t<input id=\"what\" type=\"text\">\n"); + show.append("\t\t<input type=\"button\" value=\""); + show.append(i18nc("Text on a search button, like at a search engine", "Search").toUtf8()); + show.append("\" onClick=\"search()\">\n"); + show.append("\t</body>\n"); + show.append("</html>\n"); + data(show); +} + +void gopher::addIcon(const QString &type, const QByteArray &url, QByteArray &show) +{ + QString icon; + if (type == "1") icon = "inode-directory.png"; + else if (type == "3") icon = "dialog-error.png"; + else if (type == "7") icon = "system-search.png"; + else if (type == "g") icon = "image-gif.png"; + else if (type == "I") icon = "image-x-generic.png"; + else + { + KMimeType::Ptr mime = KMimeType::findByUrl(KUrl(url), 0, false, true); + icon = mime->iconName(); + } + QFile file(m_iconLoader.iconPath(icon, -16)); + file.open(QIODevice::ReadOnly); + QByteArray ba = file.readAll(); + show.append("<img width=\"16\" height=\"16\" src=\"data:image/png;base64,"); + show.append(KCodecs::base64Encode(ba)); + show.append("\" /> "); +} |