mirror of
https://bitbucket.org/smil3y/kde-extraapps.git
synced 2025-02-24 10:52:53 +00:00
273 lines
9.2 KiB
C++
273 lines
9.2 KiB
C++
/***************************************************************************
|
|
* Copyright (C) 2010-2013 by Daniel Nicoletti *
|
|
* dantti12@gmail.com *
|
|
* *
|
|
* 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; see the file COPYING. If not, write to *
|
|
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, *
|
|
* Boston, MA 02110-1301, USA. *
|
|
***************************************************************************/
|
|
|
|
#include "KIppRequest.h"
|
|
#include "KIppRequest_p.h"
|
|
|
|
#include <QStringBuilder>
|
|
|
|
#include <KDebug>
|
|
|
|
KIppRequest::KIppRequest() :
|
|
d_ptr(new KIppRequestPrivate)
|
|
{
|
|
}
|
|
|
|
KIppRequest::KIppRequest(const KIppRequest &other) :
|
|
d_ptr(new KIppRequestPrivate)
|
|
{
|
|
*this = other;
|
|
}
|
|
|
|
KIppRequest::KIppRequest(ipp_op_t operation, const char *resource, const QString &filename) :
|
|
d_ptr(new KIppRequestPrivate)
|
|
{
|
|
Q_D(KIppRequest);
|
|
|
|
d->operation = operation;
|
|
d->resource = resource;
|
|
d->filename = filename;
|
|
|
|
// send our user name on the request too
|
|
addString(IPP_TAG_OPERATION, IPP_TAG_NAME, KCUPS_REQUESTING_USER_NAME, cupsUser());
|
|
}
|
|
|
|
KIppRequest::~KIppRequest()
|
|
{
|
|
Q_D(KIppRequest);
|
|
delete d;
|
|
}
|
|
|
|
ipp_op_t KIppRequest::operation() const
|
|
{
|
|
Q_D(const KIppRequest);
|
|
return d->operation;
|
|
}
|
|
|
|
QString KIppRequest::resource() const
|
|
{
|
|
Q_D(const KIppRequest);
|
|
return d->resource;
|
|
}
|
|
|
|
QString KIppRequest::filename() const
|
|
{
|
|
Q_D(const KIppRequest);
|
|
return d->filename;
|
|
}
|
|
|
|
ipp_t *KIppRequest::sendIppRequest() const
|
|
{
|
|
Q_D(const KIppRequest);
|
|
|
|
ipp_t *request = ippNewRequest(d->operation);
|
|
|
|
d->addRawRequestsToIpp(request);
|
|
|
|
if (d->filename.isNull()) {
|
|
return cupsDoRequest(CUPS_HTTP_DEFAULT, request, d->resource.toUtf8());
|
|
} else {
|
|
return cupsDoFileRequest(CUPS_HTTP_DEFAULT, request, d->resource.toUtf8(), d->filename.toUtf8());
|
|
}
|
|
}
|
|
|
|
void KIppRequest::addString(ipp_tag_t group, ipp_tag_t valueTag, const QString &name, const QString &value)
|
|
{
|
|
Q_D(KIppRequest);
|
|
|
|
d->addRequest(group, valueTag, name.toUtf8(), value);
|
|
}
|
|
|
|
void KIppRequest::addStringList(ipp_tag_t group, ipp_tag_t valueTag, const QString &name, const QStringList &value)
|
|
{
|
|
Q_D(KIppRequest);
|
|
|
|
d->addRequest(group, valueTag, name.toUtf8(), value);
|
|
}
|
|
|
|
void KIppRequest::addInteger(ipp_tag_t group, ipp_tag_t valueTag, const QString &name, int value)
|
|
{
|
|
Q_D(KIppRequest);
|
|
|
|
d->addRequest(group, valueTag, name.toUtf8(), value);
|
|
}
|
|
|
|
void KIppRequest::addBoolean(ipp_tag_t group, const QString &name, bool value)
|
|
{
|
|
Q_D(KIppRequest);
|
|
|
|
d->addRequest(group, IPP_TAG_ZERO, name.toUtf8(), value);
|
|
}
|
|
|
|
void KIppRequest::addVariantValues(const QVariantHash &values)
|
|
{
|
|
QVariantHash::ConstIterator i = values.constBegin();
|
|
while (i != values.constEnd()) {
|
|
QString key = i.key();
|
|
QVariant value = i.value();
|
|
switch (value.type()) {
|
|
case QVariant::Bool:
|
|
// Still in use at add-printer/PageAddPrinter.cpp
|
|
if (key == QLatin1String(KCUPS_PRINTER_IS_ACCEPTING_JOBS)) {
|
|
addBoolean(IPP_TAG_PRINTER, key, value.toBool());
|
|
} else {
|
|
addBoolean(IPP_TAG_OPERATION, key, value.toBool());
|
|
}
|
|
break;
|
|
case QVariant::Int:
|
|
// Still in use at add-printer/PageAddPrinter.cpp
|
|
if (key == QLatin1String(KCUPS_PRINTER_STATE)) {
|
|
addInteger(IPP_TAG_PRINTER, IPP_TAG_ENUM, key, value.toInt());
|
|
} else {
|
|
addInteger(IPP_TAG_OPERATION, IPP_TAG_ENUM, key, value.toInt());
|
|
}
|
|
break;
|
|
case QVariant::String:
|
|
// Still in use at add-printer/*
|
|
if (key == QLatin1String(KCUPS_DEVICE_URI)) {
|
|
// device uri has a different TAG
|
|
addString(IPP_TAG_PRINTER, IPP_TAG_URI, key, value.toString());
|
|
} else if (key == QLatin1String(KCUPS_PRINTER_OP_POLICY) ||
|
|
key == QLatin1String(KCUPS_PRINTER_ERROR_POLICY) ||
|
|
key == QLatin1String("ppd-name")) {
|
|
// printer-op-policy has a different TAG
|
|
addString(IPP_TAG_PRINTER, IPP_TAG_NAME, key, value.toString());
|
|
} else if (key == QLatin1String(KCUPS_JOB_NAME)) {
|
|
addString(IPP_TAG_OPERATION, IPP_TAG_NAME, key, value.toString());
|
|
} else if (key == QLatin1String(KCUPS_WHICH_JOBS)) {
|
|
addString(IPP_TAG_OPERATION, IPP_TAG_KEYWORD, key, value.toString());
|
|
} else {
|
|
addString(IPP_TAG_PRINTER, IPP_TAG_TEXT, key, value.toString());
|
|
}
|
|
break;
|
|
case QVariant::StringList:
|
|
if (key == QLatin1String(KCUPS_MEMBER_URIS)) {
|
|
addStringList(IPP_TAG_PRINTER, IPP_TAG_URI, key, value.toStringList());
|
|
} else {
|
|
addStringList(IPP_TAG_PRINTER, IPP_TAG_NAME, key, value.toStringList());
|
|
}
|
|
break;
|
|
case QVariant::UInt:
|
|
addInteger(IPP_TAG_OPERATION, IPP_TAG_ENUM, key, value.toInt());
|
|
break;
|
|
default:
|
|
kWarning() << "type NOT recognized! This will be ignored:" << key << "values" << i.value();
|
|
}
|
|
++i;
|
|
}
|
|
}
|
|
|
|
void KIppRequest::addPrinterUri(const QString &printerName, bool isClass)
|
|
{
|
|
QString uri = assembleUrif(printerName, isClass);
|
|
addString(IPP_TAG_OPERATION, IPP_TAG_URI, KCUPS_PRINTER_URI, uri);
|
|
}
|
|
|
|
QString KIppRequest::assembleUrif(const QString &name, bool isClass)
|
|
{
|
|
char uri[HTTP_MAX_URI]; // printer URI
|
|
|
|
QString destination;
|
|
if (isClass) {
|
|
destination = QLatin1String("/classes/") % name;
|
|
} else {
|
|
destination = QLatin1String("/printers/") % name;
|
|
}
|
|
|
|
httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", cupsUser(), "localhost",
|
|
ippPort(), destination.toUtf8());
|
|
return uri;
|
|
}
|
|
|
|
KIppRequest &KIppRequest::operator =(const KIppRequest &other)
|
|
{
|
|
Q_D(KIppRequest);
|
|
if (this == &other)
|
|
return *this;
|
|
|
|
*d = *other.d_ptr;
|
|
|
|
return *this;
|
|
}
|
|
|
|
void KIppRequestPrivate::addRequest(ipp_tag_t group, ipp_tag_t valueTag, const QString &name, const QVariant &value)
|
|
{
|
|
KCupsRawRequest request;
|
|
request.group = group;
|
|
request.valueTag = valueTag;
|
|
request.name = name;
|
|
request.value = value;
|
|
|
|
rawRequests << request;
|
|
}
|
|
|
|
void KIppRequestPrivate::addRawRequestsToIpp(ipp_t *ipp) const
|
|
{
|
|
// sort the values as CUPS requires it
|
|
qSort(rawRequests.begin(), rawRequests.end(), rawRequestGroupLessThan);
|
|
|
|
foreach (const KCupsRawRequest &request, rawRequests) {
|
|
switch (request.value.type()) {
|
|
case QVariant::Bool:
|
|
ippAddBoolean(ipp,
|
|
request.group,
|
|
request.name.toUtf8(),
|
|
request.value.toBool());
|
|
break;
|
|
case QVariant::Int:
|
|
case QVariant::UInt:
|
|
ippAddInteger(ipp,
|
|
request.group,
|
|
request.valueTag,
|
|
request.name.toUtf8(),
|
|
request.value.toInt());
|
|
break;
|
|
case QVariant::String:
|
|
ippAddString(ipp,
|
|
request.group,
|
|
request.valueTag,
|
|
request.name.toUtf8(),
|
|
"utf-8",
|
|
request.value.toString().toUtf8());
|
|
break;
|
|
case QVariant::StringList:
|
|
{
|
|
QStringList list = request.value.toStringList();
|
|
QList<QByteArray> valuesQByteArrayList;
|
|
const char **values = qStringListToCharPtrPtr(list, valuesQByteArrayList);
|
|
|
|
ippAddStrings(ipp,
|
|
request.group,
|
|
request.valueTag,
|
|
request.name.toUtf8(),
|
|
list.size(),
|
|
"utf-8",
|
|
values);
|
|
|
|
// ippAddStrings deep copies everything so we can throw away the values.
|
|
// the QBAList and content is auto discarded when going out of scope.
|
|
delete [] values;
|
|
break;
|
|
}
|
|
default:
|
|
kWarning() << "type NOT recognized! This will be ignored:" << request.name << "values" << request.value;
|
|
}
|
|
}
|
|
}
|