Blame gtksourceview/gtksourcefilesaver.c

Packit a7d494
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8; coding: utf-8 -*- */
Packit a7d494
/* gtksourcefilesaver.c
Packit a7d494
 * This file is part of GtkSourceView
Packit a7d494
 *
Packit a7d494
 * Copyright (C) 2005-2007 - Paolo Borelli and Paolo Maggi
Packit a7d494
 * Copyright (C) 2007 - Steve Frécinaux
Packit a7d494
 * Copyright (C) 2008 - Jesse van den Kieboom
Packit a7d494
 * Copyright (C) 2014, 2016 - Sébastien Wilmet
Packit a7d494
 *
Packit a7d494
 * GtkSourceView is free software; you can redistribute it and/or
Packit a7d494
 * modify it under the terms of the GNU Lesser General Public
Packit a7d494
 * License as published by the Free Software Foundation; either
Packit a7d494
 * version 2.1 of the License, or (at your option) any later version.
Packit a7d494
 *
Packit a7d494
 * GtkSourceView is distributed in the hope that it will be useful,
Packit a7d494
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
Packit a7d494
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
Packit a7d494
 * Lesser General Public License for more details.
Packit a7d494
 *
Packit a7d494
 * You should have received a copy of the GNU Lesser General Public
Packit a7d494
 * License along with this library; if not, write to the Free Software
Packit a7d494
 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
Packit a7d494
 */
Packit a7d494
Packit a7d494
#ifdef HAVE_CONFIG_H
Packit a7d494
#include <config.h>
Packit a7d494
#endif
Packit a7d494
Packit a7d494
#include "gtksourcefilesaver.h"
Packit a7d494
#include "gtksourcefile.h"
Packit a7d494
#include "gtksourcebufferinputstream.h"
Packit a7d494
#include "gtksourceencoding.h"
Packit a7d494
#include "gtksourcebuffer.h"
Packit a7d494
#include "gtksourcebuffer-private.h"
Packit a7d494
#include "gtksourceview-enumtypes.h"
Packit a7d494
#include "gtksourceview-i18n.h"
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * SECTION:filesaver
Packit a7d494
 * @Short_description: Save a GtkSourceBuffer into a file
Packit a7d494
 * @Title: GtkSourceFileSaver
Packit a7d494
 * @See_also: #GtkSourceFile, #GtkSourceFileLoader
Packit a7d494
 *
Packit a7d494
 * A #GtkSourceFileSaver object permits to save a #GtkSourceBuffer into a
Packit a7d494
 * #GFile.
Packit a7d494
 *
Packit a7d494
 * A file saver should be used only for one save operation, including errors
Packit a7d494
 * handling. If an error occurs, you can reconfigure the saver and relaunch the
Packit a7d494
 * operation with gtk_source_file_saver_save_async().
Packit a7d494
 */
Packit a7d494
Packit a7d494
/* The code has been written initially in gedit (GeditDocumentSaver).
Packit a7d494
 * It uses a GtkSourceBufferInputStream as input, create converter(s) if needed
Packit a7d494
 * for the encoding and the compression, and write the contents to a
Packit a7d494
 * GOutputStream (the file).
Packit a7d494
 */
