File indexing completed on 2025-10-19 05:26:54

0001 /*
0002  * perldoc.cpp
0003  *
0004  * Borrowed from KDevelop's perldoc ioslave, and improved.
0005  * Copyright 2007 Michael Pyne <michael.pyne@kdemail.net>
0006  * Copyright 2017 Luigi Toscano <luigi.toscano@tiscali.it>
0007  *
0008  * No copyright header was present in KDevelop's perldoc io slave source
0009  * code.  However, source code revision history indicates it was written and
0010  * imported by Bernd Gehrmann <bernd@mail.berlios.de>.  KDevelop is distributed
0011  * under the terms of the GNU General Public License v2.  Therefore, so is
0012  * this software.
0013  *
0014  * All changes made by Michael Pyne are licensed under the terms of the GNU
0015  * GPL version 2 or (at your option) any later version.
0016  *
0017  * Uses the Pod::HtmlEasy Perl module by M. P. Graciliano and
0018  * Geoffrey Leach.  It is distributed under the same terms as Perl.
0019  * See pod2html.pl for more information.
0020  */
0021 
0022 #include "perldoc.h"
0023 
0024 // KIO worker
0025 #include "version.h"
0026 // KF
0027 #include <KAboutData>
0028 #include <KLocalizedString>
0029 // Qt
0030 #include <QByteArray>
0031 #include <QCoreApplication>
0032 #include <QProcess>
0033 #include <QStandardPaths>
0034 #include <QStringList>
0035 #include <QUrl>
0036 
0037 class KIOPluginForMetaData : public QObject
0038 {
0039     Q_OBJECT
0040     Q_PLUGIN_METADATA(IID "org.kde.kio.worker.perldoc" FILE "perldoc.json")
0041 };
0042 
0043 PerldocProtocol::PerldocProtocol(const QByteArray &pool, const QByteArray &app)
0044     : KIO::WorkerBase("perldoc", pool, app)
0045 {
0046     m_pod2htmlPath = QStandardPaths::locate(QStandardPaths::GenericDataLocation, QStringLiteral("kio_perldoc/pod2html.pl"));
0047     m_cssLocation = QStandardPaths::locate(QStandardPaths::GenericDataLocation, QStringLiteral("kio_docfilter/kio_docfilter.css"));
0048 }
0049 
0050 PerldocProtocol::~PerldocProtocol()
0051 {
0052 }
0053 
0054 KIO::WorkerResult PerldocProtocol::get(const QUrl &url)
0055 {
0056     const QStringList l = url.path().split(QLatin1Char('/'), Qt::SkipEmptyParts);
0057 
0058     // Check for perldoc://foo
0059     if(!url.host().isEmpty()) {
0060         QUrl newURL(url);
0061 
0062         newURL.setPath(url.host() + url.path());
0063         newURL.setHost(QString());
0064 
0065         redirection(newURL);
0066         return KIO::WorkerResult::pass();
0067     }
0068 
0069     mimeType(QStringLiteral("text/html"));
0070 
0071     if(l[0].isEmpty() || url.path() == QLatin1String("/")) {
0072         QByteArray output = i18n("<html><head><title>No page requested</title>"
0073             "<body>No page was requested.  You can search for:<ul><li>functions "
0074             "using perldoc:/functions/foo</li>\n\n"
0075             "<li>faq entries using perldoc:/faq/search_terms</li>"
0076             "<li>All other perldoc documents using the name of the document, like"
0077             "<ul><li><a href='perldoc:/perlreftut'>perldoc:/perlreftut</a></li>"
0078             "<li>or <a href='perldoc:/Net::HTTP'>perldoc:/Net::HTTP</a></li></ul>"
0079             "</li></ul>\n\n</body></html>\n"
0080         ).toLocal8Bit();
0081 
0082         data(output);
0083         return KIO::WorkerResult::pass();
0084     }
0085 
0086     if(l[0] != QLatin1String("functions") && l[0] != QLatin1String("faq")) {
0087         // See if it exists first.
0088         if(!topicExists(l[0])) {
0089             // Failed
0090             QByteArray errstr =
0091                 i18n("<html><head><title>No documentation for %1</title><body>"
0092                 "Unable to find documentation for <b>%2</b></body></html>\n",
0093                 l[0], l[0]).toLocal8Bit();
0094 
0095             data(errstr);
0096             return KIO::WorkerResult::pass();
0097         }
0098     }
0099 
0100     QStringList pod2htmlArguments;
0101     if (l[0] == QLatin1String("functions")) {
0102         pod2htmlArguments = QStringList{QStringLiteral("-f"), l[1]};
0103     } else if (l[0] == QLatin1String("faq")) {
0104         pod2htmlArguments = QStringList{QStringLiteral("-q"), l[1]};
0105     } else if (!l[0].isEmpty()) {
0106         pod2htmlArguments = QStringList{l[0]};
0107     }
0108 
0109     QProcess pod2htmlProcess;
0110 
0111     QProcessEnvironment env = QProcessEnvironment::systemEnvironment();
0112     env.insert(QStringLiteral("KIO_PERLDOC_VERSION"), QStringLiteral(KIO_PERLDOC_VERSION_STRING));
0113     env.insert(QStringLiteral("KIO_PERLDOC_CSSLOCATION"), m_cssLocation);
0114     pod2htmlProcess.setProcessEnvironment(env);
0115 
0116     pod2htmlProcess.start(m_pod2htmlPath, pod2htmlArguments);
0117     if (!pod2htmlProcess.waitForFinished()) {
0118         return failAndQuit();
0119     }
0120 
0121     if ((pod2htmlProcess.exitStatus() != QProcess::NormalExit) ||
0122         (pod2htmlProcess.exitCode() < 0)) {
0123         return KIO::WorkerResult::fail(KIO::ERR_CANNOT_LAUNCH_PROCESS, m_pod2htmlPath);
0124     }
0125 
0126     data(pod2htmlProcess.readAllStandardOutput());
0127     return KIO::WorkerResult::pass();
0128 }
0129 
0130 KIO::WorkerResult PerldocProtocol::failAndQuit()
0131 {
0132     data(errorMessage());
0133     return KIO::WorkerResult::pass();
0134 }
0135 
0136 QByteArray PerldocProtocol::errorMessage()
0137 {
0138     return QByteArray("<html><body bgcolor=\"#FFFFFF\">" +
0139            i18n("Error in perldoc").toLocal8Bit() +
0140            "</body></html>");
0141 }
0142 
0143 KIO::WorkerResult PerldocProtocol::stat(const QUrl &/*url*/)
0144 {
0145     KIO::UDSEntry uds_entry;
0146     uds_entry.fastInsert(KIO::UDSEntry::UDS_FILE_TYPE, S_IFREG | S_IRWXU | S_IRWXG | S_IRWXO);
0147 
0148     statEntry(uds_entry);
0149     return KIO::WorkerResult::pass();
0150 }
0151 
0152 KIO::WorkerResult PerldocProtocol::listDir(const QUrl &url)
0153 {
0154     return KIO::WorkerResult::fail( KIO::ERR_CANNOT_ENTER_DIRECTORY, url.path() );
0155 }
0156 
0157 bool PerldocProtocol::topicExists(const QString &topic)
0158 {
0159     // Run perldoc in query mode to see if the given manpage exists.
0160     QProcess perldocProcess;
0161     perldocProcess.start(QStringLiteral("perldoc"), QStringList{QStringLiteral("-l"), topic});
0162     if (!perldocProcess.waitForFinished()) {
0163         return false;
0164     }
0165 
0166     if ((perldocProcess.exitStatus() != QProcess::NormalExit) ||
0167         (perldocProcess.exitCode() < 0)) {
0168         return false;
0169     }
0170 
0171     return true;
0172 }
0173 
0174 extern "C" {
0175 
0176     int Q_DECL_EXPORT kdemain(int argc, char **argv)
0177     {
0178         QCoreApplication app(argc, argv);
0179 
0180         KAboutData aboutData(
0181             QStringLiteral("kio_perldoc"),
0182             i18n("perldoc KIO worker"),
0183             QStringLiteral(KIO_PERLDOC_VERSION_STRING),
0184             i18n("KIO worker to provide access to perldoc documentation"),
0185             KAboutLicense::GPL_V2,
0186             i18n("Copyright 2007, 2008 Michael Pyne"),
0187             i18n("Uses Pod::HtmlEasy by M. P. Graciliano and Geoffrey Leach")
0188         );
0189 
0190         aboutData.addAuthor(i18n("Michael Pyne"), i18n("Maintainer, port to KDE 4"),
0191             QStringLiteral("michael.pyne@kdemail.net"), QStringLiteral("http://purinchu.net/wp/"));
0192         aboutData.addAuthor(i18n("Bernd Gehrmann"), i18n("Initial implementation"));
0193         aboutData.addCredit(i18n("M. P. Graciliano"), i18n("Pod::HtmlEasy"));
0194         aboutData.addCredit(i18n("Geoffrey Leach"), i18n("Pod::HtmlEasy current maintainer"));
0195         aboutData.setTranslator(i18nc("NAME OF TRANSLATORS", "Your names"),
0196             i18nc("EMAIL OF TRANSLATORS", "Your emails"));
0197 
0198         app.setOrganizationDomain(QStringLiteral("kde.org"));
0199         app.setOrganizationName(QStringLiteral("KDE"));
0200 
0201         KAboutData::setApplicationData(aboutData);
0202 
0203         if (argc != 4) {
0204             fprintf(stderr, "Usage: kio_perldoc protocol domain-socket1 domain-socket2\n");
0205             exit(5);
0206         }
0207 
0208         PerldocProtocol worker(argv[2], argv[3]);
0209         worker.dispatchLoop();
0210 
0211         return 0;
0212     }
0213 }
0214 
0215 #include "perldoc.moc"
0216 
0217 // vim: set et sw=4 ts=8: