/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
*
* Copyright 2015 Red Hat, Inc,
*
* 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, see .
*
* Author: Felipe Borges
*/
#include "pp-job.h"
#include
#include
#if (CUPS_VERSION_MAJOR > 1) || (CUPS_VERSION_MINOR > 5)
#define HAVE_CUPS_1_6 1
#endif
#ifndef HAVE_CUPS_1_6
#define ippGetBoolean(attr, element) attr->values[element].boolean
#define ippGetCount(attr) attr->num_values
#define ippGetInteger(attr, element) attr->values[element].integer
#define ippGetString(attr, element, language) attr->values[element].string.text
#define ippGetValueTag(attr) attr->value_tag
static int
ippGetRange (ipp_attribute_t *attr,
int element,
int *upper)
{
*upper = attr->values[element].range.upper;
return (attr->values[element].range.lower);
}
#endif
struct _PpJob
{
GObject parent_instance;
gint id;
gchar *title;
gint state;
gint priority;
gboolean sensitive;
GStrv auth_info_required;
};
G_DEFINE_TYPE (PpJob, pp_job, G_TYPE_OBJECT)
static void
pp_job_cancel_purge_async_dbus_cb (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
g_autoptr(GVariant) output = NULL;
output = g_dbus_connection_call_finish (G_DBUS_CONNECTION (source_object),
res,
NULL);
}
PpJob *
pp_job_new (gint id, const gchar *title, gint state, gint priority, GStrv auth_info_required)
{
PpJob *job = g_object_new (pp_job_get_type (), NULL);
job->id = id;
job->title = g_strdup (title);
job->state = state;
job->priority = priority;
job->sensitive = FALSE;
job->auth_info_required = g_strdupv (auth_info_required);
return job;
}
const gchar *
pp_job_get_title (PpJob *self)
{
g_return_val_if_fail (PP_IS_JOB(self), NULL);
return self->title;
}
gint
pp_job_get_state (PpJob *self)
{
g_return_val_if_fail (PP_IS_JOB(self), -1);
return self->state;
}
void
pp_job_priority_set_sensitive (PpJob *self,
gboolean sensitive)
{
self->sensitive = sensitive;
}
gboolean
pp_job_priority_get_sensitive (PpJob *self)
{
g_return_val_if_fail (PP_IS_JOB (self), FALSE);
return self->sensitive;
}
gint
pp_job_get_priority (PpJob *self)
{
g_return_val_if_fail (PP_IS_JOB (self), -1);
return self->priority;
}
void
pp_job_set_priority (PpJob *self,
gint priority)
{
self->priority = priority;
}
GStrv
pp_job_get_auth_info_required (PpJob *self)
{
g_return_val_if_fail (PP_IS_JOB(self), NULL);
return self->auth_info_required;
}
void
pp_job_cancel_purge_async (PpJob *self,
gboolean job_purge)
{
g_autoptr(GDBusConnection) bus = NULL;
g_autoptr(GError) error = NULL;
bus = g_bus_get_sync (G_BUS_TYPE_SYSTEM, NULL, &error);
if (!bus)
{
g_warning ("Failed to get session bus: %s", error->message);
return;
}
g_dbus_connection_call (bus,
MECHANISM_BUS,
"/",
MECHANISM_BUS,
"JobCancelPurge",
g_variant_new ("(ib)",
self->id,
job_purge),
G_VARIANT_TYPE ("(s)"),
G_DBUS_CALL_FLAGS_NONE,
-1,
NULL,
pp_job_cancel_purge_async_dbus_cb,
NULL);
}
static void
pp_job_set_hold_until_async_dbus_cb (GObject *source_object,
GAsyncResult *res,
gpointer user_data)
{
g_autoptr(GVariant) output = NULL;
output = g_dbus_connection_call_finish (G_DBUS_CONNECTION (source_object),
res,
NULL);
}
void
pp_job_set_hold_until_async (PpJob *self,
const gchar *job_hold_until)
{
g_autoptr(GDBusConnection) bus = NULL;
g_autoptr(GError) error = NULL;
bus = g_bus_get_sync (G_BUS_TYPE_SYSTEM, NULL, &error);
if (!bus)
{
g_warning ("Failed to get session bus: %s", error->message);
return;
}
g_dbus_connection_call (bus,
MECHANISM_BUS,
"/",
MECHANISM_BUS,
"JobSetHoldUntil",
g_variant_new ("(is)",
self->id,
job_hold_until),
G_VARIANT_TYPE ("(s)"),
G_DBUS_CALL_FLAGS_NONE,
-1,
NULL,
pp_job_set_hold_until_async_dbus_cb,
NULL);
}
static void
pp_job_init (PpJob *obj)
{
}
static void
pp_job_finalize (GObject *object)
{
PpJob *self = PP_JOB (object);
g_clear_pointer (&self->title, g_free);
g_clear_pointer (&self->auth_info_required, g_strfreev);
G_OBJECT_CLASS (pp_job_parent_class)->finalize (object);
}
static void
pp_job_class_init (PpJobClass *class)
{
GObjectClass *object_class = G_OBJECT_CLASS (class);
object_class->finalize = pp_job_finalize;
}
static void
_pp_job_get_attributes_thread (GTask *task,
gpointer source_object,
gpointer task_data,
GCancellable *cancellable)
{
PpJob *self = PP_JOB (source_object);
ipp_attribute_t *attr = NULL;
GVariantBuilder builder;
GVariant *attributes = NULL;
gchar **attributes_names = task_data;
ipp_t *request;
ipp_t *response = NULL;
g_autofree gchar *job_uri = NULL;
gint i, j, length = 0, n_attrs = 0;
job_uri = g_strdup_printf ("ipp://localhost/jobs/%d", self->id);
if (attributes_names != NULL)
{
length = g_strv_length (attributes_names);
request = ippNewRequest (IPP_GET_JOB_ATTRIBUTES);
ippAddString (request, IPP_TAG_OPERATION, IPP_TAG_URI,
"job-uri", NULL, job_uri);
ippAddString (request, IPP_TAG_OPERATION, IPP_TAG_NAME,
"requesting-user-name", NULL, cupsUser ());
ippAddStrings (request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD,
"requested-attributes", length, NULL, (const char **) attributes_names);
response = cupsDoRequest (CUPS_HTTP_DEFAULT, request, "/");
}
if (response != NULL)
{
g_variant_builder_init (&builder, G_VARIANT_TYPE ("a{sv}"));
for (j = 0; j < length; j++)
{
attr = ippFindAttribute (response, attributes_names[j], IPP_TAG_ZERO);
n_attrs = ippGetCount (attr);
if (attr != NULL && n_attrs > 0 && ippGetValueTag (attr) != IPP_TAG_NOVALUE)
{
const GVariantType *type = NULL;
GVariant **values;
GVariant *range[2];
gint range_uppervalue;
values = g_new (GVariant*, n_attrs);
switch (ippGetValueTag (attr))
{
case IPP_TAG_INTEGER:
case IPP_TAG_ENUM:
type = G_VARIANT_TYPE_INT32;
for (i = 0; i < n_attrs; i++)
values[i] = g_variant_new_int32 (ippGetInteger (attr, i));
break;
case IPP_TAG_NAME:
case IPP_TAG_STRING:
case IPP_TAG_TEXT:
case IPP_TAG_URI:
case IPP_TAG_KEYWORD:
case IPP_TAG_URISCHEME:
type = G_VARIANT_TYPE_STRING;
for (i = 0; i < n_attrs; i++)
values[i] = g_variant_new_string (ippGetString (attr, i, NULL));
break;
case IPP_TAG_RANGE:
type = G_VARIANT_TYPE_TUPLE;
for (i = 0; i < n_attrs; i++)
{
range[0] = g_variant_new_int32 (ippGetRange (attr, i, &(range_uppervalue)));
range[1] = g_variant_new_int32 (range_uppervalue);
values[i] = g_variant_new_tuple (range, 2);
}
break;
case IPP_TAG_BOOLEAN:
type = G_VARIANT_TYPE_BOOLEAN;
for (i = 0; i < n_attrs; i++)
values[i] = g_variant_new_boolean (ippGetBoolean (attr, i));
break;
default:
/* do nothing (switch w/ enumeration type) */
break;
}
if (type != NULL)
{
g_variant_builder_add (&builder, "{sv}",
attributes_names[j],
g_variant_new_array (type, values, n_attrs));
}
g_free (values);
}
}
attributes = g_variant_builder_end (&builder);
}
g_task_return_pointer (task, attributes, (GDestroyNotify) g_variant_unref);
}
void
pp_job_get_attributes_async (PpJob *self,
gchar **attributes_names,
GCancellable *cancellable,
GAsyncReadyCallback callback,
gpointer user_data)
{
g_autoptr(GTask) task = NULL;
task = g_task_new (self, cancellable, callback, user_data);
g_task_set_task_data (task, g_strdupv (attributes_names), (GDestroyNotify) g_strfreev);
g_task_run_in_thread (task, _pp_job_get_attributes_thread);
}
GVariant *
pp_job_get_attributes_finish (PpJob *self,
GAsyncResult *result,
GError **error)
{
g_return_val_if_fail (g_task_is_valid (result, self), NULL);
return g_task_propagate_pointer (G_TASK (result), error);
}
static void
_pp_job_authenticate_thread (GTask *task,
gpointer source_object,
gpointer task_data,
GCancellable *cancellable)
{
PpJob *self = source_object;
gboolean result = FALSE;
gchar **auth_info = task_data;
ipp_t *request;
ipp_t *response = NULL;
gint length;
if (auth_info != NULL)
{
g_autofree gchar *job_uri = g_strdup_printf ("ipp://localhost/jobs/%d", self->id);
length = g_strv_length (auth_info);
request = ippNewRequest (IPP_OP_CUPS_AUTHENTICATE_JOB);
ippAddString (request, IPP_TAG_OPERATION, IPP_TAG_URI,
"job-uri", NULL, job_uri);
ippAddString (request, IPP_TAG_OPERATION, IPP_TAG_NAME,
"requesting-user-name", NULL, cupsUser ());
ippAddStrings (request, IPP_TAG_OPERATION, IPP_TAG_TEXT,
"auth-info", length, NULL, (const char **) auth_info);
response = cupsDoRequest (CUPS_HTTP_DEFAULT, request, "/");
result = response != NULL && ippGetStatusCode (response) <= IPP_OK;
if (response != NULL)
ippDelete (response);
}
g_task_return_boolean (task, result);
}
void
pp_job_authenticate_async (PpJob *self,
gchar **auth_info,
GCancellable *cancellable,
GAsyncReadyCallback callback,
gpointer user_data)
{
g_autoptr(GTask) task = NULL;
task = g_task_new (self, cancellable, callback, user_data);
g_task_set_task_data (task, g_strdupv (auth_info), (GDestroyNotify) g_strfreev);
g_task_run_in_thread (task, _pp_job_authenticate_thread);
}
gboolean
pp_job_authenticate_finish (PpJob *self,
GAsyncResult *result,
GError **error)
{
g_return_val_if_fail (g_task_is_valid (result, self), FALSE);
return g_task_propagate_boolean (G_TASK (result), error);
}
static void
pp_job_set_priority_thread (GTask *task,
gpointer source_object,
gpointer task_data,
GCancellable *cancellable)
{
PpJob *self = source_object;
gint priority = GPOINTER_TO_INT (task_data);
ipp_t *request;
gboolean result = TRUE;
g_autofree gchar *uri = NULL;
request = ippNewRequest (IPP_SET_JOB_ATTRIBUTES);
uri = g_strdup_printf ("ipp://localhost/jobs/%d", self->id);
ippAddString (request, IPP_TAG_OPERATION, IPP_TAG_URI,
"job-uri", NULL, uri);
ippAddString (request, IPP_TAG_OPERATION, IPP_TAG_NAME,
"requesting-user-name", NULL, cupsUser ());
ippAddInteger (request, IPP_TAG_JOB, IPP_TAG_INTEGER,
"job-priority", priority);
ippDelete (cupsDoRequest (CUPS_HTTP_DEFAULT, request, "/"));
if (cupsLastError () > IPP_OK_CONFLICT)
{
g_warning ("Failed to set job priority: %s", cupsLastErrorString ());
result = FALSE;
}
g_task_return_boolean (task, result);
}
void
pp_job_set_priority_async (PpJob *self,
gint priority,
GCancellable *cancellable,
GAsyncReadyCallback callback,
gpointer user_data)
{
g_autoptr(GTask) task = NULL;
task = g_task_new (self, cancellable, callback, user_data);
g_task_set_task_data (task, GINT_TO_POINTER (priority), NULL);
g_task_run_in_thread (task, pp_job_set_priority_thread);
}
gboolean
pp_job_set_priority_finish (PpJob *self,
GAsyncResult *result,
GError **error)
{
g_return_val_if_fail (g_task_is_valid (result, self), FALSE);
return g_task_propagate_boolean (G_TASK (result), error);
}