Packit a7d494
Packit a7d494
#if 0
Packit a7d494
#define DEBUG(x) (x)
Packit a7d494
#else
Packit a7d494
#define DEBUG(x)
Packit a7d494
#endif
Packit a7d494
Packit a7d494
#define WRITE_CHUNK_SIZE 8192
Packit a7d494
Packit a7d494
#define QUERY_ATTRIBUTES G_FILE_ATTRIBUTE_TIME_MODIFIED
Packit a7d494
Packit a7d494
enum
Packit a7d494
{
Packit a7d494
	PROP_0,
Packit a7d494
	PROP_BUFFER,
Packit a7d494
	PROP_FILE,
Packit a7d494
	PROP_LOCATION,
Packit a7d494
	PROP_ENCODING,
Packit a7d494
	PROP_NEWLINE_TYPE,
Packit a7d494
	PROP_COMPRESSION_TYPE,
Packit a7d494
	PROP_FLAGS
Packit a7d494
};
Packit a7d494
Packit a7d494
struct _GtkSourceFileSaverPrivate
Packit a7d494
{
Packit a7d494
	/* Weak ref to the GtkSourceBuffer. A strong ref could create a
Packit a7d494
	 * reference cycle in an application. For example a subclass of
Packit a7d494
	 * GtkSourceBuffer can have a strong ref to the FileSaver.
Packit a7d494
	 */
Packit a7d494
	GtkSourceBuffer *source_buffer;
Packit a7d494
Packit a7d494
	/* Weak ref to the GtkSourceFile. A strong ref could create a reference
Packit a7d494
	 * cycle in an application. For example a subclass of GtkSourceFile can
Packit a7d494
	 * have a strong ref to the FileSaver.
Packit a7d494
	 */
Packit a7d494
	GtkSourceFile *file;
Packit a7d494
Packit a7d494
	GFile *location;
Packit a7d494
Packit a7d494
	const GtkSourceEncoding *encoding;
Packit a7d494
	GtkSourceNewlineType newline_type;
Packit a7d494
	GtkSourceCompressionType compression_type;
Packit a7d494
	GtkSourceFileSaverFlags flags;
Packit a7d494
Packit a7d494
	GTask *task;
Packit a7d494
};
Packit a7d494
Packit a7d494
typedef struct _TaskData TaskData;
Packit a7d494
struct _TaskData
Packit a7d494
{
Packit a7d494
	/* The output_stream contains the required converter(s) for the encoding
Packit a7d494
	 * and the compression type.
Packit a7d494
	 * The two streams cannot be spliced directly, because:
Packit a7d494
	 * (1) We need to call the progress callback.
Packit a7d494
	 * (2) Sync methods must be used for the input stream, and async
Packit a7d494
	 *     methods for the output stream.
Packit a7d494
	 */
Packit a7d494
	GtkSourceBufferInputStream *input_stream;
Packit a7d494
	GOutputStream *output_stream;
Packit a7d494
Packit a7d494
	GFileInfo *info;
Packit a7d494
Packit a7d494
	goffset total_size;
Packit a7d494
	GFileProgressCallback progress_cb;
Packit a7d494
	gpointer progress_cb_data;
Packit a7d494
	GDestroyNotify progress_cb_notify;
Packit a7d494
Packit a7d494
	/* This field is used when cancelling the output stream: an error occurs
Packit a7d494
	 * and is stored in this field, the output stream is cancelled
Packit a7d494
	 * asynchronously, and then the error is reported to the task.
Packit a7d494
	 */
Packit a7d494
	GError *error;
Packit a7d494
Packit a7d494
	gssize chunk_bytes_read;
Packit a7d494
	gssize chunk_bytes_written;
Packit a7d494
	gchar chunk_buffer[WRITE_CHUNK_SIZE];
Packit a7d494
Packit a7d494
	guint tried_mount : 1;
Packit a7d494
};
Packit a7d494
Packit a7d494
G_DEFINE_TYPE_WITH_PRIVATE (GtkSourceFileSaver, gtk_source_file_saver, G_TYPE_OBJECT)
Packit a7d494
Packit a7d494
static void read_file_chunk (GTask *task);
Packit a7d494
static void write_file_chunk (GTask *task);
Packit a7d494
static void recover_not_mounted (GTask *task);
Packit a7d494
Packit a7d494
static TaskData *
Packit a7d494
task_data_new (void)
Packit a7d494
{
Packit a7d494
	return g_new0 (TaskData, 1);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
task_data_free (gpointer data)
Packit a7d494
{
Packit a7d494
	TaskData *task_data = data;
Packit a7d494
Packit a7d494
	if (task_data == NULL)
Packit a7d494
	{
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_clear_object (&task_data->input_stream);
Packit a7d494
	g_clear_object (&task_data->output_stream);
Packit a7d494
	g_clear_object (&task_data->info);
Packit a7d494
	g_clear_error (&task_data->error);
Packit a7d494
Packit a7d494
	if (task_data->progress_cb_notify != NULL)
Packit a7d494
	{
Packit a7d494
		task_data->progress_cb_notify (task_data->progress_cb_data);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_free (task_data);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_saver_set_property (GObject      *object,
Packit a7d494
				    guint         prop_id,
Packit a7d494
				    const GValue *value,
Packit a7d494
				    GParamSpec   *pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceFileSaver *saver = GTK_SOURCE_FILE_SAVER (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_BUFFER:
Packit a7d494
			g_assert (saver->priv->source_buffer == NULL);
Packit a7d494
			saver->priv->source_buffer = g_value_get_object (value);
Packit a7d494
			g_object_add_weak_pointer (G_OBJECT (saver->priv->source_buffer),
Packit a7d494
						   (gpointer *)&saver->priv->source_buffer);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_FILE:
Packit a7d494
			g_assert (saver->priv->file == NULL);
Packit a7d494
			saver->priv->file = g_value_get_object (value);
Packit a7d494
			g_object_add_weak_pointer (G_OBJECT (saver->priv->file),
Packit a7d494
						   (gpointer *)&saver->priv->file);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_LOCATION:
Packit a7d494
			g_assert (saver->priv->location == NULL);
Packit a7d494
			saver->priv->location = g_value_dup_object (value);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_ENCODING:
Packit a7d494
			gtk_source_file_saver_set_encoding (saver, g_value_get_boxed (value));
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_NEWLINE_TYPE:
Packit a7d494
			gtk_source_file_saver_set_newline_type (saver, g_value_get_enum (value));
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_COMPRESSION_TYPE:
Packit a7d494
			gtk_source_file_saver_set_compression_type (saver, g_value_get_enum (value));
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_FLAGS:
Packit a7d494
			gtk_source_file_saver_set_flags (saver, g_value_get_flags (value));
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		default:
Packit a7d494
			G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit a7d494
			break;
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_saver_get_property (GObject    *object,
Packit a7d494
				    guint       prop_id,
Packit a7d494
				    GValue     *value,
Packit a7d494
				    GParamSpec *pspec)
Packit a7d494
{
Packit a7d494
	GtkSourceFileSaver *saver = GTK_SOURCE_FILE_SAVER (object);
Packit a7d494
Packit a7d494
	switch (prop_id)
Packit a7d494
	{
Packit a7d494
		case PROP_BUFFER:
Packit a7d494
			g_value_set_object (value, saver->priv->source_buffer);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_FILE:
Packit a7d494
			g_value_set_object (value, saver->priv->file);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_LOCATION:
Packit a7d494
			g_value_set_object (value, saver->priv->location);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_ENCODING:
Packit a7d494
			g_value_set_boxed (value, saver->priv->encoding);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_NEWLINE_TYPE:
Packit a7d494
			g_value_set_enum (value, saver->priv->newline_type);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_COMPRESSION_TYPE:
Packit a7d494
			g_value_set_enum (value, saver->priv->compression_type);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		case PROP_FLAGS:
Packit a7d494
			g_value_set_flags (value, saver->priv->flags);
Packit a7d494
			break;
Packit a7d494
Packit a7d494
		default:
Packit a7d494
			G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
Packit a7d494
			break;
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_saver_dispose (GObject *object)
Packit a7d494
{
Packit a7d494
	GtkSourceFileSaver *saver = GTK_SOURCE_FILE_SAVER (object);
Packit a7d494
Packit a7d494
	if (saver->priv->source_buffer != NULL)
Packit a7d494
	{
Packit a7d494
		g_object_remove_weak_pointer (G_OBJECT (saver->priv->source_buffer),
Packit a7d494
					      (gpointer *)&saver->priv->source_buffer);
Packit a7d494
Packit a7d494
		saver->priv->source_buffer = NULL;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (saver->priv->file != NULL)
Packit a7d494
	{
Packit a7d494
		g_object_remove_weak_pointer (G_OBJECT (saver->priv->file),
Packit a7d494
					      (gpointer *)&saver->priv->file);
Packit a7d494
Packit a7d494
		saver->priv->file = NULL;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_clear_object (&saver->priv->location);
Packit a7d494
	g_clear_object (&saver->priv->task);
Packit a7d494
Packit a7d494
	G_OBJECT_CLASS (gtk_source_file_saver_parent_class)->dispose (object);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_saver_constructed (GObject *object)
Packit a7d494
{
Packit a7d494
	GtkSourceFileSaver *saver = GTK_SOURCE_FILE_SAVER (object);
Packit a7d494
Packit a7d494
	if (saver->priv->file != NULL)
Packit a7d494
	{
Packit a7d494
		const GtkSourceEncoding *encoding;
Packit a7d494
		GtkSourceNewlineType newline_type;
Packit a7d494
		GtkSourceCompressionType compression_type;
Packit a7d494
Packit a7d494
		encoding = gtk_source_file_get_encoding (saver->priv->file);
Packit a7d494
		gtk_source_file_saver_set_encoding (saver, encoding);
Packit a7d494
Packit a7d494
		newline_type = gtk_source_file_get_newline_type (saver->priv->file);
Packit a7d494
		gtk_source_file_saver_set_newline_type (saver, newline_type);
Packit a7d494
Packit a7d494
		compression_type = gtk_source_file_get_compression_type (saver->priv->file);
Packit a7d494
		gtk_source_file_saver_set_compression_type (saver, compression_type);
Packit a7d494
Packit a7d494
		if (saver->priv->location == NULL)
Packit a7d494
		{
Packit a7d494
			saver->priv->location = gtk_source_file_get_location (saver->priv->file);
Packit a7d494
Packit a7d494
			if (saver->priv->location != NULL)
Packit a7d494
			{
Packit a7d494
				g_object_ref (saver->priv->location);
Packit a7d494
			}
Packit a7d494
			else
Packit a7d494
			{
Packit a7d494
				g_warning ("GtkSourceFileSaver: the GtkSourceFile's location is NULL. "
Packit a7d494
					   "Use gtk_source_file_saver_new_with_target().");
Packit a7d494
			}
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	G_OBJECT_CLASS (gtk_source_file_saver_parent_class)->constructed (object);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_saver_class_init (GtkSourceFileSaverClass *klass)
Packit a7d494
{
Packit a7d494
	GObjectClass *object_class = G_OBJECT_CLASS (klass);
Packit a7d494
Packit a7d494
	object_class->dispose = gtk_source_file_saver_dispose;
Packit a7d494
	object_class->set_property = gtk_source_file_saver_set_property;
Packit a7d494
	object_class->get_property = gtk_source_file_saver_get_property;
Packit a7d494
	object_class->constructed = gtk_source_file_saver_constructed;
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileSaver:buffer:
Packit a7d494
	 *
Packit a7d494
	 * The #GtkSourceBuffer to save. The #GtkSourceFileSaver object has a
Packit a7d494
	 * weak reference to the buffer.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_BUFFER,
Packit a7d494
					 g_param_spec_object ("buffer",
Packit a7d494
							      "GtkSourceBuffer",
Packit a7d494
							      "",
Packit a7d494
							      GTK_SOURCE_TYPE_BUFFER,
Packit a7d494
							      G_PARAM_READWRITE |
Packit a7d494
							      G_PARAM_CONSTRUCT_ONLY |
Packit a7d494
							      G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileSaver:file:
Packit a7d494
	 *
Packit a7d494
	 * The #GtkSourceFile. The #GtkSourceFileSaver object has a weak
Packit a7d494
	 * reference to the file.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_FILE,
Packit a7d494
					 g_param_spec_object ("file",
Packit a7d494
							      "GtkSourceFile",
Packit a7d494
							      "",
Packit a7d494
							      GTK_SOURCE_TYPE_FILE,
Packit a7d494
							      G_PARAM_READWRITE |
Packit a7d494
							      G_PARAM_CONSTRUCT_ONLY |
Packit a7d494
							      G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileSaver:location:
Packit a7d494
	 *
Packit a7d494
	 * The #GFile where to save the buffer. By default the location is taken
Packit a7d494
	 * from the #GtkSourceFile at construction time.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_LOCATION,
Packit a7d494
					 g_param_spec_object ("location",
Packit a7d494
							      "Location",
Packit a7d494
							      "",
Packit a7d494
							      G_TYPE_FILE,
Packit a7d494
							      G_PARAM_READWRITE |
Packit a7d494
							      G_PARAM_CONSTRUCT_ONLY |
Packit a7d494
							      G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileSaver:encoding:
Packit a7d494
	 *
Packit a7d494
	 * The file's encoding.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_ENCODING,
Packit a7d494
					 g_param_spec_boxed ("encoding",
Packit a7d494
							     "Encoding",
Packit a7d494
							     "",
Packit a7d494
							     GTK_SOURCE_TYPE_ENCODING,
Packit a7d494
							     G_PARAM_READWRITE |
Packit a7d494
							     G_PARAM_CONSTRUCT |
Packit a7d494
							     G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileSaver:newline-type:
Packit a7d494
	 *
Packit a7d494
	 * The newline type.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_NEWLINE_TYPE,
Packit a7d494
					 g_param_spec_enum ("newline-type",
Packit a7d494
					                    "Newline type",
Packit a7d494
							    "",
Packit a7d494
					                    GTK_SOURCE_TYPE_NEWLINE_TYPE,
Packit a7d494
					                    GTK_SOURCE_NEWLINE_TYPE_LF,
Packit a7d494
					                    G_PARAM_READWRITE |
Packit a7d494
					                    G_PARAM_CONSTRUCT |
Packit a7d494
							    G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileSaver:compression-type:
Packit a7d494
	 *
Packit a7d494
	 * The compression type.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_COMPRESSION_TYPE,
Packit a7d494
					 g_param_spec_enum ("compression-type",
Packit a7d494
					                    "Compression type",
Packit a7d494
					                    "",
Packit a7d494
					                    GTK_SOURCE_TYPE_COMPRESSION_TYPE,
Packit a7d494
					                    GTK_SOURCE_COMPRESSION_TYPE_NONE,
Packit a7d494
					                    G_PARAM_READWRITE |
Packit a7d494
					                    G_PARAM_CONSTRUCT |
Packit a7d494
							    G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	/**
Packit a7d494
	 * GtkSourceFileSaver:flags:
Packit a7d494
	 *
Packit a7d494
	 * File saving flags.
Packit a7d494
	 *
Packit a7d494
	 * Since: 3.14
Packit a7d494
	 */
Packit a7d494
	g_object_class_install_property (object_class,
Packit a7d494
					 PROP_FLAGS,
Packit a7d494
					 g_param_spec_flags ("flags",
Packit a7d494
							     "Flags",
Packit a7d494
							     "",
Packit a7d494
							     GTK_SOURCE_TYPE_FILE_SAVER_FLAGS,
Packit a7d494
							     GTK_SOURCE_FILE_SAVER_FLAGS_NONE,
Packit a7d494
							     G_PARAM_READWRITE |
Packit a7d494
							     G_PARAM_CONSTRUCT |
Packit a7d494
							     G_PARAM_STATIC_STRINGS));
Packit a7d494
Packit a7d494
	/* Due to potential deadlocks when registering types, we need to
Packit a7d494
	 * ensure the dependent private class GtkSourceBufferInputStream
Packit a7d494
	 * has been registered up front.
Packit a7d494
	 *
Packit a7d494
	 * See https://bugzilla.gnome.org/show_bug.cgi?id=780216
Packit a7d494
	 */
Packit a7d494
	g_type_ensure (GTK_SOURCE_TYPE_BUFFER_INPUT_STREAM);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
gtk_source_file_saver_init (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	saver->priv = gtk_source_file_saver_get_instance_private (saver);
Packit a7d494
}
Packit a7d494
Packit a7d494
/* BEGIN NOTE:
Packit a7d494
 *
Packit a7d494
 * This fixes an issue in GOutputStream that applies the atomic replace save
Packit a7d494
 * strategy. The stream moves the written file to the original file when the
Packit a7d494
 * stream is closed. However, there is no way currently to tell the stream that
Packit a7d494
 * the save should be aborted (there could be a conversion error). The patch
Packit a7d494
 * explicitly closes the output stream in all these cases with a GCancellable in
Packit a7d494
 * the cancelled state, causing the output stream to close, but not move the
Packit a7d494
 * file. This makes use of an implementation detail in the local file stream
Packit a7d494
 * and should be properly fixed by adding the appropriate API in GIO. Until
Packit a7d494
 * then, at least we prevent data corruption for now.
Packit a7d494
 *
Packit a7d494
 * Relevant bug reports:
Packit a7d494
 *
Packit a7d494
 * Bug 615110 - write file ignore encoding errors (gedit)
Packit a7d494
 * https://bugzilla.gnome.org/show_bug.cgi?id=615110
Packit a7d494
 *
Packit a7d494
 * Bug 602412 - g_file_replace does not restore original file when there is
Packit a7d494
 *              errors while writing (glib/gio)
Packit a7d494
 * https://bugzilla.gnome.org/show_bug.cgi?id=602412
Packit a7d494
 */
Packit a7d494
static void
Packit a7d494
cancel_output_stream_ready_cb (GObject      *source_object,
Packit a7d494
			       GAsyncResult *result,
Packit a7d494
			       gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GOutputStream *output_stream = G_OUTPUT_STREAM (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	TaskData *task_data;
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	g_output_stream_close_finish (output_stream, result, NULL);
Packit a7d494
Packit a7d494
	if (task_data->error != NULL)
Packit a7d494
	{
Packit a7d494
		GError *error = task_data->error;
Packit a7d494
		task_data->error = NULL;
Packit a7d494
		g_task_return_error (task, error);
Packit a7d494
	}
Packit a7d494
	else
Packit a7d494
	{
Packit a7d494
		g_task_return_boolean (task, FALSE);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
cancel_output_stream (GTask *task)
Packit a7d494
{
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GCancellable *cancellable;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Cancel output stream\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	cancellable = g_cancellable_new ();
Packit a7d494
	g_cancellable_cancel (cancellable);
Packit a7d494
Packit a7d494
	g_output_stream_close_async (task_data->output_stream,
Packit a7d494
				     g_task_get_priority (task),
Packit a7d494
				     cancellable,
Packit a7d494
				     cancel_output_stream_ready_cb,
Packit a7d494
				     task);
Packit a7d494
Packit a7d494
	g_object_unref (cancellable);
Packit a7d494
}
Packit a7d494
Packit a7d494
/*
Packit a7d494
 * END NOTE
Packit a7d494
 */
Packit a7d494
Packit a7d494
static void
Packit a7d494
query_info_cb (GObject      *source_object,
Packit a7d494
	       GAsyncResult *result,
Packit a7d494
	       gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GFile *location = G_FILE (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Finished query info on file\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	g_clear_object (&task_data->info);
Packit a7d494
	task_data->info = g_file_query_info_finish (location, result, &error);
Packit a7d494
Packit a7d494
	if (error != NULL)
Packit a7d494
	{
Packit a7d494
		DEBUG ({
Packit a7d494
		       g_print ("Query info failed: %s\n", error->message);
Packit a7d494
		});
Packit a7d494
Packit a7d494
		g_task_return_error (task, error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_task_return_boolean (task, TRUE);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
close_output_stream_cb (GObject      *source_object,
Packit a7d494
			GAsyncResult *result,
Packit a7d494
			gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GOutputStream *output_stream = G_OUTPUT_STREAM (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	GtkSourceFileSaver *saver;
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	saver = g_task_get_source_object (task);
Packit a7d494
Packit a7d494
	g_output_stream_close_finish (output_stream, result, &error);
Packit a7d494
Packit a7d494
	if (error != NULL)
Packit a7d494
	{
Packit a7d494
		DEBUG ({
Packit a7d494
		       g_print ("Closing stream error: %s\n", error->message);
Packit a7d494
		});
Packit a7d494
Packit a7d494
		g_task_return_error (task, error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* Get the file info: note we cannot use
Packit a7d494
	 * g_file_output_stream_query_info_async() since it is not able to get
Packit a7d494
	 * the modification time.
Packit a7d494
	 */
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Query info on file\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	g_file_query_info_async (saver->priv->location,
Packit a7d494
			         QUERY_ATTRIBUTES,
Packit a7d494
			         G_FILE_QUERY_INFO_NONE,
Packit a7d494
				 g_task_get_priority (task),
Packit a7d494
				 g_task_get_cancellable (task),
Packit a7d494
			         query_info_cb,
Packit a7d494
			         task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
write_complete (GTask *task)
Packit a7d494
{
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Close input stream\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	g_input_stream_close (G_INPUT_STREAM (task_data->input_stream),
Packit a7d494
			      g_task_get_cancellable (task),
Packit a7d494
			      &error);
Packit a7d494
Packit a7d494
	if (error != NULL)
Packit a7d494
	{
Packit a7d494
		DEBUG ({
Packit a7d494
		       g_print ("Closing input stream error: %s\n", error->message);
Packit a7d494
		});
Packit a7d494
Packit a7d494
		g_clear_error (&task_data->error);
Packit a7d494
		task_data->error = error;
Packit a7d494
		cancel_output_stream (task);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Close output stream\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	g_output_stream_close_async (task_data->output_stream,
Packit a7d494
				     g_task_get_priority (task),
Packit a7d494
				     g_task_get_cancellable (task),
Packit a7d494
				     close_output_stream_cb,
Packit a7d494
				     task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
write_file_chunk_cb (GObject      *source_object,
Packit a7d494
		     GAsyncResult *result,
Packit a7d494
		     gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GOutputStream *output_stream = G_OUTPUT_STREAM (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	TaskData *task_data;
Packit a7d494
	gssize bytes_written;
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	bytes_written = g_output_stream_write_finish (output_stream, result, &error);
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Written: %" G_GSSIZE_FORMAT "\n", bytes_written);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	if (error != NULL)
Packit a7d494
	{
Packit a7d494
		DEBUG ({
Packit a7d494
		       g_print ("Write error: %s\n", error->message);
Packit a7d494
		});
Packit a7d494
Packit a7d494
		g_clear_error (&task_data->error);
Packit a7d494
		task_data->error = error;
Packit a7d494
		cancel_output_stream (task);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	task_data->chunk_bytes_written += bytes_written;
Packit a7d494
Packit a7d494
	/* Write again */
Packit a7d494
	if (task_data->chunk_bytes_written < task_data->chunk_bytes_read)
Packit a7d494
	{
Packit a7d494
		write_file_chunk (task);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (task_data->progress_cb != NULL)
Packit a7d494
	{
Packit a7d494
		gsize total_chars_written;
Packit a7d494
Packit a7d494
		total_chars_written = _gtk_source_buffer_input_stream_tell (task_data->input_stream);
Packit a7d494
Packit a7d494
		task_data->progress_cb (total_chars_written,
Packit a7d494
					task_data->total_size,
Packit a7d494
					task_data->progress_cb_data);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	read_file_chunk (task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
write_file_chunk (GTask *task)
Packit a7d494
{
Packit a7d494
	TaskData *task_data;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	g_output_stream_write_async (task_data->output_stream,
Packit a7d494
				     task_data->chunk_buffer + task_data->chunk_bytes_written,
Packit a7d494
				     task_data->chunk_bytes_read - task_data->chunk_bytes_written,
Packit a7d494
				     g_task_get_priority (task),
Packit a7d494
				     g_task_get_cancellable (task),
Packit a7d494
				     write_file_chunk_cb,
Packit a7d494
				     task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
read_file_chunk (GTask *task)
Packit a7d494
{
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	task_data->chunk_bytes_written = 0;
Packit a7d494
Packit a7d494
	/* We use sync methods on doc stream since it is in memory. Using async
Packit a7d494
	 * would be racy and we could end up with invalid iters.
Packit a7d494
	 */
Packit a7d494
	task_data->chunk_bytes_read = g_input_stream_read (G_INPUT_STREAM (task_data->input_stream),
Packit a7d494
							   task_data->chunk_buffer,
Packit a7d494
							   WRITE_CHUNK_SIZE,
Packit a7d494
							   g_task_get_cancellable (task),
Packit a7d494
							   &error);
Packit a7d494
Packit a7d494
	if (error != NULL)
Packit a7d494
	{
Packit a7d494
		g_clear_error (&task_data->error);
Packit a7d494
		task_data->error = error;
Packit a7d494
		cancel_output_stream (task);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* Check if we finished reading and writing. */
Packit a7d494
	if (task_data->chunk_bytes_read == 0)
Packit a7d494
	{
Packit a7d494
		write_complete (task);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	write_file_chunk (task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
replace_file_cb (GObject      *source_object,
Packit a7d494
		 GAsyncResult *result,
Packit a7d494
		 gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GFile *location = G_FILE (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	GtkSourceFileSaver *saver;
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GFileOutputStream *file_output_stream;
Packit a7d494
	GOutputStream *output_stream;
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	saver = g_task_get_source_object (task);
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	file_output_stream = g_file_replace_finish (location, result, &error);
Packit a7d494
Packit a7d494
	if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_NOT_MOUNTED) &&
Packit a7d494
	    !task_data->tried_mount)
Packit a7d494
	{
Packit a7d494
		recover_not_mounted (task);
Packit a7d494
		g_error_free (error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
	else if (error != NULL)
Packit a7d494
	{
Packit a7d494
		DEBUG ({
Packit a7d494
		       g_print ("Opening file failed: %s\n", error->message);
Packit a7d494
		});
Packit a7d494
Packit a7d494
		g_task_return_error (task, error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (saver->priv->compression_type == GTK_SOURCE_COMPRESSION_TYPE_GZIP)
Packit a7d494
	{
Packit a7d494
		GZlibCompressor *compressor;
Packit a7d494
Packit a7d494
		DEBUG ({
Packit a7d494
		       g_print ("Use gzip compressor\n");
Packit a7d494
		});
Packit a7d494
Packit a7d494
		compressor = g_zlib_compressor_new (G_ZLIB_COMPRESSOR_FORMAT_GZIP, -1);
Packit a7d494
Packit a7d494
		output_stream = g_converter_output_stream_new (G_OUTPUT_STREAM (file_output_stream),
Packit a7d494
							       G_CONVERTER (compressor));
Packit a7d494
Packit a7d494
		g_object_unref (compressor);
Packit a7d494
		g_object_unref (file_output_stream);
Packit a7d494
	}
Packit a7d494
	else
Packit a7d494
	{
Packit a7d494
		output_stream = G_OUTPUT_STREAM (file_output_stream);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* FIXME: manage converter error? */
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Encoding charset: %s\n",
Packit a7d494
			gtk_source_encoding_get_charset (saver->priv->encoding));
Packit a7d494
	});
Packit a7d494
Packit a7d494
	if (saver->priv->encoding != gtk_source_encoding_get_utf8 ())
Packit a7d494
	{
Packit a7d494
		GCharsetConverter *converter;
Packit a7d494
Packit a7d494
		converter = g_charset_converter_new (gtk_source_encoding_get_charset (saver->priv->encoding),
Packit a7d494
						     "UTF-8",
Packit a7d494
						     NULL);
Packit a7d494
Packit a7d494
		g_clear_object (&task_data->output_stream);
Packit a7d494
		task_data->output_stream = g_converter_output_stream_new (output_stream,
Packit a7d494
									  G_CONVERTER (converter));
Packit a7d494
Packit a7d494
		g_object_unref (converter);
Packit a7d494
		g_object_unref (output_stream);
Packit a7d494
	}
Packit a7d494
	else
Packit a7d494
	{
Packit a7d494
		g_clear_object (&task_data->output_stream);
Packit a7d494
		task_data->output_stream = G_OUTPUT_STREAM (output_stream);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	task_data->total_size = _gtk_source_buffer_input_stream_get_total_size (task_data->input_stream);
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Total number of characters: %" G_GINT64_FORMAT "\n", task_data->total_size);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	read_file_chunk (task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
begin_write (GTask *task)
Packit a7d494
{
Packit a7d494
	GtkSourceFileSaver *saver;
Packit a7d494
	gboolean create_backup;
Packit a7d494
Packit a7d494
	saver = g_task_get_source_object (task);
Packit a7d494
Packit a7d494
	create_backup = (saver->priv->flags & GTK_SOURCE_FILE_SAVER_FLAGS_CREATE_BACKUP) != 0;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Start replacing file contents\n");
Packit a7d494
	       g_print ("Make backup: %s\n", create_backup ? "yes" : "no");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	g_file_replace_async (saver->priv->location,
Packit a7d494
			      NULL,
Packit a7d494
			      create_backup,
Packit a7d494
			      G_FILE_CREATE_NONE,
Packit a7d494
			      g_task_get_priority (task),
Packit a7d494
			      g_task_get_cancellable (task),
Packit a7d494
			      replace_file_cb,
Packit a7d494
			      task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
check_externally_modified_cb (GObject      *source_object,
Packit a7d494
			      GAsyncResult *result,
Packit a7d494
			      gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GFile *location = G_FILE (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	GtkSourceFileSaver *saver;
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GFileInfo *info;
Packit a7d494
	GTimeVal old_mtime;
Packit a7d494
	GTimeVal cur_mtime;
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	saver = g_task_get_source_object (task);
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	info = g_file_query_info_finish (location, result, &error);
Packit a7d494
Packit a7d494
	if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_NOT_MOUNTED) &&
Packit a7d494
	    !task_data->tried_mount)
Packit a7d494
	{
Packit a7d494
		recover_not_mounted (task);
Packit a7d494
		g_error_free (error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	/* It's perfectly fine if the file doesn't exist yet. */
Packit a7d494
	if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_NOT_FOUND))
Packit a7d494
	{
Packit a7d494
		g_clear_error (&error);
Packit a7d494
	}
Packit a7d494
	else if (error != NULL)
Packit a7d494
	{
Packit a7d494
		DEBUG ({
Packit a7d494
		       g_print ("Check externally modified failed: %s\n", error->message);
Packit a7d494
		});
Packit a7d494
Packit a7d494
		g_task_return_error (task, error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (_gtk_source_file_get_modification_time (saver->priv->file, &old_mtime) &&
Packit a7d494
	    info != NULL &&
Packit a7d494
	    g_file_info_has_attribute (info, G_FILE_ATTRIBUTE_TIME_MODIFIED))
Packit a7d494
	{
Packit a7d494
		g_file_info_get_modification_time (info, &cur_mtime);
Packit a7d494
Packit a7d494
		if (old_mtime.tv_sec != cur_mtime.tv_sec ||
Packit a7d494
		    old_mtime.tv_usec != cur_mtime.tv_usec)
Packit a7d494
		{
Packit a7d494
			DEBUG ({
Packit a7d494
			       g_print ("The file is externally modified\n");
Packit a7d494
			});
Packit a7d494
Packit a7d494
			g_task_return_new_error (task,
Packit a7d494
						 GTK_SOURCE_FILE_SAVER_ERROR,
Packit a7d494
						 GTK_SOURCE_FILE_SAVER_ERROR_EXTERNALLY_MODIFIED,
Packit a7d494
						 _("The file is externally modified."));
Packit a7d494
			g_object_unref (info);
Packit a7d494
			return;
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	begin_write (task);
Packit a7d494
Packit a7d494
	if (info != NULL)
Packit a7d494
	{
Packit a7d494
		g_object_unref (info);
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
check_externally_modified (GTask *task)
Packit a7d494
{
Packit a7d494
	GtkSourceFileSaver *saver;
Packit a7d494
	gboolean save_as = FALSE;
Packit a7d494
Packit a7d494
	saver = g_task_get_source_object (task);
Packit a7d494
Packit a7d494
	if (saver->priv->file != NULL)
Packit a7d494
	{
Packit a7d494
		GFile *prev_location;
Packit a7d494
Packit a7d494
		prev_location = gtk_source_file_get_location (saver->priv->file);
Packit a7d494
Packit a7d494
		/* Don't check for externally modified for a "save as" operation,
Packit a7d494
		 * because the user has normally accepted to overwrite the file if it
Packit a7d494
		 * already exists.
Packit a7d494
		 */
Packit a7d494
		save_as = (prev_location == NULL ||
Packit a7d494
			   !g_file_equal (prev_location, saver->priv->location));
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (saver->priv->flags & GTK_SOURCE_FILE_SAVER_FLAGS_IGNORE_MODIFICATION_TIME ||
Packit a7d494
	    save_as)
Packit a7d494
	{
Packit a7d494
		begin_write (task);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Check externally modified\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	g_file_query_info_async (saver->priv->location,
Packit a7d494
			         G_FILE_ATTRIBUTE_TIME_MODIFIED,
Packit a7d494
			         G_FILE_QUERY_INFO_NONE,
Packit a7d494
				 g_task_get_priority (task),
Packit a7d494
				 g_task_get_cancellable (task),
Packit a7d494
			         check_externally_modified_cb,
Packit a7d494
			         task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
mount_cb (GObject      *source_object,
Packit a7d494
	  GAsyncResult *result,
Packit a7d494
	  gpointer      user_data)
Packit a7d494
{
Packit a7d494
	GFile *location = G_FILE (source_object);
Packit a7d494
	GTask *task = G_TASK (user_data);
Packit a7d494
	GError *error = NULL;
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	g_file_mount_enclosing_volume_finish (location, result, &error);
Packit a7d494
Packit a7d494
	if (error != NULL)
Packit a7d494
	{
Packit a7d494
		g_task_return_error (task, error);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	check_externally_modified (task);
Packit a7d494
}
Packit a7d494
Packit a7d494
static void
Packit a7d494
recover_not_mounted (GTask *task)
Packit a7d494
{
Packit a7d494
	GtkSourceFileSaver *saver;
Packit a7d494
	TaskData *task_data;
Packit a7d494
	GMountOperation *mount_operation;
Packit a7d494
Packit a7d494
	saver = g_task_get_source_object (task);
Packit a7d494
	task_data = g_task_get_task_data (task);
Packit a7d494
Packit a7d494
	mount_operation = _gtk_source_file_create_mount_operation (saver->priv->file);
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("%s\n", G_STRFUNC);
Packit a7d494
	});
Packit a7d494
Packit a7d494
	task_data->tried_mount = TRUE;
Packit a7d494
Packit a7d494
	g_file_mount_enclosing_volume (saver->priv->location,
Packit a7d494
				       G_MOUNT_MOUNT_NONE,
Packit a7d494
				       mount_operation,
Packit a7d494
				       g_task_get_cancellable (task),
Packit a7d494
				       mount_cb,
Packit a7d494
				       task);
Packit a7d494
Packit a7d494
	g_object_unref (mount_operation);
Packit a7d494
}
Packit a7d494
Packit a7d494
GQuark
Packit a7d494
gtk_source_file_saver_error_quark (void)
Packit a7d494
{
Packit a7d494
	static GQuark quark = 0;
Packit a7d494
Packit a7d494
	if (G_UNLIKELY (quark == 0))
Packit a7d494
	{
Packit a7d494
		quark = g_quark_from_static_string ("gtk-source-file-saver-error");
Packit a7d494
	}
Packit a7d494
Packit a7d494
	return quark;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_new:
Packit a7d494
 * @buffer: the #GtkSourceBuffer to save.
Packit a7d494
 * @file: the #GtkSourceFile.
Packit a7d494
 *
Packit a7d494
 * Creates a new #GtkSourceFileSaver object. The @buffer will be saved to the
Packit a7d494
 * #GtkSourceFile's location.
Packit a7d494
 *
Packit a7d494
 * This constructor is suitable for a simple "save" operation, when the @file
Packit a7d494
 * already contains a non-%NULL #GtkSourceFile:location.
Packit a7d494
 *
Packit a7d494
 * Returns: a new #GtkSourceFileSaver object.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceFileSaver *
Packit a7d494
gtk_source_file_saver_new (GtkSourceBuffer *buffer,
Packit a7d494
			   GtkSourceFile   *file)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_BUFFER (buffer), NULL);
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE (file), NULL);
Packit a7d494
Packit a7d494
	return g_object_new (GTK_SOURCE_TYPE_FILE_SAVER,
Packit a7d494
			     "buffer", buffer,
Packit a7d494
			     "file", file,
Packit a7d494
			     NULL);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_new_with_target:
Packit a7d494
 * @buffer: the #GtkSourceBuffer to save.
Packit a7d494
 * @file: the #GtkSourceFile.
Packit a7d494
 * @target_location: the #GFile where to save the buffer to.
Packit a7d494
 *
Packit a7d494
 * Creates a new #GtkSourceFileSaver object with a target location. When the
Packit a7d494
 * file saving is finished successfully, @target_location is set to the @file's
Packit a7d494
 * #GtkSourceFile:location property. If an error occurs, the previous valid
Packit a7d494
 * location is still available in #GtkSourceFile.
Packit a7d494
 *
Packit a7d494
 * This constructor is suitable for a "save as" operation, or for saving a new
Packit a7d494
 * buffer for the first time.
Packit a7d494
 *
Packit a7d494
 * Returns: a new #GtkSourceFileSaver object.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceFileSaver *
Packit a7d494
gtk_source_file_saver_new_with_target (GtkSourceBuffer *buffer,
Packit a7d494
				       GtkSourceFile   *file,
Packit a7d494
				       GFile           *target_location)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_BUFFER (buffer), NULL);
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE (file), NULL);
Packit a7d494
	g_return_val_if_fail (G_IS_FILE (target_location), NULL);
Packit a7d494
Packit a7d494
	return g_object_new (GTK_SOURCE_TYPE_FILE_SAVER,
Packit a7d494
			     "buffer", buffer,
Packit a7d494
			     "file", file,
Packit a7d494
			     "location", target_location,
Packit a7d494
			     NULL);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_get_buffer:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 *
Packit a7d494
 * Returns: (transfer none): the #GtkSourceBuffer to save.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceBuffer *
Packit a7d494
gtk_source_file_saver_get_buffer (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), NULL);
Packit a7d494
Packit a7d494
	return saver->priv->source_buffer;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_get_file:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 *
Packit a7d494
 * Returns: (transfer none): the #GtkSourceFile.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceFile *
Packit a7d494
gtk_source_file_saver_get_file (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), NULL);
Packit a7d494
Packit a7d494
	return saver->priv->file;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_get_location:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 *
Packit a7d494
 * Returns: (transfer none): the #GFile where to save the buffer to.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GFile *
Packit a7d494
gtk_source_file_saver_get_location (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), NULL);
Packit a7d494
Packit a7d494
	return saver->priv->location;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_set_encoding:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 * @encoding: (nullable): the new encoding, or %NULL for UTF-8.
Packit a7d494
 *
Packit a7d494
 * Sets the encoding. If @encoding is %NULL, the UTF-8 encoding will be set.
Packit a7d494
 * By default the encoding is taken from the #GtkSourceFile.
Packit a7d494
 *
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_file_saver_set_encoding (GtkSourceFileSaver      *saver,
Packit a7d494
				    const GtkSourceEncoding *encoding)
Packit a7d494
{
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver));
Packit a7d494
	g_return_if_fail (saver->priv->task == NULL);
Packit a7d494
Packit a7d494
	if (encoding == NULL)
Packit a7d494
	{
Packit a7d494
		encoding = gtk_source_encoding_get_utf8 ();
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (saver->priv->encoding != encoding)
Packit a7d494
	{
Packit a7d494
		saver->priv->encoding = encoding;
Packit a7d494
		g_object_notify (G_OBJECT (saver), "encoding");
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_get_encoding:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 *
Packit a7d494
 * Returns: the encoding.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
const GtkSourceEncoding *
Packit a7d494
gtk_source_file_saver_get_encoding (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), NULL);
Packit a7d494
Packit a7d494
	return saver->priv->encoding;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_set_newline_type:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 * @newline_type: the new newline type.
Packit a7d494
 *
Packit a7d494
 * Sets the newline type. By default the newline type is taken from the
Packit a7d494
 * #GtkSourceFile.
Packit a7d494
 *
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_file_saver_set_newline_type (GtkSourceFileSaver   *saver,
Packit a7d494
					GtkSourceNewlineType  newline_type)
Packit a7d494
{
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver));
Packit a7d494
	g_return_if_fail (saver->priv->task == NULL);
Packit a7d494
Packit a7d494
	if (saver->priv->newline_type != newline_type)
Packit a7d494
	{
Packit a7d494
		saver->priv->newline_type = newline_type;
Packit a7d494
		g_object_notify (G_OBJECT (saver), "newline-type");
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_get_newline_type:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 *
Packit a7d494
 * Returns: the newline type.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceNewlineType
Packit a7d494
gtk_source_file_saver_get_newline_type (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), GTK_SOURCE_NEWLINE_TYPE_DEFAULT);
Packit a7d494
Packit a7d494
	return saver->priv->newline_type;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_set_compression_type:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 * @compression_type: the new compression type.
Packit a7d494
 *
Packit a7d494
 * Sets the compression type. By default the compression type is taken from the
Packit a7d494
 * #GtkSourceFile.
Packit a7d494
 *
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_file_saver_set_compression_type (GtkSourceFileSaver       *saver,
Packit a7d494
					    GtkSourceCompressionType  compression_type)
Packit a7d494
{
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver));
Packit a7d494
	g_return_if_fail (saver->priv->task == NULL);
Packit a7d494
Packit a7d494
	if (saver->priv->compression_type != compression_type)
Packit a7d494
	{
Packit a7d494
		saver->priv->compression_type = compression_type;
Packit a7d494
		g_object_notify (G_OBJECT (saver), "compression-type");
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_get_compression_type:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 *
Packit a7d494
 * Returns: the compression type.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceCompressionType
Packit a7d494
gtk_source_file_saver_get_compression_type (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), GTK_SOURCE_COMPRESSION_TYPE_NONE);
Packit a7d494
Packit a7d494
	return saver->priv->compression_type;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_set_flags:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 * @flags: the new flags.
Packit a7d494
 *
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_file_saver_set_flags (GtkSourceFileSaver      *saver,
Packit a7d494
				 GtkSourceFileSaverFlags  flags)
Packit a7d494
{
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver));
Packit a7d494
	g_return_if_fail (saver->priv->task == NULL);
Packit a7d494
Packit a7d494
	if (saver->priv->flags != flags)
Packit a7d494
	{
Packit a7d494
		saver->priv->flags = flags;
Packit a7d494
		g_object_notify (G_OBJECT (saver), "flags");
Packit a7d494
	}
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_get_flags:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 *
Packit a7d494
 * Returns: the flags.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
GtkSourceFileSaverFlags
Packit a7d494
gtk_source_file_saver_get_flags (GtkSourceFileSaver *saver)
Packit a7d494
{
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), GTK_SOURCE_FILE_SAVER_FLAGS_NONE);
Packit a7d494
Packit a7d494
	return saver->priv->flags;
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_save_async:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 * @io_priority: the I/O priority of the request. E.g. %G_PRIORITY_LOW,
Packit a7d494
 *   %G_PRIORITY_DEFAULT or %G_PRIORITY_HIGH.
Packit a7d494
 * @cancellable: (nullable): optional #GCancellable object, %NULL to ignore.
Packit a7d494
 * @progress_callback: (scope notified) (nullable): function to call back with
Packit a7d494
 *   progress information, or %NULL if progress information is not needed.
Packit a7d494
 * @progress_callback_data: (closure): user data to pass to @progress_callback.
Packit a7d494
 * @progress_callback_notify: (nullable): function to call on
Packit a7d494
 *   @progress_callback_data when the @progress_callback is no longer needed, or
Packit a7d494
 *   %NULL.
Packit a7d494
 * @callback: (scope async): a #GAsyncReadyCallback to call when the request is
Packit a7d494
 *   satisfied.
Packit a7d494
 * @user_data: user data to pass to @callback.
Packit a7d494
 *
Packit a7d494
 * Saves asynchronously the buffer into the file. See the #GAsyncResult
Packit a7d494
 * documentation to know how to use this function.
Packit a7d494
 *
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
Packit a7d494
/* The GDestroyNotify is needed, currently the following bug is not fixed:
Packit a7d494
 * https://bugzilla.gnome.org/show_bug.cgi?id=616044
Packit a7d494
 */
Packit a7d494
void
Packit a7d494
gtk_source_file_saver_save_async (GtkSourceFileSaver     *saver,
Packit a7d494
				  gint                    io_priority,
Packit a7d494
				  GCancellable           *cancellable,
Packit a7d494
				  GFileProgressCallback   progress_callback,
Packit a7d494
				  gpointer                progress_callback_data,
Packit a7d494
				  GDestroyNotify          progress_callback_notify,
Packit a7d494
				  GAsyncReadyCallback     callback,
Packit a7d494
				  gpointer                user_data)
Packit a7d494
{
Packit a7d494
	TaskData *task_data;
Packit a7d494
	gboolean check_invalid_chars;
Packit a7d494
	gboolean implicit_trailing_newline;
Packit a7d494
Packit a7d494
	g_return_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver));
Packit a7d494
	g_return_if_fail (cancellable == NULL || G_IS_CANCELLABLE (cancellable));
Packit a7d494
	g_return_if_fail (saver->priv->task == NULL);
Packit a7d494
Packit a7d494
	saver->priv->task = g_task_new (saver, cancellable, callback, user_data);
Packit a7d494
	g_task_set_priority (saver->priv->task, io_priority);
Packit a7d494
Packit a7d494
	task_data = task_data_new ();
Packit a7d494
	g_task_set_task_data (saver->priv->task, task_data, task_data_free);
Packit a7d494
Packit a7d494
	task_data->progress_cb = progress_callback;
Packit a7d494
	task_data->progress_cb_data = progress_callback_data;
Packit a7d494
	task_data->progress_cb_notify = progress_callback_notify;
Packit a7d494
Packit a7d494
	if (saver->priv->source_buffer == NULL ||
Packit a7d494
	    saver->priv->file == NULL ||
Packit a7d494
	    saver->priv->location == NULL)
Packit a7d494
	{
Packit a7d494
		g_task_return_boolean (saver->priv->task, FALSE);
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	check_invalid_chars = (saver->priv->flags & GTK_SOURCE_FILE_SAVER_FLAGS_IGNORE_INVALID_CHARS) == 0;
Packit a7d494
Packit a7d494
	if (check_invalid_chars && _gtk_source_buffer_has_invalid_chars (saver->priv->source_buffer))
Packit a7d494
	{
Packit a7d494
		g_task_return_new_error (saver->priv->task,
Packit a7d494
					 GTK_SOURCE_FILE_SAVER_ERROR,
Packit a7d494
					 GTK_SOURCE_FILE_SAVER_ERROR_INVALID_CHARS,
Packit a7d494
					 _("The buffer contains invalid characters."));
Packit a7d494
		return;
Packit a7d494
	}
Packit a7d494
Packit a7d494
	DEBUG ({
Packit a7d494
	       g_print ("Start saving\n");
Packit a7d494
	});
Packit a7d494
Packit a7d494
	implicit_trailing_newline = gtk_source_buffer_get_implicit_trailing_newline (saver->priv->source_buffer);
Packit a7d494
Packit a7d494
	/* The BufferInputStream has a strong reference to the buffer.
Packit a7d494
	 * We create the BufferInputStream here so we are sure that the
Packit a7d494
	 * buffer will not be destroyed during the file saving.
Packit a7d494
	 */
Packit a7d494
	task_data->input_stream = _gtk_source_buffer_input_stream_new (GTK_TEXT_BUFFER (saver->priv->source_buffer),
Packit a7d494
								       saver->priv->newline_type,
Packit a7d494
								       implicit_trailing_newline);
Packit a7d494
Packit a7d494
	check_externally_modified (saver->priv->task);
Packit a7d494
}
Packit a7d494
Packit a7d494
/**
Packit a7d494
 * gtk_source_file_saver_save_finish:
Packit a7d494
 * @saver: a #GtkSourceFileSaver.
Packit a7d494
 * @result: a #GAsyncResult.
Packit a7d494
 * @error: a #GError, or %NULL.
Packit a7d494
 *
Packit a7d494
 * Finishes a file saving started with gtk_source_file_saver_save_async().
Packit a7d494
 *
Packit a7d494
 * If the file has been saved successfully, the following #GtkSourceFile
Packit a7d494
 * properties will be updated: the location, the encoding, the newline type and
Packit a7d494
 * the compression type.
Packit a7d494
 *
Packit a7d494
 * Since the 3.20 version, gtk_text_buffer_set_modified() is called with %FALSE
Packit a7d494
 * if the file has been saved successfully.
Packit a7d494
 *
Packit a7d494
 * Returns: whether the file was saved successfully.
Packit a7d494
 * Since: 3.14
Packit a7d494
 */
Packit a7d494
gboolean
Packit a7d494
gtk_source_file_saver_save_finish (GtkSourceFileSaver  *saver,
Packit a7d494
				   GAsyncResult        *result,
Packit a7d494
				   GError             **error)
Packit a7d494
{
Packit a7d494
	gboolean ok;
Packit a7d494
Packit a7d494
	g_return_val_if_fail (GTK_SOURCE_IS_FILE_SAVER (saver), FALSE);
Packit a7d494
	g_return_val_if_fail (error == NULL || *error == NULL, FALSE);
Packit a7d494
	g_return_val_if_fail (g_task_is_valid (result, saver), FALSE);
Packit a7d494
Packit a7d494
	ok = g_task_propagate_boolean (G_TASK (result), error);
Packit a7d494
Packit a7d494
	if (ok && saver->priv->file != NULL)
Packit a7d494
	{
Packit a7d494
		TaskData *task_data;
Packit a7d494
Packit a7d494
		gtk_source_file_set_location (saver->priv->file,
Packit a7d494
					      saver->priv->location);
Packit a7d494
Packit a7d494
		_gtk_source_file_set_encoding (saver->priv->file,
Packit a7d494
					       saver->priv->encoding);
Packit a7d494
Packit a7d494
		_gtk_source_file_set_newline_type (saver->priv->file,
Packit a7d494
						   saver->priv->newline_type);
Packit a7d494
Packit a7d494
		_gtk_source_file_set_compression_type (saver->priv->file,
Packit a7d494
						       saver->priv->compression_type);
Packit a7d494
Packit a7d494
		_gtk_source_file_set_externally_modified (saver->priv->file, FALSE);
Packit a7d494
		_gtk_source_file_set_deleted (saver->priv->file, FALSE);
Packit a7d494
		_gtk_source_file_set_readonly (saver->priv->file, FALSE);
Packit a7d494
Packit a7d494
		task_data = g_task_get_task_data (G_TASK (result));
Packit a7d494
Packit a7d494
		if (g_file_info_has_attribute (task_data->info, G_FILE_ATTRIBUTE_TIME_MODIFIED))
Packit a7d494
		{
Packit a7d494
			GTimeVal modification_time;
Packit a7d494
Packit a7d494
			g_file_info_get_modification_time (task_data->info, &modification_time);
Packit a7d494
			_gtk_source_file_set_modification_time (saver->priv->file, modification_time);
Packit a7d494
		}
Packit a7d494
	}
Packit a7d494
Packit a7d494
	if (ok && saver->priv->source_buffer != NULL)
Packit a7d494
	{
Packit a7d494
		gtk_text_buffer_set_modified (GTK_TEXT_BUFFER (saver->priv->source_buffer),
Packit a7d494
					      FALSE);
Packit a7d494
	}
Packit a7d494
Packit a7d494
	g_clear_object (&saver->priv->task);
Packit a7d494
Packit a7d494
	return ok;
Packit a7d494
}